1use alloy_primitives::{Address, B256, U256};
4use clap::Parser;
5use foundry_config::{
6 Chain, Config,
7 figment::{
8 self, Metadata, Profile, Provider,
9 error::Kind::InvalidType,
10 value::{Dict, Map, Value},
11 },
12};
13use foundry_evm_networks::NetworkConfigs;
14use serde::Serialize;
15
16use crate::opts::RpcCommonOpts;
17use foundry_common::shell;
18
19#[derive(Clone, Debug, Default, Serialize, Parser)]
42#[command(next_help_heading = "EVM options", about = None, long_about = None)] pub struct EvmArgs {
44 #[command(flatten)]
46 #[serde(flatten)]
47 pub rpc: RpcCommonOpts,
48
49 #[arg(long, requires = "rpc_url", value_name = "BLOCK")]
53 #[serde(skip_serializing_if = "Option::is_none")]
54 pub fork_block_number: Option<u64>,
55
56 #[arg(long, requires = "rpc_url", value_name = "RETRIES")]
60 #[serde(skip_serializing_if = "Option::is_none")]
61 pub fork_retries: Option<u32>,
62
63 #[arg(long, requires = "rpc_url", value_name = "BACKOFF")]
67 #[serde(skip_serializing_if = "Option::is_none")]
68 pub fork_retry_backoff: Option<u64>,
69
70 #[arg(long)]
78 #[serde(skip)]
79 pub no_storage_caching: bool,
80
81 #[arg(long, value_name = "BALANCE")]
83 #[serde(skip_serializing_if = "Option::is_none")]
84 pub initial_balance: Option<U256>,
85
86 #[arg(long, value_name = "ADDRESS")]
88 #[serde(skip_serializing_if = "Option::is_none")]
89 pub sender: Option<Address>,
90
91 #[arg(long)]
93 #[serde(skip)]
94 pub ffi: bool,
95
96 #[arg(long)]
98 #[serde(skip)]
99 pub live_logs: bool,
100
101 #[arg(long)]
103 #[serde(skip)]
104 pub always_use_create_2_factory: bool,
105
106 #[arg(long, value_name = "ADDRESS")]
108 #[serde(skip_serializing_if = "Option::is_none")]
109 pub create2_deployer: Option<Address>,
110
111 #[command(flatten)]
113 #[serde(flatten)]
114 pub env: EnvArgs,
115
116 #[arg(long)]
120 #[serde(skip)]
121 pub isolate: bool,
122
123 #[command(flatten)]
125 #[serde(skip)]
126 pub networks: NetworkConfigs,
127}
128
129impl Provider for EvmArgs {
131 fn metadata(&self) -> Metadata {
132 Metadata::named("Evm Opts Provider")
133 }
134
135 fn data(&self) -> Result<Map<Profile, Dict>, figment::Error> {
136 let value = Value::serialize(self)?;
137 let error = InvalidType(value.to_actual(), "map".into());
138 let mut dict = value.into_dict().ok_or(error)?;
139
140 if shell::verbosity() > 0 {
141 dict.insert("verbosity".to_string(), shell::verbosity().into());
143 }
144
145 if self.ffi {
146 dict.insert("ffi".to_string(), self.ffi.into());
147 }
148
149 if self.live_logs {
150 dict.insert("live_logs".to_string(), self.live_logs.into());
151 }
152
153 if self.isolate {
154 dict.insert("isolate".to_string(), self.isolate.into());
155 }
156
157 if self.always_use_create_2_factory {
158 dict.insert(
159 "always_use_create_2_factory".to_string(),
160 self.always_use_create_2_factory.into(),
161 );
162 }
163
164 if self.no_storage_caching {
165 dict.insert("no_storage_caching".to_string(), self.no_storage_caching.into());
166 }
167
168 if self.rpc.no_rpc_rate_limit {
170 dict.insert("no_rpc_rate_limit".to_string(), true.into());
171 }
172 if self.rpc.accept_invalid_certs {
173 dict.insert("eth_rpc_accept_invalid_certs".to_string(), true.into());
174 }
175 if self.rpc.no_proxy {
176 dict.insert("eth_rpc_no_proxy".to_string(), true.into());
177 }
178
179 if let Some(name) = self.networks.active_network_name() {
182 dict.insert("network".to_string(), name.into());
183 }
184 if self.networks.is_celo() {
185 dict.insert("celo".to_string(), true.into());
186 }
187
188 Ok(Map::from([(Config::selected_profile(), dict)]))
189 }
190}
191
192#[derive(Clone, Debug, Default, Serialize, Parser)]
194#[command(next_help_heading = "Executor environment config")]
195pub struct EnvArgs {
196 #[arg(long, value_name = "CODE_SIZE")]
199 #[serde(skip_serializing_if = "Option::is_none")]
200 pub code_size_limit: Option<usize>,
201
202 #[arg(long, visible_alias = "chain-id", value_name = "CHAIN")]
204 #[serde(rename = "chain_id", skip_serializing_if = "Option::is_none", serialize_with = "id")]
205 pub chain: Option<Chain>,
206
207 #[arg(long, value_name = "GAS_PRICE")]
209 #[serde(skip_serializing_if = "Option::is_none")]
210 pub gas_price: Option<u64>,
211
212 #[arg(long, visible_alias = "base-fee", value_name = "FEE")]
214 #[serde(skip_serializing_if = "Option::is_none")]
215 pub block_base_fee_per_gas: Option<u64>,
216
217 #[arg(long, value_name = "ADDRESS")]
219 #[serde(skip_serializing_if = "Option::is_none")]
220 pub tx_origin: Option<Address>,
221
222 #[arg(long, value_name = "ADDRESS")]
224 #[serde(skip_serializing_if = "Option::is_none")]
225 pub block_coinbase: Option<Address>,
226
227 #[arg(long, value_name = "TIMESTAMP")]
229 #[serde(skip_serializing_if = "Option::is_none")]
230 pub block_timestamp: Option<u64>,
231
232 #[arg(long, value_name = "BLOCK")]
234 #[serde(skip_serializing_if = "Option::is_none")]
235 pub block_number: Option<u64>,
236
237 #[arg(long, value_name = "DIFFICULTY")]
239 #[serde(skip_serializing_if = "Option::is_none")]
240 pub block_difficulty: Option<u64>,
241
242 #[arg(long, value_name = "PREVRANDAO")]
244 #[serde(skip_serializing_if = "Option::is_none")]
245 pub block_prevrandao: Option<B256>,
246
247 #[arg(long, visible_alias = "gas-limit", value_name = "BLOCK_GAS_LIMIT")]
249 #[serde(skip_serializing_if = "Option::is_none")]
250 pub block_gas_limit: Option<u64>,
251
252 #[arg(long, value_name = "MEMORY_LIMIT")]
257 #[serde(skip_serializing_if = "Option::is_none")]
258 pub memory_limit: Option<u64>,
259
260 #[arg(long, visible_aliases = &["no-block-gas-limit", "no-gas-limit"])]
262 #[serde(skip_serializing_if = "std::ops::Not::not")]
263 pub disable_block_gas_limit: bool,
264
265 #[arg(long, visible_alias = "tx-gas-limit")]
267 #[serde(skip_serializing_if = "std::ops::Not::not")]
268 pub enable_tx_gas_limit: bool,
269}
270
271fn id<S: serde::Serializer>(chain: &Option<Chain>, s: S) -> Result<S::Ok, S::Error> {
274 if let Some(chain) = chain {
275 s.serialize_u64(chain.id())
276 } else {
277 unreachable!()
279 }
280}
281
282#[cfg(test)]
283mod tests {
284 use super::*;
285 use foundry_config::NamedChain;
286
287 #[test]
288 fn compute_units_per_second_skips_when_none() {
289 let args = EvmArgs::default();
290 let data = args.data().expect("provider data");
291 let dict = data.get(&Config::selected_profile()).expect("profile dict");
292 assert!(
293 !dict.contains_key("compute_units_per_second"),
294 "compute_units_per_second should be skipped when None"
295 );
296 }
297
298 #[test]
299 fn compute_units_per_second_present_when_some() {
300 let args = EvmArgs {
301 rpc: RpcCommonOpts { compute_units_per_second: Some(1000), ..Default::default() },
302 ..Default::default()
303 };
304 let data = args.data().expect("provider data");
305 let dict = data.get(&Config::selected_profile()).expect("profile dict");
306 let val = dict.get("compute_units_per_second").expect("cups present");
307 assert_eq!(val, &Value::from(1000u64));
308 }
309
310 #[test]
311 fn rpc_url_arg_does_not_read_eth_rpc_url_env() {
312 use clap::CommandFactory;
313
314 let command = EvmArgs::command();
315 let rpc_url =
316 command.get_arguments().find(|arg| arg.get_id() == "rpc_url").expect("rpc_url arg");
317
318 assert!(rpc_url.get_env().is_none());
319 }
320
321 #[test]
322 fn can_parse_chain_id() {
323 let args = EvmArgs {
324 env: EnvArgs { chain: Some(NamedChain::Mainnet.into()), ..Default::default() },
325 ..Default::default()
326 };
327 let config = Config::from_provider(Config::figment().merge(args)).unwrap();
328 assert_eq!(config.chain, Some(NamedChain::Mainnet.into()));
329
330 let env = EnvArgs::parse_from(["foundry-cli", "--chain-id", "goerli"]);
331 assert_eq!(env.chain, Some(NamedChain::Goerli.into()));
332 }
333
334 #[test]
335 fn test_memory_limit() {
336 let args = EvmArgs {
337 env: EnvArgs { chain: Some(NamedChain::Mainnet.into()), ..Default::default() },
338 ..Default::default()
339 };
340 let config = Config::from_provider(Config::figment().merge(args)).unwrap();
341 assert_eq!(config.memory_limit, Config::default().memory_limit);
342
343 let env = EnvArgs::parse_from(["foundry-cli", "--memory-limit", "100"]);
344 assert_eq!(env.memory_limit, Some(100));
345 }
346
347 #[test]
348 fn test_chain_id() {
349 let env = EnvArgs::parse_from(["foundry-cli", "--chain-id", "1"]);
350 assert_eq!(env.chain, Some(Chain::mainnet()));
351
352 let env = EnvArgs::parse_from(["foundry-cli", "--chain-id", "mainnet"]);
353 assert_eq!(env.chain, Some(Chain::mainnet()));
354 let args = EvmArgs { env, ..Default::default() };
355 let config = Config::from_provider(Config::figment().merge(args)).unwrap();
356 assert_eq!(config.chain, Some(Chain::mainnet()));
357 }
358}