2020-06-01 16:31:25 -04:00
// Copyright (c) 2020 The Bitcoin Core developers
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
# include <wallet/dump.h>
# include <util/translation.h>
# include <wallet/wallet.h>
static const std : : string DUMP_MAGIC = " BITCOIN_CORE_WALLET_DUMP " ;
uint32_t DUMP_VERSION = 1 ;
bool DumpWallet ( CWallet & wallet , bilingual_str & error )
{
// Get the dumpfile
std : : string dump_filename = gArgs . GetArg ( " -dumpfile " , " " ) ;
if ( dump_filename . empty ( ) ) {
error = _ ( " No dump file provided. To use dump, -dumpfile=<filename> must be provided. " ) ;
return false ;
}
fs : : path path = dump_filename ;
path = fs : : absolute ( path ) ;
if ( fs : : exists ( path ) ) {
error = strprintf ( _ ( " File %s already exists. If you are sure this is what you want, move it out of the way first. " ) , path . string ( ) ) ;
return false ;
}
fsbridge : : ofstream dump_file ;
dump_file . open ( path ) ;
if ( dump_file . fail ( ) ) {
error = strprintf ( _ ( " Unable to open %s for writing " ) , path . string ( ) ) ;
return false ;
}
CHashWriter hasher ( 0 , 0 ) ;
WalletDatabase & db = wallet . GetDatabase ( ) ;
std : : unique_ptr < DatabaseBatch > batch = db . MakeBatch ( ) ;
bool ret = true ;
if ( ! batch - > StartCursor ( ) ) {
error = _ ( " Error: Couldn't create cursor into database " ) ;
ret = false ;
}
// Write out a magic string with version
std : : string line = strprintf ( " %s,%u \n " , DUMP_MAGIC , DUMP_VERSION ) ;
dump_file . write ( line . data ( ) , line . size ( ) ) ;
hasher . write ( line . data ( ) , line . size ( ) ) ;
// Write out the file format
line = strprintf ( " %s,%s \n " , " format " , db . Format ( ) ) ;
dump_file . write ( line . data ( ) , line . size ( ) ) ;
hasher . write ( line . data ( ) , line . size ( ) ) ;
if ( ret ) {
// Read the records
while ( true ) {
CDataStream ss_key ( SER_DISK , CLIENT_VERSION ) ;
CDataStream ss_value ( SER_DISK , CLIENT_VERSION ) ;
bool complete ;
ret = batch - > ReadAtCursor ( ss_key , ss_value , complete ) ;
if ( complete ) {
ret = true ;
break ;
} else if ( ! ret ) {
error = _ ( " Error reading next record from wallet database " ) ;
break ;
}
std : : string key_str = HexStr ( ss_key ) ;
std : : string value_str = HexStr ( ss_value ) ;
line = strprintf ( " %s,%s \n " , key_str , value_str ) ;
dump_file . write ( line . data ( ) , line . size ( ) ) ;
hasher . write ( line . data ( ) , line . size ( ) ) ;
}
}
batch - > CloseCursor ( ) ;
batch . reset ( ) ;
// Close the wallet after we're done with it. The caller won't be doing this
wallet . Close ( ) ;
if ( ret ) {
// Write the hash
tfm : : format ( dump_file , " checksum,%s \n " , HexStr ( hasher . GetHash ( ) ) ) ;
dump_file . close ( ) ;
} else {
// Remove the dumpfile on failure
dump_file . close ( ) ;
fs : : remove ( path ) ;
}
return ret ;
}
2020-06-01 18:00:22 -04:00
// The standard wallet deleter function blocks on the validation interface
// queue, which doesn't exist for the bitcoin-wallet. Define our own
// deleter here.
static void WalletToolReleaseWallet ( CWallet * wallet )
{
wallet - > WalletLogPrintf ( " Releasing wallet \n " ) ;
wallet - > Close ( ) ;
delete wallet ;
}
bool CreateFromDump ( const std : : string & name , const fs : : path & wallet_path , bilingual_str & error , std : : vector < bilingual_str > & warnings )
{
// Get the dumpfile
std : : string dump_filename = gArgs . GetArg ( " -dumpfile " , " " ) ;
if ( dump_filename . empty ( ) ) {
error = _ ( " No dump file provided. To use createfromdump, -dumpfile=<filename> must be provided. " ) ;
return false ;
}
fs : : path dump_path = dump_filename ;
dump_path = fs : : absolute ( dump_path ) ;
if ( ! fs : : exists ( dump_path ) ) {
error = strprintf ( _ ( " Dump file %s does not exist. " ) , dump_path . string ( ) ) ;
return false ;
}
fsbridge : : ifstream dump_file ( dump_path ) ;
// Compute the checksum
CHashWriter hasher ( 0 , 0 ) ;
uint256 checksum ;
// Check the magic and version
std : : string magic_key ;
std : : getline ( dump_file , magic_key , ' , ' ) ;
std : : string version_value ;
std : : getline ( dump_file , version_value , ' \n ' ) ;
if ( magic_key ! = DUMP_MAGIC ) {
error = strprintf ( _ ( " Error: Dumpfile identifier record is incorrect. Got \" %s \" , expected \" %s \" . " ) , magic_key , DUMP_MAGIC ) ;
dump_file . close ( ) ;
return false ;
}
// Check the version number (value of first record)
uint32_t ver ;
if ( ! ParseUInt32 ( version_value , & ver ) ) {
error = strprintf ( _ ( " Error: Unable to parse version %u as a uint32_t " ) , version_value ) ;
dump_file . close ( ) ;
return false ;
}
if ( ver ! = DUMP_VERSION ) {
error = strprintf ( _ ( " Error: Dumpfile version is not supported. This version of bitcoin-wallet only supports version 1 dumpfiles. Got dumpfile with version %s " ) , version_value ) ;
dump_file . close ( ) ;
return false ;
}
std : : string magic_hasher_line = strprintf ( " %s,%s \n " , magic_key , version_value ) ;
hasher . write ( magic_hasher_line . data ( ) , magic_hasher_line . size ( ) ) ;
// Get the stored file format
std : : string format_key ;
std : : getline ( dump_file , format_key , ' , ' ) ;
std : : string format_value ;
std : : getline ( dump_file , format_value , ' \n ' ) ;
if ( format_key ! = " format " ) {
error = strprintf ( _ ( " Error: Dumpfile format record is incorrect. Got \" %s \" , expected \" format \" . " ) , format_key ) ;
dump_file . close ( ) ;
return false ;
}
// Get the data file format with format_value as the default
std : : string file_format = gArgs . GetArg ( " -format " , format_value ) ;
if ( file_format . empty ( ) ) {
error = _ ( " No wallet file format provided. To use createfromdump, -format=<format> must be provided. " ) ;
return false ;
}
DatabaseFormat data_format ;
if ( file_format = = " bdb " ) {
data_format = DatabaseFormat : : BERKELEY ;
} else if ( file_format = = " sqlite " ) {
data_format = DatabaseFormat : : SQLITE ;
} else {
error = strprintf ( _ ( " Unknown wallet file format \" %s \" provided. Please provide one of \" bdb \" or \" sqlite \" . " ) , file_format ) ;
return false ;
}
if ( file_format ! = format_value ) {
warnings . push_back ( strprintf ( _ ( " Warning: Dumpfile wallet format \" %s \" does not match command line specified format \" %s \" . " ) , format_value , file_format ) ) ;
}
std : : string format_hasher_line = strprintf ( " %s,%s \n " , format_key , format_value ) ;
hasher . write ( format_hasher_line . data ( ) , format_hasher_line . size ( ) ) ;
DatabaseOptions options ;
DatabaseStatus status ;
options . require_create = true ;
options . require_format = data_format ;
std : : unique_ptr < WalletDatabase > database = MakeDatabase ( wallet_path , options , status , error ) ;
if ( ! database ) return false ;
// dummy chain interface
bool ret = true ;
std : : shared_ptr < CWallet > wallet ( new CWallet ( nullptr /* chain */ , name , std : : move ( database ) ) , WalletToolReleaseWallet ) ;
{
LOCK ( wallet - > cs_wallet ) ;
bool first_run = true ;
DBErrors load_wallet_ret = wallet - > LoadWallet ( first_run ) ;
if ( load_wallet_ret ! = DBErrors : : LOAD_OK ) {
error = strprintf ( _ ( " Error creating %s " ) , name ) ;
return false ;
}
// Get the database handle
WalletDatabase & db = wallet - > GetDatabase ( ) ;
std : : unique_ptr < DatabaseBatch > batch = db . MakeBatch ( ) ;
batch - > TxnBegin ( ) ;
// Read the records from the dump file and write them to the database
while ( dump_file . good ( ) ) {
std : : string key ;
std : : getline ( dump_file , key , ' , ' ) ;
std : : string value ;
std : : getline ( dump_file , value , ' \n ' ) ;
if ( key = = " checksum " ) {
std : : vector < unsigned char > parsed_checksum = ParseHex ( value ) ;
std : : copy ( parsed_checksum . begin ( ) , parsed_checksum . end ( ) , checksum . begin ( ) ) ;
break ;
}
std : : string line = strprintf ( " %s,%s \n " , key , value ) ;
hasher . write ( line . data ( ) , line . size ( ) ) ;
if ( key . empty ( ) | | value . empty ( ) ) {
continue ;
}
if ( ! IsHex ( key ) ) {
error = strprintf ( _ ( " Error: Got key that was not hex: %s " ) , key ) ;
ret = false ;
break ;
}
if ( ! IsHex ( value ) ) {
error = strprintf ( _ ( " Error: Got value that was not hex: %s " ) , value ) ;
ret = false ;
break ;
}
std : : vector < unsigned char > k = ParseHex ( key ) ;
std : : vector < unsigned char > v = ParseHex ( value ) ;
CDataStream ss_key ( k , SER_DISK , CLIENT_VERSION ) ;
CDataStream ss_value ( v , SER_DISK , CLIENT_VERSION ) ;
if ( ! batch - > Write ( ss_key , ss_value ) ) {
error = strprintf ( _ ( " Error: Unable to write record to new wallet " ) ) ;
ret = false ;
break ;
}
}
if ( ret ) {
uint256 comp_checksum = hasher . GetHash ( ) ;
if ( checksum . IsNull ( ) ) {
error = _ ( " Error: Missing checksum " ) ;
ret = false ;
} else if ( checksum ! = comp_checksum ) {
error = strprintf ( _ ( " Error: Dumpfile checksum does not match. Computed %s, expected %s " ) , HexStr ( comp_checksum ) , HexStr ( checksum ) ) ;
ret = false ;
}
}
if ( ret ) {
batch - > TxnCommit ( ) ;
} else {
batch - > TxnAbort ( ) ;
}
batch . reset ( ) ;
dump_file . close ( ) ;
}
wallet . reset ( ) ; // The pointer deleter will close the wallet for us.
// Remove the wallet dir if we have a failure
if ( ! ret ) {
fs : : remove_all ( wallet_path ) ;
}
return ret ;
}