2017-07-13 17:49:46 +02:00
#!/usr/bin/env python3
# Copyright (c) 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-06-15 09:05:32 -04:00
""" Test multiwallet.
Verify that a bitcoind node can load multiple wallet files
"""
2017-07-27 00:57:02 +01:00
import os
2017-10-10 15:27:26 -04:00
import shutil
2017-07-27 00:57:02 +01:00
2017-07-13 17:49:46 +02:00
from test_framework . test_framework import BitcoinTestFramework
2017-07-12 10:33:46 -04:00
from test_framework . util import assert_equal , assert_raises_rpc_error
2017-07-13 17:49:46 +02:00
class MultiWalletTest ( BitcoinTestFramework ) :
2017-06-09 18:21:21 -04:00
def set_test_params ( self ) :
2017-07-13 17:49:46 +02:00
self . setup_clean_chain = True
2017-12-15 11:15:18 +13:00
self . num_nodes = 2
self . extra_args = [ [ ' -wallet=w1 ' , ' -wallet=w2 ' , ' -wallet=w3 ' , ' -wallet=w ' ] , [ ] ]
2017-12-20 18:37:34 -05:00
self . supports_cli = True
2017-07-13 17:49:46 +02:00
def run_test ( self ) :
2017-12-20 18:37:34 -05:00
node = self . nodes [ 0 ]
data_dir = lambda * p : os . path . join ( node . datadir , ' regtest ' , * p )
wallet_dir = lambda * p : data_dir ( ' wallets ' , * p )
wallet = lambda name : node . get_wallet_rpc ( name )
assert_equal ( set ( node . listwallets ( ) ) , { " w1 " , " w2 " , " w3 " , " w " } )
2017-09-13 00:41:14 +03:00
2017-12-15 11:15:18 +13:00
self . stop_nodes ( )
2017-07-27 00:56:30 +01:00
2018-01-18 13:15:00 -05:00
self . assert_start_raises_init_error ( 0 , [ ' -walletdir=wallets ' ] , ' Error: Specified -walletdir " wallets " does not exist ' )
self . assert_start_raises_init_error ( 0 , [ ' -walletdir=wallets ' ] , ' Error: Specified -walletdir " wallets " is a relative path ' , cwd = data_dir ( ) )
self . assert_start_raises_init_error ( 0 , [ ' -walletdir=debug.log ' ] , ' Error: Specified -walletdir " debug.log " is not a directory ' , cwd = data_dir ( ) )
2017-07-27 00:56:30 +01:00
# should not initialize if there are duplicate wallets
2017-06-09 16:35:17 -04:00
self . assert_start_raises_init_error ( 0 , [ ' -wallet=w1 ' , ' -wallet=w1 ' ] , ' Error loading wallet w1. Duplicate -wallet filename specified. ' )
2017-07-27 00:56:30 +01:00
2017-07-27 00:57:02 +01:00
# should not initialize if wallet file is a directory
2017-12-20 18:37:34 -05:00
os . mkdir ( wallet_dir ( ' w11 ' ) )
2017-06-09 16:35:17 -04:00
self . assert_start_raises_init_error ( 0 , [ ' -wallet=w11 ' ] , ' Error loading wallet w11. -wallet filename must be a regular file. ' )
2017-07-27 00:57:02 +01:00
2017-10-10 15:27:26 -04:00
# should not initialize if one wallet is a copy of another
2017-12-20 18:37:34 -05:00
shutil . copyfile ( wallet_dir ( ' w2 ' ) , wallet_dir ( ' w22 ' ) )
2017-10-10 15:27:26 -04:00
self . assert_start_raises_init_error ( 0 , [ ' -wallet=w2 ' , ' -wallet=w22 ' ] , ' duplicates fileid ' )
2017-07-27 00:57:02 +01:00
# should not initialize if wallet file is a symlink
2017-12-20 18:37:34 -05:00
os . symlink ( wallet_dir ( ' w1 ' ) , wallet_dir ( ' w12 ' ) )
2017-06-09 16:35:17 -04:00
self . assert_start_raises_init_error ( 0 , [ ' -wallet=w12 ' ] , ' Error loading wallet w12. -wallet filename must be a regular file. ' )
2017-07-27 00:57:02 +01:00
2017-10-09 21:32:07 +13:00
# should not initialize if the specified walletdir does not exist
2017-11-18 13:36:37 +13:00
self . assert_start_raises_init_error ( 0 , [ ' -walletdir=bad ' ] , ' Error: Specified -walletdir " bad " does not exist ' )
# should not initialize if the specified walletdir is not a directory
2017-12-20 18:37:34 -05:00
not_a_dir = wallet_dir ( ' notadir ' )
2017-11-18 13:36:37 +13:00
open ( not_a_dir , ' a ' ) . close ( )
2017-12-20 18:37:34 -05:00
self . assert_start_raises_init_error ( 0 , [ ' -walletdir= ' + not_a_dir ] , ' Error: Specified -walletdir " ' + not_a_dir + ' " is not a directory ' )
2017-10-09 21:32:07 +13:00
2018-04-19 11:13:47 -04:00
self . log . info ( " Do not allow -zapwallettxes with multiwallet " )
self . assert_start_raises_init_error ( 0 , [ ' -zapwallettxes ' , ' -wallet=w1 ' , ' -wallet=w2 ' ] , " Error: -zapwallettxes is only allowed with a single wallet file " )
self . assert_start_raises_init_error ( 0 , [ ' -zapwallettxes=1 ' , ' -wallet=w1 ' , ' -wallet=w2 ' ] , " Error: -zapwallettxes is only allowed with a single wallet file " )
self . assert_start_raises_init_error ( 0 , [ ' -zapwallettxes=2 ' , ' -wallet=w1 ' , ' -wallet=w2 ' ] , " Error: -zapwallettxes is only allowed with a single wallet file " )
self . log . info ( " Do not allow -salvagewallet with multiwallet " )
self . assert_start_raises_init_error ( 0 , [ ' -salvagewallet ' , ' -wallet=w1 ' , ' -wallet=w2 ' ] , " Error: -salvagewallet is only allowed with a single wallet file " )
self . assert_start_raises_init_error ( 0 , [ ' -salvagewallet=1 ' , ' -wallet=w1 ' , ' -wallet=w2 ' ] , " Error: -salvagewallet is only allowed with a single wallet file " )
self . log . info ( " Do not allow -upgradewallet with multiwallet " )
self . assert_start_raises_init_error ( 0 , [ ' -upgradewallet ' , ' -wallet=w1 ' , ' -wallet=w2 ' ] , " Error: -upgradewallet is only allowed with a single wallet file " )
self . assert_start_raises_init_error ( 0 , [ ' -upgradewallet=1 ' , ' -wallet=w1 ' , ' -wallet=w2 ' ] , " Error: -upgradewallet is only allowed with a single wallet file " )
2017-10-12 22:04:46 +13:00
# if wallets/ doesn't exist, datadir should be the default wallet dir
2017-12-20 18:37:34 -05:00
wallet_dir2 = data_dir ( ' walletdir ' )
os . rename ( wallet_dir ( ) , wallet_dir2 )
2017-10-12 22:04:46 +13:00
self . start_node ( 0 , [ ' -wallet=w4 ' , ' -wallet=w5 ' ] )
2017-12-20 18:37:34 -05:00
assert_equal ( set ( node . listwallets ( ) ) , { " w4 " , " w5 " } )
w5 = wallet ( " w5 " )
2017-10-12 22:04:46 +13:00
w5 . generate ( 1 )
# now if wallets/ exists again, but the rootdir is specified as the walletdir, w4 and w5 should still be loaded
2017-12-20 18:37:34 -05:00
os . rename ( wallet_dir2 , wallet_dir ( ) )
2017-12-15 11:15:18 +13:00
self . restart_node ( 0 , [ ' -wallet=w4 ' , ' -wallet=w5 ' , ' -walletdir= ' + data_dir ( ) ] )
2017-12-20 18:37:34 -05:00
assert_equal ( set ( node . listwallets ( ) ) , { " w4 " , " w5 " } )
w5 = wallet ( " w5 " )
2017-10-09 21:32:07 +13:00
w5_info = w5 . getwalletinfo ( )
2019-02-07 13:50:53 -08:00
assert_equal ( w5_info [ ' immature_balance ' ] , 500 )
2017-10-09 21:32:07 +13:00
2017-12-15 11:15:18 +13:00
competing_wallet_dir = os . path . join ( self . options . tmpdir , ' competing_walletdir ' )
os . mkdir ( competing_wallet_dir )
self . restart_node ( 0 , [ ' -walletdir= ' + competing_wallet_dir ] )
2017-12-24 12:45:33 +13:00
self . assert_start_raises_init_error ( 1 , [ ' -walletdir= ' + competing_wallet_dir ] , ' Error initializing wallet database environment ' )
2017-10-09 21:32:07 +13:00
2017-12-15 11:15:18 +13:00
self . restart_node ( 0 , self . extra_args [ 0 ] )
2017-07-27 00:56:30 +01:00
2017-12-20 18:37:34 -05:00
w1 = wallet ( " w1 " )
w2 = wallet ( " w2 " )
w3 = wallet ( " w3 " )
w4 = wallet ( " w " )
wallet_bad = wallet ( " bad " )
2017-06-02 14:30:36 -04:00
2017-07-13 17:49:46 +02:00
w1 . generate ( 1 )
2017-07-17 05:42:30 -04:00
# accessing invalid wallet fails
2017-07-12 10:33:46 -04:00
assert_raises_rpc_error ( - 18 , " Requested wallet does not exist or is not loaded " , wallet_bad . getwalletinfo )
2017-07-17 05:42:30 -04:00
2017-06-15 09:05:32 -04:00
# accessing wallet RPC without using wallet endpoint fails
2017-12-20 18:37:34 -05:00
assert_raises_rpc_error ( - 19 , " Wallet file not specified " , node . getwalletinfo )
2017-07-13 17:49:46 +02:00
2017-06-15 09:05:32 -04:00
# check w1 wallet balance
w1_info = w1 . getwalletinfo ( )
2019-02-07 13:50:53 -08:00
assert_equal ( w1_info [ ' immature_balance ' ] , 500 )
2017-06-15 09:05:32 -04:00
w1_name = w1_info [ ' walletname ' ]
assert_equal ( w1_name , " w1 " )
2017-07-13 17:49:46 +02:00
2017-06-02 14:30:36 -04:00
# check w2 wallet balance
2017-06-15 09:05:32 -04:00
w2_info = w2 . getwalletinfo ( )
assert_equal ( w2_info [ ' immature_balance ' ] , 0 )
w2_name = w2_info [ ' walletname ' ]
assert_equal ( w2_name , " w2 " )
2017-07-13 17:49:46 +02:00
2017-06-15 09:05:32 -04:00
w3_name = w3 . getwalletinfo ( ) [ ' walletname ' ]
assert_equal ( w3_name , " w3 " )
2017-11-15 11:03:49 -05:00
w4_name = w4 . getwalletinfo ( ) [ ' walletname ' ]
assert_equal ( w4_name , " w " )
2017-06-15 09:05:32 -04:00
2017-07-13 17:49:46 +02:00
w1 . generate ( 101 )
2019-02-07 13:50:53 -08:00
assert_equal ( w1 . getbalance ( ) , 1000 )
2017-07-13 17:49:46 +02:00
assert_equal ( w2 . getbalance ( ) , 0 )
assert_equal ( w3 . getbalance ( ) , 0 )
2017-11-15 11:03:49 -05:00
assert_equal ( w4 . getbalance ( ) , 0 )
2017-07-13 17:49:46 +02:00
w1 . sendtoaddress ( w2 . getnewaddress ( ) , 1 )
w1 . sendtoaddress ( w3 . getnewaddress ( ) , 2 )
2017-11-15 11:03:49 -05:00
w1 . sendtoaddress ( w4 . getnewaddress ( ) , 3 )
2017-07-13 17:49:46 +02:00
w1 . generate ( 1 )
assert_equal ( w2 . getbalance ( ) , 1 )
assert_equal ( w3 . getbalance ( ) , 2 )
2017-11-15 11:03:49 -05:00
assert_equal ( w4 . getbalance ( ) , 3 )
2017-07-13 17:49:46 +02:00
2017-09-07 17:40:25 -04:00
batch = w1 . batch ( [ w1 . getblockchaininfo . get_request ( ) , w1 . getwalletinfo . get_request ( ) ] )
assert_equal ( batch [ 0 ] [ " result " ] [ " chain " ] , " regtest " )
assert_equal ( batch [ 1 ] [ " result " ] [ " walletname " ] , " w1 " )
2017-07-13 17:49:46 +02:00
if __name__ == ' __main__ ' :
MultiWalletTest ( ) . main ( )