use std::{env, sync::Arc}; use proptest::prelude::*; use proptest_derive::Arbitrary; use zebra_chain::{block::Block, chain_tip::ChainTip}; use crate::{service::chain_tip::ChainTipBlock, FinalizedBlock}; use super::super::ChainTipSender; const DEFAULT_BLOCK_VEC_PROPTEST_CASES: u32 = 4; proptest! { #![proptest_config( proptest::test_runner::Config::with_cases(env::var("PROPTEST_CASES") .ok() .and_then(|v| v.parse().ok()) .unwrap_or(DEFAULT_BLOCK_VEC_PROPTEST_CASES)) )] /// Check that the best tip uses the non-finalized tip if available, /// or otherwise the finalized tip. #[test] fn best_tip_is_latest_non_finalized_then_latest_finalized( tip_updates in any::>(), ) { let (mut chain_tip_sender, chain_tip_receiver) = ChainTipSender::new(None); let mut latest_finalized_tip = None; let mut latest_non_finalized_tip = None; let mut seen_non_finalized_tip = false; for update in tip_updates { match update { BlockUpdate::Finalized(block) => { let block = block.map(FinalizedBlock::from).map(ChainTipBlock::from); chain_tip_sender.set_finalized_tip(block.clone()); if block.is_some() { latest_finalized_tip = block; } } BlockUpdate::NonFinalized(block) => { let block = block.map(FinalizedBlock::from).map(ChainTipBlock::from); chain_tip_sender.set_best_non_finalized_tip(block.clone()); if block.is_some() { latest_non_finalized_tip = block; seen_non_finalized_tip = true; } } } } let expected_tip = if seen_non_finalized_tip { latest_non_finalized_tip } else { latest_finalized_tip }; let expected_height = expected_tip.as_ref().and_then(|block| block.block.coinbase_height()); prop_assert_eq!(chain_tip_receiver.best_tip_height(), expected_height); let expected_hash = expected_tip.as_ref().map(|block| block.block.hash()); prop_assert_eq!(chain_tip_receiver.best_tip_hash(), expected_hash); let expected_transaction_ids: Vec<_> = expected_tip .as_ref() .iter() .flat_map(|block| block.block.transactions.clone()) .map(|transaction| transaction.hash()) .collect(); prop_assert_eq!( chain_tip_receiver.best_tip_mined_transaction_ids(), expected_transaction_ids ); } } #[derive(Arbitrary, Clone, Debug)] enum BlockUpdate { Finalized(Option>), NonFinalized(Option>), }