mirror of
https://github.com/hl-archive-node/nanoreth.git
synced 2025-12-06 10:59:55 +00:00
feat: Forward incoming txs to upstream server
This commit is contained in:
1
Cargo.lock
generated
1
Cargo.lock
generated
@ -6659,6 +6659,7 @@ dependencies = [
|
|||||||
"eyre",
|
"eyre",
|
||||||
"futures",
|
"futures",
|
||||||
"jsonrpsee",
|
"jsonrpsee",
|
||||||
|
"jsonrpsee-core",
|
||||||
"lz4_flex",
|
"lz4_flex",
|
||||||
"once_cell",
|
"once_cell",
|
||||||
"parking_lot",
|
"parking_lot",
|
||||||
|
|||||||
@ -97,6 +97,7 @@ reth-e2e-test-utils.workspace = true
|
|||||||
once_cell.workspace = true
|
once_cell.workspace = true
|
||||||
reth-ethereum-forks.workspace = true
|
reth-ethereum-forks.workspace = true
|
||||||
jsonrpsee.workspace = true
|
jsonrpsee.workspace = true
|
||||||
|
jsonrpsee-core.workspace = true
|
||||||
reth-rpc-layer.workspace = true
|
reth-rpc-layer.workspace = true
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
|
|||||||
45
bin/reth/src/forwarder.rs
Normal file
45
bin/reth/src/forwarder.rs
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
use alloy_primitives::{Bytes, B256};
|
||||||
|
use jsonrpsee::{
|
||||||
|
http_client::{HttpClient, HttpClientBuilder},
|
||||||
|
proc_macros::rpc,
|
||||||
|
types::{error::INTERNAL_ERROR_CODE, ErrorObject},
|
||||||
|
};
|
||||||
|
use jsonrpsee_core::{async_trait, client::ClientT, ClientError, RpcResult};
|
||||||
|
|
||||||
|
#[rpc(server, namespace = "eth")]
|
||||||
|
pub(crate) trait EthForwarderApi {
|
||||||
|
/// Returns block 0.
|
||||||
|
#[method(name = "sendRawTransaction")]
|
||||||
|
async fn send_raw_transaction(&self, tx: Bytes) -> RpcResult<B256>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) struct EthForwarderExt {
|
||||||
|
client: HttpClient,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl EthForwarderExt {
|
||||||
|
pub(crate) fn new(upstream_rpc_url: String) -> Self {
|
||||||
|
let client =
|
||||||
|
HttpClientBuilder::default().build(upstream_rpc_url).expect("Failed to build client");
|
||||||
|
|
||||||
|
Self { client }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl EthForwarderApiServer for EthForwarderExt {
|
||||||
|
async fn send_raw_transaction(&self, tx: Bytes) -> RpcResult<B256> {
|
||||||
|
let txhash =
|
||||||
|
self.client.clone().request("eth_sendRawTransaction", vec![tx]).await.map_err(|e| {
|
||||||
|
match e {
|
||||||
|
ClientError::Call(e) => e,
|
||||||
|
_ => ErrorObject::owned(
|
||||||
|
INTERNAL_ERROR_CODE,
|
||||||
|
format!("Failed to send transaction: {:?}", e),
|
||||||
|
Some(()),
|
||||||
|
),
|
||||||
|
}
|
||||||
|
})?;
|
||||||
|
Ok(txhash)
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -4,22 +4,28 @@
|
|||||||
static ALLOC: reth_cli_util::allocator::Allocator = reth_cli_util::allocator::new_allocator();
|
static ALLOC: reth_cli_util::allocator::Allocator = reth_cli_util::allocator::new_allocator();
|
||||||
|
|
||||||
mod block_ingest;
|
mod block_ingest;
|
||||||
|
mod forwarder;
|
||||||
mod serialized;
|
mod serialized;
|
||||||
|
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use block_ingest::BlockIngest;
|
use block_ingest::BlockIngest;
|
||||||
use clap::{Args, Parser};
|
use clap::{Args, Parser};
|
||||||
|
use forwarder::EthForwarderApiServer;
|
||||||
use reth::cli::Cli;
|
use reth::cli::Cli;
|
||||||
use reth_ethereum_cli::chainspec::EthereumChainSpecParser;
|
use reth_ethereum_cli::chainspec::EthereumChainSpecParser;
|
||||||
use reth_node_ethereum::EthereumNode;
|
use reth_node_ethereum::EthereumNode;
|
||||||
use tracing::info;
|
use tracing::info;
|
||||||
|
|
||||||
#[derive(Args, Debug, Clone)]
|
#[derive(Args, Debug, Clone)]
|
||||||
struct IngestArgs {
|
struct HyperliquidExtArgs {
|
||||||
/// EVM blocks base directory
|
/// EVM blocks base directory
|
||||||
#[arg(long, default_value="/tmp/evm-blocks")]
|
#[arg(long, default_value = "/tmp/evm-blocks")]
|
||||||
pub ingest_dir: PathBuf,
|
pub ingest_dir: PathBuf,
|
||||||
|
|
||||||
|
/// Upstream RPC URL to forward incoming transactions.
|
||||||
|
#[arg(long, default_value = "https://rpc.hyperliquid.xyz/evm")]
|
||||||
|
pub upstream_rpc_url: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
@ -30,15 +36,30 @@ fn main() {
|
|||||||
std::env::set_var("RUST_BACKTRACE", "1");
|
std::env::set_var("RUST_BACKTRACE", "1");
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Err(err) = Cli::<EthereumChainSpecParser, IngestArgs>::parse().run(|builder, ingest_args| async move {
|
if let Err(err) = Cli::<EthereumChainSpecParser, HyperliquidExtArgs>::parse().run(
|
||||||
|
|builder, ingest_args| async move {
|
||||||
info!(target: "reth::cli", "Launching node");
|
info!(target: "reth::cli", "Launching node");
|
||||||
let handle = builder.launch_node(EthereumNode::default()).await?;
|
let handle = builder
|
||||||
|
.node(EthereumNode::default())
|
||||||
|
.extend_rpc_modules(move |ctx| {
|
||||||
|
let upstream_rpc_url = ingest_args.upstream_rpc_url.clone();
|
||||||
|
ctx.modules.remove_method_from_configured("eth_sendRawTransaction");
|
||||||
|
ctx.modules.merge_configured(
|
||||||
|
forwarder::EthForwarderExt::new(upstream_rpc_url).into_rpc(),
|
||||||
|
)?;
|
||||||
|
|
||||||
|
info!("Transaction forwarder extension enabled");
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.launch()
|
||||||
|
.await?;
|
||||||
|
|
||||||
let ingest_dir = ingest_args.ingest_dir;
|
let ingest_dir = ingest_args.ingest_dir;
|
||||||
let ingest = BlockIngest(ingest_dir);
|
let ingest = BlockIngest(ingest_dir);
|
||||||
ingest.run(handle.node).await.unwrap();
|
ingest.run(handle.node).await.unwrap();
|
||||||
handle.node_exit_future.await
|
handle.node_exit_future.await
|
||||||
}) {
|
},
|
||||||
|
) {
|
||||||
eprintln!("Error: {err:?}");
|
eprintln!("Error: {err:?}");
|
||||||
std::process::exit(1);
|
std::process::exit(1);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user