bitcoin/test/functional/wallet_backup.py

217 lines
7.8 KiB
Python
Raw Normal View History

2016-03-19 20:58:06 +01:00
#!/usr/bin/env python3
2019-04-06 18:38:51 -04:00
# Copyright (c) 2014-2019 The Bitcoin Core developers
2014-12-13 12:09:33 +08:00
# Distributed under the MIT software license, see the accompanying
2014-12-04 16:09:58 -05:00
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
"""Test the wallet backup features.
2014-12-04 16:09:58 -05:00
Test case is:
4 nodes. 1 2 and 3 send transactions between each other,
fourth node is a miner.
1 2 3 each mine a block to start, then
Miner creates 100 blocks so 1 2 3 each have 50 mature
coins to spend.
Then 5 iterations of 1/2/3 sending coins amongst
themselves to get transactions in the wallets,
and the miner mining one block.
Wallets are backed up using dumpwallet/backupwallet.
Then 5 more iterations of transactions and mining a block.
Miner then generates 101 more blocks, so any
transaction fees paid mature.
Sanity check:
Sum(1,2,3,4 balances) == 114*50
1/2/3 are shutdown, and their wallets erased.
Then restore using wallet.dat backup. And
confirm 1/2/3/4 balances are same as before.
Shutdown again, restore using importwallet,
and confirm again balances are correct.
"""
2018-07-07 00:10:35 +02:00
from decimal import Decimal
import os
from random import randint
import shutil
2014-12-04 16:09:58 -05:00
from test_framework.test_framework import BitcoinTestFramework
2019-04-06 18:38:51 -04:00
from test_framework.util import (
assert_equal,
assert_raises_rpc_error,
connect_nodes,
)
2014-12-04 16:09:58 -05:00
class WalletBackupTest(BitcoinTestFramework):
def set_test_params(self):
self.num_nodes = 4
self.setup_clean_chain = True
# nodes 1, 2,3 are spenders, let's give them a keypool=100
self.extra_args = [["-keypool=100"], ["-keypool=100"], ["-keypool=100"], []]
self.rpc_timeout = 120
2014-12-04 16:09:58 -05:00
def skip_test_if_missing_module(self):
self.skip_if_no_wallet()
def setup_network(self):
self.setup_nodes()
2014-12-04 16:09:58 -05:00
connect_nodes(self.nodes[0], 3)
connect_nodes(self.nodes[1], 3)
connect_nodes(self.nodes[2], 3)
connect_nodes(self.nodes[2], 0)
self.sync_all()
def one_send(self, from_node, to_address):
if (randint(1,2) == 1):
amount = Decimal(randint(1,10)) / Decimal(10)
self.nodes[from_node].sendtoaddress(to_address, amount)
def do_one_round(self):
a0 = self.nodes[0].getnewaddress()
a1 = self.nodes[1].getnewaddress()
a2 = self.nodes[2].getnewaddress()
self.one_send(0, a1)
self.one_send(0, a2)
self.one_send(1, a0)
self.one_send(1, a2)
self.one_send(2, a0)
self.one_send(2, a1)
# Have the miner (node3) mine a block.
# Must sync mempools before mining.
self.sync_mempools()
2015-03-31 20:28:28 -07:00
self.nodes[3].generate(1)
self.sync_blocks()
2014-12-04 16:09:58 -05:00
# As above, this mirrors the original bash test.
def start_three(self):
self.start_node(0)
self.start_node(1)
self.start_node(2)
2014-12-04 16:09:58 -05:00
connect_nodes(self.nodes[0], 3)
connect_nodes(self.nodes[1], 3)
connect_nodes(self.nodes[2], 3)
connect_nodes(self.nodes[2], 0)
def stop_three(self):
self.stop_node(0)
self.stop_node(1)
self.stop_node(2)
2014-12-04 16:09:58 -05:00
def erase_three(self):
os.remove(os.path.join(self.nodes[0].datadir, 'regtest', 'wallets', 'wallet.dat'))
os.remove(os.path.join(self.nodes[1].datadir, 'regtest', 'wallets', 'wallet.dat'))
os.remove(os.path.join(self.nodes[2].datadir, 'regtest', 'wallets', 'wallet.dat'))
2014-12-04 16:09:58 -05:00
def run_test(self):
2017-03-07 18:46:17 -05:00
self.log.info("Generating initial blockchain")
2015-03-31 20:28:28 -07:00
self.nodes[0].generate(1)
self.sync_blocks()
2015-03-31 20:28:28 -07:00
self.nodes[1].generate(1)
self.sync_blocks()
2015-03-31 20:28:28 -07:00
self.nodes[2].generate(1)
self.sync_blocks()
2015-03-31 20:28:28 -07:00
self.nodes[3].generate(100)
self.sync_blocks()
2014-12-04 16:09:58 -05:00
assert_equal(self.nodes[0].getbalance(), 50)
assert_equal(self.nodes[1].getbalance(), 50)
assert_equal(self.nodes[2].getbalance(), 50)
assert_equal(self.nodes[3].getbalance(), 0)
2017-03-07 18:46:17 -05:00
self.log.info("Creating transactions")
2014-12-04 16:09:58 -05:00
# Five rounds of sending each other transactions.
for i in range(5):
self.do_one_round()
2017-03-07 18:46:17 -05:00
self.log.info("Backing up")
self.nodes[0].backupwallet(os.path.join(self.nodes[0].datadir, 'wallet.bak'))
self.nodes[0].dumpwallet(os.path.join(self.nodes[0].datadir, 'wallet.dump'))
self.nodes[1].backupwallet(os.path.join(self.nodes[1].datadir, 'wallet.bak'))
self.nodes[1].dumpwallet(os.path.join(self.nodes[1].datadir, 'wallet.dump'))
self.nodes[2].backupwallet(os.path.join(self.nodes[2].datadir, 'wallet.bak'))
self.nodes[2].dumpwallet(os.path.join(self.nodes[2].datadir, 'wallet.dump'))
2014-12-04 16:09:58 -05:00
2017-03-07 18:46:17 -05:00
self.log.info("More transactions")
2014-12-04 16:09:58 -05:00
for i in range(5):
self.do_one_round()
# Generate 101 more blocks, so any fees paid mature
2015-03-31 20:28:28 -07:00
self.nodes[3].generate(101)
2014-12-04 16:09:58 -05:00
self.sync_all()
balance0 = self.nodes[0].getbalance()
balance1 = self.nodes[1].getbalance()
balance2 = self.nodes[2].getbalance()
balance3 = self.nodes[3].getbalance()
total = balance0 + balance1 + balance2 + balance3
# At this point, there are 214 blocks (103 for setup, then 10 rounds, then 101.)
# 114 are mature, so the sum of all wallets should be 114 * 50 = 5700.
assert_equal(total, 5700)
##
# Test restoring spender wallets from backups
##
2017-03-07 18:46:17 -05:00
self.log.info("Restoring using wallet.dat")
2014-12-04 16:09:58 -05:00
self.stop_three()
self.erase_three()
# Start node2 with no chain
shutil.rmtree(os.path.join(self.nodes[2].datadir, 'regtest', 'blocks'))
shutil.rmtree(os.path.join(self.nodes[2].datadir, 'regtest', 'chainstate'))
2014-12-04 16:09:58 -05:00
# Restore wallets from backup
shutil.copyfile(os.path.join(self.nodes[0].datadir, 'wallet.bak'), os.path.join(self.nodes[0].datadir, 'regtest', 'wallets', 'wallet.dat'))
shutil.copyfile(os.path.join(self.nodes[1].datadir, 'wallet.bak'), os.path.join(self.nodes[1].datadir, 'regtest', 'wallets', 'wallet.dat'))
shutil.copyfile(os.path.join(self.nodes[2].datadir, 'wallet.bak'), os.path.join(self.nodes[2].datadir, 'regtest', 'wallets', 'wallet.dat'))
2014-12-04 16:09:58 -05:00
2017-03-07 18:46:17 -05:00
self.log.info("Re-starting nodes")
2014-12-04 16:09:58 -05:00
self.start_three()
self.sync_blocks()
2014-12-04 16:09:58 -05:00
assert_equal(self.nodes[0].getbalance(), balance0)
assert_equal(self.nodes[1].getbalance(), balance1)
assert_equal(self.nodes[2].getbalance(), balance2)
2017-03-07 18:46:17 -05:00
self.log.info("Restoring using dumped wallet")
2014-12-04 16:09:58 -05:00
self.stop_three()
self.erase_three()
#start node2 with no chain
shutil.rmtree(os.path.join(self.nodes[2].datadir, 'regtest', 'blocks'))
shutil.rmtree(os.path.join(self.nodes[2].datadir, 'regtest', 'chainstate'))
2014-12-04 16:09:58 -05:00
self.start_three()
assert_equal(self.nodes[0].getbalance(), 0)
assert_equal(self.nodes[1].getbalance(), 0)
assert_equal(self.nodes[2].getbalance(), 0)
self.nodes[0].importwallet(os.path.join(self.nodes[0].datadir, 'wallet.dump'))
self.nodes[1].importwallet(os.path.join(self.nodes[1].datadir, 'wallet.dump'))
self.nodes[2].importwallet(os.path.join(self.nodes[2].datadir, 'wallet.dump'))
2014-12-04 16:09:58 -05:00
self.sync_blocks()
2014-12-04 16:09:58 -05:00
assert_equal(self.nodes[0].getbalance(), balance0)
assert_equal(self.nodes[1].getbalance(), balance1)
assert_equal(self.nodes[2].getbalance(), balance2)
# Backup to source wallet file must fail
sourcePaths = [
os.path.join(self.nodes[0].datadir, 'regtest', 'wallets', 'wallet.dat'),
os.path.join(self.nodes[0].datadir, 'regtest', '.', 'wallets', 'wallet.dat'),
os.path.join(self.nodes[0].datadir, 'regtest', 'wallets', ''),
os.path.join(self.nodes[0].datadir, 'regtest', 'wallets')]
for sourcePath in sourcePaths:
assert_raises_rpc_error(-4, "backup failed", self.nodes[0].backupwallet, sourcePath)
2014-12-04 16:09:58 -05:00
if __name__ == '__main__':
WalletBackupTest().main()