blob: fc5a4656c263f2aab9308d2a71c8b31a30d2f400 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.ignite.util;
import org.apache.ignite.failure.FailureHandler;
import org.apache.ignite.failure.StopNodeFailureHandler;
import org.junit.Test;
import static org.apache.ignite.internal.commandline.CommandHandler.EXIT_CODE_OK;
import static org.apache.ignite.internal.commandline.cache.CheckIndexInlineSizes.INDEXES_INLINE_SIZE_ARE_THE_SAME;
import static org.apache.ignite.testframework.GridTestUtils.assertContains;
import static org.apache.ignite.util.GridCommandHandlerIndexingUtils.CACHE_NAME;
import static org.apache.ignite.util.GridCommandHandlerIndexingUtils.GROUP_NAME;
import static org.apache.ignite.util.GridCommandHandlerIndexingUtils.breakCacheDataTree;
import static org.apache.ignite.util.GridCommandHandlerIndexingUtils.breakSqlIndex;
import static org.apache.ignite.util.GridCommandHandlerIndexingUtils.createAndFillCache;
/**
* You can use this class if you don't need create nodes for each test because
* here create {@link #SERVER_NODE_CNT} server and 1 client nodes at before all
* tests. If you need create nodes for each test you can use
* {@link GridCommandHandlerIndexingTest}.
*/
public class GridCommandHandlerIndexingClusterByClassTest extends GridCommandHandlerClusterByClassAbstractTest {
/** {@inheritDoc} */
@Override protected void beforeTest() throws Exception {
super.beforeTest();
createAndFillCache(client, CACHE_NAME, GROUP_NAME);
}
/** {@inheritDoc} */
@Override protected FailureHandler getFailureHandler(String igniteInstanceName) {
return new StopNodeFailureHandler();
}
/**
* Tests --cache check_index_inline_sizes works in case of all indexes have the same inline size.
*/
@Test
public void testCheckIndexInlineSizesNoError() {
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "check_index_inline_sizes"));
String output = testOut.toString();
assertContains(log, output, "Found 2 secondary indexes.");
assertContains(log, output, INDEXES_INLINE_SIZE_ARE_THE_SAME);
}
/**
* Tests that validation doesn't fail if nothing is broken.
*/
@Test
public void testValidateIndexesNoErrors() {
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "validate_indexes", CACHE_NAME));
assertContains(log, testOut.toString(), "no issues found");
}
/**
* Tests that validation with CRC checking doesn't fail if nothing is broken.
*/
@Test
public void testValidateIndexesWithCrcNoErrors() {
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "validate_indexes", "--check-crc", CACHE_NAME));
assertContains(log, testOut.toString(), "no issues found");
}
/**
* Test verifies that validate_indexes command finishes successfully when no cache names are specified.
*/
@Test
public void testValidateIndexesNoErrorEmptyCacheNameArg() {
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "validate_indexes"));
assertContains(log, testOut.toString(), "no issues found");
}
/**
* Tests that missing rows in CacheDataTree are detected.
*/
@Test
public void testBrokenCacheDataTreeShouldFailValidation() {
breakCacheDataTreeOnCrd();
injectTestSystemOut();
assertEquals(EXIT_CODE_OK,
execute(
"--cache",
"validate_indexes",
CACHE_NAME,
"--check-first", "10000",
"--check-through", "10"));
String out = testOut.toString();
assertContains(log, out, "issues found (listed above)");
assertContains(log, out, "Key is present in SQL index, but is missing in corresponding data page.");
}
/**
* Checks that missing lines were detected in CacheDataTree with the output
* of cache group name and id.
*/
@Test
public void testBrokenCacheDataTreeShouldFailValidationWithCacheGroupInfo() {
breakCacheDataTreeOnCrd();
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "validate_indexes", CACHE_NAME));
assertContains(
log,
testOut.toString(),
"[cacheGroup=group1, cacheGroupId=-1237460590, cache=persons-cache-vi, cacheId=-528791027, idx=_key_PK]"
);
}
/**
* Tests that missing rows in H2 indexes are detected.
*/
@Test
public void testBrokenSqlIndexShouldFailValidation() throws Exception {
breakSqlIndexOnCrd();
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "validate_indexes", CACHE_NAME));
assertContains(log, testOut.toString(), "issues found (listed above)");
}
/**
* Test to validate only specified cache, not all cache group.
*/
@Test
public void testValidateSingleCacheShouldNotTriggerCacheGroupValidation() throws Exception {
createAndFillCache(crd, DEFAULT_CACHE_NAME, GROUP_NAME);
forceCheckpoint();
breakCacheDataTreeOnCrd();
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "validate_indexes", DEFAULT_CACHE_NAME, "--check-through", "10"));
assertContains(log, testOut.toString(), "no issues found");
}
/**
* Test validate_indexes with empty cache list.
*/
@Test
public void testCacheValidateIndexesPassEmptyCacheList() throws Exception {
injectTestSystemOut();
assertEquals(EXIT_CODE_OK, execute("--cache", "validate_indexes"));
assertContains(log, testOut.toString(), "no issues found");
}
/**
* Removes some entries from a partition skipping index update.
*/
private void breakCacheDataTreeOnCrd() {
breakCacheDataTree(log, crd.cachex(CACHE_NAME), 1, (i, entry) -> i % 5 == 0);
}
/**
* Removes some entries from H2 trees skipping partition updates.
* This effectively breaks the index.
*
* @throws Exception If failed.
*/
private void breakSqlIndexOnCrd() throws Exception {
breakSqlIndex(crd.cachex(CACHE_NAME), 0, null);
}
}