Skip to content

Commit 16324e9

Browse files
authored
Merge pull request #950 from dfinity/alex/add-p2tr-key-to-motoko-basic-bitcoin
Add P2TR untweaked key spend implementation to motoko. This PR adds pay-to-taproot (P2TR) raw key path transactions to the motoko basic_bitcoin example (see the modified README.md for more details). P2TR raw key spend is the encoded untweaked public key. This type of address is insecure to use with more than one signer, i.e., with multisignatures, but this is not a concern for using it on the IC, since it does not support multisignatures in the first place. Note that this type of address does not support scripts.
2 parents 38bc6f7 + 62ccf28 commit 16324e9

12 files changed

+662
-287
lines changed

motoko/basic_bitcoin/README.md

+83-24
Original file line numberDiff line numberDiff line change
@@ -11,14 +11,18 @@ This tutorial will walk you through how to deploy a sample [canister smart contr
1111

1212
## Architecture
1313

14-
This example internally leverages the [ECDSA API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-ecdsa_public_key)
15-
and [Bitcoin API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin-api) of the Internet Computer.
14+
This example internally leverages the [ECDSA
15+
API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-ecdsa_public_key),
16+
[Schnorr API](https://internetcomputer.org/docs/current/references/ic-interface-spec#ic-sign_with_schnorr), and [Bitcoin
17+
API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin-api)
18+
of the Internet Computer.
1619

1720
For a deeper understanding of the ICP < > BTC integration, see the [Bitcoin integration documentation](/docs/current/developer-docs/multi-chain/bitcoin/overview).
1821

1922
## Prerequisites
2023

21-
* [x] Install the [IC SDK](https://internetcomputer.org/docs/current/developer-docs/setup/install/index.mdx).
24+
* [x] Install the [IC
25+
SDK](https://internetcomputer.org/docs/current/developer-docs/setup/install/index.mdx). For local testing, `dfx >= 0.22.0` is required.
2226

2327
:::info
2428
This example is designed to be deployed on the mainnet. It will return errors when deployed locally; these errors are expected.
@@ -59,7 +63,12 @@ dfx deploy --network=ic basic_bitcoin --argument '(variant { testnet })'
5963
- `--network=ic` tells the command line to deploy the smart contract to the mainnet ICP blockchain
6064
- `--argument '(variant { Testnet })'` passes the argument `Testnet` to initialize the smart contract, telling it to connect to the Bitcoin testnet
6165

62-
**We're initializing the canister with `variant { Testnet }`, so that the canister connects to the the [Bitcoin testnet](https://en.bitcoin.it/wiki/Testnet). To be specific, this connects to `Testnet3`, which is the current Bitcoin test network used by the Bitcoin community.**
66+
**We're initializing the canister with `variant { testnet }` so that the
67+
canister connects to the [Bitcoin testnet](https://en.bitcoin.it/wiki/Testnet).
68+
To be specific, this connects to `Testnet3`, which is the current Bitcoin test
69+
network used by the Bitcoin community. This means that the addresses generated
70+
in the smart contract can only be used to receive or send funds only on Bitcoin
71+
testnet.**
6372

6473

6574
If successful, you should see an output that looks like this:
@@ -83,17 +92,48 @@ In the output above, to see the Candid Web UI for your bitcoin canister, you wou
8392

8493
## Step 2: Generating a Bitcoin address
8594

86-
Bitcoin has different types of addresses (e.g. P2PKH, P2SH). Most of these
87-
addresses can be generated from an ECDSA public key. The example code
88-
showcases how your canister can generate a [P2PKH address](https://en.bitcoin.it/wiki/Transaction#Pay-to-PubkeyHash) using the [ecdsa_public_key](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-ecdsa_public_key) API.
89-
90-
On the Candid UI of your canister, click the "Call" button under `get_p2pkh_address` to
91-
generate a P2PKH Bitcoin address.
95+
Bitcoin has different types of addresses (e.g. P2PKH, P2SH, P2TR). You may want
96+
to check [this
97+
article](https://bitcoinmagazine.com/technical/bitcoin-address-types-compared-p2pkh-p2sh-p2wpkh-and-more)
98+
if you are interested in a high-level comparison of different address types.
99+
These addresses can be generated from an ECDSA public key or a Schnorr
100+
([BIP340](https://github.com/bitcoin/bips/blob/master/bip-0340.mediawiki))
101+
public key. The example code showcases how your canister can generate and spend
102+
from two types of addresses:
103+
1. A [P2PKH address](https://en.bitcoin.it/wiki/Transaction#Pay-to-PubkeyHash)
104+
using the
105+
[ecdsa_public_key](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-method-ecdsa_public_key)
106+
API.
107+
2. A [P2TR
108+
address](https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki)
109+
where the funds can be spent using the raw (untweaked) internal key
110+
(so-called P2TR key path spend, but untweaked). The advantage of this
111+
approach compared to P2TR script spends is its significantly smaller fee per
112+
transaction because checking the transaction signature is analogous to P2PK
113+
but uses Schnorr instead of ECDSA. IMPORTANT: Note that
114+
[BIP341](https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki#cite_note-23)
115+
advises against using taproot addresses that can be spent with an untweaked
116+
key. This precaution is to prevent attacks that can occur when creating
117+
taproot multisigner addresses using specific multisignature schemes. However,
118+
the Schnorr API of the internet computer does not support Schnorr
119+
multisignatures.
120+
121+
Note that P2TR *key path* spending with a tweaked key is currently not available
122+
on the IC because the threshold Schnorr signing interface does not allow
123+
applying BIP341 tweaks to the private key. In contrast, the
124+
tweaked public key is used to spend in the script path, which is availble on the
125+
IC. For a technical comparison of different ways of how single-signer P2TR
126+
addresses can be constructed and used, you may want to take a look at [this
127+
post](https://bitcoin.stackexchange.com/a/111100) by Pieter Wuille.
128+
129+
On the Candid UI of your canister, click the "Call" button under
130+
`get_${type}_address` to generate a `${type}` Bitcoin address, where `${type}`
131+
is one of `[p2pkh, p2tr_raw_key_spend]`.
92132

93133
Or, if you prefer the command line:
94134

95135
```bash
96-
dfx canister --network=ic call basic_bitcoin get_p2pkh_address
136+
dfx canister --network=ic call basic_bitcoin get_${type}_address
97137
```
98138

99139
* The Bitcoin address you see will be different from the one above because the
@@ -109,7 +149,8 @@ Now that the canister is deployed and you have a Bitcoin address, it's time to r
109149
some testnet bitcoin. You can use one of the Bitcoin faucets, such as [coinfaucet.eu](https://coinfaucet.eu),
110150
to receive some bitcoin.
111151

112-
Enter your address and click on "Send testnet bitcoins". In the example below we will use Bitcoin address `n31eU1K11m1r58aJMgTyxGonu7wSMoUYe7`, but you will use your address. The canister will be receiving 0.011 test BTC on the Bitcoin Testnet.
152+
Enter your address and click on "Send testnet bitcoins". In the example below we will use Bitcoin address `n31eU1K11m1r58aJMgTyxGonu7wSMoUYe7`, but you will use your address. The Bitcoin address you see will be different from the one above
153+
because the ECDSA/Schnorr public key your canister retrieves is unique.
113154

114155

115156
Once the transaction has at least one confirmation, which can take a few minutes,
@@ -121,7 +162,7 @@ You can check a Bitcoin address's balance by using the `get_balance` endpoint on
121162

122163
In the Candid UI, paste in your canister's address, and click on "Call".
123164

124-
Alternatively, make the call using the command line. Be sure to replace `mheyfRsAQ1XrjtzjfU1cCH2B6G1KmNarNL` with your own generated P2PKH address:
165+
Alternatively, make the call using the command line. Be sure to replace `mheyfRsAQ1XrjtzjfU1cCH2B6G1KmNarNL` with your own generated address:
125166

126167
```bash
127168
dfx canister --network=ic call basic_bitcoin get_balance '("mheyfRsAQ1XrjtzjfU1cCH2B6G1KmNarNL")'
@@ -131,28 +172,33 @@ Checking the balance of a Bitcoin address relies on the [bitcoin_get_balance](ht
131172

132173
## Step 5: Sending bitcoin
133174

134-
You can send bitcoin using the `send` endpoint on your canister.
175+
You can send bitcoin using the `send_from_${type}` endpoint on your canister, where
176+
`${type}` is on of `[p2pkh, p2tr_raw_key_spend]`.
135177

136178
In the Candid UI, add a destination address and an amount to send. In the example
137179
below, we're sending 4'321 Satoshi (0.00004321 BTC) back to the testnet faucet.
138180

139181
Via the command line, the same call would look like this:
140182

141183
```bash
142-
dfx canister --network=ic call basic_bitcoin send '(record { destination_address = "tb1ql7w62elx9ucw4pj5lgw4l028hmuw80sndtntxt"; amount_in_satoshi = 4321; })'
184+
dfx canister --network=ic call basic_bitcoin send_from_p2pkh '(record { destination_address = "tb1ql7w62elx9ucw4pj5lgw4l028hmuw80sndtntxt"; amount_in_satoshi = 4321; })'
143185
```
144186

145-
The `send` endpoint can send bitcoin by:
187+
The `send_from_${type}` endpoint can send bitcoin by:
146188

147-
1. Getting the percentiles of the most recent fees on the Bitcoin network using the [bitcoin_get_current_fee_percentiles API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin_get_current_fee_percentiles).
148-
2. Fetching your unspent transaction outputs (UTXOs), using the [bitcoin_get_utxos API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin_get_utxos).
189+
1. Getting the percentiles of the most recent fees on the Bitcoin network using the [bitcoin_get_current_fee_percentiles API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-method-bitcoin_get_current_fee_percentiles).
190+
2. Fetching your unspent transaction outputs (UTXOs), using the [bitcoin_get_utxos API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-method-bitcoin_get_utxos).
149191
3. Building a transaction, using some of the UTXOs from step 2 as input and the destination address and amount to send as output.
150192
The fee percentiles obtained from step 1 are used to set an appropriate fee.
151-
4. Signing the inputs of the transaction using the [sign_with_ecdsa API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-sign_with_ecdsa).
152-
5. Sending the signed transaction to the Bitcoin network using the [bitcoin_send_transaction API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin_send_transaction).
153-
154-
The `send` endpoint returns the ID of the transaction it sent to the network.
155-
You can track the status of this transaction using a block explorer. Once the
193+
4. Signing the inputs of the transaction using the
194+
[sign_with_ecdsa
195+
API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-method-sign_with_ecdsa)/\
196+
[sign_with_schnorr](https://org5p-7iaaa-aaaak-qckna-cai.icp0.io/docs#ic-sign_with_schnorr).
197+
5. Sending the signed transaction to the Bitcoin network using the [bitcoin_send_transaction API](https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-method-bitcoin_send_transaction).
198+
199+
This canister's `send_from_${type}` endpoint returns the ID of the transaction
200+
it sent to the network. You can track the status of this transaction using a
201+
[block explorer](https://en.bitcoin.it/wiki/Block_chain_browser). Once the
156202
transaction has at least one confirmation, you should be able to see it
157203
reflected in your current balance.
158204

@@ -165,7 +211,7 @@ In this tutorial, you were able to:
165211
* Connect the canister to the Bitcoin testnet.
166212
* Send the canister some testnet BTC.
167213
* Check the testnet BTC balance of the canister.
168-
* Use the canister to send testnet BTC to another BTC address.
214+
* Use the canister to send testnet BTC to another testnet BTC address.
169215

170216
This example is extensively documented in the following tutorials:
171217

@@ -179,3 +225,16 @@ If you base your application on this example, we recommend you familiarize yours
179225
For example, the following aspects are particularly relevant for this app:
180226
* [Certify query responses if they are relevant for security](https://internetcomputer.org/docs/current/references/security/general-security-best-practices#certify-query-responses-if-they-are-relevant-for-security), since the app e.g. offers a method to read balances.
181227
* [Use a decentralized governance system like SNS to make a canister have a decentralized controller](https://internetcomputer.org/docs/current/developer-docs/security/security-best-practices/overview)
228+
229+
## Taproot transactions
230+
In addition to P2PKH transactions, this example now also suppots P2TR
231+
transactions, namely the so-called untweaked key path P2TR transactions, which
232+
is the most efficient way of performing a P2TR transaction. The limitation of
233+
this type of transactions is that it cannot be used in combination with scripts.
234+
IMPORTANT: Note that BIP341 advises against using taproot addresses that can be
235+
spent with an untweaked key. This precaution is to prevent attacks that can
236+
occur when creating taproot multisigner addresses using specific multisignature
237+
schemes. However, the Schnorr API of the internet computer does not support
238+
Schnorr multisignatures.
239+
240+
This implementation has only been tested locally with regtest.

motoko/basic_bitcoin/mops.toml

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,3 @@
11
[dependencies]
22
base = "0.11.2"
3-
bitcoin = "https://github.com/dfinity/motoko-bitcoin#ad3709363bf980d2cab45cce0dea7eda5c97a7ff"
4-
# sha2 is a transitive dependency and should be removed when `bitcoin` is published on `mops`
5-
sha2 = "0.1.0"
3+
bitcoin = "0.1.0"

motoko/basic_bitcoin/src/basic_bitcoin/basic_bitcoin.did

+10-3
Original file line numberDiff line numberDiff line change
@@ -33,15 +33,22 @@ type get_utxos_response = record {
3333
};
3434

3535
service : (network) -> {
36-
"get_p2pkh_address": () -> (bitcoin_address);
37-
3836
"get_balance": (address: bitcoin_address) -> (satoshi);
3937

4038
"get_utxos": (bitcoin_address) -> (get_utxos_response);
4139

4240
"get_current_fee_percentiles": () -> (vec millisatoshi_per_vbyte);
4341

44-
"send": (record {
42+
"get_p2pkh_address": () -> (bitcoin_address);
43+
44+
"send_from_p2pkh_address": (record {
45+
destination_address: bitcoin_address;
46+
amount_in_satoshi: satoshi;
47+
}) -> (transaction_id);
48+
49+
"get_p2tr_raw_key_spend_address": () -> (bitcoin_address);
50+
51+
"send_from_p2tr_raw_key_spend_address": (record {
4552
destination_address: bitcoin_address;
4653
amount_in_satoshi: satoshi;
4754
}) -> (transaction_id);

motoko/basic_bitcoin/src/basic_bitcoin/src/BitcoinApi.mo

+4-4
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ module {
3636
/// Relies on the `bitcoin_get_balance` endpoint.
3737
/// See https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin_get_balance
3838
public func get_balance(network : Network, address : BitcoinAddress) : async Satoshi {
39-
ExperimentalCycles.add(GET_BALANCE_COST_CYCLES);
39+
ExperimentalCycles.add<system>(GET_BALANCE_COST_CYCLES);
4040
await management_canister_actor.bitcoin_get_balance({
4141
address;
4242
network;
@@ -49,7 +49,7 @@ module {
4949
/// NOTE: Relies on the `bitcoin_get_utxos` endpoint.
5050
/// See https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin_get_utxos
5151
public func get_utxos(network : Network, address : BitcoinAddress) : async GetUtxosResponse {
52-
ExperimentalCycles.add(GET_UTXOS_COST_CYCLES);
52+
ExperimentalCycles.add<system>(GET_UTXOS_COST_CYCLES);
5353
await management_canister_actor.bitcoin_get_utxos({
5454
address;
5555
network;
@@ -63,7 +63,7 @@ module {
6363
/// Relies on the `bitcoin_get_current_fee_percentiles` endpoint.
6464
/// See https://internetcomputer.org/docs/current/references/ic-interface-spec/#ic-bitcoin_get_current_fee_percentiles
6565
public func get_current_fee_percentiles(network : Network) : async [MillisatoshiPerVByte] {
66-
ExperimentalCycles.add(GET_CURRENT_FEE_PERCENTILES_COST_CYCLES);
66+
ExperimentalCycles.add<system>(GET_CURRENT_FEE_PERCENTILES_COST_CYCLES);
6767
await management_canister_actor.bitcoin_get_current_fee_percentiles({
6868
network;
6969
})
@@ -77,7 +77,7 @@ module {
7777
let transaction_fee =
7878
SEND_TRANSACTION_BASE_COST_CYCLES + transaction.size() * SEND_TRANSACTION_COST_CYCLES_PER_BYTE;
7979

80-
ExperimentalCycles.add(transaction_fee);
80+
ExperimentalCycles.add<system>(transaction_fee);
8181
await management_canister_actor.bitcoin_send_transaction({
8282
network;
8383
transaction;

0 commit comments

Comments
 (0)