0c3fdcd57c
* Renamed fork_choice::process_attestation_from_block * Processing attestation in fork choice * Retrieving state from store and checking signature * Looser check on beacon state validity. * Cleaned up get_attestation_state * Expanded fork choice api to provide latest validator message. * Checking if the an attestation contains a latest message * Correct process_attestation error handling. * Copy paste error in comment fixed. * Tidy ancestor iterators * Getting attestation slot via helper method * Refactored attestation creation in test utils * Revert "Refactored attestation creation in test utils" This reverts commit 4d277fe4239a7194758b18fb5c00dfe0b8231306. * Integration tests for free attestation processing * Implicit conflicts resolved. * formatting * Do first pass on Grants code * Add another attestation processing test * Tidy attestation processing * Remove old code fragment * Add non-compiling half finished changes * Simplify, fix bugs, add tests for chain iters * Remove attestation processing from op pool * Fix bug with fork choice, tidy * Fix overly restrictive check in fork choice. * Ensure committee cache is build during attn proc * Ignore unknown blocks at fork choice * Various minor fixes * Make fork choice write lock in to read lock * Remove unused method * Tidy comments * Fix attestation prod. target roots change * Fix compile error in store iters * Reject any attestation prior to finalization * Begin metrics refactor * Move beacon_chain to new metrics structure. * Make metrics not panic if already defined * Use global prometheus gather at rest api * Unify common metric fns into a crate * Add heavy metering to block processing * Remove hypen from prometheus metric name * Add more beacon chain metrics * Add beacon chain persistence metric * Prune op pool on finalization * Add extra prom beacon chain metrics * Prefix BeaconChain metrics with "beacon_" * Add more store metrics * Add basic metrics to libp2p * Add metrics to HTTP server * Remove old `http_server` crate * Update metrics names to be more like standard * Fix broken beacon chain metrics, add slot clock metrics * Add lighthouse_metrics gather fn * Remove http args * Fix wrong state given to op pool prune * Make prom metric names more consistent * Add more metrics, tidy existing metrics * Fix store block read metrics * Tidy attestation metrics * Fix minor PR comments * Fix minor PR comments * Remove duplicated attestation finalization check * Remove awkward `let` statement * Add first attempts at HTTP bootstrap * Add beacon_block methods to rest api * Fix serde for block.body.grafitti * Allow travis failures on beta (see desc) There's a non-backward compatible change in `cargo fmt`. Stable and beta do not agree. * Add network routes to API * Fix rustc warnings * Add best_slot method * Add --bootstrap arg to beacon node * Get bootstrapper working for ENR address * Store intermediate states during block processing * Allow bootstrapper to scrape libp2p address * Update bootstrapper libp2p address finding * Add comments * Tidy API to be more consistent with recent decisions * Address some review comments * Make BeaconChainTypes Send + Sync + 'static * Add `/network/listen_port` API endpoint * Abandon starting the node if libp2p doesn't start * Update bootstrapper for API changes * Remove unnecessary trait bounds
426 lines
14 KiB
Rust
426 lines
14 KiB
Rust
use crate::{BeaconChain, BeaconChainTypes, BlockProcessingOutcome};
|
|
use lmd_ghost::LmdGhost;
|
|
use sloggers::{null::NullLoggerBuilder, Build};
|
|
use slot_clock::SlotClock;
|
|
use slot_clock::TestingSlotClock;
|
|
use state_processing::per_slot_processing;
|
|
use std::marker::PhantomData;
|
|
use std::sync::Arc;
|
|
use store::MemoryStore;
|
|
use store::Store;
|
|
use tree_hash::{SignedRoot, TreeHash};
|
|
use types::{
|
|
test_utils::TestingBeaconStateBuilder, AggregateSignature, Attestation,
|
|
AttestationDataAndCustodyBit, BeaconBlock, BeaconState, BitList, ChainSpec, Domain, EthSpec,
|
|
Hash256, Keypair, RelativeEpoch, SecretKey, Signature, Slot,
|
|
};
|
|
|
|
pub use crate::persisted_beacon_chain::{PersistedBeaconChain, BEACON_CHAIN_DB_KEY};
|
|
|
|
/// Indicates how the `BeaconChainHarness` should produce blocks.
|
|
#[derive(Clone, Copy, Debug)]
|
|
pub enum BlockStrategy {
|
|
/// Produce blocks upon the canonical head (normal case).
|
|
OnCanonicalHead,
|
|
/// Ignore the canonical head and produce blocks upon the block at the given slot.
|
|
///
|
|
/// Useful for simulating forks.
|
|
ForkCanonicalChainAt {
|
|
/// The slot of the parent of the first block produced.
|
|
previous_slot: Slot,
|
|
/// The slot of the first block produced (must be higher than `previous_slot`.
|
|
first_slot: Slot,
|
|
},
|
|
}
|
|
|
|
/// Indicates how the `BeaconChainHarness` should produce attestations.
|
|
#[derive(Clone, Debug)]
|
|
pub enum AttestationStrategy {
|
|
/// All validators attest to whichever block the `BeaconChainHarness` has produced.
|
|
AllValidators,
|
|
/// Only the given validators should attest. All others should fail to produce attestations.
|
|
SomeValidators(Vec<usize>),
|
|
}
|
|
|
|
/// Used to make the `BeaconChainHarness` generic over some types.
|
|
pub struct CommonTypes<L, E>
|
|
where
|
|
L: LmdGhost<MemoryStore, E>,
|
|
E: EthSpec,
|
|
{
|
|
_phantom_l: PhantomData<L>,
|
|
_phantom_e: PhantomData<E>,
|
|
}
|
|
|
|
impl<L, E> BeaconChainTypes for CommonTypes<L, E>
|
|
where
|
|
L: LmdGhost<MemoryStore, E> + 'static,
|
|
E: EthSpec,
|
|
{
|
|
type Store = MemoryStore;
|
|
type SlotClock = TestingSlotClock;
|
|
type LmdGhost = L;
|
|
type EthSpec = E;
|
|
}
|
|
|
|
/// A testing harness which can instantiate a `BeaconChain` and populate it with blocks and
|
|
/// attestations.
|
|
///
|
|
/// Used for testing.
|
|
pub struct BeaconChainHarness<L, E>
|
|
where
|
|
L: LmdGhost<MemoryStore, E> + 'static,
|
|
E: EthSpec,
|
|
{
|
|
pub chain: BeaconChain<CommonTypes<L, E>>,
|
|
pub keypairs: Vec<Keypair>,
|
|
pub spec: ChainSpec,
|
|
}
|
|
|
|
impl<L, E> BeaconChainHarness<L, E>
|
|
where
|
|
L: LmdGhost<MemoryStore, E>,
|
|
E: EthSpec,
|
|
{
|
|
/// Instantiate a new harness with `validator_count` initial validators.
|
|
pub fn new(validator_count: usize) -> Self {
|
|
let state_builder = TestingBeaconStateBuilder::from_default_keypairs_file_if_exists(
|
|
validator_count,
|
|
&E::default_spec(),
|
|
);
|
|
let (genesis_state, keypairs) = state_builder.build();
|
|
|
|
Self::from_state_and_keypairs(genesis_state, keypairs)
|
|
}
|
|
|
|
/// Instantiate a new harness with an initial validator for each key supplied.
|
|
pub fn from_keypairs(keypairs: Vec<Keypair>) -> Self {
|
|
let state_builder = TestingBeaconStateBuilder::from_keypairs(keypairs, &E::default_spec());
|
|
let (genesis_state, keypairs) = state_builder.build();
|
|
|
|
Self::from_state_and_keypairs(genesis_state, keypairs)
|
|
}
|
|
|
|
/// Instantiate a new harness with the given genesis state and a keypair for each of the
|
|
/// initial validators in the given state.
|
|
pub fn from_state_and_keypairs(genesis_state: BeaconState<E>, keypairs: Vec<Keypair>) -> Self {
|
|
let spec = E::default_spec();
|
|
|
|
let store = Arc::new(MemoryStore::open());
|
|
|
|
let mut genesis_block = BeaconBlock::empty(&spec);
|
|
genesis_block.state_root = Hash256::from_slice(&genesis_state.tree_hash_root());
|
|
|
|
let builder = NullLoggerBuilder;
|
|
let log = builder.build().expect("logger should build");
|
|
|
|
// Slot clock
|
|
let slot_clock = TestingSlotClock::new(
|
|
spec.genesis_slot,
|
|
genesis_state.genesis_time,
|
|
spec.seconds_per_slot,
|
|
);
|
|
|
|
let chain = BeaconChain::from_genesis(
|
|
store,
|
|
slot_clock,
|
|
genesis_state,
|
|
genesis_block,
|
|
spec.clone(),
|
|
log,
|
|
)
|
|
.expect("Terminate if beacon chain generation fails");
|
|
|
|
Self {
|
|
chain,
|
|
keypairs,
|
|
spec,
|
|
}
|
|
}
|
|
|
|
/// Advance the slot of the `BeaconChain`.
|
|
///
|
|
/// Does not produce blocks or attestations.
|
|
pub fn advance_slot(&self) {
|
|
self.chain.slot_clock.advance_slot();
|
|
self.chain.catchup_state().expect("should catchup state");
|
|
}
|
|
|
|
/// Extend the `BeaconChain` with some blocks and attestations. Returns the root of the
|
|
/// last-produced block (the head of the chain).
|
|
///
|
|
/// Chain will be extended by `num_blocks` blocks.
|
|
///
|
|
/// The `block_strategy` dictates where the new blocks will be placed.
|
|
///
|
|
/// The `attestation_strategy` dictates which validators will attest to the newly created
|
|
/// blocks.
|
|
pub fn extend_chain(
|
|
&self,
|
|
num_blocks: usize,
|
|
block_strategy: BlockStrategy,
|
|
attestation_strategy: AttestationStrategy,
|
|
) -> Hash256 {
|
|
let mut state = {
|
|
// Determine the slot for the first block (or skipped block).
|
|
let state_slot = match block_strategy {
|
|
BlockStrategy::OnCanonicalHead => self.chain.read_slot_clock().unwrap() - 1,
|
|
BlockStrategy::ForkCanonicalChainAt { previous_slot, .. } => previous_slot,
|
|
};
|
|
|
|
self.get_state_at_slot(state_slot)
|
|
};
|
|
|
|
// Determine the first slot where a block should be built.
|
|
let mut slot = match block_strategy {
|
|
BlockStrategy::OnCanonicalHead => self.chain.read_slot_clock().unwrap(),
|
|
BlockStrategy::ForkCanonicalChainAt { first_slot, .. } => first_slot,
|
|
};
|
|
|
|
let mut head_block_root = None;
|
|
|
|
for _ in 0..num_blocks {
|
|
while self.chain.read_slot_clock().expect("should have a slot") < slot {
|
|
self.advance_slot();
|
|
}
|
|
|
|
let (block, new_state) = self.build_block(state.clone(), slot, block_strategy);
|
|
|
|
let outcome = self
|
|
.chain
|
|
.process_block(block)
|
|
.expect("should not error during block processing");
|
|
|
|
if let BlockProcessingOutcome::Processed { block_root } = outcome {
|
|
head_block_root = Some(block_root);
|
|
|
|
self.add_free_attestations(&attestation_strategy, &new_state, block_root, slot);
|
|
} else {
|
|
panic!("block should be successfully processed: {:?}", outcome);
|
|
}
|
|
|
|
state = new_state;
|
|
slot += 1;
|
|
}
|
|
|
|
head_block_root.expect("did not produce any blocks")
|
|
}
|
|
|
|
fn get_state_at_slot(&self, state_slot: Slot) -> BeaconState<E> {
|
|
let state_root = self
|
|
.chain
|
|
.rev_iter_state_roots()
|
|
.find(|(_hash, slot)| *slot == state_slot)
|
|
.map(|(hash, _slot)| hash)
|
|
.expect("could not find state root");
|
|
|
|
self.chain
|
|
.store
|
|
.get(&state_root)
|
|
.expect("should read db")
|
|
.expect("should find state root")
|
|
}
|
|
|
|
/// Returns a newly created block, signed by the proposer for the given slot.
|
|
fn build_block(
|
|
&self,
|
|
mut state: BeaconState<E>,
|
|
slot: Slot,
|
|
block_strategy: BlockStrategy,
|
|
) -> (BeaconBlock<E>, BeaconState<E>) {
|
|
if slot < state.slot {
|
|
panic!("produce slot cannot be prior to the state slot");
|
|
}
|
|
|
|
while state.slot < slot {
|
|
per_slot_processing(&mut state, &self.spec)
|
|
.expect("should be able to advance state to slot");
|
|
}
|
|
|
|
state.build_all_caches(&self.spec).unwrap();
|
|
|
|
let proposer_index = match block_strategy {
|
|
BlockStrategy::OnCanonicalHead => self
|
|
.chain
|
|
.block_proposer(slot)
|
|
.expect("should get block proposer from chain"),
|
|
_ => state
|
|
.get_beacon_proposer_index(slot, RelativeEpoch::Current, &self.spec)
|
|
.expect("should get block proposer from state"),
|
|
};
|
|
|
|
let sk = &self.keypairs[proposer_index].sk;
|
|
let fork = &state.fork.clone();
|
|
|
|
let randao_reveal = {
|
|
let epoch = slot.epoch(E::slots_per_epoch());
|
|
let message = epoch.tree_hash_root();
|
|
let domain = self.spec.get_domain(epoch, Domain::Randao, fork);
|
|
Signature::new(&message, domain, sk)
|
|
};
|
|
|
|
let (mut block, state) = self
|
|
.chain
|
|
.produce_block_on_state(state, slot, randao_reveal)
|
|
.expect("should produce block");
|
|
|
|
block.signature = {
|
|
let message = block.signed_root();
|
|
let epoch = block.slot.epoch(E::slots_per_epoch());
|
|
let domain = self.spec.get_domain(epoch, Domain::BeaconProposer, fork);
|
|
Signature::new(&message, domain, sk)
|
|
};
|
|
|
|
(block, state)
|
|
}
|
|
|
|
/// Adds attestations to the `BeaconChain` operations pool and fork choice.
|
|
///
|
|
/// The `attestation_strategy` dictates which validators should attest.
|
|
fn add_free_attestations(
|
|
&self,
|
|
attestation_strategy: &AttestationStrategy,
|
|
state: &BeaconState<E>,
|
|
head_block_root: Hash256,
|
|
head_block_slot: Slot,
|
|
) {
|
|
self.get_free_attestations(
|
|
attestation_strategy,
|
|
state,
|
|
head_block_root,
|
|
head_block_slot,
|
|
)
|
|
.into_iter()
|
|
.for_each(|attestation| {
|
|
self.chain
|
|
.process_attestation(attestation)
|
|
.expect("should process attestation");
|
|
});
|
|
}
|
|
|
|
/// Generates a `Vec<Attestation>` for some attestation strategy and head_block.
|
|
pub fn get_free_attestations(
|
|
&self,
|
|
attestation_strategy: &AttestationStrategy,
|
|
state: &BeaconState<E>,
|
|
head_block_root: Hash256,
|
|
head_block_slot: Slot,
|
|
) -> Vec<Attestation<E>> {
|
|
let spec = &self.spec;
|
|
let fork = &state.fork;
|
|
|
|
let attesting_validators: Vec<usize> = match attestation_strategy {
|
|
AttestationStrategy::AllValidators => (0..self.keypairs.len()).collect(),
|
|
AttestationStrategy::SomeValidators(vec) => vec.clone(),
|
|
};
|
|
|
|
let mut vec = vec![];
|
|
|
|
state
|
|
.get_crosslink_committees_at_slot(state.slot)
|
|
.expect("should get committees")
|
|
.iter()
|
|
.for_each(|cc| {
|
|
let committee_size = cc.committee.len();
|
|
|
|
for (i, validator_index) in cc.committee.iter().enumerate() {
|
|
// Note: searching this array is worst-case `O(n)`. A hashset could be a better
|
|
// alternative.
|
|
if attesting_validators.contains(validator_index) {
|
|
let data = self
|
|
.chain
|
|
.produce_attestation_data_for_block(
|
|
cc.shard,
|
|
head_block_root,
|
|
head_block_slot,
|
|
state,
|
|
)
|
|
.expect("should produce attestation data");
|
|
|
|
let mut aggregation_bits = BitList::with_capacity(committee_size).unwrap();
|
|
aggregation_bits.set(i, true).unwrap();
|
|
let custody_bits = BitList::with_capacity(committee_size).unwrap();
|
|
|
|
let signature = {
|
|
let message = AttestationDataAndCustodyBit {
|
|
data: data.clone(),
|
|
custody_bit: false,
|
|
}
|
|
.tree_hash_root();
|
|
|
|
let domain =
|
|
spec.get_domain(data.target.epoch, Domain::Attestation, fork);
|
|
|
|
let mut agg_sig = AggregateSignature::new();
|
|
agg_sig.add(&Signature::new(
|
|
&message,
|
|
domain,
|
|
self.get_sk(*validator_index),
|
|
));
|
|
|
|
agg_sig
|
|
};
|
|
|
|
vec.push(Attestation {
|
|
aggregation_bits,
|
|
data,
|
|
custody_bits,
|
|
signature,
|
|
})
|
|
}
|
|
}
|
|
});
|
|
|
|
vec
|
|
}
|
|
|
|
/// Creates two forks:
|
|
///
|
|
/// - The "honest" fork: created by the `honest_validators` who have built `honest_fork_blocks`
|
|
/// on the head
|
|
/// - The "faulty" fork: created by the `faulty_validators` who skipped a slot and
|
|
/// then built `faulty_fork_blocks`.
|
|
///
|
|
/// Returns `(honest_head, faulty_head)`, the roots of the blocks at the top of each chain.
|
|
pub fn generate_two_forks_by_skipping_a_block(
|
|
&self,
|
|
honest_validators: &[usize],
|
|
faulty_validators: &[usize],
|
|
honest_fork_blocks: usize,
|
|
faulty_fork_blocks: usize,
|
|
) -> (Hash256, Hash256) {
|
|
let initial_head_slot = self.chain.head().beacon_block.slot;
|
|
|
|
// Move to the next slot so we may produce some more blocks on the head.
|
|
self.advance_slot();
|
|
|
|
// Extend the chain with blocks where only honest validators agree.
|
|
let honest_head = self.extend_chain(
|
|
honest_fork_blocks,
|
|
BlockStrategy::OnCanonicalHead,
|
|
AttestationStrategy::SomeValidators(honest_validators.to_vec()),
|
|
);
|
|
|
|
// Go back to the last block where all agreed, and build blocks upon it where only faulty nodes
|
|
// agree.
|
|
let faulty_head = self.extend_chain(
|
|
faulty_fork_blocks,
|
|
BlockStrategy::ForkCanonicalChainAt {
|
|
previous_slot: initial_head_slot,
|
|
// `initial_head_slot + 2` means one slot is skipped.
|
|
first_slot: initial_head_slot + 2,
|
|
},
|
|
AttestationStrategy::SomeValidators(faulty_validators.to_vec()),
|
|
);
|
|
|
|
assert!(honest_head != faulty_head, "forks should be distinct");
|
|
|
|
(honest_head, faulty_head)
|
|
}
|
|
|
|
/// Returns the secret key for the given validator index.
|
|
fn get_sk(&self, validator_index: usize) -> &SecretKey {
|
|
&self.keypairs[validator_index].sk
|
|
}
|
|
}
|