|
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,49 @@ 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 | + non_witness_utxo: None, |
| 76 | + witness_utxo: Some(TxOut { value: txout.value, script_pubkey: txout.script_pubkey }), |
| 77 | + partial_sigs: Default::default(), |
| 78 | + sighash_type: None, |
| 79 | + redeem_script: None, |
| 80 | + witness_script: None, |
| 81 | + bip32_derivation: Default::default(), |
| 82 | + final_script_sig: None, |
| 83 | + final_script_witness: None, |
| 84 | + ripemd160_preimages: Default::default(), |
| 85 | + sha256_preimages: Default::default(), |
| 86 | + hash160_preimages: Default::default(), |
| 87 | + hash256_preimages: Default::default(), |
| 88 | + tap_key_sig: None, |
| 89 | + tap_script_sigs: Default::default(), |
| 90 | + tap_scripts: Default::default(), |
| 91 | + tap_key_origins: Default::default(), |
| 92 | + tap_internal_key: None, |
| 93 | + tap_merkle_root: None, |
| 94 | + proprietary: Default::default(), |
| 95 | + unknown: Default::default(), |
| 96 | + }; |
| 97 | + let input_pair = Self { txin, psbtin }; |
| 98 | + let raw = InternalInputPair::from(&input_pair); |
| 99 | + raw.validate_utxo()?; |
| 100 | + |
| 101 | + Ok(input_pair) |
| 102 | + } |
| 103 | + |
61 | 104 | pub(crate) fn previous_txout(&self) -> TxOut { |
62 | 105 | InternalInputPair::from(self) |
63 | 106 | .previous_txout() |
@@ -88,3 +131,32 @@ pub(crate) fn parse_payload( |
88 | 131 |
|
89 | 132 | Ok((psbt, params)) |
90 | 133 | } |
| 134 | + |
| 135 | +#[cfg(test)] |
| 136 | +mod tests { |
| 137 | + use bitcoin::hashes::Hash; |
| 138 | + use bitcoin::key::WPubkeyHash; |
| 139 | + use bitcoin::{ScriptBuf, ScriptHash, Txid}; |
| 140 | + |
| 141 | + use super::*; |
| 142 | + use crate::psbt::InternalPsbtInputError::InvalidP2wpkhScriptPubkey; |
| 143 | + |
| 144 | + #[test] |
| 145 | + fn create_p2wpkh_input_pair() { |
| 146 | + let outpoint = OutPoint { txid: Txid::all_zeros(), vout: 0 }; |
| 147 | + let p2wpkh_txout = TxOut { |
| 148 | + value: Default::default(), |
| 149 | + script_pubkey: ScriptBuf::new_p2wpkh(&WPubkeyHash::all_zeros()), |
| 150 | + }; |
| 151 | + assert!(InputPair::new_p2wpkh(p2wpkh_txout, outpoint).is_ok()); |
| 152 | + |
| 153 | + let p2sh_txout = TxOut { |
| 154 | + value: Default::default(), |
| 155 | + script_pubkey: ScriptBuf::new_p2sh(&ScriptHash::all_zeros()), |
| 156 | + }; |
| 157 | + assert_eq!( |
| 158 | + InputPair::new_p2wpkh(p2sh_txout, outpoint).err().unwrap(), |
| 159 | + PsbtInputError::from(InvalidP2wpkhScriptPubkey) |
| 160 | + ) |
| 161 | + } |
| 162 | +} |
0 commit comments