Branch data Line data Source code
1 : : // Copyright (c) 2009-present The Bitcoin Core developers
2 : : // Distributed under the MIT software license, see the accompanying
3 : : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
4 : :
5 : : #include <psbt.h>
6 : :
7 : : #include <common/types.h>
8 : : #include <node/types.h>
9 : : #include <policy/policy.h>
10 : : #include <primitives/transaction.h>
11 : : #include <script/signingprovider.h>
12 : : #include <util/check.h>
13 : : #include <util/result.h>
14 : : #include <util/strencodings.h>
15 : :
16 : : using common::PSBTError;
17 : :
18 [ + - ]: 4 : PartiallySignedTransaction::PartiallySignedTransaction(const CMutableTransaction& tx, uint32_t version) : m_version(version)
19 : : {
20 [ + - - + ]: 4 : assert(m_version == 0 || m_version == 2);
21 : :
22 : 4 : tx_version = tx.version;
23 [ - + ]: 4 : fallback_locktime = tx.nLockTime;
24 [ - + + - ]: 4 : inputs.reserve(tx.vin.size());
25 [ + + ]: 6 : for (const CTxIn& input : tx.vin) {
26 [ + - + - ]: 2 : inputs.emplace_back(GetVersion(), input.prevout.hash, input.prevout.n, input.nSequence);
27 : : }
28 [ - + + - ]: 4 : outputs.reserve(tx.vout.size());
29 [ + + ]: 6 : for (const CTxOut& output : tx.vout) {
30 [ + - + - ]: 2 : outputs.emplace_back(GetVersion(), output.nValue, output.scriptPubKey);
31 : : }
32 : 4 : }
33 : :
34 : 0 : bool PartiallySignedTransaction::IsNull() const
35 : : {
36 [ # # # # : 0 : return inputs.empty() && outputs.empty() && unknown.empty();
# # ]
37 : : }
38 : :
39 : 1 : bool PartiallySignedTransaction::Merge(const PartiallySignedTransaction& psbt)
40 : : {
41 : : // Prohibited to merge two PSBTs over different transactions
42 : 1 : std::optional<Txid> this_id = GetUniqueID();
43 : 1 : std::optional<Txid> psbt_id = psbt.GetUniqueID();
44 [ + - + - : 1 : if (!this_id || !psbt_id || this_id != psbt_id) {
+ - ]
45 : : return false;
46 : : }
47 [ + - ]: 1 : if (GetVersion() != psbt.GetVersion()) {
48 : : return false;
49 : : }
50 : :
51 [ - + + + ]: 2 : for (unsigned int i = 0; i < inputs.size(); ++i) {
52 [ + - ]: 1 : if (!inputs[i].Merge(psbt.inputs[i])) {
53 : : return false;
54 : : }
55 : : }
56 [ - + + + ]: 2 : for (unsigned int i = 0; i < outputs.size(); ++i) {
57 [ + - ]: 1 : if (!outputs[i].Merge(psbt.outputs[i])) {
58 : : return false;
59 : : }
60 : : }
61 [ - + ]: 1 : for (auto& xpub_pair : psbt.m_xpubs) {
62 [ # # ]: 0 : if (!m_xpubs.contains(xpub_pair.first)) {
63 : 0 : m_xpubs[xpub_pair.first] = xpub_pair.second;
64 : : } else {
65 : 0 : m_xpubs[xpub_pair.first].insert(xpub_pair.second.begin(), xpub_pair.second.end());
66 : : }
67 : : }
68 [ - + - - ]: 1 : if (fallback_locktime == std::nullopt && psbt.fallback_locktime != std::nullopt) fallback_locktime = psbt.fallback_locktime;
69 : :
70 : : // Set m_tx_modifiable only if either PSBT had it set
71 [ + - - + ]: 1 : if (m_tx_modifiable.has_value() || psbt.m_tx_modifiable.has_value()) {
72 : : // In general, we AND the modifiable flags
73 [ # # ]: 0 : std::bitset<8> this_modifiable = m_tx_modifiable.value_or(0);
74 [ # # ]: 0 : std::bitset<8> psbt_modifiable = psbt.m_tx_modifiable.value_or(0);
75 [ # # ]: 0 : std::bitset<8> final_modifiable = this_modifiable & psbt_modifiable;
76 : : // SIGHASH_SINGLE Modifiable (bit 2) needs to be bitwise OR'd
77 [ # # # # : 0 : final_modifiable.set(2, this_modifiable[2] || psbt_modifiable[2]);
# # ]
78 : :
79 [ # # ]: 0 : m_tx_modifiable = final_modifiable;
80 : : }
81 : :
82 : 1 : m_proprietary.insert(psbt.m_proprietary.begin(), psbt.m_proprietary.end());
83 : 1 : unknown.insert(psbt.unknown.begin(), psbt.unknown.end());
84 : :
85 : 1 : return true;
86 : : }
87 : :
88 : 45 : std::optional<uint32_t> PartiallySignedTransaction::ComputeTimeLock() const
89 : : {
90 [ + + ]: 45 : if (GetVersion() >= 2) {
91 : 26 : std::optional<uint32_t> time_lock{0};
92 : 26 : std::optional<uint32_t> height_lock{0};
93 [ + + ]: 78 : for (const PSBTInput& input : inputs) {
94 [ + + + + ]: 54 : if (input.time_locktime.has_value() && !input.height_locktime.has_value()) {
95 [ + - ]: 10 : height_lock.reset(); // Transaction can no longer have a height locktime
96 [ + + ]: 10 : if (!time_lock.has_value()) {
97 : 2 : return std::nullopt;
98 : : }
99 [ + + + + ]: 44 : } else if (!input.time_locktime.has_value() && input.height_locktime.has_value()) {
100 [ + + ]: 10 : time_lock.reset(); // Transaction can no longer have a time locktime
101 [ - + ]: 10 : if (!height_lock.has_value()) {
102 : 0 : return std::nullopt;
103 : : }
104 : : }
105 [ + + + + ]: 52 : if (input.time_locktime && time_lock.has_value()) {
106 : 19 : time_lock = std::max(time_lock, input.time_locktime);
107 : : }
108 [ + + + + ]: 52 : if (input.height_locktime && height_lock.has_value()) {
109 : 21 : height_lock = std::max(height_lock, input.height_locktime);
110 : : }
111 : : }
112 [ + + + + ]: 24 : if (height_lock.has_value() && *height_lock > 0) {
113 : 9 : return *height_lock;
114 : : }
115 [ + - + + ]: 15 : if (time_lock.has_value() && *time_lock > 0) {
116 : 8 : return *time_lock;
117 : : }
118 : : }
119 [ + + ]: 50 : return fallback_locktime.value_or(0);
120 : : }
121 : :
122 : 31 : std::optional<CMutableTransaction> PartiallySignedTransaction::GetUnsignedTx() const
123 : : {
124 : 31 : CMutableTransaction mtx;
125 : 31 : mtx.version = tx_version;
126 [ + - ]: 31 : std::optional<uint32_t> locktime = ComputeTimeLock();
127 [ + + ]: 31 : if (!locktime) {
128 : 1 : return std::nullopt;
129 : : }
130 : 30 : mtx.nLockTime = *locktime;
131 : 30 : uint32_t max_sequence = CTxIn::SEQUENCE_FINAL;
132 [ + + ]: 87 : for (const PSBTInput& input : inputs) {
133 : 57 : CTxIn txin;
134 : 57 : txin.prevout.hash = input.prev_txid;
135 : 57 : txin.prevout.n = input.prev_out;
136 [ + + ]: 57 : txin.nSequence = input.sequence.value_or(max_sequence);
137 [ + - ]: 57 : mtx.vin.push_back(txin);
138 : 57 : }
139 [ + + ]: 80 : for (const PSBTOutput& output : outputs) {
140 : 50 : CTxOut txout;
141 : 50 : txout.nValue = output.amount;
142 : 50 : txout.scriptPubKey = output.script;
143 [ + - ]: 50 : mtx.vout.push_back(txout);
144 : 50 : }
145 : 30 : return mtx;
146 : 31 : }
147 : :
148 : 2 : std::optional<Txid> PartiallySignedTransaction::GetUniqueID() const
149 : : {
150 : : // Get the unsigned transaction
151 : 2 : std::optional<CMutableTransaction> mtx = GetUnsignedTx();
152 [ - + ]: 2 : if (!mtx) {
153 : 0 : return std::nullopt;
154 : : }
155 [ + - + - ]: 2 : if (GetVersion() >= 2) {
156 [ + + ]: 4 : for (CTxIn& txin : mtx->vin) {
157 : 2 : txin.nSequence = 0;
158 : : }
159 : : }
160 [ + - ]: 2 : return mtx->GetHash();
161 : 2 : }
162 : :
163 : 15 : bool PartiallySignedTransaction::AddInput(const PSBTInput& psbtin)
164 : : {
165 : : // The input being added must be for this PSBT's version
166 [ + + ]: 15 : if (psbtin.GetVersion() != GetVersion()) {
167 : : return false;
168 : : }
169 : :
170 : : // Prevent duplicate inputs
171 [ + - ]: 14 : if (std::find_if(inputs.begin(), inputs.end(),
172 [ + - + + ]: 84 : [psbtin](const PSBTInput& psbt) {
173 [ + + - + ]: 47 : return psbt.prev_txid == psbtin.prev_txid && psbt.prev_out == psbtin.prev_out;
174 : : }
175 : 28 : ) != inputs.end()) {
176 : : return false;
177 : : }
178 : :
179 [ - + ]: 8 : if (GetVersion() < 2) {
180 : : // This is a v0 psbt, so do the v0 AddInput
181 : 0 : inputs.push_back(psbtin);
182 : 0 : inputs.back().partial_sigs.clear();
183 : 0 : inputs.back().final_script_sig.clear();
184 : 0 : inputs.back().final_script_witness.SetNull();
185 : 0 : return true;
186 : : }
187 : :
188 : : // Check inputs modifiable flag
189 [ + - + + ]: 8 : if (!m_tx_modifiable.has_value() || !m_tx_modifiable->test(0)) {
190 : : return false;
191 : : }
192 : :
193 : : // Determine if we need to iterate the inputs.
194 : : // For now, we only do this if the new input has a required time lock.
195 : : // BIP 370 states that we should also do this if m_tx_modifiable's bit 2 is set
196 : : // (Has SIGHASH_SINGLE flag) but since we are only adding inputs at the end of the vector,
197 : : // we don't care about that.
198 [ + + + + ]: 7 : bool iterate_inputs = psbtin.time_locktime != std::nullopt || psbtin.height_locktime != std::nullopt;
199 : 4 : if (iterate_inputs) {
200 : 4 : std::optional<uint32_t> old_timelock = ComputeTimeLock();
201 [ + - ]: 4 : if (!old_timelock) {
202 : : return false;
203 : : }
204 : :
205 : 4 : std::optional<uint32_t> time_lock = psbtin.time_locktime;
206 : 4 : std::optional<uint32_t> height_lock = psbtin.height_locktime;
207 : 4 : bool has_sigs = false;
208 [ + + ]: 17 : for (const PSBTInput& input : inputs) {
209 [ + + + + ]: 14 : if (input.time_locktime.has_value() && !input.height_locktime.has_value()) {
210 [ + + ]: 2 : height_lock.reset(); // Transaction can no longer have a height locktime
211 [ + + ]: 2 : if (time_lock == std::nullopt) {
212 : : return false;
213 : : }
214 [ + + - + ]: 12 : } else if (!input.time_locktime.has_value() && input.height_locktime.has_value()) {
215 [ # # ]: 0 : time_lock.reset(); // Transaction can no longer have a time locktime
216 [ # # ]: 0 : if (height_lock == std::nullopt) {
217 : : return false;
218 : : }
219 : : }
220 [ + + + + ]: 13 : if (input.time_locktime && time_lock.has_value()) {
221 : 3 : time_lock = std::max(time_lock, input.time_locktime);
222 : : }
223 [ + + + + ]: 13 : if (input.height_locktime && height_lock.has_value()) {
224 : 1 : height_lock = std::max(height_lock, input.height_locktime);
225 : : }
226 [ + + ]: 13 : if (input.HasSignatures()) {
227 : 1 : has_sigs = true;
228 : : }
229 : : }
230 [ + - ]: 3 : uint32_t new_timelock = fallback_locktime.value_or(0);
231 [ + + - + ]: 3 : if (height_lock.has_value() && *height_lock > 0) {
232 : : new_timelock = *height_lock;
233 [ + - - + ]: 2 : } else if (time_lock.has_value() && *time_lock > 0) {
234 : : new_timelock = *time_lock;
235 : : }
236 [ + + - + ]: 3 : if (has_sigs && *old_timelock != new_timelock) {
237 : : return false;
238 : : }
239 : : }
240 : :
241 : : // Add the input to the end
242 : 5 : inputs.push_back(psbtin);
243 : 5 : return true;
244 : : }
245 : :
246 : 4 : bool PartiallySignedTransaction::AddOutput(const PSBTOutput& psbtout)
247 : : {
248 : : // The output being added must be for this PSBT's version
249 [ + + ]: 4 : if (psbtout.GetVersion() != GetVersion()) {
250 : : return false;
251 : : }
252 : :
253 [ - + ]: 3 : if (GetVersion() < 2) {
254 : : // This is a v0 psbt, do the v0 AddOutput
255 : 0 : outputs.push_back(psbtout);
256 : 0 : return true;
257 : : }
258 : :
259 : : // No global tx, must be PSBTv2
260 : : // Check outputs are modifiable
261 [ + - + + ]: 3 : if (!m_tx_modifiable.has_value() || !m_tx_modifiable->test(1)) {
262 : : return false;
263 : : }
264 : 2 : outputs.push_back(psbtout);
265 : :
266 : 2 : return true;
267 : : }
268 : :
269 : 4 : bool PSBTInput::GetUTXO(CTxOut& utxo) const
270 : : {
271 [ + - ]: 4 : if (non_witness_utxo) {
272 [ - + + + ]: 4 : if (prev_out >= non_witness_utxo->vout.size()) {
273 : : return false;
274 : : }
275 [ + - ]: 3 : if (non_witness_utxo->GetHash() != prev_txid) {
276 : : return false;
277 : : }
278 : 3 : utxo = non_witness_utxo->vout[prev_out];
279 [ # # ]: 0 : } else if (!witness_utxo.IsNull()) {
280 : 0 : utxo = witness_utxo;
281 : : } else {
282 : : return false;
283 : : }
284 : : return true;
285 : : }
286 : :
287 : 14 : COutPoint PSBTInput::GetOutPoint() const
288 : : {
289 : 14 : return COutPoint(prev_txid, prev_out);
290 : : }
291 : :
292 : 0 : bool PSBTInput::IsNull() const
293 : : {
294 [ # # # # : 0 : return !non_witness_utxo && witness_utxo.IsNull() && partial_sigs.empty() && unknown.empty() && hd_keypaths.empty() && redeem_script.empty() && witness_script.empty();
# # # # #
# # # # #
# # # # ]
295 : : }
296 : :
297 : 6 : void PSBTInput::FillSignatureData(SignatureData& sigdata) const
298 : : {
299 [ - + - + ]: 6 : if (!final_script_sig.empty()) {
300 : 0 : sigdata.scriptSig = final_script_sig;
301 : 0 : sigdata.complete = true;
302 : : }
303 [ - + ]: 6 : if (!final_script_witness.IsNull()) {
304 : 0 : sigdata.scriptWitness = final_script_witness;
305 : 0 : sigdata.complete = true;
306 : : }
307 [ + - ]: 6 : if (sigdata.complete) {
308 : : return;
309 : : }
310 : :
311 : 6 : sigdata.signatures.insert(partial_sigs.begin(), partial_sigs.end());
312 [ + + + + ]: 7 : if (!redeem_script.empty()) {
313 : 1 : sigdata.redeem_script = redeem_script;
314 : : }
315 [ - + - + ]: 6 : if (!witness_script.empty()) {
316 : 0 : sigdata.witness_script = witness_script;
317 : : }
318 [ + + ]: 8 : for (const auto& key_pair : hd_keypaths) {
319 : 2 : sigdata.misc_pubkeys.emplace(key_pair.first.GetID(), key_pair);
320 : : }
321 [ - + ]: 6 : if (!m_tap_key_sig.empty()) {
322 : 0 : sigdata.taproot_key_path_sig = m_tap_key_sig;
323 : : }
324 [ - + ]: 6 : for (const auto& [pubkey_leaf, sig] : m_tap_script_sigs) {
325 : 0 : sigdata.taproot_script_sigs.emplace(pubkey_leaf, sig);
326 : : }
327 [ - + ]: 12 : if (!m_tap_internal_key.IsNull()) {
328 : 0 : sigdata.tr_spenddata.internal_key = m_tap_internal_key;
329 : : }
330 [ - + ]: 12 : if (!m_tap_merkle_root.IsNull()) {
331 : 0 : sigdata.tr_spenddata.merkle_root = m_tap_merkle_root;
332 : : }
333 [ - + ]: 6 : for (const auto& [leaf_script, control_block] : m_tap_scripts) {
334 : 0 : sigdata.tr_spenddata.scripts.emplace(leaf_script, control_block);
335 : : }
336 [ - + ]: 6 : for (const auto& [pubkey, leaf_origin] : m_tap_bip32_paths) {
337 : 0 : sigdata.taproot_misc_pubkeys.emplace(pubkey, leaf_origin);
338 : 0 : sigdata.tap_pubkeys.emplace(Hash160(pubkey), pubkey);
339 : : }
340 [ - + ]: 6 : for (const auto& [hash, preimage] : ripemd160_preimages) {
341 [ # # ]: 0 : sigdata.ripemd160_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
342 : : }
343 [ - + ]: 6 : for (const auto& [hash, preimage] : sha256_preimages) {
344 [ # # ]: 0 : sigdata.sha256_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
345 : : }
346 [ - + ]: 6 : for (const auto& [hash, preimage] : hash160_preimages) {
347 [ # # ]: 0 : sigdata.hash160_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
348 : : }
349 [ - + ]: 6 : for (const auto& [hash, preimage] : hash256_preimages) {
350 [ # # ]: 0 : sigdata.hash256_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
351 : : }
352 : 6 : sigdata.musig2_pubkeys.insert(m_musig2_participants.begin(), m_musig2_participants.end());
353 [ - + ]: 6 : for (const auto& [agg_key_lh, pubnonces] : m_musig2_pubnonces) {
354 : 0 : sigdata.musig2_pubnonces[agg_key_lh].insert(pubnonces.begin(), pubnonces.end());
355 : : }
356 [ - + ]: 6 : for (const auto& [agg_key_lh, psigs] : m_musig2_partial_sigs) {
357 : 0 : sigdata.musig2_partial_sigs[agg_key_lh].insert(psigs.begin(), psigs.end());
358 : : }
359 : : }
360 : :
361 : 6 : void PSBTInput::FromSignatureData(const SignatureData& sigdata)
362 : : {
363 [ - + ]: 6 : if (sigdata.complete) {
364 : 0 : partial_sigs.clear();
365 : 0 : hd_keypaths.clear();
366 : 0 : redeem_script.clear();
367 : 0 : witness_script.clear();
368 : :
369 [ # # # # ]: 0 : if (!sigdata.scriptSig.empty()) {
370 : 0 : final_script_sig = sigdata.scriptSig;
371 : : }
372 [ # # ]: 0 : if (!sigdata.scriptWitness.IsNull()) {
373 : 0 : final_script_witness = sigdata.scriptWitness;
374 : : }
375 : 0 : return;
376 : : }
377 : :
378 : 6 : partial_sigs.insert(sigdata.signatures.begin(), sigdata.signatures.end());
379 [ + + + + : 8 : if (redeem_script.empty() && !sigdata.redeem_script.empty()) {
+ + + + ]
380 : 2 : redeem_script = sigdata.redeem_script;
381 : : }
382 [ - + + - : 7 : if (witness_script.empty() && !sigdata.witness_script.empty()) {
+ + + + ]
383 : 1 : witness_script = sigdata.witness_script;
384 : : }
385 [ + + ]: 12 : for (const auto& entry : sigdata.misc_pubkeys) {
386 : 6 : hd_keypaths.emplace(entry.second);
387 : : }
388 [ - + ]: 6 : if (!sigdata.taproot_key_path_sig.empty()) {
389 : 0 : m_tap_key_sig = sigdata.taproot_key_path_sig;
390 : : }
391 [ - + ]: 6 : for (const auto& [pubkey_leaf, sig] : sigdata.taproot_script_sigs) {
392 : 0 : m_tap_script_sigs.emplace(pubkey_leaf, sig);
393 : : }
394 [ - + ]: 12 : if (!sigdata.tr_spenddata.internal_key.IsNull()) {
395 : 0 : m_tap_internal_key = sigdata.tr_spenddata.internal_key;
396 : : }
397 [ - + ]: 12 : if (!sigdata.tr_spenddata.merkle_root.IsNull()) {
398 : 0 : m_tap_merkle_root = sigdata.tr_spenddata.merkle_root;
399 : : }
400 [ - + ]: 6 : for (const auto& [leaf_script, control_block] : sigdata.tr_spenddata.scripts) {
401 : 0 : m_tap_scripts.emplace(leaf_script, control_block);
402 : : }
403 [ - + ]: 6 : for (const auto& [pubkey, leaf_origin] : sigdata.taproot_misc_pubkeys) {
404 : 0 : m_tap_bip32_paths.emplace(pubkey, leaf_origin);
405 : : }
406 : 6 : m_musig2_participants.insert(sigdata.musig2_pubkeys.begin(), sigdata.musig2_pubkeys.end());
407 [ - + ]: 6 : for (const auto& [agg_key_lh, pubnonces] : sigdata.musig2_pubnonces) {
408 : 0 : m_musig2_pubnonces[agg_key_lh].insert(pubnonces.begin(), pubnonces.end());
409 : : }
410 [ - + ]: 6 : for (const auto& [agg_key_lh, psigs] : sigdata.musig2_partial_sigs) {
411 : 0 : m_musig2_partial_sigs[agg_key_lh].insert(psigs.begin(), psigs.end());
412 : : }
413 [ - + ]: 6 : for (const auto& [hash, preimage] : sigdata.ripemd160_preimages) {
414 : 0 : ripemd160_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
415 : : }
416 [ - + ]: 6 : for (const auto& [hash, preimage] : sigdata.sha256_preimages) {
417 : 0 : sha256_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
418 : : }
419 [ - + ]: 6 : for (const auto& [hash, preimage] : sigdata.hash160_preimages) {
420 : 0 : hash160_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
421 : : }
422 [ - + ]: 6 : for (const auto& [hash, preimage] : sigdata.hash256_preimages) {
423 : 0 : hash256_preimages.emplace(std::vector<unsigned char>(hash.begin(), hash.end()), preimage);
424 : : }
425 : : }
426 : :
427 : 1 : bool PSBTInput::Merge(const PSBTInput& input)
428 : : {
429 [ + - - + ]: 1 : if (!non_witness_utxo && input.non_witness_utxo) non_witness_utxo = input.non_witness_utxo;
430 [ + - - + ]: 1 : if (witness_utxo.IsNull() && !input.witness_utxo.IsNull()) {
431 : 0 : witness_utxo = input.witness_utxo;
432 : : }
433 : :
434 : 1 : partial_sigs.insert(input.partial_sigs.begin(), input.partial_sigs.end());
435 : 1 : ripemd160_preimages.insert(input.ripemd160_preimages.begin(), input.ripemd160_preimages.end());
436 : 1 : sha256_preimages.insert(input.sha256_preimages.begin(), input.sha256_preimages.end());
437 : 1 : hash160_preimages.insert(input.hash160_preimages.begin(), input.hash160_preimages.end());
438 : 1 : hash256_preimages.insert(input.hash256_preimages.begin(), input.hash256_preimages.end());
439 : 1 : hd_keypaths.insert(input.hd_keypaths.begin(), input.hd_keypaths.end());
440 : 1 : m_proprietary.insert(input.m_proprietary.begin(), input.m_proprietary.end());
441 : 1 : unknown.insert(input.unknown.begin(), input.unknown.end());
442 : 1 : m_tap_script_sigs.insert(input.m_tap_script_sigs.begin(), input.m_tap_script_sigs.end());
443 : 1 : m_tap_scripts.insert(input.m_tap_scripts.begin(), input.m_tap_scripts.end());
444 : 1 : m_tap_bip32_paths.insert(input.m_tap_bip32_paths.begin(), input.m_tap_bip32_paths.end());
445 : :
446 [ - + + - : 1 : if (redeem_script.empty() && !input.redeem_script.empty()) redeem_script = input.redeem_script;
- + - + ]
447 [ - + + - : 1 : if (witness_script.empty() && !input.witness_script.empty()) witness_script = input.witness_script;
- + - + ]
448 [ - + + - : 1 : if (final_script_sig.empty() && !input.final_script_sig.empty()) final_script_sig = input.final_script_sig;
- + - + ]
449 [ + - - + ]: 1 : if (final_script_witness.IsNull() && !input.final_script_witness.IsNull()) final_script_witness = input.final_script_witness;
450 [ + - - + ]: 1 : if (m_tap_key_sig.empty() && !input.m_tap_key_sig.empty()) m_tap_key_sig = input.m_tap_key_sig;
451 [ + - - + ]: 3 : if (m_tap_internal_key.IsNull() && !input.m_tap_internal_key.IsNull()) m_tap_internal_key = input.m_tap_internal_key;
452 [ + - - + ]: 3 : if (m_tap_merkle_root.IsNull() && !input.m_tap_merkle_root.IsNull()) m_tap_merkle_root = input.m_tap_merkle_root;
453 : 1 : m_musig2_participants.insert(input.m_musig2_participants.begin(), input.m_musig2_participants.end());
454 [ - + ]: 1 : for (const auto& [agg_key_lh, pubnonces] : input.m_musig2_pubnonces) {
455 : 0 : m_musig2_pubnonces[agg_key_lh].insert(pubnonces.begin(), pubnonces.end());
456 : : }
457 [ - + ]: 1 : for (const auto& [agg_key_lh, psigs] : input.m_musig2_partial_sigs) {
458 : 0 : m_musig2_partial_sigs[agg_key_lh].insert(psigs.begin(), psigs.end());
459 : : }
460 [ - + - - ]: 1 : if (sequence == std::nullopt && input.sequence != std::nullopt) sequence = input.sequence;
461 [ + - - + ]: 1 : if (time_locktime == std::nullopt && input.time_locktime != std::nullopt) time_locktime = input.time_locktime;
462 [ + - - + ]: 1 : if (height_locktime == std::nullopt && input.height_locktime != std::nullopt) height_locktime = input.height_locktime;
463 : :
464 : 1 : return true;
465 : : }
466 : :
467 : 13 : bool PSBTInput::HasSignatures() const
468 : : {
469 [ - + ]: 13 : return !final_script_sig.empty()
470 [ + - ]: 12 : || !final_script_witness.IsNull()
471 [ + - ]: 12 : || !partial_sigs.empty()
472 [ + - ]: 12 : || !m_tap_key_sig.empty()
473 [ + - ]: 12 : || !m_tap_script_sigs.empty()
474 [ + + - + ]: 25 : || !m_musig2_partial_sigs.empty();
475 : : }
476 : :
477 : 2 : void PSBTOutput::FillSignatureData(SignatureData& sigdata) const
478 : : {
479 [ - + - + ]: 2 : if (!redeem_script.empty()) {
480 : 0 : sigdata.redeem_script = redeem_script;
481 : : }
482 [ - + - + ]: 2 : if (!witness_script.empty()) {
483 : 0 : sigdata.witness_script = witness_script;
484 : : }
485 [ - + ]: 2 : for (const auto& key_pair : hd_keypaths) {
486 : 0 : sigdata.misc_pubkeys.emplace(key_pair.first.GetID(), key_pair);
487 : : }
488 [ - + - - ]: 2 : if (!m_tap_tree.empty() && m_tap_internal_key.IsFullyValid()) {
489 : 0 : TaprootBuilder builder;
490 [ # # # # ]: 0 : for (const auto& [depth, leaf_ver, script] : m_tap_tree) {
491 [ # # # # ]: 0 : builder.Add((int)depth, script, (int)leaf_ver, /*track=*/true);
492 : : }
493 [ # # ]: 0 : assert(builder.IsComplete());
494 [ # # ]: 0 : builder.Finalize(m_tap_internal_key);
495 [ # # ]: 0 : TaprootSpendData spenddata = builder.GetSpendData();
496 : :
497 : 0 : sigdata.tr_spenddata.internal_key = m_tap_internal_key;
498 [ # # # # ]: 0 : sigdata.tr_spenddata.Merge(spenddata);
499 [ # # ]: 0 : sigdata.tr_builder = builder;
500 : 0 : }
501 [ - + ]: 2 : for (const auto& [pubkey, leaf_origin] : m_tap_bip32_paths) {
502 : 0 : sigdata.taproot_misc_pubkeys.emplace(pubkey, leaf_origin);
503 : 0 : sigdata.tap_pubkeys.emplace(Hash160(pubkey), pubkey);
504 : : }
505 : 2 : sigdata.musig2_pubkeys.insert(m_musig2_participants.begin(), m_musig2_participants.end());
506 : 2 : }
507 : :
508 : 2 : void PSBTOutput::FromSignatureData(const SignatureData& sigdata)
509 : : {
510 [ - + + - : 2 : if (redeem_script.empty() && !sigdata.redeem_script.empty()) {
- + - + ]
511 : 0 : redeem_script = sigdata.redeem_script;
512 : : }
513 [ - + + - : 2 : if (witness_script.empty() && !sigdata.witness_script.empty()) {
- + - + ]
514 : 0 : witness_script = sigdata.witness_script;
515 : : }
516 [ + + ]: 4 : for (const auto& entry : sigdata.misc_pubkeys) {
517 : 2 : hd_keypaths.emplace(entry.second);
518 : : }
519 [ - + ]: 4 : if (!sigdata.tr_spenddata.internal_key.IsNull()) {
520 : 0 : m_tap_internal_key = sigdata.tr_spenddata.internal_key;
521 : : }
522 [ - + - - ]: 2 : if (sigdata.tr_builder.has_value() && sigdata.tr_builder->HasScripts()) {
523 : 0 : m_tap_tree = sigdata.tr_builder->GetTreeTuples();
524 : : }
525 [ - + ]: 2 : for (const auto& [pubkey, leaf_origin] : sigdata.taproot_misc_pubkeys) {
526 : 0 : m_tap_bip32_paths.emplace(pubkey, leaf_origin);
527 : : }
528 : 2 : m_musig2_participants.insert(sigdata.musig2_pubkeys.begin(), sigdata.musig2_pubkeys.end());
529 : 2 : }
530 : :
531 : 0 : bool PSBTOutput::IsNull() const
532 : : {
533 [ # # # # : 0 : return redeem_script.empty() && witness_script.empty() && hd_keypaths.empty() && unknown.empty();
# # # # #
# # # ]
534 : : }
535 : :
536 : 1 : bool PSBTOutput::Merge(const PSBTOutput& output)
537 : : {
538 : 1 : hd_keypaths.insert(output.hd_keypaths.begin(), output.hd_keypaths.end());
539 : 1 : m_proprietary.insert(output.m_proprietary.begin(), output.m_proprietary.end());
540 : 1 : unknown.insert(output.unknown.begin(), output.unknown.end());
541 : 1 : m_tap_bip32_paths.insert(output.m_tap_bip32_paths.begin(), output.m_tap_bip32_paths.end());
542 : :
543 [ - + + - : 1 : if (redeem_script.empty() && !output.redeem_script.empty()) redeem_script = output.redeem_script;
- + - + ]
544 [ - + + - : 1 : if (witness_script.empty() && !output.witness_script.empty()) witness_script = output.witness_script;
- + - + ]
545 [ + - - + ]: 3 : if (m_tap_internal_key.IsNull() && !output.m_tap_internal_key.IsNull()) m_tap_internal_key = output.m_tap_internal_key;
546 [ + - - + ]: 1 : if (m_tap_tree.empty() && !output.m_tap_tree.empty()) m_tap_tree = output.m_tap_tree;
547 : 1 : m_musig2_participants.insert(output.m_musig2_participants.begin(), output.m_musig2_participants.end());
548 : :
549 : 1 : return true;
550 : : }
551 : :
552 : 17 : bool PSBTInputSigned(const PSBTInput& input)
553 : : {
554 [ - + + - : 17 : return !input.final_script_sig.empty() || !input.final_script_witness.IsNull();
- + ]
555 : : }
556 : :
557 : 8 : bool PSBTInputSignedAndVerified(const PartiallySignedTransaction& psbt, unsigned int input_index, const PrecomputedTransactionData* txdata)
558 : : {
559 : 8 : CTxOut utxo;
560 [ - + - + ]: 8 : assert(input_index < psbt.inputs.size());
561 [ + - ]: 8 : const PSBTInput& input = psbt.inputs[input_index];
562 : :
563 [ + - ]: 8 : if (input.non_witness_utxo) {
564 : : // If we're taking our information from a non-witness UTXO, verify that it matches the prevout.
565 [ + - ]: 8 : COutPoint prevout = input.GetOutPoint();
566 [ - + + - ]: 8 : if (prevout.n >= input.non_witness_utxo->vout.size()) {
567 : : return false;
568 : : }
569 [ + - ]: 8 : if (input.non_witness_utxo->GetHash() != prevout.hash) {
570 : : return false;
571 : : }
572 : 8 : utxo = input.non_witness_utxo->vout[prevout.n];
573 [ # # ]: 0 : } else if (!input.witness_utxo.IsNull()) {
574 : 0 : utxo = input.witness_utxo;
575 : : } else {
576 : : return false;
577 : : }
578 : :
579 [ + - ]: 8 : std::optional<CMutableTransaction> unsigned_tx = psbt.GetUnsignedTx();
580 [ + - ]: 8 : if (!unsigned_tx) {
581 : : return false;
582 : : }
583 [ + - ]: 8 : const CMutableTransaction& tx = *unsigned_tx;
584 [ + - ]: 8 : if (txdata) {
585 [ + - ]: 8 : return VerifyScript(input.final_script_sig, utxo.scriptPubKey, &input.final_script_witness, STANDARD_SCRIPT_VERIFY_FLAGS, MutableTransactionSignatureChecker{&tx, input_index, utxo.nValue, *txdata, MissingDataBehavior::FAIL});
586 : : } else {
587 [ # # ]: 0 : return VerifyScript(input.final_script_sig, utxo.scriptPubKey, &input.final_script_witness, STANDARD_SCRIPT_VERIFY_FLAGS, MutableTransactionSignatureChecker{&tx, input_index, utxo.nValue, MissingDataBehavior::FAIL});
588 : : }
589 : 16 : }
590 : :
591 : 0 : size_t CountPSBTUnsignedInputs(const PartiallySignedTransaction& psbt) {
592 : 0 : size_t count = 0;
593 [ # # ]: 0 : for (const auto& input : psbt.inputs) {
594 [ # # ]: 0 : if (!PSBTInputSigned(input)) {
595 : 0 : count++;
596 : : }
597 : : }
598 : :
599 : 0 : return count;
600 : : }
601 : :
602 : 2 : void UpdatePSBTOutput(const SigningProvider& provider, PartiallySignedTransaction& psbt, int index)
603 : : {
604 : 2 : std::optional<CMutableTransaction> unsigned_tx = psbt.GetUnsignedTx();
605 [ - + ]: 2 : if (!unsigned_tx) {
606 : 0 : return;
607 : : }
608 [ + - ]: 2 : CMutableTransaction& tx = *unsigned_tx;
609 [ + - ]: 2 : const CTxOut& out = tx.vout.at(index);
610 [ + - ]: 2 : PSBTOutput& psbt_out = psbt.outputs.at(index);
611 : :
612 : : // Fill a SignatureData with output info
613 : 2 : SignatureData sigdata;
614 [ + - ]: 2 : psbt_out.FillSignatureData(sigdata);
615 : :
616 : : // Construct a would-be spend of this output, to update sigdata with.
617 : : // Note that ProduceSignature is used to fill in metadata (not actual signatures),
618 : : // so provider does not need to provide any private keys (it can be a HidingSigningProvider).
619 [ + - ]: 2 : MutableTransactionSignatureCreator creator(tx, /*input_idx=*/0, out.nValue, {.sighash_type = SIGHASH_ALL});
620 [ + - ]: 2 : ProduceSignature(provider, creator, out.scriptPubKey, sigdata);
621 : :
622 : : // Put redeem_script, witness_script, key paths, into PSBTOutput.
623 [ + - ]: 2 : psbt_out.FromSignatureData(sigdata);
624 [ + - ]: 4 : }
625 : :
626 : 2 : std::optional<PrecomputedTransactionData> PrecomputePSBTData(const PartiallySignedTransaction& psbt)
627 : : {
628 : 2 : std::optional<CMutableTransaction> unsigned_tx = psbt.GetUnsignedTx();
629 [ - + ]: 2 : if (!unsigned_tx) {
630 : 0 : return std::nullopt;
631 : : }
632 : 2 : const CMutableTransaction& tx = *unsigned_tx;
633 : 2 : bool have_all_spent_outputs = true;
634 : 2 : std::vector<CTxOut> utxos;
635 [ + + ]: 6 : for (const PSBTInput& input : psbt.inputs) {
636 [ + - + - : 4 : if (!input.GetUTXO(utxos.emplace_back())) have_all_spent_outputs = false;
+ + ]
637 : : }
638 : 2 : PrecomputedTransactionData txdata;
639 [ + + ]: 2 : if (have_all_spent_outputs) {
640 [ + - ]: 1 : txdata.Init(tx, std::move(utxos), true);
641 : : } else {
642 [ + - ]: 1 : txdata.Init(tx, {}, true);
643 : : }
644 : 2 : return txdata;
645 : 4 : }
646 : :
647 : 6 : PSBTError SignPSBTInput(const SigningProvider& provider, PartiallySignedTransaction& psbt, int index, const PrecomputedTransactionData* txdata, const common::PSBTFillOptions& options, SignatureData* out_sigdata)
648 : : {
649 : 6 : PSBTInput& input = psbt.inputs.at(index);
650 : 6 : std::optional<CMutableTransaction> unsigned_tx = psbt.GetUnsignedTx();
651 [ + - ]: 6 : if (!unsigned_tx) {
652 : : return PSBTError::INVALID_TX;
653 : : }
654 [ + - ]: 6 : const CMutableTransaction& tx = *unsigned_tx;
655 : :
656 [ + - + - ]: 6 : if (PSBTInputSignedAndVerified(psbt, index, txdata)) {
657 : : return PSBTError::OK;
658 : : }
659 : :
660 : : // Fill SignatureData with input info
661 : 6 : SignatureData sigdata;
662 [ + - ]: 6 : input.FillSignatureData(sigdata);
663 : :
664 : : // Get UTXO
665 : 6 : bool require_witness_sig = false;
666 : 6 : CTxOut utxo;
667 : :
668 [ + - ]: 6 : if (input.non_witness_utxo) {
669 : : // If we're taking our information from a non-witness UTXO, verify that it matches the prevout.
670 [ + - ]: 6 : COutPoint prevout = input.GetOutPoint();
671 [ - + + - ]: 6 : if (prevout.n >= input.non_witness_utxo->vout.size()) {
672 : : return PSBTError::MISSING_INPUTS;
673 : : }
674 [ + - ]: 6 : if (input.non_witness_utxo->GetHash() != prevout.hash) {
675 : : return PSBTError::MISSING_INPUTS;
676 : : }
677 : 6 : utxo = input.non_witness_utxo->vout[prevout.n];
678 [ # # ]: 0 : } else if (!input.witness_utxo.IsNull()) {
679 : 0 : utxo = input.witness_utxo;
680 : : // When we're taking our information from a witness UTXO, we can't verify it is actually data from
681 : : // the output being spent. This is safe in case a witness signature is produced (which includes this
682 : : // information directly in the hash), but not for non-witness signatures. Remember that we require
683 : : // a witness signature in this situation.
684 : 0 : require_witness_sig = true;
685 : : } else {
686 : : return PSBTError::MISSING_INPUTS;
687 : : }
688 : :
689 : : // Get the sighash type
690 : : // If both the field and the parameter are provided, they must match
691 : : // If only the parameter is provided, use it and add it to the PSBT if it is other than SIGHASH_DEFAULT
692 : : // for all input types, and not SIGHASH_ALL for non-taproot input types.
693 : : // If neither are provided, use SIGHASH_DEFAULT if it is taproot, and SIGHASH_ALL for everything else.
694 [ + - + - : 12 : int sighash{options.sighash_type.value_or(utxo.scriptPubKey.IsPayToTaproot() ? SIGHASH_DEFAULT : SIGHASH_ALL)};
- + ]
695 : :
696 : : // For user safety, the desired sighash must be provided if the PSBT wants something other than the default set in the previous line.
697 [ - + - - ]: 12 : if (input.sighash_type && input.sighash_type != sighash) {
698 : : return PSBTError::SIGHASH_MISMATCH;
699 : : }
700 : : // Set the PSBT sighash field when sighash is not DEFAULT or ALL
701 : : // DEFAULT is allowed for non-taproot inputs since DEFAULT may be passed for them (e.g. the psbt being signed also has taproot inputs)
702 : : // Note that signing already aliases DEFAULT to ALL for non-taproot inputs.
703 [ + - - + : 12 : if (utxo.scriptPubKey.IsPayToTaproot() ? sighash != SIGHASH_DEFAULT :
- + ]
704 : 6 : (sighash != SIGHASH_DEFAULT && sighash != SIGHASH_ALL)) {
705 : 0 : input.sighash_type = sighash;
706 : : }
707 : :
708 : : // Check all existing signatures use the sighash type
709 [ - + ]: 6 : if (sighash == SIGHASH_DEFAULT) {
710 [ # # # # : 0 : if (!input.m_tap_key_sig.empty() && input.m_tap_key_sig.size() != 64) {
# # ]
711 : : return PSBTError::SIGHASH_MISMATCH;
712 : : }
713 [ # # # # ]: 0 : for (const auto& [_, sig] : input.m_tap_script_sigs) {
714 [ # # # # ]: 0 : if (sig.size() != 64) return PSBTError::SIGHASH_MISMATCH;
715 : : }
716 : : } else {
717 [ - + - - : 6 : if (!input.m_tap_key_sig.empty() && (input.m_tap_key_sig.size() != 65 || input.m_tap_key_sig.back() != sighash)) {
- - - - ]
718 : : return PSBTError::SIGHASH_MISMATCH;
719 : : }
720 [ - - - + ]: 6 : for (const auto& [_, sig] : input.m_tap_script_sigs) {
721 [ # # # # : 0 : if (sig.size() != 65 || sig.back() != sighash) return PSBTError::SIGHASH_MISMATCH;
# # ]
722 : : }
723 [ - - - + ]: 6 : for (const auto& [_, sig] : input.partial_sigs) {
724 [ # # ]: 0 : if (sig.second.back() != sighash) return PSBTError::SIGHASH_MISMATCH;
725 : : }
726 : : }
727 : :
728 : 6 : sigdata.witness = false;
729 : 6 : bool sig_complete;
730 [ - + ]: 6 : if (txdata == nullptr) {
731 [ # # ]: 0 : sig_complete = ProduceSignature(provider, DUMMY_SIGNATURE_CREATOR, utxo.scriptPubKey, sigdata);
732 : : } else {
733 [ + - ]: 6 : MutableTransactionSignatureCreator creator(tx, index, utxo.nValue, txdata, {.sighash_type = sighash});
734 [ + - ]: 6 : sig_complete = ProduceSignature(provider, creator, utxo.scriptPubKey, sigdata);
735 : 6 : }
736 : : // Verify that a witness signature was produced in case one was required.
737 [ - + - - ]: 6 : if (require_witness_sig && !sigdata.witness) return PSBTError::INCOMPLETE;
738 : :
739 : : // If we are not finalizing, set sigdata.complete to false to not set the scriptWitness
740 [ - + - - ]: 6 : if (!options.finalize && sigdata.complete) sigdata.complete = false;
741 : :
742 [ + - ]: 6 : input.FromSignatureData(sigdata);
743 : :
744 : : // If we have a witness signature, put a witness UTXO.
745 [ + + ]: 6 : if (sigdata.witness) {
746 : 1 : input.witness_utxo = utxo;
747 : : // We can remove the non_witness_utxo if and only if there are no non-segwit or segwit v0
748 : : // inputs in this transaction. Since this requires inspecting the entire transaction, this
749 : : // is something for the caller to deal with (i.e. FillPSBT).
750 : : }
751 : :
752 : : // Fill in the missing info
753 [ - + ]: 6 : if (out_sigdata) {
754 [ # # ]: 0 : out_sigdata->missing_pubkeys = sigdata.missing_pubkeys;
755 [ # # ]: 0 : out_sigdata->missing_sigs = sigdata.missing_sigs;
756 : 0 : out_sigdata->missing_redeem_script = sigdata.missing_redeem_script;
757 : 0 : out_sigdata->missing_witness_script = sigdata.missing_witness_script;
758 : : }
759 : :
760 [ + - ]: 6 : return sig_complete ? PSBTError::OK : PSBTError::INCOMPLETE;
761 : 12 : }
762 : :
763 : 1 : void RemoveUnnecessaryTransactions(PartiallySignedTransaction& psbtx)
764 : : {
765 : : // Figure out if any non_witness_utxos should be dropped
766 : 1 : std::vector<unsigned int> to_drop;
767 [ - + + - ]: 1 : for (unsigned int i = 0; i < psbtx.inputs.size(); ++i) {
768 [ + - ]: 1 : const auto& input = psbtx.inputs.at(i);
769 : 1 : int wit_ver;
770 : 1 : std::vector<unsigned char> wit_prog;
771 [ - + - - : 1 : if (input.witness_utxo.IsNull() || !input.witness_utxo.scriptPubKey.IsWitnessProgram(wit_ver, wit_prog)) {
- - ]
772 : : // There's a non-segwit input, so we cannot drop any non_witness_utxos
773 [ - + ]: 1 : to_drop.clear();
774 : : break;
775 : : }
776 [ # # ]: 0 : if (wit_ver == 0) {
777 : : // Segwit v0, so we cannot drop any non_witness_utxos
778 [ # # ]: 0 : to_drop.clear();
779 : : break;
780 : : }
781 : : // non_witness_utxos cannot be dropped if the sighash type includes SIGHASH_ANYONECANPAY
782 : : // Since callers should have called SignPSBTInput which updates the sighash type in the PSBT, we only
783 : : // need to look at that field. If it is not present, then we can assume SIGHASH_DEFAULT or SIGHASH_ALL.
784 [ # # # # ]: 0 : if (input.sighash_type != std::nullopt && (*input.sighash_type & 0x80) == SIGHASH_ANYONECANPAY) {
785 [ - - ]: 1 : to_drop.clear();
786 : : break;
787 : : }
788 : :
789 [ # # ]: 0 : if (input.non_witness_utxo) {
790 [ # # ]: 0 : to_drop.push_back(i);
791 : : }
792 : 1 : }
793 : :
794 : : // Drop the non_witness_utxos that we can drop
795 [ - + ]: 1 : for (unsigned int i : to_drop) {
796 [ # # # # ]: 0 : psbtx.inputs.at(i).non_witness_utxo = nullptr;
797 : : }
798 : 1 : }
799 : :
800 : 0 : bool FinalizePSBT(PartiallySignedTransaction& psbtx)
801 : : {
802 : : // Finalize input signatures -- in case we have partial signatures that add up to a complete
803 : : // signature, but have not combined them yet (e.g. because the combiner that created this
804 : : // PartiallySignedTransaction did not understand them), this will combine them into a final
805 : : // script.
806 : 0 : bool complete = true;
807 : 0 : std::optional<PrecomputedTransactionData> txdata_res = PrecomputePSBTData(psbtx);
808 [ # # ]: 0 : if (!txdata_res) {
809 : : return false;
810 : : }
811 : : const PrecomputedTransactionData& txdata = *txdata_res;
812 [ # # # # ]: 0 : for (unsigned int i = 0; i < psbtx.inputs.size(); ++i) {
813 [ # # ]: 0 : PSBTInput& input = psbtx.inputs.at(i);
814 [ # # ]: 0 : complete &= (SignPSBTInput(DUMMY_SIGNING_PROVIDER, psbtx, i, &txdata, {.sighash_type = input.sighash_type, .finalize = true}, /*out_sigdata=*/nullptr) == PSBTError::OK);
815 : : }
816 : :
817 : : return complete;
818 : 0 : }
819 : :
820 : 0 : bool FinalizeAndExtractPSBT(PartiallySignedTransaction& psbtx, CMutableTransaction& result)
821 : : {
822 : : // It's not safe to extract a PSBT that isn't finalized, and there's no easy way to check
823 : : // whether a PSBT is finalized without finalizing it, so we just do this.
824 [ # # ]: 0 : if (!FinalizePSBT(psbtx)) {
825 : : return false;
826 : : }
827 : :
828 : 0 : std::optional<CMutableTransaction> unsigned_tx = psbtx.GetUnsignedTx();
829 [ # # ]: 0 : if (!unsigned_tx) {
830 : : return false;
831 : : }
832 [ # # ]: 0 : result = *unsigned_tx;
833 [ # # # # ]: 0 : for (unsigned int i = 0; i < result.vin.size(); ++i) {
834 : 0 : result.vin[i].scriptSig = psbtx.inputs[i].final_script_sig;
835 [ # # ]: 0 : result.vin[i].scriptWitness = psbtx.inputs[i].final_script_witness;
836 : : }
837 : : return true;
838 : 0 : }
839 : :
840 : 0 : std::optional<PartiallySignedTransaction> CombinePSBTs(const std::vector<PartiallySignedTransaction>& psbtxs)
841 : : {
842 : 0 : PartiallySignedTransaction out = psbtxs[0]; // Copy the first one
843 : :
844 : : // Merge
845 [ # # ]: 0 : for (auto it = std::next(psbtxs.begin()); it != psbtxs.end(); ++it) {
846 [ # # # # ]: 0 : if (!out.Merge(*it)) {
847 : 0 : return std::nullopt;
848 : : }
849 : : }
850 : 0 : return out;
851 : 0 : }
852 : :
853 : 0 : std::string PSBTRoleName(PSBTRole role) {
854 [ # # # # : 0 : switch (role) {
# # ]
855 : 0 : case PSBTRole::CREATOR: return "creator";
856 : 0 : case PSBTRole::UPDATER: return "updater";
857 : 0 : case PSBTRole::SIGNER: return "signer";
858 : 0 : case PSBTRole::FINALIZER: return "finalizer";
859 : 0 : case PSBTRole::EXTRACTOR: return "extractor";
860 : : } // no default case, so the compiler can warn about missing cases
861 : 0 : assert(false);
862 : : }
863 : :
864 : 10 : util::Result<PartiallySignedTransaction> DecodeBase64PSBT(const std::string& base64_tx)
865 : : {
866 [ - + ]: 10 : auto tx_data = DecodeBase64(base64_tx);
867 [ - + ]: 10 : if (!tx_data) {
868 [ # # # # ]: 0 : return util::Error{Untranslated("invalid base64")};
869 : : }
870 [ + - ]: 10 : return DecodeRawPSBT(MakeByteSpan(*tx_data));
871 : 10 : }
872 : :
873 : 10 : util::Result<PartiallySignedTransaction> DecodeRawPSBT(std::span<const std::byte> tx_data)
874 : : {
875 [ + - ]: 10 : SpanReader ss_data{tx_data};
876 : 10 : try {
877 [ + - ]: 10 : PartiallySignedTransaction psbt(deserialize, ss_data);
878 [ - + ]: 10 : if (!ss_data.empty()) {
879 [ # # # # ]: 0 : return util::Error{Untranslated("extra data after PSBT")};
880 : : }
881 : 10 : return psbt;
882 [ - - ]: 10 : } catch (const std::exception& e) {
883 [ - - - - ]: 0 : return util::Error{Untranslated(e.what())};
884 : 0 : }
885 : : }
886 : :
887 : 97 : uint32_t PartiallySignedTransaction::GetVersion() const
888 : : {
889 [ + + ]: 97 : if (m_version != std::nullopt) {
890 : 74 : return *m_version;
891 : : }
892 : : return 0;
893 : : }
|