2017-07-18 17:15: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-07-21 15:34:11 +02:00
|
|
|
"""Test multiwallet.
|
|
|
|
|
2020-06-11 10:39:04 +02:00
|
|
|
Verify that a dashd node can load multiple wallet files
|
2017-07-21 15:34:11 +02:00
|
|
|
"""
|
2017-07-28 12:42:28 +02:00
|
|
|
import os
|
2017-10-19 18:12:59 +02:00
|
|
|
import shutil
|
2017-07-28 12:42:28 +02:00
|
|
|
|
2017-07-18 17:15:46 +02:00
|
|
|
from test_framework.test_framework import BitcoinTestFramework
|
2019-09-25 11:34:51 +02:00
|
|
|
from test_framework.util import assert_equal, assert_raises_rpc_error
|
2017-07-18 17:15:46 +02:00
|
|
|
|
|
|
|
class MultiWalletTest(BitcoinTestFramework):
|
2017-09-01 18:47:13 +02:00
|
|
|
def set_test_params(self):
|
2017-07-18 17:15:46 +02:00
|
|
|
self.setup_clean_chain = True
|
2018-01-16 10:54:13 +01:00
|
|
|
self.num_nodes = 2
|
2018-01-12 23:24:36 +01:00
|
|
|
self.supports_cli = True
|
2017-07-18 17:15:46 +02:00
|
|
|
|
|
|
|
def run_test(self):
|
2018-01-12 23:24:36 +01: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)
|
|
|
|
|
2018-03-07 17:05:08 +01:00
|
|
|
# check wallet.dat is created
|
2018-01-16 10:54:13 +01:00
|
|
|
self.stop_nodes()
|
2018-03-07 17:05:08 +01:00
|
|
|
assert_equal(os.path.isfile(wallet_dir('wallet.dat')), True)
|
|
|
|
|
|
|
|
# create symlink to verify wallet directory path can be referenced
|
|
|
|
# through symlink
|
|
|
|
os.mkdir(wallet_dir('w7'))
|
|
|
|
os.symlink('w7', wallet_dir('w7_symlink'))
|
|
|
|
|
|
|
|
# rename wallet.dat to make sure plain wallet file paths (as opposed to
|
|
|
|
# directory paths) can be loaded
|
|
|
|
os.rename(wallet_dir("wallet.dat"), wallet_dir("w8"))
|
|
|
|
|
|
|
|
# restart node with a mix of wallet names:
|
|
|
|
# w1, w2, w3 - to verify new wallets created when non-existing paths specified
|
|
|
|
# w - to verify wallet name matching works when one wallet path is prefix of another
|
|
|
|
# sub/w5 - to verify relative wallet path is created correctly
|
|
|
|
# extern/w6 - to verify absolute wallet path is created correctly
|
|
|
|
# w7_symlink - to verify symlinked wallet path is initialized correctly
|
|
|
|
# w8 - to verify existing wallet file is loaded correctly
|
|
|
|
# '' - to verify default wallet file is created correctly
|
|
|
|
wallet_names = ['w1', 'w2', 'w3', 'w', 'sub/w5', os.path.join(self.options.tmpdir, 'extern/w6'), 'w7_symlink', 'w8', '']
|
|
|
|
extra_args = ['-wallet={}'.format(n) for n in wallet_names]
|
|
|
|
self.start_node(0, extra_args)
|
|
|
|
assert_equal(set(node.listwallets()), set(wallet_names))
|
|
|
|
|
|
|
|
# check that all requested wallets were created
|
|
|
|
self.stop_node(0)
|
|
|
|
for wallet_name in wallet_names:
|
|
|
|
if os.path.isdir(wallet_dir(wallet_name)):
|
|
|
|
assert_equal(os.path.isfile(wallet_dir(wallet_name, "wallet.dat")), True)
|
|
|
|
else:
|
|
|
|
assert_equal(os.path.isfile(wallet_dir(wallet_name)), True)
|
|
|
|
|
|
|
|
# should not initialize if wallet path can't be created
|
|
|
|
self.assert_start_raises_init_error(0, ['-wallet=wallet.dat/bad'], 'Not a directory')
|
2017-07-28 12:42:28 +02:00
|
|
|
|
2018-01-19 17:44:27 +01: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-28 12:42:28 +02:00
|
|
|
# should not initialize if there are duplicate wallets
|
2017-09-01 18:47:13 +02:00
|
|
|
self.assert_start_raises_init_error(0, ['-wallet=w1', '-wallet=w1'], 'Error loading wallet w1. Duplicate -wallet filename specified.')
|
2017-07-28 12:42:28 +02:00
|
|
|
|
2017-10-19 18:12:59 +02:00
|
|
|
# should not initialize if one wallet is a copy of another
|
2018-03-07 17:05:08 +01:00
|
|
|
shutil.copyfile(wallet_dir('w8'), wallet_dir('w8_copy'))
|
|
|
|
self.assert_start_raises_init_error(0, ['-wallet=w8', '-wallet=w8_copy'], 'duplicates fileid')
|
2017-10-19 18:12:59 +02:00
|
|
|
|
2017-07-28 12:42:28 +02:00
|
|
|
# should not initialize if wallet file is a symlink
|
2018-03-07 17:05:08 +01:00
|
|
|
os.symlink('w8', wallet_dir('w8_symlink'))
|
|
|
|
self.assert_start_raises_init_error(0, ['-wallet=w8_symlink'], 'Invalid -wallet path')
|
2017-07-28 12:42:28 +02:00
|
|
|
|
2017-11-18 14:32:50 +01:00
|
|
|
# should not initialize if the specified walletdir does not exist
|
2017-12-20 23:37:52 +01: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
|
2018-01-12 23:24:36 +01:00
|
|
|
not_a_dir = wallet_dir('notadir')
|
2017-12-20 23:37:52 +01:00
|
|
|
open(not_a_dir, 'a').close()
|
2018-01-12 23:24:36 +01:00
|
|
|
self.assert_start_raises_init_error(0, ['-walletdir=' + not_a_dir], 'Error: Specified -walletdir "' + not_a_dir + '" is not a directory')
|
2017-11-18 14:32:50 +01:00
|
|
|
|
|
|
|
# if wallets/ doesn't exist, datadir should be the default wallet dir
|
2018-01-12 23:24:36 +01:00
|
|
|
wallet_dir2 = data_dir('walletdir')
|
|
|
|
os.rename(wallet_dir(), wallet_dir2)
|
2017-11-18 14:32:50 +01:00
|
|
|
self.start_node(0, ['-wallet=w4', '-wallet=w5'])
|
2018-01-12 23:24:36 +01:00
|
|
|
assert_equal(set(node.listwallets()), {"w4", "w5"})
|
|
|
|
w5 = wallet("w5")
|
2017-11-18 14:32:50 +01:00
|
|
|
w5.generate(1)
|
|
|
|
|
|
|
|
# now if wallets/ exists again, but the rootdir is specified as the walletdir, w4 and w5 should still be loaded
|
2018-01-12 23:24:36 +01:00
|
|
|
os.rename(wallet_dir2, wallet_dir())
|
2018-01-16 10:54:13 +01:00
|
|
|
self.restart_node(0, ['-wallet=w4', '-wallet=w5', '-walletdir=' + data_dir()])
|
2018-01-12 23:24:36 +01:00
|
|
|
assert_equal(set(node.listwallets()), {"w4", "w5"})
|
|
|
|
w5 = wallet("w5")
|
2017-11-18 14:32:50 +01:00
|
|
|
w5_info = w5.getwalletinfo()
|
2020-02-06 16:45:57 +01:00
|
|
|
assert_equal(w5_info['immature_balance'], 500)
|
2017-11-18 14:32:50 +01:00
|
|
|
|
2018-01-16 10:54:13 +01: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])
|
|
|
|
self.assert_start_raises_init_error(1, ['-walletdir='+competing_wallet_dir], 'Error initializing wallet database environment')
|
2017-11-18 14:32:50 +01:00
|
|
|
|
2018-03-07 17:05:08 +01:00
|
|
|
self.restart_node(0, extra_args)
|
2017-07-28 12:42:28 +02:00
|
|
|
|
2018-03-07 17:05:08 +01:00
|
|
|
wallets = [wallet(w) for w in wallet_names]
|
2018-01-12 23:24:36 +01:00
|
|
|
wallet_bad = wallet("bad")
|
2017-08-15 23:34:07 +02:00
|
|
|
|
2018-03-07 17:05:08 +01:00
|
|
|
# check wallet names and balances
|
|
|
|
wallets[0].generate(1)
|
|
|
|
for wallet_name, wallet in zip(wallet_names, wallets):
|
|
|
|
info = wallet.getwalletinfo()
|
2020-04-15 01:27:49 +02:00
|
|
|
assert_equal(info['immature_balance'], 500 if wallet is wallets[0] else 0)
|
2018-03-07 17:05:08 +01:00
|
|
|
assert_equal(info['walletname'], wallet_name)
|
2017-07-18 17:15:46 +02:00
|
|
|
|
2017-07-27 17:51:14 +02:00
|
|
|
# accessing invalid wallet fails
|
2019-09-25 11:34:51 +02:00
|
|
|
assert_raises_rpc_error(-18, "Requested wallet does not exist or is not loaded", wallet_bad.getwalletinfo)
|
2017-07-27 17:51:14 +02:00
|
|
|
|
2017-07-21 15:34:11 +02:00
|
|
|
# accessing wallet RPC without using wallet endpoint fails
|
2018-01-12 23:24:36 +01:00
|
|
|
assert_raises_rpc_error(-19, "Wallet file not specified", node.getwalletinfo)
|
2017-07-18 17:15:46 +02:00
|
|
|
|
2018-03-07 17:05:08 +01:00
|
|
|
w1, w2, w3, w4, *_ = wallets
|
2017-07-18 17:15:46 +02:00
|
|
|
w1.generate(101)
|
2019-08-03 17:56:18 +02:00
|
|
|
assert_equal(w1.getbalance(), 1000)
|
2017-07-18 17:15:46 +02:00
|
|
|
assert_equal(w2.getbalance(), 0)
|
|
|
|
assert_equal(w3.getbalance(), 0)
|
2017-11-22 17:31:32 +01:00
|
|
|
assert_equal(w4.getbalance(), 0)
|
2017-07-18 17:15:46 +02:00
|
|
|
|
|
|
|
w1.sendtoaddress(w2.getnewaddress(), 1)
|
|
|
|
w1.sendtoaddress(w3.getnewaddress(), 2)
|
2017-11-22 17:31:32 +01:00
|
|
|
w1.sendtoaddress(w4.getnewaddress(), 3)
|
2017-07-18 17:15:46 +02:00
|
|
|
w1.generate(1)
|
|
|
|
assert_equal(w2.getbalance(), 1)
|
|
|
|
assert_equal(w3.getbalance(), 2)
|
2017-11-22 17:31:32 +01:00
|
|
|
assert_equal(w4.getbalance(), 3)
|
2017-07-18 17:15:46 +02:00
|
|
|
|
2017-10-12 14:53:33 +02: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")
|
|
|
|
|
2018-04-19 23:42:40 +02:00
|
|
|
|
|
|
|
self.log.info("Test dynamic wallet loading")
|
|
|
|
|
|
|
|
self.restart_node(0, ['-nowallet'])
|
|
|
|
assert_equal(node.listwallets(), [])
|
|
|
|
assert_raises_rpc_error(-32601, "Method not found", node.getwalletinfo)
|
|
|
|
|
|
|
|
self.log.info("Load first wallet")
|
|
|
|
loadwallet_name = node.loadwallet(wallet_names[0])
|
|
|
|
assert_equal(loadwallet_name['name'], wallet_names[0])
|
|
|
|
assert_equal(node.listwallets(), wallet_names[0:1])
|
|
|
|
node.getwalletinfo()
|
|
|
|
w1 = node.get_wallet_rpc(wallet_names[0])
|
|
|
|
w1.getwalletinfo()
|
|
|
|
|
|
|
|
self.log.info("Load second wallet")
|
|
|
|
loadwallet_name = node.loadwallet(wallet_names[1])
|
|
|
|
assert_equal(loadwallet_name['name'], wallet_names[1])
|
|
|
|
assert_equal(node.listwallets(), wallet_names[0:2])
|
|
|
|
assert_raises_rpc_error(-19, "Wallet file not specified", node.getwalletinfo)
|
|
|
|
w2 = node.get_wallet_rpc(wallet_names[1])
|
|
|
|
w2.getwalletinfo()
|
|
|
|
|
|
|
|
self.log.info("Load remaining wallets")
|
|
|
|
for wallet_name in wallet_names[2:]:
|
|
|
|
loadwallet_name = self.nodes[0].loadwallet(wallet_name)
|
|
|
|
assert_equal(loadwallet_name['name'], wallet_name)
|
|
|
|
|
|
|
|
assert_equal(set(self.nodes[0].listwallets()), set(wallet_names))
|
|
|
|
|
|
|
|
# Fail to load if wallet doesn't exist
|
|
|
|
assert_raises_rpc_error(-18, 'Wallet wallets not found.', self.nodes[0].loadwallet, 'wallets')
|
|
|
|
|
|
|
|
# Fail to load duplicate wallets
|
|
|
|
assert_raises_rpc_error(-4, 'Wallet file verification failed: Error loading wallet w1. Duplicate -wallet filename specified.', self.nodes[0].loadwallet, wallet_names[0])
|
|
|
|
|
|
|
|
# Fail to load if one wallet is a copy of another
|
|
|
|
assert_raises_rpc_error(-1, "BerkeleyBatch: Can't open database w8_copy (duplicates fileid", self.nodes[0].loadwallet, 'w8_copy')
|
|
|
|
|
|
|
|
# Fail to load if wallet file is a symlink
|
|
|
|
assert_raises_rpc_error(-4, "Wallet file verification failed: Invalid -wallet path 'w8_symlink'", self.nodes[0].loadwallet, 'w8_symlink')
|
|
|
|
|
2020-07-08 14:55:33 +02:00
|
|
|
# Fail to load if a directory is specified that doesn't contain a wallet
|
|
|
|
os.mkdir(wallet_dir('empty_wallet_dir'))
|
|
|
|
assert_raises_rpc_error(-18, "Directory empty_wallet_dir does not contain a wallet.dat file", self.nodes[0].loadwallet, 'empty_wallet_dir')
|
|
|
|
|
2017-07-18 17:15:46 +02:00
|
|
|
if __name__ == '__main__':
|
|
|
|
MultiWalletTest().main()
|