HBASE-2733. Replacement of LATEST_TIMESTAMP with real timestamp was broken by HBASE-2353
git-svn-id: https://svn.apache.org/repos/asf/hbase/trunk@955077 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
fef909299a
commit
609ce409a0
|
@ -388,6 +388,8 @@ Release 0.21.0 - Unreleased
|
||||||
HBASE-2732 TestZooKeeper was broken, HBASE-2691 showed it
|
HBASE-2732 TestZooKeeper was broken, HBASE-2691 showed it
|
||||||
HBASE-2670 Provide atomicity for readers even when new insert has
|
HBASE-2670 Provide atomicity for readers even when new insert has
|
||||||
same timestamp as current row.
|
same timestamp as current row.
|
||||||
|
HBASE-2733 Replacement of LATEST_TIMESTAMP with real timestamp was broken
|
||||||
|
by HBASE-2353.
|
||||||
|
|
||||||
IMPROVEMENTS
|
IMPROVEMENTS
|
||||||
HBASE-1760 Cleanup TODOs in HTable
|
HBASE-1760 Cleanup TODOs in HTable
|
||||||
|
|
|
@ -1327,7 +1327,7 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
// bunch up all edits across all column families into a
|
// bunch up all edits across all column families into a
|
||||||
// single WALEdit.
|
// single WALEdit.
|
||||||
WALEdit walEdit = new WALEdit();
|
WALEdit walEdit = new WALEdit();
|
||||||
addFamilyMapToWALEdit(familyMap, byteNow, walEdit);
|
addFamilyMapToWALEdit(familyMap, walEdit);
|
||||||
this.log.append(regionInfo, regionInfo.getTableDesc().getName(),
|
this.log.append(regionInfo, regionInfo.getTableDesc().getName(),
|
||||||
walEdit, now);
|
walEdit, now);
|
||||||
}
|
}
|
||||||
|
@ -1522,7 +1522,16 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
assert numReadyToWrite > 0;
|
assert numReadyToWrite > 0;
|
||||||
|
|
||||||
// ------------------------------------
|
// ------------------------------------
|
||||||
// STEP 2. Write to WAL
|
// STEP 2. Update any LATEST_TIMESTAMP timestamps
|
||||||
|
// ----------------------------------
|
||||||
|
for (int i = firstIndex; i < lastIndexExclusive; i++) {
|
||||||
|
updateKVTimestamps(
|
||||||
|
batchOp.operations[i].getFirst().getFamilyMap().values(),
|
||||||
|
byteNow);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------
|
||||||
|
// STEP 3. Write to WAL
|
||||||
// ----------------------------------
|
// ----------------------------------
|
||||||
WALEdit walEdit = new WALEdit();
|
WALEdit walEdit = new WALEdit();
|
||||||
for (int i = firstIndex; i < lastIndexExclusive; i++) {
|
for (int i = firstIndex; i < lastIndexExclusive; i++) {
|
||||||
|
@ -1531,7 +1540,7 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
|
|
||||||
Put p = batchOp.operations[i].getFirst();
|
Put p = batchOp.operations[i].getFirst();
|
||||||
if (!p.getWriteToWAL()) continue;
|
if (!p.getWriteToWAL()) continue;
|
||||||
addFamilyMapToWALEdit(p.getFamilyMap(), byteNow, walEdit);
|
addFamilyMapToWALEdit(p.getFamilyMap(), walEdit);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Append the edit to WAL
|
// Append the edit to WAL
|
||||||
|
@ -1539,7 +1548,7 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
walEdit, now);
|
walEdit, now);
|
||||||
|
|
||||||
// ------------------------------------
|
// ------------------------------------
|
||||||
// STEP 3. Write back to memstore
|
// STEP 4. Write back to memstore
|
||||||
// ----------------------------------
|
// ----------------------------------
|
||||||
long addedSize = 0;
|
long addedSize = 0;
|
||||||
for (int i = firstIndex; i < lastIndexExclusive; i++) {
|
for (int i = firstIndex; i < lastIndexExclusive; i++) {
|
||||||
|
@ -1636,21 +1645,17 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if any stamps is Long.MAX_VALUE. If so, sets them to now.
|
* Replaces any KV timestamps set to {@link HConstants#LATEST_TIMESTAMP}
|
||||||
* <p>
|
* with the provided current timestamp.
|
||||||
* This acts to replace {@link HConstants#LATEST_TIMESTAMP} with {@code now}.
|
|
||||||
* @param keys
|
|
||||||
* @param now
|
|
||||||
* @return <code>true</code> when updating the time stamp completed.
|
|
||||||
*/
|
*/
|
||||||
private boolean updateKeys(final List<KeyValue> keys, final byte[] now) {
|
private void updateKVTimestamps(
|
||||||
if (keys == null || keys.isEmpty()) {
|
final Iterable<List<KeyValue>> keyLists, final byte[] now) {
|
||||||
return false;
|
for (List<KeyValue> keys: keyLists) {
|
||||||
|
if (keys == null) continue;
|
||||||
|
for (KeyValue key : keys) {
|
||||||
|
key.updateLatestStamp(now);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
for (KeyValue key : keys) {
|
|
||||||
key.updateLatestStamp(now);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// /*
|
// /*
|
||||||
|
@ -1754,7 +1759,7 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
this.updatesLock.readLock().lock();
|
this.updatesLock.readLock().lock();
|
||||||
try {
|
try {
|
||||||
checkFamilies(familyMap.keySet());
|
checkFamilies(familyMap.keySet());
|
||||||
|
updateKVTimestamps(familyMap.values(), byteNow);
|
||||||
// write/sync to WAL should happen before we touch memstore.
|
// write/sync to WAL should happen before we touch memstore.
|
||||||
//
|
//
|
||||||
// If order is reversed, i.e. we write to memstore first, and
|
// If order is reversed, i.e. we write to memstore first, and
|
||||||
|
@ -1762,7 +1767,7 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
// will contain uncommitted transactions.
|
// will contain uncommitted transactions.
|
||||||
if (writeToWAL) {
|
if (writeToWAL) {
|
||||||
WALEdit walEdit = new WALEdit();
|
WALEdit walEdit = new WALEdit();
|
||||||
addFamilyMapToWALEdit(familyMap, byteNow, walEdit);
|
addFamilyMapToWALEdit(familyMap, walEdit);
|
||||||
this.log.append(regionInfo, regionInfo.getTableDesc().getName(),
|
this.log.append(regionInfo, regionInfo.getTableDesc().getName(),
|
||||||
walEdit, now);
|
walEdit, now);
|
||||||
}
|
}
|
||||||
|
@ -1822,22 +1827,15 @@ public class HRegion implements HeapSize { // , Writable{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Append the given map of family->edits to a WALEdit data structure.
|
* Append the given map of family->edits to a WALEdit data structure.
|
||||||
* Also updates the timestamps of the edits where they have not
|
* This does not write to the HLog itself.
|
||||||
* been specified by the user. This does not write to the HLog itself.
|
|
||||||
* @param familyMap map of family->edits
|
* @param familyMap map of family->edits
|
||||||
* @param byteNow timestamp to use when unspecified
|
|
||||||
* @param walEdit the destination entry to append into
|
* @param walEdit the destination entry to append into
|
||||||
*/
|
*/
|
||||||
private void addFamilyMapToWALEdit(Map<byte[], List<KeyValue>> familyMap,
|
private void addFamilyMapToWALEdit(Map<byte[], List<KeyValue>> familyMap,
|
||||||
byte[] byteNow, WALEdit walEdit) {
|
WALEdit walEdit) {
|
||||||
for (List<KeyValue> edits : familyMap.values()) {
|
for (List<KeyValue> edits : familyMap.values()) {
|
||||||
// update timestamp on keys if required.
|
for (KeyValue kv : edits) {
|
||||||
if (updateKeys(edits, byteNow)) {
|
walEdit.add(kv);
|
||||||
// bunch up all edits across all column families into a
|
|
||||||
// single WALEdit.
|
|
||||||
for (KeyValue kv : edits) {
|
|
||||||
walEdit.add(kv);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -345,6 +345,7 @@ public class TestHRegion extends HBaseTestCase {
|
||||||
byte[] val = Bytes.toBytes("val");
|
byte[] val = Bytes.toBytes("val");
|
||||||
initHRegion(b, getName(), cf);
|
initHRegion(b, getName(), cf);
|
||||||
|
|
||||||
|
HLog.getSyncOps(); // clear counter from prior tests
|
||||||
assertEquals(0, HLog.getSyncOps());
|
assertEquals(0, HLog.getSyncOps());
|
||||||
|
|
||||||
LOG.info("First a batch put with all valid puts");
|
LOG.info("First a batch put with all valid puts");
|
||||||
|
@ -834,6 +835,53 @@ public class TestHRegion extends HBaseTestCase {
|
||||||
assertEquals(0, result.size());
|
assertEquals(0, result.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests that the special LATEST_TIMESTAMP option for puts gets
|
||||||
|
* replaced by the actual timestamp
|
||||||
|
*/
|
||||||
|
public void testPutWithLatestTS() throws IOException {
|
||||||
|
byte [] tableName = Bytes.toBytes("testtable");
|
||||||
|
byte [] fam = Bytes.toBytes("info");
|
||||||
|
byte [][] families = {fam};
|
||||||
|
String method = this.getName();
|
||||||
|
initHRegion(tableName, method, families);
|
||||||
|
|
||||||
|
byte [] row = Bytes.toBytes("row1");
|
||||||
|
// column names
|
||||||
|
byte [] qual = Bytes.toBytes("qual");
|
||||||
|
|
||||||
|
// add data with LATEST_TIMESTAMP, put without WAL
|
||||||
|
Put put = new Put(row);
|
||||||
|
put.add(fam, qual, HConstants.LATEST_TIMESTAMP, Bytes.toBytes("value"));
|
||||||
|
region.put(put, false);
|
||||||
|
|
||||||
|
// Make sure it shows up with an actual timestamp
|
||||||
|
Get get = new Get(row).addColumn(fam, qual);
|
||||||
|
Result result = region.get(get, null);
|
||||||
|
assertEquals(1, result.size());
|
||||||
|
KeyValue kv = result.raw()[0];
|
||||||
|
LOG.info("Got: " + kv);
|
||||||
|
assertTrue("LATEST_TIMESTAMP was not replaced with real timestamp",
|
||||||
|
kv.getTimestamp() != HConstants.LATEST_TIMESTAMP);
|
||||||
|
|
||||||
|
// Check same with WAL enabled (historically these took different
|
||||||
|
// code paths, so check both)
|
||||||
|
row = Bytes.toBytes("row2");
|
||||||
|
put = new Put(row);
|
||||||
|
put.add(fam, qual, HConstants.LATEST_TIMESTAMP, Bytes.toBytes("value"));
|
||||||
|
region.put(put, true);
|
||||||
|
|
||||||
|
// Make sure it shows up with an actual timestamp
|
||||||
|
get = new Get(row).addColumn(fam, qual);
|
||||||
|
result = region.get(get, null);
|
||||||
|
assertEquals(1, result.size());
|
||||||
|
kv = result.raw()[0];
|
||||||
|
LOG.info("Got: " + kv);
|
||||||
|
assertTrue("LATEST_TIMESTAMP was not replaced with real timestamp",
|
||||||
|
kv.getTimestamp() != HConstants.LATEST_TIMESTAMP);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
public void testScanner_DeleteOneFamilyNotAnother() throws IOException {
|
public void testScanner_DeleteOneFamilyNotAnother() throws IOException {
|
||||||
byte [] tableName = Bytes.toBytes("test_table");
|
byte [] tableName = Bytes.toBytes("test_table");
|
||||||
byte [] fam1 = Bytes.toBytes("columnA");
|
byte [] fam1 = Bytes.toBytes("columnA");
|
||||||
|
|
Loading…
Reference in New Issue