Skip to content

Commit 780dbc6

Browse files
committed
Merge #592: Add the planning module (updated PR)
a358076 test: absolute/relative timelocks in satisfaction (Daniela Brozzoni) cb9a769 tests: plan capabilities (Daniela Brozzoni) d29c298 Add plan capabilities to miniscript (Alekos Filini) fc20eb0 Fix test_cpp (sanket1729) 448fbd8 Add full_derivation_paths on DescriptorPublicKey (Daniela Brozzoni) 7ca9ba1 Add relative and absolute timelock in Satisfaction (Alekos Filini) Pull request description: This PR builds on top of #481, fixing all the review comments. I didn't squash my last commits on purpose to make review easier, I can squash them before merging if preferred. ACKs for top commit: apoelstra: ACK a358076 sanket1729: ACK a358076 Tree-SHA512: 32e547eedaf56d7ddb9ab8069ab394b655f46f6eae7b971d521befc800abadb785335a84c977875b050bcb202517381aba0fb9d8f2d418cd59a1f87147491d67
2 parents ff99a1f + a358076 commit 780dbc6

File tree

13 files changed

+2386
-217
lines changed

13 files changed

+2386
-217
lines changed

bitcoind-tests/tests/test_cpp.rs

+8-10
Original file line numberDiff line numberDiff line change
@@ -11,28 +11,24 @@ use std::path::Path;
1111

1212
use bitcoin::hashes::{sha256d, Hash};
1313
use bitcoin::psbt::Psbt;
14-
use bitcoin::secp256k1::{self, Secp256k1};
15-
use bitcoin::{psbt, Amount, OutPoint, Sequence, Transaction, TxIn, TxOut, Txid};
14+
use bitcoin::{psbt, secp256k1, Amount, OutPoint, Sequence, Transaction, TxIn, TxOut, Txid};
1615
use bitcoind::bitcoincore_rpc::{json, Client, RpcApi};
1716
use miniscript::bitcoin::absolute;
1817
use miniscript::psbt::PsbtExt;
19-
use miniscript::{bitcoin, Descriptor};
18+
use miniscript::{bitcoin, DefiniteDescriptorKey, Descriptor};
2019

2120
mod setup;
2221
use setup::test_util::{self, PubData, TestData};
2322

