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:
parent
f0c271bcfe
commit
f94033df08
|
@ -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)| {
|
||||||
|
if ledger_state.genesis_previous_block_hash_override {
|
||||||
|
header.previous_block_hash = GENESIS_PREVIOUS_BLOCK_HASH;
|
||||||
|
}
|
||||||
|
Self {
|
||||||
header,
|
header,
|
||||||
transactions,
|
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()
|
||||||
|
|
|
@ -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!(
|
||||||
|
|
Loading…
Reference in New Issue