HBASE-11512 Write region open/close events to WAL
This commit is contained in:
parent
05e67eba9d
commit
cbc1bc7d49
|
@ -120,6 +120,8 @@ import org.apache.hadoop.hbase.protobuf.generated.RegionServerStatusProtos.Regio
|
|||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.CompactionDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor.FlushAction;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.RegionEventDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.RegionEventDescriptor.EventType;
|
||||
import org.apache.hadoop.hbase.security.access.Permission;
|
||||
import org.apache.hadoop.hbase.security.access.TablePermission;
|
||||
import org.apache.hadoop.hbase.security.access.UserPermission;
|
||||
|
@ -2531,6 +2533,30 @@ public final class ProtobufUtil {
|
|||
return desc.build();
|
||||
}
|
||||
|
||||
public static RegionEventDescriptor toRegionEventDescriptor(
|
||||
EventType eventType, HRegionInfo hri, long seqId, ServerName server,
|
||||
Map<byte[], List<Path>> storeFiles) {
|
||||
RegionEventDescriptor.Builder desc = RegionEventDescriptor.newBuilder()
|
||||
.setEventType(eventType)
|
||||
.setTableName(ByteStringer.wrap(hri.getTable().getName()))
|
||||
.setEncodedRegionName(ByteStringer.wrap(hri.getEncodedNameAsBytes()))
|
||||
.setLogSequenceNumber(seqId)
|
||||
.setServer(toServerName(server));
|
||||
|
||||
for (Map.Entry<byte[], List<Path>> entry : storeFiles.entrySet()) {
|
||||
RegionEventDescriptor.StoreDescriptor.Builder builder
|
||||
= RegionEventDescriptor.StoreDescriptor.newBuilder()
|
||||
.setFamilyName(ByteStringer.wrap(entry.getKey()))
|
||||
.setStoreHomeDir(Bytes.toString(entry.getKey()));
|
||||
for (Path path : entry.getValue()) {
|
||||
builder.addStoreFile(path.getName());
|
||||
}
|
||||
|
||||
desc.addStores(builder);
|
||||
}
|
||||
return desc.build();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return short version of Message toString'd, shorter than TextFormat#shortDebugString.
|
||||
* Tries to NOT print out data both because it can be big but also so we do not have data in our
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -120,6 +120,29 @@ message FlushDescriptor {
|
|||
repeated StoreFlushDescriptor store_flushes = 5;
|
||||
}
|
||||
|
||||
/**
|
||||
* Special WAL entry to hold all related to a region event (open/close).
|
||||
*/
|
||||
message RegionEventDescriptor {
|
||||
enum EventType {
|
||||
REGION_OPEN = 0;
|
||||
REGION_CLOSE = 1;
|
||||
}
|
||||
|
||||
message StoreDescriptor {
|
||||
required bytes family_name = 1;
|
||||
required string store_home_dir = 2; //relative to region dir
|
||||
repeated string store_file = 3; // relative to store dir
|
||||
}
|
||||
|
||||
required EventType event_type = 1;
|
||||
required bytes table_name = 2;
|
||||
required bytes encoded_region_name = 3;
|
||||
optional uint64 log_sequence_number = 4;
|
||||
repeated StoreDescriptor stores = 5;
|
||||
optional ServerName server = 6; // Server who opened the region
|
||||
}
|
||||
|
||||
/**
|
||||
* A trailer that is appended to the end of a properly closed HLog WAL file.
|
||||
* If missing, this is either a legacy or a corrupted WAL file.
|
||||
|
|
|
@ -120,6 +120,7 @@ import org.apache.hadoop.hbase.protobuf.generated.ClientProtos.CoprocessorServic
|
|||
import org.apache.hadoop.hbase.protobuf.generated.HBaseProtos.SnapshotDescription;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.CompactionDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.RegionEventDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor.FlushAction;
|
||||
import org.apache.hadoop.hbase.regionserver.MultiVersionConsistencyControl.WriteEntry;
|
||||
import org.apache.hadoop.hbase.regionserver.compactions.CompactionContext;
|
||||
|
@ -725,6 +726,8 @@ public class HRegion implements HeapSize { // , Writable{
|
|||
status.setStatus("Writing region info on filesystem");
|
||||
fs.checkRegionInfoOnFilesystem();
|
||||
|
||||
|
||||
|
||||
// Initialize all the HStores
|
||||
status.setStatus("Initializing all the Stores");
|
||||
long maxSeqId = initializeRegionStores(reporter, status);
|
||||
|
@ -761,6 +764,7 @@ public class HRegion implements HeapSize { // , Writable{
|
|||
// overlaps used sequence numbers
|
||||
nextSeqid += this.flushPerChanges + 10000000; // add another extra 10million
|
||||
}
|
||||
|
||||
LOG.info("Onlined " + this.getRegionInfo().getShortNameToLog() +
|
||||
"; next sequenceid=" + nextSeqid);
|
||||
|
||||
|
@ -850,6 +854,44 @@ public class HRegion implements HeapSize { // , Writable{
|
|||
return maxSeqId;
|
||||
}
|
||||
|
||||
private void writeRegionOpenMarker(HLog log, long openSeqId) throws IOException {
|
||||
Map<byte[], List<Path>> storeFiles
|
||||
= new TreeMap<byte[], List<Path>>(Bytes.BYTES_COMPARATOR);
|
||||
for (Map.Entry<byte[], Store> entry : getStores().entrySet()) {
|
||||
Store store = entry.getValue();
|
||||
ArrayList<Path> storeFileNames = new ArrayList<Path>();
|
||||
for (StoreFile storeFile: store.getStorefiles()) {
|
||||
storeFileNames.add(storeFile.getPath());
|
||||
}
|
||||
storeFiles.put(entry.getKey(), storeFileNames);
|
||||
}
|
||||
|
||||
RegionEventDescriptor regionOpenDesc = ProtobufUtil.toRegionEventDescriptor(
|
||||
RegionEventDescriptor.EventType.REGION_OPEN, getRegionInfo(), openSeqId,
|
||||
getRegionServerServices().getServerName(), storeFiles);
|
||||
HLogUtil.writeRegionEventMarker(log, getTableDesc(), getRegionInfo(), regionOpenDesc,
|
||||
getSequenceId());
|
||||
}
|
||||
|
||||
private void writeRegionCloseMarker(HLog log) throws IOException {
|
||||
Map<byte[], List<Path>> storeFiles
|
||||
= new TreeMap<byte[], List<Path>>(Bytes.BYTES_COMPARATOR);
|
||||
for (Map.Entry<byte[], Store> entry : getStores().entrySet()) {
|
||||
Store store = entry.getValue();
|
||||
ArrayList<Path> storeFileNames = new ArrayList<Path>();
|
||||
for (StoreFile storeFile: store.getStorefiles()) {
|
||||
storeFileNames.add(storeFile.getPath());
|
||||
}
|
||||
storeFiles.put(entry.getKey(), storeFileNames);
|
||||
}
|
||||
|
||||
RegionEventDescriptor regionEventDesc = ProtobufUtil.toRegionEventDescriptor(
|
||||
RegionEventDescriptor.EventType.REGION_CLOSE, getRegionInfo(), getSequenceId().get(),
|
||||
getRegionServerServices().getServerName(), storeFiles);
|
||||
HLogUtil.writeRegionEventMarker(log, getTableDesc(), getRegionInfo(), regionEventDesc,
|
||||
getSequenceId());
|
||||
}
|
||||
|
||||
/**
|
||||
* @return True if this region has references.
|
||||
*/
|
||||
|
@ -1227,6 +1269,12 @@ public class HRegion implements HeapSize { // , Writable{
|
|||
storeCloserThreadPool.shutdownNow();
|
||||
}
|
||||
}
|
||||
|
||||
status.setStatus("Writing region close event to WAL");
|
||||
if (!abort && log != null && getRegionServerServices() != null) {
|
||||
writeRegionCloseMarker(log);
|
||||
}
|
||||
|
||||
this.closed.set(true);
|
||||
if (memstoreSize.get() != 0) LOG.error("Memstore size is " + memstoreSize.get());
|
||||
if (coprocessorHost != null) {
|
||||
|
@ -3550,6 +3598,7 @@ public class HRegion implements HeapSize { // , Writable{
|
|||
}
|
||||
return storeFileNames;
|
||||
}
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Support code
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
@ -4601,6 +4650,9 @@ public class HRegion implements HeapSize { // , Writable{
|
|||
|
||||
this.openSeqNum = initialize(reporter);
|
||||
this.setSequenceId(openSeqNum);
|
||||
if (log != null && getRegionServerServices() != null) {
|
||||
writeRegionOpenMarker(log, openSeqNum);
|
||||
}
|
||||
return this;
|
||||
}
|
||||
|
||||
|
|
|
@ -40,6 +40,7 @@ import org.apache.hadoop.hbase.ServerName;
|
|||
import org.apache.hadoop.hbase.TableName;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.CompactionDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.RegionEventDescriptor;
|
||||
import org.apache.hadoop.hbase.util.FSUtils;
|
||||
|
||||
import com.google.protobuf.TextFormat;
|
||||
|
@ -284,4 +285,20 @@ public class HLogUtil {
|
|||
}
|
||||
return trx;
|
||||
}
|
||||
|
||||
/**
|
||||
* Write a region open marker indicating that the region is opened
|
||||
*/
|
||||
public static long writeRegionEventMarker(HLog log, HTableDescriptor htd, HRegionInfo info,
|
||||
final RegionEventDescriptor r, AtomicLong sequenceId) throws IOException {
|
||||
TableName tn = TableName.valueOf(r.getTableName().toByteArray());
|
||||
HLogKey key = new HLogKey(info.getEncodedNameAsBytes(), tn);
|
||||
long trx = log.appendNoSync(htd, info, key, WALEdit.createRegionEventWALEdit(info, r),
|
||||
sequenceId, false, null);
|
||||
log.sync(trx);
|
||||
if (LOG.isTraceEnabled()) {
|
||||
LOG.trace("Appended region event marker " + TextFormat.shortDebugString(r));
|
||||
}
|
||||
return trx;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -37,6 +37,7 @@ import org.apache.hadoop.hbase.codec.Codec;
|
|||
import org.apache.hadoop.hbase.io.HeapSize;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.CompactionDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.RegionEventDescriptor;
|
||||
import org.apache.hadoop.hbase.util.Bytes;
|
||||
import org.apache.hadoop.hbase.util.ClassSize;
|
||||
import org.apache.hadoop.hbase.util.EnvironmentEdgeManager;
|
||||
|
@ -86,6 +87,7 @@ public class WALEdit implements Writable, HeapSize {
|
|||
static final byte [] METAROW = Bytes.toBytes("METAROW");
|
||||
static final byte[] COMPACTION = Bytes.toBytes("HBASE::COMPACTION");
|
||||
static final byte [] FLUSH = Bytes.toBytes("HBASE::FLUSH");
|
||||
static final byte [] REGION_EVENT = Bytes.toBytes("HBASE::REGION_EVENT");
|
||||
|
||||
private final int VERSION_2 = -1;
|
||||
private final boolean isReplay;
|
||||
|
@ -277,6 +279,20 @@ public class WALEdit implements Writable, HeapSize {
|
|||
return null;
|
||||
}
|
||||
|
||||
public static WALEdit createRegionEventWALEdit(HRegionInfo hri,
|
||||
RegionEventDescriptor regionEventDesc) {
|
||||
KeyValue kv = new KeyValue(getRowForRegion(hri), METAFAMILY, REGION_EVENT,
|
||||
EnvironmentEdgeManager.currentTimeMillis(), regionEventDesc.toByteArray());
|
||||
return new WALEdit().add(kv);
|
||||
}
|
||||
|
||||
public static RegionEventDescriptor getRegionEventDescriptor(Cell cell) throws IOException {
|
||||
if (CellUtil.matchingColumn(cell, METAFAMILY, REGION_EVENT)) {
|
||||
return RegionEventDescriptor.parseFrom(cell.getValue());
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a compacion WALEdit
|
||||
* @param c
|
||||
|
|
|
@ -118,7 +118,7 @@ public class TestDistributedLogSplitting {
|
|||
Logger.getLogger("org.apache.hadoop.hbase").setLevel(Level.DEBUG);
|
||||
|
||||
// test ThreeRSAbort fails under hadoop2 (2.0.2-alpha) if shortcircuit-read (scr) is on. this
|
||||
// turns it off for this test. TODO: Figure out why scr breaks recovery.
|
||||
// turns it off for this test. TODO: Figure out why scr breaks recovery.
|
||||
System.setProperty("hbase.tests.use.shortcircuit.reads", "false");
|
||||
|
||||
}
|
||||
|
@ -178,7 +178,7 @@ public class TestDistributedLogSplitting {
|
|||
// refresh configuration
|
||||
conf = HBaseConfiguration.create(originalConf);
|
||||
}
|
||||
|
||||
|
||||
@After
|
||||
public void after() throws Exception {
|
||||
try {
|
||||
|
@ -193,7 +193,7 @@ public class TestDistributedLogSplitting {
|
|||
ZKUtil.deleteNodeRecursively(TEST_UTIL.getZooKeeperWatcher(), "/hbase");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Test (timeout=300000)
|
||||
public void testRecoveredEdits() throws Exception {
|
||||
LOG.info("testRecoveredEdits");
|
||||
|
@ -484,7 +484,7 @@ public class TestDistributedLogSplitting {
|
|||
ht.close();
|
||||
zkw.close();
|
||||
}
|
||||
|
||||
|
||||
@Test(timeout = 300000)
|
||||
public void testMasterStartsUpWithLogReplayWork() throws Exception {
|
||||
LOG.info("testMasterStartsUpWithLogReplayWork");
|
||||
|
@ -706,7 +706,7 @@ public class TestDistributedLogSplitting {
|
|||
|
||||
this.prepareData(ht, Bytes.toBytes("family"), Bytes.toBytes("c1"));
|
||||
String originalCheckSum = TEST_UTIL.checksumRows(ht);
|
||||
|
||||
|
||||
// abort RA and trigger replay
|
||||
abortRSAndWaitForRecovery(hrs, zkw, NUM_REGIONS_TO_CREATE);
|
||||
|
||||
|
@ -779,10 +779,10 @@ public class TestDistributedLogSplitting {
|
|||
}
|
||||
makeHLog(hrs.getWAL(), regions, "disableTable", "family", NUM_LOG_LINES, 100, false);
|
||||
makeHLog(hrs.getWAL(), regions, "table", "family", NUM_LOG_LINES, 100);
|
||||
|
||||
|
||||
LOG.info("Disabling table\n");
|
||||
TEST_UTIL.getHBaseAdmin().disableTable(TableName.valueOf("disableTable"));
|
||||
|
||||
|
||||
// abort RS
|
||||
LOG.info("Aborting region server: " + hrs.getServerName());
|
||||
hrs.abort("testing");
|
||||
|
@ -839,7 +839,7 @@ public class TestDistributedLogSplitting {
|
|||
assertEquals(NUM_LOG_LINES, count);
|
||||
LOG.info("Verify replayed edits");
|
||||
assertEquals(NUM_LOG_LINES, TEST_UTIL.countRows(ht));
|
||||
|
||||
|
||||
// clean up
|
||||
for (HRegionInfo hri : regions) {
|
||||
Path editsdir =
|
||||
|
@ -881,7 +881,7 @@ public class TestDistributedLogSplitting {
|
|||
dstRS = rsts.get((i+1) % NUM_RS).getRegionServer();
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
slm.markRegionsRecoveringInZK(hrs.getServerName(), regionSet);
|
||||
// move region in order for the region opened in recovering state
|
||||
final HRegionInfo hri = region;
|
||||
|
@ -898,7 +898,7 @@ public class TestDistributedLogSplitting {
|
|||
return (sn != null && sn.equals(tmpRS.getServerName()));
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
try {
|
||||
byte[] key = region.getStartKey();
|
||||
if (key == null || key.length == 0) {
|
||||
|
@ -957,6 +957,7 @@ public class TestDistributedLogSplitting {
|
|||
"table", "family", NUM_LOG_LINES, 100);
|
||||
|
||||
new Thread() {
|
||||
@Override
|
||||
public void run() {
|
||||
waitForCounter(tot_wkr_task_acquired, 0, 1, 1000);
|
||||
for (RegionServerThread rst : rsts) {
|
||||
|
@ -1147,7 +1148,7 @@ public class TestDistributedLogSplitting {
|
|||
assertTrue(isMetaRegionInRecovery);
|
||||
|
||||
master.getMasterFileSystem().splitMetaLog(hrs.getServerName());
|
||||
|
||||
|
||||
isMetaRegionInRecovery = false;
|
||||
recoveringRegions =
|
||||
zkw.getRecoverableZooKeeper().getChildren(zkw.recoveringRegionsZNode, false);
|
||||
|
@ -1319,7 +1320,7 @@ public class TestDistributedLogSplitting {
|
|||
WALEdit e = new WALEdit();
|
||||
value++;
|
||||
e.add(new KeyValue(row, family, qualifier, timeStamp, Bytes.toBytes(value)));
|
||||
hrs.getWAL().append(curRegionInfo, TableName.valueOf(tableName), e,
|
||||
hrs.getWAL().append(curRegionInfo, TableName.valueOf(tableName), e,
|
||||
System.currentTimeMillis(), htd, sequenceId);
|
||||
}
|
||||
hrs.getWAL().sync();
|
||||
|
@ -1327,7 +1328,7 @@ public class TestDistributedLogSplitting {
|
|||
|
||||
// wait for abort completes
|
||||
this.abortRSAndWaitForRecovery(hrs, zkw, NUM_REGIONS_TO_CREATE);
|
||||
|
||||
|
||||
// verify we got the last value
|
||||
LOG.info("Verification Starts...");
|
||||
Get g = new Get(row);
|
||||
|
@ -1339,7 +1340,7 @@ public class TestDistributedLogSplitting {
|
|||
LOG.info("Verification after flush...");
|
||||
TEST_UTIL.getHBaseAdmin().flush(tableName);
|
||||
TEST_UTIL.getHBaseAdmin().compact(tableName);
|
||||
|
||||
|
||||
// wait for compaction completes
|
||||
TEST_UTIL.waitFor(30000, 200, new Waiter.Predicate<Exception>() {
|
||||
@Override
|
||||
|
@ -1358,7 +1359,7 @@ public class TestDistributedLogSplitting {
|
|||
return installTable(zkw, tname, fname, nrs, 0);
|
||||
}
|
||||
|
||||
HTable installTable(ZooKeeperWatcher zkw, String tname, String fname, int nrs,
|
||||
HTable installTable(ZooKeeperWatcher zkw, String tname, String fname, int nrs,
|
||||
int existingRegions) throws Exception {
|
||||
// Create a table with regions
|
||||
TableName table = TableName.valueOf(tname);
|
||||
|
@ -1499,8 +1500,11 @@ public class TestDistributedLogSplitting {
|
|||
throws IOException {
|
||||
int count = 0;
|
||||
HLog.Reader in = HLogFactory.createReader(fs, log, conf);
|
||||
while (in.next() != null) {
|
||||
count++;
|
||||
HLog.Entry e;
|
||||
while ((e = in.next()) != null) {
|
||||
if (!WALEdit.isMetaEditFamily(e.getEdit().getKeyValues().get(0))) {
|
||||
count++;
|
||||
}
|
||||
}
|
||||
return count;
|
||||
}
|
||||
|
|
|
@ -34,8 +34,10 @@ import static org.junit.Assert.assertNull;
|
|||
import static org.junit.Assert.assertTrue;
|
||||
import static org.junit.Assert.fail;
|
||||
import static org.mockito.Matchers.any;
|
||||
import static org.mockito.Matchers.anyBoolean;
|
||||
import static org.mockito.Matchers.anyLong;
|
||||
import static org.mockito.Matchers.argThat;
|
||||
import static org.mockito.Mockito.mock;
|
||||
import static org.mockito.Mockito.never;
|
||||
import static org.mockito.Mockito.spy;
|
||||
import static org.mockito.Mockito.times;
|
||||
|
@ -86,6 +88,7 @@ import org.apache.hadoop.hbase.MultithreadedTestUtil;
|
|||
import org.apache.hadoop.hbase.MultithreadedTestUtil.RepeatingTestThread;
|
||||
import org.apache.hadoop.hbase.MultithreadedTestUtil.TestThread;
|
||||
import org.apache.hadoop.hbase.NotServingRegionException;
|
||||
import org.apache.hadoop.hbase.ServerName;
|
||||
import org.apache.hadoop.hbase.TableName;
|
||||
import org.apache.hadoop.hbase.Waiter;
|
||||
import org.apache.hadoop.hbase.client.Append;
|
||||
|
@ -114,8 +117,10 @@ import org.apache.hadoop.hbase.monitoring.TaskMonitor;
|
|||
import org.apache.hadoop.hbase.protobuf.ProtobufUtil;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.CompactionDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.RegionEventDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor.FlushAction;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.FlushDescriptor.StoreFlushDescriptor;
|
||||
import org.apache.hadoop.hbase.protobuf.generated.WALProtos.RegionEventDescriptor.StoreDescriptor;
|
||||
import org.apache.hadoop.hbase.regionserver.HRegion.RegionScannerImpl;
|
||||
import org.apache.hadoop.hbase.regionserver.HRegion.RowLock;
|
||||
import org.apache.hadoop.hbase.regionserver.TestStore.FaultyFileSystem;
|
||||
|
@ -141,6 +146,7 @@ import org.junit.Rule;
|
|||
import org.junit.Test;
|
||||
import org.junit.experimental.categories.Category;
|
||||
import org.junit.rules.TestName;
|
||||
import org.mockito.ArgumentCaptor;
|
||||
import org.mockito.ArgumentMatcher;
|
||||
import org.mockito.Mockito;
|
||||
|
||||
|
@ -5481,6 +5487,136 @@ public class TestHRegion {
|
|||
this.region = null;
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("unchecked")
|
||||
public void testOpenRegionWrittenToWAL() throws Exception {
|
||||
final ServerName serverName = ServerName.valueOf("testOpenRegionWrittenToWAL", 100, 42);
|
||||
final RegionServerServices rss = spy(TEST_UTIL.createMockRegionServerService(serverName));
|
||||
|
||||
HTableDescriptor htd
|
||||
= new HTableDescriptor(TableName.valueOf("testOpenRegionWrittenToWAL"));
|
||||
htd.addFamily(new HColumnDescriptor(fam1));
|
||||
htd.addFamily(new HColumnDescriptor(fam2));
|
||||
|
||||
HRegionInfo hri = new HRegionInfo(htd.getTableName(),
|
||||
HConstants.EMPTY_BYTE_ARRAY, HConstants.EMPTY_BYTE_ARRAY);
|
||||
|
||||
// open the region w/o rss and log and flush some files
|
||||
HRegion region =
|
||||
HRegion.createHRegion(hri, TEST_UTIL.getDataTestDir(), TEST_UTIL
|
||||
.getConfiguration(), htd);
|
||||
assertNotNull(region);
|
||||
|
||||
// create a file in fam1 for the region before opening in OpenRegionHandler
|
||||
region.put(new Put(Bytes.toBytes("a")).add(fam1, fam1, fam1));
|
||||
region.flushcache();
|
||||
region.close();
|
||||
|
||||
ArgumentCaptor<WALEdit> editCaptor = ArgumentCaptor.forClass(WALEdit.class);
|
||||
|
||||
// capture appendNoSync() calls
|
||||
HLog log = mock(HLog.class);
|
||||
when(rss.getWAL((HRegionInfo) any())).thenReturn(log);
|
||||
|
||||
try {
|
||||
region = HRegion.openHRegion(hri, htd, rss.getWAL(hri),
|
||||
TEST_UTIL.getConfiguration(), rss, null);
|
||||
|
||||
verify(log, times(1)).appendNoSync((HTableDescriptor)any(), (HRegionInfo)any(), (HLogKey)any()
|
||||
, editCaptor.capture(), (AtomicLong)any(), anyBoolean(), (List<KeyValue>)any());
|
||||
|
||||
WALEdit edit = editCaptor.getValue();
|
||||
assertNotNull(edit);
|
||||
assertNotNull(edit.getKeyValues());
|
||||
assertEquals(1, edit.getKeyValues().size());
|
||||
RegionEventDescriptor desc = WALEdit.getRegionEventDescriptor(edit.getKeyValues().get(0));
|
||||
assertNotNull(desc);
|
||||
|
||||
LOG.info("RegionEventDescriptor from WAL: " + desc);
|
||||
|
||||
assertEquals(RegionEventDescriptor.EventType.REGION_OPEN, desc.getEventType());
|
||||
assertTrue(Bytes.equals(desc.getTableName().toByteArray(), htd.getName()));
|
||||
assertTrue(Bytes.equals(desc.getEncodedRegionName().toByteArray(),
|
||||
hri.getEncodedNameAsBytes()));
|
||||
assertTrue(desc.getLogSequenceNumber() > 0);
|
||||
assertEquals(serverName, ProtobufUtil.toServerName(desc.getServer()));
|
||||
assertEquals(2, desc.getStoresCount());
|
||||
|
||||
StoreDescriptor store = desc.getStores(0);
|
||||
assertTrue(Bytes.equals(store.getFamilyName().toByteArray(), fam1));
|
||||
assertEquals(store.getStoreHomeDir(), Bytes.toString(fam1));
|
||||
assertEquals(1, store.getStoreFileCount()); // 1store file
|
||||
assertFalse(store.getStoreFile(0).contains("/")); // ensure path is relative
|
||||
|
||||
store = desc.getStores(1);
|
||||
assertTrue(Bytes.equals(store.getFamilyName().toByteArray(), fam2));
|
||||
assertEquals(store.getStoreHomeDir(), Bytes.toString(fam2));
|
||||
assertEquals(0, store.getStoreFileCount()); // no store files
|
||||
|
||||
} finally {
|
||||
HRegion.closeHRegion(region);
|
||||
}
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("unchecked")
|
||||
public void testCloseRegionWrittenToWAL() throws Exception {
|
||||
final ServerName serverName = ServerName.valueOf("testCloseRegionWrittenToWAL", 100, 42);
|
||||
final RegionServerServices rss = spy(TEST_UTIL.createMockRegionServerService(serverName));
|
||||
|
||||
HTableDescriptor htd
|
||||
= new HTableDescriptor(TableName.valueOf("testOpenRegionWrittenToWAL"));
|
||||
htd.addFamily(new HColumnDescriptor(fam1));
|
||||
htd.addFamily(new HColumnDescriptor(fam2));
|
||||
|
||||
HRegionInfo hri = new HRegionInfo(htd.getTableName(),
|
||||
HConstants.EMPTY_BYTE_ARRAY, HConstants.EMPTY_BYTE_ARRAY);
|
||||
|
||||
ArgumentCaptor<WALEdit> editCaptor = ArgumentCaptor.forClass(WALEdit.class);
|
||||
|
||||
// capture appendNoSync() calls
|
||||
HLog log = mock(HLog.class);
|
||||
when(rss.getWAL((HRegionInfo) any())).thenReturn(log);
|
||||
|
||||
// open a region first so that it can be closed later
|
||||
region = HRegion.openHRegion(hri, htd, rss.getWAL(hri),
|
||||
TEST_UTIL.getConfiguration(), rss, null);
|
||||
|
||||
// close the region
|
||||
region.close(false);
|
||||
|
||||
// 2 times, one for region open, the other close region
|
||||
verify(log, times(2)).appendNoSync((HTableDescriptor)any(), (HRegionInfo)any(), (HLogKey)any(),
|
||||
editCaptor.capture(), (AtomicLong)any(), anyBoolean(), (List<KeyValue>)any());
|
||||
|
||||
WALEdit edit = editCaptor.getAllValues().get(1);
|
||||
assertNotNull(edit);
|
||||
assertNotNull(edit.getKeyValues());
|
||||
assertEquals(1, edit.getKeyValues().size());
|
||||
RegionEventDescriptor desc = WALEdit.getRegionEventDescriptor(edit.getKeyValues().get(0));
|
||||
assertNotNull(desc);
|
||||
|
||||
LOG.info("RegionEventDescriptor from WAL: " + desc);
|
||||
|
||||
assertEquals(RegionEventDescriptor.EventType.REGION_CLOSE, desc.getEventType());
|
||||
assertTrue(Bytes.equals(desc.getTableName().toByteArray(), htd.getName()));
|
||||
assertTrue(Bytes.equals(desc.getEncodedRegionName().toByteArray(),
|
||||
hri.getEncodedNameAsBytes()));
|
||||
assertTrue(desc.getLogSequenceNumber() > 0);
|
||||
assertEquals(serverName, ProtobufUtil.toServerName(desc.getServer()));
|
||||
assertEquals(2, desc.getStoresCount());
|
||||
|
||||
StoreDescriptor store = desc.getStores(0);
|
||||
assertTrue(Bytes.equals(store.getFamilyName().toByteArray(), fam1));
|
||||
assertEquals(store.getStoreHomeDir(), Bytes.toString(fam1));
|
||||
assertEquals(0, store.getStoreFileCount()); // no store files
|
||||
|
||||
store = desc.getStores(1);
|
||||
assertTrue(Bytes.equals(store.getFamilyName().toByteArray(), fam2));
|
||||
assertEquals(store.getStoreHomeDir(), Bytes.toString(fam2));
|
||||
assertEquals(0, store.getStoreFileCount()); // no store files
|
||||
}
|
||||
|
||||
private static HRegion initHRegion(byte[] tableName, String callingMethod,
|
||||
byte[]... families) throws IOException {
|
||||
return initHRegion(tableName, callingMethod, HBaseConfiguration.create(),
|
||||
|
|
|
@ -21,6 +21,7 @@ package org.apache.hadoop.hbase.regionserver.wal;
|
|||
import static org.junit.Assert.assertEquals;
|
||||
import static org.junit.Assert.assertTrue;
|
||||
import static org.junit.Assert.fail;
|
||||
import static org.mockito.Mockito.when;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.security.PrivilegedExceptionAction;
|
||||
|
@ -100,7 +101,7 @@ public class TestWALReplay {
|
|||
private FileSystem fs;
|
||||
private Configuration conf;
|
||||
private RecoveryMode mode;
|
||||
|
||||
|
||||
|
||||
@BeforeClass
|
||||
public static void setUpBeforeClass() throws Exception {
|
||||
|
@ -131,7 +132,7 @@ public class TestWALReplay {
|
|||
if (TEST_UTIL.getDFSCluster().getFileSystem().exists(this.hbaseRootDir)) {
|
||||
TEST_UTIL.getDFSCluster().getFileSystem().delete(this.hbaseRootDir, true);
|
||||
}
|
||||
this.mode = (conf.getBoolean(HConstants.DISTRIBUTED_LOG_REPLAY_KEY, false) ?
|
||||
this.mode = (conf.getBoolean(HConstants.DISTRIBUTED_LOG_REPLAY_KEY, false) ?
|
||||
RecoveryMode.LOG_REPLAY : RecoveryMode.LOG_SPLITTING);
|
||||
}
|
||||
|
||||
|
@ -152,7 +153,7 @@ public class TestWALReplay {
|
|||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*
|
||||
* @throws Exception
|
||||
*/
|
||||
@Test
|
||||
|
@ -354,6 +355,7 @@ public class TestWALReplay {
|
|||
User user = HBaseTestingUtility.getDifferentUser(newConf,
|
||||
tableName.getNameAsString());
|
||||
user.runAs(new PrivilegedExceptionAction() {
|
||||
@Override
|
||||
public Object run() throws Exception {
|
||||
runWALSplit(newConf);
|
||||
HLog wal2 = createWAL(newConf);
|
||||
|
@ -425,6 +427,7 @@ public class TestWALReplay {
|
|||
User user = HBaseTestingUtility.getDifferentUser(newConf,
|
||||
tableName.getNameAsString());
|
||||
user.runAs(new PrivilegedExceptionAction() {
|
||||
@Override
|
||||
public Object run() throws Exception {
|
||||
runWALSplit(newConf);
|
||||
HLog wal2 = createWAL(newConf);
|
||||
|
@ -518,6 +521,7 @@ public class TestWALReplay {
|
|||
User user = HBaseTestingUtility.getDifferentUser(newConf,
|
||||
tableName.getNameAsString());
|
||||
user.runAs(new PrivilegedExceptionAction() {
|
||||
@Override
|
||||
public Object run() throws Exception {
|
||||
runWALSplit(newConf);
|
||||
FileSystem newFS = FileSystem.get(newConf);
|
||||
|
@ -669,6 +673,7 @@ public class TestWALReplay {
|
|||
HLog wal = createWAL(this.conf);
|
||||
RegionServerServices rsServices = Mockito.mock(RegionServerServices.class);
|
||||
Mockito.doReturn(false).when(rsServices).isAborted();
|
||||
when(rsServices.getServerName()).thenReturn(ServerName.valueOf("foo", 10, 10));
|
||||
Configuration customConf = new Configuration(this.conf);
|
||||
customConf.set(DefaultStoreEngine.DEFAULT_STORE_FLUSHER_CLASS_KEY,
|
||||
CustomStoreFlusher.class.getName());
|
||||
|
@ -802,6 +807,7 @@ public class TestWALReplay {
|
|||
User user = HBaseTestingUtility.getDifferentUser(newConf,
|
||||
".replay.wal.secondtime");
|
||||
user.runAs(new PrivilegedExceptionAction() {
|
||||
@Override
|
||||
public Object run() throws Exception {
|
||||
runWALSplit(newConf);
|
||||
FileSystem newFS = FileSystem.get(newConf);
|
||||
|
@ -813,6 +819,7 @@ public class TestWALReplay {
|
|||
try {
|
||||
final HRegion region =
|
||||
new HRegion(basedir, newWal, newFS, newConf, hri, htd, null) {
|
||||
@Override
|
||||
protected FlushResult internalFlushcache(
|
||||
final HLog wal, final long myseqid, MonitoredTask status)
|
||||
throws IOException {
|
||||
|
@ -886,7 +893,7 @@ public class TestWALReplay {
|
|||
for (FileStatus fileStatus : listStatus1) {
|
||||
editCount = Integer.parseInt(fileStatus.getPath().getName());
|
||||
}
|
||||
// The sequence number should be same
|
||||
// The sequence number should be same
|
||||
assertEquals(
|
||||
"The sequence number of the recoverd.edits and the current edit seq should be same",
|
||||
lastestSeqNumber, editCount);
|
||||
|
@ -914,7 +921,7 @@ public class TestWALReplay {
|
|||
htd.addFamily(a);
|
||||
return htd;
|
||||
}
|
||||
|
||||
|
||||
private MockHLog createMockWAL(Configuration conf) throws IOException {
|
||||
MockHLog wal = new MockHLog(FileSystem.get(conf), hbaseRootDir, logName, conf);
|
||||
// Set down maximum recovery so we dfsclient doesn't linger retrying something
|
||||
|
@ -940,7 +947,7 @@ public class TestWALReplay {
|
|||
@Override
|
||||
public void requestDelayedFlush(HRegion region, long when) {
|
||||
// TODO Auto-generated method stub
|
||||
|
||||
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -1021,7 +1028,7 @@ public class TestWALReplay {
|
|||
* @throws IOException
|
||||
*/
|
||||
private HLog createWAL(final Configuration c) throws IOException {
|
||||
HLog wal = HLogFactory.createHLog(FileSystem.get(c),
|
||||
HLog wal = HLogFactory.createHLog(FileSystem.get(c),
|
||||
hbaseRootDir, logName, c);
|
||||
// Set down maximum recovery so we dfsclient doesn't linger retrying something
|
||||
// long gone.
|
||||
|
|
Loading…
Reference in New Issue