Make arbitrary block chains pass some genesis checks (#2208)

* Clarify the finalized state assertion that checks the genesis block

* Make arbitrary block chains pass some genesis checks

Use the genesis previous block hash for
- the first arbitrary block in each chain, and
- individual arbitrary blocks.

This setting can be adjusted by individual proptests as needed.
This commit is contained in:
teor 2021-05-28 01:41:20 +10:00 committed by GitHub
parent f0c271bcfe
commit f94033df08
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 31 additions and 6 deletions

View File

@ -6,7 +6,7 @@ use proptest::{
use std::sync::Arc; use std::sync::Arc;
use crate::{ use crate::{
parameters::{Network, NetworkUpgrade}, parameters::{Network, NetworkUpgrade, GENESIS_PREVIOUS_BLOCK_HASH},
serialization, serialization,
work::{difficulty::CompactDifficulty, equihash}, work::{difficulty::CompactDifficulty, equihash},
}; };
@ -20,6 +20,9 @@ pub struct LedgerState {
/// The tip height of the block or start of the chain. /// The tip height of the block or start of the chain.
/// ///
/// To get the network upgrade, use the `network_upgrade` method. /// To get the network upgrade, use the `network_upgrade` method.
///
/// If `network_upgrade_override` is not set, the network upgrade is derived
/// from the height and network.
pub tip_height: Height, pub tip_height: Height,
/// The network to generate fake blocks for. /// The network to generate fake blocks for.
@ -38,6 +41,12 @@ pub struct LedgerState {
/// For an individual transaction, make the transaction a coinbase /// For an individual transaction, make the transaction a coinbase
/// transaction. /// transaction.
pub(crate) has_coinbase: bool, pub(crate) has_coinbase: bool,
/// Should this block have a genesis (all-zeroes) previous block hash?
///
/// In Zebra's proptests, the previous block hash can be overriden with
/// genesis at any block height.
genesis_previous_block_hash_override: bool,
} }
impl LedgerState { impl LedgerState {
@ -53,6 +62,14 @@ impl LedgerState {
} }
} }
/// Should this block have a genesis (all-zeroes) previous block hash?
///
/// In Zebra's proptests, the previous block hash can be overriden with
/// genesis at any block height.
pub fn use_genesis_previous_block_hash(&self) -> bool {
self.tip_height == Height(0) || self.genesis_previous_block_hash_override
}
/// Returns a strategy for creating `LedgerState`s that always have coinbase /// Returns a strategy for creating `LedgerState`s that always have coinbase
/// transactions. /// transactions.
pub fn coinbase_strategy() -> BoxedStrategy<Self> { pub fn coinbase_strategy() -> BoxedStrategy<Self> {
@ -79,6 +96,8 @@ impl Default for LedgerState {
network, network,
network_upgrade_override: nu5_override, network_upgrade_override: nu5_override,
has_coinbase: true, has_coinbase: true,
// start each chain with a genesis previous block hash, regardless of height
genesis_previous_block_hash_override: true,
} }
} }
} }
@ -110,6 +129,7 @@ impl Arbitrary for LedgerState {
network, network,
network_upgrade_override, network_upgrade_override,
has_coinbase: require_coinbase || has_coinbase, has_coinbase: require_coinbase || has_coinbase,
genesis_previous_block_hash_override: true,
} }
}) })
.boxed() .boxed()
@ -125,9 +145,14 @@ impl Arbitrary for Block {
let transactions_strategy = Transaction::vec_strategy(ledger_state, 2); let transactions_strategy = Transaction::vec_strategy(ledger_state, 2);
(any::<Header>(), transactions_strategy) (any::<Header>(), transactions_strategy)
.prop_map(|(header, transactions)| Self { .prop_map(move |(mut header, transactions)| {
header, if ledger_state.genesis_previous_block_hash_override {
transactions, header.previous_block_hash = GENESIS_PREVIOUS_BLOCK_HASH;
}
Self {
header,
transactions,
}
}) })
.boxed() .boxed()
} }
@ -147,6 +172,7 @@ impl Block {
for _ in 0..count { for _ in 0..count {
vec.push(Block::arbitrary_with(current).prop_map(Arc::new)); vec.push(Block::arbitrary_with(current).prop_map(Arc::new));
current.tip_height.0 += 1; current.tip_height.0 += 1;
current.genesis_previous_block_hash_override = false;
} }
vec.boxed() vec.boxed()

View File

@ -198,8 +198,7 @@ impl FinalizedState {
// Assert that callers (including unit tests) get the chain order correct // Assert that callers (including unit tests) get the chain order correct
if self.is_empty(hash_by_height) { if self.is_empty(hash_by_height) {
assert_eq!( assert_eq!(
block::Hash([0; 32]), GENESIS_PREVIOUS_BLOCK_HASH, block.header.previous_block_hash,
block.header.previous_block_hash,
"the first block added to an empty state must be a genesis block" "the first block added to an empty state must be a genesis block"
); );
assert_eq!( assert_eq!(