Skip to content

Commit d384b63

Browse files
Add taproot compiler example usage
1 parent 227da1a commit d384b63

File tree

2 files changed

+88
-0
lines changed

2 files changed

+88
-0
lines changed

Cargo.toml

+5
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ serde = { version = "1.0", optional = true}
2525
[dev-dependencies]
2626
bitcoind = {version = "0.26.1", features=["22_0"]}
2727
actual-rand = { package = "rand", version = "0.8.4"}
28+
secp256k1 = {version = "0.22.1", features = ["rand-std"]}
2829

2930
[[example]]
3031
name = "htlc"
@@ -44,3 +45,7 @@ name = "psbt"
4445

4546
[[example]]
4647
name = "xpub_descriptors"
48+
49+
[[example]]
50+
name = "taproot"
51+
required-features = ["compiler"]

examples/taproot.rs

+83
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
extern crate bitcoin;
2+
extern crate miniscript;
3+
extern crate secp256k1;
4+
5+
use std::str::FromStr;
6+
7+
use bitcoin::util::address::WitnessVersion;
8+
use bitcoin::Network;
9+
use miniscript::descriptor::DescriptorType;
10+
use miniscript::policy::Concrete;
11+
use secp256k1::{rand, KeyPair};
12+
13+
// Example shamelessly copied from https://github.com/sanket1729/adv_btc_workshop/blob/master/workshop.md#creating-a-taproot-descriptor
14+
15+
const PUBKEY_1: &str = "5102bb3ee77f65a02d79489bebc6e6d86ffc5ff6c4a2ec2b5265a5acb4ba3106";
16+
const PUBKEY_2: &str = "3fc863588b1b48ec8e7a077a26112f2356a27ad4717cbb56f289f71a3740eb49";
17+
const PUBKEY_3: &str = "8d8ce6639a64a60c02d7d97b1b5bc4ef3b01467f306e0913c275aeb3cc9cb3dc";
18+
const PUBKEY_4: &str = "86fe856dc0f9adc12b863d13d03d13dec99bd1112d0425668a6f7237238aa485";
19+
20+
fn main() {
21+
let pol_str = format!(
22+
"or(
23+
99@thresh(2,
24+
pk({}),
25+
pk({})
26+
),1@or(
27+
99@pk({}),
28+
1@and(pk({}),
29+
older(9))
30+
)
31+
)",
32+
PUBKEY_1, PUBKEY_2, PUBKEY_3, PUBKEY_4
33+
)
34+
.replace(&[' ', '\n', '\t'][..], "");
35+
36+
let pol: Concrete<secp256k1::XOnlyPublicKey> = Concrete::from_str(&pol_str).unwrap();
37+
38+
// We require secp for generating a random XOnlyPublicKey
39+
let secp = secp256k1::Secp256k1::new();
40+
let key_pair = KeyPair::new(&secp, &mut rand::thread_rng());
41+
// Random unspendable XOnlyPublicKey provided for compilation to Taproot Descriptor
42+
let unspendable_key = secp256k1::XOnlyPublicKey::from_keypair(&key_pair);
43+
44+
let private_desc = pol
45+
.compile_tr_private(Some(unspendable_key.clone()))
46+
.unwrap();
47+
// println!("{:}", private_desc);
48+
// Printing the policy shows that the internal_key (PUBKEY_3) is inferred from the given policy
49+
// let opt_desc = pol.compile_tr(Some(unspendable_key.clone())).unwrap();
50+
// println!("{:}", opt_desc);
51+
52+
// Check whether the descriptors are safe.
53+
assert!(private_desc.sanity_check().is_ok());
54+
// assert!(opt_desc.sanity_check().is_ok());
55+
56+
// Descriptor Type and Version should match respectively for Taproot
57+
let priv_desc_type = private_desc.desc_type();
58+
assert_eq!(priv_desc_type, DescriptorType::Tr);
59+
// let opt_desc_type = opt_desc.desc_type();
60+
// assert_eq!(opt_desc_type, DescriptorType::Tr);
61+
assert_eq!(priv_desc_type.segwit_version().unwrap(), WitnessVersion::V1);
62+
// assert_eq!(opt_desc_type.segwit_version().unwrap(), WitnessVersion::V1);
63+
64+
// Compute the bitcoin address and check if it matches
65+
let network = Network::Bitcoin;
66+
let priv_addr = private_desc.address(network).unwrap();
67+
let expected_addr = bitcoin::Address::from_str(
68+
"bc1p0y6a05cz95qhj6wlt5fuw7ykh5pq7580hsjkt7tdnd4y8tfv5g0s2w60nu",
69+
)
70+
.unwrap();
71+
assert_eq!(priv_addr, expected_addr);
72+
73+
// Computing the underlying script should fail for Taproot Descriptor
74+
assert!(private_desc.explicit_script().is_err());
75+
76+
// Max Satisfaction Weight for compilation, corresponding to the script-path spend
77+
// `multi_a(2,PUBKEY_1,PUBKEY_2) at taptree depth 1, having
78+
// Max Witness Size = scriptSig len + control_block size + varint(script_size) + script_size +
79+
// varint(max satisfaction elements) + max satisfaction size
80+
// = 4 + 65 + 1 + 70 + 1 + 132
81+
let max_sat_wt = private_desc.max_satisfaction_weight().unwrap();
82+
assert_eq!(max_sat_wt, 273);
83+
}

0 commit comments

Comments
 (0)