You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
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.
For a deeper understanding of the ICP < > BTC integration, see the [Bitcoin integration documentation](/docs/current/developer-docs/multi-chain/bitcoin/overview).
18
21
19
22
## Prerequisites
20
23
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.
22
26
23
27
:::info
24
28
This example is designed to be deployed on the mainnet. It will return errors when deployed locally; these errors are expected.
-`--network=ic` tells the command line to deploy the smart contract to the mainnet ICP blockchain
60
64
-`--argument '(variant { Testnet })'` passes the argument `Testnet` to initialize the smart contract, telling it to connect to the Bitcoin testnet
61
65
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.**
63
72
64
73
65
74
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
83
92
84
93
## Step 2: Generating a Bitcoin address
85
94
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
* 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
109
149
some testnet bitcoin. You can use one of the Bitcoin faucets, such as [coinfaucet.eu](https://coinfaucet.eu),
110
150
to receive some bitcoin.
111
151
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.
113
154
114
155
115
156
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
121
162
122
163
In the Candid UI, paste in your canister's address, and click on "Call".
123
164
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:
The `send_from_${type}` endpoint can send bitcoin by:
146
188
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).
149
191
3. Building a transaction, using some of the UTXOs from step 2 as input and the destination address and amount to send as output.
150
192
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
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
156
202
transaction has at least one confirmation, you should be able to see it
157
203
reflected in your current balance.
158
204
@@ -165,7 +211,7 @@ In this tutorial, you were able to:
165
211
* Connect the canister to the Bitcoin testnet.
166
212
* Send the canister some testnet BTC.
167
213
* 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.
169
215
170
216
This example is extensively documented in the following tutorials:
171
217
@@ -179,3 +225,16 @@ If you base your application on this example, we recommend you familiarize yours
179
225
For example, the following aspects are particularly relevant for this app:
180
226
*[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.
181
227
*[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.
0 commit comments