Signed-off-by: Jan Hentschel <jan.hentschel@ultratendency.com>
This commit is contained in:
parent
ae4f1b4830
commit
950237566b
|
@ -424,7 +424,11 @@ public class CatalogJanitor extends ScheduledChore {
|
||||||
// It doesn't have merge qualifier, no need to clean
|
// It doesn't have merge qualifier, no need to clean
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return cleanMergeRegion(region, parents);
|
|
||||||
|
// If a parent region is a merged child region and GC has not kicked in/finish its work yet,
|
||||||
|
// return false in this case to avoid kicking in a merge, trying later.
|
||||||
|
cleanMergeRegion(region, parents);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -19,6 +19,7 @@ package org.apache.hadoop.hbase.client;
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
import static org.junit.Assert.assertFalse;
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
|
@ -36,6 +37,8 @@ import org.apache.hadoop.hbase.ServerName;
|
||||||
import org.apache.hadoop.hbase.TableName;
|
import org.apache.hadoop.hbase.TableName;
|
||||||
import org.apache.hadoop.hbase.TableNotFoundException;
|
import org.apache.hadoop.hbase.TableNotFoundException;
|
||||||
import org.apache.hadoop.hbase.exceptions.MergeRegionException;
|
import org.apache.hadoop.hbase.exceptions.MergeRegionException;
|
||||||
|
import org.apache.hadoop.hbase.master.CatalogJanitor;
|
||||||
|
import org.apache.hadoop.hbase.master.HMaster;
|
||||||
import org.apache.hadoop.hbase.regionserver.DisabledRegionSplitPolicy;
|
import org.apache.hadoop.hbase.regionserver.DisabledRegionSplitPolicy;
|
||||||
import org.apache.hadoop.hbase.regionserver.HRegion;
|
import org.apache.hadoop.hbase.regionserver.HRegion;
|
||||||
import org.apache.hadoop.hbase.regionserver.HStore;
|
import org.apache.hadoop.hbase.regionserver.HStore;
|
||||||
|
@ -523,25 +526,42 @@ public class TestAdmin1 extends TestAdminBase {
|
||||||
List<RegionInfo> tableRegions;
|
List<RegionInfo> tableRegions;
|
||||||
RegionInfo regionA;
|
RegionInfo regionA;
|
||||||
RegionInfo regionB;
|
RegionInfo regionB;
|
||||||
|
RegionInfo regionC;
|
||||||
|
RegionInfo mergedChildRegion = null;
|
||||||
|
|
||||||
// merge with full name
|
// merge with full name
|
||||||
tableRegions = ADMIN.getRegions(tableName);
|
tableRegions = ADMIN.getRegions(tableName);
|
||||||
assertEquals(3, tableRegions.size());
|
assertEquals(3, tableRegions.size());
|
||||||
regionA = tableRegions.get(0);
|
regionA = tableRegions.get(0);
|
||||||
regionB = tableRegions.get(1);
|
regionB = tableRegions.get(1);
|
||||||
|
regionC = tableRegions.get(2);
|
||||||
// TODO convert this to version that is synchronous (See HBASE-16668)
|
// TODO convert this to version that is synchronous (See HBASE-16668)
|
||||||
ADMIN.mergeRegionsAsync(regionA.getRegionName(), regionB.getRegionName(), false).get(60,
|
ADMIN.mergeRegionsAsync(regionA.getRegionName(), regionB.getRegionName(),
|
||||||
TimeUnit.SECONDS);
|
false).get(60, TimeUnit.SECONDS);
|
||||||
|
|
||||||
assertEquals(2, ADMIN.getRegions(tableName).size());
|
|
||||||
|
|
||||||
// merge with encoded name
|
|
||||||
tableRegions = ADMIN.getRegions(tableName);
|
tableRegions = ADMIN.getRegions(tableName);
|
||||||
regionA = tableRegions.get(0);
|
|
||||||
regionB = tableRegions.get(1);
|
assertEquals(2, tableRegions.size());
|
||||||
|
for (RegionInfo ri : tableRegions) {
|
||||||
|
if (regionC.compareTo(ri) != 0) {
|
||||||
|
mergedChildRegion = ri;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertNotNull(mergedChildRegion);
|
||||||
|
// Need to wait GC for merged child region is done.
|
||||||
|
HMaster services = TEST_UTIL.getHBaseCluster().getMaster();
|
||||||
|
CatalogJanitor cj = services.getCatalogJanitor();
|
||||||
|
cj.cleanMergeQualifier(mergedChildRegion);
|
||||||
|
// Wait until all procedures settled down
|
||||||
|
while (!services.getMasterProcedureExecutor().getActiveProcIds().isEmpty()) {
|
||||||
|
Thread.sleep(200);
|
||||||
|
}
|
||||||
|
|
||||||
// TODO convert this to version that is synchronous (See HBASE-16668)
|
// TODO convert this to version that is synchronous (See HBASE-16668)
|
||||||
ADMIN
|
ADMIN.mergeRegionsAsync(regionC.getEncodedNameAsBytes(),
|
||||||
.mergeRegionsAsync(regionA.getEncodedNameAsBytes(), regionB.getEncodedNameAsBytes(), false)
|
mergedChildRegion.getEncodedNameAsBytes(), false)
|
||||||
.get(60, TimeUnit.SECONDS);
|
.get(60, TimeUnit.SECONDS);
|
||||||
|
|
||||||
assertEquals(1, ADMIN.getRegions(tableName).size());
|
assertEquals(1, ADMIN.getRegions(tableName).size());
|
||||||
|
|
|
@ -21,6 +21,7 @@ import static org.apache.hadoop.hbase.TableName.META_TABLE_NAME;
|
||||||
import static org.hamcrest.CoreMatchers.instanceOf;
|
import static org.hamcrest.CoreMatchers.instanceOf;
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
import static org.junit.Assert.assertFalse;
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
import static org.junit.Assert.assertThat;
|
import static org.junit.Assert.assertThat;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.junit.Assert.fail;
|
import static org.junit.Assert.fail;
|
||||||
|
@ -35,6 +36,8 @@ import org.apache.hadoop.hbase.HBaseClassTestRule;
|
||||||
import org.apache.hadoop.hbase.HConstants;
|
import org.apache.hadoop.hbase.HConstants;
|
||||||
import org.apache.hadoop.hbase.HRegionLocation;
|
import org.apache.hadoop.hbase.HRegionLocation;
|
||||||
import org.apache.hadoop.hbase.TableName;
|
import org.apache.hadoop.hbase.TableName;
|
||||||
|
import org.apache.hadoop.hbase.master.CatalogJanitor;
|
||||||
|
import org.apache.hadoop.hbase.master.HMaster;
|
||||||
import org.apache.hadoop.hbase.testclassification.ClientTests;
|
import org.apache.hadoop.hbase.testclassification.ClientTests;
|
||||||
import org.apache.hadoop.hbase.testclassification.LargeTests;
|
import org.apache.hadoop.hbase.testclassification.LargeTests;
|
||||||
import org.apache.hadoop.hbase.util.Bytes;
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
|
@ -161,20 +164,39 @@ public class TestAsyncRegionAdminApi2 extends TestAsyncAdminBase {
|
||||||
.getTableHRegionLocations(metaTable, tableName).get();
|
.getTableHRegionLocations(metaTable, tableName).get();
|
||||||
RegionInfo regionA;
|
RegionInfo regionA;
|
||||||
RegionInfo regionB;
|
RegionInfo regionB;
|
||||||
|
RegionInfo regionC;
|
||||||
|
RegionInfo mergedChildRegion = null;
|
||||||
|
|
||||||
// merge with full name
|
// merge with full name
|
||||||
assertEquals(3, regionLocations.size());
|
assertEquals(3, regionLocations.size());
|
||||||
regionA = regionLocations.get(0).getRegion();
|
regionA = regionLocations.get(0).getRegion();
|
||||||
regionB = regionLocations.get(1).getRegion();
|
regionB = regionLocations.get(1).getRegion();
|
||||||
|
regionC = regionLocations.get(2).getRegion();
|
||||||
admin.mergeRegions(regionA.getRegionName(), regionB.getRegionName(), false).get();
|
admin.mergeRegions(regionA.getRegionName(), regionB.getRegionName(), false).get();
|
||||||
|
|
||||||
regionLocations = AsyncMetaTableAccessor
|
regionLocations = AsyncMetaTableAccessor
|
||||||
.getTableHRegionLocations(metaTable, tableName).get();
|
.getTableHRegionLocations(metaTable, tableName).get();
|
||||||
|
|
||||||
assertEquals(2, regionLocations.size());
|
assertEquals(2, regionLocations.size());
|
||||||
|
for (HRegionLocation rl : regionLocations) {
|
||||||
|
if (regionC.compareTo(rl.getRegion()) != 0) {
|
||||||
|
mergedChildRegion = rl.getRegion();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertNotNull(mergedChildRegion);
|
||||||
|
// Need to wait GC for merged child region is done.
|
||||||
|
HMaster services = TEST_UTIL.getHBaseCluster().getMaster();
|
||||||
|
CatalogJanitor cj = services.getCatalogJanitor();
|
||||||
|
cj.cleanMergeQualifier(mergedChildRegion);
|
||||||
|
// Wait until all procedures settled down
|
||||||
|
while (!services.getMasterProcedureExecutor().getActiveProcIds().isEmpty()) {
|
||||||
|
Thread.sleep(200);
|
||||||
|
}
|
||||||
// merge with encoded name
|
// merge with encoded name
|
||||||
regionA = regionLocations.get(0).getRegion();
|
admin.mergeRegions(regionC.getRegionName(), mergedChildRegion.getRegionName(),
|
||||||
regionB = regionLocations.get(1).getRegion();
|
false).get();
|
||||||
admin.mergeRegions(regionA.getRegionName(), regionB.getRegionName(), false).get();
|
|
||||||
|
|
||||||
regionLocations = AsyncMetaTableAccessor
|
regionLocations = AsyncMetaTableAccessor
|
||||||
.getTableHRegionLocations(metaTable, tableName).get();
|
.getTableHRegionLocations(metaTable, tableName).get();
|
||||||
|
|
|
@ -26,11 +26,15 @@ import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.function.BooleanSupplier;
|
import java.util.function.BooleanSupplier;
|
||||||
|
import org.apache.hadoop.hbase.Cell;
|
||||||
|
import org.apache.hadoop.hbase.CellBuilderFactory;
|
||||||
|
import org.apache.hadoop.hbase.CellBuilderType;
|
||||||
import org.apache.hadoop.hbase.HBaseClassTestRule;
|
import org.apache.hadoop.hbase.HBaseClassTestRule;
|
||||||
import org.apache.hadoop.hbase.HBaseTestingUtility;
|
import org.apache.hadoop.hbase.HBaseTestingUtility;
|
||||||
import org.apache.hadoop.hbase.HConstants;
|
import org.apache.hadoop.hbase.HConstants;
|
||||||
import org.apache.hadoop.hbase.MetaTableAccessor;
|
import org.apache.hadoop.hbase.MetaTableAccessor;
|
||||||
import org.apache.hadoop.hbase.TableName;
|
import org.apache.hadoop.hbase.TableName;
|
||||||
|
import org.apache.hadoop.hbase.client.Put;
|
||||||
import org.apache.hadoop.hbase.client.RegionInfo;
|
import org.apache.hadoop.hbase.client.RegionInfo;
|
||||||
import org.apache.hadoop.hbase.client.RegionInfoBuilder;
|
import org.apache.hadoop.hbase.client.RegionInfoBuilder;
|
||||||
import org.apache.hadoop.hbase.client.Result;
|
import org.apache.hadoop.hbase.client.Result;
|
||||||
|
@ -43,6 +47,7 @@ import org.apache.hadoop.hbase.master.procedure.MasterProcedureEnv;
|
||||||
import org.apache.hadoop.hbase.procedure2.ProcedureExecutor;
|
import org.apache.hadoop.hbase.procedure2.ProcedureExecutor;
|
||||||
import org.apache.hadoop.hbase.testclassification.LargeTests;
|
import org.apache.hadoop.hbase.testclassification.LargeTests;
|
||||||
import org.apache.hadoop.hbase.testclassification.MasterTests;
|
import org.apache.hadoop.hbase.testclassification.MasterTests;
|
||||||
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
import org.apache.hadoop.hbase.util.Pair;
|
import org.apache.hadoop.hbase.util.Pair;
|
||||||
import org.apache.hadoop.hbase.util.Threads;
|
import org.apache.hadoop.hbase.util.Threads;
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
|
@ -141,7 +146,7 @@ public class TestMetaFixer {
|
||||||
assertEquals(0, ris.size());
|
assertEquals(0, ris.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void makeOverlap(MasterServices services, RegionInfo a, RegionInfo b)
|
private static RegionInfo makeOverlap(MasterServices services, RegionInfo a, RegionInfo b)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
RegionInfo overlapRegion = RegionInfoBuilder.newBuilder(a.getTable()).
|
RegionInfo overlapRegion = RegionInfoBuilder.newBuilder(a.getTable()).
|
||||||
setStartKey(a.getStartKey()).
|
setStartKey(a.getStartKey()).
|
||||||
|
@ -152,6 +157,7 @@ public class TestMetaFixer {
|
||||||
System.currentTimeMillis())));
|
System.currentTimeMillis())));
|
||||||
// TODO: Add checks at assign time to PREVENT being able to assign over existing assign.
|
// TODO: Add checks at assign time to PREVENT being able to assign over existing assign.
|
||||||
services.getAssignmentManager().assign(overlapRegion);
|
services.getAssignmentManager().assign(overlapRegion);
|
||||||
|
return overlapRegion;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void testOverlapCommon(final TableName tn) throws Exception {
|
private void testOverlapCommon(final TableName tn) throws Exception {
|
||||||
|
@ -167,7 +173,6 @@ public class TestMetaFixer {
|
||||||
makeOverlap(services, ris.get(1), ris.get(3));
|
makeOverlap(services, ris.get(1), ris.get(3));
|
||||||
makeOverlap(services, ris.get(2), ris.get(3));
|
makeOverlap(services, ris.get(2), ris.get(3));
|
||||||
makeOverlap(services, ris.get(2), ris.get(4));
|
makeOverlap(services, ris.get(2), ris.get(4));
|
||||||
Threads.sleep(10000);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -308,6 +313,74 @@ public class TestMetaFixer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This test covers the case that one of merged parent regions is a merged child region that
|
||||||
|
* has not been GCed but there is no reference files anymore. In this case, it will kick off
|
||||||
|
* a GC procedure, but no merge will happen.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testMergeWithMergedChildRegion() throws Exception {
|
||||||
|
TableName tn = TableName.valueOf(this.name.getMethodName());
|
||||||
|
Table t = TEST_UTIL.createMultiRegionTable(tn, HConstants.CATALOG_FAMILY);
|
||||||
|
List<RegionInfo> ris = MetaTableAccessor.getTableRegions(TEST_UTIL.getConnection(), tn);
|
||||||
|
assertTrue(ris.size() > 5);
|
||||||
|
HMaster services = TEST_UTIL.getHBaseCluster().getMaster();
|
||||||
|
CatalogJanitor cj = services.getCatalogJanitor();
|
||||||
|
cj.scan();
|
||||||
|
CatalogJanitor.Report report = cj.getLastReport();
|
||||||
|
assertTrue(report.isEmpty());
|
||||||
|
RegionInfo overlapRegion = makeOverlap(services, ris.get(1), ris.get(2));
|
||||||
|
|
||||||
|
cj.scan();
|
||||||
|
report = cj.getLastReport();
|
||||||
|
assertEquals(2, report.getOverlaps().size());
|
||||||
|
|
||||||
|
// Mark it as a merged child region.
|
||||||
|
RegionInfo fakedParentRegion = RegionInfoBuilder.newBuilder(tn).
|
||||||
|
setStartKey(overlapRegion.getStartKey()).
|
||||||
|
build();
|
||||||
|
|
||||||
|
Table meta = MetaTableAccessor.getMetaHTable(TEST_UTIL.getConnection());
|
||||||
|
Put putOfMerged = MetaTableAccessor.makePutFromRegionInfo(overlapRegion,
|
||||||
|
HConstants.LATEST_TIMESTAMP);
|
||||||
|
String qualifier = String.format(HConstants.MERGE_QUALIFIER_PREFIX_STR + "%04d", 0);
|
||||||
|
putOfMerged.add(CellBuilderFactory.create(CellBuilderType.SHALLOW_COPY).setRow(
|
||||||
|
putOfMerged.getRow()).
|
||||||
|
setFamily(HConstants.CATALOG_FAMILY).
|
||||||
|
setQualifier(Bytes.toBytes(qualifier)).
|
||||||
|
setTimestamp(putOfMerged.getTimestamp()).
|
||||||
|
setType(Cell.Type.Put).
|
||||||
|
setValue(RegionInfo.toByteArray(fakedParentRegion)).
|
||||||
|
build());
|
||||||
|
|
||||||
|
meta.put(putOfMerged);
|
||||||
|
|
||||||
|
MetaFixer fixer = new MetaFixer(services);
|
||||||
|
fixer.fixOverlaps(report);
|
||||||
|
|
||||||
|
// Wait until all procedures settled down
|
||||||
|
await(200, () -> {
|
||||||
|
return services.getMasterProcedureExecutor().getActiveProcIds().isEmpty();
|
||||||
|
});
|
||||||
|
|
||||||
|
// No merge is done, overlap is still there.
|
||||||
|
cj.scan();
|
||||||
|
report = cj.getLastReport();
|
||||||
|
assertEquals(2, report.getOverlaps().size());
|
||||||
|
|
||||||
|
fixer.fixOverlaps(report);
|
||||||
|
|
||||||
|
// Wait until all procedures settled down
|
||||||
|
await(200, () -> {
|
||||||
|
return services.getMasterProcedureExecutor().getActiveProcIds().isEmpty();
|
||||||
|
});
|
||||||
|
|
||||||
|
// Merge is done and no more overlaps
|
||||||
|
cj.scan();
|
||||||
|
report = cj.getLastReport();
|
||||||
|
assertEquals(0, report.getOverlaps().size());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Make it so a big overlap spans many Regions, some of which are non-contiguous. Make it so
|
* Make it so a big overlap spans many Regions, some of which are non-contiguous. Make it so
|
||||||
* we can fix this condition. HBASE-24247
|
* we can fix this condition. HBASE-24247
|
||||||
|
@ -336,7 +409,6 @@ public class TestMetaFixer {
|
||||||
while (!services.getMasterProcedureExecutor().isFinished(pid)) {
|
while (!services.getMasterProcedureExecutor().isFinished(pid)) {
|
||||||
Threads.sleep(100);
|
Threads.sleep(100);
|
||||||
}
|
}
|
||||||
Threads.sleep(10000);
|
|
||||||
services.getCatalogJanitor().scan();
|
services.getCatalogJanitor().scan();
|
||||||
report = services.getCatalogJanitor().getLastReport();
|
report = services.getCatalogJanitor().getLastReport();
|
||||||
assertEquals(1, MetaFixer.calculateMerges(10, report.getOverlaps()).size());
|
assertEquals(1, MetaFixer.calculateMerges(10, report.getOverlaps()).size());
|
||||||
|
|
Loading…
Reference in New Issue