-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathmerge_signed.rs
90 lines (80 loc) · 3.04 KB
/
merge_signed.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
use transaction::{Transaction, TxIn, TxOut};
use hash::Hashable;
use util;
/**/
fn match_input(in1: &TxIn, in2: &TxIn) -> bool
{
/* We don't check scriptSig since that will be different
* for different transactions. */
in1.prev_hash == in2.prev_hash &&
in1.prev_index == in2.prev_index &&
in1.nSequence == in2.nSequence
}
fn match_output(in1: &TxOut, in2: &TxOut) -> bool
{
in1.nValue == in2.nValue &&
in1.scriptPubKey == in2.scriptPubKey
}
/**
* Merge signed transactions
* This function verifies that all the transactions are the same modulo
* signatures, then combines them all into one big transaction that has
* all the available signatures in place.
*/
pub fn merge_signed_transactions (txlist: &[Transaction]) -> Option<Transaction>
{
if txlist.len() == 0 { return None; }
/* The first transaction will be our ``master'' list of inputs and outputs.
* Every other transaction needs to match this or else that's a failure.
*/
let mut master = txlist[0].clone();
/* Rust needs me to copy the master transaction's hash here, since
* I later mutate the master (by adding signatures) and it fears this
* will somehow bojangle the call to .to_hash(). */
let master_hash = util::u8_to_hex_string (master.to_hash());
/* Loop through all transactions, merging onto master */
for tx in txlist.iter() {
/* Check that version and locktime at least match */
if tx.nVersion != master.nVersion {
println (format! ("err: Tx {:s} did not match {:s} (version {:u} vs {:u})!",
util::u8_to_hex_string (master.to_hash()),
util::u8_to_hex_string (tx.to_hash()),
master.nVersion, tx.nVersion));
return None;
}
if tx.nLockTime != master.nLockTime {
println (format! ("err: Tx {:s} did not match {:s} (locktime {:u} vs {:u})!",
util::u8_to_hex_string (master.to_hash()),
util::u8_to_hex_string (tx.to_hash()),
master.nLockTime, tx.nLockTime));
return None;
}
/* Check that outputs match */
for (tx1, tx2) in tx.output.iter().zip(master.output.iter()) {
if !match_output (tx1, tx2) {
println (format! ("err: Tx {:s} did not match {:s} (output {:s}:{:u} vs {:s}:{:u})!",
master_hash,
util::u8_to_hex_string (tx.to_hash()),
util::u8_to_hex_string (tx1.scriptPubKey), tx1.nValue,
util::u8_to_hex_string (tx2.scriptPubKey), tx2.nValue));
return None;
}
}
/* Check that inputs match -- if they do, and a signature exists, take it */
for (tx1, tx2) in tx.input.iter().zip(master.input.mut_iter()) {
if match_input (tx1, tx2) {
if tx1.scriptSig.len() > 0 {
tx2.scriptSig = tx1.scriptSig.clone();
}
} else {
println (format! ("err: Tx {:s} did not match {:s} (input {:s}:{:u} vs {:s}:{:u})!",
master_hash,
util::u8_to_hex_string (tx.to_hash()),
util::u8_to_hex_string (tx1.prev_hash), tx1.prev_index,
util::u8_to_hex_string (tx2.prev_hash), tx2.prev_index));
return None;
}
}
}
Some(master)
}