spawn geth node

Signed-off-by: Cyrill Leutwiler <bigcyrill@hotmail.com>
This commit is contained in:
Cyrill Leutwiler
2025-03-23 14:12:08 +01:00
parent 6cd4519d89
commit 487eefe908
8 changed files with 214 additions and 75 deletions
Generated
+2
View File
@@ -2874,12 +2874,14 @@ name = "revive-dt-core"
version = "0.1.0" version = "0.1.0"
dependencies = [ dependencies = [
"anyhow", "anyhow",
"clap",
"env_logger", "env_logger",
"log", "log",
"rayon", "rayon",
"revive-dt-compiler", "revive-dt-compiler",
"revive-dt-config", "revive-dt-config",
"revive-dt-format", "revive-dt-format",
"revive-dt-node",
"revive-solc-json-interface", "revive-solc-json-interface",
"semver 1.0.26", "semver 1.0.26",
"serde", "serde",
-1
View File
@@ -45,7 +45,6 @@ version = "0.12.6"
default-features = false default-features = false
features = [ features = [
"json-abi", "json-abi",
"genesis",
"providers", "providers",
"provider-debug-api", "provider-debug-api",
"reqwest", "reqwest",
+29 -14
View File
@@ -1,21 +1,11 @@
//! The global configuration used accross all revive differential testing crates. //! The global configuration used accross all revive differential testing crates.
use std::{path::PathBuf, sync::OnceLock}; use std::{env, path::PathBuf};
use clap::Parser; use clap::Parser;
static ARGUMENTS: OnceLock<Arguments> = OnceLock::new();
/// Get the command line arguments.
pub fn get_args() -> &'static Arguments {
ARGUMENTS.get_or_init(Arguments::parse)
}
#[derive(Debug, Parser, Clone)] #[derive(Debug, Parser, Clone)]
#[command( #[command(name = "retester")]
name = "revive compiler differential tester utility",
arg_required_else_help = true
)]
pub struct Arguments { pub struct Arguments {
/// The path to the `resolc` executable to be tested. /// The path to the `resolc` executable to be tested.
/// ///
@@ -28,12 +18,37 @@ pub struct Arguments {
pub corpus: Vec<PathBuf>, pub corpus: Vec<PathBuf>,
/// A place to store temporary artifacts during test execution. /// A place to store temporary artifacts during test execution.
#[arg(long = "workdir", short)] #[arg(long = "workdir", short, default_value_t = cwd())]
pub working_directory: PathBuf, pub working_directory: String,
/// The path to the `geth` executable. /// The path to the `geth` executable.
/// ///
/// By default it uses `geth` binary found in `$PATH`. /// By default it uses `geth` binary found in `$PATH`.
#[arg(short, long = "geth", default_value = "geth")] #[arg(short, long = "geth", default_value = "geth")]
pub geth: PathBuf, pub geth: PathBuf,
/// The maximum time in milliseconds to wait for geth to start.
#[arg(long = "geth-start-timeout", default_value = "2000")]
pub geth_start_timeout: u64,
/// The test network chain ID.
#[arg(short, long = "network-id", default_value = "420420420")]
pub network_id: u64,
/// Configure nodes according to this genesis.json file.
#[arg(long = "genesis-file")]
pub genesis_file: Option<PathBuf>,
}
fn cwd() -> String {
env::current_dir()
.expect("should be able to access current woring directory")
.to_string_lossy()
.to_string()
}
impl Default for Arguments {
fn default() -> Self {
Arguments::parse_from(["retester"])
}
} }
+3 -1
View File
@@ -14,11 +14,13 @@ path = "src/main.rs"
[dependencies] [dependencies]
revive-dt-compiler = { workspace = true } revive-dt-compiler = { workspace = true }
revive-dt-config = { workspace = true }
revive-dt-format = { workspace = true } revive-dt-format = { workspace = true }
revive-solc-json-interface = { workspace = true } revive-solc-json-interface = { workspace = true }
revive-dt-config = { workspace = true } revive-dt-node = { workspace = true }
anyhow = { workspace = true } anyhow = { workspace = true }
clap = { workspace = true }
log = { workspace = true } log = { workspace = true }
env_logger = { workspace = true } env_logger = { workspace = true }
rayon = { workspace = true } rayon = { workspace = true }
+7 -1
View File
@@ -1,5 +1,6 @@
use std::collections::BTreeSet; use std::collections::BTreeSet;
use clap::Parser;
use rayon::prelude::*; use rayon::prelude::*;
use revive_dt_config::*; use revive_dt_config::*;
@@ -9,7 +10,12 @@ use revive_dt_format::corpus::Corpus;
fn main() -> anyhow::Result<()> { fn main() -> anyhow::Result<()> {
env_logger::init(); env_logger::init();
for path in get_args().corpus.iter().collect::<BTreeSet<_>>() { let config = Arguments::parse();
if config.corpus.is_empty() {
anyhow::bail!("no test corpus specified");
}
for path in config.corpus.iter().collect::<BTreeSet<_>>() {
log::trace!("attempting corpus {path:?}"); log::trace!("attempting corpus {path:?}");
let corpus = Corpus::try_from_path(path)?; let corpus = Corpus::try_from_path(path)?;
log::info!("found corpus: {corpus:?}"); log::info!("found corpus: {corpus:?}");
+2 -5
View File
@@ -13,12 +13,9 @@ pub trait EthereumNode {
/// Execute the [TransactionRequest] and return a [TransactionReceipt]. /// Execute the [TransactionRequest] and return a [TransactionReceipt].
fn execute_transaction( fn execute_transaction(
&self, &self,
transaction_request: TransactionRequest, transaction: TransactionRequest,
) -> anyhow::Result<TransactionReceipt>; ) -> anyhow::Result<TransactionReceipt>;
/// Trace the transaction in the [TransactionReceipt] and return a [GethTrace]. /// Trace the transaction in the [TransactionReceipt] and return a [GethTrace].
fn trace_transaction( fn trace_transaction(&self, transaction: TransactionReceipt) -> anyhow::Result<GethTrace>;
&self,
transaction_receipt: TransactionReceipt,
) -> anyhow::Result<GethTrace>;
} }
+165 -47
View File
@@ -1,17 +1,20 @@
//! The go-ethereum node implementation. //! The go-ethereum node implementation.
use std::{ use std::{
fs::{File, create_dir, exists}, fs::{File, create_dir, exists, remove_dir_all},
io::{BufRead, BufReader, Read, Write},
path::PathBuf, path::PathBuf,
process::{Command, Stdio}, process::{Child, Command, Stdio},
sync::atomic::{AtomicU32, Ordering}, sync::atomic::{AtomicU32, Ordering},
thread,
time::{Duration, Instant},
}; };
use alloy::{ use alloy::rpc::types::{
genesis::Genesis, TransactionReceipt, TransactionRequest,
rpc::types::trace::geth::{DiffMode, PreStateFrame}, trace::geth::{DiffMode, PreStateFrame},
}; };
use revive_dt_config::get_args; use revive_dt_config::Arguments;
use revive_dt_node_interaction::{ use revive_dt_node_interaction::{
EthereumNode, trace::trace_transaction, transaction::execute_transaction, EthereumNode, trace::trace_transaction, transaction::execute_transaction,
}; };
@@ -20,78 +23,157 @@ use crate::Node;
static NODE_COUNT: AtomicU32 = AtomicU32::new(0); static NODE_COUNT: AtomicU32 = AtomicU32::new(0);
/// The go-ethereum node instance implementation.
///
/// Implements helpers to initialize, spawn and wait the node.
///
/// Assumes dev mode and IPC only (`P2P`, `http`` etc. are kept disabled).
///
/// Prunes the child process and the base directory on drop.
#[derive(Debug)] #[derive(Debug)]
pub struct Instance { pub struct Instance {
connection_string: String, connection_string: String,
directory: PathBuf, base_directory: PathBuf,
data_directory: PathBuf,
geth: PathBuf, geth: PathBuf,
id: u32, id: u32,
handle: Option<Child>,
network_id: u64,
start_timeout: u64,
} }
impl Instance { impl Instance {
pub fn new() -> anyhow::Result<Self> { const BASE_DIRECTORY: &str = "geth";
let args = get_args(); const DATA_DIRECTORY: &str = "data";
let geth_directory = args.working_directory.join("geth"); const IPC_FILE: &str = "geth.ipc";
if !exists(&geth_directory)? { const GENESIS_JSON_FILE: &str = "genesis.json";
create_dir(&geth_directory)?;
}
const READY_MARKER: &str = "IPC endpoint opened";
const ERROR_MARKER: &str = "Fatal:";
/// Create a new uninitialized instance.
pub fn new(config: &Arguments) -> anyhow::Result<Self> {
let geth_directory = PathBuf::from(&config.working_directory).join(Self::BASE_DIRECTORY);
let id = NODE_COUNT.fetch_add(1, Ordering::SeqCst); let id = NODE_COUNT.fetch_add(1, Ordering::SeqCst);
let directory = geth_directory.join(id.to_string()); let base_directory = geth_directory.join(id.to_string());
let connection_string = directory.join("geth.ipc").display().to_string();
Ok(Self { Ok(Self {
connection_string, connection_string: base_directory.join(Self::IPC_FILE).display().to_string(),
directory, data_directory: base_directory.join(Self::DATA_DIRECTORY),
geth: args.geth.clone(), base_directory,
geth: config.geth.clone(),
id, id,
handle: None,
network_id: config.network_id,
start_timeout: config.geth_start_timeout,
}) })
} }
}
impl Instance { /// Create the node directory and call `geth init` to configure the genesis.
/// Call `init` on the node to configure it's genesis. fn init(&mut self, genesis: String) -> anyhow::Result<&mut Self> {
fn init(&mut self, genesis: Genesis) -> anyhow::Result<()> { let geth_directory = self.base_directory.parent().expect("the id should be set");
let genesis_path = self.directory.join("genesis.json"); if !exists(geth_directory)? {
create_dir(geth_directory)?;
}
create_dir(&self.base_directory)?;
let mut file = File::create(&genesis_path)?; let genesis_path = self.base_directory.join(Self::GENESIS_JSON_FILE);
serde_json::to_writer_pretty(&mut file, &genesis)?; File::create(&genesis_path)?.write_all(genesis.as_bytes())?;
if !Command::new(&self.geth) let mut child = Command::new(&self.geth)
.arg("init")
.arg("--datadir") .arg("--datadir")
.arg(self.directory.join("data")) .arg(&self.data_directory)
.stderr(Stdio::null()) .arg(genesis_path)
.stderr(Stdio::piped())
.stdout(Stdio::null()) .stdout(Stdio::null())
.spawn()? .spawn()?;
.wait()?
.success() let mut stderr = String::new();
{ child
anyhow::bail!("failed to initialize geth node {:?}", &self); .stderr
.take()
.expect("should be piped")
.read_to_string(&mut stderr)?;
if !child.wait()?.success() {
anyhow::bail!("failed to initialize geth node #{:?}: {stderr}", &self.id);
} }
Ok(()) Ok(self)
}
/// Spawn the go-ethereum node child process.
///
/// [Instance::init] must be called priorly.
fn spawn_process(&mut self) -> anyhow::Result<&mut Self> {
self.handle = Command::new(&self.geth)
.arg("--dev")
.arg("--datadir")
.arg(&self.data_directory)
.arg("--ipcpath")
.arg(&self.connection_string)
.arg("--networkid")
.arg(self.network_id.to_string())
.arg("--nodiscover")
.arg("--maxpeers")
.arg("0")
.stderr(Stdio::piped())
.stdout(Stdio::null())
.spawn()?
.into();
Ok(self)
}
/// Wait for the g-ethereum node child process getting ready.
///
/// [Instance::spawn_process] must be called priorly.
fn wait_ready(&mut self) -> anyhow::Result<&mut Self> {
// Thanks clippy but geth is a server; we don't `wait` but eventually kill it.
#[allow(clippy::zombie_processes)]
let mut child = self.handle.take().expect("should be spawned");
let start_time = Instant::now();
let maximum_wait_time = Duration::from_millis(self.start_timeout);
let mut stderr = BufReader::new(child.stderr.take().expect("should be piped")).lines();
let error = loop {
let Some(Ok(line)) = stderr.next() else {
break "child process stderr reading error".to_string();
};
if line.contains(Self::ERROR_MARKER) {
break line;
}
if line.contains(Self::READY_MARKER) {
// Keep stderr alive
// https://github.com/alloy-rs/alloy/issues/2091#issuecomment-2676134147
thread::spawn(move || for _ in stderr.by_ref() {});
self.handle = child.into();
return Ok(self);
}
if Instant::now().duration_since(start_time) > maximum_wait_time {
break "spawn timeout".to_string();
}
};
let _ = child.kill();
anyhow::bail!("geth node #{} spawn error: {error}", self.id)
} }
} }
impl EthereumNode for Instance { impl EthereumNode for Instance {
fn execute_transaction( fn execute_transaction(
&self, &self,
transaction_request: alloy::rpc::types::TransactionRequest, transaction: TransactionRequest,
) -> anyhow::Result<alloy::rpc::types::TransactionReceipt> { ) -> anyhow::Result<alloy::rpc::types::TransactionReceipt> {
execute_transaction(transaction_request, self.connection_string()) execute_transaction(transaction, self.connection_string())
} }
fn trace_transaction( fn trace_transaction(
&self, &self,
transaction_receipt: alloy::rpc::types::TransactionReceipt, transaction: TransactionReceipt,
) -> anyhow::Result<alloy::rpc::types::trace::geth::GethTrace> { ) -> anyhow::Result<alloy::rpc::types::trace::geth::GethTrace> {
trace_transaction( trace_transaction(transaction, Default::default(), self.connection_string())
transaction_receipt,
Default::default(),
self.connection_string(),
)
} }
} }
@@ -104,10 +186,9 @@ impl Node for Instance {
Ok(()) Ok(())
} }
fn spawn(&mut self, genesis: Genesis) -> anyhow::Result<&mut Self> { fn spawn(&mut self, genesis: String) -> anyhow::Result<()> {
self.init(genesis)?; self.init(genesis)?.spawn_process()?.wait_ready()?;
Ok(())
Ok(self)
} }
fn state_diff( fn state_diff(
@@ -123,3 +204,40 @@ impl Node for Instance {
} }
} }
} }
impl Drop for Instance {
fn drop(&mut self) {
if let Some(child) = self.handle.as_mut() {
let _ = child.kill();
}
if self.base_directory.exists() {
let _ = remove_dir_all(&self.base_directory);
}
}
}
#[cfg(test)]
mod tests {
use revive_dt_config::Arguments;
use crate::{GENESIS_JSON, Node};
use super::Instance;
#[test]
fn init_works() {
Instance::new(&Arguments::default())
.unwrap()
.init(GENESIS_JSON.to_string())
.unwrap();
}
#[test]
fn spawn_works() {
Instance::new(&Arguments::default())
.unwrap()
.spawn(GENESIS_JSON.to_string())
.unwrap();
}
}
+6 -6
View File
@@ -1,19 +1,19 @@
//! This crate implements the testing nodes. //! This crate implements the testing nodes.
use alloy::{ use alloy::rpc::types::{TransactionReceipt, trace::geth::DiffMode};
genesis::Genesis,
rpc::types::{TransactionReceipt, trace::geth::DiffMode},
};
use revive_dt_node_interaction::EthereumNode; use revive_dt_node_interaction::EthereumNode;
pub mod geth; pub mod geth;
/// The default genesis configuration.
pub const GENESIS_JSON: &str = include_str!("../../../genesis.json");
/// An abstract interface for testing nodes. /// An abstract interface for testing nodes.
pub trait Node: EthereumNode { pub trait Node: EthereumNode {
/// Spawns a node configured according to the [Genesis]. /// Spawns a node configured according to the genesis json.
/// ///
/// Blocking until it's ready to accept transactions. /// Blocking until it's ready to accept transactions.
fn spawn(&mut self, genesis: Genesis) -> anyhow::Result<&mut Self>; fn spawn(&mut self, genesis: String) -> anyhow::Result<()>;
/// Prune the node instance and related data. /// Prune the node instance and related data.
/// ///