2423
// parse ~30 miniscripts from file
25-
pub(crate) fn parse_miniscripts(
26-
secp: &Secp256k1<secp256k1::All>,
27-
pubdata: &PubData,
28-
) -> Vec<Descriptor<bitcoin::PublicKey>> {
24+
pub(crate) fn parse_miniscripts(pubdata: &PubData) -> Vec<Descriptor<DefiniteDescriptorKey>> {
2925
// File must exist in current path before this produces output
3026
let mut desc_vec = vec![];
3127
// Consumes the iterator, returns an (Optional) String
3228
for line in read_lines("tests/data/random_ms.txt") {
3329
let ms = test_util::parse_insane_ms(&line.unwrap(), pubdata);
3430
let wsh = Descriptor::new_wsh(ms).unwrap();
35-
desc_vec.push(wsh.derived_descriptor(secp, 0).unwrap());
31+
desc_vec.push(wsh.at_derivation_index(0).unwrap());
3632
}
3733
desc_vec
3834
}
@@ -71,7 +67,7 @@ fn get_vout(cl: &Client, txid: Txid, value: u64) -> (OutPoint, TxOut) {
7167

7268
pub fn test_from_cpp_ms(cl: &Client, testdata: &TestData) {
7369
let secp = secp256k1::Secp256k1::new();
74-
let desc_vec = parse_miniscripts(&secp, &testdata.pubdata);
70+
let desc_vec = parse_miniscripts(&testdata.pubdata);
7571
let sks = &testdata.secretdata.sks;
7672
let pks = &testdata.pubdata.pks;
7773
// Generate some blocks
@@ -152,6 +148,7 @@ pub fn test_from_cpp_ms(cl: &Client, testdata: &TestData) {
152148
input.witness_utxo = Some(witness_utxo);
153149
input.witness_script = Some(desc.explicit_script().unwrap());
154150
psbt.inputs.push(input);
151+
psbt.update_input_with_descriptor(0, &desc).unwrap();
155152
psbt.outputs.push(psbt::Output::default());
156153
psbts.push(psbt);
157154
}
@@ -160,7 +157,8 @@ pub fn test_from_cpp_ms(cl: &Client, testdata: &TestData) {
160157
// Sign the transactions with all keys
161158
// AKA the signer role of psbt
162159
for i in 0..psbts.len() {
163-
let ms = if let Descriptor::Wsh(wsh) = &desc_vec[i] {
160+
let wsh_derived = desc_vec[i].derived_descriptor(&secp).unwrap();
161+
let ms = if let Descriptor::Wsh(wsh) = &wsh_derived {
164162
match wsh.as_inner() {
165163
miniscript::descriptor::WshInner::Ms(ms) => ms,
166164
_ => unreachable!(),

src/descriptor/bare.rs

+66
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,11 @@ use bitcoin::script::{self, PushBytes};
1313
use bitcoin::{Address, Network, ScriptBuf};
1414

1515
use super::checksum::{self, verify_checksum};
16+
use crate::descriptor::DefiniteDescriptorKey;
1617
use crate::expression::{self, FromTree};
1718
use crate::miniscript::context::{ScriptContext, ScriptContextError};
19+
use crate::miniscript::satisfy::{Placeholder, Satisfaction, Witness};
20+
use crate::plan::AssetProvider;
1821
use crate::policy::{semantic, Liftable};
1922
use crate::prelude::*;
2023
use crate::util::{varint_len, witness_to_scriptsig};
@@ -134,6 +137,30 @@ impl<Pk: MiniscriptKey + ToPublicKey> Bare<Pk> {
134137
}
135138
}
136139

140+
impl Bare<DefiniteDescriptorKey> {
141+
/// Returns a plan if the provided assets are sufficient to produce a non-malleable satisfaction
142+
pub fn plan_satisfaction<P>(
143+
&self,
144+
provider: &P,
145+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
146+
where
147+
P: AssetProvider<DefiniteDescriptorKey>,
148+
{
149+
self.ms.build_template(provider)
150+
}
151+
152+
/// Returns a plan if the provided assets are sufficient to produce a malleable satisfaction
153+
pub fn plan_satisfaction_mall<P>(
154+
&self,
155+
provider: &P,
156+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
157+
where
158+
P: AssetProvider<DefiniteDescriptorKey>,
159+
{
160+
self.ms.build_template_mall(provider)
161+
}
162+
}
163+
137164
impl<Pk: MiniscriptKey> fmt::Debug for Bare<Pk> {
138165
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
139166
write!(f, "{:?}", self.ms)
@@ -311,6 +338,45 @@ impl<Pk: MiniscriptKey + ToPublicKey> Pkh<Pk> {
311338
}
312339
}
313340

341+
impl Pkh<DefiniteDescriptorKey> {
342+
/// Returns a plan if the provided assets are sufficient to produce a non-malleable satisfaction
343+
pub fn plan_satisfaction<P>(
344+
&self,
345+
provider: &P,
346+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
347+
where
348+
P: AssetProvider<DefiniteDescriptorKey>,
349+
{
350+
let stack = if provider.provider_lookup_ecdsa_sig(&self.pk) {
351+
let stack = vec![
352+
Placeholder::EcdsaSigPk(self.pk.clone()),
353+
Placeholder::Pubkey(self.pk.clone(), BareCtx::pk_len(&self.pk)),
354+
];
355+
Witness::Stack(stack)
356+
} else {
357+
Witness::Unavailable
358+
};
359+
360+
Satisfaction {
361+
stack,
362+
has_sig: true,
363+
relative_timelock: None,
364+
absolute_timelock: None,
365+
}
366+
}
367+
368+
/// Returns a plan if the provided assets are sufficient to produce a malleable satisfaction
369+
pub fn plan_satisfaction_mall<P>(
370+
&self,
371+
provider: &P,
372+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
373+
where
374+
P: AssetProvider<DefiniteDescriptorKey>,
375+
{
376+
self.plan_satisfaction(provider)
377+
}
378+
}
379+
314380
impl<Pk: MiniscriptKey> fmt::Debug for Pkh<Pk> {
315381
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
316382
write!(f, "pkh({:?})", self.pk)

src/descriptor/key.rs

+41
Original file line numberDiff line numberDiff line change
@@ -592,6 +592,33 @@ impl DescriptorPublicKey {
592592
}
593593
}
594594

595+
/// Returns a vector containing the full derivation paths from the master key.
596+
/// The vector will contain just one element for single keys, and multiple elements
597+
/// for multipath extended keys.
598+
///
599+
/// For wildcard keys this will return the path up to the wildcard, so you
600+
/// can get full paths by appending one additional derivation step, according
601+
/// to the wildcard type (hardened or normal).
602+
pub fn full_derivation_paths(&self) -> Vec<bip32::DerivationPath> {
603+
match self {
604+
DescriptorPublicKey::MultiXPub(xpub) => {
605+
let origin_path = if let Some((_, ref path)) = xpub.origin {
606+
path.clone()
607+
} else {
608+
bip32::DerivationPath::from(vec![])
609+
};
610+
xpub.derivation_paths
611+
.paths()
612+
.into_iter()
613+
.map(|p| origin_path.extend(p))
614+
.collect()
615+
}
616+
_ => vec![self
617+
.full_derivation_path()
618+
.expect("Must be Some for non-multipath keys")],
619+
}
620+
}
621+
595622
/// Whether or not the key has a wildcard
596623
#[deprecated(note = "use has_wildcard instead")]
597624
pub fn is_deriveable(&self) -> bool {
@@ -1075,6 +1102,12 @@ impl DefiniteDescriptorKey {
10751102
self.0.full_derivation_path()
10761103
}
10771104

1105+
/// Full paths from the master key. The vector will contain just one path for single
1106+
/// keys, and multiple ones for multipath extended keys
1107+
pub fn full_derivation_paths(&self) -> Vec<bip32::DerivationPath> {
1108+
self.0.full_derivation_paths()
1109+
}
1110+
10781111
/// Reference to the underlying `DescriptorPublicKey`
10791112
pub fn as_descriptor_public_key(&self) -> &DescriptorPublicKey {
10801113
&self.0
@@ -1476,6 +1509,14 @@ mod test {
14761509
let desc_key = DescriptorPublicKey::from_str("[abcdef00/0'/1']tpubDBrgjcxBxnXyL575sHdkpKohWu5qHKoQ7TJXKNrYznh5fVEGBv89hA8ENW7A8MFVpFUSvgLqc4Nj1WZcpePX6rrxviVtPowvMuGF5rdT2Vi/9478'/<0';1>/8h/*'").unwrap();
14771510
assert!(desc_key.full_derivation_path().is_none());
14781511
assert!(desc_key.is_multipath());
1512+
// But you can get all the derivation paths
1513+
assert_eq!(
1514+
desc_key.full_derivation_paths(),
1515+
vec![
1516+
bip32::DerivationPath::from_str("m/0'/1'/9478'/0'/8'").unwrap(),
1517+
bip32::DerivationPath::from_str("m/0'/1'/9478'/1/8'").unwrap(),
1518+
],
1519+
);
14791520
assert_eq!(desc_key.into_single_keys(), vec![DescriptorPublicKey::from_str("[abcdef00/0'/1']tpubDBrgjcxBxnXyL575sHdkpKohWu5qHKoQ7TJXKNrYznh5fVEGBv89hA8ENW7A8MFVpFUSvgLqc4Nj1WZcpePX6rrxviVtPowvMuGF5rdT2Vi/9478'/0'/8h/*'").unwrap(), DescriptorPublicKey::from_str("[abcdef00/0'/1']tpubDBrgjcxBxnXyL575sHdkpKohWu5qHKoQ7TJXKNrYznh5fVEGBv89hA8ENW7A8MFVpFUSvgLqc4Nj1WZcpePX6rrxviVtPowvMuGF5rdT2Vi/9478'/1/8h/*'").unwrap()]);
14801521

14811522
// All the same but with extended private keys instead of xpubs.

src/descriptor/mod.rs

+62-3
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,8 @@ use sync::Arc;
2222

2323
use self::checksum::verify_checksum;
2424
use crate::miniscript::decode::Terminal;
25-
use crate::miniscript::{Legacy, Miniscript, Segwitv0};
25+
use crate::miniscript::{satisfy, Legacy, Miniscript, Segwitv0};
26+
use crate::plan::{AssetProvider, Plan};
2627
use crate::prelude::*;
2728
use crate::{
2829
expression, hash256, BareCtx, Error, ForEachKey, MiniscriptKey, Satisfier, ToPublicKey,
@@ -474,7 +475,7 @@ impl<Pk: MiniscriptKey + ToPublicKey> Descriptor<Pk> {
474475
Descriptor::Wpkh(ref wpkh) => wpkh.get_satisfaction(satisfier),
475476
Descriptor::Wsh(ref wsh) => wsh.get_satisfaction(satisfier),
476477
Descriptor::Sh(ref sh) => sh.get_satisfaction(satisfier),
477-
Descriptor::Tr(ref tr) => tr.get_satisfaction(satisfier),
478+
Descriptor::Tr(ref tr) => tr.get_satisfaction(&satisfier),
478479
}
479480
}
480481

@@ -491,7 +492,7 @@ impl<Pk: MiniscriptKey + ToPublicKey> Descriptor<Pk> {
491492
Descriptor::Wpkh(ref wpkh) => wpkh.get_satisfaction_mall(satisfier),
492493
Descriptor::Wsh(ref wsh) => wsh.get_satisfaction_mall(satisfier),
493494
Descriptor::Sh(ref sh) => sh.get_satisfaction_mall(satisfier),
494-
Descriptor::Tr(ref tr) => tr.get_satisfaction_mall(satisfier),
495+
Descriptor::Tr(ref tr) => tr.get_satisfaction_mall(&satisfier),
495496
}
496497
}
497498

@@ -509,6 +510,64 @@ impl<Pk: MiniscriptKey + ToPublicKey> Descriptor<Pk> {
509510
}
510511
}
511512

513+
impl Descriptor<DefiniteDescriptorKey> {
514+
/// Returns a plan if the provided assets are sufficient to produce a non-malleable satisfaction
515+
///
516+
/// If the assets aren't sufficient for generating a Plan, the descriptor is returned
517+
pub fn plan<P>(self, provider: &P) -> Result<Plan, Self>
518+
where
519+
P: AssetProvider<DefiniteDescriptorKey>,
520+
{
521+
let satisfaction = match self {
522+
Descriptor::Bare(ref bare) => bare.plan_satisfaction(provider),
523+
Descriptor::Pkh(ref pkh) => pkh.plan_satisfaction(provider),
524+
Descriptor::Wpkh(ref wpkh) => wpkh.plan_satisfaction(provider),
525+
Descriptor::Wsh(ref wsh) => wsh.plan_satisfaction(provider),
526+
Descriptor::Sh(ref sh) => sh.plan_satisfaction(provider),
527+
Descriptor::Tr(ref tr) => tr.plan_satisfaction(provider),
528+
};
529+
530+
if let satisfy::Witness::Stack(stack) = satisfaction.stack {
531+
Ok(Plan {
532+
descriptor: self,
533+
template: stack,
534+
absolute_timelock: satisfaction.absolute_timelock.map(Into::into),
535+
relative_timelock: satisfaction.relative_timelock,
536+
})
537+
} else {
538+
Err(self)
539+
}
540+
}
541+
542+
/// Returns a plan if the provided assets are sufficient to produce a malleable satisfaction
543+
///
544+
/// If the assets aren't sufficient for generating a Plan, the descriptor is returned
545+
pub fn plan_mall<P>(self, provider: &P) -> Result<Plan, Self>
546+
where
547+
P: AssetProvider<DefiniteDescriptorKey>,
548+
{
549+
let satisfaction = match self {
550+
Descriptor::Bare(ref bare) => bare.plan_satisfaction_mall(provider),
551+
Descriptor::Pkh(ref pkh) => pkh.plan_satisfaction_mall(provider),
552+
Descriptor::Wpkh(ref wpkh) => wpkh.plan_satisfaction_mall(provider),
553+
Descriptor::Wsh(ref wsh) => wsh.plan_satisfaction_mall(provider),
554+
Descriptor::Sh(ref sh) => sh.plan_satisfaction_mall(provider),
555+
Descriptor::Tr(ref tr) => tr.plan_satisfaction_mall(provider),
556+
};
557+
558+
if let satisfy::Witness::Stack(stack) = satisfaction.stack {
559+
Ok(Plan {
560+
descriptor: self,
561+
template: stack,
562+
absolute_timelock: satisfaction.absolute_timelock.map(Into::into),
563+
relative_timelock: satisfaction.relative_timelock,
564+
})
565+
} else {
566+
Err(self)
567+
}
568+
}
569+
}
570+
512571
impl<P, Q> TranslatePk<P, Q> for Descriptor<P>
513572
where
514573
P: MiniscriptKey,

src/descriptor/segwitv0.rs

+72
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,11 @@ use bitcoin::{Address, Network, ScriptBuf};
1111

1212
use super::checksum::{self, verify_checksum};
1313
use super::SortedMultiVec;
14+
use crate::descriptor::DefiniteDescriptorKey;
1415
use crate::expression::{self, FromTree};
1516
use crate::miniscript::context::{ScriptContext, ScriptContextError};
17+
use crate::miniscript::satisfy::{Placeholder, Satisfaction, Witness};
18+
use crate::plan::AssetProvider;
1619
use crate::policy::{semantic, Liftable};
1720
use crate::prelude::*;
1821
use crate::util::varint_len;
@@ -191,6 +194,36 @@ impl<Pk: MiniscriptKey + ToPublicKey> Wsh<Pk> {
191194
}
192195
}
193196

197+
impl Wsh<DefiniteDescriptorKey> {
198+
/// Returns a plan if the provided assets are sufficient to produce a non-malleable satisfaction
199+
pub fn plan_satisfaction<P>(
200+
&self,
201+
provider: &P,
202+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
203+
where
204+
P: AssetProvider<DefiniteDescriptorKey>,
205+
{
206+
match &self.inner {
207+
WshInner::SortedMulti(sm) => sm.build_template(provider),
208+
WshInner::Ms(ms) => ms.build_template(provider),
209+
}
210+
}
211+
212+
/// Returns a plan if the provided assets are sufficient to produce a malleable satisfaction
213+
pub fn plan_satisfaction_mall<P>(
214+
&self,
215+
provider: &P,
216+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
217+
where
218+
P: AssetProvider<DefiniteDescriptorKey>,
219+
{
220+
match &self.inner {
221+
WshInner::SortedMulti(sm) => sm.build_template(provider),
222+
WshInner::Ms(ms) => ms.build_template_mall(provider),
223+
}
224+
}
225+
}
226+
194227
/// Wsh Inner
195228
#[derive(Clone, Ord, PartialOrd, Eq, PartialEq, Hash)]
196229
pub enum WshInner<Pk: MiniscriptKey> {
@@ -418,6 +451,45 @@ impl<Pk: MiniscriptKey + ToPublicKey> Wpkh<Pk> {
418451
}
419452
}
420453

454+
impl Wpkh<DefiniteDescriptorKey> {
455+
/// Returns a plan if the provided assets are sufficient to produce a non-malleable satisfaction
456+
pub fn plan_satisfaction<P>(
457+
&self,
458+
provider: &P,
459+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
460+
where
461+
P: AssetProvider<DefiniteDescriptorKey>,
462+
{
463+
let stack = if provider.provider_lookup_ecdsa_sig(&self.pk) {
464+
let stack = vec![
465+
Placeholder::EcdsaSigPk(self.pk.clone()),
466+
Placeholder::Pubkey(self.pk.clone(), Segwitv0::pk_len(&self.pk)),
467+
];
468+
Witness::Stack(stack)
469+
} else {
470+
Witness::Unavailable
471+
};
472+
473+
Satisfaction {
474+
stack,
475+
has_sig: true,
476+
relative_timelock: None,
477+
absolute_timelock: None,
478+
}
479+
}
480+
481+
/// Returns a plan if the provided assets are sufficient to produce a malleable satisfaction
482+
pub fn plan_satisfaction_mall<P>(
483+
&self,
484+
provider: &P,
485+
) -> Satisfaction<Placeholder<DefiniteDescriptorKey>>
486+
where
487+
P: AssetProvider<DefiniteDescriptorKey>,
488+
{
489+
self.plan_satisfaction(provider)
490+
}
491+
}
492+
421493
impl<Pk: MiniscriptKey> fmt::Debug for Wpkh<Pk> {
422494
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
423495
write!(f, "wpkh({:?})", self.pk)

0 commit comments

Comments
 (0)