|
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, TxIn, TxOut}; |
15 | 15 | pub(crate) use error::InternalPayloadError; |
16 | 16 | pub use error::{ |
17 | 17 | Error, InputContributionError, JsonReply, OutputSubstitutionError, PayloadError, |
@@ -58,6 +58,30 @@ 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(txout: TxOut, outpoint: OutPoint) -> Result<Self, PsbtInputError> { |
| 63 | + if !txout.script_pubkey.is_p2wpkh() { |
| 64 | + return Err(InternalPsbtInputError::InvalidP2wpkhScriptPubkey.into()); |
| 65 | + } |
| 66 | + |
| 67 | + let txin = TxIn { |
| 68 | + previous_output: OutPoint { txid: outpoint.txid, vout: outpoint.vout }, |
| 69 | + script_sig: Default::default(), |
| 70 | + sequence: Default::default(), |
| 71 | + witness: Default::default(), |
| 72 | + }; |
| 73 | + |
| 74 | + let psbtin = psbt::Input { |
| 75 | + witness_utxo: Some(TxOut { value: txout.value, script_pubkey: txout.script_pubkey }), |
| 76 | + ..psbt::Input::default() |
| 77 | + }; |
| 78 | + let input_pair = Self { txin, psbtin }; |
| 79 | + let raw = InternalInputPair::from(&input_pair); |
| 80 | + raw.validate_utxo()?; |
| 81 | + |
| 82 | + Ok(input_pair) |
| 83 | + } |
| 84 | + |
61 | 85 | pub(crate) fn previous_txout(&self) -> TxOut { |
62 | 86 | InternalInputPair::from(self) |
63 | 87 | .previous_txout() |
@@ -88,3 +112,35 @@ pub(crate) fn parse_payload( |
88 | 112 |
|
89 | 113 | Ok((psbt, params)) |
90 | 114 | } |
| 115 | + |
| 116 | +#[cfg(test)] |
| 117 | +mod tests { |
| 118 | + use bitcoin::hashes::Hash; |
| 119 | + use bitcoin::key::WPubkeyHash; |
| 120 | + use bitcoin::{Amount, ScriptBuf, ScriptHash, Txid}; |
| 121 | + use payjoin_test_utils::{DUMMY20, DUMMY32}; |
| 122 | + |
| 123 | + use super::*; |
| 124 | + use crate::psbt::InternalPsbtInputError::InvalidP2wpkhScriptPubkey; |
| 125 | + |
| 126 | + #[test] |
| 127 | + fn create_p2wpkh_input_pair() { |
| 128 | + let outpoint = OutPoint { txid: Txid::from_byte_array(DUMMY32), vout: 31 }; |
| 129 | + let p2wpkh_txout = TxOut { |
| 130 | + value: Amount::from_sat(12345), |
| 131 | + script_pubkey: ScriptBuf::new_p2wpkh(&WPubkeyHash::from_byte_array(DUMMY20)), |
| 132 | + }; |
| 133 | + let p2wpkh_pair = InputPair::new_p2wpkh(p2wpkh_txout.clone(), outpoint).unwrap(); |
| 134 | + assert_eq!(p2wpkh_pair.txin.previous_output, outpoint); |
| 135 | + assert_eq!(p2wpkh_pair.psbtin.witness_utxo.unwrap(), p2wpkh_txout); |
| 136 | + |
| 137 | + let p2sh_txout = TxOut { |
| 138 | + value: Default::default(), |
| 139 | + script_pubkey: ScriptBuf::new_p2sh(&ScriptHash::all_zeros()), |
| 140 | + }; |
| 141 | + assert_eq!( |
| 142 | + InputPair::new_p2wpkh(p2sh_txout, outpoint).err().unwrap(), |
| 143 | + PsbtInputError::from(InvalidP2wpkhScriptPubkey) |
| 144 | + ) |
| 145 | + } |
| 146 | +} |
0 commit comments