reth_rpc_eth_types/builder/
config.rs

1//! Configuration for `eth` namespace APIs.
2
3use std::time::Duration;
4
5use crate::{
6    EthStateCacheConfig, FeeHistoryCacheConfig, ForwardConfig, GasPriceOracleConfig,
7    RPC_DEFAULT_GAS_CAP,
8};
9use reqwest::Url;
10use reth_rpc_server_types::constants::{
11    default_max_tracing_requests, DEFAULT_ETH_PROOF_WINDOW, DEFAULT_MAX_BLOCKS_PER_FILTER,
12    DEFAULT_MAX_LOGS_PER_RESPONSE, DEFAULT_MAX_SIMULATE_BLOCKS, DEFAULT_MAX_TRACE_FILTER_BLOCKS,
13    DEFAULT_PROOF_PERMITS,
14};
15use serde::{Deserialize, Serialize};
16
17/// Default value for stale filter ttl
18pub const DEFAULT_STALE_FILTER_TTL: Duration = Duration::from_secs(5 * 60);
19
20/// Config for the locally built pending block
21#[derive(Debug, Clone, Copy, Eq, PartialEq, Serialize, Deserialize, Default)]
22#[serde(rename_all = "lowercase")]
23pub enum PendingBlockKind {
24    /// Return a pending block with header only, no transactions included
25    Empty,
26    /// Return null/no pending block
27    None,
28    /// Return a pending block with all transactions from the mempool (default behavior)
29    #[default]
30    Full,
31}
32
33impl std::str::FromStr for PendingBlockKind {
34    type Err = String;
35
36    fn from_str(s: &str) -> Result<Self, Self::Err> {
37        match s.to_lowercase().as_str() {
38            "empty" => Ok(Self::Empty),
39            "none" => Ok(Self::None),
40            "full" => Ok(Self::Full),
41            _ => Err(format!(
42                "Invalid pending block kind: {s}. Valid options are: empty, none, full"
43            )),
44        }
45    }
46}
47
48impl PendingBlockKind {
49    /// Returns true if the pending block kind is `None`
50    pub const fn is_none(&self) -> bool {
51        matches!(self, Self::None)
52    }
53
54    /// Returns true if the pending block kind is `Empty`
55    pub const fn is_empty(&self) -> bool {
56        matches!(self, Self::Empty)
57    }
58}
59
60/// Additional config values for the eth namespace.
61#[derive(Debug, Clone, Eq, PartialEq, Serialize, Deserialize)]
62pub struct EthConfig {
63    /// Settings for the caching layer
64    pub cache: EthStateCacheConfig,
65    /// Settings for the gas price oracle
66    pub gas_oracle: GasPriceOracleConfig,
67    /// The maximum number of blocks into the past for generating state proofs.
68    pub eth_proof_window: u64,
69    /// The maximum number of tracing calls that can be executed in concurrently.
70    pub max_tracing_requests: usize,
71    /// Maximum number of blocks for `trace_filter` requests.
72    pub max_trace_filter_blocks: u64,
73    /// Maximum number of blocks that could be scanned per filter request in `eth_getLogs` calls.
74    pub max_blocks_per_filter: u64,
75    /// Maximum number of logs that can be returned in a single response in `eth_getLogs` calls.
76    pub max_logs_per_response: usize,
77    /// Gas limit for `eth_call` and call tracing RPC methods.
78    ///
79    /// Defaults to [`RPC_DEFAULT_GAS_CAP`]
80    pub rpc_gas_cap: u64,
81    /// Max number of blocks for `eth_simulateV1`.
82    pub rpc_max_simulate_blocks: u64,
83    ///
84    /// Sets TTL for stale filters
85    pub stale_filter_ttl: Duration,
86    /// Settings for the fee history cache
87    pub fee_history_cache: FeeHistoryCacheConfig,
88    /// The maximum number of getproof calls that can be executed concurrently.
89    pub proof_permits: usize,
90    /// Maximum batch size for transaction pool insertions.
91    pub max_batch_size: usize,
92    /// Controls how pending blocks are built when requested via RPC methods
93    pub pending_block_kind: PendingBlockKind,
94    /// The raw transaction forwarder.
95    pub raw_tx_forwarder: ForwardConfig,
96}
97
98impl EthConfig {
99    /// Returns the filter config for the `eth_filter` handler.
100    pub fn filter_config(&self) -> EthFilterConfig {
101        EthFilterConfig::default()
102            .max_blocks_per_filter(self.max_blocks_per_filter)
103            .max_logs_per_response(self.max_logs_per_response)
104            .stale_filter_ttl(self.stale_filter_ttl)
105    }
106}
107
108impl Default for EthConfig {
109    fn default() -> Self {
110        Self {
111            cache: EthStateCacheConfig::default(),
112            gas_oracle: GasPriceOracleConfig::default(),
113            eth_proof_window: DEFAULT_ETH_PROOF_WINDOW,
114            max_tracing_requests: default_max_tracing_requests(),
115            max_trace_filter_blocks: DEFAULT_MAX_TRACE_FILTER_BLOCKS,
116            max_blocks_per_filter: DEFAULT_MAX_BLOCKS_PER_FILTER,
117            max_logs_per_response: DEFAULT_MAX_LOGS_PER_RESPONSE,
118            rpc_gas_cap: RPC_DEFAULT_GAS_CAP.into(),
119            rpc_max_simulate_blocks: DEFAULT_MAX_SIMULATE_BLOCKS,
120            stale_filter_ttl: DEFAULT_STALE_FILTER_TTL,
121            fee_history_cache: FeeHistoryCacheConfig::default(),
122            proof_permits: DEFAULT_PROOF_PERMITS,
123            max_batch_size: 1,
124            pending_block_kind: PendingBlockKind::Full,
125            raw_tx_forwarder: ForwardConfig::default(),
126        }
127    }
128}
129
130impl EthConfig {
131    /// Configures the caching layer settings
132    pub const fn state_cache(mut self, cache: EthStateCacheConfig) -> Self {
133        self.cache = cache;
134        self
135    }
136
137    /// Configures the gas price oracle settings
138    pub const fn gpo_config(mut self, gas_oracle_config: GasPriceOracleConfig) -> Self {
139        self.gas_oracle = gas_oracle_config;
140        self
141    }
142
143    /// Configures the maximum number of tracing requests
144    pub const fn max_tracing_requests(mut self, max_requests: usize) -> Self {
145        self.max_tracing_requests = max_requests;
146        self
147    }
148
149    /// Configures the maximum block length to scan per `eth_getLogs` request
150    pub const fn max_blocks_per_filter(mut self, max_blocks: u64) -> Self {
151        self.max_blocks_per_filter = max_blocks;
152        self
153    }
154
155    /// Configures the maximum number of blocks for `trace_filter` requests
156    pub const fn max_trace_filter_blocks(mut self, max_blocks: u64) -> Self {
157        self.max_trace_filter_blocks = max_blocks;
158        self
159    }
160
161    /// Configures the maximum number of logs per response
162    pub const fn max_logs_per_response(mut self, max_logs: usize) -> Self {
163        self.max_logs_per_response = max_logs;
164        self
165    }
166
167    /// Configures the maximum gas limit for `eth_call` and call tracing RPC methods
168    pub const fn rpc_gas_cap(mut self, rpc_gas_cap: u64) -> Self {
169        self.rpc_gas_cap = rpc_gas_cap;
170        self
171    }
172
173    /// Configures the maximum gas limit for `eth_call` and call tracing RPC methods
174    pub const fn rpc_max_simulate_blocks(mut self, max_blocks: u64) -> Self {
175        self.rpc_max_simulate_blocks = max_blocks;
176        self
177    }
178
179    /// Configures the maximum proof window for historical proof generation.
180    pub const fn eth_proof_window(mut self, window: u64) -> Self {
181        self.eth_proof_window = window;
182        self
183    }
184
185    /// Configures the number of getproof requests
186    pub const fn proof_permits(mut self, permits: usize) -> Self {
187        self.proof_permits = permits;
188        self
189    }
190
191    /// Configures the maximum batch size for transaction pool insertions
192    pub const fn max_batch_size(mut self, max_batch_size: usize) -> Self {
193        self.max_batch_size = max_batch_size;
194        self
195    }
196
197    /// Configures the pending block config
198    pub const fn pending_block_kind(mut self, pending_block_kind: PendingBlockKind) -> Self {
199        self.pending_block_kind = pending_block_kind;
200        self
201    }
202
203    /// Configures the raw transaction forwarder.
204    pub fn raw_tx_forwarder(mut self, tx_forwarder: Option<Url>) -> Self {
205        if let Some(tx_forwarder) = tx_forwarder {
206            self.raw_tx_forwarder.tx_forwarder = Some(tx_forwarder);
207        }
208        self
209    }
210}
211
212/// Config for the filter
213#[derive(Debug, Clone, PartialEq, Eq)]
214pub struct EthFilterConfig {
215    /// Maximum number of blocks that a filter can scan for logs.
216    ///
217    /// If `None` then no limit is enforced.
218    pub max_blocks_per_filter: Option<u64>,
219    /// Maximum number of logs that can be returned in a single response in `eth_getLogs` calls.
220    ///
221    /// If `None` then no limit is enforced.
222    pub max_logs_per_response: Option<usize>,
223    /// How long a filter remains valid after the last poll.
224    ///
225    /// A filter is considered stale if it has not been polled for longer than this duration and
226    /// will be removed.
227    pub stale_filter_ttl: Duration,
228}
229
230impl EthFilterConfig {
231    /// Sets the maximum number of blocks that a filter can scan for logs.
232    pub const fn max_blocks_per_filter(mut self, num: u64) -> Self {
233        self.max_blocks_per_filter = Some(num);
234        self
235    }
236
237    /// Sets the maximum number of logs that can be returned in a single response in `eth_getLogs`
238    /// calls.
239    pub const fn max_logs_per_response(mut self, num: usize) -> Self {
240        self.max_logs_per_response = Some(num);
241        self
242    }
243
244    /// Sets how long a filter remains valid after the last poll before it will be removed.
245    pub const fn stale_filter_ttl(mut self, duration: Duration) -> Self {
246        self.stale_filter_ttl = duration;
247        self
248    }
249}
250
251impl Default for EthFilterConfig {
252    fn default() -> Self {
253        Self {
254            max_blocks_per_filter: None,
255            max_logs_per_response: None,
256            // 5min
257            stale_filter_ttl: Duration::from_secs(5 * 60),
258        }
259    }
260}