use beacon_chain::test_utils::{AttestationStrategy, BeaconChainHarness, BlockStrategy}; use lmd_ghost::ThreadSafeReducedTree; use store::MemoryStore; use types::{EthSpec, MinimalEthSpec, Slot}; // Should ideally be divisible by 3. pub const VALIDATOR_COUNT: usize = 24; fn get_harness( validator_count: usize, ) -> BeaconChainHarness, MinimalEthSpec> { let harness = BeaconChainHarness::new(validator_count); // Move past the zero slot. harness.advance_slot(); harness } #[test] fn fork() { let harness = get_harness(VALIDATOR_COUNT); let two_thirds = (VALIDATOR_COUNT / 3) * 2; let delay = MinimalEthSpec::default_spec().min_attestation_inclusion_delay as usize; let honest_validators: Vec = (0..two_thirds).collect(); let faulty_validators: Vec = (two_thirds..VALIDATOR_COUNT).collect(); let initial_blocks = delay + 1; let honest_fork_blocks = delay + 1; let faulty_fork_blocks = delay + 2; // Build an initial chain were all validators agree. harness.extend_chain( initial_blocks, BlockStrategy::OnCanonicalHead, AttestationStrategy::AllValidators, ); // Move to the next slot so we may produce some more blocks on the head. harness.advance_slot(); // Extend the chain with blocks where only honest validators agree. let honest_head = harness.extend_chain( honest_fork_blocks, BlockStrategy::OnCanonicalHead, AttestationStrategy::SomeValidators(honest_validators.clone()), ); // Go back to the last block where all agreed, and build blocks upon it where only faulty nodes // agree. let faulty_head = harness.extend_chain( faulty_fork_blocks, BlockStrategy::ForkCanonicalChainAt { previous_slot: Slot::from(initial_blocks), first_slot: Slot::from(initial_blocks + 2), }, AttestationStrategy::SomeValidators(faulty_validators.clone()), ); assert!(honest_head != faulty_head, "forks should be distinct"); let state = &harness.chain.head().beacon_state; assert_eq!( state.slot, Slot::from(initial_blocks + honest_fork_blocks), "head should be at the current slot" ); assert_eq!( harness.chain.head().beacon_block_root, honest_head, "the honest chain should be the canonical chain" ); } #[test] fn finalizes_with_full_participation() { let num_blocks_produced = MinimalEthSpec::slots_per_epoch() * 5; let harness = get_harness(VALIDATOR_COUNT); harness.extend_chain( num_blocks_produced as usize, BlockStrategy::OnCanonicalHead, AttestationStrategy::AllValidators, ); let state = &harness.chain.head().beacon_state; assert_eq!( state.slot, num_blocks_produced, "head should be at the current slot" ); assert_eq!( state.current_epoch(), num_blocks_produced / MinimalEthSpec::slots_per_epoch(), "head should be at the expected epoch" ); assert_eq!( state.current_justified_epoch, state.current_epoch() - 1, "the head should be justified one behind the current epoch" ); assert_eq!( state.finalized_epoch, state.current_epoch() - 2, "the head should be finalized two behind the current epoch" ); } #[test] fn finalizes_with_two_thirds_participation() { let num_blocks_produced = MinimalEthSpec::slots_per_epoch() * 5; let harness = get_harness(VALIDATOR_COUNT); let two_thirds = (VALIDATOR_COUNT / 3) * 2; let attesters = (0..two_thirds).collect(); harness.extend_chain( num_blocks_produced as usize, BlockStrategy::OnCanonicalHead, AttestationStrategy::SomeValidators(attesters), ); let state = &harness.chain.head().beacon_state; assert_eq!( state.slot, num_blocks_produced, "head should be at the current slot" ); assert_eq!( state.current_epoch(), num_blocks_produced / MinimalEthSpec::slots_per_epoch(), "head should be at the expected epoch" ); // Note: the 2/3rds tests are not justifying the immediately prior epochs because the // `MIN_ATTESTATION_INCLUSION_DELAY` is preventing an adequate number of attestations being // included in blocks during that epoch. assert_eq!( state.current_justified_epoch, state.current_epoch() - 2, "the head should be justified two behind the current epoch" ); assert_eq!( state.finalized_epoch, state.current_epoch() - 4, "the head should be finalized three behind the current epoch" ); } #[test] fn does_not_finalize_with_less_than_two_thirds_participation() { let num_blocks_produced = MinimalEthSpec::slots_per_epoch() * 5; let harness = get_harness(VALIDATOR_COUNT); let two_thirds = (VALIDATOR_COUNT / 3) * 2; let less_than_two_thirds = two_thirds - 1; let attesters = (0..less_than_two_thirds).collect(); harness.extend_chain( num_blocks_produced as usize, BlockStrategy::OnCanonicalHead, AttestationStrategy::SomeValidators(attesters), ); let state = &harness.chain.head().beacon_state; assert_eq!( state.slot, num_blocks_produced, "head should be at the current slot" ); assert_eq!( state.current_epoch(), num_blocks_produced / MinimalEthSpec::slots_per_epoch(), "head should be at the expected epoch" ); assert_eq!( state.current_justified_epoch, 0, "no epoch should have been justified" ); assert_eq!( state.finalized_epoch, 0, "no epoch should have been finalized" ); } #[test] fn does_not_finalize_without_attestation() { let num_blocks_produced = MinimalEthSpec::slots_per_epoch() * 5; let harness = get_harness(VALIDATOR_COUNT); harness.extend_chain( num_blocks_produced as usize, BlockStrategy::OnCanonicalHead, AttestationStrategy::SomeValidators(vec![]), ); let state = &harness.chain.head().beacon_state; assert_eq!( state.slot, num_blocks_produced, "head should be at the current slot" ); assert_eq!( state.current_epoch(), num_blocks_produced / MinimalEthSpec::slots_per_epoch(), "head should be at the expected epoch" ); assert_eq!( state.current_justified_epoch, 0, "no epoch should have been justified" ); assert_eq!( state.finalized_epoch, 0, "no epoch should have been finalized" ); }