mirror of
https://github.com/hl-archive-node/nanoreth.git
synced 2025-12-06 10:59:55 +00:00
chore: improve TestConsensusEngineBuilder ergonomics (#3891)
This commit is contained in:
@ -72,6 +72,9 @@ mod metrics;
|
||||
pub(crate) mod prune;
|
||||
pub(crate) mod sync;
|
||||
|
||||
#[cfg(any(test, feature = "test-utils"))]
|
||||
pub mod test_utils;
|
||||
|
||||
/// The maximum number of invalid headers that can be tracked by the engine.
|
||||
const MAX_INVALID_HEADERS: u32 = 512u32;
|
||||
|
||||
@ -1709,413 +1712,17 @@ where
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use crate::{BeaconForkChoiceUpdateError, BeaconOnNewPayloadError};
|
||||
use crate::{
|
||||
test_utils::{spawn_consensus_engine, TestConsensusEngineBuilder},
|
||||
BeaconForkChoiceUpdateError,
|
||||
};
|
||||
use assert_matches::assert_matches;
|
||||
use reth_blockchain_tree::{
|
||||
config::BlockchainTreeConfig, externals::TreeExternals, post_state::PostState,
|
||||
BlockchainTree, ShareableBlockchainTree,
|
||||
};
|
||||
use reth_db::{test_utils::create_test_rw_db, DatabaseEnv};
|
||||
use reth_downloaders::{
|
||||
bodies::bodies::BodiesDownloaderBuilder,
|
||||
headers::reverse_headers::ReverseHeadersDownloaderBuilder,
|
||||
};
|
||||
use reth_interfaces::{
|
||||
consensus::Consensus,
|
||||
p2p::either::EitherDownloader,
|
||||
sync::NoopSyncStateUpdater,
|
||||
test_utils::{NoopFullBlockClient, TestConsensus},
|
||||
};
|
||||
use reth_payload_builder::test_utils::spawn_test_payload_service;
|
||||
use reth_primitives::{
|
||||
stage::StageCheckpoint, ChainSpec, ChainSpecBuilder, PruneModes, H256, MAINNET,
|
||||
};
|
||||
use reth_provider::{
|
||||
providers::BlockchainProvider, test_utils::TestExecutorFactory, BlockExecutor, BlockWriter,
|
||||
ExecutorFactory, ProviderFactory, StateProvider,
|
||||
};
|
||||
use reth_prune::BatchSizes;
|
||||
use reth_revm::Factory;
|
||||
use reth_rpc_types::engine::{
|
||||
ExecutionPayload, ForkchoiceState, ForkchoiceUpdated, PayloadStatus,
|
||||
};
|
||||
use reth_stages::{
|
||||
sets::DefaultStages, stages::HeaderSyncMode, test_utils::TestStages, ExecOutput,
|
||||
PipelineError, StageError,
|
||||
};
|
||||
use reth_tasks::TokioTaskExecutor;
|
||||
use reth_primitives::{stage::StageCheckpoint, ChainSpec, ChainSpecBuilder, H256, MAINNET};
|
||||
use reth_provider::{BlockWriter, ProviderFactory};
|
||||
use reth_rpc_types::engine::{ForkchoiceState, ForkchoiceUpdated, PayloadStatus};
|
||||
use reth_stages::{ExecOutput, PipelineError, StageError};
|
||||
use std::{collections::VecDeque, sync::Arc, time::Duration};
|
||||
use tokio::sync::{
|
||||
oneshot::{self, error::TryRecvError},
|
||||
watch,
|
||||
};
|
||||
|
||||
type TestBeaconConsensusEngine<Client> = BeaconConsensusEngine<
|
||||
Arc<DatabaseEnv>,
|
||||
BlockchainProvider<
|
||||
Arc<DatabaseEnv>,
|
||||
ShareableBlockchainTree<
|
||||
Arc<DatabaseEnv>,
|
||||
Arc<TestConsensus>,
|
||||
EitherExecutorFactory<TestExecutorFactory, Factory>,
|
||||
>,
|
||||
>,
|
||||
Arc<EitherDownloader<Client, NoopFullBlockClient>>,
|
||||
>;
|
||||
|
||||
struct TestEnv<DB> {
|
||||
db: DB,
|
||||
// Keep the tip receiver around, so it's not dropped.
|
||||
#[allow(dead_code)]
|
||||
tip_rx: watch::Receiver<H256>,
|
||||
engine_handle: BeaconConsensusEngineHandle,
|
||||
}
|
||||
|
||||
impl<DB> TestEnv<DB> {
|
||||
fn new(
|
||||
db: DB,
|
||||
tip_rx: watch::Receiver<H256>,
|
||||
engine_handle: BeaconConsensusEngineHandle,
|
||||
) -> Self {
|
||||
Self { db, tip_rx, engine_handle }
|
||||
}
|
||||
|
||||
async fn send_new_payload(
|
||||
&self,
|
||||
payload: ExecutionPayload,
|
||||
) -> Result<PayloadStatus, BeaconOnNewPayloadError> {
|
||||
self.engine_handle.new_payload(payload).await
|
||||
}
|
||||
|
||||
/// Sends the `ExecutionPayload` message to the consensus engine and retries if the engine
|
||||
/// is syncing.
|
||||
async fn send_new_payload_retry_on_syncing(
|
||||
&self,
|
||||
payload: ExecutionPayload,
|
||||
) -> Result<PayloadStatus, BeaconOnNewPayloadError> {
|
||||
loop {
|
||||
let result = self.send_new_payload(payload.clone()).await?;
|
||||
if !result.is_syncing() {
|
||||
return Ok(result)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
async fn send_forkchoice_updated(
|
||||
&self,
|
||||
state: ForkchoiceState,
|
||||
) -> Result<ForkchoiceUpdated, BeaconForkChoiceUpdateError> {
|
||||
self.engine_handle.fork_choice_updated(state, None).await
|
||||
}
|
||||
|
||||
/// Sends the `ForkchoiceUpdated` message to the consensus engine and retries if the engine
|
||||
/// is syncing.
|
||||
async fn send_forkchoice_retry_on_syncing(
|
||||
&self,
|
||||
state: ForkchoiceState,
|
||||
) -> Result<ForkchoiceUpdated, BeaconForkChoiceUpdateError> {
|
||||
loop {
|
||||
let result = self.engine_handle.fork_choice_updated(state, None).await?;
|
||||
if !result.is_syncing() {
|
||||
return Ok(result)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Represents either test pipeline outputs, or real pipeline configuration.
|
||||
enum TestPipelineConfig {
|
||||
/// Test pipeline outputs.
|
||||
Test(VecDeque<Result<ExecOutput, StageError>>),
|
||||
/// Real pipeline configuration.
|
||||
Real,
|
||||
}
|
||||
|
||||
impl Default for TestPipelineConfig {
|
||||
fn default() -> Self {
|
||||
Self::Test(VecDeque::new())
|
||||
}
|
||||
}
|
||||
|
||||
/// Represents either test executor results, or real executor configuration.
|
||||
enum TestExecutorConfig {
|
||||
/// Test executor results.
|
||||
Test(Vec<PostState>),
|
||||
/// Real executor configuration.
|
||||
Real,
|
||||
}
|
||||
|
||||
impl Default for TestExecutorConfig {
|
||||
fn default() -> Self {
|
||||
Self::Test(Vec::new())
|
||||
}
|
||||
}
|
||||
|
||||
/// A type that represents one of two possible executor factories.
|
||||
#[derive(Debug, Clone)]
|
||||
enum EitherExecutorFactory<A: ExecutorFactory, B: ExecutorFactory> {
|
||||
/// The first factory variant
|
||||
Left(A),
|
||||
/// The second factory variant
|
||||
Right(B),
|
||||
}
|
||||
|
||||
// A type that represents one of two possible BlockExecutor types.
|
||||
#[derive(Debug)]
|
||||
enum EitherBlockExecutor<A, B> {
|
||||
/// The first executor variant
|
||||
Left(A),
|
||||
/// The second executor variant
|
||||
Right(B),
|
||||
}
|
||||
|
||||
impl<A, B, SP> BlockExecutor<SP> for EitherBlockExecutor<A, B>
|
||||
where
|
||||
A: BlockExecutor<SP>,
|
||||
B: BlockExecutor<SP>,
|
||||
SP: StateProvider,
|
||||
{
|
||||
fn execute(
|
||||
&mut self,
|
||||
block: &reth_primitives::Block,
|
||||
total_difficulty: U256,
|
||||
senders: Option<Vec<reth_primitives::Address>>,
|
||||
) -> Result<PostState, BlockExecutionError> {
|
||||
match self {
|
||||
EitherBlockExecutor::Left(a) => a.execute(block, total_difficulty, senders),
|
||||
EitherBlockExecutor::Right(b) => b.execute(block, total_difficulty, senders),
|
||||
}
|
||||
}
|
||||
|
||||
fn execute_and_verify_receipt(
|
||||
&mut self,
|
||||
block: &reth_primitives::Block,
|
||||
total_difficulty: U256,
|
||||
senders: Option<Vec<reth_primitives::Address>>,
|
||||
) -> Result<PostState, BlockExecutionError> {
|
||||
match self {
|
||||
EitherBlockExecutor::Left(a) => {
|
||||
a.execute_and_verify_receipt(block, total_difficulty, senders)
|
||||
}
|
||||
EitherBlockExecutor::Right(b) => {
|
||||
b.execute_and_verify_receipt(block, total_difficulty, senders)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<A, B> ExecutorFactory for EitherExecutorFactory<A, B>
|
||||
where
|
||||
A: ExecutorFactory,
|
||||
B: ExecutorFactory,
|
||||
{
|
||||
type Executor<T: StateProvider> = EitherBlockExecutor<A::Executor<T>, B::Executor<T>>;
|
||||
|
||||
fn chain_spec(&self) -> &ChainSpec {
|
||||
match self {
|
||||
EitherExecutorFactory::Left(a) => a.chain_spec(),
|
||||
EitherExecutorFactory::Right(b) => b.chain_spec(),
|
||||
}
|
||||
}
|
||||
|
||||
fn with_sp<SP: reth_provider::StateProvider>(&self, sp: SP) -> Self::Executor<SP> {
|
||||
match self {
|
||||
EitherExecutorFactory::Left(a) => EitherBlockExecutor::Left(a.with_sp(sp)),
|
||||
EitherExecutorFactory::Right(b) => EitherBlockExecutor::Right(b.with_sp(sp)),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// A builder for `TestConsensusEngine`, allows configuration of mocked pipeline outputs and
|
||||
/// mocked executor results.
|
||||
struct TestConsensusEngineBuilder<Client> {
|
||||
chain_spec: Arc<ChainSpec>,
|
||||
pipeline_config: TestPipelineConfig,
|
||||
executor_config: TestExecutorConfig,
|
||||
pipeline_run_threshold: Option<u64>,
|
||||
max_block: Option<BlockNumber>,
|
||||
client: Option<Client>,
|
||||
}
|
||||
|
||||
impl<Client> TestConsensusEngineBuilder<Client>
|
||||
where
|
||||
Client: HeadersClient + BodiesClient + 'static,
|
||||
{
|
||||
/// Create a new `TestConsensusEngineBuilder` with the given `ChainSpec`.
|
||||
fn new(chain_spec: Arc<ChainSpec>) -> Self {
|
||||
Self {
|
||||
chain_spec,
|
||||
pipeline_config: Default::default(),
|
||||
executor_config: Default::default(),
|
||||
pipeline_run_threshold: None,
|
||||
client: None,
|
||||
max_block: None,
|
||||
}
|
||||
}
|
||||
|
||||
/// Set the pipeline execution outputs to use for the test consensus engine.
|
||||
fn with_pipeline_exec_outputs(
|
||||
mut self,
|
||||
pipeline_exec_outputs: VecDeque<Result<ExecOutput, StageError>>,
|
||||
) -> Self {
|
||||
self.pipeline_config = TestPipelineConfig::Test(pipeline_exec_outputs);
|
||||
self
|
||||
}
|
||||
|
||||
/// Set the executor results to use for the test consensus engine.
|
||||
fn with_executor_results(mut self, executor_results: Vec<PostState>) -> Self {
|
||||
self.executor_config = TestExecutorConfig::Test(executor_results);
|
||||
self
|
||||
}
|
||||
|
||||
/// Sets the max block for the pipeline to run.
|
||||
fn with_max_block(mut self, max_block: BlockNumber) -> Self {
|
||||
self.max_block = Some(max_block);
|
||||
self
|
||||
}
|
||||
|
||||
/// Uses the real pipeline instead of a pipeline with empty exec outputs.
|
||||
fn with_real_pipeline(mut self) -> Self {
|
||||
self.pipeline_config = TestPipelineConfig::Real;
|
||||
self
|
||||
}
|
||||
|
||||
/// Uses the real executor instead of a executor with empty results.
|
||||
fn with_real_executor(mut self) -> Self {
|
||||
self.executor_config = TestExecutorConfig::Real;
|
||||
self
|
||||
}
|
||||
|
||||
/// Sets the client to use for network operations.
|
||||
#[allow(dead_code)]
|
||||
fn with_client(mut self, client: Client) -> Self {
|
||||
self.client = Some(client);
|
||||
self
|
||||
}
|
||||
|
||||
/// Disables blockchain tree driven sync. This is the same as setting the pipeline run
|
||||
/// threshold to 0.
|
||||
fn disable_blockchain_tree_sync(mut self) -> Self {
|
||||
self.pipeline_run_threshold = Some(0);
|
||||
self
|
||||
}
|
||||
|
||||
/// Builds the test consensus engine into a `TestConsensusEngine` and `TestEnv`.
|
||||
fn build(self) -> (TestBeaconConsensusEngine<Client>, TestEnv<Arc<DatabaseEnv>>) {
|
||||
reth_tracing::init_test_tracing();
|
||||
let db = create_test_rw_db();
|
||||
let consensus = Arc::new(TestConsensus::default());
|
||||
let payload_builder = spawn_test_payload_service();
|
||||
|
||||
// use either noop client or a user provided client (for example TestFullBlockClient)
|
||||
let client = Arc::new(
|
||||
self.client
|
||||
.map(EitherDownloader::Left)
|
||||
.unwrap_or_else(|| EitherDownloader::Right(NoopFullBlockClient::default())),
|
||||
);
|
||||
|
||||
// use either test executor or real executor
|
||||
let executor_factory = match self.executor_config {
|
||||
TestExecutorConfig::Test(results) => {
|
||||
let executor_factory = TestExecutorFactory::new(self.chain_spec.clone());
|
||||
executor_factory.extend(results);
|
||||
EitherExecutorFactory::Left(executor_factory)
|
||||
}
|
||||
TestExecutorConfig::Real => {
|
||||
EitherExecutorFactory::Right(Factory::new(self.chain_spec.clone()))
|
||||
}
|
||||
};
|
||||
|
||||
// Setup pipeline
|
||||
let (tip_tx, tip_rx) = watch::channel(H256::default());
|
||||
let mut pipeline = match self.pipeline_config {
|
||||
TestPipelineConfig::Test(outputs) => Pipeline::builder()
|
||||
.add_stages(TestStages::new(outputs, Default::default()))
|
||||
.with_tip_sender(tip_tx),
|
||||
TestPipelineConfig::Real => {
|
||||
let header_downloader = ReverseHeadersDownloaderBuilder::default()
|
||||
.build(client.clone(), consensus.clone())
|
||||
.into_task();
|
||||
|
||||
let body_downloader = BodiesDownloaderBuilder::default()
|
||||
.build(client.clone(), consensus.clone(), db.clone())
|
||||
.into_task();
|
||||
|
||||
Pipeline::builder().add_stages(DefaultStages::new(
|
||||
HeaderSyncMode::Tip(tip_rx.clone()),
|
||||
Arc::clone(&consensus) as Arc<dyn Consensus>,
|
||||
header_downloader,
|
||||
body_downloader,
|
||||
executor_factory.clone(),
|
||||
))
|
||||
}
|
||||
};
|
||||
|
||||
if let Some(max_block) = self.max_block {
|
||||
pipeline = pipeline.with_max_block(max_block);
|
||||
}
|
||||
|
||||
let pipeline = pipeline.build(db.clone(), self.chain_spec.clone());
|
||||
|
||||
// Setup blockchain tree
|
||||
let externals = TreeExternals::new(
|
||||
db.clone(),
|
||||
consensus,
|
||||
executor_factory,
|
||||
self.chain_spec.clone(),
|
||||
);
|
||||
let config = BlockchainTreeConfig::new(1, 2, 3, 2);
|
||||
let (canon_state_notification_sender, _) = tokio::sync::broadcast::channel(3);
|
||||
let tree = ShareableBlockchainTree::new(
|
||||
BlockchainTree::new(externals, canon_state_notification_sender, config)
|
||||
.expect("failed to create tree"),
|
||||
);
|
||||
let shareable_db = ProviderFactory::new(db.clone(), self.chain_spec.clone());
|
||||
let latest = self.chain_spec.genesis_header().seal_slow();
|
||||
let blockchain_provider = BlockchainProvider::with_latest(shareable_db, tree, latest);
|
||||
|
||||
let pruner = Pruner::new(
|
||||
db.clone(),
|
||||
self.chain_spec.clone(),
|
||||
5,
|
||||
0,
|
||||
PruneModes::default(),
|
||||
BatchSizes::default(),
|
||||
);
|
||||
|
||||
let (mut engine, handle) = BeaconConsensusEngine::new(
|
||||
client,
|
||||
pipeline,
|
||||
blockchain_provider,
|
||||
Box::<TokioTaskExecutor>::default(),
|
||||
Box::<NoopSyncStateUpdater>::default(),
|
||||
None,
|
||||
false,
|
||||
payload_builder,
|
||||
None,
|
||||
self.pipeline_run_threshold.unwrap_or(MIN_BLOCKS_FOR_PIPELINE_RUN),
|
||||
Some(pruner),
|
||||
)
|
||||
.expect("failed to create consensus engine");
|
||||
|
||||
if let Some(max_block) = self.max_block {
|
||||
engine.sync.set_max_block(max_block)
|
||||
}
|
||||
|
||||
(engine, TestEnv::new(db, tip_rx, handle))
|
||||
}
|
||||
}
|
||||
|
||||
fn spawn_consensus_engine<Client: HeadersClient + BodiesClient + 'static>(
|
||||
engine: TestBeaconConsensusEngine<Client>,
|
||||
) -> oneshot::Receiver<Result<(), BeaconConsensusEngineError>> {
|
||||
let (tx, rx) = oneshot::channel();
|
||||
tokio::spawn(async move {
|
||||
let result = engine.await;
|
||||
tx.send(result).expect("failed to forward consensus engine result");
|
||||
});
|
||||
rx
|
||||
}
|
||||
use tokio::sync::oneshot::error::TryRecvError;
|
||||
|
||||
// Pipeline error is propagated.
|
||||
#[tokio::test]
|
||||
@ -2128,12 +1735,11 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Err(StageError::ChannelClosed)]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.with_max_block(1)
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Err(StageError::ChannelClosed)]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.with_max_block(1)
|
||||
.build();
|
||||
|
||||
let res = spawn_consensus_engine(consensus_engine);
|
||||
|
||||
@ -2160,12 +1766,11 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Err(StageError::ChannelClosed)]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.with_max_block(1)
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Err(StageError::ChannelClosed)]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.with_max_block(1)
|
||||
.build();
|
||||
|
||||
let mut rx = spawn_consensus_engine(consensus_engine);
|
||||
|
||||
@ -2223,15 +1828,14 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(1), done: true }),
|
||||
Err(StageError::ChannelClosed),
|
||||
]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.with_max_block(2)
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(1), done: true }),
|
||||
Err(StageError::ChannelClosed),
|
||||
]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.with_max_block(2)
|
||||
.build();
|
||||
|
||||
let rx = spawn_consensus_engine(consensus_engine);
|
||||
|
||||
@ -2259,15 +1863,14 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(max_block),
|
||||
done: true,
|
||||
})]))
|
||||
.with_max_block(max_block)
|
||||
.disable_blockchain_tree_sync()
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(max_block),
|
||||
done: true,
|
||||
})]))
|
||||
.with_max_block(max_block)
|
||||
.disable_blockchain_tree_sync()
|
||||
.build();
|
||||
|
||||
let rx = spawn_consensus_engine(consensus_engine);
|
||||
|
||||
@ -2309,13 +1912,12 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
|
||||
let mut engine_rx = spawn_consensus_engine(consensus_engine);
|
||||
|
||||
@ -2341,13 +1943,12 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
|
||||
let genesis = random_block(&mut rng, 0, None, None, Some(0));
|
||||
let block1 = random_block(&mut rng, 1, Some(genesis.hash), None, Some(0));
|
||||
@ -2391,14 +1992,13 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.build();
|
||||
|
||||
let genesis = random_block(&mut rng, 0, None, None, Some(0));
|
||||
let block1 = random_block(&mut rng, 1, Some(genesis.hash), None, Some(0));
|
||||
@ -2442,14 +2042,13 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.disable_blockchain_tree_sync()
|
||||
.build();
|
||||
|
||||
let genesis = random_block(&mut rng, 0, None, None, Some(0));
|
||||
let block1 = random_block(&mut rng, 1, Some(genesis.hash), None, Some(0));
|
||||
@ -2481,13 +2080,12 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
]))
|
||||
.build();
|
||||
|
||||
let genesis = random_block(&mut rng, 0, None, None, Some(0));
|
||||
let mut block1 = random_block(&mut rng, 1, Some(genesis.hash), None, Some(0));
|
||||
@ -2535,13 +2133,12 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
Ok(ExecOutput { checkpoint: StageCheckpoint::new(0), done: true }),
|
||||
]))
|
||||
.build();
|
||||
|
||||
let genesis = random_block(&mut rng, 0, None, None, Some(0));
|
||||
let block1 = random_block(&mut rng, 1, Some(genesis.hash), None, Some(0));
|
||||
@ -2586,13 +2183,12 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
|
||||
let mut engine_rx = spawn_consensus_engine(consensus_engine);
|
||||
|
||||
@ -2622,13 +2218,12 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
|
||||
let genesis = random_block(&mut rng, 0, None, None, Some(0));
|
||||
let block1 = random_block(&mut rng, 1, Some(genesis.hash), None, Some(0));
|
||||
@ -2684,11 +2279,11 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_real_pipeline()
|
||||
.with_real_executor()
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_real_pipeline()
|
||||
.with_real_executor()
|
||||
.with_real_consensus()
|
||||
.build();
|
||||
|
||||
let genesis =
|
||||
SealedBlock { header: chain_spec.sealed_genesis_header(), ..Default::default() };
|
||||
@ -2726,13 +2321,12 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.build();
|
||||
|
||||
let genesis = random_block(&mut rng, 0, None, None, Some(0));
|
||||
|
||||
@ -2783,14 +2377,13 @@ mod tests {
|
||||
.build(),
|
||||
);
|
||||
|
||||
let (consensus_engine, env) =
|
||||
TestConsensusEngineBuilder::<NoopFullBlockClient>::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.with_executor_results(Vec::from([exec_result2]))
|
||||
.build();
|
||||
let (consensus_engine, env) = TestConsensusEngineBuilder::new(chain_spec.clone())
|
||||
.with_pipeline_exec_outputs(VecDeque::from([Ok(ExecOutput {
|
||||
checkpoint: StageCheckpoint::new(0),
|
||||
done: true,
|
||||
})]))
|
||||
.with_executor_results(Vec::from([exec_result2]))
|
||||
.build();
|
||||
|
||||
insert_blocks(
|
||||
env.db.as_ref(),
|
||||
|
||||
509
crates/consensus/beacon/src/engine/test_utils.rs
Normal file
509
crates/consensus/beacon/src/engine/test_utils.rs
Normal file
@ -0,0 +1,509 @@
|
||||
use crate::{
|
||||
BeaconConsensus, BeaconConsensusEngine, BeaconConsensusEngineError,
|
||||
BeaconConsensusEngineHandle, BeaconForkChoiceUpdateError, BeaconOnNewPayloadError,
|
||||
MIN_BLOCKS_FOR_PIPELINE_RUN,
|
||||
};
|
||||
use reth_blockchain_tree::{
|
||||
config::BlockchainTreeConfig, externals::TreeExternals, post_state::PostState, BlockchainTree,
|
||||
ShareableBlockchainTree,
|
||||
};
|
||||
use reth_db::{test_utils::create_test_rw_db, DatabaseEnv};
|
||||
use reth_downloaders::{
|
||||
bodies::bodies::BodiesDownloaderBuilder,
|
||||
headers::reverse_headers::ReverseHeadersDownloaderBuilder,
|
||||
};
|
||||
use reth_interfaces::{
|
||||
consensus::Consensus,
|
||||
executor::BlockExecutionError,
|
||||
p2p::{bodies::client::BodiesClient, either::EitherDownloader, headers::client::HeadersClient},
|
||||
sync::NoopSyncStateUpdater,
|
||||
test_utils::{NoopFullBlockClient, TestConsensus},
|
||||
};
|
||||
use reth_payload_builder::test_utils::spawn_test_payload_service;
|
||||
use reth_primitives::{BlockNumber, ChainSpec, PruneModes, H256, U256};
|
||||
use reth_provider::{
|
||||
providers::BlockchainProvider, test_utils::TestExecutorFactory, BlockExecutor, ExecutorFactory,
|
||||
ProviderFactory, StateProvider,
|
||||
};
|
||||
use reth_prune::{BatchSizes, Pruner};
|
||||
use reth_revm::Factory;
|
||||
use reth_rpc_types::engine::{ExecutionPayload, ForkchoiceState, ForkchoiceUpdated, PayloadStatus};
|
||||
use reth_stages::{
|
||||
sets::DefaultStages, stages::HeaderSyncMode, test_utils::TestStages, ExecOutput, Pipeline,
|
||||
StageError,
|
||||
};
|
||||
use reth_tasks::TokioTaskExecutor;
|
||||
use std::{collections::VecDeque, sync::Arc};
|
||||
use tokio::sync::{oneshot, watch};
|
||||
|
||||
type TestBeaconConsensusEngine<Client> = BeaconConsensusEngine<
|
||||
Arc<DatabaseEnv>,
|
||||
BlockchainProvider<
|
||||
Arc<DatabaseEnv>,
|
||||
ShareableBlockchainTree<
|
||||
Arc<DatabaseEnv>,
|
||||
Arc<dyn Consensus>,
|
||||
EitherExecutorFactory<TestExecutorFactory, Factory>,
|
||||
>,
|
||||
>,
|
||||
Arc<EitherDownloader<Client, NoopFullBlockClient>>,
|
||||
>;
|
||||
|
||||
pub struct TestEnv<DB> {
|
||||
pub db: DB,
|
||||
// Keep the tip receiver around, so it's not dropped.
|
||||
#[allow(dead_code)]
|
||||
tip_rx: watch::Receiver<H256>,
|
||||
engine_handle: BeaconConsensusEngineHandle,
|
||||
}
|
||||
|
||||
impl<DB> TestEnv<DB> {
|
||||
fn new(
|
||||
db: DB,
|
||||
tip_rx: watch::Receiver<H256>,
|
||||
engine_handle: BeaconConsensusEngineHandle,
|
||||
) -> Self {
|
||||
Self { db, tip_rx, engine_handle }
|
||||
}
|
||||
|
||||
pub async fn send_new_payload(
|
||||
&self,
|
||||
payload: ExecutionPayload,
|
||||
) -> Result<PayloadStatus, BeaconOnNewPayloadError> {
|
||||
self.engine_handle.new_payload(payload).await
|
||||
}
|
||||
|
||||
/// Sends the `ExecutionPayload` message to the consensus engine and retries if the engine
|
||||
/// is syncing.
|
||||
pub async fn send_new_payload_retry_on_syncing(
|
||||
&self,
|
||||
payload: ExecutionPayload,
|
||||
) -> Result<PayloadStatus, BeaconOnNewPayloadError> {
|
||||
loop {
|
||||
let result = self.send_new_payload(payload.clone()).await?;
|
||||
if !result.is_syncing() {
|
||||
return Ok(result)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub async fn send_forkchoice_updated(
|
||||
&self,
|
||||
state: ForkchoiceState,
|
||||
) -> Result<ForkchoiceUpdated, BeaconForkChoiceUpdateError> {
|
||||
self.engine_handle.fork_choice_updated(state, None).await
|
||||
}
|
||||
|
||||
/// Sends the `ForkchoiceUpdated` message to the consensus engine and retries if the engine
|
||||
/// is syncing.
|
||||
pub async fn send_forkchoice_retry_on_syncing(
|
||||
&self,
|
||||
state: ForkchoiceState,
|
||||
) -> Result<ForkchoiceUpdated, BeaconForkChoiceUpdateError> {
|
||||
loop {
|
||||
let result = self.engine_handle.fork_choice_updated(state, None).await?;
|
||||
if !result.is_syncing() {
|
||||
return Ok(result)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: add with_consensus in case we want to use the TestConsensus purposeful failure - this
|
||||
// would require similar patterns to how we use with_client and the EitherDownloader
|
||||
/// Represents either a real consensus engine, or a test consensus engine.
|
||||
#[derive(Default)]
|
||||
enum TestConsensusConfig {
|
||||
/// Test consensus engine
|
||||
#[default]
|
||||
Test,
|
||||
/// Real consensus engine
|
||||
Real,
|
||||
}
|
||||
|
||||
/// Represents either test pipeline outputs, or real pipeline configuration.
|
||||
enum TestPipelineConfig {
|
||||
/// Test pipeline outputs.
|
||||
Test(VecDeque<Result<ExecOutput, StageError>>),
|
||||
/// Real pipeline configuration.
|
||||
Real,
|
||||
}
|
||||
|
||||
impl Default for TestPipelineConfig {
|
||||
fn default() -> Self {
|
||||
Self::Test(VecDeque::new())
|
||||
}
|
||||
}
|
||||
|
||||
/// Represents either test executor results, or real executor configuration.
|
||||
enum TestExecutorConfig {
|
||||
/// Test executor results.
|
||||
Test(Vec<PostState>),
|
||||
/// Real executor configuration.
|
||||
Real,
|
||||
}
|
||||
|
||||
impl Default for TestExecutorConfig {
|
||||
fn default() -> Self {
|
||||
Self::Test(Vec::new())
|
||||
}
|
||||
}
|
||||
|
||||
/// A type that represents one of two possible executor factories.
|
||||
#[derive(Debug, Clone)]
|
||||
pub enum EitherExecutorFactory<A: ExecutorFactory, B: ExecutorFactory> {
|
||||
/// The first factory variant
|
||||
Left(A),
|
||||
/// The second factory variant
|
||||
Right(B),
|
||||
}
|
||||
|
||||
// A type that represents one of two possible BlockExecutor types.
|
||||
#[derive(Debug)]
|
||||
pub enum EitherBlockExecutor<A, B> {
|
||||
/// The first executor variant
|
||||
Left(A),
|
||||
/// The second executor variant
|
||||
Right(B),
|
||||
}
|
||||
|
||||
impl<A, B, SP> BlockExecutor<SP> for EitherBlockExecutor<A, B>
|
||||
where
|
||||
A: BlockExecutor<SP>,
|
||||
B: BlockExecutor<SP>,
|
||||
SP: StateProvider,
|
||||
{
|
||||
fn execute(
|
||||
&mut self,
|
||||
block: &reth_primitives::Block,
|
||||
total_difficulty: U256,
|
||||
senders: Option<Vec<reth_primitives::Address>>,
|
||||
) -> Result<PostState, BlockExecutionError> {
|
||||
match self {
|
||||
EitherBlockExecutor::Left(a) => a.execute(block, total_difficulty, senders),
|
||||
EitherBlockExecutor::Right(b) => b.execute(block, total_difficulty, senders),
|
||||
}
|
||||
}
|
||||
|
||||
fn execute_and_verify_receipt(
|
||||
&mut self,
|
||||
block: &reth_primitives::Block,
|
||||
total_difficulty: U256,
|
||||
senders: Option<Vec<reth_primitives::Address>>,
|
||||
) -> Result<PostState, BlockExecutionError> {
|
||||
match self {
|
||||
EitherBlockExecutor::Left(a) => {
|
||||
a.execute_and_verify_receipt(block, total_difficulty, senders)
|
||||
}
|
||||
EitherBlockExecutor::Right(b) => {
|
||||
b.execute_and_verify_receipt(block, total_difficulty, senders)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<A, B> ExecutorFactory for EitherExecutorFactory<A, B>
|
||||
where
|
||||
A: ExecutorFactory,
|
||||
B: ExecutorFactory,
|
||||
{
|
||||
type Executor<T: StateProvider> = EitherBlockExecutor<A::Executor<T>, B::Executor<T>>;
|
||||
|
||||
fn chain_spec(&self) -> &ChainSpec {
|
||||
match self {
|
||||
EitherExecutorFactory::Left(a) => a.chain_spec(),
|
||||
EitherExecutorFactory::Right(b) => b.chain_spec(),
|
||||
}
|
||||
}
|
||||
|
||||
fn with_sp<SP: reth_provider::StateProvider>(&self, sp: SP) -> Self::Executor<SP> {
|
||||
match self {
|
||||
EitherExecutorFactory::Left(a) => EitherBlockExecutor::Left(a.with_sp(sp)),
|
||||
EitherExecutorFactory::Right(b) => EitherBlockExecutor::Right(b.with_sp(sp)),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// The basic configuration for a `TestConsensusEngine`, without generics for the client or
|
||||
/// consensus engine.
|
||||
pub struct TestConsensusEngineBuilder {
|
||||
chain_spec: Arc<ChainSpec>,
|
||||
pipeline_config: TestPipelineConfig,
|
||||
executor_config: TestExecutorConfig,
|
||||
pipeline_run_threshold: Option<u64>,
|
||||
max_block: Option<BlockNumber>,
|
||||
consensus: TestConsensusConfig,
|
||||
}
|
||||
|
||||
impl TestConsensusEngineBuilder {
|
||||
/// Create a new `TestConsensusEngineBuilder` with the given `ChainSpec`.
|
||||
pub fn new(chain_spec: Arc<ChainSpec>) -> Self {
|
||||
Self {
|
||||
chain_spec,
|
||||
pipeline_config: Default::default(),
|
||||
executor_config: Default::default(),
|
||||
pipeline_run_threshold: None,
|
||||
max_block: None,
|
||||
consensus: Default::default(),
|
||||
}
|
||||
}
|
||||
|
||||
/// Set the pipeline execution outputs to use for the test consensus engine.
|
||||
pub fn with_pipeline_exec_outputs(
|
||||
mut self,
|
||||
pipeline_exec_outputs: VecDeque<Result<ExecOutput, StageError>>,
|
||||
) -> Self {
|
||||
self.pipeline_config = TestPipelineConfig::Test(pipeline_exec_outputs);
|
||||
self
|
||||
}
|
||||
|
||||
/// Set the executor results to use for the test consensus engine.
|
||||
pub fn with_executor_results(mut self, executor_results: Vec<PostState>) -> Self {
|
||||
self.executor_config = TestExecutorConfig::Test(executor_results);
|
||||
self
|
||||
}
|
||||
|
||||
/// Sets the max block for the pipeline to run.
|
||||
pub fn with_max_block(mut self, max_block: BlockNumber) -> Self {
|
||||
self.max_block = Some(max_block);
|
||||
self
|
||||
}
|
||||
|
||||
/// Uses the real pipeline instead of a pipeline with empty exec outputs.
|
||||
pub fn with_real_pipeline(mut self) -> Self {
|
||||
self.pipeline_config = TestPipelineConfig::Real;
|
||||
self
|
||||
}
|
||||
|
||||
/// Uses the real executor instead of a executor with empty results.
|
||||
pub fn with_real_executor(mut self) -> Self {
|
||||
self.executor_config = TestExecutorConfig::Real;
|
||||
self
|
||||
}
|
||||
|
||||
/// Uses a real consensus engine instead of a test consensus engine.
|
||||
pub fn with_real_consensus(mut self) -> Self {
|
||||
self.consensus = TestConsensusConfig::Real;
|
||||
self
|
||||
}
|
||||
|
||||
/// Disables blockchain tree driven sync. This is the same as setting the pipeline run
|
||||
/// threshold to 0.
|
||||
pub fn disable_blockchain_tree_sync(mut self) -> Self {
|
||||
self.pipeline_run_threshold = Some(0);
|
||||
self
|
||||
}
|
||||
|
||||
/// Sets the client to use for network operations.
|
||||
#[allow(dead_code)]
|
||||
pub fn with_client<Client>(self, client: Client) -> NetworkedTestConsensusEngineBuilder<Client>
|
||||
where
|
||||
Client: HeadersClient + BodiesClient + 'static,
|
||||
{
|
||||
NetworkedTestConsensusEngineBuilder { base_config: self, client: Some(client) }
|
||||
}
|
||||
|
||||
/// Builds the test consensus engine into a `TestConsensusEngine` and `TestEnv`.
|
||||
pub fn build(
|
||||
self,
|
||||
) -> (TestBeaconConsensusEngine<NoopFullBlockClient>, TestEnv<Arc<DatabaseEnv>>) {
|
||||
let networked = NetworkedTestConsensusEngineBuilder { base_config: self, client: None };
|
||||
|
||||
networked.build()
|
||||
}
|
||||
}
|
||||
|
||||
/// A builder for `TestConsensusEngine`, allows configuration of mocked pipeline outputs and
|
||||
/// mocked executor results.
|
||||
///
|
||||
/// This optionally includes a client for network operations.
|
||||
pub struct NetworkedTestConsensusEngineBuilder<Client> {
|
||||
base_config: TestConsensusEngineBuilder,
|
||||
client: Option<Client>,
|
||||
}
|
||||
|
||||
impl<Client> NetworkedTestConsensusEngineBuilder<Client>
|
||||
where
|
||||
Client: HeadersClient + BodiesClient + 'static,
|
||||
{
|
||||
/// Set the pipeline execution outputs to use for the test consensus engine.
|
||||
#[allow(dead_code)]
|
||||
pub fn with_pipeline_exec_outputs(
|
||||
mut self,
|
||||
pipeline_exec_outputs: VecDeque<Result<ExecOutput, StageError>>,
|
||||
) -> Self {
|
||||
self.base_config.pipeline_config = TestPipelineConfig::Test(pipeline_exec_outputs);
|
||||
self
|
||||
}
|
||||
|
||||
/// Set the executor results to use for the test consensus engine.
|
||||
#[allow(dead_code)]
|
||||
pub fn with_executor_results(mut self, executor_results: Vec<PostState>) -> Self {
|
||||
self.base_config.executor_config = TestExecutorConfig::Test(executor_results);
|
||||
self
|
||||
}
|
||||
|
||||
/// Sets the max block for the pipeline to run.
|
||||
#[allow(dead_code)]
|
||||
pub fn with_max_block(mut self, max_block: BlockNumber) -> Self {
|
||||
self.base_config.max_block = Some(max_block);
|
||||
self
|
||||
}
|
||||
|
||||
/// Uses the real pipeline instead of a pipeline with empty exec outputs.
|
||||
#[allow(dead_code)]
|
||||
pub fn with_real_pipeline(mut self) -> Self {
|
||||
self.base_config.pipeline_config = TestPipelineConfig::Real;
|
||||
self
|
||||
}
|
||||
|
||||
/// Uses the real executor instead of a executor with empty results.
|
||||
#[allow(dead_code)]
|
||||
pub fn with_real_executor(mut self) -> Self {
|
||||
self.base_config.executor_config = TestExecutorConfig::Real;
|
||||
self
|
||||
}
|
||||
|
||||
/// Disables blockchain tree driven sync. This is the same as setting the pipeline run
|
||||
/// threshold to 0.
|
||||
#[allow(dead_code)]
|
||||
pub fn disable_blockchain_tree_sync(mut self) -> Self {
|
||||
self.base_config.pipeline_run_threshold = Some(0);
|
||||
self
|
||||
}
|
||||
|
||||
/// Sets the client to use for network operations.
|
||||
#[allow(dead_code)]
|
||||
pub fn with_client<ClientType>(
|
||||
self,
|
||||
client: ClientType,
|
||||
) -> NetworkedTestConsensusEngineBuilder<ClientType>
|
||||
where
|
||||
ClientType: HeadersClient + BodiesClient + 'static,
|
||||
{
|
||||
NetworkedTestConsensusEngineBuilder { base_config: self.base_config, client: Some(client) }
|
||||
}
|
||||
|
||||
/// Builds the test consensus engine into a `TestConsensusEngine` and `TestEnv`.
|
||||
pub fn build(self) -> (TestBeaconConsensusEngine<Client>, TestEnv<Arc<DatabaseEnv>>) {
|
||||
reth_tracing::init_test_tracing();
|
||||
let db = create_test_rw_db();
|
||||
|
||||
let consensus: Arc<dyn Consensus> = match self.base_config.consensus {
|
||||
TestConsensusConfig::Real => {
|
||||
Arc::new(BeaconConsensus::new(Arc::clone(&self.base_config.chain_spec)))
|
||||
}
|
||||
TestConsensusConfig::Test => Arc::new(TestConsensus::default()),
|
||||
};
|
||||
let payload_builder = spawn_test_payload_service();
|
||||
|
||||
// use either noop client or a user provided client (for example TestFullBlockClient)
|
||||
let client = Arc::new(
|
||||
self.client
|
||||
.map(EitherDownloader::Left)
|
||||
.unwrap_or_else(|| EitherDownloader::Right(NoopFullBlockClient::default())),
|
||||
);
|
||||
|
||||
// use either test executor or real executor
|
||||
let executor_factory = match self.base_config.executor_config {
|
||||
TestExecutorConfig::Test(results) => {
|
||||
let executor_factory =
|
||||
TestExecutorFactory::new(self.base_config.chain_spec.clone());
|
||||
executor_factory.extend(results);
|
||||
EitherExecutorFactory::Left(executor_factory)
|
||||
}
|
||||
TestExecutorConfig::Real => {
|
||||
EitherExecutorFactory::Right(Factory::new(self.base_config.chain_spec.clone()))
|
||||
}
|
||||
};
|
||||
|
||||
// Setup pipeline
|
||||
let (tip_tx, tip_rx) = watch::channel(H256::default());
|
||||
let mut pipeline = match self.base_config.pipeline_config {
|
||||
TestPipelineConfig::Test(outputs) => Pipeline::builder()
|
||||
.add_stages(TestStages::new(outputs, Default::default()))
|
||||
.with_tip_sender(tip_tx),
|
||||
TestPipelineConfig::Real => {
|
||||
let header_downloader = ReverseHeadersDownloaderBuilder::default()
|
||||
.build(client.clone(), consensus.clone())
|
||||
.into_task();
|
||||
|
||||
let body_downloader = BodiesDownloaderBuilder::default()
|
||||
.build(client.clone(), consensus.clone(), db.clone())
|
||||
.into_task();
|
||||
|
||||
Pipeline::builder().add_stages(DefaultStages::new(
|
||||
HeaderSyncMode::Tip(tip_rx.clone()),
|
||||
Arc::clone(&consensus) as Arc<dyn Consensus>,
|
||||
header_downloader,
|
||||
body_downloader,
|
||||
executor_factory.clone(),
|
||||
))
|
||||
}
|
||||
};
|
||||
|
||||
if let Some(max_block) = self.base_config.max_block {
|
||||
pipeline = pipeline.with_max_block(max_block);
|
||||
}
|
||||
|
||||
let pipeline = pipeline.build(db.clone(), self.base_config.chain_spec.clone());
|
||||
|
||||
// Setup blockchain tree
|
||||
let externals = TreeExternals::new(
|
||||
db.clone(),
|
||||
consensus,
|
||||
executor_factory,
|
||||
self.base_config.chain_spec.clone(),
|
||||
);
|
||||
let config = BlockchainTreeConfig::new(1, 2, 3, 2);
|
||||
let (canon_state_notification_sender, _) = tokio::sync::broadcast::channel(3);
|
||||
let tree = ShareableBlockchainTree::new(
|
||||
BlockchainTree::new(externals, canon_state_notification_sender, config)
|
||||
.expect("failed to create tree"),
|
||||
);
|
||||
let shareable_db = ProviderFactory::new(db.clone(), self.base_config.chain_spec.clone());
|
||||
let latest = self.base_config.chain_spec.genesis_header().seal_slow();
|
||||
let blockchain_provider = BlockchainProvider::with_latest(shareable_db, tree, latest);
|
||||
|
||||
let pruner = Pruner::new(
|
||||
db.clone(),
|
||||
self.base_config.chain_spec.clone(),
|
||||
5,
|
||||
0,
|
||||
PruneModes::default(),
|
||||
BatchSizes::default(),
|
||||
);
|
||||
|
||||
let (mut engine, handle) = BeaconConsensusEngine::new(
|
||||
client,
|
||||
pipeline,
|
||||
blockchain_provider,
|
||||
Box::<TokioTaskExecutor>::default(),
|
||||
Box::<NoopSyncStateUpdater>::default(),
|
||||
None,
|
||||
false,
|
||||
payload_builder,
|
||||
None,
|
||||
self.base_config.pipeline_run_threshold.unwrap_or(MIN_BLOCKS_FOR_PIPELINE_RUN),
|
||||
Some(pruner),
|
||||
)
|
||||
.expect("failed to create consensus engine");
|
||||
|
||||
if let Some(max_block) = self.base_config.max_block {
|
||||
engine.sync.set_max_block(max_block)
|
||||
}
|
||||
|
||||
(engine, TestEnv::new(db, tip_rx, handle))
|
||||
}
|
||||
}
|
||||
|
||||
pub fn spawn_consensus_engine<Client: HeadersClient + BodiesClient + 'static>(
|
||||
engine: TestBeaconConsensusEngine<Client>,
|
||||
) -> oneshot::Receiver<Result<(), BeaconConsensusEngineError>> {
|
||||
let (tx, rx) = oneshot::channel();
|
||||
tokio::spawn(async move {
|
||||
let result = engine.await;
|
||||
tx.send(result).expect("failed to forward consensus engine result");
|
||||
});
|
||||
rx
|
||||
}
|
||||
Reference in New Issue
Block a user