Avoid side-effect in VersionMap when assertion enabled (#29585)

Today when a version map does not require safe access, we will skip that
document. However, if the assertion is enabled, we remove the delete
tombstone of that document if existed. This side-effect may accidentally
hide bugs in which stale delete tombstone can be accessed.

This change ensures putAssertionMap not modify the tombstone maps.
This commit is contained in:
Nhat Nguyen 2018-04-19 12:38:10 -04:00 committed by GitHub
parent 7fa7dea044
commit 1b24d4e68b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 84 additions and 73 deletions

View File

@ -23,7 +23,7 @@ import org.apache.lucene.util.RamUsageEstimator;
/** Holds a deleted version, which just adds a timestamp to {@link VersionValue} so we know when we can expire the deletion. */ /** Holds a deleted version, which just adds a timestamp to {@link VersionValue} so we know when we can expire the deletion. */
class DeleteVersionValue extends VersionValue { final class DeleteVersionValue extends VersionValue {
private static final long BASE_RAM_BYTES_USED = RamUsageEstimator.shallowSizeOfInstance(DeleteVersionValue.class); private static final long BASE_RAM_BYTES_USED = RamUsageEstimator.shallowSizeOfInstance(DeleteVersionValue.class);

View File

@ -24,13 +24,13 @@ import org.elasticsearch.index.translog.Translog;
import java.util.Objects; import java.util.Objects;
final class TranslogVersionValue extends VersionValue { final class IndexVersionValue extends VersionValue {
private static final long RAM_BYTES_USED = RamUsageEstimator.shallowSizeOfInstance(TranslogVersionValue.class); private static final long RAM_BYTES_USED = RamUsageEstimator.shallowSizeOfInstance(IndexVersionValue.class);
private final Translog.Location translogLocation; private final Translog.Location translogLocation;
TranslogVersionValue(Translog.Location translogLocation, long version, long seqNo, long term) { IndexVersionValue(Translog.Location translogLocation, long version, long seqNo, long term) {
super(version, seqNo, term); super(version, seqNo, term);
this.translogLocation = translogLocation; this.translogLocation = translogLocation;
} }
@ -45,7 +45,7 @@ final class TranslogVersionValue extends VersionValue {
if (this == o) return true; if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false; if (o == null || getClass() != o.getClass()) return false;
if (!super.equals(o)) return false; if (!super.equals(o)) return false;
TranslogVersionValue that = (TranslogVersionValue) o; IndexVersionValue that = (IndexVersionValue) o;
return Objects.equals(translogLocation, that.translogLocation); return Objects.equals(translogLocation, that.translogLocation);
} }
@ -56,7 +56,7 @@ final class TranslogVersionValue extends VersionValue {
@Override @Override
public String toString() { public String toString() {
return "TranslogVersionValue{" + return "IndexVersionValue{" +
"version=" + version + "version=" + version +
", seqNo=" + seqNo + ", seqNo=" + seqNo +
", term=" + term + ", term=" + term +

View File

@ -623,7 +623,7 @@ public class InternalEngine extends Engine {
assert incrementIndexVersionLookup(); // used for asserting in tests assert incrementIndexVersionLookup(); // used for asserting in tests
final long currentVersion = loadCurrentVersionFromIndex(op.uid()); final long currentVersion = loadCurrentVersionFromIndex(op.uid());
if (currentVersion != Versions.NOT_FOUND) { if (currentVersion != Versions.NOT_FOUND) {
versionValue = new VersionValue(currentVersion, SequenceNumbers.UNASSIGNED_SEQ_NO, 0L); versionValue = new IndexVersionValue(null, currentVersion, SequenceNumbers.UNASSIGNED_SEQ_NO, 0L);
} }
} else if (engineConfig.isEnableGcDeletes() && versionValue.isDelete() && } else if (engineConfig.isEnableGcDeletes() && versionValue.isDelete() &&
(engineConfig.getThreadPool().relativeTimeInMillis() - ((DeleteVersionValue)versionValue).time) > getGcDeletesInMillis()) { (engineConfig.getThreadPool().relativeTimeInMillis() - ((DeleteVersionValue)versionValue).time) > getGcDeletesInMillis()) {
@ -785,8 +785,9 @@ public class InternalEngine extends Engine {
indexResult.setTranslogLocation(location); indexResult.setTranslogLocation(location);
} }
if (plan.indexIntoLucene && indexResult.hasFailure() == false) { if (plan.indexIntoLucene && indexResult.hasFailure() == false) {
versionMap.maybePutUnderLock(index.uid().bytes(), final Translog.Location translogLocation = trackTranslogLocation.get() ? indexResult.getTranslogLocation() : null;
getVersionValue(plan.versionForIndexing, plan.seqNoForIndexing, index.primaryTerm(), indexResult.getTranslogLocation())); versionMap.maybePutIndexUnderLock(index.uid().bytes(),
new IndexVersionValue(translogLocation, plan.versionForIndexing, plan.seqNoForIndexing, index.primaryTerm()));
} }
if (indexResult.getSeqNo() != SequenceNumbers.UNASSIGNED_SEQ_NO) { if (indexResult.getSeqNo() != SequenceNumbers.UNASSIGNED_SEQ_NO) {
localCheckpointTracker.markSeqNoAsCompleted(indexResult.getSeqNo()); localCheckpointTracker.markSeqNoAsCompleted(indexResult.getSeqNo());
@ -937,13 +938,6 @@ public class InternalEngine extends Engine {
} }
} }
private VersionValue getVersionValue(long version, long seqNo, long term, Translog.Location location) {
if (location != null && trackTranslogLocation.get()) {
return new TranslogVersionValue(location, version, seqNo, term);
}
return new VersionValue(version, seqNo, term);
}
/** /**
* returns true if the indexing operation may have already be processed by this engine. * returns true if the indexing operation may have already be processed by this engine.
* Note that it is OK to rarely return true even if this is not the case. However a `false` * Note that it is OK to rarely return true even if this is not the case. However a `false`
@ -1193,7 +1187,7 @@ public class InternalEngine extends Engine {
indexWriter.deleteDocuments(delete.uid()); indexWriter.deleteDocuments(delete.uid());
numDocDeletes.inc(); numDocDeletes.inc();
} }
versionMap.putUnderLock(delete.uid().bytes(), versionMap.putDeleteUnderLock(delete.uid().bytes(),
new DeleteVersionValue(plan.versionOfDeletion, plan.seqNoOfDeletion, delete.primaryTerm(), new DeleteVersionValue(plan.versionOfDeletion, plan.seqNoOfDeletion, delete.primaryTerm(),
engineConfig.getThreadPool().relativeTimeInMillis())); engineConfig.getThreadPool().relativeTimeInMillis()));
return new DeleteResult( return new DeleteResult(

View File

@ -268,7 +268,7 @@ final class LiveVersionMap implements ReferenceManager.RefreshListener, Accounta
} }
private VersionValue getUnderLock(final BytesRef uid, Maps currentMaps) { private VersionValue getUnderLock(final BytesRef uid, Maps currentMaps) {
assert keyedLock.isHeldByCurrentThread(uid); assert assertKeyedLockHeldByCurrentThread(uid);
// First try to get the "live" value: // First try to get the "live" value:
VersionValue value = currentMaps.current.get(uid); VersionValue value = currentMaps.current.get(uid);
if (value != null) { if (value != null) {
@ -306,44 +306,36 @@ final class LiveVersionMap implements ReferenceManager.RefreshListener, Accounta
/** /**
* Adds this uid/version to the pending adds map iff the map needs safe access. * Adds this uid/version to the pending adds map iff the map needs safe access.
*/ */
void maybePutUnderLock(BytesRef uid, VersionValue version) { void maybePutIndexUnderLock(BytesRef uid, IndexVersionValue version) {
assert keyedLock.isHeldByCurrentThread(uid); assert assertKeyedLockHeldByCurrentThread(uid);
Maps maps = this.maps; Maps maps = this.maps;
if (maps.isSafeAccessMode()) { if (maps.isSafeAccessMode()) {
putUnderLock(uid, version, maps); putIndexUnderLock(uid, version);
} else { } else {
maps.current.markAsUnsafe(); maps.current.markAsUnsafe();
assert putAssertionMap(uid, version); assert putAssertionMap(uid, version);
} }
} }
private boolean putAssertionMap(BytesRef uid, VersionValue version) { void putIndexUnderLock(BytesRef uid, IndexVersionValue version) {
putUnderLock(uid, version, unsafeKeysMap); assert assertKeyedLockHeldByCurrentThread(uid);
assert uid.bytes.length == uid.length : "Oversized _uid! UID length: " + uid.length + ", bytes length: " + uid.bytes.length;
maps.put(uid, version);
removeTombstoneUnderLock(uid);
}
private boolean putAssertionMap(BytesRef uid, IndexVersionValue version) {
assert assertKeyedLockHeldByCurrentThread(uid);
assert uid.bytes.length == uid.length : "Oversized _uid! UID length: " + uid.length + ", bytes length: " + uid.bytes.length;
unsafeKeysMap.put(uid, version);
return true; return true;
} }
/** void putDeleteUnderLock(BytesRef uid, DeleteVersionValue version) {
* Adds this uid/version to the pending adds map. assert assertKeyedLockHeldByCurrentThread(uid);
*/
void putUnderLock(BytesRef uid, VersionValue version) {
Maps maps = this.maps;
putUnderLock(uid, version, maps);
}
/**
* Adds this uid/version to the pending adds map.
*/
private void putUnderLock(BytesRef uid, VersionValue version, Maps maps) {
assert keyedLock.isHeldByCurrentThread(uid);
assert uid.bytes.length == uid.length : "Oversized _uid! UID length: " + uid.length + ", bytes length: " + uid.bytes.length; assert uid.bytes.length == uid.length : "Oversized _uid! UID length: " + uid.length + ", bytes length: " + uid.bytes.length;
if (version.isDelete() == false) { putTombstone(uid, version);
maps.put(uid, version); maps.remove(uid, version);
removeTombstoneUnderLock(uid);
} else {
DeleteVersionValue versionValue = (DeleteVersionValue) version;
putTombstone(uid, versionValue);
maps.remove(uid, versionValue);
}
} }
private void putTombstone(BytesRef uid, DeleteVersionValue version) { private void putTombstone(BytesRef uid, DeleteVersionValue version) {
@ -365,7 +357,7 @@ final class LiveVersionMap implements ReferenceManager.RefreshListener, Accounta
* Removes this uid from the pending deletes map. * Removes this uid from the pending deletes map.
*/ */
void removeTombstoneUnderLock(BytesRef uid) { void removeTombstoneUnderLock(BytesRef uid) {
assert keyedLock.isHeldByCurrentThread(uid); assert assertKeyedLockHeldByCurrentThread(uid);
long uidRAMBytesUsed = BASE_BYTES_PER_BYTESREF + uid.bytes.length; long uidRAMBytesUsed = BASE_BYTES_PER_BYTESREF + uid.bytes.length;
final VersionValue prev = tombstones.remove(uid); final VersionValue prev = tombstones.remove(uid);
if (prev != null) { if (prev != null) {
@ -465,4 +457,9 @@ final class LiveVersionMap implements ReferenceManager.RefreshListener, Accounta
Releasable acquireLock(BytesRef uid) { Releasable acquireLock(BytesRef uid) {
return keyedLock.acquire(uid); return keyedLock.acquire(uid);
} }
private boolean assertKeyedLockHeldByCurrentThread(BytesRef uid) {
assert keyedLock.isHeldByCurrentThread(uid) : "Thread [" + Thread.currentThread().getName() + "], uid [" + uid.utf8ToString() + "]";
return true;
}
} }

View File

@ -27,7 +27,7 @@ import org.elasticsearch.index.translog.Translog;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
class VersionValue implements Accountable { abstract class VersionValue implements Accountable {
private static final long BASE_RAM_BYTES_USED = RamUsageEstimator.shallowSizeOfInstance(VersionValue.class); private static final long BASE_RAM_BYTES_USED = RamUsageEstimator.shallowSizeOfInstance(VersionValue.class);

View File

@ -25,6 +25,7 @@ import org.apache.lucene.util.Constants;
import org.apache.lucene.util.RamUsageTester; import org.apache.lucene.util.RamUsageTester;
import org.apache.lucene.util.TestUtil; import org.apache.lucene.util.TestUtil;
import org.elasticsearch.common.lease.Releasable; import org.elasticsearch.common.lease.Releasable;
import org.elasticsearch.index.translog.Translog;
import org.elasticsearch.test.ESTestCase; import org.elasticsearch.test.ESTestCase;
import java.io.IOException; import java.io.IOException;
@ -47,9 +48,8 @@ public class LiveVersionMapTests extends ESTestCase {
for (int i = 0; i < 100000; ++i) { for (int i = 0; i < 100000; ++i) {
BytesRefBuilder uid = new BytesRefBuilder(); BytesRefBuilder uid = new BytesRefBuilder();
uid.copyChars(TestUtil.randomSimpleString(random(), 10, 20)); uid.copyChars(TestUtil.randomSimpleString(random(), 10, 20));
VersionValue version = new VersionValue(randomLong(), randomLong(), randomLong());
try (Releasable r = map.acquireLock(uid.toBytesRef())) { try (Releasable r = map.acquireLock(uid.toBytesRef())) {
map.putUnderLock(uid.toBytesRef(), version); map.putIndexUnderLock(uid.toBytesRef(), randomIndexVersionValue());
} }
} }
long actualRamBytesUsed = RamUsageTester.sizeOf(map); long actualRamBytesUsed = RamUsageTester.sizeOf(map);
@ -64,9 +64,8 @@ public class LiveVersionMapTests extends ESTestCase {
for (int i = 0; i < 100000; ++i) { for (int i = 0; i < 100000; ++i) {
BytesRefBuilder uid = new BytesRefBuilder(); BytesRefBuilder uid = new BytesRefBuilder();
uid.copyChars(TestUtil.randomSimpleString(random(), 10, 20)); uid.copyChars(TestUtil.randomSimpleString(random(), 10, 20));
VersionValue version = new VersionValue(randomLong(), randomLong(), randomLong());
try (Releasable r = map.acquireLock(uid.toBytesRef())) { try (Releasable r = map.acquireLock(uid.toBytesRef())) {
map.putUnderLock(uid.toBytesRef(), version); map.putIndexUnderLock(uid.toBytesRef(), randomIndexVersionValue());
} }
} }
actualRamBytesUsed = RamUsageTester.sizeOf(map); actualRamBytesUsed = RamUsageTester.sizeOf(map);
@ -100,14 +99,15 @@ public class LiveVersionMapTests extends ESTestCase {
public void testBasics() throws IOException { public void testBasics() throws IOException {
LiveVersionMap map = new LiveVersionMap(); LiveVersionMap map = new LiveVersionMap();
try (Releasable r = map.acquireLock(uid("test"))) { try (Releasable r = map.acquireLock(uid("test"))) {
map.putUnderLock(uid("test"), new VersionValue(1,1,1)); Translog.Location tlogLoc = randomTranslogLocation();
assertEquals(new VersionValue(1,1,1), map.getUnderLock(uid("test"))); map.putIndexUnderLock(uid("test"), new IndexVersionValue(tlogLoc, 1, 1, 1));
assertEquals(new IndexVersionValue(tlogLoc, 1, 1, 1), map.getUnderLock(uid("test")));
map.beforeRefresh(); map.beforeRefresh();
assertEquals(new VersionValue(1,1,1), map.getUnderLock(uid("test"))); assertEquals(new IndexVersionValue(tlogLoc, 1, 1, 1), map.getUnderLock(uid("test")));
map.afterRefresh(randomBoolean()); map.afterRefresh(randomBoolean());
assertNull(map.getUnderLock(uid("test"))); assertNull(map.getUnderLock(uid("test")));
map.putUnderLock(uid("test"), new DeleteVersionValue(1,1,1,1)); map.putDeleteUnderLock(uid("test"), new DeleteVersionValue(1,1,1,1));
assertEquals(new DeleteVersionValue(1,1,1,1), map.getUnderLock(uid("test"))); assertEquals(new DeleteVersionValue(1,1,1,1), map.getUnderLock(uid("test")));
map.beforeRefresh(); map.beforeRefresh();
assertEquals(new DeleteVersionValue(1,1,1,1), map.getUnderLock(uid("test"))); assertEquals(new DeleteVersionValue(1,1,1,1), map.getUnderLock(uid("test")));
@ -154,21 +154,24 @@ public class LiveVersionMapTests extends ESTestCase {
BytesRef bytesRef = randomFrom(random(), keyList); BytesRef bytesRef = randomFrom(random(), keyList);
try (Releasable r = map.acquireLock(bytesRef)) { try (Releasable r = map.acquireLock(bytesRef)) {
VersionValue versionValue = values.computeIfAbsent(bytesRef, VersionValue versionValue = values.computeIfAbsent(bytesRef,
v -> new VersionValue(randomLong(), maxSeqNo.incrementAndGet(), randomLong())); v -> new IndexVersionValue(
randomTranslogLocation(), randomLong(), maxSeqNo.incrementAndGet(), randomLong()));
boolean isDelete = versionValue instanceof DeleteVersionValue; boolean isDelete = versionValue instanceof DeleteVersionValue;
if (isDelete) { if (isDelete) {
map.removeTombstoneUnderLock(bytesRef); map.removeTombstoneUnderLock(bytesRef);
deletes.remove(bytesRef); deletes.remove(bytesRef);
} }
if (isDelete == false && rarely()) { if (isDelete == false && rarely()) {
versionValue = new DeleteVersionValue(versionValue.version + 1, maxSeqNo.incrementAndGet(), versionValue = new DeleteVersionValue(versionValue.version + 1,
versionValue.term, clock.getAndIncrement()); maxSeqNo.incrementAndGet(), versionValue.term, clock.getAndIncrement());
deletes.put(bytesRef, (DeleteVersionValue) versionValue); deletes.put(bytesRef, (DeleteVersionValue) versionValue);
map.putDeleteUnderLock(bytesRef, (DeleteVersionValue) versionValue);
} else { } else {
versionValue = new VersionValue(versionValue.version + 1, maxSeqNo.incrementAndGet(), versionValue.term); versionValue = new IndexVersionValue(randomTranslogLocation(),
versionValue.version + 1, maxSeqNo.incrementAndGet(), versionValue.term);
map.putIndexUnderLock(bytesRef, (IndexVersionValue) versionValue);
} }
values.put(bytesRef, versionValue); values.put(bytesRef, versionValue);
map.putUnderLock(bytesRef, versionValue);
} }
if (rarely()) { if (rarely()) {
final long pruneSeqNo = randomLongBetween(0, maxSeqNo.get()); final long pruneSeqNo = randomLongBetween(0, maxSeqNo.get());
@ -268,7 +271,7 @@ public class LiveVersionMapTests extends ESTestCase {
} }
try (Releasable r = map.acquireLock(uid(""))) { try (Releasable r = map.acquireLock(uid(""))) {
map.maybePutUnderLock(new BytesRef(""), new VersionValue(randomLong(), randomLong(), randomLong())); map.maybePutIndexUnderLock(new BytesRef(""), randomIndexVersionValue());
} }
assertFalse(map.isUnsafe()); assertFalse(map.isUnsafe());
assertEquals(1, map.getAllCurrent().size()); assertEquals(1, map.getAllCurrent().size());
@ -278,7 +281,7 @@ public class LiveVersionMapTests extends ESTestCase {
assertFalse(map.isUnsafe()); assertFalse(map.isUnsafe());
assertFalse(map.isSafeAccessRequired()); assertFalse(map.isSafeAccessRequired());
try (Releasable r = map.acquireLock(uid(""))) { try (Releasable r = map.acquireLock(uid(""))) {
map.maybePutUnderLock(new BytesRef(""), new VersionValue(randomLong(), randomLong(), randomLong())); map.maybePutIndexUnderLock(new BytesRef(""), randomIndexVersionValue());
} }
assertTrue(map.isUnsafe()); assertTrue(map.isUnsafe());
assertFalse(map.isSafeAccessRequired()); assertFalse(map.isSafeAccessRequired());
@ -288,7 +291,7 @@ public class LiveVersionMapTests extends ESTestCase {
public void testRefreshTransition() throws IOException { public void testRefreshTransition() throws IOException {
LiveVersionMap map = new LiveVersionMap(); LiveVersionMap map = new LiveVersionMap();
try (Releasable r = map.acquireLock(uid("1"))) { try (Releasable r = map.acquireLock(uid("1"))) {
map.maybePutUnderLock(uid("1"), new VersionValue(randomLong(), randomLong(), randomLong())); map.maybePutIndexUnderLock(uid("1"), randomIndexVersionValue());
assertTrue(map.isUnsafe()); assertTrue(map.isUnsafe());
assertNull(map.getUnderLock(uid("1"))); assertNull(map.getUnderLock(uid("1")));
map.beforeRefresh(); map.beforeRefresh();
@ -299,7 +302,7 @@ public class LiveVersionMapTests extends ESTestCase {
assertFalse(map.isUnsafe()); assertFalse(map.isUnsafe());
map.enforceSafeAccess(); map.enforceSafeAccess();
map.maybePutUnderLock(uid("1"), new VersionValue(randomLong(), randomLong(), randomLong())); map.maybePutIndexUnderLock(uid("1"), randomIndexVersionValue());
assertFalse(map.isUnsafe()); assertFalse(map.isUnsafe());
assertNotNull(map.getUnderLock(uid("1"))); assertNotNull(map.getUnderLock(uid("1")));
map.beforeRefresh(); map.beforeRefresh();
@ -320,9 +323,10 @@ public class LiveVersionMapTests extends ESTestCase {
AtomicLong version = new AtomicLong(); AtomicLong version = new AtomicLong();
CountDownLatch start = new CountDownLatch(2); CountDownLatch start = new CountDownLatch(2);
BytesRef uid = uid("1"); BytesRef uid = uid("1");
VersionValue initialVersion = new VersionValue(version.incrementAndGet(), 1, 1); VersionValue initialVersion;
try (Releasable ignore = map.acquireLock(uid)) { try (Releasable ignore = map.acquireLock(uid)) {
map.putUnderLock(uid, initialVersion); initialVersion = new IndexVersionValue(randomTranslogLocation(), version.incrementAndGet(), 1, 1);
map.putIndexUnderLock(uid, (IndexVersionValue) initialVersion);
} }
Thread t = new Thread(() -> { Thread t = new Thread(() -> {
start.countDown(); start.countDown();
@ -337,14 +341,13 @@ public class LiveVersionMapTests extends ESTestCase {
} else { } else {
underLock = nextVersionValue; underLock = nextVersionValue;
} }
if (underLock.isDelete()) { if (underLock.isDelete() || randomBoolean()) {
nextVersionValue = new VersionValue(version.incrementAndGet(), 1, 1); nextVersionValue = new IndexVersionValue(randomTranslogLocation(), version.incrementAndGet(), 1, 1);
} else if (randomBoolean()) { map.putIndexUnderLock(uid, (IndexVersionValue) nextVersionValue);
nextVersionValue = new VersionValue(version.incrementAndGet(), 1, 1);
} else { } else {
nextVersionValue = new DeleteVersionValue(version.incrementAndGet(), 1, 1, 0); nextVersionValue = new DeleteVersionValue(version.incrementAndGet(), 1, 1, 0);
map.putDeleteUnderLock(uid, (DeleteVersionValue) nextVersionValue);
} }
map.putUnderLock(uid, nextVersionValue);
} }
} }
} catch (Exception e) { } catch (Exception e) {
@ -375,7 +378,7 @@ public class LiveVersionMapTests extends ESTestCase {
BytesRef uid = uid("1"); BytesRef uid = uid("1");
; ;
try (Releasable ignore = map.acquireLock(uid)) { try (Releasable ignore = map.acquireLock(uid)) {
map.putUnderLock(uid, new DeleteVersionValue(0, 0, 0, 0)); map.putDeleteUnderLock(uid, new DeleteVersionValue(0, 0, 0, 0));
map.beforeRefresh(); // refresh otherwise we won't prune since it's tracked by the current map map.beforeRefresh(); // refresh otherwise we won't prune since it's tracked by the current map
map.afterRefresh(false); map.afterRefresh(false);
Thread thread = new Thread(() -> { Thread thread = new Thread(() -> {
@ -392,4 +395,16 @@ public class LiveVersionMapTests extends ESTestCase {
thread.join(); thread.join();
assertEquals(0, map.getAllTombstones().size()); assertEquals(0, map.getAllTombstones().size());
} }
IndexVersionValue randomIndexVersionValue() {
return new IndexVersionValue(randomTranslogLocation(), randomNonNegativeLong(), randomNonNegativeLong(), randomNonNegativeLong());
}
Translog.Location randomTranslogLocation() {
if (randomBoolean()) {
return null;
} else {
return new Translog.Location(randomNonNegativeLong(), randomNonNegativeLong(), randomInt());
}
}
} }

View File

@ -20,12 +20,17 @@
package org.elasticsearch.index.engine; package org.elasticsearch.index.engine;
import org.apache.lucene.util.RamUsageTester; import org.apache.lucene.util.RamUsageTester;
import org.elasticsearch.index.translog.Translog;
import org.elasticsearch.test.ESTestCase; import org.elasticsearch.test.ESTestCase;
public class VersionValueTests extends ESTestCase { public class VersionValueTests extends ESTestCase {
public void testRamBytesUsed() { public void testIndexRamBytesUsed() {
VersionValue versionValue = new VersionValue(randomLong(), randomLong(), randomLong()); Translog.Location translogLoc = null;
if (randomBoolean()) {
translogLoc = new Translog.Location(randomNonNegativeLong(), randomNonNegativeLong(), randomInt());
}
IndexVersionValue versionValue = new IndexVersionValue(translogLoc, randomLong(), randomLong(), randomLong());
assertEquals(RamUsageTester.sizeOf(versionValue), versionValue.ramBytesUsed()); assertEquals(RamUsageTester.sizeOf(versionValue), versionValue.ramBytesUsed());
} }