2017-08-02 07:19:28 -04:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
// Copyright (c) 2009-2017 The Bitcoin Core developers
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-12-12 17:32:50 -03:00
# include <chainparams.h>
2018-03-30 05:48:29 -03:00
# include <init.h>
2017-11-09 21:57:53 -03:00
# include <net.h>
# include <util.h>
# include <utilmoneystr.h>
# include <validation.h>
2018-03-29 06:08:38 -03:00
# include <walletinitinterface.h>
2017-11-09 21:57:53 -03:00
# include <wallet/rpcwallet.h>
2017-10-08 17:48:07 -03:00
# include <wallet/wallet.h>
# include <wallet/walletutil.h>
2017-08-02 07:48:52 -04:00
2018-03-29 06:08:38 -03:00
class WalletInit : public WalletInitInterface {
public :
//! Return the wallets help message.
2018-04-28 17:54:58 -03:00
void AddWalletOptions ( ) const override ;
2018-03-29 06:08:38 -03:00
//! Wallets parameter interaction
2018-04-13 10:02:59 -03:00
bool ParameterInteraction ( ) const override ;
2018-03-29 06:08:38 -03:00
//! Register wallet RPCs.
2018-04-13 10:02:59 -03:00
void RegisterRPC ( CRPCTable & tableRPC ) const override ;
2018-03-29 06:08:38 -03:00
//! Responsible for reading and validating the -wallet arguments and verifying the wallet database.
// This function will perform salvage on the wallet if requested, as long as only one wallet is
// being loaded (WalletParameterInteraction forbids -salvagewallet, -zapwallettxes or -upgradewallet with multiwallet).
2018-04-13 10:02:59 -03:00
bool Verify ( ) const override ;
2018-03-29 06:08:38 -03:00
//! Load wallet databases.
2018-04-13 10:02:59 -03:00
bool Open ( ) const override ;
2018-03-29 06:08:38 -03:00
//! Complete startup of wallets.
2018-04-13 10:02:59 -03:00
void Start ( CScheduler & scheduler ) const override ;
2018-03-29 06:08:38 -03:00
//! Flush all wallets in preparation for shutdown.
2018-04-13 10:02:59 -03:00
void Flush ( ) const override ;
2018-03-29 06:08:38 -03:00
//! Stop all wallets. Wallets will be flushed first.
2018-04-13 10:02:59 -03:00
void Stop ( ) const override ;
2018-03-29 06:08:38 -03:00
//! Close all wallets.
2018-04-13 10:02:59 -03:00
void Close ( ) const override ;
2018-03-29 06:08:38 -03:00
} ;
2018-04-13 10:05:55 -03:00
const WalletInitInterface & g_wallet_init_interface = WalletInit ( ) ;
2018-03-29 06:08:38 -03:00
2018-04-28 17:54:58 -03:00
void WalletInit : : AddWalletOptions ( ) const
2017-08-02 07:48:52 -04:00
{
2018-04-28 17:54:58 -03:00
gArgs . AddArg ( " -addresstype " , strprintf ( " What type of addresses to use ( \" legacy \" , \" p2sh-segwit \" , or \" bech32 \" , default: \" %s \" ) " , FormatOutputType ( DEFAULT_ADDRESS_TYPE ) ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -changetype " , " What type of change to use ( \" legacy \" , \" p2sh-segwit \" , or \" bech32 \" ). Default is same as -addresstype, except when -addresstype=p2sh-segwit a native segwit output is used when sending to a native segwit address) " , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -disablewallet " , _ ( " Do not load the wallet and disable wallet RPC calls " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -discardfee=<amt> " , strprintf ( _ ( " The fee rate (in %s/kB) that indicates your tolerance for discarding change by adding it to the fee (default: %s). "
2017-08-02 07:48:52 -04:00
" Note: An output is discarded if it is dust at this rate, but we will always discard up to the dust relay fee and a discard fee above that is limited by the fee estimate for the longest target " ) ,
2018-04-28 17:54:58 -03:00
CURRENCY_UNIT , FormatMoney ( DEFAULT_DISCARD_FEE ) ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -fallbackfee=<amt> " , strprintf ( _ ( " A fee rate (in %s/kB) that will be used when fee estimation has insufficient data (default: %s) " ) ,
CURRENCY_UNIT , FormatMoney ( DEFAULT_FALLBACK_FEE ) ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -keypool=<n> " , strprintf ( _ ( " Set key pool size to <n> (default: %u) " ) , DEFAULT_KEYPOOL_SIZE ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -mintxfee=<amt> " , strprintf ( _ ( " Fees (in %s/kB) smaller than this are considered zero fee for transaction creation (default: %s) " ) ,
CURRENCY_UNIT , FormatMoney ( DEFAULT_TRANSACTION_MINFEE ) ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -paytxfee=<amt> " , strprintf ( _ ( " Fee (in %s/kB) to add to transactions you send (default: %s) " ) ,
CURRENCY_UNIT , FormatMoney ( CFeeRate { DEFAULT_PAY_TX_FEE } . GetFeePerK ( ) ) ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -rescan " , _ ( " Rescan the block chain for missing wallet transactions on startup " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -salvagewallet " , _ ( " Attempt to recover private keys from a corrupt wallet on startup " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -spendzeroconfchange " , strprintf ( _ ( " Spend unconfirmed change when sending transactions (default: %u) " ) , DEFAULT_SPEND_ZEROCONF_CHANGE ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -txconfirmtarget=<n> " , strprintf ( _ ( " If paytxfee is not set, include enough fee so transactions begin confirmation on average within n blocks (default: %u) " ) , DEFAULT_TX_CONFIRM_TARGET ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -upgradewallet " , _ ( " Upgrade wallet to latest format on startup " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -wallet=<path> " , _ ( " Specify wallet database path. Can be specified multiple times to load multiple wallets. Path is interpreted relative to <walletdir> if it is not absolute, and will be created if it does not exist (as a directory containing a wallet.dat file and log files). For backwards compatibility this will also accept names of existing data files in <walletdir>.) " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -walletbroadcast " , _ ( " Make the wallet broadcast transactions " ) + " " + strprintf ( _ ( " (default: %u) " ) , DEFAULT_WALLETBROADCAST ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -walletdir=<dir> " , _ ( " Specify directory to hold wallets (default: <datadir>/wallets if it exists, otherwise <datadir>) " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -walletnotify=<cmd> " , _ ( " Execute command when a wallet transaction changes (%s in cmd is replaced by TxID) " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -walletrbf " , strprintf ( _ ( " Send transactions with full-RBF opt-in enabled (RPC only, default: %u) " ) , DEFAULT_WALLET_RBF ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -zapwallettxes=<mode> " , _ ( " Delete all wallet transactions and only recover those parts of the blockchain through -rescan on startup " ) +
" " + _ ( " (1 = keep tx meta data e.g. account owner and payment request information, 2 = drop tx meta data) " ) , false , OptionsCategory : : WALLET ) ;
gArgs . AddArg ( " -dblogsize=<n> " , strprintf ( " Flush wallet database activity from memory to disk log every <n> megabytes (default: %u) " , DEFAULT_WALLET_DBLOGSIZE ) , true , OptionsCategory : : WALLET_DEBUG_TEST ) ;
gArgs . AddArg ( " -flushwallet " , strprintf ( " Run a thread to flush wallet periodically (default: %u) " , DEFAULT_FLUSHWALLET ) , true , OptionsCategory : : WALLET_DEBUG_TEST ) ;
gArgs . AddArg ( " -privdb " , strprintf ( " Sets the DB_PRIVATE flag in the wallet db environment (default: %u) " , DEFAULT_WALLET_PRIVDB ) , true , OptionsCategory : : WALLET_DEBUG_TEST ) ;
gArgs . AddArg ( " -walletrejectlongchains " , strprintf ( _ ( " Wallet will not create transactions that violate mempool chain limits (default: %u) " ) , DEFAULT_WALLET_REJECT_LONG_CHAINS ) , true , OptionsCategory : : WALLET_DEBUG_TEST ) ;
2017-08-02 07:48:52 -04:00
}
2018-04-13 10:02:59 -03:00
bool WalletInit : : ParameterInteraction ( ) const
2017-08-02 07:48:52 -04:00
{
2017-11-01 21:39:23 -03:00
if ( gArgs . GetBoolArg ( " -disablewallet " , DEFAULT_DISABLE_WALLET ) ) {
for ( const std : : string & wallet : gArgs . GetArgs ( " -wallet " ) ) {
LogPrintf ( " %s: parameter interaction: -disablewallet -> ignoring -wallet=%s \n " , __func__ , wallet ) ;
}
2017-08-02 07:48:52 -04:00
return true ;
2017-11-01 21:39:23 -03:00
}
2017-11-15 17:44:36 -03:00
gArgs . SoftSetArg ( " -wallet " , " " ) ;
2017-11-01 21:39:23 -03:00
const bool is_multiwallet = gArgs . GetArgs ( " -wallet " ) . size ( ) > 1 ;
2017-08-02 07:48:52 -04:00
if ( gArgs . GetBoolArg ( " -blocksonly " , DEFAULT_BLOCKSONLY ) & & gArgs . SoftSetBoolArg ( " -walletbroadcast " , false ) ) {
LogPrintf ( " %s: parameter interaction: -blocksonly=1 -> setting -walletbroadcast=0 \n " , __func__ ) ;
}
if ( gArgs . GetBoolArg ( " -salvagewallet " , false ) ) {
if ( is_multiwallet ) {
return InitError ( strprintf ( " %s is only allowed with a single wallet file " , " -salvagewallet " ) ) ;
}
// Rewrite just private keys: rescan to find transactions
if ( gArgs . SoftSetBoolArg ( " -rescan " , true ) ) {
LogPrintf ( " %s: parameter interaction: -salvagewallet=1 -> setting -rescan=1 \n " , __func__ ) ;
}
}
2018-04-19 12:07:54 -03:00
bool zapwallettxes = gArgs . GetBoolArg ( " -zapwallettxes " , false ) ;
2017-08-02 07:48:52 -04:00
// -zapwallettxes implies dropping the mempool on startup
2018-04-19 12:07:54 -03:00
if ( zapwallettxes & & gArgs . SoftSetBoolArg ( " -persistmempool " , false ) ) {
LogPrintf ( " %s: parameter interaction: -zapwallettxes enabled -> setting -persistmempool=0 \n " , __func__ ) ;
2017-08-02 07:48:52 -04:00
}
// -zapwallettxes implies a rescan
2018-04-19 12:07:54 -03:00
if ( zapwallettxes ) {
2017-08-02 07:48:52 -04:00
if ( is_multiwallet ) {
return InitError ( strprintf ( " %s is only allowed with a single wallet file " , " -zapwallettxes " ) ) ;
}
if ( gArgs . SoftSetBoolArg ( " -rescan " , true ) ) {
2018-04-19 12:07:54 -03:00
LogPrintf ( " %s: parameter interaction: -zapwallettxes enabled -> setting -rescan=1 \n " , __func__ ) ;
2017-08-02 07:48:52 -04:00
}
}
if ( is_multiwallet ) {
if ( gArgs . GetBoolArg ( " -upgradewallet " , false ) ) {
return InitError ( strprintf ( " %s is only allowed with a single wallet file " , " -upgradewallet " ) ) ;
}
}
if ( gArgs . GetBoolArg ( " -sysperms " , false ) )
return InitError ( " -sysperms is not allowed in combination with enabled wallet functionality " ) ;
if ( gArgs . GetArg ( " -prune " , 0 ) & & gArgs . GetBoolArg ( " -rescan " , false ) )
return InitError ( _ ( " Rescans are not possible in pruned mode. You will need to use -reindex which will download the whole blockchain again. " ) ) ;
if ( : : minRelayTxFee . GetFeePerK ( ) > HIGH_TX_FEE_PER_KB )
InitWarning ( AmountHighWarn ( " -minrelaytxfee " ) + " " +
_ ( " The wallet will avoid paying less than the minimum relay fee. " ) ) ;
if ( gArgs . IsArgSet ( " -maxtxfee " ) )
{
CAmount nMaxFee = 0 ;
if ( ! ParseMoney ( gArgs . GetArg ( " -maxtxfee " , " " ) , nMaxFee ) )
return InitError ( AmountErrMsg ( " maxtxfee " , gArgs . GetArg ( " -maxtxfee " , " " ) ) ) ;
if ( nMaxFee > HIGH_MAX_TX_FEE )
InitWarning ( _ ( " -maxtxfee is set very high! Fees this large could be paid on a single transaction. " ) ) ;
maxTxFee = nMaxFee ;
if ( CFeeRate ( maxTxFee , 1000 ) < : : minRelayTxFee )
{
return InitError ( strprintf ( _ ( " Invalid amount for -maxtxfee=<amount>: '%s' (must be at least the minrelay fee of %s to prevent stuck transactions) " ) ,
gArgs . GetArg ( " -maxtxfee " , " " ) , : : minRelayTxFee . ToString ( ) ) ) ;
}
}
return true ;
}
2018-04-13 10:02:59 -03:00
void WalletInit : : RegisterRPC ( CRPCTable & t ) const
2017-08-28 14:33:59 -03:00
{
2017-11-01 21:41:29 -03:00
if ( gArgs . GetBoolArg ( " -disablewallet " , DEFAULT_DISABLE_WALLET ) ) {
return ;
}
2017-08-28 14:33:59 -03:00
RegisterWalletRPCCommands ( t ) ;
}
2018-04-13 10:02:59 -03:00
bool WalletInit : : Verify ( ) const
2017-08-02 07:48:52 -04:00
{
2017-11-01 21:41:29 -03:00
if ( gArgs . GetBoolArg ( " -disablewallet " , DEFAULT_DISABLE_WALLET ) ) {
2017-08-02 07:48:52 -04:00
return true ;
2017-11-01 21:41:29 -03:00
}
2017-08-02 07:48:52 -04:00
2018-01-18 15:15:00 -03:00
if ( gArgs . IsArgSet ( " -walletdir " ) ) {
fs : : path wallet_dir = gArgs . GetArg ( " -walletdir " , " " ) ;
if ( ! fs : : exists ( wallet_dir ) ) {
return InitError ( strprintf ( _ ( " Specified -walletdir \" %s \" does not exist " ) , wallet_dir . string ( ) ) ) ;
} else if ( ! fs : : is_directory ( wallet_dir ) ) {
return InitError ( strprintf ( _ ( " Specified -walletdir \" %s \" is not a directory " ) , wallet_dir . string ( ) ) ) ;
} else if ( ! wallet_dir . is_absolute ( ) ) {
return InitError ( strprintf ( _ ( " Specified -walletdir \" %s \" is a relative path " ) , wallet_dir . string ( ) ) ) ;
2017-11-17 21:36:37 -03:00
}
2017-10-12 06:04:46 -03:00
}
LogPrintf ( " Using wallet directory %s \n " , GetWalletDir ( ) . string ( ) ) ;
2017-08-02 07:48:52 -04:00
uiInterface . InitMessage ( _ ( " Verifying wallet(s)... " ) ) ;
// Keep track of each wallet absolute path to detect duplicates.
std : : set < fs : : path > wallet_paths ;
for ( const std : : string & walletFile : gArgs . GetArgs ( " -wallet " ) ) {
2017-11-15 17:44:36 -03:00
// Do some checking on wallet path. It should be either a:
//
// 1. Path where a directory can be created.
// 2. Path to an existing directory.
// 3. Path to a symlink to a directory.
// 4. For backwards compatibility, the name of a data file in -walletdir.
2017-10-08 17:48:07 -03:00
fs : : path wallet_path = fs : : absolute ( walletFile , GetWalletDir ( ) ) ;
2017-11-15 17:44:36 -03:00
fs : : file_type path_type = fs : : symlink_status ( wallet_path ) . type ( ) ;
if ( ! ( path_type = = fs : : file_not_found | | path_type = = fs : : directory_file | |
( path_type = = fs : : symlink_file & & fs : : is_directory ( wallet_path ) ) | |
( path_type = = fs : : regular_file & & fs : : path ( walletFile ) . filename ( ) = = walletFile ) ) ) {
return InitError ( strprintf (
_ ( " Invalid -wallet path '%s'. -wallet path should point to a directory where wallet.dat and "
" database/log.?????????? files can be stored, a location where such a directory could be created, "
" or (for backwards compatibility) the name of an existing data file in -walletdir (%s) " ) ,
walletFile , GetWalletDir ( ) ) ) ;
2017-08-02 07:48:52 -04:00
}
if ( ! wallet_paths . insert ( wallet_path ) . second ) {
return InitError ( strprintf ( _ ( " Error loading wallet %s. Duplicate -wallet filename specified. " ) , walletFile ) ) ;
}
std : : string strError ;
2017-12-08 08:39:22 -03:00
if ( ! WalletBatch : : VerifyEnvironment ( wallet_path , strError ) ) {
2017-08-02 07:48:52 -04:00
return InitError ( strError ) ;
}
if ( gArgs . GetBoolArg ( " -salvagewallet " , false ) ) {
// Recover readable keypairs:
2017-12-08 08:39:22 -03:00
CWallet dummyWallet ( " dummy " , WalletDatabase : : CreateDummy ( ) ) ;
2017-08-02 07:48:52 -04:00
std : : string backup_filename ;
2017-12-08 08:39:22 -03:00
if ( ! WalletBatch : : Recover ( wallet_path , ( void * ) & dummyWallet , WalletBatch : : RecoverKeysOnlyFilter , backup_filename ) ) {
2017-08-02 07:48:52 -04:00
return false ;
}
}
std : : string strWarning ;
2017-12-08 08:39:22 -03:00
bool dbV = WalletBatch : : VerifyDatabaseFile ( wallet_path , strWarning , strError ) ;
2017-08-02 07:48:52 -04:00
if ( ! strWarning . empty ( ) ) {
InitWarning ( strWarning ) ;
}
if ( ! dbV ) {
InitError ( strError ) ;
return false ;
}
}
return true ;
}
2018-04-13 10:02:59 -03:00
bool WalletInit : : Open ( ) const
2017-08-02 07:48:52 -04:00
{
if ( gArgs . GetBoolArg ( " -disablewallet " , DEFAULT_DISABLE_WALLET ) ) {
LogPrintf ( " Wallet disabled! \n " ) ;
return true ;
}
for ( const std : : string & walletFile : gArgs . GetArgs ( " -wallet " ) ) {
2017-11-13 23:25:46 -03:00
CWallet * const pwallet = CWallet : : CreateWalletFromFile ( walletFile , fs : : absolute ( walletFile , GetWalletDir ( ) ) ) ;
2017-08-02 07:48:52 -04:00
if ( ! pwallet ) {
return false ;
}
2018-04-17 14:22:23 -03:00
AddWallet ( pwallet ) ;
2017-08-02 07:48:52 -04:00
}
return true ;
}
2017-08-28 13:31:53 -03:00
2018-04-13 10:02:59 -03:00
void WalletInit : : Start ( CScheduler & scheduler ) const
2017-07-06 06:16:40 -04:00
{
2018-04-17 14:22:23 -03:00
for ( CWallet * pwallet : GetWallets ( ) ) {
2017-08-29 12:47:06 -03:00
pwallet - > postInitProcess ( scheduler ) ;
}
}
2018-04-13 10:02:59 -03:00
void WalletInit : : Flush ( ) const
2017-07-06 06:16:40 -04:00
{
2018-04-17 14:22:23 -03:00
for ( CWallet * pwallet : GetWallets ( ) ) {
2017-08-28 13:53:56 -03:00
pwallet - > Flush ( false ) ;
}
}
2018-04-13 10:02:59 -03:00
void WalletInit : : Stop ( ) const
2017-07-06 06:16:40 -04:00
{
2018-04-17 14:22:23 -03:00
for ( CWallet * pwallet : GetWallets ( ) ) {
2017-08-28 13:53:56 -03:00
pwallet - > Flush ( true ) ;
2017-08-28 13:31:53 -03:00
}
}
2017-08-28 13:41:33 -03:00
2018-04-13 10:02:59 -03:00
void WalletInit : : Close ( ) const
2017-07-06 06:16:40 -04:00
{
2018-04-17 14:22:23 -03:00
for ( CWallet * pwallet : GetWallets ( ) ) {
RemoveWallet ( pwallet ) ;
2017-08-28 13:41:33 -03:00
delete pwallet ;
}
}