@ -10,7 +10,9 @@
# if !defined(ROCKSDB_LITE) && !defined(OS_WIN)
# if !defined(ROCKSDB_LITE) && !defined(OS_WIN)
# include <algorithm>
# include <algorithm>
# include <limits>
# include <string>
# include <string>
# include <utility>
# include "db/db_impl/db_impl.h"
# include "db/db_impl/db_impl.h"
# include "env/env_chroot.h"
# include "env/env_chroot.h"
@ -515,6 +517,15 @@ static void AssertEmpty(DB* db, int from, int to) {
class BackupableDBTest : public testing : : Test {
class BackupableDBTest : public testing : : Test {
public :
public :
enum ShareOption {
kNoShare ,
kShareNoChecksum ,
kShareWithChecksum ,
} ;
const std : : vector < ShareOption > kAllShareOptions = {
kNoShare , kShareNoChecksum , kShareWithChecksum } ;
BackupableDBTest ( ) {
BackupableDBTest ( ) {
// set up files
// set up files
std : : string db_chroot = test : : PerThreadDBPath ( " backupable_db " ) ;
std : : string db_chroot = test : : PerThreadDBPath ( " backupable_db " ) ;
@ -560,15 +571,8 @@ class BackupableDBTest : public testing::Test {
return db ;
return db ;
}
}
void OpenDBAndBackupEngineShareWithChecksum (
bool destroy_old_data = false , bool dummy = false ,
bool /*share_table_files*/ = true , bool share_with_checksums = false ) {
backupable_options_ - > share_files_with_checksum = share_with_checksums ;
OpenDBAndBackupEngine ( destroy_old_data , dummy , share_with_checksums ) ;
}
void OpenDBAndBackupEngine ( bool destroy_old_data = false , bool dummy = false ,
void OpenDBAndBackupEngine ( bool destroy_old_data = false , bool dummy = false ,
bool share_table_files = true ) {
ShareOption shared_option = kShareNoChecksum ) {
// reset all the defaults
// reset all the defaults
test_backup_env_ - > SetLimitWrittenFiles ( 1000000 ) ;
test_backup_env_ - > SetLimitWrittenFiles ( 1000000 ) ;
test_db_env_ - > SetLimitWrittenFiles ( 1000000 ) ;
test_db_env_ - > SetLimitWrittenFiles ( 1000000 ) ;
@ -583,7 +587,9 @@ class BackupableDBTest : public testing::Test {
}
}
db_ . reset ( db ) ;
db_ . reset ( db ) ;
backupable_options_ - > destroy_old_data = destroy_old_data ;
backupable_options_ - > destroy_old_data = destroy_old_data ;
backupable_options_ - > share_table_files = share_table_files ;
backupable_options_ - > share_table_files = shared_option ! = kNoShare ;
backupable_options_ - > share_files_with_checksum =
shared_option = = kShareWithChecksum ;
BackupEngine * backup_engine ;
BackupEngine * backup_engine ;
ASSERT_OK ( BackupEngine : : Open ( test_db_env_ . get ( ) , * backupable_options_ ,
ASSERT_OK ( BackupEngine : : Open ( test_db_env_ . get ( ) , * backupable_options_ ,
& backup_engine ) ) ;
& backup_engine ) ) ;
@ -1204,7 +1210,7 @@ TEST_F(BackupableDBTest, FailOverwritingBackups) {
TEST_F ( BackupableDBTest , NoShareTableFiles ) {
TEST_F ( BackupableDBTest , NoShareTableFiles ) {
const int keys_iteration = 5000 ;
const int keys_iteration = 5000 ;
OpenDBAndBackupEngine ( true , false , fals e) ;
OpenDBAndBackupEngine ( true , false , kNoShar e) ;
for ( int i = 0 ; i < 5 ; + + i ) {
for ( int i = 0 ; i < 5 ; + + i ) {
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , ! ! ( i % 2 ) ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , ! ! ( i % 2 ) ) ) ;
@ -1220,7 +1226,7 @@ TEST_F(BackupableDBTest, NoShareTableFiles) {
// Verify that you can backup and restore with share_files_with_checksum on
// Verify that you can backup and restore with share_files_with_checksum on
TEST_F ( BackupableDBTest , ShareTableFilesWithChecksums ) {
TEST_F ( BackupableDBTest , ShareTableFilesWithChecksums ) {
const int keys_iteration = 5000 ;
const int keys_iteration = 5000 ;
OpenDBAndBackupEngineShareWithChecksum ( true , false , true , true ) ;
OpenDBAndBackupEngine ( true , false , kShareWithChecksum ) ;
for ( int i = 0 ; i < 5 ; + + i ) {
for ( int i = 0 ; i < 5 ; + + i ) {
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , ! ! ( i % 2 ) ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , ! ! ( i % 2 ) ) ) ;
@ -1238,7 +1244,7 @@ TEST_F(BackupableDBTest, ShareTableFilesWithChecksums) {
TEST_F ( BackupableDBTest , ShareTableFilesWithChecksumsTransition ) {
TEST_F ( BackupableDBTest , ShareTableFilesWithChecksumsTransition ) {
const int keys_iteration = 5000 ;
const int keys_iteration = 5000 ;
// set share_files_with_checksum to false
// set share_files_with_checksum to false
OpenDBAndBackupEngineShareWithChecksum ( true , false , true , false ) ;
OpenDBAndBackupEngine ( true , false , kShareNoChecksum ) ;
for ( int i = 0 ; i < 5 ; + + i ) {
for ( int i = 0 ; i < 5 ; + + i ) {
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , true ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , true ) ) ;
@ -1251,65 +1257,107 @@ TEST_F(BackupableDBTest, ShareTableFilesWithChecksumsTransition) {
}
}
// set share_files_with_checksum to true and do some more backups
// set share_files_with_checksum to true and do some more backups
OpenDBAndBackupEngineShareWithChecksum ( true , false , true , true ) ;
OpenDBAndBackupEngine ( false /* destroy_old_data */ , false ,
kShareWithChecksum ) ;
for ( int i = 5 ; i < 10 ; + + i ) {
for ( int i = 5 ; i < 10 ; + + i ) {
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
FillDB ( db_ . get ( ) , keys_iteration * i , keys_iteration * ( i + 1 ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , true ) ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) , true ) ) ;
}
}
CloseDBAndBackupEngine ( ) ;
CloseDBAndBackupEngine ( ) ;
for ( int i = 0 ; i < 5 ; + + i ) {
// Verify first (about to delete)
AssertBackupConsistency ( i + 1 , 0 , keys_iteration * ( i + 5 + 1 ) ,
AssertBackupConsistency ( 1 , 0 , keys_iteration , keys_iteration * 11 ) ;
// For an extra challenge, make sure that GarbageCollect / DeleteBackup
// is OK even if we open without share_table_files
OpenDBAndBackupEngine ( false /* destroy_old_data */ , false , kNoShare ) ;
backup_engine_ - > DeleteBackup ( 1 ) ;
backup_engine_ - > GarbageCollect ( ) ;
CloseDBAndBackupEngine ( ) ;
// Verify rest (not deleted)
for ( int i = 1 ; i < 10 ; + + i ) {
AssertBackupConsistency ( i + 1 , 0 , keys_iteration * ( i + 1 ) ,
keys_iteration * 11 ) ;
keys_iteration * 11 ) ;
}
}
}
}
// This test simulates cleaning up after aborted or incomplete creation
// of a new backup.
TEST_F ( BackupableDBTest , DeleteTmpFiles ) {
TEST_F ( BackupableDBTest , DeleteTmpFiles ) {
for ( int cleanup_fn : { 1 , 2 , 3 } ) {
for ( int cleanup_fn : { 1 , 2 , 3 , 4 } ) {
for ( bool shared_checksum : { false , true } ) {
for ( ShareOption shared_option : kAllShareOptions ) {
OpenDBAndBackupEngineShareWithChecksum (
OpenDBAndBackupEngine ( false /* destroy_old_data */ , false /* dummy */ ,
false /* destroy_old_data */ , false /* dummy */ ,
shared_option ) ;
true /* share_table_files */ , shared_checksum ) ;
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) ) ) ;
CloseDBAndBackupEngine ( ) ;
BackupID next_id = 1 ;
std : : string shared_tmp = backupdir_ ;
BackupID oldest_id = std : : numeric_limits < BackupID > : : max ( ) ;
if ( shared_checksum ) {
{
shared_tmp + = " /shared_checksum " ;
std : : vector < BackupInfo > backup_info ;
} else {
backup_engine_ - > GetBackupInfo ( & backup_info ) ;
shared_tmp + = " /shared " ;
for ( const auto & bi : backup_info ) {
}
next_id = std : : max ( next_id , bi . backup_id + 1 ) ;
shared_tmp + = " /.00006.sst.tmp " ;
oldest_id = std : : min ( oldest_id , bi . backup_id ) ;
std : : string private_tmp_dir = backupdir_ + " /private/10 " ;
std : : string private_tmp_file = private_tmp_dir + " /00003.sst " ;
file_manager_ - > WriteToFile ( shared_tmp , " tmp " ) ;
file_manager_ - > CreateDir ( private_tmp_dir ) ;
file_manager_ - > WriteToFile ( private_tmp_file , " tmp " ) ;
ASSERT_OK ( file_manager_ - > FileExists ( private_tmp_dir ) ) ;
if ( shared_checksum ) {
OpenDBAndBackupEngineShareWithChecksum (
false /* destroy_old_data */ , false /* dummy */ ,
true /* share_table_files */ , true /* share_with_checksums */ ) ;
} else {
OpenDBAndBackupEngine ( ) ;
}
}
}
CloseDBAndBackupEngine ( ) ;
// An aborted or incomplete new backup will always be in the next
// id (maybe more)
std : : string next_private = " private/ " + std : : to_string ( next_id ) ;
// NOTE: both shared and shared_checksum should be cleaned up
// regardless of how the backup engine is opened.
std : : vector < std : : string > tmp_files_and_dirs ;
for ( const auto & dir_and_file : {
std : : make_pair ( std : : string ( " shared " ) ,
std : : string ( " .00006.sst.tmp " ) ) ,
std : : make_pair ( std : : string ( " shared_checksum " ) ,
std : : string ( " .00007.sst.tmp " ) ) ,
std : : make_pair ( next_private , std : : string ( " 00003.sst " ) ) ,
} ) {
std : : string dir = backupdir_ + " / " + dir_and_file . first ;
file_manager_ - > CreateDir ( dir ) ;
ASSERT_OK ( file_manager_ - > FileExists ( dir ) ) ;
std : : string file = dir + " / " + dir_and_file . second ;
file_manager_ - > WriteToFile ( file , " tmp " ) ;
ASSERT_OK ( file_manager_ - > FileExists ( file ) ) ;
tmp_files_and_dirs . push_back ( file ) ;
}
if ( cleanup_fn ! = /*CreateNewBackup*/ 4 ) {
// This exists after CreateNewBackup because it's deleted then
// re-created.
tmp_files_and_dirs . push_back ( backupdir_ + " / " + next_private ) ;
}
OpenDBAndBackupEngine ( false /* destroy_old_data */ , false /* dummy */ ,
shared_option ) ;
// Need to call one of these explicitly to delete tmp files
// Need to call one of these explicitly to delete tmp files
switch ( cleanup_fn ) {
switch ( cleanup_fn ) {
case 1 :
case 1 :
( void ) backup_engine_ - > GarbageCollect ( ) ;
ASSERT_OK ( backup_engine_ - > GarbageCollect ( ) ) ;
break ;
break ;
case 2 :
case 2 :
( void ) backup_engine_ - > DeleteBackup ( 1 ) ;
ASSERT_OK ( backup_engine_ - > DeleteBackup ( oldest_id ) ) ;
break ;
break ;
case 3 :
case 3 :
( void ) backup_engine_ - > PurgeOldBackups ( 1 ) ;
ASSERT_OK ( backup_engine_ - > PurgeOldBackups ( 1 ) ) ;
break ;
case 4 :
// Does a garbage collect if it sees that next private dir exists
ASSERT_OK ( backup_engine_ - > CreateNewBackup ( db_ . get ( ) ) ) ;
break ;
break ;
default :
default :
assert ( false ) ;
assert ( false ) ;
}
}
CloseDBAndBackupEngine ( ) ;
CloseDBAndBackupEngine ( ) ;
ASSERT_EQ ( Status : : NotFound ( ) , file_manager_ - > FileExists ( shared_tmp ) ) ;
for ( std : : string file_or_dir : tmp_files_and_dirs ) {
ASSERT_EQ ( Status : : NotFound ( ) ,
if ( file_manager_ - > FileExists ( file_or_dir ) ! = Status : : NotFound ( ) ) {
file_manager_ - > FileExists ( private_tmp_file ) ) ;
FAIL ( ) < < file_or_dir < < " was expected to be deleted. " < < cleanup_fn ;
ASSERT_EQ ( Status : : NotFound ( ) , file_manager_ - > FileExists ( private_tmp_dir ) ) ;
}
}
}
}
}
}
}
}