stratum_apps/config_helpers/coinbase_output/
mod.rs1mod errors;
2mod serde_types;
3
4use miniscript::{
5 bitcoin::{address::NetworkUnchecked, Address, Network, ScriptBuf},
6 DefiniteDescriptorKey, Descriptor,
7};
8
9pub use errors::Error;
10
11#[derive(Debug, serde::Deserialize, Clone)]
15#[serde(try_from = "serde_types::SerdeCoinbaseOutput")]
16pub struct CoinbaseRewardScript {
17 script_pubkey: ScriptBuf,
18 ok_for_mainnet: bool,
19}
20
21impl CoinbaseRewardScript {
22 pub fn from_descriptor(s: &str) -> Result<Self, Error> {
24 if s.starts_with("tr") {
28 let desc = s.parse::<Descriptor<DefiniteDescriptorKey>>()?;
29 return Ok(Self {
30 script_pubkey: desc.script_pubkey(),
31 ok_for_mainnet: true,
34 });
35 }
36
37 let tree = miniscript::expression::Tree::from_str(s)?;
38 let root = tree.root();
39 match root.name() {
40 "addr" => {
41 let addr: Address<NetworkUnchecked> = root
42 .verify_terminal_parent("addr", "a valid Bitcoin address")
43 .map_err(miniscript::Error::Parse)?;
44
45 Ok(Self {
46 script_pubkey: addr.assume_checked_ref().script_pubkey(),
47 ok_for_mainnet: addr.is_valid_for_network(Network::Bitcoin),
48 })
49 }
50 "raw" => {
51 let script_hex: String = root
52 .verify_terminal_parent(
53 "raw",
54 "a hex-encoded Bitcoin script without length prefix",
55 )
56 .map_err(miniscript::Error::Parse)?;
57
58 Ok(Self {
59 script_pubkey: ScriptBuf::from_hex(&script_hex)?,
60 ok_for_mainnet: true,
62 })
63 }
64 _ => {
65 use miniscript::expression::FromTree as _;
66
67 let desc = Descriptor::<DefiniteDescriptorKey>::from_tree(root)?;
68 Ok(Self {
69 script_pubkey: desc.script_pubkey(),
70 ok_for_mainnet: true,
73 })
74 }
75 }
76 }
77
78 pub fn ok_for_mainnet(&self) -> bool {
84 self.ok_for_mainnet
85 }
86
87 pub fn script_pubkey(&self) -> ScriptBuf {
89 self.script_pubkey.clone()
90 }
91}
92
93#[cfg(test)]
94mod tests {
95 use super::*;
96
97 #[test]
98 fn fixed_vector_addr() {
99 assert_eq!(
101 CoinbaseRewardScript::from_descriptor(
102 "addr(1BvBMSEYstWetqTFn5Au4m4GFg7xJaNVN2)#wdnlkpe8"
103 )
104 .unwrap()
105 .script_pubkey()
106 .to_hex_string(),
107 "76a91477bff20c60e522dfaa3350c39b030a5d004e839a88ac",
108 );
109 assert_eq!(
110 CoinbaseRewardScript::from_descriptor(
111 "addr(3J98t1WpEZ73CNmQviecrnyiWrnqRhWNLy)#rsjl0crt"
112 )
113 .unwrap()
114 .script_pubkey()
115 .to_hex_string(),
116 "a914b472a266d0bd89c13706a4132ccfb16f7c3b9fcb87",
117 );
118 assert_eq!(
119 CoinbaseRewardScript::from_descriptor(
120 "addr(bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t4)#uyjndxcw"
121 )
122 .unwrap()
123 .script_pubkey()
124 .to_hex_string(),
125 "0014751e76e8199196d454941c45d1b3a323f1433bd6",
126 );
127 assert_eq!(
128 CoinbaseRewardScript::from_descriptor(
129 "addr(bc1qrp33g0q5c5txsp9arysrx4k6zdkfs4nce4xj0gdcccefvpysxf3qccfmv3)#8kzm8txf"
130 )
131 .unwrap()
132 .script_pubkey()
133 .to_hex_string(),
134 "00201863143c14c5166804bd19203356da136c985678cd4d27a1b8c6329604903262",
135 );
136 assert_eq!(
138 CoinbaseRewardScript::from_descriptor("addr(1BvBMSEYstWetqTFn5Au4m4GFg7xJaNVN2)")
139 .unwrap()
140 .script_pubkey()
141 .to_hex_string(),
142 "76a91477bff20c60e522dfaa3350c39b030a5d004e839a88ac",
143 );
144 assert_eq!(
145 CoinbaseRewardScript::from_descriptor("addr(1BvBMSEYstWetqTFn5Au4m4GFg7xJaNVN2,)")
146 .unwrap_err()
147 .to_string(),
148 "Miniscript: addr must have 1 children, but found 2",
149 );
150
151 assert_eq!(
154 CoinbaseRewardScript::from_descriptor("addr(1BvBMSEYstWetqTFn5Au4m4GFg7xJaNVN2)#")
155 .unwrap_err()
156 .to_string(),
157 "Miniscript: invalid checksum (length 0, expected 8)",
158 );
159 assert_eq!(
160 CoinbaseRewardScript::from_descriptor(
161 "addr(1BvBMSEYstWetqTFn5Au4m4GFg7xJaNVN2)#wdnlkpe7"
162 )
163 .unwrap_err()
164 .to_string(),
165 "Miniscript: invalid checksum wdnlkpe7; expected wdnlkpe8",
166 );
167 assert!(CoinbaseRewardScript::from_descriptor(
174 "addr(1BvBMSEYstWetqTFn5Au4m4GFg7xJaNVN3)#5v55uzec"
175 )
176 .is_err());
177 assert!(CoinbaseRewardScript::from_descriptor(
181 "addr(bc1qw508d6qejxtdg4y5r3zarvary0c5xw7kv8f3t3)#wfr7lfxf"
182 )
183 .is_err());
184 assert!(CoinbaseRewardScript::from_descriptor("addr()").is_err());
189 assert_eq!(
190 CoinbaseRewardScript::from_descriptor("addr(It's a mad mad world!?! 🙃)")
191 .unwrap_err()
192 .to_string(),
193 "Miniscript: invalid character '🙃' (position 29)",
194 );
195 assert_eq!(
197 CoinbaseRewardScript::from_descriptor("addr(It's a mad mad world!?! 🙃)#abcdefg")
198 .unwrap_err()
199 .to_string(),
200 "Miniscript: invalid character '🙃' (position 29)",
201 );
202 assert!(
206 CoinbaseRewardScript::from_descriptor("addr(It's a mad mad world!?!)#hmeprl29")
207 .is_err()
208 );
209 assert_eq!(
210 CoinbaseRewardScript::from_descriptor("addr(It's a mad mad world!?!)#🙃🙃🙃🙃🙃🙃")
211 .unwrap_err()
212 .to_string(),
213 "Miniscript: invalid character '🙃' (position 30)",
214 );
215 }
216
217 #[test]
218 fn fixed_vector_combo() {
219 assert_eq!(
221 CoinbaseRewardScript::from_descriptor(
222 "combo(0279be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798)"
223 )
224 .unwrap_err()
225 .to_string(),
226 "Miniscript: unrecognized name 'combo'",
227 );
228 }
229
230 #[test]
231 fn fixed_vector_musig() {
232 assert_eq!(
234 CoinbaseRewardScript::from_descriptor("musig(0279be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798,03fff97bd5755eeea420453a14355235d382f6472f8568a18b2f057a1460297556)").unwrap_err().to_string(),
235 "Miniscript: unrecognized name '03fff97bd5755eeea420453a14355235d382f6472f8568a18b2f057a1460297556'",
236 );
237 assert_eq!(
238 CoinbaseRewardScript::from_descriptor("tr(musig(0279be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798,03fff97bd5755eeea420453a14355235d382f6472f8568a18b2f057a1460297556))").unwrap_err().to_string(),
239 "Miniscript: internal key must have no children, but found 2",
240 );
241 }
242
243 #[test]
244 fn fixed_vector_raw() {
245 assert_eq!(
247 CoinbaseRewardScript::from_descriptor("raw()")
248 .unwrap()
249 .script_pubkey()
250 .to_hex_string(),
251 "",
252 );
253 assert_eq!(
254 CoinbaseRewardScript::from_descriptor("raw(deadbeef)")
255 .unwrap()
256 .script_pubkey()
257 .to_hex_string(),
258 "deadbeef",
259 );
260 assert_eq!(
261 CoinbaseRewardScript::from_descriptor("raw(DEADBEEF)")
262 .unwrap()
263 .script_pubkey()
264 .to_hex_string(),
265 "deadbeef",
266 );
267 assert_eq!(
269 CoinbaseRewardScript::from_descriptor("raw(DEADbeef)")
270 .unwrap()
271 .script_pubkey()
272 .to_hex_string(),
273 "deadbeef",
274 );
275 assert!(CoinbaseRewardScript::from_descriptor("raw(0)").is_err());
280 assert_eq!(
281 CoinbaseRewardScript::from_descriptor("raw(0,1)")
282 .unwrap_err()
283 .to_string(),
284 "Miniscript: raw must have 1 children, but found 2",
285 );
286 }
287
288 #[test]
289 fn fixed_vector_miniscript() {
290 assert_eq!(
291 CoinbaseRewardScript::from_descriptor("sh(wsh(multi(2,0279be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798,03fff97bd5755eeea420453a14355235d382f6472f8568a18b2f057a1460297556)))#qpcmf2lu").unwrap().script_pubkey().to_hex_string(),
292 "a9141cb55de50b72c67709ab16307d69557e6bb1a98787",
293 );
294 assert_eq!(
295 CoinbaseRewardScript::from_descriptor(
296 "tr(0279be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798)"
297 )
298 .unwrap()
299 .script_pubkey()
300 .to_hex_string(),
301 "5120da4710964f7852695de2da025290e24af6d8c281de5a0b902b7135fd9fd74d21",
302 );
303 assert_eq!(
304 CoinbaseRewardScript::from_descriptor("tr(0279be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798,{pk(03fff97bd5755eeea420453a14355235d382f6472f8568a18b2f057a1460297556),{multi_a(2,026a245bf6dc698504c89a20cfded60853152b695336c28063b61c65cbd269e6b4,0231ecbfac95d972f0b8f81ec6e01e9c621d91a4b48d5f9d12d7e95febe9f34d64),multi_a(2,026a245bf6dc698504c89a20cfded60853152b695336c28063b61c65cbd269e6b4,0231ecbfac95d972f0b8f81ec6e01e9c621d91a4b48d5f9d12d7e95febe9f34d64)}})")
305 .unwrap()
306 .script_pubkey()
307 .to_hex_string(),
308 "5120493bdae0d225af5cb88c4cb2a1e1e89e391153ba7699c91ebee2fd082ed1636c",
309 );
310 }
311
312 #[test]
313 fn fixed_vector_keys() {
314 assert_eq!(
316 CoinbaseRewardScript::from_descriptor("pkh(xpub661MyMwAqRbcFtXgS5sYJABqqG9YLmC4Q1Rdap9gSE8NqtwybGhePY2gZ29ESFjqJoCu1Rupje8YtGqsefD265TMg7usUDFdp6W1EGMcet8)").unwrap().script_pubkey().to_hex_string(),
317 "76a9143442193e1bb70916e914552172cd4e2dbc9df81188ac",
318 );
319 assert_eq!(
321 CoinbaseRewardScript::from_descriptor("pkh(xpub661MyMwAqRbcFtXgS5sYJABqqG9YLmC4Q1Rdap9gSE8NqtwybGhePY2gZ29ESFjqJoCu1Rupje8YtGqsefD265TMg7usUDFdp6W1EGMcet8/1/2/3)").unwrap().script_pubkey().to_hex_string(),
322 "76a914f2d2e1401c88353c2298d1a928d4ed827ff46ff688ac",
323 );
324 assert_eq!(
326 CoinbaseRewardScript::from_descriptor("pkh(xpub661MyMwAqRbcFtXgS5sYJABqqG9YLmC4Q1Rdap9gSE8NqtwybGhePY2gZ29ESFjqJoCu1Rupje8YtGqsefD265TMg7usUDFdp6W1EGMcet8/1'/2/3)").unwrap_err().to_string(),
327 "Miniscript: key with hardened derivation steps cannot be a DerivedDescriptorKey",
328 );
329 assert_eq!(
332 CoinbaseRewardScript::from_descriptor("pkh(xpub661MyMwAqRbcFtXgS5sYJABqqG9YLmC4Q1Rdap9gSE8NqtwybGhePY2gZ29ESFjqJoCu1Rupje8YtGqsefD265TMg7usUDFdp6W1EGMcet8/*)").unwrap_err().to_string(),
333 "Miniscript: key with a wildcard cannot be a DerivedDescriptorKey",
334 );
335 assert_eq!(
337 CoinbaseRewardScript::from_descriptor("pkh(xpub661MyMwAqRbcFtXgS5sYJABqqG9YLmC4Q1Rdap9gSE8NqtwybGhePY2gZ29ESFjqJoCu1Rupje8YtGqsefD265TMg7usUDFdp6W1EGMcet8/<0;1>)").unwrap_err().to_string(),
338 "Miniscript: multipath key cannot be a DerivedDescriptorKey",
339 );
340 assert_eq!(
342 CoinbaseRewardScript::from_descriptor(
343 "pkh(L4rK1yDtCWekvXuE6oXD9jCYfFNV2cWRpVuPLBcCU2z8TrisoyY1)"
344 )
345 .unwrap_err()
346 .to_string(),
347 "Miniscript: key too short",
348 );
349 assert_eq!(
351 CoinbaseRewardScript::from_descriptor("pkh(xprv9s21ZrQH143K3QTDL4LXw2F7HEK3wJUD2nW2nRk4stbPy6cq3jPPqjiChkVvvNKmPGJxWUtg6LnF5kejMRNNU3TGtRBeJgk33yuGBxrMPHi)").unwrap_err().to_string(),
352 "Miniscript: public keys must be 64, 66 or 130 characters in size",
353 );
354 }
355}