HBASE-5121 MajorCompaction may affect scan's correctness (chunhui shen and Lars H)
git-svn-id: https://svn.apache.org/repos/asf/hbase/trunk@1229806 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
9cbf9e9f4e
commit
15c62ed21a
|
@ -476,6 +476,7 @@ Release 0.92.0 - Unreleased
|
||||||
HBASE-5152 Region is on service before completing initialization when doing rollback of split,
|
HBASE-5152 Region is on service before completing initialization when doing rollback of split,
|
||||||
it will affect read correctness (Chunhui)
|
it will affect read correctness (Chunhui)
|
||||||
HBASE-5137 MasterFileSystem.splitLog() should abort even if waitOnSafeMode() throws IOException(Ted)
|
HBASE-5137 MasterFileSystem.splitLog() should abort even if waitOnSafeMode() throws IOException(Ted)
|
||||||
|
HBASE-5121 MajorCompaction may affect scan's correctness (chunhui shen and Lars H)
|
||||||
|
|
||||||
TESTS
|
TESTS
|
||||||
HBASE-4450 test for number of blocks read: to serve as baseline for expected
|
HBASE-4450 test for number of blocks read: to serve as baseline for expected
|
||||||
|
|
|
@ -20,14 +20,14 @@
|
||||||
|
|
||||||
package org.apache.hadoop.hbase.regionserver;
|
package org.apache.hadoop.hbase.regionserver;
|
||||||
|
|
||||||
import org.apache.hadoop.hbase.KeyValue;
|
|
||||||
import org.apache.hadoop.hbase.KeyValue.KVComparator;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.PriorityQueue;
|
import java.util.PriorityQueue;
|
||||||
|
|
||||||
|
import org.apache.hadoop.hbase.KeyValue;
|
||||||
|
import org.apache.hadoop.hbase.KeyValue.KVComparator;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Implements a heap merge across any number of KeyValueScanners.
|
* Implements a heap merge across any number of KeyValueScanners.
|
||||||
* <p>
|
* <p>
|
||||||
|
@ -124,7 +124,7 @@ public class KeyValueHeap extends NonLazyKeyValueScanner
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
InternalScanner currentAsInternal = (InternalScanner)this.current;
|
InternalScanner currentAsInternal = (InternalScanner)this.current;
|
||||||
boolean mayContainsMoreRows = currentAsInternal.next(result, limit);
|
boolean mayContainMoreRows = currentAsInternal.next(result, limit);
|
||||||
KeyValue pee = this.current.peek();
|
KeyValue pee = this.current.peek();
|
||||||
/*
|
/*
|
||||||
* By definition, any InternalScanner must return false only when it has no
|
* By definition, any InternalScanner must return false only when it has no
|
||||||
|
@ -133,7 +133,7 @@ public class KeyValueHeap extends NonLazyKeyValueScanner
|
||||||
* more efficient to close scanners which are not needed than keep them in
|
* more efficient to close scanners which are not needed than keep them in
|
||||||
* the heap. This is also required for certain optimizations.
|
* the heap. This is also required for certain optimizations.
|
||||||
*/
|
*/
|
||||||
if (pee == null || !mayContainsMoreRows) {
|
if (pee == null || !mayContainMoreRows) {
|
||||||
this.current.close();
|
this.current.close();
|
||||||
} else {
|
} else {
|
||||||
this.heap.add(this.current);
|
this.heap.add(this.current);
|
||||||
|
|
|
@ -298,7 +298,9 @@ class StoreScanner extends NonLazyKeyValueScanner
|
||||||
@Override
|
@Override
|
||||||
public synchronized boolean next(List<KeyValue> outResult, int limit) throws IOException {
|
public synchronized boolean next(List<KeyValue> outResult, int limit) throws IOException {
|
||||||
|
|
||||||
checkReseek();
|
if (checkReseek()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// if the heap was left null, then the scanners had previously run out anyways, close and
|
// if the heap was left null, then the scanners had previously run out anyways, close and
|
||||||
// return.
|
// return.
|
||||||
|
@ -448,12 +450,25 @@ class StoreScanner extends NonLazyKeyValueScanner
|
||||||
// Let the next() call handle re-creating and seeking
|
// Let the next() call handle re-creating and seeking
|
||||||
}
|
}
|
||||||
|
|
||||||
private void checkReseek() throws IOException {
|
/**
|
||||||
|
* @return true if top of heap has changed (and KeyValueHeap has to try the
|
||||||
|
* next KV)
|
||||||
|
* @throws IOException
|
||||||
|
*/
|
||||||
|
private boolean checkReseek() throws IOException {
|
||||||
if (this.heap == null && this.lastTop != null) {
|
if (this.heap == null && this.lastTop != null) {
|
||||||
resetScannerStack(this.lastTop);
|
resetScannerStack(this.lastTop);
|
||||||
|
if (this.heap.peek() == null
|
||||||
|
|| store.comparator.compare(this.lastTop, this.heap.peek()) != 0) {
|
||||||
|
LOG.debug("Storescanner.peek() is changed where before = "
|
||||||
|
+ this.lastTop.toString() + ",and after = " + this.heap.peek());
|
||||||
|
this.lastTop = null;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
this.lastTop = null; // gone!
|
this.lastTop = null; // gone!
|
||||||
}
|
}
|
||||||
// else dont need to reseek
|
// else dont need to reseek
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void resetScannerStack(KeyValue lastTopKey) throws IOException {
|
private void resetScannerStack(KeyValue lastTopKey) throws IOException {
|
||||||
|
|
|
@ -27,7 +27,15 @@ import java.util.List;
|
||||||
|
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
import org.apache.hadoop.hbase.*;
|
import org.apache.hadoop.hbase.HBaseTestCase;
|
||||||
|
import org.apache.hadoop.hbase.HColumnDescriptor;
|
||||||
|
import org.apache.hadoop.hbase.HConstants;
|
||||||
|
import org.apache.hadoop.hbase.HRegionInfo;
|
||||||
|
import org.apache.hadoop.hbase.HTableDescriptor;
|
||||||
|
import org.apache.hadoop.hbase.KeyValue;
|
||||||
|
import org.apache.hadoop.hbase.SmallTests;
|
||||||
|
import org.apache.hadoop.hbase.UnknownScannerException;
|
||||||
|
import org.apache.hadoop.hbase.client.Delete;
|
||||||
import org.apache.hadoop.hbase.client.Get;
|
import org.apache.hadoop.hbase.client.Get;
|
||||||
import org.apache.hadoop.hbase.client.Put;
|
import org.apache.hadoop.hbase.client.Put;
|
||||||
import org.apache.hadoop.hbase.client.Result;
|
import org.apache.hadoop.hbase.client.Result;
|
||||||
|
@ -39,7 +47,6 @@ import org.apache.hadoop.hbase.filter.WhileMatchFilter;
|
||||||
import org.apache.hadoop.hbase.io.hfile.Compression;
|
import org.apache.hadoop.hbase.io.hfile.Compression;
|
||||||
import org.apache.hadoop.hbase.util.Bytes;
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
import org.apache.hadoop.hbase.util.Writables;
|
import org.apache.hadoop.hbase.util.Writables;
|
||||||
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
|
||||||
import org.junit.experimental.categories.Category;
|
import org.junit.experimental.categories.Category;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -78,6 +85,23 @@ public class TestScanner extends HBaseTestCase {
|
||||||
private HRegion r;
|
private HRegion r;
|
||||||
private HRegionIncommon region;
|
private HRegionIncommon region;
|
||||||
|
|
||||||
|
private byte[] firstRowBytes, secondRowBytes, thirdRowBytes;
|
||||||
|
final private byte[] col1, col2;
|
||||||
|
|
||||||
|
public TestScanner() throws Exception {
|
||||||
|
super();
|
||||||
|
|
||||||
|
firstRowBytes = START_KEY.getBytes(HConstants.UTF8_ENCODING);
|
||||||
|
secondRowBytes = START_KEY.getBytes(HConstants.UTF8_ENCODING);
|
||||||
|
// Increment the least significant character so we get to next row.
|
||||||
|
secondRowBytes[START_KEY_BYTES.length - 1]++;
|
||||||
|
thirdRowBytes = START_KEY.getBytes(HConstants.UTF8_ENCODING);
|
||||||
|
thirdRowBytes[START_KEY_BYTES.length - 1]++;
|
||||||
|
thirdRowBytes[START_KEY_BYTES.length - 1]++;
|
||||||
|
col1 = "column1".getBytes(HConstants.UTF8_ENCODING);
|
||||||
|
col2 = "column2".getBytes(HConstants.UTF8_ENCODING);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test basic stop row filter works.
|
* Test basic stop row filter works.
|
||||||
* @throws Exception
|
* @throws Exception
|
||||||
|
@ -466,6 +490,68 @@ public class TestScanner extends HBaseTestCase {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make sure scanner returns correct result when we run a major compaction
|
||||||
|
* with deletes.
|
||||||
|
*
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("deprecation")
|
||||||
|
public void testScanAndConcurrentMajorCompact() throws Exception {
|
||||||
|
HTableDescriptor htd = createTableDescriptor(getName());
|
||||||
|
this.r = createNewHRegion(htd, null, null);
|
||||||
|
HRegionIncommon hri = new HRegionIncommon(r);
|
||||||
|
|
||||||
|
try {
|
||||||
|
addContent(hri, Bytes.toString(fam1), Bytes.toString(col1),
|
||||||
|
firstRowBytes, secondRowBytes);
|
||||||
|
addContent(hri, Bytes.toString(fam2), Bytes.toString(col1),
|
||||||
|
firstRowBytes, secondRowBytes);
|
||||||
|
|
||||||
|
Delete dc = new Delete(firstRowBytes);
|
||||||
|
/* delete column1 of firstRow */
|
||||||
|
dc.deleteColumns(fam1, col1);
|
||||||
|
r.delete(dc, null, true);
|
||||||
|
r.flushcache();
|
||||||
|
|
||||||
|
addContent(hri, Bytes.toString(fam1), Bytes.toString(col1),
|
||||||
|
secondRowBytes, thirdRowBytes);
|
||||||
|
addContent(hri, Bytes.toString(fam2), Bytes.toString(col1),
|
||||||
|
secondRowBytes, thirdRowBytes);
|
||||||
|
r.flushcache();
|
||||||
|
|
||||||
|
InternalScanner s = r.getScanner(new Scan());
|
||||||
|
// run a major compact, column1 of firstRow will be cleaned.
|
||||||
|
r.compactStores(true);
|
||||||
|
|
||||||
|
List<KeyValue> results = new ArrayList<KeyValue>();
|
||||||
|
s.next(results);
|
||||||
|
|
||||||
|
// make sure returns column2 of firstRow
|
||||||
|
assertTrue("result is not correct, keyValues : " + results,
|
||||||
|
results.size() == 1);
|
||||||
|
assertTrue(Bytes.BYTES_COMPARATOR.compare(firstRowBytes, results.get(0)
|
||||||
|
.getRow()) == 0);
|
||||||
|
assertTrue(Bytes.BYTES_COMPARATOR.compare(fam2, results.get(0)
|
||||||
|
.getFamily()) == 0);
|
||||||
|
|
||||||
|
results = new ArrayList<KeyValue>();
|
||||||
|
s.next(results);
|
||||||
|
|
||||||
|
// get secondRow
|
||||||
|
assertTrue(results.size() == 2);
|
||||||
|
assertTrue(Bytes.BYTES_COMPARATOR.compare(secondRowBytes, results.get(0)
|
||||||
|
.getRow()) == 0);
|
||||||
|
assertTrue(Bytes.BYTES_COMPARATOR.compare(fam1, results.get(0)
|
||||||
|
.getFamily()) == 0);
|
||||||
|
assertTrue(Bytes.BYTES_COMPARATOR.compare(fam2, results.get(1)
|
||||||
|
.getFamily()) == 0);
|
||||||
|
} finally {
|
||||||
|
this.r.close();
|
||||||
|
this.r.getLog().closeAndDelete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @param hri Region
|
* @param hri Region
|
||||||
|
|
Loading…
Reference in New Issue