@ -89,7 +89,9 @@ class BackupEngineImpl : public BackupEngine {
BackupEngineImpl ( Env * db_env , const BackupableDBOptions & options ,
BackupEngineImpl ( Env * db_env , const BackupableDBOptions & options ,
bool read_only = false ) ;
bool read_only = false ) ;
~ BackupEngineImpl ( ) ;
~ BackupEngineImpl ( ) ;
Status CreateNewBackup ( DB * db , bool flush_before_backup = false ) override ;
Status CreateNewBackup ( DB * db , bool flush_before_backup = false ,
std : : function < void ( ) > progress_callback = [ ] ( ) {
} ) override ;
Status PurgeOldBackups ( uint32_t num_backups_to_keep ) override ;
Status PurgeOldBackups ( uint32_t num_backups_to_keep ) override ;
Status DeleteBackup ( BackupID backup_id ) override ;
Status DeleteBackup ( BackupID backup_id ) override ;
void StopBackup ( ) override {
void StopBackup ( ) override {
@ -267,15 +269,11 @@ class BackupEngineImpl : public BackupEngine {
Status PutLatestBackupFileContents ( uint32_t latest_backup ) ;
Status PutLatestBackupFileContents ( uint32_t latest_backup ) ;
// if size_limit == 0, there is no size limit, copy everything
// if size_limit == 0, there is no size limit, copy everything
Status CopyFile ( const std : : string & src ,
Status CopyFile ( const std : : string & src , const std : : string & dst , Env * src_env ,
const std : : string & dst ,
Env * dst_env , bool sync , RateLimiter * rate_limiter ,
Env * src_env ,
uint64_t * size = nullptr , uint32_t * checksum_value = nullptr ,
Env * dst_env ,
uint64_t size_limit = 0 ,
bool sync ,
std : : function < void ( ) > progress_callback = [ ] ( ) { } ) ;
RateLimiter * rate_limiter ,
uint64_t * size = nullptr ,
uint32_t * checksum_value = nullptr ,
uint64_t size_limit = 0 ) ;
Status CalculateChecksum ( const std : : string & src ,
Status CalculateChecksum ( const std : : string & src ,
Env * src_env ,
Env * src_env ,
@ -296,6 +294,7 @@ class BackupEngineImpl : public BackupEngine {
RateLimiter * rate_limiter ;
RateLimiter * rate_limiter ;
uint64_t size_limit ;
uint64_t size_limit ;
std : : promise < CopyResult > result ;
std : : promise < CopyResult > result ;
std : : function < void ( ) > progress_callback ;
CopyWorkItem ( ) { }
CopyWorkItem ( ) { }
CopyWorkItem ( const CopyWorkItem & ) = delete ;
CopyWorkItem ( const CopyWorkItem & ) = delete ;
@ -312,23 +311,22 @@ class BackupEngineImpl : public BackupEngine {
rate_limiter = o . rate_limiter ;
rate_limiter = o . rate_limiter ;
size_limit = o . size_limit ;
size_limit = o . size_limit ;
result = std : : move ( o . result ) ;
result = std : : move ( o . result ) ;
progress_callback = std : : move ( o . progress_callback ) ;
return * this ;
return * this ;
}
}
CopyWorkItem ( std : : string _src_path ,
CopyWorkItem ( std : : string _src_path , std : : string _dst_path , Env * _src_env ,
std : : string _dst_path ,
Env * _dst_env , bool _sync , RateLimiter * _rate_limiter ,
Env * _src_env ,
uint64_t _size_limit ,
Env * _dst_env ,
std : : function < void ( ) > _progress_callback = [ ] ( ) { } )
bool _sync ,
RateLimiter * _rate_limiter ,
uint64_t _size_limit )
: src_path ( std : : move ( _src_path ) ) ,
: src_path ( std : : move ( _src_path ) ) ,
dst_path ( std : : move ( _dst_path ) ) ,
dst_path ( std : : move ( _dst_path ) ) ,
src_env ( _src_env ) ,
src_env ( _src_env ) ,
dst_env ( _dst_env ) ,
dst_env ( _dst_env ) ,
sync ( _sync ) ,
sync ( _sync ) ,
rate_limiter ( _rate_limiter ) ,
rate_limiter ( _rate_limiter ) ,
size_limit ( _size_limit ) { }
size_limit ( _size_limit ) ,
progress_callback ( _progress_callback ) { }
} ;
} ;
struct BackupAfterCopyWorkItem {
struct BackupAfterCopyWorkItem {
@ -388,19 +386,18 @@ class BackupEngineImpl : public BackupEngine {
} ;
} ;
bool initialized_ ;
bool initialized_ ;
std : : mutex byte_report_mutex_ ;
channel < CopyWorkItem > files_to_copy_ ;
channel < CopyWorkItem > files_to_copy_ ;
std : : vector < std : : thread > threads_ ;
std : : vector < std : : thread > threads_ ;
Status AddBackupFileWorkItem (
Status AddBackupFileWorkItem (
std : : unordered_set < std : : string > & live_dst_paths ,
std : : unordered_set < std : : string > & live_dst_paths ,
std : : vector < BackupAfterCopyWorkItem > & backup_items_to_finish ,
std : : vector < BackupAfterCopyWorkItem > & backup_items_to_finish ,
BackupID backup_id ,
BackupID backup_id , bool shared , const std : : string & src_dir ,
bool shared ,
const std : : string & src_fname , // starts with "/"
const std : : string & src_dir ,
RateLimiter * rate_limiter , uint64_t size_limit = 0 ,
const std : : string & src_fname , // starts with "/"
bool shared_checksum = false ,
RateLimiter * rate_limiter ,
std : : function < void ( ) > progress_callback = [ ] ( ) { } ) ;
uint64_t size_limit = 0 ,
bool shared_checksum = false ) ;
// backup state data
// backup state data
BackupID latest_backup_id_ ;
BackupID latest_backup_id_ ;
@ -578,15 +575,11 @@ Status BackupEngineImpl::Initialize() {
CopyWorkItem work_item ;
CopyWorkItem work_item ;
while ( files_to_copy_ . read ( work_item ) ) {
while ( files_to_copy_ . read ( work_item ) ) {
CopyResult result ;
CopyResult result ;
result . status = CopyFile ( work_item . src_path ,
result . status =
work_item . dst_path ,
CopyFile ( work_item . src_path , work_item . dst_path , work_item . src_env ,
work_item . src_env ,
work_item . dst_env , work_item . sync , work_item . rate_limiter ,
work_item . dst_env ,
& result . size , & result . checksum_value , work_item . size_limit ,
work_item . sync ,
work_item . progress_callback ) ;
work_item . rate_limiter ,
& result . size ,
& result . checksum_value ,
work_item . size_limit ) ;
work_item . result . set_value ( std : : move ( result ) ) ;
work_item . result . set_value ( std : : move ( result ) ) ;
}
}
} ) ;
} ) ;
@ -597,7 +590,8 @@ Status BackupEngineImpl::Initialize() {
return Status : : OK ( ) ;
return Status : : OK ( ) ;
}
}
Status BackupEngineImpl : : CreateNewBackup ( DB * db , bool flush_before_backup ) {
Status BackupEngineImpl : : CreateNewBackup (
DB * db , bool flush_before_backup , std : : function < void ( ) > progress_callback ) {
assert ( initialized_ ) ;
assert ( initialized_ ) ;
assert ( ! read_only_ ) ;
assert ( ! read_only_ ) ;
Status s ;
Status s ;
@ -672,15 +666,12 @@ Status BackupEngineImpl::CreateNewBackup(DB* db, bool flush_before_backup) {
// * if it's kTableFile, then it's shared
// * if it's kTableFile, then it's shared
// * if it's kDescriptorFile, limit the size to manifest_file_size
// * if it's kDescriptorFile, limit the size to manifest_file_size
s = AddBackupFileWorkItem (
s = AddBackupFileWorkItem (
live_dst_paths ,
live_dst_paths , backup_items_to_finish , new_backup_id ,
backup_items_to_finish ,
options_ . share_table_files & & type = = kTableFile , db - > GetName ( ) ,
new_backup_id ,
live_files [ i ] , rate_limiter . get ( ) ,
options_ . share_table_files & & type = = kTableFile ,
( type = = kDescriptorFile ) ? manifest_file_size : 0 ,
db - > GetName ( ) ,
options_ . share_files_with_checksum & & type = = kTableFile ,
live_files [ i ] ,
progress_callback ) ;
rate_limiter . get ( ) ,
( type = = kDescriptorFile ) ? manifest_file_size : 0 ,
options_ . share_files_with_checksum & & type = = kTableFile ) ;
}
}
// Add a CopyWorkItem to the channel for each WAL file
// Add a CopyWorkItem to the channel for each WAL file
for ( size_t i = 0 ; s . ok ( ) & & i < live_wal_files . size ( ) ; + + i ) {
for ( size_t i = 0 ; s . ok ( ) & & i < live_wal_files . size ( ) ; + + i ) {
@ -1085,13 +1076,12 @@ Status BackupEngineImpl::PutLatestBackupFileContents(uint32_t latest_backup) {
return s ;
return s ;
}
}
Status BackupEngineImpl : : CopyFile (
Status BackupEngineImpl : : CopyFile ( const std : : string & src ,
const std : : string & src ,
const std : : string & dst , Env * src_env ,
const std : : string & dst , Env * src_env ,
Env * dst_env , bool sync ,
Env * dst_env , bool sync ,
RateLimiter * rate_limiter , uint64_t * size ,
RateLimiter * rate_limiter , uint64_t * size ,
uint32_t * checksum_value , uint64_t size_limit ,
uint32_t * checksum_value ,
std : : function < void ( ) > progress_callback ) {
uint64_t size_limit ) {
Status s ;
Status s ;
unique_ptr < WritableFile > dst_file ;
unique_ptr < WritableFile > dst_file ;
unique_ptr < SequentialFile > src_file ;
unique_ptr < SequentialFile > src_file ;
@ -1125,6 +1115,7 @@ Status BackupEngineImpl::CopyFile(
unique_ptr < char [ ] > buf ( new char [ copy_file_buffer_size_ ] ) ;
unique_ptr < char [ ] > buf ( new char [ copy_file_buffer_size_ ] ) ;
Slice data ;
Slice data ;
uint64_t processed_buffer_size = 0 ;
do {
do {
if ( stop_backup_ . load ( std : : memory_order_acquire ) ) {
if ( stop_backup_ . load ( std : : memory_order_acquire ) ) {
return Status : : Incomplete ( " Backup stopped " ) ;
return Status : : Incomplete ( " Backup stopped " ) ;
@ -1149,6 +1140,12 @@ Status BackupEngineImpl::CopyFile(
if ( rate_limiter ! = nullptr ) {
if ( rate_limiter ! = nullptr ) {
rate_limiter - > Request ( data . size ( ) , Env : : IO_LOW ) ;
rate_limiter - > Request ( data . size ( ) , Env : : IO_LOW ) ;
}
}
processed_buffer_size + = buffer_to_read ;
if ( processed_buffer_size > options_ . callback_trigger_interval_size ) {
processed_buffer_size - = options_ . callback_trigger_interval_size ;
std : : lock_guard < std : : mutex > lock ( byte_report_mutex_ ) ;
progress_callback ( ) ;
}
} while ( s . ok ( ) & & data . size ( ) > 0 & & size_limit > 0 ) ;
} while ( s . ok ( ) & & data . size ( ) > 0 & & size_limit > 0 ) ;
if ( s . ok ( ) & & sync ) {
if ( s . ok ( ) & & sync ) {
@ -1160,15 +1157,12 @@ Status BackupEngineImpl::CopyFile(
// src_fname will always start with "/"
// src_fname will always start with "/"
Status BackupEngineImpl : : AddBackupFileWorkItem (
Status BackupEngineImpl : : AddBackupFileWorkItem (
std : : unordered_set < std : : string > & live_dst_paths ,
std : : unordered_set < std : : string > & live_dst_paths ,
std : : vector < BackupAfterCopyWorkItem > & backup_items_to_finish ,
std : : vector < BackupAfterCopyWorkItem > & backup_items_to_finish ,
BackupID backup_id ,
BackupID backup_id , bool shared , const std : : string & src_dir ,
bool shared ,
const std : : string & src_fname , RateLimiter * rate_limiter ,
const std : : string & src_dir ,
uint64_t size_limit , bool shared_checksum ,
const std : : string & src_fname ,
std : : function < void ( ) > progress_callback ) {
RateLimiter * rate_limiter ,
uint64_t size_limit ,
bool shared_checksum ) {
assert ( src_fname . size ( ) > 0 & & src_fname [ 0 ] = = ' / ' ) ;
assert ( src_fname . size ( ) > 0 & & src_fname [ 0 ] = = ' / ' ) ;
std : : string dst_relative = src_fname . substr ( 1 ) ;
std : : string dst_relative = src_fname . substr ( 1 ) ;
std : : string dst_relative_tmp ;
std : : string dst_relative_tmp ;
@ -1252,13 +1246,9 @@ Status BackupEngineImpl::AddBackupFileWorkItem(
if ( need_to_copy ) {
if ( need_to_copy ) {
Log ( options_ . info_log , " Copying %s to %s " , src_fname . c_str ( ) ,
Log ( options_ . info_log , " Copying %s to %s " , src_fname . c_str ( ) ,
dst_path_tmp . c_str ( ) ) ;
dst_path_tmp . c_str ( ) ) ;
CopyWorkItem copy_work_item ( src_dir + src_fname ,
CopyWorkItem copy_work_item ( src_dir + src_fname , dst_path_tmp , db_env_ ,
dst_path_tmp ,
backup_env_ , options_ . sync , rate_limiter ,
db_env_ ,
size_limit , progress_callback ) ;
backup_env_ ,
options_ . sync ,
rate_limiter ,
size_limit ) ;
BackupAfterCopyWorkItem after_copy_work_item (
BackupAfterCopyWorkItem after_copy_work_item (
copy_work_item . result . get_future ( ) ,
copy_work_item . result . get_future ( ) ,
shared ,
shared ,