Skip to content

Commit 6fd861d

Browse files
committed
new test wallet_descriptor_upgradetohd
1 parent 3bc5159 commit 6fd861d

File tree

1 file changed

+215
-0
lines changed

1 file changed

+215
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,215 @@
1+
#!/usr/bin/env python3
2+
# Copyright (c) 2016 The Bitcoin Core developers
3+
# Distributed under the MIT software license, see the accompanying
4+
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
5+
"""
6+
wallet_upgradetohd.py
7+
8+
Test upgrade to a Hierarchical Deterministic wallet via upgradetohd rpc
9+
"""
10+
11+
import shutil
12+
import os
13+
14+
from test_framework.test_framework import BitcoinTestFramework
15+
from test_framework.util import (
16+
assert_equal,
17+
assert_raises_rpc_error,
18+
)
19+
20+
21+
class WalletUpgradeToHDTest(BitcoinTestFramework):
22+
def set_test_params(self):
23+
self.num_nodes = 1
24+
25+
def skip_test_if_missing_module(self):
26+
self.skip_if_no_wallet()
27+
28+
def recover_blank(self):
29+
self.log.info("Recover non-HD wallet to check different upgrade paths")
30+
node = self.nodes[0]
31+
self.stop_node(0)
32+
shutil.copyfile(os.path.join(node.datadir, "blank.bak"), os.path.join(node.datadir, self.chain, self.default_wallet_name, self.wallet_data_filename))
33+
self.start_node(0)
34+
assert 'hdchainid' not in node.get_wallet_rpc("w-1").getwalletinfo()
35+
36+
def run_test(self):
37+
node = self.nodes[0]
38+
node.createwallet("w-1", False, True, "", False, True, True)
39+
40+
wallet = self.nodes[0].get_wallet_rpc(self.default_wallet_name)
41+
w1 = self.nodes[0].get_wallet_rpc("w-1")
42+
w1.backupwallet(os.path.join(node.datadir, "blank.bak"))
43+
44+
45+
self.log.info("No mnemonic, no mnemonic passphrase, no wallet passphrase")
46+
assert_equal(len(w1.listdescriptors()['descriptors']), 0)
47+
balance_before = w1.getbalance()
48+
balance_non_HD = wallet.getbalance()
49+
assert w1.upgradetohd()
50+
mnemonic = w1.listdescriptors(True)['descriptors'][0]['mnemonic']
51+
desc = w1.listdescriptors()['descriptors'][0]['desc']
52+
assert_equal(len(desc), 149)
53+
assert_equal(balance_before, w1.getbalance())
54+
55+
self.log.info("Should be spendable and should use correct paths")
56+
for i in range(5):
57+
txid = wallet.sendtoaddress(w1.getnewaddress(), 1)
58+
self.sync_all()
59+
outs = node.decoderawtransaction(w1.gettransaction(txid)['hex'])['vout']
60+
for out in outs:
61+
if out['value'] == 1:
62+
keypath =w1.getaddressinfo(out['scriptPubKey']['address'])['hdkeypath']
63+
assert_equal(keypath, "m/44'/1'/0'/0/%d" % i)
64+
else:
65+
# change doesn't belong to descriptor wallet, skip it
66+
pass
67+
68+
self.bump_mocktime(1)
69+
self.generate(node, 1)
70+
71+
self.log.info("Should no longer be able to start it with HD disabled")
72+
self.stop_node(0)
73+
node.assert_start_raises_init_error(['-usehd=0'], "Error: Error loading %s: You can't disable HD on an already existing HD wallet" % self.default_wallet_name)
74+
self.extra_args = []
75+
self.start_node(0, [])
76+
77+
wallet = self.nodes[0].get_wallet_rpc(self.default_wallet_name)
78+
w1 = self.nodes[0].get_wallet_rpc("w-1")
79+
balance_after = w1.getbalance()
80+
81+
self.recover_blank()
82+
83+
# We spent some coins from default wallet to descriptorwallet earlier
84+
assert balance_non_HD != wallet.getbalance()
85+
balance_non_HD = wallet.getbalance()
86+
87+
self.log.info("No mnemonic, no mnemonic passphrase, no wallet passphrase, should result in completely different keys")
88+
assert node.upgradetohd()
89+
assert mnemonic != node.dumphdinfo()['mnemonic']
90+
assert desc != node.getwalletinfo()['hdchainid']
91+
assert_equal(balance_non_HD, node.getbalance())
92+
node.keypoolrefill(5)
93+
node.rescanblockchain()
94+
# Completely different keys, no HD coins should be recovered
95+
assert_equal(balance_non_HD, node.getbalance())
96+
97+
self.recover_blank()
98+
99+
self.log.info("No mnemonic, no mnemonic passphrase, no wallet passphrase, should result in completely different keys")
100+
self.restart_node(0, extra_args=['-keypool=10'])
101+
assert node.upgradetohd("", "", "", True)
102+
# Completely different keys, no HD coins should be recovered
103+
assert mnemonic != node.dumphdinfo()['mnemonic']
104+
assert desc != node.getwalletinfo()['hdchainid']
105+
assert_equal(balance_non_HD, node.getbalance())
106+
107+
self.recover_blank()
108+
109+
self.log.info("Same mnemonic, another mnemonic passphrase, no wallet passphrase, should result in a different set of keys")
110+
new_mnemonic_passphrase = "somewords"
111+
assert node.upgradetohd(mnemonic, new_mnemonic_passphrase)
112+
assert_equal(mnemonic, node.dumphdinfo()['mnemonic'])
113+
new_chainid = node.getwalletinfo()['hdchainid']
114+
assert desc != new_chainid
115+
assert_equal(balance_non_HD, node.getbalance())
116+
node.keypoolrefill(5)
117+
node.rescanblockchain()
118+
# A different set of keys, no HD coins should be recovered
119+
new_addresses = (node.getnewaddress(), node.getrawchangeaddress())
120+
assert_equal(balance_non_HD, node.getbalance())
121+
122+
self.recover_blank()
123+
124+
self.log.info("Same mnemonic, another mnemonic passphrase, no wallet passphrase, should result in a different set of keys (again)")
125+
assert node.upgradetohd(mnemonic, new_mnemonic_passphrase)
126+
assert_equal(mnemonic, node.dumphdinfo()['mnemonic'])
127+
assert_equal(new_chainid, node.getwalletinfo()['hdchainid'])
128+
assert_equal(balance_non_HD, node.getbalance())
129+
node.keypoolrefill(5)
130+
node.rescanblockchain()
131+
# A different set of keys, no HD coins should be recovered, keys should be the same as they were the previous time
132+
assert_equal(new_addresses, (node.getnewaddress(), node.getrawchangeaddress()))
133+
assert_equal(balance_non_HD, node.getbalance())
134+
135+
self.recover_blank()
136+
137+
self.log.info("Same mnemonic, no mnemonic passphrase, no wallet passphrase, should recover all coins after rescan")
138+
assert node.upgradetohd(mnemonic)
139+
assert_equal(mnemonic, node.dumphdinfo()['mnemonic'])
140+
assert_equal(desc, node.getwalletinfo()['hdchainid'])
141+
node.keypoolrefill(5)
142+
assert balance_after != node.getbalance()
143+
node.rescanblockchain()
144+
assert_equal(balance_after, node.getbalance())
145+
146+
self.recover_blank()
147+
148+
self.log.info("Same mnemonic, no mnemonic passphrase, no wallet passphrase, large enough keepool, should recover all coins with no extra rescan")
149+
self.restart_node(0, extra_args=['-keypool=10'])
150+
assert node.upgradetohd(mnemonic)
151+
assert_equal(mnemonic, node.dumphdinfo()['mnemonic'])
152+
assert_equal(desc, node.getwalletinfo()['hdchainid'])
153+
# All coins should be recovered
154+
assert_equal(balance_after, node.getbalance())
155+
156+
self.recover_blank()
157+
158+
self.log.info("Same mnemonic, no mnemonic passphrase, no wallet passphrase, large enough keepool, rescan is skipped initially, should recover all coins after rescanblockchain")
159+
self.restart_node(0, extra_args=['-keypool=10'])
160+
assert node.upgradetohd(mnemonic, "", "", False)
161+
assert_equal(mnemonic, node.dumphdinfo()['mnemonic'])
162+
assert_equal(desc, node.getwalletinfo()['hdchainid'])
163+
assert balance_after != node.getbalance()
164+
node.rescanblockchain()
165+
# All coins should be recovered
166+
assert_equal(balance_after, node.getbalance())
167+
168+
self.recover_blank()
169+
170+
self.log.info("Same mnemonic, same mnemonic passphrase, encrypt wallet on upgrade, should recover all coins after rescan")
171+
walletpass = "111pass222"
172+
assert node.upgradetohd(mnemonic, "", walletpass)
173+
node.stop()
174+
node.wait_until_stopped()
175+
self.start_node(0, extra_args=['-rescan'])
176+
assert_raises_rpc_error(-13, "Error: Please enter the wallet passphrase with walletpassphrase first.", node.dumphdinfo)
177+
node.walletpassphrase(walletpass, 100)
178+
assert_equal(mnemonic, node.dumphdinfo()['mnemonic'])
179+
assert_equal(desc, node.getwalletinfo()['hdchainid'])
180+
# Note: wallet encryption results in additional keypool topup,
181+
# so we can't compare new balance to balance_non_HD here,
182+
# assert_equal(balance_non_HD, node.getbalance()) # won't work
183+
assert balance_non_HD != node.getbalance()
184+
node.keypoolrefill(4)
185+
node.rescanblockchain()
186+
# All coins should be recovered
187+
assert_equal(balance_after, node.getbalance())
188+
189+
self.recover_blank()
190+
191+
self.log.info("Same mnemonic, same mnemonic passphrase, encrypt wallet first, should recover all coins on upgrade after rescan")
192+
walletpass = "111pass222"
193+
node.encryptwallet(walletpass)
194+
node.stop()
195+
node.wait_until_stopped()
196+
self.start_node(0, extra_args=['-rescan'])
197+
assert_raises_rpc_error(-13, "Error: Wallet encrypted but passphrase not supplied to RPC.", node.upgradetohd, mnemonic)
198+
assert_raises_rpc_error(-1, "Error: The wallet passphrase entered was incorrect", node.upgradetohd, mnemonic, "", "wrongpass")
199+
assert node.upgradetohd(mnemonic, "", walletpass)
200+
assert_raises_rpc_error(-13, "Error: Please enter the wallet passphrase with walletpassphrase first.", node.dumphdinfo)
201+
node.walletpassphrase(walletpass, 100)
202+
assert_equal(mnemonic, node.dumphdinfo()['mnemonic'])
203+
assert_equal(desc, node.getwalletinfo()['hdchainid'])
204+
# Note: wallet encryption results in additional keypool topup,
205+
# so we can't compare new balance to balance_non_HD here,
206+
# assert_equal(balance_non_HD, node.getbalance()) # won't work
207+
assert balance_non_HD != node.getbalance()
208+
node.keypoolrefill(4)
209+
node.rescanblockchain()
210+
# All coins should be recovered
211+
assert_equal(balance_after, node.getbalance())
212+
213+
214+
if __name__ == '__main__':
215+
WalletUpgradeToHDTest().main ()

0 commit comments

Comments
 (0)