Skip to content

Commit dde44e2

Browse files
committed
Add psbt finalizer support
1 parent bf49604 commit dde44e2

File tree

4 files changed

+129
-16
lines changed

4 files changed

+129
-16
lines changed

src/lib.rs

+3
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,9 @@ pub use miniscript::decode::Terminal;
129129
pub use miniscript::satisfy::{Preimage32, Satisfier};
130130
pub use miniscript::Miniscript;
131131

132+
// Use schnorr pubkey as Xonlykey
133+
pub(crate) use bitcoin::schnorr::PublicKey as XOnlyKey;
134+
132135
///Public key trait which can be converted to Hash type
133136
pub trait MiniscriptKey: Clone + Eq + Ord + fmt::Debug + fmt::Display + hash::Hash {
134137
/// Check if the publicKey is uncompressed. The default

src/psbt/finalizer.rs

+104-16
Original file line numberDiff line numberDiff line change
@@ -19,15 +19,79 @@
1919
//! `https://github.com/bitcoin/bips/blob/master/bip-0174.mediawiki`
2020
//!
2121
22+
use util::{script_is_v1_tr, witness_size};
23+
2224
use super::{sanity_check, Psbt};
2325
use super::{Error, InputError, PsbtInputSatisfier};
2426
use bitcoin::secp256k1::{self, Secp256k1};
27+
use bitcoin::util::taproot::LeafVersion;
2528
use bitcoin::{self, PublicKey, Script};
2629
use descriptor::DescriptorTrait;
2730
use interpreter;
2831
use Descriptor;
2932
use Miniscript;
30-
use {BareCtx, Legacy, Segwitv0};
33+
use Satisfier;
34+
use XOnlyKey;
35+
use {BareCtx, Legacy, Segwitv0, Tap};
36+
37+
// Satisfy the taproot descriptor. It is not possible to infer the complete
38+
// descriptor from psbt because the information about all the scripts might not
39+
// be present. Also, currently the spec does not support hidden branches, so
40+
// inferring a descriptor is not possible
41+
fn construct_tap_witness(
42+
spk: &Script,
43+
sat: &PsbtInputSatisfier,
44+
allow_mall: bool,
45+
) -> Result<Vec<Vec<u8>>, InputError> {
46+
assert!(script_is_v1_tr(&spk));
47+
48+
// try the script spend path first
49+
if let Some(sig) = <PsbtInputSatisfier as Satisfier<XOnlyKey>>::lookup_tap_key_spend_sig(sat) {
50+
return Ok(vec![sig.to_vec()]);
51+
}
52+
// Next script spends
53+
let (mut min_wit, mut min_wit_len) = (None, None);
54+
if let Some(block_map) =
55+
<PsbtInputSatisfier as Satisfier<XOnlyKey>>::lookup_tap_control_block_map(sat)
56+
{
57+
for (control_block, (script, ver)) in block_map {
58+
if *ver != LeafVersion::default() {
59+
// We don't know how to satisfy non default version scripts yet
60+
continue;
61+
}
62+
let ms = match Miniscript::<XOnlyKey, Tap>::parse_insane(script) {
63+
Ok(ms) => ms,
64+
Err(..) => continue, // try another script
65+
};
66+
let mut wit = if allow_mall {
67+
match ms.satisfy_malleable(sat) {
68+
Ok(ms) => ms,
69+
Err(..) => continue,
70+
}
71+
} else {
72+
match ms.satisfy(sat) {
73+
Ok(ms) => ms,
74+
Err(..) => continue,
75+
}
76+
};
77+
wit.push(ms.encode().into_bytes());
78+
wit.push(control_block.serialize());
79+
let wit_len = Some(witness_size(&wit));
80+
if min_wit_len.is_some() && wit_len > min_wit_len {
81+
continue;
82+
} else {
83+
// store the minimum
84+
min_wit = Some(wit);
85+
min_wit_len = wit_len;
86+
}
87+
}
88+
min_wit.ok_or(InputError::CouldNotSatisfyTr)
89+
} else {
90+
// No control blocks found
91+
Err(InputError::CouldNotSatisfyTr)
92+
}
93+
}
94+
3195
// Get the scriptpubkey for the psbt input
3296
fn get_scriptpubkey(psbt: &Psbt, index: usize) -> Result<&Script, InputError> {
3397
let script_pubkey;
@@ -320,16 +384,28 @@ pub fn finalize_helper<C: secp256k1::Verification>(
320384

321385
// Actually construct the witnesses
322386
for index in 0..psbt.inputs.len() {
323-
// Get a descriptor for this input
324-
let desc = get_descriptor(&psbt, index).map_err(|e| Error::InputError(e, index))?;
387+
let (witness, script_sig) = {
388+
let spk = get_scriptpubkey(psbt, index).map_err(|e| Error::InputError(e, index))?;
389+
let sat = PsbtInputSatisfier::new(&psbt, index);
325390

326-
//generate the satisfaction witness and scriptsig
327-
let (witness, script_sig) = if !allow_mall {
328-
desc.get_satisfaction(PsbtInputSatisfier::new(&psbt, index))
329-
} else {
330-
desc.get_satisfaction_mall(PsbtInputSatisfier::new(&psbt, index))
331-
}
332-
.map_err(|e| Error::InputError(InputError::MiniscriptError(e), index))?;
391+
if script_is_v1_tr(spk) {
392+
// Deal with tr case separately, unfortunately we cannot infer the full descriptor for Tr
393+
let wit = construct_tap_witness(spk, &sat, allow_mall)
394+
.map_err(|e| Error::InputError(e, index))?;
395+
(wit, Script::new())
396+
} else {
397+
// Get a descriptor for this input.
398+
let desc = get_descriptor(&psbt, index).map_err(|e| Error::InputError(e, index))?;
399+
400+
//generate the satisfaction witness and scriptsig
401+
if !allow_mall {
402+
desc.get_satisfaction(PsbtInputSatisfier::new(&psbt, index))
403+
} else {
404+
desc.get_satisfaction_mall(PsbtInputSatisfier::new(&psbt, index))
405+
}
406+
.map_err(|e| Error::InputError(InputError::MiniscriptError(e), index))?
407+
}
408+
};
333409

334410
let input = &mut psbt.inputs[index];
335411
//Fill in the satisfactions
@@ -344,12 +420,24 @@ pub fn finalize_helper<C: secp256k1::Verification>(
344420
Some(witness)
345421
};
346422
//reset everything
347-
input.redeem_script = None;
348-
input.partial_sigs.clear();
349-
input.sighash_type = None;
350-
input.redeem_script = None;
351-
input.bip32_derivation.clear();
352-
input.witness_script = None;
423+
input.partial_sigs.clear(); // 0x02
424+
input.sighash_type = None; // 0x03
425+
input.redeem_script = None; // 0x04
426+
input.witness_script = None; // 0x05
427+
input.bip32_derivation.clear(); // 0x05
428+
// finalized witness 0x06 and 0x07 are not clear
429+
// 0x09 Proof of reserves not yet supported
430+
input.ripemd160_preimages.clear(); // 0x0a
431+
input.sha256_preimages.clear(); // 0x0b
432+
input.hash160_preimages.clear(); // 0x0c
433+
input.hash256_preimages.clear(); // 0x0d
434+
// psbt v2 fields till 0x012 not supported
435+
input.tap_key_sig = None; // 0x013
436+
input.tap_script_sigs.clear(); // 0x014
437+
input.tap_scripts.clear(); // 0x015
438+
input.tap_key_origins.clear(); // 0x16
439+
input.tap_internal_key = None; // x017
440+
input.tap_merkle_root = None; // 0x018
353441
}
354442
// Double check everything with the interpreter
355443
// This only checks whether the script will be executed

src/psbt/mod.rs

+8
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,11 @@ pub enum InputError {
4444
SecpErr(bitcoin::secp256k1::Error),
4545
/// Key errors
4646
KeyErr(bitcoin::util::key::Error),
47+
/// Could not satisfy taproot descriptor
48+
/// This error is returned when both script path and key paths could not be
49+
/// satisfied. We cannot return a detailed error because we try all miniscripts
50+
/// in script spend path, we cannot know which miniscript failed.
51+
CouldNotSatisfyTr,
4752
/// Error doing an interpreter-check on a finalized psbt
4853
Interpreter(interpreter::Error),
4954
/// Redeem script does not match the p2sh hash
@@ -159,6 +164,9 @@ impl fmt::Display for InputError {
159164
sighashflag {:?} rather than required {:?}",
160165
pubkey.key, got, required
161166
),
167+
InputError::CouldNotSatisfyTr => {
168+
write!(f, "Could not satisfy Tr descriptor")
169+
}
162170
}
163171
}
164172
}

src/util.rs

+14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
use bitcoin;
2+
use bitcoin::blockdata::opcodes;
23
use bitcoin::blockdata::script;
4+
use bitcoin::blockdata::script::Instruction;
35
use bitcoin::Script;
46

57
use {ScriptContext, ToPublicKey};
@@ -46,3 +48,15 @@ impl MsKeyBuilder for script::Builder {
4648
}
4749
}
4850
}
51+
52+
// This belongs in rust-bitcoin, make this upstream later
53+
pub(crate) fn script_is_v1_tr(s: &Script) -> bool {
54+
let mut iter = s.instructions_minimal();
55+
if s.len() != 32
56+
|| iter.next() != Some(Ok(Instruction::Op(opcodes::all::OP_PUSHNUM_1)))
57+
|| iter.next() != Some(Ok(Instruction::Op(opcodes::all::OP_PUSHBYTES_32)))
58+
{
59+
return false;
60+
}
61+
return true;
62+
}

0 commit comments

Comments
 (0)