//! Provides tools for checking if a node is ready for the Electra upgrade and following merge //! transition. use crate::{BeaconChain, BeaconChainTypes}; use execution_layer::http::{ENGINE_GET_PAYLOAD_V4, ENGINE_NEW_PAYLOAD_V4}; use serde::{Deserialize, Serialize}; use std::fmt; use std::time::Duration; use types::*; /// The time before the Electra fork when we will start issuing warnings about preparation. use super::bellatrix_readiness::SECONDS_IN_A_WEEK; pub const ELECTRA_READINESS_PREPARATION_SECONDS: u64 = SECONDS_IN_A_WEEK * 2; pub const ENGINE_CAPABILITIES_REFRESH_INTERVAL: u64 = 300; #[derive(Debug, Serialize, Deserialize)] #[serde(rename_all = "snake_case")] #[serde(tag = "type")] pub enum ElectraReadiness { /// The execution engine is electra-enabled (as far as we can tell) Ready, /// We are connected to an execution engine which doesn't support the V4 engine api methods V4MethodsNotSupported { error: String }, /// The transition configuration with the EL failed, there might be a problem with /// connectivity, authentication or a difference in configuration. ExchangeCapabilitiesFailed { error: String }, /// The user has not configured an execution endpoint NoExecutionEndpoint, } impl fmt::Display for ElectraReadiness { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match self { ElectraReadiness::Ready => { write!(f, "This node appears ready for Electra.") } ElectraReadiness::ExchangeCapabilitiesFailed { error } => write!( f, "Could not exchange capabilities with the \ execution endpoint: {}", error ), ElectraReadiness::NoExecutionEndpoint => write!( f, "The --execution-endpoint flag is not specified, this is a \ requirement post-merge" ), ElectraReadiness::V4MethodsNotSupported { error } => write!( f, "Execution endpoint does not support Electra methods: {}", error ), } } } impl BeaconChain { /// Returns `true` if electra epoch is set and Electra fork has occurred or will /// occur within `ELECTRA_READINESS_PREPARATION_SECONDS` pub fn is_time_to_prepare_for_electra(&self, current_slot: Slot) -> bool { if let Some(electra_epoch) = self.spec.electra_fork_epoch { let electra_slot = electra_epoch.start_slot(T::EthSpec::slots_per_epoch()); let electra_readiness_preparation_slots = ELECTRA_READINESS_PREPARATION_SECONDS / self.spec.seconds_per_slot; // Return `true` if Electra has happened or is within the preparation time. current_slot + electra_readiness_preparation_slots > electra_slot } else { // The Electra fork epoch has not been defined yet, no need to prepare. false } } /// Attempts to connect to the EL and confirm that it is ready for electra. pub async fn check_electra_readiness(&self) -> ElectraReadiness { if let Some(el) = self.execution_layer.as_ref() { match el .get_engine_capabilities(Some(Duration::from_secs( ENGINE_CAPABILITIES_REFRESH_INTERVAL, ))) .await { Err(e) => { // The EL was either unreachable or responded with an error ElectraReadiness::ExchangeCapabilitiesFailed { error: format!("{:?}", e), } } Ok(capabilities) => { let mut missing_methods = String::from("Required Methods Unsupported:"); let mut all_good = true; if !capabilities.get_payload_v4 { missing_methods.push(' '); missing_methods.push_str(ENGINE_GET_PAYLOAD_V4); all_good = false; } if !capabilities.new_payload_v4 { missing_methods.push(' '); missing_methods.push_str(ENGINE_NEW_PAYLOAD_V4); all_good = false; } if all_good { ElectraReadiness::Ready } else { ElectraReadiness::V4MethodsNotSupported { error: missing_methods, } } } } } else { ElectraReadiness::NoExecutionEndpoint } } }