Use the SHA-256d HashReader and HashWriter for transaction hashing.

This hashes the transaction as it is read, rather than hashing a re-serialization.

Co-authored-by: Jack Grigg <str4d@electriccoin.co>
Signed-off-by: Daira Hopwood <daira@jacaranda.org>
This commit is contained in:
Daira Hopwood 2020-11-16 19:08:00 +00:00
parent 1f18da239f
commit 62675f4b4f
1 changed files with 35 additions and 25 deletions

View File

@ -1,12 +1,16 @@
//! Structs and methods for handling Zcash transactions.
use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt};
use sha2::{Digest, Sha256};
use std::fmt;
use std::io::{self, Read, Write};
use std::ops::Deref;
use crate::{consensus::BlockHeight, redjubjub::Signature, serialize::Vector};
use crate::{
consensus::BlockHeight,
redjubjub::Signature,
serialize::Vector,
util::sha256d::{HashReader, HashWriter},
};
pub mod builder;
pub mod components;
@ -194,11 +198,9 @@ impl Transaction {
txid: TxId([0; 32]),
data,
};
let mut raw = vec![];
tx.write(&mut raw)?;
tx.txid
.0
.copy_from_slice(&Sha256::digest(&Sha256::digest(&raw)));
let mut writer = HashWriter::default();
tx.write(&mut writer)?;
tx.txid.0.copy_from_slice(&writer.into_hash());
Ok(tx)
}
@ -206,7 +208,9 @@ impl Transaction {
self.txid
}
pub fn read<R: Read>(mut reader: R) -> io::Result<Self> {
pub fn read<R: Read>(reader: R) -> io::Result<Self> {
let mut reader = HashReader::new(reader);
let header = reader.read_u32::<LittleEndian>()?;
let overwintered = (header >> 31) == 1;
let version = header & 0x7FFFFFFF;
@ -291,7 +295,12 @@ impl Transaction {
None
};
Transaction::from_data(TransactionData {
let mut txid = [0; 32];
txid.copy_from_slice(&reader.into_hash());
Ok(Transaction {
txid: TxId(txid),
data: TransactionData {
overwintered,
version,
version_group_id,
@ -308,6 +317,7 @@ impl Transaction {
joinsplit_pubkey,
joinsplit_sig,
binding_sig,
},
})
}