Skip to content

Commit

Permalink
Merge pull request #2 from mempool/fix/p2tr
Browse files Browse the repository at this point in the history
Fix: Parse the inner witness_script properly for p2tr
  • Loading branch information
wiz authored Jul 13, 2023
2 parents 0c7335d + c876aca commit ac8968a
Showing 1 changed file with 34 additions and 1 deletion.
35 changes: 34 additions & 1 deletion src/util/script.rs
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@ pub fn get_innerscripts(txin: &TxIn, prevout: &TxOut) -> InnerScripts {

// Wrapped witnessScript for P2WSH or P2SH-P2WSH spends
let witness_script = if prevout.script_pubkey.is_v0_p2wsh()
|| prevout.script_pubkey.is_v1_p2tr()
|| redeem_script.as_ref().map_or(false, |s| s.is_v0_p2wsh())
{
let witness = &txin.witness;
Expand All @@ -63,7 +64,39 @@ pub fn get_innerscripts(txin: &TxIn, prevout: &TxOut) -> InnerScripts {
#[cfg(feature = "liquid")]
let wit_to_vec = Clone::clone;

witness.iter().last().map(wit_to_vec).map(Script::from)
let inner_script_slice = if prevout.script_pubkey.is_v1_p2tr() {
// Witness stack is potentially very large
// so we avoid to_vec() or iter().collect() for performance
let w_len = witness.len();
witness
.last()
// Get the position of the script spend script (if it exists)
.map(|last_elem| {
// From BIP341:
// If there are at least two witness elements, and the first byte of
// the last element is 0x50, this last element is called annex a
// and is removed from the witness stack.
if w_len >= 2 && last_elem.get(0).filter(|&&v| v == 0x50).is_some() {
// account for the extra item removed from the end
3
} else {
// otherwise script is 2nd from last
2
}
})
// Convert to None if not script spend
// Note: Option doesn't have filter_map() method
.filter(|&script_pos_from_last| w_len >= script_pos_from_last)
.map(|script_pos_from_last| {
// Can't use second_to_last() since it might be 3rd to last
witness.iter().nth(w_len - script_pos_from_last)
})
.flatten()
} else {
witness.last()
};

inner_script_slice.map(wit_to_vec).map(Script::from)
} else {
None
};
Expand Down

0 comments on commit ac8968a

Please sign in to comment.