use bitcoin_hashes::{sha256d, Hash};
use blockdata::constants::max_target;
use blockdata::transaction::Transaction;
use consensus::encode::VarInt;
use network::constants::Network;
use util;
use util::hash::BitcoinHash;
use util::uint::Uint256;
use util::Error::{SpvBadProofOfWork, SpvBadTarget};
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct BlockHeader {
pub version: u32,
pub prev_blockhash: sha256d::Hash,
pub merkle_root: sha256d::Hash,
pub time: u32,
pub bits: u32,
pub nonce: u32,
pub coinbase_txn: Transaction,
pub block_hash: sha256d::Hash,
pub coinbase_branch_hashes: Vec<sha256d::Hash>,
pub coinbase_branch_side_mask: u32,
pub blockchain_branch_hashes: Vec<sha256d::Hash>,
pub blockchain_branch_side_mask: u32,
pub parent_version: u32,
pub parent_prev_blockhash: sha256d::Hash,
pub parent_merkle_root: sha256d::Hash,
pub parent_time: u32,
pub parent_bits: u32,
pub parent_nonce: u32,
}
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct Block {
pub header: BlockHeader,
pub txdata: Vec<Transaction>,
}
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct LoneBlockHeader {
pub header: BlockHeader,
pub tx_count: VarInt,
}
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct BaseHeader {
pub version: u32,
pub prev_blockhash: sha256d::Hash,
pub merkle_root: sha256d::Hash,
pub time: u32,
pub bits: u32,
pub nonce: u32,
}
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct BaseBlock {
pub header: BaseHeader,
pub txdata: Vec<Transaction>,
}
impl BlockHeader {
pub fn target(&self) -> Uint256 {
let (mant, expt) = {
let unshifted_expt = self.bits >> 24;
if unshifted_expt <= 3 {
(
(self.bits & 0xFFFFFF) >> (8 * (3 - unshifted_expt as usize)),
0,
)
} else {
(self.bits & 0xFFFFFF, 8 * ((self.bits >> 24) - 3))
}
};
if mant > 0x7FFFFF {
Default::default()
} else {
Uint256::from_u64(mant as u64).unwrap() << (expt as usize)
}
}
pub fn compact_target_from_u256(value: &Uint256) -> u32 {
let mut size = (value.bits() + 7) / 8;
let mut compact = if size <= 3 {
(value.low_u64() << (8 * (3 - size))) as u32
} else {
let bn = *value >> (8 * (size - 3));
bn.low_u32()
};
if (compact & 0x00800000) != 0 {
compact >>= 8;
size += 1;
}
compact | (size << 24) as u32
}
pub fn difficulty(&self, network: Network) -> u64 {
(max_target(network) / self.target()).low_u64()
}
pub fn spv_validate(&self, required_target: &Uint256) -> Result<(), util::Error> {
use byteorder::{ByteOrder, LittleEndian};
let target = &self.target();
if target != required_target {
return Err(SpvBadTarget);
}
let data: [u8; 32] = self.bitcoin_hash().into_inner();
let mut ret = [0u64; 4];
LittleEndian::read_u64_into(&data, &mut ret);
let hash = &Uint256(ret);
if hash <= target {
Ok(())
} else {
Err(SpvBadProofOfWork)
}
}
pub fn work(&self) -> Uint256 {
let mut ret = !self.target();
let mut ret1 = self.target();
ret1.increment();
ret = ret / ret1;
ret.increment();
ret
}
}
impl From<&BlockHeader> for BaseHeader {
fn from(item: &BlockHeader) -> Self {
BaseHeader {
version: item.version,
prev_blockhash: item.prev_blockhash,
merkle_root: item.merkle_root,
time: item.time,
bits: item.bits,
nonce: item.nonce,
}
}
}
impl BitcoinHash for BaseHeader {
fn bitcoin_hash(&self) -> sha256d::Hash {
use consensus::encode::serialize;
sha256d::Hash::hash(&serialize(self))
}
}
impl BitcoinHash for BlockHeader {
fn bitcoin_hash(&self) -> sha256d::Hash {
let base_header = BaseHeader::from(self);
base_header.bitcoin_hash()
}
}
impl BitcoinHash for BaseBlock {
fn bitcoin_hash(&self) -> sha256d::Hash {
self.header.bitcoin_hash()
}
}
impl BitcoinHash for Block {
fn bitcoin_hash(&self) -> sha256d::Hash {
let base_header = BaseHeader::from(&self.header);
base_header.bitcoin_hash()
}
}
impl_consensus_encoding!(
BlockHeader,
version,
prev_blockhash,
merkle_root,
time,
bits,
nonce,
coinbase_txn,
block_hash,
coinbase_branch_hashes,
coinbase_branch_side_mask,
blockchain_branch_hashes,
blockchain_branch_side_mask,
parent_version,
parent_prev_blockhash,
parent_merkle_root,
parent_time,
parent_bits,
parent_nonce
);
impl_consensus_encoding!(Block, header, txdata);
impl_consensus_encoding!(
BaseHeader,
version,
prev_blockhash,
merkle_root,
time,
bits,
nonce
);
impl_consensus_encoding!(BaseBlock, header, txdata);
impl_consensus_encoding!(LoneBlockHeader, header, tx_count);
#[cfg(test)]
mod tests {
use hex::decode as hex_decode;
use blockdata::block::{Block, BlockHeader};
use consensus::encode::{deserialize, serialize};
#[test]
fn block_test() {
let some_block = hex_decode("040100106f378876737d2a2bfffd51ca4c7e3d6281dad5fbd0b8ce61cde88674440f0000aaa7bee217ab6c525b55734dfb013e6e7b7bb70848adb50407b6cc74fbf2011ee121525cf0ff0f1e0000000002000000010000000000000000000000000000000000000000000000000000000000000000ffffffff292890e8b09682761713f0f091aef58669e8c37149aedd282337da6193edc7ffc8f50100000000000000ffffffff0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000000000000c69c3a951476397def6efd0d481944058f9d56497c040ff1c36c9f2ef090cbfb00000000000000005a5b000001020000000001010000000000000000000000000000000000000000000000000000000000000000ffffffff03510109ffffffff020040874e115cbd002321025687e93e908ab873cb37174215bed989791fe93e0ab5f5cc95f4250deb11fa8aac0000000000000000266a24aa21a9ede2f61c3f71d1defd3fa999dfa36953755c690689799962b48bebd836974e8cf90120000000000000000000000000000000000000000000000000000000000000000000000000").unwrap();
let cutoff_block = hex_decode("010000004ddccd549d28f385ab457e98d1b11ce80bfea2c5ab93015ade4973e400000000bf4473e53794beae34e64fccc471dace6ae544180816f89591894e0f417a914cd74d6e49ffff001d323b3a7b0201000000010000000000000000000000000000000000000000000000000000000000000000ffffffff0804ffff001d026e04ffffffff0100f2052a0100000043410446ef0102d1ec5240f0d061a4246c1bdef63fc3dbab7733052fbbf0ecd8f41fc26bf049ebb4f9527f374280259e7cfa99c48b0e3f39c51347a19a5819651503a5ac00000000010000000321f75f3139a013f50f315b23b0c9a2b6eac31e2bec98e5891c924664889942260000000049483045022100cb2c6b346a978ab8c61b18b5e9397755cbd17d6eb2fe0083ef32e067fa6c785a02206ce44e613f31d9a6b0517e46f3db1576e9812cc98d159bfdaf759a5014081b5c01ffffffff79cda0945903627c3da1f85fc95d0b8ee3e76ae0cfdc9a65d09744b1f8fc85430000000049483045022047957cdd957cfd0becd642f6b84d82f49b6cb4c51a91f49246908af7c3cfdf4a022100e96b46621f1bffcf5ea5982f88cef651e9354f5791602369bf5a82a6cd61a62501fffffffffe09f5fe3ffbf5ee97a54eb5e5069e9da6b4856ee86fc52938c2f979b0f38e82000000004847304402204165be9a4cbab8049e1af9723b96199bfd3e85f44c6b4c0177e3962686b26073022028f638da23fc003760861ad481ead4099312c60030d4cb57820ce4d33812a5ce01ffffffff01009d966b01000000434104ea1feff861b51fe3f5f8a3b12d0f4712db80e919548a80839fc47c6a21e66d957e9c5d8cd108c7a2d2324bad71f9904ac0ae7336507d785b17a2c115e427a32fac00000000").unwrap();
let prevhash =
hex_decode("6f378876737d2a2bfffd51ca4c7e3d6281dad5fbd0b8ce61cde88674440f0000").unwrap();
let merkle =
hex_decode("aaa7bee217ab6c525b55734dfb013e6e7b7bb70848adb50407b6cc74fbf2011e").unwrap();
let decode: Result<Block, _> = deserialize(&some_block);
let bad_decode: Result<Block, _> = deserialize(&cutoff_block);
assert!(decode.is_ok());
assert!(bad_decode.is_err());
let real_decode = decode.unwrap();
assert_eq!(real_decode.header.version, 268435716);
assert_eq!(serialize(&real_decode.header.prev_blockhash), prevhash);
assert_eq!(serialize(&real_decode.header.merkle_root), merkle);
assert_eq!(real_decode.header.time, 1548886497);
assert_eq!(real_decode.header.bits, 504365040);
assert_eq!(real_decode.header.nonce, 0);
assert_eq!(serialize(&real_decode), some_block);
}
#[test]
fn compact_roundrtip_test() {
let some_header = hex_decode("0401001041abad32db63c4097cab01e5b63398405ac7fb749a4213631ba2760f2c050000bf46c1fb889f0291fb830240748fb1c054242eb3d0e3a38ef11467f95c4ee84ef0f6575cf0ff0f1e0000000002000000010000000000000000000000000000000000000000000000000000000000000000ffffffff2928ff047727ad39e2fe1442251297280aff03bab479c3f17f0f9027b2380b504fb70100000000000000ffffffff00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000000000000c2cfb0be45baa44a337be963abf90c90c90a1a56131fb917e330871d2f7521d0000000000000000c30e0000").unwrap();
let header: BlockHeader =
deserialize(&some_header).expect("Can't deserialize correct block header");
assert_eq!(
header.bits,
BlockHeader::compact_target_from_u256(&header.target())
);
}
}