2019-06-23 21:45:34 +00:00
|
|
|
#![cfg(not(debug_assertions))]
|
|
|
|
|
2019-06-23 04:47:23 +00:00
|
|
|
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<ThreadSafeReducedTree<MemoryStore, MinimalEthSpec>, 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<usize> = (0..two_thirds).collect();
|
|
|
|
let faulty_validators: Vec<usize> = (two_thirds..VALIDATOR_COUNT).collect();
|
|
|
|
|
|
|
|
let initial_blocks = delay + 1;
|
|
|
|
let honest_fork_blocks = delay + 1;
|
|
|
|
let faulty_fork_blocks = delay + 2;
|
|
|
|
|
2019-06-24 05:31:36 +00:00
|
|
|
// Build an initial chain where all validators agree.
|
2019-06-23 04:47:23 +00:00
|
|
|
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"
|
|
|
|
);
|
|
|
|
}
|