mirror of
https://github.com/hl-archive-node/nanoreth.git
synced 2025-12-06 02:49:55 +00:00
97
src/call_forwarder.rs
Normal file
97
src/call_forwarder.rs
Normal file
@ -0,0 +1,97 @@
|
||||
use alloy_eips::BlockId;
|
||||
use alloy_primitives::{Bytes, U256};
|
||||
use alloy_rpc_types_eth::{state::StateOverride, transaction::TransactionRequest, BlockOverrides};
|
||||
use jsonrpsee::{
|
||||
http_client::{HttpClient, HttpClientBuilder},
|
||||
proc_macros::rpc,
|
||||
rpc_params,
|
||||
types::{error::INTERNAL_ERROR_CODE, ErrorObject},
|
||||
};
|
||||
use jsonrpsee_core::{async_trait, client::ClientT, ClientError, RpcResult};
|
||||
|
||||
#[rpc(server, namespace = "eth")]
|
||||
pub(crate) trait CallForwarderApi {
|
||||
/// Executes a new message call immediately without creating a transaction on the block chain.
|
||||
#[method(name = "call")]
|
||||
async fn call(
|
||||
&self,
|
||||
request: TransactionRequest,
|
||||
block_number: Option<BlockId>,
|
||||
state_overrides: Option<StateOverride>,
|
||||
block_overrides: Option<Box<BlockOverrides>>,
|
||||
) -> RpcResult<Bytes>;
|
||||
|
||||
/// Generates and returns an estimate of how much gas is necessary to allow the transaction to
|
||||
/// complete.
|
||||
#[method(name = "estimateGas")]
|
||||
async fn estimate_gas(
|
||||
&self,
|
||||
request: TransactionRequest,
|
||||
block_number: Option<BlockId>,
|
||||
state_override: Option<StateOverride>,
|
||||
) -> RpcResult<U256>;
|
||||
}
|
||||
|
||||
pub struct CallForwarderExt {
|
||||
client: HttpClient,
|
||||
}
|
||||
|
||||
impl CallForwarderExt {
|
||||
pub 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 CallForwarderApiServer for CallForwarderExt {
|
||||
async fn call(
|
||||
&self,
|
||||
request: TransactionRequest,
|
||||
block_number: Option<BlockId>,
|
||||
state_overrides: Option<StateOverride>,
|
||||
block_overrides: Option<Box<BlockOverrides>>,
|
||||
) -> RpcResult<Bytes> {
|
||||
let result = self
|
||||
.client
|
||||
.clone()
|
||||
.request(
|
||||
"eth_call",
|
||||
rpc_params![request, block_number, state_overrides, block_overrides],
|
||||
)
|
||||
.await
|
||||
.map_err(|e| match e {
|
||||
ClientError::Call(e) => e,
|
||||
_ => ErrorObject::owned(
|
||||
INTERNAL_ERROR_CODE,
|
||||
format!("Failed to call: {e:?}"),
|
||||
Some(()),
|
||||
),
|
||||
})?;
|
||||
Ok(result)
|
||||
}
|
||||
|
||||
async fn estimate_gas(
|
||||
&self,
|
||||
request: TransactionRequest,
|
||||
block_number: Option<BlockId>,
|
||||
state_override: Option<StateOverride>,
|
||||
) -> RpcResult<U256> {
|
||||
let result = self
|
||||
.client
|
||||
.clone()
|
||||
.request("eth_estimateGas", rpc_params![request, block_number, state_override])
|
||||
.await
|
||||
.map_err(|e| match e {
|
||||
ClientError::Call(e) => e,
|
||||
_ => ErrorObject::owned(
|
||||
INTERNAL_ERROR_CODE,
|
||||
format!("Failed to estimate gas: {e:?}"),
|
||||
Some(()),
|
||||
),
|
||||
})?;
|
||||
Ok(result)
|
||||
}
|
||||
}
|
||||
@ -6,5 +6,6 @@ pub mod hl_node_compliance;
|
||||
pub mod node;
|
||||
pub mod pseudo_peer;
|
||||
pub mod tx_forwarder;
|
||||
pub mod call_forwarder;
|
||||
|
||||
pub use node::primitives::{HlBlock, HlBlockBody, HlPrimitives};
|
||||
|
||||
@ -1,6 +1,7 @@
|
||||
use clap::Parser;
|
||||
use reth::builder::NodeHandle;
|
||||
use reth_hl::{
|
||||
call_forwarder::{self, CallForwarderApiServer},
|
||||
chainspec::parser::HlChainSpecParser,
|
||||
hl_node_compliance::install_hl_node_compliance,
|
||||
node::{
|
||||
@ -42,6 +43,13 @@ fn main() -> eyre::Result<()> {
|
||||
)?;
|
||||
info!("Transaction will be forwarded to {}", upstream_rpc_url);
|
||||
|
||||
if ext.forward_call {
|
||||
ctx.modules.replace_configured(
|
||||
call_forwarder::CallForwarderExt::new(upstream_rpc_url.clone()).into_rpc(),
|
||||
)?;
|
||||
info!("Call/gas estimation will be forwarded to {}", upstream_rpc_url);
|
||||
}
|
||||
|
||||
if ext.hl_node_compliant {
|
||||
install_hl_node_compliance(ctx)?;
|
||||
info!("hl-node compliant mode enabled");
|
||||
|
||||
@ -46,6 +46,12 @@ pub struct HlNodeArgs {
|
||||
/// 3. filters out logs and transactions from subscription.
|
||||
#[arg(long, env = "HL_NODE_COMPLIANT")]
|
||||
pub hl_node_compliant: bool,
|
||||
|
||||
/// Forward eth_call and eth_estimateGas to the upstream RPC.
|
||||
///
|
||||
/// This is useful when read precompile is needed for gas estimation.
|
||||
#[arg(long, env = "FORWARD_CALL")]
|
||||
pub forward_call: bool,
|
||||
}
|
||||
|
||||
/// The main reth_hl cli interface.
|
||||
|
||||
Reference in New Issue
Block a user