@ -4534,7 +4534,50 @@ TEST_P(DBTestUniversalCompactionMultiLevels, UniversalCompactionMultiLevels) {
ASSERT_EQ ( Get ( 1 , Key ( i % num_keys ) ) , Key ( i ) ) ;
ASSERT_EQ ( Get ( 1 , Key ( i % num_keys ) ) , Key ( i ) ) ;
}
}
}
}
// Tests universal compaction with trivial move enabled
TEST_P ( DBTestUniversalCompactionMultiLevels , UniversalCompactionTrivialMove ) {
int32_t trivial_move = 0 ;
int32_t non_trivial_move = 0 ;
rocksdb : : SyncPoint : : GetInstance ( ) - > SetCallBack (
" DBImpl::BackgroundCompaction:TrivialMove " ,
[ & ] ( void * arg ) { trivial_move + + ; } ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > SetCallBack (
" DBImpl::BackgroundCompaction:NonTrivial " ,
[ & ] ( void * arg ) { non_trivial_move + + ; } ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;
Options options ;
options . compaction_style = kCompactionStyleUniversal ;
options . compaction_options_universal . allow_trivial_move = true ;
options . num_levels = 3 ;
options . write_buffer_size = 100 < < 10 ; // 100KB
options . level0_file_num_compaction_trigger = 3 ;
options . max_background_compactions = 1 ;
options . target_file_size_base = 32 * 1024 ;
options = CurrentOptions ( options ) ;
DestroyAndReopen ( options ) ;
CreateAndReopenWithCF ( { " pikachu " } , options ) ;
// Trigger compaction if size amplification exceeds 110%
options . compaction_options_universal . max_size_amplification_percent = 110 ;
options = CurrentOptions ( options ) ;
ReopenWithColumnFamilies ( { " default " , " pikachu " } , options ) ;
Random rnd ( 301 ) ;
int num_keys = 15000 ;
for ( int i = 0 ; i < num_keys ; i + + ) {
ASSERT_OK ( Put ( 1 , Key ( i ) , Key ( i ) ) ) ;
}
std : : vector < std : : string > values ;
ASSERT_OK ( Flush ( 1 ) ) ;
dbfull ( ) - > TEST_WaitForCompact ( ) ;
ASSERT_GT ( trivial_move , 0 ) ;
ASSERT_EQ ( non_trivial_move , 0 ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > DisableProcessing ( ) ;
}
INSTANTIATE_TEST_CASE_P ( DBTestUniversalCompactionMultiLevels ,
INSTANTIATE_TEST_CASE_P ( DBTestUniversalCompactionMultiLevels ,
DBTestUniversalCompactionMultiLevels ,
DBTestUniversalCompactionMultiLevels ,
: : testing : : Values ( 3 , 20 ) ) ;
: : testing : : Values ( 3 , 20 ) ) ;
@ -8669,28 +8712,78 @@ TEST_F(DBTest, TransactionLogIteratorCorruptedLog) {
//
//
// Test WAL recovery for the various modes available
// Test WAL recovery for the various modes available
// TODO krad:
// 1. Add tests when there are more than one log file
//
//
class RecoveryTestHelper {
class RecoveryTestHelper {
public :
public :
// Recreate and fill the store with some data
static size_t FillData ( DBTest * test , const Options & options ) {
size_t count = 0 ;
test - > DestroyAndReopen ( options ) ;
// Number of WAL files to generate
static const int kWALFilesCount = 10 ;
// Starting number for the WAL file name like 00010.log
static const int kWALFileOffset = 10 ;
// Keys to be written per WAL file
static const int kKeysPerWALFile = 1024 ;
// Size of the value
static const int kValueSize = 10 ;
for ( int i = 0 ; i < 1024 ; i + + ) {
// Create WAL files with values filled in
test - > Put ( " key " + ToString ( i ) , test - > DummyString ( 10 ) ) ;
static void FillData ( DBTest * test , Options & options ,
+ + count ;
const size_t wal_count , size_t & count ) {
DBOptions & db_options = options ;
count = 0 ;
shared_ptr < Cache > table_cache = NewLRUCache ( 50000 , 16 ) ;
EnvOptions env_options ;
WriteBuffer write_buffer ( db_options . db_write_buffer_size ) ;
unique_ptr < VersionSet > versions ;
unique_ptr < WalManager > wal_manager ;
WriteController write_controller ;
versions . reset ( new VersionSet ( test - > dbname_ , & db_options , env_options ,
table_cache . get ( ) , & write_buffer ,
& write_controller ) ) ;
wal_manager . reset ( new WalManager ( db_options , env_options ) ) ;
std : : unique_ptr < log : : Writer > current_log_writer ;
for ( size_t j = kWALFileOffset ; j < wal_count + kWALFileOffset ; j + + ) {
uint64_t current_log_number = j ;
std : : string fname = LogFileName ( test - > dbname_ , current_log_number ) ;
unique_ptr < WritableFile > file ;
ASSERT_OK ( db_options . env - > NewWritableFile ( fname , & file , env_options ) ) ;
current_log_writer . reset ( new log : : Writer ( std : : move ( file ) ) ) ;
for ( int i = 0 ; i < kKeysPerWALFile ; i + + ) {
std : : string key = " key " + ToString ( count + + ) ;
std : : string value = test - > DummyString ( kValueSize ) ;
assert ( current_log_writer . get ( ) ! = nullptr ) ;
uint64_t seq = versions - > LastSequence ( ) + 1 ;
WriteBatch batch ;
batch . Put ( key , value ) ;
WriteBatchInternal : : SetSequence ( & batch , seq ) ;
current_log_writer - > AddRecord ( WriteBatchInternal : : Contents ( & batch ) ) ;
versions - > SetLastSequence ( seq ) ;
}
}
}
}
// Recreate and fill the store with some data
static size_t FillData ( DBTest * test , Options & options ) {
options . create_if_missing = true ;
test - > DestroyAndReopen ( options ) ;
test - > Close ( ) ;
size_t count = 0 ;
FillData ( test , options , kWALFilesCount , count ) ;
return count ;
return count ;
}
}
// Read back all the keys we wrote and return the number of keys found
// Read back all the keys we wrote and return the number of keys found
static size_t GetData ( DBTest * test ) {
static size_t GetData ( DBTest * test ) {
size_t count = 0 ;
size_t count = 0 ;
for ( size_t i = 0 ; i < 1024 ; i + + ) {
for ( size_t i = 0 ; i < kWALFilesCount * kKeysPerWALFile ; i + + ) {
if ( test - > Get ( " key " + ToString ( i ) ) ! = " NOT_FOUND " ) {
if ( test - > Get ( " key " + ToString ( i ) ) ! = " NOT_FOUND " ) {
+ + count ;
+ + count ;
}
}
@ -8698,6 +8791,30 @@ class RecoveryTestHelper {
return count ;
return count ;
}
}
// Manuall corrupt the specified WAL
static void CorruptWAL ( DBTest * test , Options & options ,
const double off , const double len ,
const int wal_file_id , const bool trunc = false ) {
Env * env = options . env ;
std : : string fname = LogFileName ( test - > dbname_ , wal_file_id ) ;
uint64_t size ;
ASSERT_OK ( env - > GetFileSize ( fname , & size ) ) ;
ASSERT_GT ( size , 0 ) ;
# ifdef OS_WIN
// Windows disk cache behaves differently. When we truncate
// the original content is still in the cache due to the original
// handle is still open. Generally, in Windows, one prohibits
// shared access to files and it is not needed for WAL but we allow
// it to induce corruption at various tests.
test - > Close ( ) ;
# endif
if ( trunc ) {
ASSERT_EQ ( 0 , truncate ( fname . c_str ( ) , size * off ) ) ;
} else {
InduceCorruption ( fname , size * off , size * len ) ;
}
}
// Overwrite data with 'a' from offset for length len
// Overwrite data with 'a' from offset for length len
static void InduceCorruption ( const std : : string & filename , uint32_t offset ,
static void InduceCorruption ( const std : : string & filename , uint32_t offset ,
uint32_t len ) {
uint32_t len ) {
@ -8714,32 +8831,6 @@ class RecoveryTestHelper {
close ( fd ) ;
close ( fd ) ;
}
}
// Corrupt the last WAL file from (filesize * off) for length (filesize * len)
static void CorruptWAL ( DBTest * test , const double off , const double len ,
const bool trunc = false ) {
rocksdb : : VectorLogPtr wal_files ;
ASSERT_OK ( test - > dbfull ( ) - > GetSortedWalFiles ( wal_files ) ) ;
ASSERT_EQ ( wal_files . size ( ) , 1 ) ;
const auto logfile_path =
test - > dbname_ + " / " + wal_files . front ( ) - > PathName ( ) ;
auto size = wal_files . front ( ) - > SizeFileBytes ( ) ;
# ifdef OS_WIN
// Windows disk cache behaves differently. When we truncate
// the original content is still in the cache due to the original
// handle is still open. Generally, in Windows, one prohibits
// shared access to files and it is not needed for WAL but we allow
// it to induce corruption at various tests.
test - > Close ( ) ;
# endif
if ( trunc ) {
ASSERT_EQ ( 0 , truncate ( logfile_path . c_str ( ) , size * off ) ) ;
} else {
InduceCorruption ( logfile_path , size * off , size * len ) ;
}
}
} ;
} ;
// Test scope:
// Test scope:
@ -8747,26 +8838,32 @@ class RecoveryTestHelper {
// at the end of any of the logs
// at the end of any of the logs
// - We do not expect to open the data store for corruption
// - We do not expect to open the data store for corruption
TEST_F ( DBTest , kTolerateCorruptedTailRecords ) {
TEST_F ( DBTest , kTolerateCorruptedTailRecords ) {
for ( auto trunc : { true , false } ) {
const int jstart = RecoveryTestHelper : : kWALFileOffset ;
for ( int i = 0 ; i < 4 ; i + + ) {
const int jend = jstart + RecoveryTestHelper : : kWALFilesCount ;
// Fill data for testing
Options options = CurrentOptions ( ) ;
for ( auto trunc : { true , false } ) { /* Corruption style */
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
for ( int i = 0 ; i < 4 ; i + + ) { /* Corruption offset position */
for ( int j = jstart ; j < jend ; j + + ) { /* WAL file */
// test checksum failure or parsing
// Fill data for testing
RecoveryTestHelper : : CorruptWAL ( this , i * .3 , /*len%=*/ .1 , trunc ) ;
Options options = CurrentOptions ( ) ;
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
if ( trunc ) {
// test checksum failure or parsing
options . wal_recovery_mode =
RecoveryTestHelper : : CorruptWAL ( this , options , /*off=*/ i * .3 ,
WALRecoveryMode : : kTolerateCorruptedTailRecords ;
/*len%=*/ .1 , /*wal=*/ j , trunc ) ;
ASSERT_OK ( TryReopen ( options ) ) ;
const size_t recovered_row_count = RecoveryTestHelper : : GetData ( this ) ;
if ( trunc ) {
ASSERT_TRUE ( i = = 0 | | recovered_row_count > 0 ) ;
options . wal_recovery_mode =
ASSERT_LT ( recovered_row_count , row_count ) ;
WALRecoveryMode : : kTolerateCorruptedTailRecords ;
} else {
options . create_if_missing = false ;
options . wal_recovery_mode =
ASSERT_OK ( TryReopen ( options ) ) ;
WALRecoveryMode : : kTolerateCorruptedTailRecords ;
const size_t recovered_row_count = RecoveryTestHelper : : GetData ( this ) ;
ASSERT_NOK ( TryReopen ( options ) ) ;
ASSERT_TRUE ( i = = 0 | | recovered_row_count > 0 ) ;
ASSERT_LT ( recovered_row_count , row_count ) ;
} else {
options . wal_recovery_mode =
WALRecoveryMode : : kTolerateCorruptedTailRecords ;
ASSERT_NOK ( TryReopen ( options ) ) ;
}
}
}
}
}
}
}
@ -8776,23 +8873,34 @@ TEST_F(DBTest, kTolerateCorruptedTailRecords) {
// We don't expect the data store to be opened if there is any corruption
// We don't expect the data store to be opened if there is any corruption
// (leading, middle or trailing -- incomplete writes or corruption)
// (leading, middle or trailing -- incomplete writes or corruption)
TEST_F ( DBTest , kAbsoluteConsistency ) {
TEST_F ( DBTest , kAbsoluteConsistency ) {
const int jstart = RecoveryTestHelper : : kWALFileOffset ;
const int jend = jstart + RecoveryTestHelper : : kWALFilesCount ;
// Verify clean slate behavior
Options options = CurrentOptions ( ) ;
Options options = CurrentOptions ( ) ;
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
options . wal_recovery_mode = WALRecoveryMode : : kAbsoluteConsistency ;
options . wal_recovery_mode = WALRecoveryMode : : kAbsoluteConsistency ;
options . create_if_missing = false ;
ASSERT_OK ( TryReopen ( options ) ) ;
ASSERT_OK ( TryReopen ( options ) ) ;
ASSERT_EQ ( RecoveryTestHelper : : GetData ( this ) , row_count ) ;
ASSERT_EQ ( RecoveryTestHelper : : GetData ( this ) , row_count ) ;
for ( auto trunc : { true , false } ) {
for ( auto trunc : { true , false } ) { /* Corruption style */
for ( int i = 0 ; i < 4 ; i + + ) {
for ( int i = 0 ; i < 4 ; i + + ) { /* Corruption offset position */
if ( trunc & & i = = 0 ) {
if ( trunc & & i = = 0 ) {
continue ;
continue ;
}
}
options = CurrentOptions ( ) ;
RecoveryTestHelper : : FillData ( this , options ) ;
RecoveryTestHelper : : CorruptWAL ( this , i * .3 , /*len%=*/ .1 , trunc ) ;
for ( int j = jstart ; j < jend ; j + + ) { /* wal files */
options . wal_recovery_mode = WALRecoveryMode : : kAbsoluteConsistency ;
// fill with new date
ASSERT_NOK ( TryReopen ( options ) ) ;
RecoveryTestHelper : : FillData ( this , options ) ;
// corrupt the wal
RecoveryTestHelper : : CorruptWAL ( this , options , /*off=*/ i * .3 ,
/*len%=*/ .1 , j , trunc ) ;
// verify
options . wal_recovery_mode = WALRecoveryMode : : kAbsoluteConsistency ;
options . create_if_missing = false ;
ASSERT_NOK ( TryReopen ( options ) ) ;
}
}
}
}
}
}
}
@ -8801,34 +8909,49 @@ TEST_F(DBTest, kAbsoluteConsistency) {
// - We expect to open data store under all circumstances
// - We expect to open data store under all circumstances
// - We expect only data upto the point where the first error was encountered
// - We expect only data upto the point where the first error was encountered
TEST_F ( DBTest , kPointInTimeRecovery ) {
TEST_F ( DBTest , kPointInTimeRecovery ) {
for ( auto trunc : { true , false } ) {
const int jstart = RecoveryTestHelper : : kWALFileOffset ;
for ( int i = 0 ; i < 4 ; i + + ) {
const int jend = jstart + RecoveryTestHelper : : kWALFilesCount ;
// Fill data for testing
const int maxkeys = RecoveryTestHelper : : kWALFilesCount *
Options options = CurrentOptions ( ) ;
RecoveryTestHelper : : kKeysPerWALFile ;
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
for ( auto trunc : { true , false } ) { /* Corruption style */
// test checksum failure or parsing
for ( int i = 0 ; i < 4 ; i + + ) { /* Offset of corruption */
RecoveryTestHelper : : CorruptWAL ( this , i * .3 , /*len%=*/ .1 , trunc ) ;
for ( int j = jstart ; j < jend ; j + + ) { /* WAL file */
// Fill data for testing
options . wal_recovery_mode = WALRecoveryMode : : kPointInTimeRecovery ;
Options options = CurrentOptions ( ) ;
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
// Corrupt the wal
RecoveryTestHelper : : CorruptWAL ( this , options , /*off=*/ i * .3 ,
/*len%=*/ .1 , j , trunc ) ;
// Verify
options . wal_recovery_mode = WALRecoveryMode : : kPointInTimeRecovery ;
options . create_if_missing = false ;
ASSERT_OK ( TryReopen ( options ) ) ;
ASSERT_OK ( TryReopen ( options ) ) ;
// Probe data for invariants
size_t recovered_row_count = RecoveryTestHelper : : GetData ( this ) ;
ASSERT_LT ( recovered_row_count , row_count ) ;
size_t recovered_row_count = RecoveryTestHelper : : GetData ( this ) ;
bool expect_data = true ;
ASSERT_LT ( recovered_row_count , row_count ) ;
for ( size_t k = 0 ; k < maxkeys ; + + k ) {
bool found = Get ( " key " + ToString ( i ) ) ! = " NOT_FOUND " ;
if ( expect_data & & ! found ) {
expect_data = false ;
}
ASSERT_EQ ( found , expect_data ) ;
}
// verify that the keys are sequential and there is no break
const size_t min = RecoveryTestHelper : : kKeysPerWALFile *
bool expect_data = true ;
( j - RecoveryTestHelper : : kWALFileOffset ) ;
for ( size_t j = 0 ; j < 1024 ; + + j ) {
ASSERT_GE ( recovered_row_count , min ) ;
bool found = Get ( " key " + ToString ( i ) ) ! = " NOT_FOUND " ;
if ( ! trunc & & i ! = 0 ) {
if ( expect_data & & ! found ) {
const size_t max = RecoveryTestHelper : : kKeysPerWALFile *
expect_data = false ;
( j - RecoveryTestHelper : : kWALFileOffset + 1 ) ;
ASSERT_LE ( recovered_row_count , max ) ;
}
}
ASSERT_EQ ( found , expect_data ) ;
}
}
ASSERT_TRUE ( i ! = 0 | | recovered_row_count = = 0 ) ;
ASSERT_TRUE ( i ! = 1 | | recovered_row_count < row_count / 2 ) ;
}
}
}
}
}
}
@ -8837,22 +8960,32 @@ TEST_F(DBTest, kPointInTimeRecovery) {
// - We expect to open the data store under all scenarios
// - We expect to open the data store under all scenarios
// - We expect to have recovered records past the corruption zone
// - We expect to have recovered records past the corruption zone
TEST_F ( DBTest , kSkipAnyCorruptedRecords ) {
TEST_F ( DBTest , kSkipAnyCorruptedRecords ) {
for ( auto trunc : { true , false } ) {
const int jstart = RecoveryTestHelper : : kWALFileOffset ;
for ( int i = 0 ; i < 4 ; i + + ) {
const int jend = jstart + RecoveryTestHelper : : kWALFilesCount ;
// Fill data for testing
Options options = CurrentOptions ( ) ;
for ( auto trunc : { true , false } ) { /* Corruption style */
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
for ( int i = 0 ; i < 4 ; i + + ) { /* Corruption offset */
for ( int j = jstart ; j < jend ; j + + ) { /* wal files */
// induce leading corruption
// Fill data for testing
RecoveryTestHelper : : CorruptWAL ( this , i * .3 , /*len%=*/ .1 , trunc ) ;
Options options = CurrentOptions ( ) ;
const size_t row_count = RecoveryTestHelper : : FillData ( this , options ) ;
// Corrupt the WAL
RecoveryTestHelper : : CorruptWAL ( this , options , /*off=*/ i * .3 ,
/*len%=*/ .1 , j , trunc ) ;
// Verify behavior
options . wal_recovery_mode = WALRecoveryMode : : kSkipAnyCorruptedRecords ;
options . create_if_missing = false ;
ASSERT_OK ( TryReopen ( options ) ) ;
options . wal_recovery_mode = WALRecoveryMode : : kSkipAnyCorruptedRecords ;
// Probe data for invariants
ASSERT_OK ( TryReopen ( options ) ) ;
size_t recovered_row_count = RecoveryTestHelper : : GetData ( this ) ;
size_t recovered_row_count = RecoveryTestHelper : : GetData ( this ) ;
ASSERT_LT ( recovered_row_count , row_count ) ;
ASSERT_LT ( recovered_row_count , row_count ) ;
if ( ! trunc ) {
if ( ! trunc ) {
ASSERT_TRUE ( i ! = 0 | | recovered_row_count > 0 ) ;
ASSERT_TRUE ( i ! = 0 | | recovered_row_count > 0 ) ;
}
}
}
}
}
}
}
@ -11274,10 +11407,10 @@ TEST_F(DBTest, ThreadStatusSingleCompaction) {
{ " CompactionJob::Run():Start " , " DBTest::ThreadStatusSingleCompaction:1 " } ,
{ " CompactionJob::Run():Start " , " DBTest::ThreadStatusSingleCompaction:1 " } ,
{ " DBTest::ThreadStatusSingleCompaction:2 " , " CompactionJob::Run():End " } ,
{ " DBTest::ThreadStatusSingleCompaction:2 " , " CompactionJob::Run():End " } ,
} ) ;
} ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;
for ( int tests = 0 ; tests < 2 ; + + tests ) {
for ( int tests = 0 ; tests < 2 ; + + tests ) {
DestroyAndReopen ( options ) ;
DestroyAndReopen ( options ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > ClearTrace ( ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;
Random rnd ( 301 ) ;
Random rnd ( 301 ) ;
// The Put Phase.
// The Put Phase.
@ -11309,8 +11442,8 @@ TEST_F(DBTest, ThreadStatusSingleCompaction) {
// repeat the test with disabling thread tracking.
// repeat the test with disabling thread tracking.
options . enable_thread_tracking = false ;
options . enable_thread_tracking = false ;
rocksdb : : SyncPoint : : GetInstance ( ) - > DisableProcessing ( ) ;
}
}
rocksdb : : SyncPoint : : GetInstance ( ) - > DisableProcessing ( ) ;
}
}
TEST_F ( DBTest , PreShutdownManualCompaction ) {
TEST_F ( DBTest , PreShutdownManualCompaction ) {
@ -12183,7 +12316,7 @@ TEST_F(DBTest, DynamicLevelCompressionPerLevel2) {
" LevelCompactionPicker::PickCompaction:Return " , [ & ] ( void * arg ) {
" LevelCompactionPicker::PickCompaction:Return " , [ & ] ( void * arg ) {
Compaction * compaction = reinterpret_cast < Compaction * > ( arg ) ;
Compaction * compaction = reinterpret_cast < Compaction * > ( arg ) ;
if ( compaction - > output_level ( ) = = 4 ) {
if ( compaction - > output_level ( ) = = 4 ) {
ASSERT_TRUE ( compaction - > OutputCompressionType ( ) = = kLZ4Compression ) ;
ASSERT_TRUE ( compaction - > output_compression ( ) = = kLZ4Compression ) ;
num_lz4 . fetch_add ( 1 ) ;
num_lz4 . fetch_add ( 1 ) ;
}
}
} ) ;
} ) ;
@ -12218,10 +12351,10 @@ TEST_F(DBTest, DynamicLevelCompressionPerLevel2) {
" LevelCompactionPicker::PickCompaction:Return " , [ & ] ( void * arg ) {
" LevelCompactionPicker::PickCompaction:Return " , [ & ] ( void * arg ) {
Compaction * compaction = reinterpret_cast < Compaction * > ( arg ) ;
Compaction * compaction = reinterpret_cast < Compaction * > ( arg ) ;
if ( compaction - > output_level ( ) = = 4 & & compaction - > start_level ( ) = = 3 ) {
if ( compaction - > output_level ( ) = = 4 & & compaction - > start_level ( ) = = 3 ) {
ASSERT_TRUE ( compaction - > OutputCompressionType ( ) = = kZlibCompression ) ;
ASSERT_TRUE ( compaction - > output_compression ( ) = = kZlibCompression ) ;
num_zlib . fetch_add ( 1 ) ;
num_zlib . fetch_add ( 1 ) ;
} else {
} else {
ASSERT_TRUE ( compaction - > OutputCompressionType ( ) = = kLZ4Compression ) ;
ASSERT_TRUE ( compaction - > output_compression ( ) = = kLZ4Compression ) ;
num_lz4 . fetch_add ( 1 ) ;
num_lz4 . fetch_add ( 1 ) ;
}
}
} ) ;
} ) ;
@ -12696,6 +12829,7 @@ TEST_F(DBTest, DontDeletePendingOutputs) {
dbfull ( ) - > FindObsoleteFiles ( & job_context , true /*force*/ ) ;
dbfull ( ) - > FindObsoleteFiles ( & job_context , true /*force*/ ) ;
dbfull ( ) - > TEST_UnlockMutex ( ) ;
dbfull ( ) - > TEST_UnlockMutex ( ) ;
dbfull ( ) - > PurgeObsoleteFiles ( job_context ) ;
dbfull ( ) - > PurgeObsoleteFiles ( job_context ) ;
job_context . Clean ( ) ;
} ;
} ;
env_ - > table_write_callback_ = & purge_obsolete_files_function ;
env_ - > table_write_callback_ = & purge_obsolete_files_function ;
@ -14137,7 +14271,9 @@ TEST_F(DBTest, RowCache) {
ASSERT_EQ ( TestGetTickerCount ( options , ROW_CACHE_MISS ) , 1 ) ;
ASSERT_EQ ( TestGetTickerCount ( options , ROW_CACHE_MISS ) , 1 ) ;
}
}
TEST_F ( DBTest , PrevAfterMerge ) {
// TODO(3.13): fix the issue of Seek() + Prev() which might not necessary
// return the biggest key which is smaller than the seek key.
TEST_F ( DBTest , DISABLED_PrevAfterMerge ) {
Options options ;
Options options ;
options . create_if_missing = true ;
options . create_if_missing = true ;
options . merge_operator = MergeOperators : : CreatePutOperator ( ) ;
options . merge_operator = MergeOperators : : CreatePutOperator ( ) ;
@ -14160,6 +14296,40 @@ TEST_F(DBTest, PrevAfterMerge) {
ASSERT_EQ ( " 1 " , it - > key ( ) . ToString ( ) ) ;
ASSERT_EQ ( " 1 " , it - > key ( ) . ToString ( ) ) ;
}
}
TEST_F ( DBTest , DeletingOldWalAfterDrop ) {
rocksdb : : SyncPoint : : GetInstance ( ) - > LoadDependency (
{ { " Test:AllowFlushes " , " DBImpl::BGWorkFlush " } ,
{ " DBImpl::BGWorkFlush:done " , " Test:WaitForFlush " } } ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > ClearTrace ( ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > DisableProcessing ( ) ;
Options options = CurrentOptions ( ) ;
options . max_total_wal_size = 8192 ;
options . compression = kNoCompression ;
options . write_buffer_size = 1 < < 20 ;
options . level0_file_num_compaction_trigger = ( 1 < < 30 ) ;
options . level0_slowdown_writes_trigger = ( 1 < < 30 ) ;
options . level0_stop_writes_trigger = ( 1 < < 30 ) ;
options . disable_auto_compactions = true ;
DestroyAndReopen ( options ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;
CreateColumnFamilies ( { " cf1 " , " cf2 " } , options ) ;
ASSERT_OK ( Put ( 0 , " key1 " , DummyString ( 8192 ) ) ) ;
ASSERT_OK ( Put ( 0 , " key2 " , DummyString ( 8192 ) ) ) ;
// the oldest wal should now be getting_flushed
ASSERT_OK ( db_ - > DropColumnFamily ( handles_ [ 0 ] ) ) ;
// all flushes should now do nothing because their CF is dropped
TEST_SYNC_POINT ( " Test:AllowFlushes " ) ;
TEST_SYNC_POINT ( " Test:WaitForFlush " ) ;
uint64_t lognum1 = dbfull ( ) - > TEST_LogfileNumber ( ) ;
ASSERT_OK ( Put ( 1 , " key3 " , DummyString ( 8192 ) ) ) ;
ASSERT_OK ( Put ( 1 , " key4 " , DummyString ( 8192 ) ) ) ;
// new wal should have been created
uint64_t lognum2 = dbfull ( ) - > TEST_LogfileNumber ( ) ;
EXPECT_GT ( lognum2 , lognum1 ) ;
}
} // namespace rocksdb
} // namespace rocksdb
# endif
# endif