mirror of
https://github.com/pezkuwichain/revive-differential-tests.git
synced 2026-06-15 18:21:08 +00:00
spawn geth node
Signed-off-by: Cyrill Leutwiler <bigcyrill@hotmail.com>
This commit is contained in:
Generated
+2
@@ -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",
|
||||||
|
|||||||
@@ -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
@@ -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"])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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 }
|
||||||
|
|||||||
@@ -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:?}");
|
||||||
|
|||||||
@@ -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
@@ -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();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -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.
|
||||||
///
|
///
|
||||||
|
|||||||
Reference in New Issue
Block a user