Enable more Transaction v5 tests (#2063)

* Use NU5 and Transaction v5 in most proptests

* Stop skipping post-Canopy blocks in the block subsidy tests

Co-authored-by: teor <teor@riseup.net>
This commit is contained in:
Alfredo Garcia 2021-04-25 19:32:21 -03:00 committed by GitHub
parent 1be56166b6
commit a49b9d44f6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 106 additions and 50 deletions

View File

@ -17,39 +17,107 @@ use super::*;
#[non_exhaustive] #[non_exhaustive]
/// The configuration data for proptest when generating arbitrary chains /// The configuration data for proptest when generating arbitrary chains
pub struct LedgerState { 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.
pub tip_height: Height, pub tip_height: Height,
/// The network to generate fake blocks for
/// The network to generate fake blocks for.
pub network: Network, pub network: Network,
/// Make this fake transaction a coinbase transaction /// Overrides the network upgrade calculated from `tip_height` and `network`.
pub(crate) is_coinbase: bool, ///
/// To get the network upgrade, use the `network_upgrade` method.
pub network_upgrade_override: Option<NetworkUpgrade>,
/// Generate coinbase transactions.
///
/// In a block or transaction vector, make the first transaction a coinbase
/// transaction.
///
/// For an individual transaction, make the transaction a coinbase
/// transaction.
pub(crate) has_coinbase: bool,
} }
impl LedgerState { impl LedgerState {
/// Construct a new ledger state for generating arbitrary chains via proptest /// Returns the network upgrade for this ledger state.
pub fn new(tip_height: Height, network: Network) -> Self { ///
Self { /// If `network_upgrade_override` is set, it replaces the upgrade calculated
tip_height, /// using `tip_height` and `network`.
is_coinbase: true, pub fn network_upgrade(&self) -> NetworkUpgrade {
network, if let Some(network_upgrade_override) = self.network_upgrade_override {
network_upgrade_override
} else {
NetworkUpgrade::current(self.network, self.tip_height)
} }
} }
/// Returns a strategy for creating `LedgerState`s that always have coinbase
/// transactions.
pub fn coinbase_strategy() -> BoxedStrategy<Self> {
Self::arbitrary_with(true)
}
} }
impl Default for LedgerState { impl Default for LedgerState {
fn default() -> Self { fn default() -> Self {
let network = Network::Mainnet; let network = Network::Mainnet;
let tip_height = NetworkUpgrade::Canopy.activation_height(network).unwrap(); let most_recent_nu = NetworkUpgrade::current(network, Height::MAX);
let most_recent_activation_height = most_recent_nu.activation_height(network).unwrap();
// TODO: dynamically select any future network upgrade (#1974)
let nu5_activation_height = NetworkUpgrade::Nu5.activation_height(network);
let nu5_override = if nu5_activation_height.is_some() {
None
} else {
Some(NetworkUpgrade::Nu5)
};
Self { Self {
tip_height, tip_height: most_recent_activation_height,
is_coinbase: true,
network, network,
network_upgrade_override: nu5_override,
has_coinbase: true,
} }
} }
} }
impl Arbitrary for LedgerState {
type Parameters = bool;
/// Generate an arbitrary `LedgerState`.
///
/// The default strategy arbitrarily skips some coinbase transactions. To
/// override, use `LedgerState::coinbase_strategy`.
fn arbitrary_with(require_coinbase: Self::Parameters) -> Self::Strategy {
(
any::<Height>(),
any::<Network>(),
any::<bool>(),
any::<bool>(),
)
.prop_map(move |(tip_height, network, nu5_override, has_coinbase)| {
// TODO: dynamically select any future network upgrade (#1974)
let network_upgrade_override = if nu5_override {
Some(NetworkUpgrade::Nu5)
} else {
None
};
LedgerState {
tip_height,
network,
network_upgrade_override,
has_coinbase: require_coinbase || has_coinbase,
}
})
.boxed()
}
type Strategy = BoxedStrategy<Self>;
}
impl Arbitrary for Block { impl Arbitrary for Block {
type Parameters = LedgerState; type Parameters = LedgerState;

View File

@ -5,7 +5,7 @@ use proptest::{arbitrary::any, prelude::*, test_runner::Config};
use zebra_test::prelude::*; use zebra_test::prelude::*;
use crate::serialization::{SerializationError, ZcashDeserializeInto, ZcashSerialize}; use crate::serialization::{SerializationError, ZcashDeserializeInto, ZcashSerialize};
use crate::{block, parameters::Network, LedgerState}; use crate::{parameters::Network, LedgerState};
use super::super::{serialize::MAX_BLOCK_BYTES, *}; use super::super::{serialize::MAX_BLOCK_BYTES, *};
@ -121,13 +121,7 @@ proptest! {
fn blocks_have_coinbase() -> Result<()> { fn blocks_have_coinbase() -> Result<()> {
zebra_test::init(); zebra_test::init();
let strategy = any::<block::Height>() let strategy = LedgerState::coinbase_strategy().prop_flat_map(Block::arbitrary_with);
.prop_map(|tip_height| LedgerState {
tip_height,
is_coinbase: true,
network: Network::Mainnet,
})
.prop_flat_map(Block::arbitrary_with);
proptest!(|(block in strategy)| { proptest!(|(block in strategy)| {
let has_coinbase = block.coinbase_height().is_some(); let has_coinbase = block.coinbase_height().is_some();

View File

@ -130,7 +130,7 @@ impl Transaction {
len: usize, len: usize,
) -> BoxedStrategy<Vec<Arc<Self>>> { ) -> BoxedStrategy<Vec<Arc<Self>>> {
let coinbase = Transaction::arbitrary_with(ledger_state).prop_map(Arc::new); let coinbase = Transaction::arbitrary_with(ledger_state).prop_map(Arc::new);
ledger_state.is_coinbase = false; ledger_state.has_coinbase = false;
let remainder = vec( let remainder = vec(
Transaction::arbitrary_with(ledger_state).prop_map(Arc::new), Transaction::arbitrary_with(ledger_state).prop_map(Arc::new),
len, len,
@ -302,16 +302,7 @@ impl Arbitrary for Transaction {
type Parameters = LedgerState; type Parameters = LedgerState;
fn arbitrary_with(ledger_state: Self::Parameters) -> Self::Strategy { fn arbitrary_with(ledger_state: Self::Parameters) -> Self::Strategy {
let LedgerState { match ledger_state.network_upgrade() {
tip_height,
network,
..
} = ledger_state;
let height = block::Height(tip_height.0 + 1);
let network_upgrade = NetworkUpgrade::current(network, height);
match network_upgrade {
NetworkUpgrade::Genesis | NetworkUpgrade::BeforeOverwinter => { NetworkUpgrade::Genesis | NetworkUpgrade::BeforeOverwinter => {
Self::v1_strategy(ledger_state) Self::v1_strategy(ledger_state)
} }

View File

@ -5,9 +5,9 @@ use crate::{block, LedgerState};
use super::{CoinbaseData, Input, OutPoint, Script}; use super::{CoinbaseData, Input, OutPoint, Script};
impl Input { impl Input {
/// Construct a strategy for creating validish vecs of Inputs. /// Construct a strategy for creating valid-ish vecs of Inputs.
pub fn vec_strategy(ledger_state: LedgerState, max_size: usize) -> BoxedStrategy<Vec<Self>> { pub fn vec_strategy(ledger_state: LedgerState, max_size: usize) -> BoxedStrategy<Vec<Self>> {
if ledger_state.is_coinbase { if ledger_state.has_coinbase {
let height = block::Height(ledger_state.tip_height.0 + 1); let height = block::Height(ledger_state.tip_height.0 + 1);
Self::arbitrary_with(Some(height)) Self::arbitrary_with(Some(height))
.prop_map(|input| vec![input]) .prop_map(|input| vec![input])

View File

@ -1,6 +1,6 @@
use zebra_test::prelude::*; use zebra_test::prelude::*;
use crate::{block, parameters::Network, LedgerState}; use crate::{block, LedgerState};
use super::Input; use super::Input;
@ -24,12 +24,7 @@ fn input_coinbase_vecs_only_have_coinbase_input() -> Result<()> {
zebra_test::init(); zebra_test::init();
let max_size = 100; let max_size = 100;
let strategy = any::<block::Height>() let strategy = LedgerState::coinbase_strategy()
.prop_map(|tip_height| LedgerState {
tip_height,
is_coinbase: true,
network: Network::Mainnet,
})
.prop_flat_map(|ledger_state| Input::vec_strategy(ledger_state, max_size)); .prop_flat_map(|ledger_state| Input::vec_strategy(ledger_state, max_size));
proptest!(|(inputs in strategy)| { proptest!(|(inputs in strategy)| {

View File

@ -13,7 +13,7 @@ use tower::buffer::Buffer;
use zebra_chain::{ use zebra_chain::{
block::{self, Block, Height}, block::{self, Block, Height},
parameters::{Network, NetworkUpgrade}, parameters::Network,
serialization::{ZcashDeserialize, ZcashDeserializeInto}, serialization::{ZcashDeserialize, ZcashDeserializeInto},
work::difficulty::{ExpandedDifficulty, INVALID_COMPACT_DIFFICULTY}, work::difficulty::{ExpandedDifficulty, INVALID_COMPACT_DIFFICULTY},
}; };
@ -283,9 +283,7 @@ fn subsidy_is_valid_for_network(network: Network) -> Result<(), Report> {
.expect("block is structurally valid"); .expect("block is structurally valid");
// TODO: first halving, second halving, third halving, and very large halvings // TODO: first halving, second halving, third halving, and very large halvings
if block::Height(height) > SLOW_START_INTERVAL if block::Height(height) > SLOW_START_INTERVAL {
&& block::Height(height) < NetworkUpgrade::Canopy.activation_height(network).unwrap()
{
check::subsidy_is_valid(&block, network).expect("subsidies should pass for this block"); check::subsidy_is_valid(&block, network).expect("subsidies should pass for this block");
} }
} }

View File

@ -49,11 +49,21 @@ impl Strategy for PreparedChain {
fn new_tree(&self, runner: &mut TestRunner) -> NewTree<Self> { fn new_tree(&self, runner: &mut TestRunner) -> NewTree<Self> {
let mut chain = self.chain.lock().unwrap(); let mut chain = self.chain.lock().unwrap();
if chain.is_none() { if chain.is_none() {
let blocks = let ledger_strategy = LedgerState::coinbase_strategy();
Block::partial_chain_strategy(LedgerState::default(), MAX_PARTIAL_CHAIN_BLOCKS)
.prop_map(|vec| vec.into_iter().map(|blk| blk.prepare()).collect::<Vec<_>>()) // Disable the NU5 override until UpdateWith is implemented for Tx v5 (#1982)
.new_tree(runner)? let ledger_strategy = ledger_strategy.prop_map(|mut ledger_state| {
.current(); ledger_state.network_upgrade_override = None;
ledger_state
});
let blocks = ledger_strategy
.prop_flat_map(|ledger_state| {
Block::partial_chain_strategy(ledger_state, MAX_PARTIAL_CHAIN_BLOCKS)
})
.prop_map(|vec| vec.into_iter().map(|blk| blk.prepare()).collect::<Vec<_>>())
.new_tree(runner)?
.current();
*chain = Some(Arc::new(blocks)); *chain = Some(Arc::new(blocks));
} }