Fix on-disk consensus context format

This commit is contained in:
Michael Sproul
2024-04-18 14:26:09 +10:00
parent 49617f3e82
commit 3a16649023
4 changed files with 80 additions and 16 deletions

View File

@@ -8,8 +8,9 @@ use crate::{
use lru::LruCache;
use parking_lot::RwLock;
use ssz_derive::{Decode, Encode};
use state_processing::{BlockReplayer, ConsensusContext, StateProcessingStrategy};
use state_processing::{BlockReplayer, StateProcessingStrategy};
use std::sync::Arc;
use store::OnDiskConsensusContext;
use types::beacon_block_body::KzgCommitments;
use types::{ssz_tagged_signed_beacon_block, ssz_tagged_signed_beacon_block_arc};
use types::{BeaconState, BlindedPayload, ChainSpec, Epoch, EthSpec, Hash256, SignedBeaconBlock};
@@ -26,7 +27,7 @@ pub struct DietAvailabilityPendingExecutedBlock<E: EthSpec> {
parent_block: SignedBeaconBlock<E, BlindedPayload<E>>,
parent_eth1_finalization_data: Eth1FinalizationData,
confirmed_state_roots: Vec<Hash256>,
consensus_context: ConsensusContext<E>,
consensus_context: OnDiskConsensusContext,
payload_verification_outcome: PayloadVerificationOutcome,
}
@@ -94,7 +95,9 @@ impl<T: BeaconChainTypes> StateLRUCache<T> {
parent_block: executed_block.import_data.parent_block,
parent_eth1_finalization_data: executed_block.import_data.parent_eth1_finalization_data,
confirmed_state_roots: executed_block.import_data.confirmed_state_roots,
consensus_context: executed_block.import_data.consensus_context,
consensus_context: OnDiskConsensusContext::from_consensus_context(
&executed_block.import_data.consensus_context,
),
payload_verification_outcome: executed_block.payload_verification_outcome,
}
}
@@ -119,7 +122,9 @@ impl<T: BeaconChainTypes> StateLRUCache<T> {
parent_eth1_finalization_data: diet_executed_block
.parent_eth1_finalization_data,
confirmed_state_roots: diet_executed_block.confirmed_state_roots,
consensus_context: diet_executed_block.consensus_context,
consensus_context: diet_executed_block
.consensus_context
.into_consensus_context(),
},
payload_verification_outcome: diet_executed_block.payload_verification_outcome,
})
@@ -145,7 +150,9 @@ impl<T: BeaconChainTypes> StateLRUCache<T> {
parent_block: diet_executed_block.parent_block,
parent_eth1_finalization_data: diet_executed_block.parent_eth1_finalization_data,
confirmed_state_roots: diet_executed_block.confirmed_state_roots,
consensus_context: diet_executed_block.consensus_context,
consensus_context: diet_executed_block
.consensus_context
.into_consensus_context(),
},
payload_verification_outcome: diet_executed_block.payload_verification_outcome,
})
@@ -232,7 +239,9 @@ impl<E: EthSpec> From<AvailabilityPendingExecutedBlock<E>>
parent_block: value.import_data.parent_block,
parent_eth1_finalization_data: value.import_data.parent_eth1_finalization_data,
confirmed_state_roots: value.import_data.confirmed_state_roots,
consensus_context: value.import_data.consensus_context,
consensus_context: OnDiskConsensusContext::from_consensus_context(
&value.import_data.consensus_context,
),
payload_verification_outcome: value.payload_verification_outcome,
}
}

View File

@@ -0,0 +1,56 @@
use ssz_derive::{Decode, Encode};
use state_processing::ConsensusContext;
use types::{EthSpec, Hash256, Slot};
/// The consensus context is stored on disk as part of the data availability overflow cache.
///
/// We use this separate struct to keep the on-disk format stable in the presence of changes to the
/// in-memory `ConsensusContext`. You MUST NOT change the fields of this struct without
/// superstructing it and implementing a schema migration.
#[derive(Debug, PartialEq, Clone, Encode, Decode)]
pub struct OnDiskConsensusContext {
/// Slot to act as an identifier/safeguard
slot: Slot,
/// Proposer index of the block at `slot`.
proposer_index: Option<u64>,
/// Block root of the block at `slot`.
current_block_root: Option<Hash256>,
}
impl OnDiskConsensusContext {
pub fn from_consensus_context<E: EthSpec>(ctxt: &ConsensusContext<E>) -> Self {
// Match exhaustively on fields here so we are forced to *consider* updating the on-disk
// format when the `ConsensusContext` fields change.
let &ConsensusContext {
slot,
previous_epoch: _,
current_epoch: _,
proposer_index,
current_block_root,
indexed_attestations: _,
} = ctxt;
OnDiskConsensusContext {
slot,
proposer_index,
current_block_root,
}
}
pub fn into_consensus_context<E: EthSpec>(self) -> ConsensusContext<E> {
let OnDiskConsensusContext {
slot,
proposer_index,
current_block_root,
} = self;
let mut ctxt = ConsensusContext::new(slot);
if let Some(proposer_index) = proposer_index {
ctxt = ctxt.set_proposer_index(proposer_index);
}
if let Some(block_root) = current_block_root {
ctxt = ctxt.set_current_block_root(block_root);
}
ctxt
}
}

View File

@@ -14,6 +14,7 @@ mod chunk_writer;
pub mod chunked_iter;
pub mod chunked_vector;
pub mod config;
pub mod consensus_context;
pub mod errors;
mod forwards_iter;
mod garbage_collection;
@@ -30,6 +31,7 @@ pub mod iter;
pub use self::chunk_writer::ChunkWriter;
pub use self::config::StoreConfig;
pub use self::consensus_context::OnDiskConsensusContext;
pub use self::hot_cold_store::{HotColdDB, HotStateSummary, Split};
pub use self::leveldb_store::LevelDB;
pub use self::memory_store::MemoryStore;

View File

@@ -1,7 +1,6 @@
use crate::common::get_indexed_attestation;
use crate::per_block_processing::errors::{AttestationInvalid, BlockOperationError};
use crate::EpochCacheError;
use ssz_derive::{Decode, Encode};
use std::collections::{hash_map::Entry, HashMap};
use tree_hash::TreeHash;
use types::{
@@ -9,22 +8,20 @@ use types::{
ChainSpec, Epoch, EthSpec, Hash256, IndexedAttestation, SignedBeaconBlock, Slot,
};
#[derive(Debug, PartialEq, Clone, Encode, Decode)]
#[derive(Debug, PartialEq, Clone)]
pub struct ConsensusContext<E: EthSpec> {
/// Slot to act as an identifier/safeguard
slot: Slot,
pub slot: Slot,
/// Previous epoch of the `slot` precomputed for optimization purpose.
pub(crate) previous_epoch: Epoch,
pub previous_epoch: Epoch,
/// Current epoch of the `slot` precomputed for optimization purpose.
pub(crate) current_epoch: Epoch,
pub current_epoch: Epoch,
/// Proposer index of the block at `slot`.
proposer_index: Option<u64>,
pub proposer_index: Option<u64>,
/// Block root of the block at `slot`.
current_block_root: Option<Hash256>,
pub current_block_root: Option<Hash256>,
/// Cache of indexed attestations constructed during block processing.
/// We can skip serializing / deserializing this as the cache will just be rebuilt
#[ssz(skip_serializing, skip_deserializing)]
indexed_attestations:
pub indexed_attestations:
HashMap<(AttestationData, BitList<E::MaxValidatorsPerCommittee>), IndexedAttestation<E>>,
}