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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
use snarkos_errors::objects::TransactionError;
use snarkos_models::objects::Transaction;
use snarkos_utilities::{
bytes::{FromBytes, ToBytes},
has_duplicates,
to_bytes,
variable_length_integer::{read_variable_length_integer, variable_length_integer},
};
use std::{
io::{Read, Result as IoResult, Write},
ops::{Deref, DerefMut},
};
#[derive(Debug, Clone, Eq, PartialEq)]
pub struct DPCTransactions<T: Transaction>(pub Vec<T>);
impl<T: Transaction> DPCTransactions<T> {
pub fn new() -> Self {
Self(vec![])
}
pub fn from(transactions: &[T]) -> Self {
Self(transactions.to_vec())
}
pub fn push(&mut self, transaction: T) {
self.0.push(transaction);
}
pub fn to_transaction_ids(&self) -> Result<Vec<Vec<u8>>, TransactionError> {
self.0
.iter()
.map(|transaction| -> Result<Vec<u8>, TransactionError> {
transaction.transaction_id().map(|tx_id| tx_id.to_vec())
})
.collect::<Result<Vec<Vec<u8>>, TransactionError>>()
}
pub fn serialize(&self) -> Result<Vec<Vec<u8>>, TransactionError> {
self.0
.iter()
.map(|transaction| -> Result<Vec<u8>, TransactionError> { Ok(to_bytes![transaction]?) })
.collect::<Result<Vec<Vec<u8>>, TransactionError>>()
}
pub fn serialize_as_str(&self) -> Result<Vec<String>, TransactionError> {
self.0
.iter()
.map(|transaction| -> Result<String, TransactionError> { Ok(hex::encode(to_bytes![transaction]?)) })
.collect::<Result<Vec<String>, TransactionError>>()
}
pub fn conflicts(&self, transaction: &T) -> bool {
let mut holding_serial_numbers = vec![];
let mut holding_commitments = vec![];
let mut holding_memos = vec![];
for tx in &self.0 {
if tx.network_id() != transaction.network_id() {
return true;
};
holding_serial_numbers.extend(tx.old_serial_numbers());
holding_commitments.extend(tx.new_commitments());
holding_memos.push(tx.memorandum());
}
let transaction_serial_numbers = transaction.old_serial_numbers();
let transaction_commitments = transaction.new_commitments();
let transaction_memo = transaction.memorandum();
if has_duplicates(transaction_serial_numbers) {
return true;
}
if has_duplicates(transaction_commitments) {
return true;
}
if holding_memos.contains(&transaction_memo) {
return true;
}
for sn in transaction_serial_numbers {
if holding_serial_numbers.contains(&sn) {
return true;
}
}
for cm in transaction_commitments {
if holding_commitments.contains(&cm) {
return true;
}
}
false
}
}
impl<T: Transaction> ToBytes for DPCTransactions<T> {
#[inline]
fn write<W: Write>(&self, mut writer: W) -> IoResult<()> {
variable_length_integer(self.0.len() as u64).write(&mut writer)?;
for transaction in &self.0 {
transaction.write(&mut writer)?;
}
Ok(())
}
}
impl<T: Transaction> FromBytes for DPCTransactions<T> {
#[inline]
fn read<R: Read>(mut reader: R) -> IoResult<Self> {
let num_transactions = read_variable_length_integer(&mut reader)?;
let mut transactions = vec![];
for _ in 0..num_transactions {
let transaction: T = FromBytes::read(&mut reader)?;
transactions.push(transaction);
}
Ok(Self(transactions))
}
}
impl<T: Transaction> Default for DPCTransactions<T> {
fn default() -> Self {
Self::new()
}
}
impl<T: Transaction> Deref for DPCTransactions<T> {
type Target = Vec<T>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl<T: Transaction> DerefMut for DPCTransactions<T> {
fn deref_mut(&mut self) -> &mut Vec<T> {
&mut self.0
}
}