use std::borrow::BorrowMut;
use std::io::{self};
use byteorder::{LittleEndian, ReadBytesExt};
use crate::blockchain::proto::block::Block;
use crate::blockchain::proto::header::BlockHeader;
use crate::blockchain::proto::tx::{RawTx, TxInput, TxOutpoint, TxOutput};
use crate::blockchain::proto::varuint::VarUint;
use crate::errors::OpResult;
pub trait BlockchainRead: io::Read {
#[inline]
fn read_256hash(&mut self) -> OpResult<[u8; 32]> {
let mut arr = [0u8; 32];
self.read_exact(arr.borrow_mut())?;
Ok(arr)
}
#[inline]
fn read_u8_vec(&mut self, count: u32) -> OpResult<Vec<u8>> {
let mut arr = vec![0u8; count as usize];
self.read_exact(arr.borrow_mut())?;
Ok(arr)
}
fn read_block(&mut self, size: u32, version_id: u8) -> OpResult<Block> {
let header = self.read_block_header()?;
let tx_count = VarUint::read_from(self)?;
let txs = self.read_txs(tx_count.value, version_id)?;
Ok(Block::new(size, header, tx_count, txs))
}
#[inline]
fn read_block_header(&mut self) -> OpResult<BlockHeader> {
Ok(BlockHeader::new(
self.read_u32::<LittleEndian>()?,
self.read_256hash()?,
self.read_256hash()?,
self.read_u32::<LittleEndian>()?,
self.read_u32::<LittleEndian>()?,
self.read_u32::<LittleEndian>()?,
))
}
fn read_txs(&mut self, tx_count: u64, version_id: u8) -> OpResult<Vec<RawTx>> {
let mut txs = Vec::with_capacity(tx_count as usize);
for _ in 0..tx_count {
let mut flags = 0u8;
let version = self.read_u32::<LittleEndian>()?;
let mut in_count = VarUint::read_from(self)?;
if in_count.value == 0 {
flags = self.read_u8()?;
in_count = VarUint::read_from(self)?
}
let inputs = self.read_tx_inputs(in_count.value)?;
let out_count = VarUint::read_from(self)?;
let outputs = self.read_tx_outputs(out_count.value)?;
if flags & 1 > 0 {
for _ in 0..in_count.value {
let item_count = VarUint::read_from(self)?;
for _ in 0..item_count.value {
let witness_len = VarUint::read_from(self)?;
let _ = self.read_u8_vec(witness_len.value as u32)?;
}
}
}
let locktime = self.read_u32::<LittleEndian>()?;
txs.push(RawTx {
version,
in_count,
inputs,
out_count,
outputs,
locktime,
version_id,
});
}
Ok(txs)
}
#[inline]
fn read_tx_outpoint(&mut self) -> OpResult<TxOutpoint> {
Ok(TxOutpoint::new(
self.read_256hash()?,
self.read_u32::<LittleEndian>()?,
))
}
fn read_tx_inputs(&mut self, input_count: u64) -> OpResult<Vec<TxInput>> {
let mut inputs = Vec::with_capacity(input_count as usize);
for _ in 0..input_count {
let outpoint = self.read_tx_outpoint()?;
let script_len = VarUint::read_from(self)?;
let script_sig = self.read_u8_vec(script_len.value as u32)?;
let seq_no = self.read_u32::<LittleEndian>()?;
inputs.push(TxInput {
outpoint,
script_len,
script_sig,
seq_no,
});
}
Ok(inputs)
}
fn read_tx_outputs(&mut self, output_count: u64) -> OpResult<Vec<TxOutput>> {
let mut outputs = Vec::with_capacity(output_count as usize);
for _ in 0..output_count {
let value = self.read_u64::<LittleEndian>()?;
let script_len = VarUint::read_from(self)?;
let script_pubkey = self.read_u8_vec(script_len.value as u32)?;
outputs.push(TxOutput {
value,
script_len,
script_pubkey,
});
}
Ok(outputs)
}
}
impl<R: io::Read + ?Sized> BlockchainRead for R {}
#[cfg(test)]
mod tests {
use super::*;
use crate::blockchain::parser::types::{Bitcoin, Coin};
use crate::blockchain::proto::script;
use crate::blockchain::proto::tx::EvaluatedTx;
use crate::common::utils;
use blockchain::proto::script::ScriptPattern;
use byteorder::{LittleEndian, ReadBytesExt};
use std::io::{BufReader, Cursor};
#[test]
fn test_bitcoin_parse_genesis_block() {
let raw_data = vec![
0xf9, 0xbe, 0xb4, 0xd9, 0x1d, 0x01, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x3b, 0xa3, 0xed, 0xfd, 0x7a, 0x7b, 0x12, 0xb2, 0x7a, 0xc7, 0x2c, 0x3e,
0x67, 0x76, 0x8f, 0x61, 0x7f, 0xc8, 0x1b, 0xc3, 0x88, 0x8a, 0x51, 0x32, 0x3a, 0x9f,
0xb8, 0xaa, 0x4b, 0x1e, 0x5e, 0x4a, 0x29, 0xab, 0x5f, 0x49, 0xff, 0xff, 0x00, 0x1d,
0x1d, 0xac, 0x2b, 0x7c, 0x01, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0xff, 0xff, 0xff, 0xff, 0x4d, 0x04, 0xff, 0xff, 0x00, 0x1d, 0x01, 0x04, 0x45, 0x54,
0x68, 0x65, 0x20, 0x54, 0x69, 0x6d, 0x65, 0x73, 0x20, 0x30, 0x33, 0x2f, 0x4a, 0x61,
0x6e, 0x2f, 0x32, 0x30, 0x30, 0x39, 0x20, 0x43, 0x68, 0x61, 0x6e, 0x63, 0x65, 0x6c,
0x6c, 0x6f, 0x72, 0x20, 0x6f, 0x6e, 0x20, 0x62, 0x72, 0x69, 0x6e, 0x6b, 0x20, 0x6f,
0x66, 0x20, 0x73, 0x65, 0x63, 0x6f, 0x6e, 0x64, 0x20, 0x62, 0x61, 0x69, 0x6c, 0x6f,
0x75, 0x74, 0x20, 0x66, 0x6f, 0x72, 0x20, 0x62, 0x61, 0x6e, 0x6b, 0x73, 0xff, 0xff,
0xff, 0xff, 0x01, 0x00, 0xf2, 0x05, 0x2a, 0x01, 0x00, 0x00, 0x00, 0x43, 0x41, 0x04,
0x67, 0x8a, 0xfd, 0xb0, 0xfe, 0x55, 0x48, 0x27, 0x19, 0x67, 0xf1, 0xa6, 0x71, 0x30,
0xb7, 0x10, 0x5c, 0xd6, 0xa8, 0x28, 0xe0, 0x39, 0x09, 0xa6, 0x79, 0x62, 0xe0, 0xea,
0x1f, 0x61, 0xde, 0xb6, 0x49, 0xf6, 0xbc, 0x3f, 0x4c, 0xef, 0x38, 0xc4, 0xf3, 0x55,
0x04, 0xe5, 0x1e, 0xc1, 0x12, 0xde, 0x5c, 0x38, 0x4d, 0xf7, 0xba, 0x0b, 0x8d, 0x57,
0x8a, 0x4c, 0x70, 0x2b, 0x6b, 0xf1, 0x1d, 0x5f, 0xac, 0x00, 0x00, 0x00, 0x00, 0xf9,
0xbe, 0xb4, 0xd9, 0xd7, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x6f, 0xe2, 0x8c,
0x0a, 0xb6, 0xf1, 0xb3, 0x72, 0xc1, 0xa6, 0xa2, 0x46, 0xae, 0x63, 0xf7, 0x4f, 0x93,
0x1e, 0x83, 0x65, 0xe1, 0x5a, 0x08, 0x9c, 0x68, 0xd6, 0x19, 0x00, 0x00, 0x00, 0x00,
0x00, 0x98, 0x20, 0x51, 0xfd, 0x1e, 0x4b, 0xa7, 0x44, 0xbb, 0xbe, 0x68, 0x0e, 0x1f,
0xee, 0x14, 0x67, 0x7b, 0xa1, 0xa3, 0xc3, 0x54, 0x0b, 0xf7, 0xb1, 0xcd, 0xb6, 0x06,
0xe8, 0x57, 0x23, 0x3e, 0x0e, 0x61, 0xbc, 0x66, 0x49, 0xff, 0xff, 0x00, 0x1d, 0x01,
0xe3, 0x62, 0x99, 0x01, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff,
0xff, 0xff, 0xff, 0x07, 0x04, 0xff, 0xff, 0x00, 0x1d, 0x01, 0x04, 0xff, 0xff, 0xff,
0xff, 0x01, 0x00, 0xf2, 0x05, 0x2a, 0x01, 0x00, 0x00, 0x00, 0x43, 0x41, 0x04, 0x96,
0xb5, 0x38, 0xe8, 0x53, 0x51, 0x9c, 0x72, 0x6a, 0x2c, 0x91, 0xe6, 0x1e, 0xc1, 0x16,
0x00, 0xae, 0x13, 0x90, 0x81, 0x3a, 0x62, 0x7c, 0x66, 0xfb, 0x8b, 0xe7, 0x94, 0x7b,
0xe6, 0x3c, 0x52, 0xda, 0x75, 0x89, 0x37, 0x95, 0x15, 0xd4, 0xe0, 0xa6, 0x04, 0xf8,
0x14, 0x17, 0x81, 0xe6, 0x22, 0x94, 0x72, 0x11, 0x66, 0xbf, 0x62, 0x1e, 0x73, 0xa8,
0x2c, 0xbf, 0x23, 0x42, 0xc8, 0x58, 0xee, 0xac, 0x00, 0x00, 0x00, 0x0,
];
let inner = Cursor::new(raw_data);
let mut reader = BufReader::with_capacity(200, inner);
let magic: u32 = reader.read_u32::<LittleEndian>().unwrap();
let block_size: u32 = reader.read_u32::<LittleEndian>().unwrap();
let block = reader.read_block(block_size, Bitcoin.version_id()).unwrap();
assert_eq!(0xd9b4bef9, magic);
assert_eq!(285, block.size);
assert_eq!(0x00000001, block.header.value.version);
assert_eq!(
"0000000000000000000000000000000000000000000000000000000000000000",
utils::arr_to_hex(&block.header.value.prev_hash)
);
assert_eq!(
"3ba3edfd7a7b12b27ac72c3e67768f617fc81bc3888a51323a9fb8aa4b1e5e4a",
utils::arr_to_hex(&block.header.value.merkle_root)
);
assert_eq!(
"000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f",
utils::arr_to_hex_swapped(&block.header.hash)
);
assert_eq!(1231006505, block.header.value.timestamp);
assert_eq!(0x1d00ffff, block.header.value.bits);
assert_eq!(2083236893, block.header.value.nonce);
assert_eq!(0x01, block.tx_count.value);
assert_eq!(0x00000001, block.txs[0].value.version);
assert_eq!(0x01, block.txs[0].value.in_count.value);
assert_eq!(
"0000000000000000000000000000000000000000000000000000000000000000",
utils::arr_to_hex_swapped(&block.txs[0].value.inputs[0].outpoint.txid)
);
assert_eq!(0xffffffff, block.txs[0].value.inputs[0].outpoint.index);
assert_eq!(0x4d, block.txs[0].value.inputs[0].script_len.value);
assert_eq!("04ffff001d0104455468652054696d65732030332f4a616e2f32303039204368616e63656c6c6f72206f6e206272696e6b206f66207365636f6e64206261696c6f757420666f722062616e6b73",
utils::arr_to_hex(&block.txs[0].value.inputs[0].script_sig));
assert_eq!(0xffffffff, block.txs[0].value.inputs[0].seq_no);
assert_eq!(0x01, block.txs[0].value.out_count.value);
assert_eq!(
u64::from_be(0x00f2052a01000000),
block.txs[0].value.outputs[0].out.value
);
assert_eq!(0x43, block.txs[0].value.outputs[0].out.script_len.value);
let script_pubkey = &block.txs[0].value.outputs[0].out.script_pubkey;
assert_eq!("4104678afdb0fe5548271967f1a67130b7105cd6a828e03909a67962e0ea1f61deb649f6bc3f4cef38c4f35504e51ec112de5c384df7ba0b8d578a4c702b6bf11d5fac",
utils::arr_to_hex(&script_pubkey));
assert_eq!(0x00000000, block.txs[0].value.locktime);
assert_eq!(
Some(String::from("1A1zP1eP5QGefi2DMPTfTL5SLmv7DivfNa")),
script::eval_from_bytes(script_pubkey, Bitcoin.version_id()).address
);
}
#[test]
fn test_bitcoin_parse_segwit_tx() {
let raw_data = vec![
0x01, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x15, 0xe1, 0x80, 0xdc, 0x28, 0xa2, 0x32,
0x7e, 0x68, 0x7f, 0xac, 0xc3, 0x3f, 0x10, 0xf2, 0xa2, 0x0d, 0xa7, 0x17, 0xe5, 0x54,
0x84, 0x06, 0xf7, 0xae, 0x8b, 0x4c, 0x81, 0x10, 0x72, 0xf8, 0x56, 0x03, 0x00, 0x00,
0x00, 0x17, 0x16, 0x00, 0x14, 0x1d, 0x7c, 0xd6, 0xc7, 0x5c, 0x2e, 0x86, 0xf4, 0xcb,
0xf9, 0x8e, 0xae, 0xd2, 0x21, 0xb3, 0x0b, 0xd9, 0xa0, 0xb9, 0x28, 0xff, 0xff, 0xff,
0xff, 0x01, 0x9c, 0xae, 0xf5, 0x05, 0x00, 0x00, 0x00, 0x00, 0x19, 0x76, 0xa9, 0x14,
0x1d, 0x7c, 0xd6, 0xc7, 0x5c, 0x2e, 0x86, 0xf4, 0xcb, 0xf9, 0x8e, 0xae, 0xd2, 0x21,
0xb3, 0x0b, 0xd9, 0xa0, 0xb9, 0x28, 0x88, 0xac, 0x02, 0x48, 0x30, 0x45, 0x02, 0x21,
0x00, 0xf7, 0x64, 0x28, 0x7d, 0x3e, 0x99, 0xb1, 0x47, 0x4d, 0xa9, 0xbe, 0xc7, 0xf7,
0xed, 0x23, 0x6d, 0x6c, 0x81, 0xe7, 0x93, 0xb2, 0x0c, 0x4b, 0x5a, 0xa1, 0xf3, 0x05,
0x1b, 0x9a, 0x7d, 0xaa, 0x63, 0x02, 0x20, 0x16, 0xa1, 0x98, 0x03, 0x1d, 0x55, 0x54,
0xdb, 0xb8, 0x55, 0xbd, 0xbe, 0x85, 0x34, 0x77, 0x6a, 0x4b, 0xe6, 0x95, 0x8b, 0xd8,
0xd5, 0x30, 0xdc, 0x00, 0x1c, 0x32, 0xb8, 0x28, 0xf6, 0xf0, 0xab, 0x01, 0x21, 0x03,
0x82, 0x62, 0xa6, 0xc6, 0xce, 0xc9, 0x3c, 0x2d, 0x3e, 0xcd, 0x6c, 0x60, 0x72, 0xef,
0xea, 0x86, 0xd0, 0x2f, 0xf8, 0xe3, 0x32, 0x8b, 0xbd, 0x02, 0x42, 0xb2, 0x0a, 0xf3,
0x42, 0x59, 0x90, 0xac, 0x00, 0x00, 0x00, 0x00,
];
let inner = Cursor::new(raw_data);
let mut reader = BufReader::with_capacity(200, inner);
let txs: Vec<EvaluatedTx> = reader
.read_txs(1, 0x00)
.unwrap()
.into_iter()
.map(|raw| EvaluatedTx::from(raw))
.collect();
assert_eq!(txs.len(), 1);
let tx = txs.first().unwrap();
assert_eq!(tx.version, 1);
assert_eq!(tx.in_count.value, 1);
assert_eq!(tx.inputs.len(), 1);
let prev_hash = [
0x15, 0xe1, 0x80, 0xdc, 0x28, 0xa2, 0x32, 0x7e, 0x68, 0x7f, 0xac, 0xc3, 0x3f, 0x10,
0xf2, 0xa2, 0x0d, 0xa7, 0x17, 0xe5, 0x54, 0x84, 0x06, 0xf7, 0xae, 0x8b, 0x4c, 0x81,
0x10, 0x72, 0xf8, 0x56,
];
assert_eq!(tx.inputs[0].outpoint.txid, prev_hash);
assert_eq!(tx.inputs[0].outpoint.index, 3);
assert_eq!(tx.inputs[0].script_len.value, 23);
assert_eq!(tx.inputs[0].seq_no, 0xffffffff);
assert_eq!(tx.out_count.value, 1);
assert_eq!(tx.outputs.len(), 1);
assert_eq!(tx.outputs[0].out.value, 99987100);
assert_eq!(tx.outputs[0].out.script_len.value, 25);
assert_eq!(
tx.outputs[0].script.pattern,
ScriptPattern::Pay2PublicKeyHash
);
assert_eq!(
tx.outputs[0].script.address,
Some(String::from("13gv9XbKJPxxRF8Zm1LsVKeeiMCFguQPqm"))
);
assert_eq!(tx.locktime, 0);
}
}