|
11 | 11 |
|
12 | 12 | use std::str::FromStr; |
13 | 13 |
|
14 | | -use bitcoin::{psbt, AddressType, Psbt, TxIn, TxOut}; |
| 14 | +use bitcoin::{psbt, AddressType, OutPoint, Psbt, Sequence, TxIn, TxOut}; |
15 | 15 | pub(crate) use error::InternalPayloadError; |
16 | 16 | pub use error::{ |
17 | 17 | Error, InputContributionError, JsonReply, OutputSubstitutionError, PayloadError, |
@@ -58,6 +58,34 @@ impl InputPair { |
58 | 58 | Ok(input_pair) |
59 | 59 | } |
60 | 60 |
|
| 61 | + /// Constructs a new ['InputPair'] for spending a native SegWit P2WPKH output |
| 62 | + pub fn new_p2wpkh( |
| 63 | + txout: TxOut, |
| 64 | + outpoint: OutPoint, |
| 65 | + sequence: Option<Sequence>, |
| 66 | + ) -> Result<Self, PsbtInputError> { |
| 67 | + if !txout.script_pubkey.is_p2wpkh() { |
| 68 | + return Err(InternalPsbtInputError::InvalidP2wpkhScriptPubkey.into()); |
| 69 | + } |
| 70 | + |
| 71 | + let txin = TxIn { |
| 72 | + previous_output: OutPoint { txid: outpoint.txid, vout: outpoint.vout }, |
| 73 | + script_sig: Default::default(), |
| 74 | + sequence: sequence.unwrap_or_default(), |
| 75 | + witness: Default::default(), |
| 76 | + }; |
| 77 | + |
| 78 | + let psbtin = psbt::Input { |
| 79 | + witness_utxo: Some(TxOut { value: txout.value, script_pubkey: txout.script_pubkey }), |
| 80 | + ..psbt::Input::default() |
| 81 | + }; |
| 82 | + let input_pair = Self { txin, psbtin }; |
| 83 | + let raw = InternalInputPair::from(&input_pair); |
| 84 | + raw.validate_utxo()?; |
| 85 | + |
| 86 | + Ok(input_pair) |
| 87 | + } |
| 88 | + |
61 | 89 | pub(crate) fn previous_txout(&self) -> TxOut { |
62 | 90 | InternalInputPair::from(self) |
63 | 91 | .previous_txout() |
@@ -88,3 +116,38 @@ pub(crate) fn parse_payload( |
88 | 116 |
|
89 | 117 | Ok((psbt, params)) |
90 | 118 | } |
| 119 | + |
| 120 | +#[cfg(test)] |
| 121 | +mod tests { |
| 122 | + use bitcoin::hashes::Hash; |
| 123 | + use bitcoin::key::WPubkeyHash; |
| 124 | + use bitcoin::{Amount, ScriptBuf, ScriptHash, Txid}; |
| 125 | + use payjoin_test_utils::{DUMMY20, DUMMY32}; |
| 126 | + |
| 127 | + use super::*; |
| 128 | + use crate::psbt::InternalPsbtInputError::InvalidP2wpkhScriptPubkey; |
| 129 | + |
| 130 | + #[test] |
| 131 | + fn create_p2wpkh_input_pair() { |
| 132 | + let outpoint = OutPoint { txid: Txid::from_byte_array(DUMMY32), vout: 31 }; |
| 133 | + let sequence = Sequence::from_512_second_intervals(123); |
| 134 | + let p2wpkh_txout = TxOut { |
| 135 | + value: Amount::from_sat(12345), |
| 136 | + script_pubkey: ScriptBuf::new_p2wpkh(&WPubkeyHash::from_byte_array(DUMMY20)), |
| 137 | + }; |
| 138 | + let p2wpkh_pair = |
| 139 | + InputPair::new_p2wpkh(p2wpkh_txout.clone(), outpoint, Some(sequence)).unwrap(); |
| 140 | + assert_eq!(p2wpkh_pair.txin.previous_output, outpoint); |
| 141 | + assert_eq!(p2wpkh_pair.txin.sequence, sequence); |
| 142 | + assert_eq!(p2wpkh_pair.psbtin.witness_utxo.unwrap(), p2wpkh_txout); |
| 143 | + |
| 144 | + let p2sh_txout = TxOut { |
| 145 | + value: Default::default(), |
| 146 | + script_pubkey: ScriptBuf::new_p2sh(&ScriptHash::all_zeros()), |
| 147 | + }; |
| 148 | + assert_eq!( |
| 149 | + InputPair::new_p2wpkh(p2sh_txout, outpoint, Some(sequence)).err().unwrap(), |
| 150 | + PsbtInputError::from(InvalidP2wpkhScriptPubkey) |
| 151 | + ) |
| 152 | + } |
| 153 | +} |
0 commit comments