Skip to content

Commit 28cf756

Browse files
committed
Merge bitcoin/bitcoin#23578: Add external signer taproot support
796b020 wallet: add taproot support to external signer (Sjors Provoost) Pull request description: Builds on #22558 (merged on 2022-06-28). [HWI 2.1.0](https://github.com/bitcoin-core/HWI/releases/tag/2.1.0) or newer is required to import and use taproot descriptors. Older versions will work, but won't import a taproot descriptor. Tested with HWI 2.1.1: * Trezor T (firmware v2.5.1) on Signet: signs, change detection works * Ledger Nano S (firmware 2.1.0, Bitcoin app 2.0.6): signs, change detection works Only the most basic `tr(key)` descriptor is supported, script path spending is completely untested (if it works at all). ACKs for top commit: jb55: utACK 796b020 achow101: ACK 796b020 Tree-SHA512: 6dcb7eeb45421a3bbf2bdabeacd29979867db69077d7bf192bb77faa4bfefe446487b8df07bc40f9457009a88e598bdc09f769e6106fed2833ace7ef205a157a
2 parents bfce05c + 796b020 commit 28cf756

File tree

3 files changed

+21
-7
lines changed

3 files changed

+21
-7
lines changed

src/external_signer.cpp

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,9 @@ bool ExternalSigner::SignTransaction(PartiallySignedTransaction& psbtx, std::str
8181
for (const auto& entry : input.hd_keypaths) {
8282
if (parsed_m_fingerprint == MakeUCharSpan(entry.second.fingerprint)) return true;
8383
}
84+
for (const auto& entry : input.m_tap_bip32_paths) {
85+
if (parsed_m_fingerprint == MakeUCharSpan(entry.second.second.fingerprint)) return true;
86+
}
8487
return false;
8588
};
8689

test/functional/mocks/signer.py

Lines changed: 7 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -27,12 +27,15 @@ def getdescriptors(args):
2727
"receive": [
2828
"pkh([00000001/44'/1'/" + args.account + "']" + xpub + "/0/*)#vt6w3l3j",
2929
"sh(wpkh([00000001/49'/1'/" + args.account + "']" + xpub + "/0/*))#r0grqw5x",
30-
"wpkh([00000001/84'/1'/" + args.account + "']" + xpub + "/0/*)#x30uthjs"
30+
"wpkh([00000001/84'/1'/" + args.account + "']" + xpub + "/0/*)#x30uthjs",
31+
"tr([00000001/86'/1'/" + args.account + "']" + xpub + "/0/*)#sng9rd4t"
3132
],
3233
"internal": [
3334
"pkh([00000001/44'/1'/" + args.account + "']" + xpub + "/1/*)#all0v2p2",
3435
"sh(wpkh([00000001/49'/1'/" + args.account + "']" + xpub + "/1/*))#kwx4c3pe",
35-
"wpkh([00000001/84'/1'/" + args.account + "']" + xpub + "/1/*)#h92akzzg"
36+
"wpkh([00000001/84'/1'/" + args.account + "']" + xpub + "/1/*)#h92akzzg",
37+
"tr([00000001/86'/1'/" + args.account + "']" + xpub + "/1/*)#p8dy7c9n"
38+
3639
]
3740
}))
3841

@@ -44,7 +47,8 @@ def displayaddress(args):
4447
return sys.stdout.write(json.dumps({"error": "Unexpected fingerprint", "fingerprint": args.fingerprint}))
4548

4649
expected_desc = [
47-
"wpkh([00000001/84'/1'/0'/0/0]02c97dc3f4420402e01a113984311bf4a1b8de376cac0bdcfaf1b3ac81f13433c7)#0yneg42r"
50+
"wpkh([00000001/84'/1'/0'/0/0]02c97dc3f4420402e01a113984311bf4a1b8de376cac0bdcfaf1b3ac81f13433c7)#0yneg42r",
51+
"tr([00000001/86'/1'/0'/0/0]c97dc3f4420402e01a113984311bf4a1b8de376cac0bdcfaf1b3ac81f13433c7)#4vdj9jqk",
4852
]
4953
if args.desc not in expected_desc:
5054
return sys.stdout.write(json.dumps({"error": "Unexpected descriptor", "desc": args.desc}))

test/functional/wallet_signer.py

Lines changed: 11 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -98,7 +98,7 @@ def test_valid_signer(self):
9898
# )
9999
# self.clear_mock_result(self.nodes[1])
100100

101-
assert_equal(hww.getwalletinfo()["keypoolsize"], 30)
101+
assert_equal(hww.getwalletinfo()["keypoolsize"], 40)
102102

103103
address1 = hww.getnewaddress(address_type="bech32")
104104
assert_equal(address1, "bcrt1qm90ugl4d48jv8n6e5t9ln6t9zlpm5th68x4f8g")
@@ -121,6 +121,13 @@ def test_valid_signer(self):
121121
assert_equal(address_info['ismine'], True)
122122
assert_equal(address_info['hdkeypath'], "m/44'/1'/0'/0/0")
123123

124+
address4 = hww.getnewaddress(address_type="bech32m")
125+
assert_equal(address4, "bcrt1phw4cgpt6cd30kz9k4wkpwm872cdvhss29jga2xpmftelhqll62ms4e9sqj")
126+
address_info = hww.getaddressinfo(address4)
127+
assert_equal(address_info['solvable'], True)
128+
assert_equal(address_info['ismine'], True)
129+
assert_equal(address_info['hdkeypath'], "m/86'/1'/0'/0/0")
130+
124131
self.log.info('Test walletdisplayaddress')
125132
result = hww.walletdisplayaddress(address1)
126133
assert_equal(result, {"address": address1})
@@ -133,7 +140,7 @@ def test_valid_signer(self):
133140
self.clear_mock_result(self.nodes[1])
134141

135142
self.log.info('Prepare mock PSBT')
136-
self.nodes[0].sendtoaddress(address1, 1)
143+
self.nodes[0].sendtoaddress(address4, 1)
137144
self.generate(self.nodes[0], 1)
138145

139146
# Load private key into wallet to generate a signed PSBT for the mock
@@ -142,14 +149,14 @@ def test_valid_signer(self):
142149
assert mock_wallet.getwalletinfo()['private_keys_enabled']
143150

144151
result = mock_wallet.importdescriptors([{
145-
"desc": "wpkh([00000001/84'/1'/0']tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/0/*)#rweraev0",
152+
"desc": "tr([00000001/86'/1'/0']tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/0/*)#0jtt2jc9",
146153
"timestamp": 0,
147154
"range": [0,1],
148155
"internal": False,
149156
"active": True
150157
},
151158
{
152-
"desc": "wpkh([00000001/84'/1'/0']tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/1/*)#j6uzqvuh",
159+
"desc": "tr([00000001/86'/1'/0']tprv8ZgxMBicQKsPd7Uf69XL1XwhmjHopUGep8GuEiJDZmbQz6o58LninorQAfcKZWARbtRtfnLcJ5MQ2AtHcQJCCRUcMRvmDUjyEmNUWwx8UbK/1/*)#7xw2h8ga",
153160
"timestamp": 0,
154161
"range": [0, 0],
155162
"internal": True,

0 commit comments

Comments
 (0)