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:
parent
1be56166b6
commit
a49b9d44f6
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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])
|
||||||
|
|
|
@ -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)| {
|
||||||
|
|
|
@ -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");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue