|
|
@ -1,3 +1,7 @@ |
|
|
|
|
|
|
|
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
|
|
|
|
|
|
|
|
// This source code is licensed under both the GPLv2 (found in the
|
|
|
|
|
|
|
|
// COPYING file in the root directory) and Apache 2.0 License
|
|
|
|
|
|
|
|
// (found in the LICENSE.Apache file in the root directory).
|
|
|
|
package org.rocksdb; |
|
|
|
package org.rocksdb; |
|
|
|
|
|
|
|
|
|
|
|
import static org.assertj.core.api.Assertions.assertThat; |
|
|
|
import static org.assertj.core.api.Assertions.assertThat; |
|
|
@ -5,15 +9,13 @@ import static org.junit.Assert.*; |
|
|
|
|
|
|
|
|
|
|
|
import java.nio.file.Path; |
|
|
|
import java.nio.file.Path; |
|
|
|
import java.nio.file.Paths; |
|
|
|
import java.nio.file.Paths; |
|
|
|
import java.util.Collections; |
|
|
|
import java.util.*; |
|
|
|
import java.util.Map; |
|
|
|
|
|
|
|
import java.util.Random; |
|
|
|
|
|
|
|
import java.util.UUID; |
|
|
|
|
|
|
|
import java.util.concurrent.atomic.AtomicBoolean; |
|
|
|
import java.util.concurrent.atomic.AtomicBoolean; |
|
|
|
import org.junit.ClassRule; |
|
|
|
import org.junit.ClassRule; |
|
|
|
import org.junit.Rule; |
|
|
|
import org.junit.Rule; |
|
|
|
import org.junit.Test; |
|
|
|
import org.junit.Test; |
|
|
|
import org.junit.rules.TemporaryFolder; |
|
|
|
import org.junit.rules.TemporaryFolder; |
|
|
|
|
|
|
|
import org.rocksdb.AbstractEventListener.EnabledEventCallback; |
|
|
|
import org.rocksdb.test.TestableEventListener; |
|
|
|
import org.rocksdb.test.TestableEventListener; |
|
|
|
|
|
|
|
|
|
|
|
public class EventListenerTest { |
|
|
|
public class EventListenerTest { |
|
|
@ -41,10 +43,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onFlushCompleted() throws RocksDBException { |
|
|
|
public void onFlushCompleted() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onFlushCompletedListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onFlushCompletedListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onFlushCompleted(final RocksDB rocksDb, final FlushJobInfo flushJobInfo) { |
|
|
|
public void onFlushCompleted(final RocksDB rocksDb, final FlushJobInfo flushJobInfo) { |
|
|
|
assertNotNull(flushJobInfo.getColumnFamilyName()); |
|
|
|
assertNotNull(flushJobInfo.getColumnFamilyName()); |
|
|
@ -57,10 +57,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onFlushBegin() throws RocksDBException { |
|
|
|
public void onFlushBegin() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onFlushBeginListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onFlushBeginListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onFlushBegin(final RocksDB rocksDb, final FlushJobInfo flushJobInfo) { |
|
|
|
public void onFlushBegin(final RocksDB rocksDb, final FlushJobInfo flushJobInfo) { |
|
|
|
assertNotNull(flushJobInfo.getColumnFamilyName()); |
|
|
|
assertNotNull(flushJobInfo.getColumnFamilyName()); |
|
|
@ -72,7 +70,7 @@ public class EventListenerTest { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void deleteTableFile(final AbstractEventListener el, final AtomicBoolean wasCbCalled) |
|
|
|
void deleteTableFile(final AbstractEventListener el, final AtomicBoolean wasCbCalled) |
|
|
|
throws RocksDBException, InterruptedException { |
|
|
|
throws RocksDBException { |
|
|
|
try (final Options opt = |
|
|
|
try (final Options opt = |
|
|
|
new Options().setCreateIfMissing(true).setListeners(Collections.singletonList(el)); |
|
|
|
new Options().setCreateIfMissing(true).setListeners(Collections.singletonList(el)); |
|
|
|
final RocksDB db = RocksDB.open(opt, dbFolder.getRoot().getAbsolutePath())) { |
|
|
|
final RocksDB db = RocksDB.open(opt, dbFolder.getRoot().getAbsolutePath())) { |
|
|
@ -80,7 +78,7 @@ public class EventListenerTest { |
|
|
|
final byte[] value = new byte[24]; |
|
|
|
final byte[] value = new byte[24]; |
|
|
|
rand.nextBytes(value); |
|
|
|
rand.nextBytes(value); |
|
|
|
db.put("testKey".getBytes(), value); |
|
|
|
db.put("testKey".getBytes(), value); |
|
|
|
RocksDB.LiveFiles liveFiles = db.getLiveFiles(); |
|
|
|
final RocksDB.LiveFiles liveFiles = db.getLiveFiles(); |
|
|
|
assertNotNull(liveFiles); |
|
|
|
assertNotNull(liveFiles); |
|
|
|
assertNotNull(liveFiles.files); |
|
|
|
assertNotNull(liveFiles.files); |
|
|
|
assertFalse(liveFiles.files.isEmpty()); |
|
|
|
assertFalse(liveFiles.files.isEmpty()); |
|
|
@ -91,10 +89,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onTableFileDeleted() throws RocksDBException, InterruptedException { |
|
|
|
public void onTableFileDeleted() throws RocksDBException, InterruptedException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onTableFileDeletedListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onTableFileDeletedListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onTableFileDeleted(final TableFileDeletionInfo tableFileDeletionInfo) { |
|
|
|
public void onTableFileDeleted(final TableFileDeletionInfo tableFileDeletionInfo) { |
|
|
|
assertNotNull(tableFileDeletionInfo.getDbName()); |
|
|
|
assertNotNull(tableFileDeletionInfo.getDbName()); |
|
|
@ -120,10 +116,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onCompactionBegin() throws RocksDBException { |
|
|
|
public void onCompactionBegin() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onCompactionBeginListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onCompactionBeginListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onCompactionBegin(final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
public void onCompactionBegin(final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
assertEquals(CompactionReason.kManualCompaction, compactionJobInfo.compactionReason()); |
|
|
|
assertEquals(CompactionReason.kManualCompaction, compactionJobInfo.compactionReason()); |
|
|
@ -135,10 +129,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onCompactionCompleted() throws RocksDBException { |
|
|
|
public void onCompactionCompleted() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onCompactionCompletedListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onCompactionCompletedListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onCompactionCompleted( |
|
|
|
public void onCompactionCompleted( |
|
|
|
final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
@ -151,10 +143,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onTableFileCreated() throws RocksDBException { |
|
|
|
public void onTableFileCreated() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onTableFileCreatedListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onTableFileCreatedListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onTableFileCreated(final TableFileCreationInfo tableFileCreationInfo) { |
|
|
|
public void onTableFileCreated(final TableFileCreationInfo tableFileCreationInfo) { |
|
|
|
assertEquals(TableFileCreationReason.FLUSH, tableFileCreationInfo.getReason()); |
|
|
|
assertEquals(TableFileCreationReason.FLUSH, tableFileCreationInfo.getReason()); |
|
|
@ -166,10 +156,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onTableFileCreationStarted() throws RocksDBException { |
|
|
|
public void onTableFileCreationStarted() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onTableFileCreationStartedListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onTableFileCreationStartedListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onTableFileCreationStarted( |
|
|
|
public void onTableFileCreationStarted( |
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfo) { |
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfo) { |
|
|
@ -197,10 +185,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onColumnFamilyHandleDeletionStarted() throws RocksDBException { |
|
|
|
public void onColumnFamilyHandleDeletionStarted() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onColumnFamilyHandleDeletionStartedListener = |
|
|
|
final AbstractEventListener onColumnFamilyHandleDeletionStartedListener = |
|
|
|
new AbstractEventListener() { |
|
|
|
new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onColumnFamilyHandleDeletionStarted( |
|
|
|
public void onColumnFamilyHandleDeletionStarted( |
|
|
@ -218,9 +204,9 @@ public class EventListenerTest { |
|
|
|
new Options().setCreateIfMissing(true).setListeners(Collections.singletonList(el)); |
|
|
|
new Options().setCreateIfMissing(true).setListeners(Collections.singletonList(el)); |
|
|
|
final RocksDB db = RocksDB.open(opt, dbFolder.getRoot().getAbsolutePath())) { |
|
|
|
final RocksDB db = RocksDB.open(opt, dbFolder.getRoot().getAbsolutePath())) { |
|
|
|
assertThat(db).isNotNull(); |
|
|
|
assertThat(db).isNotNull(); |
|
|
|
String uuid = UUID.randomUUID().toString(); |
|
|
|
final String uuid = UUID.randomUUID().toString(); |
|
|
|
SstFileWriter sstFileWriter = new SstFileWriter(new EnvOptions(), opt); |
|
|
|
final SstFileWriter sstFileWriter = new SstFileWriter(new EnvOptions(), opt); |
|
|
|
Path externalFilePath = Paths.get(db.getName(), uuid); |
|
|
|
final Path externalFilePath = Paths.get(db.getName(), uuid); |
|
|
|
sstFileWriter.open(externalFilePath.toString()); |
|
|
|
sstFileWriter.open(externalFilePath.toString()); |
|
|
|
sstFileWriter.put("testKey".getBytes(), uuid.getBytes()); |
|
|
|
sstFileWriter.put("testKey".getBytes(), uuid.getBytes()); |
|
|
|
sstFileWriter.finish(); |
|
|
|
sstFileWriter.finish(); |
|
|
@ -232,10 +218,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void onExternalFileIngested() throws RocksDBException { |
|
|
|
public void onExternalFileIngested() throws RocksDBException { |
|
|
|
// Callback is synchronous, but we need mutable container to update boolean value in other
|
|
|
|
|
|
|
|
// method
|
|
|
|
|
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
final AtomicBoolean wasCbCalled = new AtomicBoolean(); |
|
|
|
AbstractEventListener onExternalFileIngestedListener = new AbstractEventListener() { |
|
|
|
final AbstractEventListener onExternalFileIngestedListener = new AbstractEventListener() { |
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onExternalFileIngested( |
|
|
|
public void onExternalFileIngested( |
|
|
|
final RocksDB db, final ExternalFileIngestionInfo externalFileIngestionInfo) { |
|
|
|
final RocksDB db, final ExternalFileIngestionInfo externalFileIngestionInfo) { |
|
|
@ -248,8 +232,8 @@ public class EventListenerTest { |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void testAllCallbacksInvocation() { |
|
|
|
public void testAllCallbacksInvocation() { |
|
|
|
final int TEST_INT_VAL = Integer.MAX_VALUE; |
|
|
|
final int TEST_INT_VAL = -1; |
|
|
|
final long TEST_LONG_VAL = Long.MAX_VALUE; |
|
|
|
final long TEST_LONG_VAL = -1; |
|
|
|
// Expected test data objects
|
|
|
|
// Expected test data objects
|
|
|
|
final Map<String, String> userCollectedPropertiesTestData = |
|
|
|
final Map<String, String> userCollectedPropertiesTestData = |
|
|
|
Collections.singletonMap("key", "value"); |
|
|
|
Collections.singletonMap("key", "value"); |
|
|
@ -263,18 +247,18 @@ public class EventListenerTest { |
|
|
|
"columnFamilyName".getBytes(), "filterPolicyName", "comparatorName", "mergeOperatorName", |
|
|
|
"columnFamilyName".getBytes(), "filterPolicyName", "comparatorName", "mergeOperatorName", |
|
|
|
"prefixExtractorName", "propertyCollectorsNames", "compressionName", |
|
|
|
"prefixExtractorName", "propertyCollectorsNames", "compressionName", |
|
|
|
userCollectedPropertiesTestData, readablePropertiesTestData, propertiesOffsetsTestData); |
|
|
|
userCollectedPropertiesTestData, readablePropertiesTestData, propertiesOffsetsTestData); |
|
|
|
final FlushJobInfo flushJobInfoTestData = new FlushJobInfo(TEST_INT_VAL, "testColumnFamily", |
|
|
|
final FlushJobInfo flushJobInfoTestData = new FlushJobInfo(Integer.MAX_VALUE, |
|
|
|
"/file/path", TEST_LONG_VAL, TEST_INT_VAL, true, true, TEST_LONG_VAL, TEST_LONG_VAL, |
|
|
|
"testColumnFamily", "/file/path", TEST_LONG_VAL, Integer.MAX_VALUE, true, true, |
|
|
|
tablePropertiesTestData, (byte) 0x0a); |
|
|
|
TEST_LONG_VAL, TEST_LONG_VAL, tablePropertiesTestData, (byte) 0x0a); |
|
|
|
final Status statusTestData = new Status(Status.Code.Incomplete, Status.SubCode.NoSpace, null); |
|
|
|
final Status statusTestData = new Status(Status.Code.Incomplete, Status.SubCode.NoSpace, null); |
|
|
|
final TableFileDeletionInfo tableFileDeletionInfoTestData = |
|
|
|
final TableFileDeletionInfo tableFileDeletionInfoTestData = |
|
|
|
new TableFileDeletionInfo("dbName", "/file/path", TEST_INT_VAL, statusTestData); |
|
|
|
new TableFileDeletionInfo("dbName", "/file/path", Integer.MAX_VALUE, statusTestData); |
|
|
|
final TableFileCreationInfo tableFileCreationInfoTestData = |
|
|
|
final TableFileCreationInfo tableFileCreationInfoTestData = |
|
|
|
new TableFileCreationInfo(TEST_LONG_VAL, tablePropertiesTestData, statusTestData, "dbName", |
|
|
|
new TableFileCreationInfo(TEST_LONG_VAL, tablePropertiesTestData, statusTestData, "dbName", |
|
|
|
"columnFamilyName", "/file/path", TEST_INT_VAL, (byte) 0x03); |
|
|
|
"columnFamilyName", "/file/path", Integer.MAX_VALUE, (byte) 0x03); |
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfoTestData = |
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfoTestData = |
|
|
|
new TableFileCreationBriefInfo( |
|
|
|
new TableFileCreationBriefInfo( |
|
|
|
"dbName", "columnFamilyName", "/file/path", TEST_INT_VAL, (byte) 0x03); |
|
|
|
"dbName", "columnFamilyName", "/file/path", Integer.MAX_VALUE, (byte) 0x03); |
|
|
|
final MemTableInfo memTableInfoTestData = new MemTableInfo( |
|
|
|
final MemTableInfo memTableInfoTestData = new MemTableInfo( |
|
|
|
"columnFamilyName", TEST_LONG_VAL, TEST_LONG_VAL, TEST_LONG_VAL, TEST_LONG_VAL); |
|
|
|
"columnFamilyName", TEST_LONG_VAL, TEST_LONG_VAL, TEST_LONG_VAL, TEST_LONG_VAL); |
|
|
|
final FileOperationInfo fileOperationInfoTestData = new FileOperationInfo("/file/path", |
|
|
|
final FileOperationInfo fileOperationInfoTestData = new FileOperationInfo("/file/path", |
|
|
@ -285,305 +269,496 @@ public class EventListenerTest { |
|
|
|
new ExternalFileIngestionInfo("columnFamilyName", "/external/file/path", |
|
|
|
new ExternalFileIngestionInfo("columnFamilyName", "/external/file/path", |
|
|
|
"/internal/file/path", TEST_LONG_VAL, tablePropertiesTestData); |
|
|
|
"/internal/file/path", TEST_LONG_VAL, tablePropertiesTestData); |
|
|
|
|
|
|
|
|
|
|
|
final int CALLBACKS_COUNT = 22; |
|
|
|
final CapturingTestableEventListener listener = new CapturingTestableEventListener() { |
|
|
|
final AtomicBoolean[] wasCalled = new AtomicBoolean[CALLBACKS_COUNT]; |
|
|
|
|
|
|
|
for (int i = 0; i < CALLBACKS_COUNT; ++i) { |
|
|
|
|
|
|
|
wasCalled[i] = new AtomicBoolean(); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
TestableEventListener listener = new TestableEventListener() { |
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onFlushCompleted(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
public void onFlushCompleted(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
|
|
|
|
super.onFlushCompleted(db, flushJobInfo); |
|
|
|
assertEquals(flushJobInfoTestData, flushJobInfo); |
|
|
|
assertEquals(flushJobInfoTestData, flushJobInfo); |
|
|
|
wasCalled[0].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onFlushBegin(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
public void onFlushBegin(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
|
|
|
|
super.onFlushBegin(db, flushJobInfo); |
|
|
|
assertEquals(flushJobInfoTestData, flushJobInfo); |
|
|
|
assertEquals(flushJobInfoTestData, flushJobInfo); |
|
|
|
wasCalled[1].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onTableFileDeleted(final TableFileDeletionInfo tableFileDeletionInfo) { |
|
|
|
public void onTableFileDeleted(final TableFileDeletionInfo tableFileDeletionInfo) { |
|
|
|
|
|
|
|
super.onTableFileDeleted(tableFileDeletionInfo); |
|
|
|
assertEquals(tableFileDeletionInfoTestData, tableFileDeletionInfo); |
|
|
|
assertEquals(tableFileDeletionInfoTestData, tableFileDeletionInfo); |
|
|
|
wasCalled[2].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onCompactionBegin(final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
public void onCompactionBegin(final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
|
|
|
|
super.onCompactionBegin(db, compactionJobInfo); |
|
|
|
assertArrayEquals( |
|
|
|
assertArrayEquals( |
|
|
|
"compactionColumnFamily".getBytes(), compactionJobInfo.columnFamilyName()); |
|
|
|
"compactionColumnFamily".getBytes(), compactionJobInfo.columnFamilyName()); |
|
|
|
assertEquals(statusTestData, compactionJobInfo.status()); |
|
|
|
assertEquals(statusTestData, compactionJobInfo.status()); |
|
|
|
assertEquals(TEST_LONG_VAL, compactionJobInfo.threadId()); |
|
|
|
assertEquals(TEST_LONG_VAL, compactionJobInfo.threadId()); |
|
|
|
assertEquals(TEST_INT_VAL, compactionJobInfo.jobId()); |
|
|
|
assertEquals(Integer.MAX_VALUE, compactionJobInfo.jobId()); |
|
|
|
assertEquals(TEST_INT_VAL, compactionJobInfo.baseInputLevel()); |
|
|
|
assertEquals(Integer.MAX_VALUE, compactionJobInfo.baseInputLevel()); |
|
|
|
assertEquals(TEST_INT_VAL, compactionJobInfo.outputLevel()); |
|
|
|
assertEquals(Integer.MAX_VALUE, compactionJobInfo.outputLevel()); |
|
|
|
assertEquals(Collections.singletonList("inputFile.sst"), compactionJobInfo.inputFiles()); |
|
|
|
assertEquals(Collections.singletonList("inputFile.sst"), compactionJobInfo.inputFiles()); |
|
|
|
assertEquals(Collections.singletonList("outputFile.sst"), compactionJobInfo.outputFiles()); |
|
|
|
assertEquals(Collections.singletonList("outputFile.sst"), compactionJobInfo.outputFiles()); |
|
|
|
assertEquals(Collections.singletonMap("tableProperties", tablePropertiesTestData), |
|
|
|
assertEquals(Collections.singletonMap("tableProperties", tablePropertiesTestData), |
|
|
|
compactionJobInfo.tableProperties()); |
|
|
|
compactionJobInfo.tableProperties()); |
|
|
|
assertEquals(CompactionReason.kFlush, compactionJobInfo.compactionReason()); |
|
|
|
assertEquals(CompactionReason.kFlush, compactionJobInfo.compactionReason()); |
|
|
|
assertEquals(CompressionType.SNAPPY_COMPRESSION, compactionJobInfo.compression()); |
|
|
|
assertEquals(CompressionType.SNAPPY_COMPRESSION, compactionJobInfo.compression()); |
|
|
|
wasCalled[3].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onCompactionCompleted( |
|
|
|
public void onCompactionCompleted( |
|
|
|
final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
|
|
|
|
super.onCompactionCompleted(db, compactionJobInfo); |
|
|
|
assertArrayEquals( |
|
|
|
assertArrayEquals( |
|
|
|
"compactionColumnFamily".getBytes(), compactionJobInfo.columnFamilyName()); |
|
|
|
"compactionColumnFamily".getBytes(), compactionJobInfo.columnFamilyName()); |
|
|
|
assertEquals(statusTestData, compactionJobInfo.status()); |
|
|
|
assertEquals(statusTestData, compactionJobInfo.status()); |
|
|
|
assertEquals(TEST_LONG_VAL, compactionJobInfo.threadId()); |
|
|
|
assertEquals(TEST_LONG_VAL, compactionJobInfo.threadId()); |
|
|
|
assertEquals(TEST_INT_VAL, compactionJobInfo.jobId()); |
|
|
|
assertEquals(Integer.MAX_VALUE, compactionJobInfo.jobId()); |
|
|
|
assertEquals(TEST_INT_VAL, compactionJobInfo.baseInputLevel()); |
|
|
|
assertEquals(Integer.MAX_VALUE, compactionJobInfo.baseInputLevel()); |
|
|
|
assertEquals(TEST_INT_VAL, compactionJobInfo.outputLevel()); |
|
|
|
assertEquals(Integer.MAX_VALUE, compactionJobInfo.outputLevel()); |
|
|
|
assertEquals(Collections.singletonList("inputFile.sst"), compactionJobInfo.inputFiles()); |
|
|
|
assertEquals(Collections.singletonList("inputFile.sst"), compactionJobInfo.inputFiles()); |
|
|
|
assertEquals(Collections.singletonList("outputFile.sst"), compactionJobInfo.outputFiles()); |
|
|
|
assertEquals(Collections.singletonList("outputFile.sst"), compactionJobInfo.outputFiles()); |
|
|
|
assertEquals(Collections.singletonMap("tableProperties", tablePropertiesTestData), |
|
|
|
assertEquals(Collections.singletonMap("tableProperties", tablePropertiesTestData), |
|
|
|
compactionJobInfo.tableProperties()); |
|
|
|
compactionJobInfo.tableProperties()); |
|
|
|
assertEquals(CompactionReason.kFlush, compactionJobInfo.compactionReason()); |
|
|
|
assertEquals(CompactionReason.kFlush, compactionJobInfo.compactionReason()); |
|
|
|
assertEquals(CompressionType.SNAPPY_COMPRESSION, compactionJobInfo.compression()); |
|
|
|
assertEquals(CompressionType.SNAPPY_COMPRESSION, compactionJobInfo.compression()); |
|
|
|
wasCalled[4].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onTableFileCreated(final TableFileCreationInfo tableFileCreationInfo) { |
|
|
|
public void onTableFileCreated(final TableFileCreationInfo tableFileCreationInfo) { |
|
|
|
|
|
|
|
super.onTableFileCreated(tableFileCreationInfo); |
|
|
|
assertEquals(tableFileCreationInfoTestData, tableFileCreationInfo); |
|
|
|
assertEquals(tableFileCreationInfoTestData, tableFileCreationInfo); |
|
|
|
wasCalled[5].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onTableFileCreationStarted( |
|
|
|
public void onTableFileCreationStarted( |
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfo) { |
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfo) { |
|
|
|
|
|
|
|
super.onTableFileCreationStarted(tableFileCreationBriefInfo); |
|
|
|
assertEquals(tableFileCreationBriefInfoTestData, tableFileCreationBriefInfo); |
|
|
|
assertEquals(tableFileCreationBriefInfoTestData, tableFileCreationBriefInfo); |
|
|
|
wasCalled[6].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onMemTableSealed(final MemTableInfo memTableInfo) { |
|
|
|
public void onMemTableSealed(final MemTableInfo memTableInfo) { |
|
|
|
|
|
|
|
super.onMemTableSealed(memTableInfo); |
|
|
|
assertEquals(memTableInfoTestData, memTableInfo); |
|
|
|
assertEquals(memTableInfoTestData, memTableInfo); |
|
|
|
wasCalled[7].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onColumnFamilyHandleDeletionStarted(final ColumnFamilyHandle columnFamilyHandle) { |
|
|
|
public void onColumnFamilyHandleDeletionStarted(final ColumnFamilyHandle columnFamilyHandle) { |
|
|
|
wasCalled[8].set(true); |
|
|
|
super.onColumnFamilyHandleDeletionStarted(columnFamilyHandle); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onExternalFileIngested( |
|
|
|
public void onExternalFileIngested( |
|
|
|
final RocksDB db, final ExternalFileIngestionInfo externalFileIngestionInfo) { |
|
|
|
final RocksDB db, final ExternalFileIngestionInfo externalFileIngestionInfo) { |
|
|
|
|
|
|
|
super.onExternalFileIngested(db, externalFileIngestionInfo); |
|
|
|
assertEquals(externalFileIngestionInfoTestData, externalFileIngestionInfo); |
|
|
|
assertEquals(externalFileIngestionInfoTestData, externalFileIngestionInfo); |
|
|
|
wasCalled[9].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onBackgroundError( |
|
|
|
public void onBackgroundError( |
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
wasCalled[10].set(true); |
|
|
|
super.onBackgroundError(backgroundErrorReason, backgroundError); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onStallConditionsChanged(final WriteStallInfo writeStallInfo) { |
|
|
|
public void onStallConditionsChanged(final WriteStallInfo writeStallInfo) { |
|
|
|
|
|
|
|
super.onStallConditionsChanged(writeStallInfo); |
|
|
|
assertEquals(writeStallInfoTestData, writeStallInfo); |
|
|
|
assertEquals(writeStallInfoTestData, writeStallInfo); |
|
|
|
wasCalled[11].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onFileReadFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
public void onFileReadFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
super.onFileReadFinish(fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
wasCalled[12].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onFileWriteFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
public void onFileWriteFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
super.onFileWriteFinish(fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
wasCalled[13].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void OnFileFlushFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
public void onFileFlushFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
super.onFileFlushFinish(fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
wasCalled[14].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void OnFileSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
public void onFileSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
super.onFileSyncFinish(fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
wasCalled[15].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void OnFileRangeSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
public void onFileRangeSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
super.onFileRangeSyncFinish(fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
wasCalled[16].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void OnFileTruncateFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
public void onFileTruncateFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
wasCalled[17].set(true); |
|
|
|
super.onFileTruncateFinish(fileOperationInfo); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void OnFileCloseFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
public void onFileCloseFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
super.onFileCloseFinish(fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
assertEquals(fileOperationInfoTestData, fileOperationInfo); |
|
|
|
wasCalled[18].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public boolean shouldBeNotifiedOnFileIO() { |
|
|
|
public boolean shouldBeNotifiedOnFileIO() { |
|
|
|
wasCalled[19].set(true); |
|
|
|
super.shouldBeNotifiedOnFileIO(); |
|
|
|
return false; |
|
|
|
return false; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public boolean onErrorRecoveryBegin( |
|
|
|
public boolean onErrorRecoveryBegin( |
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
|
|
|
|
super.onErrorRecoveryBegin(backgroundErrorReason, backgroundError); |
|
|
|
assertEquals(BackgroundErrorReason.FLUSH, backgroundErrorReason); |
|
|
|
assertEquals(BackgroundErrorReason.FLUSH, backgroundErrorReason); |
|
|
|
assertEquals(statusTestData, backgroundError); |
|
|
|
assertEquals(statusTestData, backgroundError); |
|
|
|
wasCalled[20].set(true); |
|
|
|
|
|
|
|
return true; |
|
|
|
return true; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
@Override |
|
|
|
public void onErrorRecoveryCompleted(final Status oldBackgroundError) { |
|
|
|
public void onErrorRecoveryCompleted(final Status oldBackgroundError) { |
|
|
|
|
|
|
|
super.onErrorRecoveryCompleted(oldBackgroundError); |
|
|
|
assertEquals(statusTestData, oldBackgroundError); |
|
|
|
assertEquals(statusTestData, oldBackgroundError); |
|
|
|
wasCalled[21].set(true); |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// test action
|
|
|
|
listener.invokeAllCallbacks(); |
|
|
|
listener.invokeAllCallbacks(); |
|
|
|
for (int i = 0; i < CALLBACKS_COUNT; ++i) { |
|
|
|
|
|
|
|
assertTrue("Callback method " + i + " was not called", wasCalled[i].get()); |
|
|
|
// assert
|
|
|
|
} |
|
|
|
assertAllEventsCalled(listener); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Test |
|
|
|
@Test |
|
|
|
public void testEnabledCallbacks() { |
|
|
|
public void testEnabledCallbacks() { |
|
|
|
final AtomicBoolean wasOnMemTableSealedCalled = new AtomicBoolean(); |
|
|
|
final EnabledEventCallback enabledEvents[] = { |
|
|
|
final AtomicBoolean wasOnErrorRecoveryCompletedCalled = new AtomicBoolean(); |
|
|
|
EnabledEventCallback.ON_MEMTABLE_SEALED, EnabledEventCallback.ON_ERROR_RECOVERY_COMPLETED}; |
|
|
|
final TestableEventListener listener = new TestableEventListener( |
|
|
|
|
|
|
|
AbstractEventListener.EnabledEventCallback.ON_MEMTABLE_SEALED, |
|
|
|
|
|
|
|
AbstractEventListener.EnabledEventCallback.ON_ERROR_RECOVERY_COMPLETED) { |
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFlushCompleted(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
|
|
|
|
fail("onFlushCompleted was not enabled"); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
final CapturingTestableEventListener listener = |
|
|
|
public void onFlushBegin(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
new CapturingTestableEventListener(enabledEvents); |
|
|
|
fail("onFlushBegin was not enabled"); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
// test action
|
|
|
|
public void onTableFileDeleted(final TableFileDeletionInfo tableFileDeletionInfo) { |
|
|
|
listener.invokeAllCallbacks(); |
|
|
|
fail("onTableFileDeleted was not enabled"); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
// assert
|
|
|
|
public void onCompactionBegin(final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
assertEventsCalled(listener, enabledEvents); |
|
|
|
fail("onCompactionBegin was not enabled"); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
private static void assertAllEventsCalled( |
|
|
|
public void onCompactionCompleted( |
|
|
|
final CapturingTestableEventListener capturingTestableEventListener) { |
|
|
|
final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
assertEventsCalled(capturingTestableEventListener, EnumSet.allOf(EnabledEventCallback.class)); |
|
|
|
fail("onCompactionCompleted was not enabled"); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
private static void assertEventsCalled( |
|
|
|
public void onTableFileCreated(final TableFileCreationInfo tableFileCreationInfo) { |
|
|
|
final CapturingTestableEventListener capturingTestableEventListener, |
|
|
|
fail("onTableFileCreated was not enabled"); |
|
|
|
final EnabledEventCallback[] expected) { |
|
|
|
} |
|
|
|
assertEventsCalled(capturingTestableEventListener, EnumSet.copyOf(Arrays.asList(expected))); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
private static void assertEventsCalled( |
|
|
|
public void onTableFileCreationStarted( |
|
|
|
final CapturingTestableEventListener capturingTestableEventListener, |
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfo) { |
|
|
|
final EnumSet<EnabledEventCallback> expected) { |
|
|
|
fail("onTableFileCreationStarted was not enabled"); |
|
|
|
final ListenerEvents capturedEvents = capturingTestableEventListener.capturedListenerEvents; |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_FLUSH_COMPLETED)) { |
|
|
|
public void onMemTableSealed(final MemTableInfo memTableInfo) { |
|
|
|
assertTrue("onFlushCompleted was not called", capturedEvents.flushCompleted); |
|
|
|
wasOnMemTableSealedCalled.set(true); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onFlushCompleted was not called", capturedEvents.flushCompleted); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_FLUSH_BEGIN)) { |
|
|
|
public void onColumnFamilyHandleDeletionStarted(final ColumnFamilyHandle columnFamilyHandle) { |
|
|
|
assertTrue("onFlushBegin was not called", capturedEvents.flushBegin); |
|
|
|
fail("onColumnFamilyHandleDeletionStarted was not enabled"); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onFlushBegin was called", capturedEvents.flushBegin); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_TABLE_FILE_DELETED)) { |
|
|
|
public void onExternalFileIngested( |
|
|
|
assertTrue("onTableFileDeleted was not called", capturedEvents.tableFileDeleted); |
|
|
|
final RocksDB db, final ExternalFileIngestionInfo externalFileIngestionInfo) { |
|
|
|
} else { |
|
|
|
fail("onExternalFileIngested was not enabled"); |
|
|
|
assertFalse("onTableFileDeleted was called", capturedEvents.tableFileDeleted); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_COMPACTION_BEGIN)) { |
|
|
|
public void onBackgroundError( |
|
|
|
assertTrue("onCompactionBegin was not called", capturedEvents.compactionBegin); |
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
} else { |
|
|
|
fail("onBackgroundError was not enabled"); |
|
|
|
assertFalse("onCompactionBegin was called", capturedEvents.compactionBegin); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_COMPACTION_COMPLETED)) { |
|
|
|
public void onStallConditionsChanged(final WriteStallInfo writeStallInfo) { |
|
|
|
assertTrue("onCompactionCompleted was not called", capturedEvents.compactionCompleted); |
|
|
|
fail("onStallConditionsChanged was not enabled"); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onCompactionCompleted was called", capturedEvents.compactionCompleted); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_TABLE_FILE_CREATED)) { |
|
|
|
public void onFileReadFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertTrue("onTableFileCreated was not called", capturedEvents.tableFileCreated); |
|
|
|
fail("onFileReadFinish was not enabled"); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onTableFileCreated was called", capturedEvents.tableFileCreated); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_TABLE_FILE_CREATION_STARTED)) { |
|
|
|
public void onFileWriteFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertTrue( |
|
|
|
fail("onFileWriteFinish was not enabled"); |
|
|
|
"onTableFileCreationStarted was not called", capturedEvents.tableFileCreationStarted); |
|
|
|
} |
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onTableFileCreationStarted was called", capturedEvents.tableFileCreationStarted); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_MEMTABLE_SEALED)) { |
|
|
|
public void OnFileFlushFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertTrue("onMemTableSealed was not called", capturedEvents.memTableSealed); |
|
|
|
fail("OnFileFlushFinish was not enabled"); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onMemTableSealed was called", capturedEvents.memTableSealed); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_COLUMN_FAMILY_HANDLE_DELETION_STARTED)) { |
|
|
|
public void OnFileSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertTrue("onColumnFamilyHandleDeletionStarted was not called", |
|
|
|
fail("OnFileSyncFinish was not enabled"); |
|
|
|
capturedEvents.columnFamilyHandleDeletionStarted); |
|
|
|
} |
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onColumnFamilyHandleDeletionStarted was called", |
|
|
|
|
|
|
|
capturedEvents.columnFamilyHandleDeletionStarted); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_EXTERNAL_FILE_INGESTED)) { |
|
|
|
public void OnFileRangeSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertTrue("onExternalFileIngested was not called", capturedEvents.externalFileIngested); |
|
|
|
fail("OnFileRangeSyncFinish was not enabled"); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onExternalFileIngested was called", capturedEvents.externalFileIngested); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_BACKGROUND_ERROR)) { |
|
|
|
public void OnFileTruncateFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertTrue("onBackgroundError was not called", capturedEvents.backgroundError); |
|
|
|
fail("OnFileTruncateFinish was not enabled"); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onBackgroundError was called", capturedEvents.backgroundError); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_STALL_CONDITIONS_CHANGED)) { |
|
|
|
public void OnFileCloseFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
assertTrue("onStallConditionsChanged was not called", capturedEvents.stallConditionsChanged); |
|
|
|
fail("OnFileCloseFinish was not enabled"); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onStallConditionsChanged was called", capturedEvents.stallConditionsChanged); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_FILE_READ_FINISH)) { |
|
|
|
public boolean shouldBeNotifiedOnFileIO() { |
|
|
|
assertTrue("onFileReadFinish was not called", capturedEvents.fileReadFinish); |
|
|
|
fail("shouldBeNotifiedOnFileIO was not enabled"); |
|
|
|
} else { |
|
|
|
return false; |
|
|
|
assertFalse("onFileReadFinish was called", capturedEvents.fileReadFinish); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_FILE_WRITE_FINISH)) { |
|
|
|
public boolean onErrorRecoveryBegin( |
|
|
|
assertTrue("onFileWriteFinish was not called", capturedEvents.fileWriteFinish); |
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
} else { |
|
|
|
fail("onErrorRecoveryBegin was not enabled"); |
|
|
|
assertFalse("onFileWriteFinish was called", capturedEvents.fileWriteFinish); |
|
|
|
return true; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
if (expected.contains(EnabledEventCallback.ON_FILE_FLUSH_FINISH)) { |
|
|
|
public void onErrorRecoveryCompleted(final Status oldBackgroundError) { |
|
|
|
assertTrue("onFileFlushFinish was not called", capturedEvents.fileFlushFinish); |
|
|
|
wasOnErrorRecoveryCompletedCalled.set(true); |
|
|
|
} else { |
|
|
|
} |
|
|
|
assertFalse("onFileFlushFinish was called", capturedEvents.fileFlushFinish); |
|
|
|
}; |
|
|
|
} |
|
|
|
listener.invokeAllCallbacks(); |
|
|
|
|
|
|
|
assertTrue(wasOnMemTableSealedCalled.get()); |
|
|
|
if (expected.contains(EnabledEventCallback.ON_FILE_SYNC_FINISH)) { |
|
|
|
assertTrue(wasOnErrorRecoveryCompletedCalled.get()); |
|
|
|
assertTrue("onFileSyncFinish was not called", capturedEvents.fileSyncFinish); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onFileSyncFinish was called", capturedEvents.fileSyncFinish); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (expected.contains(EnabledEventCallback.ON_FILE_RANGE_SYNC_FINISH)) { |
|
|
|
|
|
|
|
assertTrue("onFileRangeSyncFinish was not called", capturedEvents.fileRangeSyncFinish); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onFileRangeSyncFinish was called", capturedEvents.fileRangeSyncFinish); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (expected.contains(EnabledEventCallback.ON_FILE_TRUNCATE_FINISH)) { |
|
|
|
|
|
|
|
assertTrue("onFileTruncateFinish was not called", capturedEvents.fileTruncateFinish); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onFileTruncateFinish was called", capturedEvents.fileTruncateFinish); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (expected.contains(EnabledEventCallback.ON_FILE_CLOSE_FINISH)) { |
|
|
|
|
|
|
|
assertTrue("onFileCloseFinish was not called", capturedEvents.fileCloseFinish); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onFileCloseFinish was called", capturedEvents.fileCloseFinish); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (expected.contains(EnabledEventCallback.SHOULD_BE_NOTIFIED_ON_FILE_IO)) { |
|
|
|
|
|
|
|
assertTrue( |
|
|
|
|
|
|
|
"shouldBeNotifiedOnFileIO was not called", capturedEvents.shouldBeNotifiedOnFileIO); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("shouldBeNotifiedOnFileIO was called", capturedEvents.shouldBeNotifiedOnFileIO); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (expected.contains(EnabledEventCallback.ON_ERROR_RECOVERY_BEGIN)) { |
|
|
|
|
|
|
|
assertTrue("onErrorRecoveryBegin was not called", capturedEvents.errorRecoveryBegin); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onErrorRecoveryBegin was called", capturedEvents.errorRecoveryBegin); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (expected.contains(EnabledEventCallback.ON_ERROR_RECOVERY_COMPLETED)) { |
|
|
|
|
|
|
|
assertTrue("onErrorRecoveryCompleted was not called", capturedEvents.errorRecoveryCompleted); |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
assertFalse("onErrorRecoveryCompleted was called", capturedEvents.errorRecoveryCompleted); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** |
|
|
|
|
|
|
|
* Members are volatile as they may be written |
|
|
|
|
|
|
|
* and read by different threads. |
|
|
|
|
|
|
|
*/ |
|
|
|
|
|
|
|
private static class ListenerEvents { |
|
|
|
|
|
|
|
volatile boolean flushCompleted; |
|
|
|
|
|
|
|
volatile boolean flushBegin; |
|
|
|
|
|
|
|
volatile boolean tableFileDeleted; |
|
|
|
|
|
|
|
volatile boolean compactionBegin; |
|
|
|
|
|
|
|
volatile boolean compactionCompleted; |
|
|
|
|
|
|
|
volatile boolean tableFileCreated; |
|
|
|
|
|
|
|
volatile boolean tableFileCreationStarted; |
|
|
|
|
|
|
|
volatile boolean memTableSealed; |
|
|
|
|
|
|
|
volatile boolean columnFamilyHandleDeletionStarted; |
|
|
|
|
|
|
|
volatile boolean externalFileIngested; |
|
|
|
|
|
|
|
volatile boolean backgroundError; |
|
|
|
|
|
|
|
volatile boolean stallConditionsChanged; |
|
|
|
|
|
|
|
volatile boolean fileReadFinish; |
|
|
|
|
|
|
|
volatile boolean fileWriteFinish; |
|
|
|
|
|
|
|
volatile boolean fileFlushFinish; |
|
|
|
|
|
|
|
volatile boolean fileSyncFinish; |
|
|
|
|
|
|
|
volatile boolean fileRangeSyncFinish; |
|
|
|
|
|
|
|
volatile boolean fileTruncateFinish; |
|
|
|
|
|
|
|
volatile boolean fileCloseFinish; |
|
|
|
|
|
|
|
volatile boolean shouldBeNotifiedOnFileIO; |
|
|
|
|
|
|
|
volatile boolean errorRecoveryBegin; |
|
|
|
|
|
|
|
volatile boolean errorRecoveryCompleted; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private static class CapturingTestableEventListener extends TestableEventListener { |
|
|
|
|
|
|
|
final ListenerEvents capturedListenerEvents = new ListenerEvents(); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public CapturingTestableEventListener() {} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public CapturingTestableEventListener(final EnabledEventCallback... enabledEventCallbacks) { |
|
|
|
|
|
|
|
super(enabledEventCallbacks); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFlushCompleted(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.flushCompleted = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFlushBegin(final RocksDB db, final FlushJobInfo flushJobInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.flushBegin = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onTableFileDeleted(final TableFileDeletionInfo tableFileDeletionInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.tableFileDeleted = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onCompactionBegin(final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.compactionBegin = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onCompactionCompleted(final RocksDB db, final CompactionJobInfo compactionJobInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.compactionCompleted = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onTableFileCreated(final TableFileCreationInfo tableFileCreationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.tableFileCreated = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onTableFileCreationStarted( |
|
|
|
|
|
|
|
final TableFileCreationBriefInfo tableFileCreationBriefInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.tableFileCreationStarted = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onMemTableSealed(final MemTableInfo memTableInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.memTableSealed = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onColumnFamilyHandleDeletionStarted(final ColumnFamilyHandle columnFamilyHandle) { |
|
|
|
|
|
|
|
capturedListenerEvents.columnFamilyHandleDeletionStarted = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onExternalFileIngested( |
|
|
|
|
|
|
|
final RocksDB db, final ExternalFileIngestionInfo externalFileIngestionInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.externalFileIngested = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onBackgroundError( |
|
|
|
|
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
|
|
|
|
capturedListenerEvents.backgroundError = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onStallConditionsChanged(final WriteStallInfo writeStallInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.stallConditionsChanged = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFileReadFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.fileReadFinish = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFileWriteFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.fileWriteFinish = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFileFlushFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.fileFlushFinish = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFileSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.fileSyncFinish = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFileRangeSyncFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.fileRangeSyncFinish = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFileTruncateFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.fileTruncateFinish = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onFileCloseFinish(final FileOperationInfo fileOperationInfo) { |
|
|
|
|
|
|
|
capturedListenerEvents.fileCloseFinish = true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public boolean shouldBeNotifiedOnFileIO() { |
|
|
|
|
|
|
|
capturedListenerEvents.shouldBeNotifiedOnFileIO = true; |
|
|
|
|
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public boolean onErrorRecoveryBegin( |
|
|
|
|
|
|
|
final BackgroundErrorReason backgroundErrorReason, final Status backgroundError) { |
|
|
|
|
|
|
|
capturedListenerEvents.errorRecoveryBegin = true; |
|
|
|
|
|
|
|
return true; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@Override |
|
|
|
|
|
|
|
public void onErrorRecoveryCompleted(final Status oldBackgroundError) { |
|
|
|
|
|
|
|
capturedListenerEvents.errorRecoveryCompleted = true; |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|