diff --git a/crates/cheatcodes/src/config.rs b/crates/cheatcodes/src/config.rs index 9ce0117e8..44d209f5a 100644 --- a/crates/cheatcodes/src/config.rs +++ b/crates/cheatcodes/src/config.rs @@ -1,8 +1,5 @@ use super::Result; -use crate::{ - strategy::{CheatcodeInspectorStrategy, EvmCheatcodeInspectorStrategy}, - Vm::Rpc, -}; +use crate::{strategy::CheatcodeInspectorStrategy, Vm::Rpc}; use alloy_primitives::{map::AddressHashMap, U256}; use foundry_common::{fs::normalize_path, ContractsByArtifact}; use foundry_compilers::{utils::canonicalize, ProjectPathsConfig}; @@ -57,7 +54,7 @@ pub struct CheatsConfig { /// Version of the script/test contract which is currently running. pub running_version: Option, /// The behavior strategy. - pub strategy: Box, + pub strategy: CheatcodeInspectorStrategy, /// Whether to enable legacy (non-reverting) assertions. pub assertions_revert: bool, /// Optional seed for the RNG algorithm. @@ -73,7 +70,7 @@ impl CheatsConfig { available_artifacts: Option, running_contract: Option, running_version: Option, - strategy: Box, + strategy: CheatcodeInspectorStrategy, ) -> Self { let mut allowed_paths = vec![config.root.clone()]; allowed_paths.extend(config.libs.iter().cloned()); @@ -117,7 +114,7 @@ impl CheatsConfig { self.available_artifacts.clone(), self.running_contract.clone(), self.running_version.clone(), - self.strategy.new_cloned(), + self.strategy.clone(), ) } @@ -246,7 +243,7 @@ impl Default for CheatsConfig { available_artifacts: Default::default(), running_contract: Default::default(), running_version: Default::default(), - strategy: Box::new(EvmCheatcodeInspectorStrategy::default()), + strategy: CheatcodeInspectorStrategy::new_evm(), assertions_revert: true, seed: None, } @@ -265,7 +262,7 @@ mod tests { None, None, None, - Box::new(EvmCheatcodeInspectorStrategy::default()), + CheatcodeInspectorStrategy::new_evm(), ) } diff --git a/crates/cheatcodes/src/evm.rs b/crates/cheatcodes/src/evm.rs index 15332c6c2..f1546061d 100644 --- a/crates/cheatcodes/src/evm.rs +++ b/crates/cheatcodes/src/evm.rs @@ -134,7 +134,7 @@ impl Cheatcode for getNonce_0Call { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { account } = self; - ccx.with_strategy(|strategy, ccx| strategy.cheatcode_get_nonce(ccx, *account)) + ccx.state.strategy.runner.clone().cheatcode_get_nonce(ccx, *account) } } @@ -420,7 +420,7 @@ impl Cheatcode for getBlobhashesCall { impl Cheatcode for rollCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { newHeight } = self; - ccx.with_strategy(|strategy, ccx| strategy.cheatcode_roll(ccx, *newHeight)) + ccx.state.strategy.runner.clone().cheatcode_roll(ccx, *newHeight) } } @@ -442,7 +442,7 @@ impl Cheatcode for txGasPriceCall { impl Cheatcode for warpCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { newTimestamp } = self; - ccx.with_strategy(|strategy, ccx| strategy.cheatcode_warp(ccx, *newTimestamp)) + ccx.state.strategy.runner.clone().cheatcode_warp(ccx, *newTimestamp) } } @@ -477,7 +477,7 @@ impl Cheatcode for dealCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { account: address, newBalance: new_balance } = *self; - ccx.with_strategy(|strategy, ccx| strategy.cheatcode_deal(ccx, address, new_balance)) + ccx.state.strategy.runner.clone().cheatcode_deal(ccx, address, new_balance) } } @@ -485,14 +485,14 @@ impl Cheatcode for etchCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { target, newRuntimeBytecode } = self; - ccx.with_strategy(|strategy, ccx| strategy.cheatcode_etch(ccx, *target, newRuntimeBytecode)) + ccx.state.strategy.runner.clone().cheatcode_etch(ccx, *target, newRuntimeBytecode) } } impl Cheatcode for resetNonceCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { account } = self; - ccx.with_strategy(|strategy, ccx| strategy.cheatcode_reset_nonce(ccx, *account)) + ccx.state.strategy.runner.clone().cheatcode_reset_nonce(ccx, *account) } } @@ -500,7 +500,7 @@ impl Cheatcode for setNonceCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { account, newNonce } = *self; - ccx.with_strategy(|strategy, ccx| strategy.cheatcode_set_nonce(ccx, account, newNonce)) + ccx.state.strategy.runner.clone().cheatcode_set_nonce(ccx, account, newNonce) } } @@ -508,9 +508,7 @@ impl Cheatcode for setNonceUnsafeCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { account, newNonce } = *self; - ccx.with_strategy(|strategy, ccx| { - strategy.cheatcode_set_nonce_unsafe(ccx, account, newNonce) - }) + ccx.state.strategy.runner.clone().cheatcode_set_nonce_unsafe(ccx, account, newNonce) } } diff --git a/crates/cheatcodes/src/evm/fork.rs b/crates/cheatcodes/src/evm/fork.rs index 17bb9bacf..e1db17a3d 100644 --- a/crates/cheatcodes/src/evm/fork.rs +++ b/crates/cheatcodes/src/evm/fork.rs @@ -125,7 +125,11 @@ impl Cheatcode for selectForkCall { persist_caller(ccx); check_broadcast(ccx.state)?; - ccx.with_strategy(|strategy, ccx| strategy.zksync_select_fork_vm(ccx.ecx, *forkId)); + ccx.state.strategy.runner.zksync_select_fork_vm( + ccx.state.strategy.context.as_mut(), + ccx.ecx, + *forkId, + ); ccx.ecx.db.select_fork(*forkId, &mut ccx.ecx.env, &mut ccx.ecx.journaled_state)?; Ok(Default::default()) @@ -281,7 +285,11 @@ fn create_select_fork(ccx: &mut CheatsCtxt, url_or_alias: &str, block: Option Result { let Self { callee, data, returnData } = self; - ccx.with_strategy(|strategy, ccx| { - strategy.cheatcode_mock_call(ccx, *callee, data, returnData) - }) + ccx.state.strategy.runner.clone().cheatcode_mock_call(ccx, *callee, data, returnData) } } @@ -85,9 +83,7 @@ impl Cheatcode for mockCalls_1Call { impl Cheatcode for mockCallRevert_0Call { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { callee, data, revertData } = self; - ccx.with_strategy(|strategy, ccx| { - strategy.cheatcode_mock_call_revert(ccx, *callee, data, revertData) - }) + ccx.state.strategy.runner.clone().cheatcode_mock_call_revert(ccx, *callee, data, revertData) } } diff --git a/crates/cheatcodes/src/fs.rs b/crates/cheatcodes/src/fs.rs index 4dafd3926..ae955a4dc 100644 --- a/crates/cheatcodes/src/fs.rs +++ b/crates/cheatcodes/src/fs.rs @@ -283,7 +283,7 @@ impl Cheatcode for getArtifactPathByDeployedCodeCall { impl Cheatcode for getCodeCall { fn apply(&self, state: &mut Cheatcodes) -> Result { let Self { artifactPath: path } = self; - state.with_strategy(|strategy, state| strategy.get_artifact_code(state, path, false)) + state.strategy.runner.get_artifact_code(state, path, false) } } diff --git a/crates/cheatcodes/src/inspector.rs b/crates/cheatcodes/src/inspector.rs index 6725f9abe..4900c6674 100644 --- a/crates/cheatcodes/src/inspector.rs +++ b/crates/cheatcodes/src/inspector.rs @@ -69,7 +69,6 @@ pub use utils::CommonCreateInput; pub type Ecx<'a, 'b, 'c> = &'a mut EvmContext<&'b mut (dyn DatabaseExt + 'c)>; pub type InnerEcx<'a, 'b, 'c> = &'a mut InnerEvmContext<&'b mut (dyn DatabaseExt + 'c)>; -pub type Strategy<'a> = &'a mut dyn CheatcodeInspectorStrategy; /// Helper trait for obtaining complete [revm::Inspector] instance from mutable reference to /// [Cheatcodes]. @@ -528,7 +527,7 @@ pub struct Cheatcodes { pub wallets: Option, /// The behavior strategy. - pub strategy: Option>, + pub strategy: CheatcodeInspectorStrategy, } impl Clone for Cheatcodes { @@ -568,7 +567,7 @@ impl Clone for Cheatcodes { arbitrary_storage: self.arbitrary_storage.clone(), deprecated: self.deprecated.clone(), wallets: self.wallets.clone(), - strategy: self.strategy.as_ref().map(|s| s.new_cloned()), + strategy: self.strategy.clone(), } } } @@ -588,7 +587,7 @@ impl Cheatcodes { Self { fs_commit: true, labels: config.labels.clone(), - strategy: Some(config.strategy.clone()), + strategy: config.strategy.clone(), config, block: Default::default(), active_delegation: Default::default(), @@ -763,7 +762,8 @@ impl Cheatcodes { if ecx_inner.journaled_state.depth() == broadcast.depth { input.set_caller(broadcast.new_origin); - self.strategy.as_mut().unwrap().record_broadcastable_create_transactions( + self.strategy.runner.record_broadcastable_create_transactions( + self.strategy.context.as_mut(), self.config.clone(), &input, ecx_inner, @@ -801,9 +801,9 @@ impl Cheatcodes { }]); } - if let Some(result) = self.with_strategy(|strategy, cheatcodes| { - strategy.zksync_try_create(cheatcodes, ecx, &input, executor) - }) { + if let Some(result) = + self.strategy.runner.clone().zksync_try_create(self, ecx, &input, executor) + { return Some(result); } @@ -924,10 +924,7 @@ where { } } - self.strategy - .as_mut() - .expect("failed acquiring strategy") - .zksync_record_create_address(&outcome); + self.strategy.runner.zksync_record_create_address(self.strategy.context.as_mut(), &outcome); outcome } @@ -970,10 +967,12 @@ where { let prev = account.info.nonce; let nonce = prev.saturating_sub(1); account.info.nonce = nonce; - self.strategy - .as_mut() - .expect("failed acquiring strategy") - .zksync_sync_nonce(sender, nonce, ecx); + self.strategy.runner.zksync_sync_nonce( + self.strategy.context.as_mut(), + sender, + nonce, + ecx, + ); trace!(target: "cheatcodes", %sender, nonce, prev, "corrected nonce"); } @@ -1007,10 +1006,7 @@ where { return None; } - self.strategy - .as_mut() - .expect("failed acquiring strategy") - .zksync_set_deployer_call_input(call); + self.strategy.runner.zksync_set_deployer_call_input(self.strategy.context.as_mut(), call); // Handle expected calls @@ -1145,17 +1141,15 @@ where { }) } - self.strategy - .as_mut() - .expect("failed acquiring strategy") - .record_broadcastable_call_transactions( - self.config.clone(), - call, - ecx_inner, - broadcast, - &mut self.broadcastable_transactions, - &mut self.active_delegation, - ); + self.strategy.runner.record_broadcastable_call_transactions( + self.strategy.context.as_mut(), + self.config.clone(), + call, + ecx_inner, + broadcast, + &mut self.broadcastable_transactions, + &mut self.active_delegation, + ); let account = ecx_inner.journaled_state.state().get_mut(&broadcast.new_origin).unwrap(); @@ -1228,9 +1222,9 @@ where { }]); } - if let Some(result) = self.with_strategy(|strategy, cheatcodes| { - strategy.zksync_try_call(cheatcodes, ecx, call, executor) - }) { + if let Some(result) = + self.strategy.runner.clone().zksync_try_call(self, ecx, call, executor) + { return Some(result); } @@ -1272,17 +1266,6 @@ where { None => false, } } - - pub fn with_strategy(&mut self, mut f: F) -> R - where - F: FnMut(Strategy, &mut Self) -> R, - { - let mut strategy = self.strategy.take(); - let result = f(strategy.as_mut().expect("failed acquiring strategy").as_mut(), self); - self.strategy = strategy; - - result - } } impl Inspector<&mut dyn DatabaseExt> for Cheatcodes { @@ -1302,10 +1285,11 @@ impl Inspector<&mut dyn DatabaseExt> for Cheatcodes { self.gas_metering.paused_frames.push(interpreter.gas); } - self.strategy - .as_mut() - .expect("failed acquiring strategy") - .post_initialize_interp(interpreter, ecx); + self.strategy.runner.post_initialize_interp( + self.strategy.context.as_mut(), + interpreter, + ecx, + ); } #[inline] @@ -1350,8 +1334,7 @@ impl Inspector<&mut dyn DatabaseExt> for Cheatcodes { #[inline] fn step_end(&mut self, interpreter: &mut Interpreter, ecx: Ecx) { - if self.strategy.as_mut().expect("failed acquiring strategy").pre_step_end(interpreter, ecx) - { + if self.strategy.runner.pre_step_end(self.strategy.context.as_mut(), interpreter, ecx) { return; } diff --git a/crates/cheatcodes/src/lib.rs b/crates/cheatcodes/src/lib.rs index 56b6cfb43..33dec1c33 100644 --- a/crates/cheatcodes/src/lib.rs +++ b/crates/cheatcodes/src/lib.rs @@ -17,7 +17,6 @@ extern crate tracing; use alloy_primitives::Address; use foundry_evm_core::backend::DatabaseExt; -use inspector::Strategy; use revm::{ContextPrecompiles, InnerEvmContext}; use spec::Status; @@ -175,15 +174,4 @@ impl CheatsCtxt<'_, '_, '_, '_> { pub(crate) fn is_precompile(&self, address: &Address) -> bool { self.precompiles.contains(address) } - - pub(crate) fn with_strategy(&mut self, mut f: F) -> R - where - F: FnMut(Strategy, &mut Self) -> R, - { - let mut strategy = self.state.strategy.take(); - let result = f(strategy.as_mut().expect("failed acquiring strategy").as_mut(), self); - self.state.strategy = strategy; - - result - } } diff --git a/crates/cheatcodes/src/strategy.rs b/crates/cheatcodes/src/strategy.rs index 91154b660..42d1f6fbf 100644 --- a/crates/cheatcodes/src/strategy.rs +++ b/crates/cheatcodes/src/strategy.rs @@ -1,4 +1,4 @@ -use std::{fmt::Debug, sync::Arc}; +use std::{any::Any, fmt::Debug, sync::Arc}; use alloy_primitives::{Address, Bytes, FixedBytes, TxKind, U256}; use alloy_rpc_types::{TransactionInput, TransactionRequest}; @@ -18,39 +18,84 @@ use crate::{ CheatsConfig, CheatsCtxt, Result, }; -pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorStrategyExt { +/// Represents the context for [CheatcodeInspectorStrategy]. +pub trait CheatcodeInspectorStrategyContext: Debug + Send + Sync + Any { + /// Clone the strategy context. + fn new_cloned(&self) -> Box; + /// Alias as immutable reference of [Any]. + fn as_any_ref(&self) -> &dyn Any; + /// Alias as mutable reference of [Any]. + fn as_any_mut(&mut self) -> &mut dyn Any; +} + +impl CheatcodeInspectorStrategyContext for () { + fn new_cloned(&self) -> Box { + Box::new(()) + } + + fn as_any_mut(&mut self) -> &mut dyn Any { + self + } + + fn as_any_ref(&self) -> &dyn Any { + self + } +} + +/// Represents the strategy. +#[derive(Debug)] +pub struct CheatcodeInspectorStrategy { + /// Strategy runner. + pub runner: Box, + /// Strategy context. + pub context: Box, +} + +impl CheatcodeInspectorStrategy { + pub fn new_evm() -> Self { + Self { + runner: Box::new(EvmCheatcodeInspectorStrategyRunner::default()), + context: Box::new(()), + } + } +} + +impl Clone for CheatcodeInspectorStrategy { + fn clone(&self) -> Self { + Self { runner: self.runner.new_cloned(), context: self.context.new_cloned() } + } +} + +pub trait CheatcodeInspectorStrategyRunner: + Debug + Send + Sync + CheatcodeInspectorStrategyExt +{ fn name(&self) -> &'static str; - fn new_cloned(&self) -> Box; + fn new_cloned(&self) -> Box; /// Get nonce. - fn get_nonce(&mut self, ccx: &mut CheatsCtxt, address: Address) -> Result { + fn get_nonce(&self, ccx: &mut CheatsCtxt, address: Address) -> Result { let account = ccx.ecx.journaled_state.load_account(address, &mut ccx.ecx.db)?; Ok(account.info.nonce) } /// Called when the main test or script contract is deployed. - fn base_contract_deployed(&mut self) {} + fn base_contract_deployed(&self, _ctx: &mut dyn CheatcodeInspectorStrategyContext) {} /// Cheatcode: roll. - fn cheatcode_roll(&mut self, ccx: &mut CheatsCtxt, new_height: U256) -> Result { + fn cheatcode_roll(&self, ccx: &mut CheatsCtxt, new_height: U256) -> Result { ccx.ecx.env.block.number = new_height; Ok(Default::default()) } /// Cheatcode: warp. - fn cheatcode_warp(&mut self, ccx: &mut CheatsCtxt, new_timestamp: U256) -> Result { + fn cheatcode_warp(&self, ccx: &mut CheatsCtxt, new_timestamp: U256) -> Result { ccx.ecx.env.block.timestamp = new_timestamp; Ok(Default::default()) } /// Cheatcode: deal. - fn cheatcode_deal( - &mut self, - ccx: &mut CheatsCtxt, - address: Address, - new_balance: U256, - ) -> Result { + fn cheatcode_deal(&self, ccx: &mut CheatsCtxt, address: Address, new_balance: U256) -> Result { let account = journaled_account(ccx.ecx, address)?; let old_balance = std::mem::replace(&mut account.info.balance, new_balance); let record = DealRecord { address, old_balance, new_balance }; @@ -60,7 +105,7 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt /// Cheatcode: etch. fn cheatcode_etch( - &mut self, + &self, ccx: &mut CheatsCtxt, target: Address, new_runtime_bytecode: &Bytes, @@ -73,12 +118,12 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt } /// Cheatcode: getNonce. - fn cheatcode_get_nonce(&mut self, ccx: &mut CheatsCtxt, address: Address) -> Result { + fn cheatcode_get_nonce(&self, ccx: &mut CheatsCtxt, address: Address) -> Result { evm::get_nonce(ccx, &address) } /// Cheatcode: resetNonce. - fn cheatcode_reset_nonce(&mut self, ccx: &mut CheatsCtxt, account: Address) -> Result { + fn cheatcode_reset_nonce(&self, ccx: &mut CheatsCtxt, account: Address) -> Result { let account = journaled_account(ccx.ecx, account)?; // Per EIP-161, EOA nonces start at 0, but contract nonces // start at 1. Comparing by code_hash instead of code @@ -92,7 +137,7 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt /// Cheatcode: setNonce. fn cheatcode_set_nonce( - &mut self, + &self, ccx: &mut CheatsCtxt, account: Address, new_nonce: u64, @@ -111,7 +156,7 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt /// Cheatcode: setNonceUnsafe. fn cheatcode_set_nonce_unsafe( - &mut self, + &self, ccx: &mut CheatsCtxt, account: Address, new_nonce: u64, @@ -123,7 +168,7 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt /// Mocks a call to return with a value. fn cheatcode_mock_call( - &mut self, + &self, ccx: &mut CheatsCtxt, callee: Address, data: &Bytes, @@ -136,7 +181,7 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt /// Mocks a call to revert with a value. fn cheatcode_mock_call_revert( - &mut self, + &self, ccx: &mut CheatsCtxt, callee: Address, data: &Bytes, @@ -154,7 +199,8 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt /// Record broadcastable transaction during CREATE. fn record_broadcastable_create_transactions( - &mut self, + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, config: Arc, input: &dyn CommonCreateInput, ecx_inner: InnerEcx, @@ -163,8 +209,10 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt ); /// Record broadcastable transaction during CALL. + #[allow(clippy::too_many_arguments)] fn record_broadcastable_call_transactions( - &mut self, + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, config: Arc, input: &CallInputs, ecx_inner: InnerEcx, @@ -173,35 +221,55 @@ pub trait CheatcodeInspectorStrategy: Debug + Send + Sync + CheatcodeInspectorSt active_delegation: &mut Option, ); - fn post_initialize_interp(&mut self, _interpreter: &mut Interpreter, _ecx: Ecx) {} + fn post_initialize_interp( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _interpreter: &mut Interpreter, + _ecx: Ecx, + ) { + } /// Used to override opcode behaviors. Returns true if handled. - fn pre_step_end(&mut self, _interpreter: &mut Interpreter, _ecx: Ecx) -> bool { + fn pre_step_end( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _interpreter: &mut Interpreter, + _ecx: Ecx, + ) -> bool { false } } /// We define this in our fork pub trait CheatcodeInspectorStrategyExt { - fn zksync_cheatcode_skip_zkvm(&mut self) -> Result { + fn zksync_cheatcode_skip_zkvm( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + ) -> Result { Ok(Default::default()) } fn zksync_cheatcode_set_paymaster( - &mut self, + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, _paymaster_address: Address, _paymaster_input: &Bytes, ) -> Result { Ok(Default::default()) } - fn zksync_cheatcode_use_factory_deps(&mut self, _name: String) -> Result { + fn zksync_cheatcode_use_factory_deps( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _name: String, + ) -> Result { Ok(Default::default()) } #[allow(clippy::too_many_arguments)] fn zksync_cheatcode_register_contract( - &mut self, + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, _name: String, _zk_bytecode_hash: FixedBytes<32>, _zk_deployed_bytecode: Vec, @@ -213,16 +281,39 @@ pub trait CheatcodeInspectorStrategyExt { Ok(Default::default()) } - fn zksync_cheatcode_select_zk_vm(&mut self, _data: InnerEcx, _enable: bool) {} + fn zksync_cheatcode_select_zk_vm( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _data: InnerEcx, + _enable: bool, + ) { + } - fn zksync_record_create_address(&mut self, _outcome: &CreateOutcome) {} + fn zksync_record_create_address( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _outcome: &CreateOutcome, + ) { + } - fn zksync_sync_nonce(&mut self, _sender: Address, _nonce: u64, _ecx: Ecx) {} + fn zksync_sync_nonce( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _sender: Address, + _nonce: u64, + _ecx: Ecx, + ) { + } - fn zksync_set_deployer_call_input(&mut self, _call: &mut CallInputs) {} + fn zksync_set_deployer_call_input( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _call: &mut CallInputs, + ) { + } fn zksync_try_create( - &mut self, + &self, _state: &mut Cheatcodes, _ecx: Ecx<'_, '_, '_>, _input: &dyn CommonCreateInput, @@ -232,7 +323,7 @@ pub trait CheatcodeInspectorStrategyExt { } fn zksync_try_call( - &mut self, + &self, _state: &mut Cheatcodes, _ecx: Ecx, _input: &CallInputs, @@ -241,23 +332,30 @@ pub trait CheatcodeInspectorStrategyExt { None } - fn zksync_select_fork_vm(&mut self, _data: InnerEcx, _fork_id: LocalForkId) {} + fn zksync_select_fork_vm( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + _data: InnerEcx, + _fork_id: LocalForkId, + ) { + } } #[derive(Debug, Default, Clone)] -pub struct EvmCheatcodeInspectorStrategy {} +pub struct EvmCheatcodeInspectorStrategyRunner {} -impl CheatcodeInspectorStrategy for EvmCheatcodeInspectorStrategy { +impl CheatcodeInspectorStrategyRunner for EvmCheatcodeInspectorStrategyRunner { fn name(&self) -> &'static str { "evm" } - fn new_cloned(&self) -> Box { + fn new_cloned(&self) -> Box { Box::new(self.clone()) } fn record_broadcastable_create_transactions( - &mut self, + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, _config: Arc, input: &dyn CommonCreateInput, ecx_inner: InnerEcx, @@ -288,7 +386,8 @@ impl CheatcodeInspectorStrategy for EvmCheatcodeInspectorStrategy { } fn record_broadcastable_call_transactions( - &mut self, + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, _config: Arc, call: &CallInputs, ecx_inner: InnerEcx, @@ -326,13 +425,13 @@ impl CheatcodeInspectorStrategy for EvmCheatcodeInspectorStrategy { } } -impl CheatcodeInspectorStrategyExt for EvmCheatcodeInspectorStrategy {} +impl CheatcodeInspectorStrategyExt for EvmCheatcodeInspectorStrategyRunner {} -impl Clone for Box { +impl Clone for Box { fn clone(&self) -> Self { self.new_cloned() } } -struct _ObjectSafe0(dyn CheatcodeInspectorStrategy); +struct _ObjectSafe0(dyn CheatcodeInspectorStrategyRunner); struct _ObjectSafe1(dyn CheatcodeInspectorStrategyExt); diff --git a/crates/cheatcodes/src/test.rs b/crates/cheatcodes/src/test.rs index de3f1eb72..ae59439fc 100644 --- a/crates/cheatcodes/src/test.rs +++ b/crates/cheatcodes/src/test.rs @@ -15,7 +15,11 @@ impl Cheatcode for zkVmCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { enable } = *self; - ccx.with_strategy(|strategy, ccx| strategy.zksync_cheatcode_select_zk_vm(ccx.ecx, enable)); + ccx.state.strategy.runner.zksync_cheatcode_select_zk_vm( + ccx.state.strategy.context.as_mut(), + ccx.ecx, + enable, + ); Ok(Default::default()) } @@ -23,23 +27,28 @@ impl Cheatcode for zkVmCall { impl Cheatcode for zkVmSkipCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { - ccx.with_strategy(|strategy, _ccx| strategy.zksync_cheatcode_skip_zkvm()) + ccx.state.strategy.runner.zksync_cheatcode_skip_zkvm(ccx.state.strategy.context.as_mut()) } } impl Cheatcode for zkUsePaymasterCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { paymaster_address, paymaster_input } = self; - ccx.with_strategy(|strategy, _ccx| { - strategy.zksync_cheatcode_set_paymaster(*paymaster_address, paymaster_input) - }) + ccx.state.strategy.runner.zksync_cheatcode_set_paymaster( + ccx.state.strategy.context.as_mut(), + *paymaster_address, + paymaster_input, + ) } } impl Cheatcode for zkUseFactoryDepCall { fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result { let Self { name } = self; - ccx.with_strategy(|strategy, _ccx| strategy.zksync_cheatcode_use_factory_deps(name.clone())) + ccx.state + .strategy + .runner + .zksync_cheatcode_use_factory_deps(ccx.state.strategy.context.as_mut(), name.clone()) } } @@ -54,17 +63,16 @@ impl Cheatcode for zkRegisterContractCall { zkDeployedBytecode, } = self; - ccx.with_strategy(|strategy, _ccx| { - strategy.zksync_cheatcode_register_contract( - name.clone(), - zkBytecodeHash.0.into(), - zkDeployedBytecode.to_vec(), - vec![], //TODO: add argument to cheatcode - *evmBytecodeHash, - evmDeployedBytecode.to_vec(), - evmBytecode.to_vec(), - ) - }) + ccx.state.strategy.runner.zksync_cheatcode_register_contract( + ccx.state.strategy.context.as_mut(), + name.clone(), + zkBytecodeHash.0.into(), + zkDeployedBytecode.to_vec(), + vec![], //TODO: add argument to cheatcode + *evmBytecodeHash, + evmDeployedBytecode.to_vec(), + evmBytecode.to_vec(), + ) } } diff --git a/crates/chisel/src/executor.rs b/crates/chisel/src/executor.rs index 91ab88ddb..b66c61bed 100644 --- a/crates/chisel/src/executor.rs +++ b/crates/chisel/src/executor.rs @@ -316,14 +316,14 @@ impl SessionSource { let env = self.config.evm_opts.evm_env().await.expect("Could not instantiate fork environment"); - let strategy = utils::get_executor_strategy(&self.config.foundry_config); + let mut strategy = utils::get_executor_strategy(&self.config.foundry_config); // Create an in-memory backend let backend = match self.config.backend.take() { Some(backend) => backend, None => { let fork = self.config.evm_opts.get_fork(&self.config.foundry_config, env.clone()); - let backend = Backend::spawn(fork, strategy.new_backend_strategy()); + let backend = Backend::spawn(fork, strategy.runner.new_backend_strategy()); self.config.backend = Some(backend.clone()); backend } @@ -339,7 +339,7 @@ impl SessionSource { None, None, Some(self.solc.version.clone()), - strategy.new_cheatcode_inspector_strategy(), + strategy.runner.new_cheatcode_inspector_strategy(strategy.context.as_mut()), ) .into(), ) diff --git a/crates/cli/src/utils/mod.rs b/crates/cli/src/utils/mod.rs index e1a5eeb34..1adbec435 100644 --- a/crates/cli/src/utils/mod.rs +++ b/crates/cli/src/utils/mod.rs @@ -9,8 +9,8 @@ use foundry_common::{ shell, }; use foundry_config::{Chain, Config}; -use foundry_evm::executors::strategy::{EvmExecutorStrategy, ExecutorStrategy}; -use foundry_strategy_zksync::ZksyncExecutorStrategy; +use foundry_evm::executors::strategy::ExecutorStrategy; +use foundry_strategy_zksync::ZksyncExecutorStrategyBuilder; use serde::de::DeserializeOwned; use std::{ ffi::OsStr, @@ -93,13 +93,13 @@ pub fn get_provider(config: &Config) -> Result { get_provider_builder(config)?.build() } -pub fn get_executor_strategy(config: &Config) -> Box { +pub fn get_executor_strategy(config: &Config) -> ExecutorStrategy { if config.zksync.should_compile() { info!("using zksync strategy"); - Box::new(ZksyncExecutorStrategy::default()) + ExecutorStrategy::new_zksync() } else { info!("using evm strategy"); - Box::new(EvmExecutorStrategy::default()) + ExecutorStrategy::new_evm() } } diff --git a/crates/evm/core/src/backend/cow.rs b/crates/evm/core/src/backend/cow.rs index eaa4bd59a..44e91a2a8 100644 --- a/crates/evm/core/src/backend/cow.rs +++ b/crates/evm/core/src/backend/cow.rs @@ -14,10 +14,13 @@ use alloy_rpc_types::TransactionRequest; use foundry_fork_db::DatabaseError; use revm::{ db::DatabaseRef, - primitives::{Account, AccountInfo, Bytecode, Env, EnvWithHandlerCfg, HashMap as Map, SpecId}, + primitives::{ + Account, AccountInfo, Bytecode, Env, EnvWithHandlerCfg, HashMap as Map, ResultAndState, + SpecId, + }, Database, DatabaseCommit, JournaledState, }; -use std::{borrow::Cow, collections::BTreeMap}; +use std::{any::Any, borrow::Cow, collections::BTreeMap}; /// A wrapper around `Backend` that ensures only `revm::DatabaseRef` functions are called. /// @@ -53,6 +56,30 @@ impl<'a> CowBackend<'a> { Self { backend: Cow::Borrowed(backend), is_initialized: false, spec_id: SpecId::LATEST } } + /// Executes the configured transaction of the `env` without committing state changes + /// + /// Note: in case there are any cheatcodes executed that modify the environment, this will + /// update the given `env` with the new values. + #[instrument(name = "inspect", level = "debug", skip_all)] + pub fn inspect( + &mut self, + env: &mut EnvWithHandlerCfg, + inspector: &mut I, + inspect_ctx: Box, + ) -> eyre::Result { + // this is a new call to inspect with a new env, so even if we've cloned the backend + // already, we reset the initialized state + self.is_initialized = false; + self.spec_id = env.handler_cfg.spec_id; + + self.backend.strategy.runner.clone().inspect( + self.backend.to_mut(), + env, + inspector, + inspect_ctx, + ) + } + /// Returns whether there was a state snapshot failure in the backend. /// /// This is bubbled up from the underlying Copy-On-Write backend when a revert occurs. @@ -88,8 +115,8 @@ impl DatabaseExt for CowBackend<'_> { self.backend.to_mut().get_fork_info(id) } - fn get_strategy(&mut self) -> &mut dyn BackendStrategy { - self.backend.to_mut().strategy.as_mut() + fn get_strategy(&mut self) -> &mut BackendStrategy { + &mut self.backend.to_mut().strategy } fn snapshot_state(&mut self, journaled_state: &JournaledState, env: &Env) -> U256 { diff --git a/crates/evm/core/src/backend/mod.rs b/crates/evm/core/src/backend/mod.rs index 2bfff2984..621f5191d 100644 --- a/crates/evm/core/src/backend/mod.rs +++ b/crates/evm/core/src/backend/mod.rs @@ -21,11 +21,12 @@ use revm::{ precompile::{PrecompileSpecId, Precompiles}, primitives::{ Account, AccountInfo, BlobExcessGasAndPrice, Bytecode, Env, EnvWithHandlerCfg, EvmState, - EvmStorageSlot, HashMap as Map, Log, SpecId, KECCAK_EMPTY, + EvmStorageSlot, HashMap as Map, Log, ResultAndState, SpecId, KECCAK_EMPTY, }, Database, DatabaseCommit, JournaledState, }; use std::{ + any::Any, collections::{BTreeMap, HashSet}, time::Instant, }; @@ -100,8 +101,7 @@ pub trait DatabaseExt: Database + DatabaseCommit { /// and the the fork environment. fn get_fork_info(&mut self, id: LocalForkId) -> eyre::Result; - /// Retrieve the strategy. - fn get_strategy(&mut self) -> &mut dyn BackendStrategy; + fn get_strategy(&mut self) -> &mut BackendStrategy; /// Reverts the snapshot if it exists /// @@ -459,7 +459,7 @@ struct _ObjectSafe(dyn DatabaseExt); #[must_use] pub struct Backend { /// The behavior strategy. - pub strategy: Box, + pub strategy: BackendStrategy, /// The access point for managing forks forks: MultiFork, @@ -495,13 +495,13 @@ pub struct Backend { impl Clone for Backend { fn clone(&self) -> Self { Self { - strategy: self.strategy.new_cloned(), forks: self.forks.clone(), mem_db: self.mem_db.clone(), fork_init_journaled_state: self.fork_init_journaled_state.clone(), active_fork_ids: self.active_fork_ids, inner: self.inner.clone(), fork_url_type: self.fork_url_type.clone(), + strategy: self.strategy.clone(), } } } @@ -511,7 +511,7 @@ impl Backend { /// /// If `fork` is `Some` this will use a `fork` database, otherwise with an in-memory /// database. - pub fn spawn(fork: Option, strategy: Box) -> Self { + pub fn spawn(fork: Option, strategy: BackendStrategy) -> Self { Self::new(MultiFork::spawn(), fork, strategy) } @@ -521,11 +521,7 @@ impl Backend { /// database. /// /// Prefer using [`spawn`](Self::spawn) instead. - pub fn new( - forks: MultiFork, - fork: Option, - strategy: Box, - ) -> Self { + pub fn new(forks: MultiFork, fork: Option, strategy: BackendStrategy) -> Self { trace!(target: "backend", forking_mode=?fork.is_some(), "creating executor backend"); // Note: this will take of registering the `fork` let inner = BackendInner { @@ -564,10 +560,10 @@ impl Backend { /// Creates a new instance of `Backend` with fork added to the fork database and sets the fork /// as active pub(crate) fn new_with_fork( + strategy: BackendStrategy, id: &ForkId, fork: Fork, journaled_state: JournaledState, - strategy: Box, ) -> Self { let mut backend = Self::spawn(None, strategy); let fork_ids = backend.inner.insert_new_fork(id.clone(), fork.db, journaled_state); @@ -585,7 +581,7 @@ impl Backend { active_fork_ids: None, inner: Default::default(), fork_url_type: Default::default(), - strategy: self.strategy.new_cloned(), + strategy: self.strategy.clone(), } } @@ -777,11 +773,26 @@ impl Backend { /// Initializes settings we need to keep track of. /// /// We need to track these mainly to prevent issues when switching between different evms - pub fn initialize(&mut self, env: &EnvWithHandlerCfg) { + pub(crate) fn initialize(&mut self, env: &EnvWithHandlerCfg) { self.set_caller(env.tx.caller); self.set_spec_id(env.handler_cfg.spec_id); } + /// Executes the configured test call of the `env` without committing state changes. + /// + /// Note: in case there are any cheatcodes executed that modify the environment, this will + /// update the given `env` with the new values. + #[instrument(name = "inspect", level = "debug", skip_all)] + pub fn inspect( + &mut self, + env: &mut EnvWithHandlerCfg, + inspector: &mut I, + inspect_ctx: Box, + ) -> eyre::Result { + self.initialize(env); + self.strategy.runner.clone().inspect(self, env, inspector, inspect_ctx) + } + /// Returns the `EnvWithHandlerCfg` with the current `spec_id` set. fn env_with_handler_cfg(&self, env: Env) -> EnvWithHandlerCfg { EnvWithHandlerCfg::new_with_spec_id(Box::new(env), self.inner.spec_id) @@ -906,6 +917,7 @@ impl Backend { trace!(tx=?tx.tx_hash(), "committing transaction"); commit_transaction( + &mut self.strategy, &tx.inner, env.clone(), journaled_state, @@ -913,7 +925,6 @@ impl Backend { &fork_id, &persistent_accounts, &mut NoOpInspector, - self.strategy.as_mut(), )?; } @@ -937,8 +948,8 @@ impl DatabaseExt for Backend { Ok(ForkInfo { fork_type, fork_env }) } - fn get_strategy(&mut self) -> &mut dyn BackendStrategy { - self.strategy.as_mut() + fn get_strategy(&mut self) -> &mut BackendStrategy { + &mut self.strategy } fn snapshot_state(&mut self, journaled_state: &JournaledState, env: &Env) -> U256 { @@ -1168,9 +1179,12 @@ impl DatabaseExt for Backend { caller_account.into() }); - self.strategy.update_fork_db( + let active_fork = self.active_fork_ids.map(|(_, idx)| self.inner.get_fork(idx)); + // let active_fork = self.active_fork().cloned(); + self.strategy.runner.update_fork_db( + self.strategy.context.as_mut(), BackendStrategyForkInfo { - active_fork: self.active_fork(), + active_fork, active_type: current_fork_type, target_type: target_fork_type, }, @@ -1205,7 +1219,7 @@ impl DatabaseExt for Backend { let (fork_id, backend, fork_env) = self.forks.roll_fork(self.inner.ensure_fork_id(id).cloned()?, block_number)?; // this will update the local mapping - self.inner.roll_fork(id, fork_id, backend, self.strategy.as_mut())?; + self.inner.roll_fork(&mut self.strategy, id, fork_id, backend)?; if let Some((active_id, active_idx)) = self.active_fork_ids { // the currently active fork is the targeted fork of this call @@ -1227,7 +1241,8 @@ impl DatabaseExt for Backend { active.journaled_state.depth = journaled_state.depth; for addr in persistent_addrs { - self.strategy.merge_journaled_state_data( + self.strategy.runner.merge_journaled_state_data( + self.strategy.context.as_mut(), addr, journaled_state, &mut active.journaled_state, @@ -1244,7 +1259,8 @@ impl DatabaseExt for Backend { for (addr, acc) in journaled_state.state.iter() { if acc.is_created() { if acc.is_touched() { - self.strategy.merge_journaled_state_data( + self.strategy.runner.merge_journaled_state_data( + self.strategy.context.as_mut(), *addr, journaled_state, &mut active.journaled_state, @@ -1263,6 +1279,7 @@ impl DatabaseExt for Backend { fn roll_fork_to_transaction( &mut self, + id: Option, transaction: B256, env: &mut Env, @@ -1318,6 +1335,7 @@ impl DatabaseExt for Backend { let env = self.env_with_handler_cfg(env); let fork = self.inner.get_fork_by_id_mut(id)?; commit_transaction( + &mut self.strategy, &tx, env, journaled_state, @@ -1325,7 +1343,6 @@ impl DatabaseExt for Backend { &fork_id, &persistent_accounts, inspector, - self.strategy.as_mut(), ) } @@ -1804,10 +1821,10 @@ impl BackendInner { pub fn roll_fork( &mut self, + strategy: &mut BackendStrategy, id: LocalForkId, new_fork_id: ForkId, backend: SharedBackend, - strategy: &mut dyn BackendStrategy, ) -> eyre::Result { let fork_id = self.ensure_fork_id(id)?; let idx = self.ensure_fork_index(fork_id)?; @@ -1816,7 +1833,12 @@ impl BackendInner { // we initialize a _new_ `ForkDB` but keep the state of persistent accounts let mut new_db = ForkDB::new(backend); for addr in self.persistent_accounts.iter().copied() { - strategy.merge_db_account_data(addr, &active.db, &mut new_db); + strategy.runner.merge_db_account_data( + strategy.context.as_mut(), + addr, + &active.db, + &mut new_db, + ); } active.db = new_db; } @@ -1929,6 +1951,7 @@ fn update_env_block(env: &mut Env, block: &AnyRpcBlock) { /// state, with an inspector. #[allow(clippy::too_many_arguments)] fn commit_transaction( + strategy: &mut BackendStrategy, tx: &Transaction, mut env: EnvWithHandlerCfg, journaled_state: &mut JournaledState, @@ -1936,7 +1959,6 @@ fn commit_transaction( fork_id: &ForkId, persistent_accounts: &HashSet
, inspector: &mut dyn InspectorExt, - strategy: &mut dyn BackendStrategy, ) -> eyre::Result<()> { configure_tx_env(&mut env.env, tx); @@ -1945,7 +1967,7 @@ fn commit_transaction( let fork = fork.clone(); let journaled_state = journaled_state.clone(); let depth = journaled_state.depth; - let mut db = Backend::new_with_fork(fork_id, fork, journaled_state, strategy.new_cloned()); + let mut db = Backend::new_with_fork(strategy.clone(), fork_id, fork, journaled_state); let mut evm = crate::utils::new_evm_with_inspector(&mut db as _, env, inspector); // Adjust inner EVM depth to ensure that inspectors receive accurate data. @@ -1998,7 +2020,7 @@ fn apply_state_changeset( #[allow(clippy::needless_return)] mod tests { use crate::{ - backend::{strategy::EvmBackendStrategy, Backend}, + backend::{strategy::BackendStrategy, Backend}, fork::CreateFork, opts::EvmOpts, }; @@ -2032,7 +2054,7 @@ mod tests { evm_opts, }; - let backend = Backend::spawn(Some(fork), Box::new(EvmBackendStrategy)); + let backend = Backend::spawn(Some(fork), BackendStrategy::new_evm()); // some rng contract from etherscan let address: Address = "63091244180ae240c87d1f528f5f269134cb07b3".parse().unwrap(); diff --git a/crates/evm/core/src/backend/strategy.rs b/crates/evm/core/src/backend/strategy.rs index 18ebe9e6b..2160da098 100644 --- a/crates/evm/core/src/backend/strategy.rs +++ b/crates/evm/core/src/backend/strategy.rs @@ -1,8 +1,15 @@ -use std::fmt::Debug; +use std::{any::Any, fmt::Debug}; -use super::{BackendInner, Fork, ForkDB, ForkType, FoundryEvmInMemoryDB}; +use crate::InspectorExt; + +use super::{Backend, BackendInner, Fork, ForkDB, ForkType, FoundryEvmInMemoryDB}; use alloy_primitives::{Address, U256}; -use revm::{db::CacheDB, primitives::HashSet, DatabaseRef, JournaledState}; +use eyre::{Context, Result}; +use revm::{ + db::CacheDB, + primitives::{EnvWithHandlerCfg, HashSet, ResultAndState}, + DatabaseRef, JournaledState, +}; use serde::{Deserialize, Serialize}; pub struct BackendStrategyForkInfo<'a> { @@ -11,14 +18,69 @@ pub struct BackendStrategyForkInfo<'a> { pub target_type: ForkType, } -pub trait BackendStrategy: Debug + Send + Sync + BackendStrategyExt { +/// Context for [BackendStrategyRunner]. +pub trait BackendStrategyContext: Debug + Send + Sync + Any { + /// Clone the strategy context. + fn new_cloned(&self) -> Box; + /// Alias as immutable reference of [Any]. + fn as_any_ref(&self) -> &dyn Any; + /// Alias as mutable reference of [Any]. + fn as_any_mut(&mut self) -> &mut dyn Any; +} + +impl BackendStrategyContext for () { + fn new_cloned(&self) -> Box { + Box::new(()) + } + + fn as_any_ref(&self) -> &dyn Any { + self + } + + fn as_any_mut(&mut self) -> &mut dyn Any { + self + } +} + +/// Strategy for [super::Backend]. +#[derive(Debug)] +pub struct BackendStrategy { + /// Strategy runner. + pub runner: Box, + /// Strategy context. + pub context: Box, +} + +impl BackendStrategy { + /// Create a new instance of [BackendStrategy] + pub fn new_evm() -> Self { + Self { runner: Box::new(EvmBackendStrategyRunner), context: Box::new(()) } + } +} + +impl Clone for BackendStrategy { + fn clone(&self) -> Self { + Self { runner: self.runner.new_cloned(), context: self.context.new_cloned() } + } +} + +pub trait BackendStrategyRunner: Debug + Send + Sync + BackendStrategyRunnerExt { fn name(&self) -> &'static str; - fn new_cloned(&self) -> Box; + fn new_cloned(&self) -> Box; + + fn inspect( + &self, + backend: &mut Backend, + env: &mut EnvWithHandlerCfg, + inspector: &mut dyn InspectorExt, + inspect_ctx: Box, + ) -> Result; /// When creating or switching forks, we update the AccountInfo of the contract fn update_fork_db( &self, + ctx: &mut dyn BackendStrategyContext, fork_info: BackendStrategyForkInfo<'_>, mem_db: &FoundryEvmInMemoryDB, backend_inner: &BackendInner, @@ -29,40 +91,70 @@ pub trait BackendStrategy: Debug + Send + Sync + BackendStrategyExt { /// Clones the account data from the `active_journaled_state` into the `fork_journaled_state` fn merge_journaled_state_data( &self, + ctx: &mut dyn BackendStrategyContext, addr: Address, active_journaled_state: &JournaledState, fork_journaled_state: &mut JournaledState, ); - fn merge_db_account_data(&self, addr: Address, active: &ForkDB, fork_db: &mut ForkDB); + fn merge_db_account_data( + &self, + ctx: &mut dyn BackendStrategyContext, + addr: Address, + active: &ForkDB, + fork_db: &mut ForkDB, + ); } -pub trait BackendStrategyExt { +pub trait BackendStrategyRunnerExt { /// Saves the storage keys for immutable variables per address. /// /// These are required during fork to help merge the persisted addresses, as they are stored /// hashed so there is currently no way to retrieve all the address associated storage keys. /// We store all the storage keys here, even if the addresses are not marked persistent as /// they can be marked at a later stage as well. - fn zksync_save_immutable_storage(&mut self, _addr: Address, _keys: HashSet) {} + fn zksync_save_immutable_storage( + &self, + _ctx: &mut dyn BackendStrategyContext, + _addr: Address, + _keys: HashSet, + ) { + } } -struct _ObjectSafe(dyn BackendStrategy); +struct _ObjectSafe(dyn BackendStrategyRunner); #[derive(Debug, Default, Clone, Serialize, Deserialize)] -pub struct EvmBackendStrategy; +pub struct EvmBackendStrategyRunner; -impl BackendStrategy for EvmBackendStrategy { +impl BackendStrategyRunner for EvmBackendStrategyRunner { fn name(&self) -> &'static str { "evm" } - fn new_cloned(&self) -> Box { + fn new_cloned(&self) -> Box { Box::new(self.clone()) } + fn inspect( + &self, + backend: &mut Backend, + env: &mut EnvWithHandlerCfg, + inspector: &mut dyn InspectorExt, + _inspect_ctx: Box, + ) -> Result { + let mut evm = crate::utils::new_evm_with_inspector(backend, env.clone(), inspector); + + let res = evm.transact().wrap_err("backend: failed while inspecting")?; + + env.env = evm.context.evm.inner.env; + + Ok(res) + } + fn update_fork_db( &self, + _ctx: &mut dyn BackendStrategyContext, fork_info: BackendStrategyForkInfo<'_>, mem_db: &FoundryEvmInMemoryDB, backend_inner: &BackendInner, @@ -80,6 +172,7 @@ impl BackendStrategy for EvmBackendStrategy { fn merge_journaled_state_data( &self, + _ctx: &mut dyn BackendStrategyContext, addr: Address, active_journaled_state: &JournaledState, fork_journaled_state: &mut JournaledState, @@ -91,14 +184,20 @@ impl BackendStrategy for EvmBackendStrategy { ); } - fn merge_db_account_data(&self, addr: Address, active: &ForkDB, fork_db: &mut ForkDB) { + fn merge_db_account_data( + &self, + _ctx: &mut dyn BackendStrategyContext, + addr: Address, + active: &ForkDB, + fork_db: &mut ForkDB, + ) { EvmBackendMergeStrategy::merge_db_account_data(addr, active, fork_db); } } -impl BackendStrategyExt for EvmBackendStrategy {} +impl BackendStrategyRunnerExt for EvmBackendStrategyRunner {} -impl EvmBackendStrategy { +impl EvmBackendStrategyRunner { /// Merges the state of all `accounts` from the currently active db into the given `fork` pub(crate) fn update_fork_db_contracts( &self, @@ -199,3 +298,9 @@ impl EvmBackendMergeStrategy { fork_db.accounts.insert(addr, acc); } } + +impl Clone for Box { + fn clone(&self) -> Self { + self.new_cloned() + } +} diff --git a/crates/evm/evm/src/executors/builder.rs b/crates/evm/evm/src/executors/builder.rs index d44142873..9c40a47f2 100644 --- a/crates/evm/evm/src/executors/builder.rs +++ b/crates/evm/evm/src/executors/builder.rs @@ -76,7 +76,7 @@ impl ExecutorBuilder { /// Builds the executor as configured. #[inline] - pub fn build(self, env: Env, db: Backend, strategy: Box) -> Executor { + pub fn build(self, env: Env, db: Backend, strategy: ExecutorStrategy) -> Executor { let Self { mut stack, gas_limit, spec_id, legacy_assertions } = self; if stack.block.is_none() { stack.block = Some(env.block.clone()); diff --git a/crates/evm/evm/src/executors/mod.rs b/crates/evm/evm/src/executors/mod.rs index 44bb11b09..0e6c8975b 100644 --- a/crates/evm/evm/src/executors/mod.rs +++ b/crates/evm/evm/src/executors/mod.rs @@ -15,7 +15,6 @@ use alloy_primitives::{ map::{AddressHashMap, HashMap}, Address, Bytes, Log, U256, }; -use alloy_serde::OtherFields; use alloy_sol_types::{sol, SolCall}; use foundry_evm_core::{ backend::{Backend, BackendError, BackendResult, CowBackend, DatabaseExt, GLOBAL_FAIL_SLOT}, @@ -95,7 +94,7 @@ pub struct Executor { /// Whether `failed()` should be called on the test contract to determine if the test failed. legacy_assertions: bool, - strategy: Option>, + pub strategy: ExecutorStrategy, } impl Clone for Executor { @@ -106,7 +105,7 @@ impl Clone for Executor { inspector: self.inspector.clone(), gas_limit: self.gas_limit, legacy_assertions: self.legacy_assertions, - strategy: self.strategy.as_ref().map(|s| s.new_cloned()), + strategy: self.strategy.clone(), } } } @@ -126,7 +125,7 @@ impl Executor { inspector: InspectorStack, gas_limit: u64, legacy_assertions: bool, - strategy: Box, + strategy: ExecutorStrategy, ) -> Self { // Need to create a non-empty contract on the cheatcodes address so `extcodesize` checks // do not fail. @@ -141,7 +140,7 @@ impl Executor { }, ); - Self { backend, env, inspector, gas_limit, legacy_assertions, strategy: Some(strategy) } + Self { backend, env, inspector, gas_limit, legacy_assertions, strategy } } fn clone_with_backend(&self, backend: Backend) -> Self { @@ -152,7 +151,7 @@ impl Executor { self.inspector().clone(), self.gas_limit, self.legacy_assertions, - self.strategy.as_ref().map(|s| s.new_cloned()).expect("failed acquiring strategy"), + self.strategy.clone(), ) } @@ -246,21 +245,10 @@ impl Executor { Ok(()) } - pub fn with_strategy(&mut self, mut f: F) -> R - where - F: FnMut(&mut dyn ExecutorStrategy, &mut Self) -> R, - { - let mut strategy = self.strategy.take(); - let result = f(strategy.as_mut().expect("failed acquiring strategy").as_mut(), self); - self.strategy = strategy; - - result - } - /// Set the balance of an account. pub fn set_balance(&mut self, address: Address, amount: U256) -> BackendResult<()> { trace!(?address, ?amount, "setting account balance"); - self.with_strategy(|strategy, executor| strategy.set_balance(executor, address, amount)) + self.strategy.runner.clone().set_balance(self, address, amount) } /// Gets the balance of an account @@ -270,7 +258,7 @@ impl Executor { /// Set the nonce of an account. pub fn set_nonce(&mut self, address: Address, nonce: u64) -> BackendResult<()> { - self.with_strategy(|strategy, executor| strategy.set_nonce(executor, address, nonce)) + self.strategy.runner.clone().set_nonce(self, address, nonce) } /// Returns the nonce of an account. @@ -300,14 +288,6 @@ impl Executor { self.inspector().create2_deployer() } - #[inline] - pub fn set_transaction_other_fields(&mut self, other_fields: OtherFields) { - self.strategy - .as_mut() - .expect("failed acquiring strategy") - .set_inspect_context(other_fields); - } - /// Deploys a contract and commits the new state to the underlying database. /// /// Executes a CREATE transaction with the contract `code` and persistent database state @@ -483,17 +463,14 @@ impl Executor { pub fn call_with_env(&self, mut env: EnvWithHandlerCfg) -> eyre::Result { let mut inspector = self.inspector().clone(); let mut backend = CowBackend::new_borrowed(self.backend()); - // this is a new call to inspect with a new env, so even if we've cloned the backend - // already, we reset the initialized state - backend.is_initialized = false; - backend.spec_id = env.spec_id(); - - let result = self - .strategy - .as_ref() - .expect("failed acquiring strategy") - .new_cloned() - .call_inspect(&mut backend, &mut env, &mut inspector)?; + + let result = self.strategy.runner.call( + self.strategy.context.as_ref(), + &mut backend, + &mut env, + &self.env, + &mut inspector, + )?; convert_executed_result( env.clone(), @@ -506,24 +483,26 @@ impl Executor { /// Execute the transaction configured in `env.tx`. #[instrument(name = "transact", level = "debug", skip_all)] pub fn transact_with_env(&mut self, mut env: EnvWithHandlerCfg) -> eyre::Result { - self.with_strategy(|strategy, executor| { - let mut inspector = executor.inspector.clone(); - let backend = &mut executor.backend; - backend.initialize(&env); - - let result_and_state = - strategy.transact_inspect(backend, &mut env, &executor.env, &mut inspector)?; - - let mut result = convert_executed_result( - env.clone(), - inspector, - result_and_state, - backend.has_state_snapshot_failure(), - )?; - - executor.commit(&mut result); - Ok(result) - }) + let mut inspector = self.inspector.clone(); + let backend = &mut self.backend; + + let result_and_state = self.strategy.runner.transact( + self.strategy.context.as_mut(), + backend, + &mut env, + &self.env, + &mut inspector, + )?; + + let mut result = convert_executed_result( + env.clone(), + inspector, + result_and_state, + backend.has_state_snapshot_failure(), + )?; + + self.commit(&mut result); + Ok(result) } /// Commit the changeset to the database and adjust `self.inspector_config` values according to diff --git a/crates/evm/evm/src/executors/strategy.rs b/crates/evm/evm/src/executors/strategy.rs index 2d20b4159..6c6c6af03 100644 --- a/crates/evm/evm/src/executors/strategy.rs +++ b/crates/evm/evm/src/executors/strategy.rs @@ -1,134 +1,158 @@ -use std::fmt::Debug; +use std::{any::Any, fmt::Debug}; use alloy_primitives::{Address, U256}; use alloy_serde::OtherFields; -use eyre::{Context, Result}; -use foundry_cheatcodes::strategy::{CheatcodeInspectorStrategy, EvmCheatcodeInspectorStrategy}; -use foundry_evm_core::{ - backend::{ - strategy::{BackendStrategy, EvmBackendStrategy}, - BackendResult, DatabaseExt, - }, - InspectorExt, +use eyre::Result; +use foundry_cheatcodes::strategy::{ + CheatcodeInspectorStrategy, EvmCheatcodeInspectorStrategyRunner, }; +use foundry_evm_core::backend::{strategy::BackendStrategy, Backend, BackendResult, CowBackend}; use foundry_zksync_compilers::dual_compiled_contracts::DualCompiledContracts; use revm::{ primitives::{Env, EnvWithHandlerCfg, ResultAndState}, DatabaseRef, }; +use crate::inspectors::InspectorStack; + use super::Executor; -pub trait ExecutorStrategy: Debug + Send + Sync + ExecutorStrategyExt { +pub trait ExecutorStrategyContext: Debug + Send + Sync + Any { + /// Clone the strategy context. + fn new_cloned(&self) -> Box; + /// Alias as immutable reference of [Any]. + fn as_any_ref(&self) -> &dyn Any; + /// Alias as mutable reference of [Any]. + fn as_any_mut(&mut self) -> &mut dyn Any; +} + +impl ExecutorStrategyContext for () { + fn new_cloned(&self) -> Box { + Box::new(()) + } + + fn as_any_ref(&self) -> &dyn Any { + self + } + + fn as_any_mut(&mut self) -> &mut dyn Any { + self + } +} + +#[derive(Debug)] +pub struct ExecutorStrategy { + /// Strategy runner. + pub runner: Box, + /// Strategy context. + pub context: Box, +} + +impl ExecutorStrategy { + pub fn new_evm() -> Self { + Self { runner: Box::new(EvmExecutorStrategyRunner::default()), context: Box::new(()) } + } +} + +impl Clone for ExecutorStrategy { + fn clone(&self) -> Self { + Self { runner: self.runner.new_cloned(), context: self.context.new_cloned() } + } +} + +pub trait ExecutorStrategyRunner: Debug + Send + Sync + ExecutorStrategyExt { fn name(&self) -> &'static str; - fn new_cloned(&self) -> Box; + fn new_cloned(&self) -> Box; fn set_balance( - &mut self, + &self, executor: &mut Executor, address: Address, amount: U256, ) -> BackendResult<()>; - fn set_nonce( - &mut self, - executor: &mut Executor, - address: Address, - nonce: u64, - ) -> BackendResult<()>; - - fn set_inspect_context(&mut self, other_fields: OtherFields); + fn set_nonce(&self, executor: &mut Executor, address: Address, nonce: u64) + -> BackendResult<()>; - fn call_inspect( + /// Execute a transaction and *WITHOUT* applying state changes. + fn call( &self, - db: &mut dyn DatabaseExt, + ctx: &dyn ExecutorStrategyContext, + backend: &mut CowBackend<'_>, env: &mut EnvWithHandlerCfg, - inspector: &mut dyn InspectorExt, - ) -> eyre::Result; + executor_env: &EnvWithHandlerCfg, + inspector: &mut InspectorStack, + ) -> Result; - fn transact_inspect( - &mut self, - db: &mut dyn DatabaseExt, + /// Execute a transaction and apply state changes. + fn transact( + &self, + ctx: &mut dyn ExecutorStrategyContext, + backend: &mut Backend, env: &mut EnvWithHandlerCfg, - _executor_env: &EnvWithHandlerCfg, - inspector: &mut dyn InspectorExt, - ) -> eyre::Result; + executor_env: &EnvWithHandlerCfg, + inspector: &mut InspectorStack, + ) -> Result; - fn new_backend_strategy(&self) -> Box; - fn new_cheatcode_inspector_strategy(&self) -> Box; + fn new_backend_strategy(&self) -> BackendStrategy; + fn new_cheatcode_inspector_strategy( + &self, + ctx: &dyn ExecutorStrategyContext, + ) -> foundry_cheatcodes::strategy::CheatcodeInspectorStrategy; // TODO perhaps need to create fresh strategies as well } +/// Extended trait for ZKsync. pub trait ExecutorStrategyExt { + /// Set [DualCompiledContracts] on the context. fn zksync_set_dual_compiled_contracts( - &mut self, + &self, + _ctx: &mut dyn ExecutorStrategyContext, _dual_compiled_contracts: DualCompiledContracts, ) { } - fn zksync_set_fork_env(&mut self, _fork_url: &str, _env: &Env) -> Result<()> { + /// Set the fork environment on the context. + fn zksync_set_fork_env( + &self, + _ctx: &mut dyn ExecutorStrategyContext, + _fork_url: &str, + _env: &Env, + ) -> Result<()> { Ok(()) } + + /// Sets the transaction context for the next [ExecutorStrategyRunner::call] or + /// [ExecutorStrategyRunner::transact]. This selects whether to run the transaction on zkEVM + /// or the EVM. + /// This is based if the [OtherFields] contains + /// [foundry_zksync_core::ZKSYNC_TRANSACTION_OTHER_FIELDS_KEY] with + /// [foundry_zksync_core::ZkTransactionMetadata]. + fn zksync_set_transaction_context( + &self, + _ctx: &mut dyn ExecutorStrategyContext, + _other_fields: OtherFields, + ) { + } } +/// Implements [ExecutorStrategyRunner] for EVM. #[derive(Debug, Default, Clone)] -pub struct EvmExecutorStrategy {} +pub struct EvmExecutorStrategyRunner {} -impl ExecutorStrategy for EvmExecutorStrategy { +impl ExecutorStrategyRunner for EvmExecutorStrategyRunner { fn name(&self) -> &'static str { "evm" } - fn new_cloned(&self) -> Box { + fn new_cloned(&self) -> Box { Box::new(self.clone()) } - fn set_inspect_context(&mut self, _other_fields: OtherFields) {} - - /// Executes the configured test call of the `env` without committing state changes. - /// - /// Note: in case there are any cheatcodes executed that modify the environment, this will - /// update the given `env` with the new values. - fn call_inspect( - &self, - db: &mut dyn DatabaseExt, - env: &mut EnvWithHandlerCfg, - inspector: &mut dyn InspectorExt, - ) -> eyre::Result { - let mut evm = crate::utils::new_evm_with_inspector(db, env.clone(), inspector); - - let res = evm.transact().wrap_err("backend: failed while inspecting")?; - - env.env = evm.context.evm.inner.env; - - Ok(res) - } - - /// Executes the configured test call of the `env` without committing state changes. - /// Modifications to the state are however allowed. - /// - /// Note: in case there are any cheatcodes executed that modify the environment, this will - /// update the given `env` with the new values. - fn transact_inspect( - &mut self, - db: &mut dyn DatabaseExt, - env: &mut EnvWithHandlerCfg, - _executor_env: &EnvWithHandlerCfg, - inspector: &mut dyn InspectorExt, - ) -> eyre::Result { - let mut evm = crate::utils::new_evm_with_inspector(db, env.clone(), inspector); - - let res = evm.transact().wrap_err("backend: failed while inspecting")?; - - env.env = evm.context.evm.inner.env; - - Ok(res) - } - fn set_balance( - &mut self, + &self, executor: &mut Executor, address: Address, amount: U256, @@ -142,7 +166,7 @@ impl ExecutorStrategy for EvmExecutorStrategy { } fn set_nonce( - &mut self, + &self, executor: &mut Executor, address: Address, nonce: u64, @@ -154,13 +178,47 @@ impl ExecutorStrategy for EvmExecutorStrategy { Ok(()) } - fn new_backend_strategy(&self) -> Box { - Box::new(EvmBackendStrategy) + fn call( + &self, + _ctx: &dyn ExecutorStrategyContext, + backend: &mut CowBackend<'_>, + env: &mut EnvWithHandlerCfg, + _executor_env: &EnvWithHandlerCfg, + inspector: &mut InspectorStack, + ) -> Result { + backend.inspect(env, inspector, Box::new(())) + } + + fn transact( + &self, + _ctx: &mut dyn ExecutorStrategyContext, + backend: &mut Backend, + env: &mut EnvWithHandlerCfg, + _executor_env: &EnvWithHandlerCfg, + inspector: &mut InspectorStack, + ) -> Result { + backend.inspect(env, inspector, Box::new(())) + } + + fn new_backend_strategy(&self) -> BackendStrategy { + BackendStrategy::new_evm() } - fn new_cheatcode_inspector_strategy(&self) -> Box { - Box::new(EvmCheatcodeInspectorStrategy::default()) + fn new_cheatcode_inspector_strategy( + &self, + _ctx: &dyn ExecutorStrategyContext, + ) -> CheatcodeInspectorStrategy { + CheatcodeInspectorStrategy { + runner: Box::new(EvmCheatcodeInspectorStrategyRunner::default()), + context: Box::new(()), + } } } -impl ExecutorStrategyExt for EvmExecutorStrategy {} +impl ExecutorStrategyExt for EvmExecutorStrategyRunner {} + +impl Clone for Box { + fn clone(&self) -> Self { + self.new_cloned() + } +} diff --git a/crates/evm/evm/src/executors/trace.rs b/crates/evm/evm/src/executors/trace.rs index 3bac12991..d98e194d3 100644 --- a/crates/evm/evm/src/executors/trace.rs +++ b/crates/evm/evm/src/executors/trace.rs @@ -22,9 +22,9 @@ impl TracingExecutor { trace_mode: TraceMode, odyssey: bool, create2_deployer: Address, - strategy: Box, + strategy: ExecutorStrategy, ) -> Self { - let db = Backend::spawn(fork, strategy.new_backend_strategy()); + let db = Backend::spawn(fork, strategy.runner.new_backend_strategy()); Self { // configures a bare version of the evm executor: no cheatcode inspector is enabled, // tracing will be enabled only for the targeted transaction diff --git a/crates/forge/bin/cmd/test/mod.rs b/crates/forge/bin/cmd/test/mod.rs index c8a9546ea..e7dd34cbc 100644 --- a/crates/forge/bin/cmd/test/mod.rs +++ b/crates/forge/bin/cmd/test/mod.rs @@ -363,7 +363,10 @@ impl TestArgs { // Prepare the test builder. let config = Arc::new(config); - strategy.zksync_set_dual_compiled_contracts(dual_compiled_contracts.unwrap_or_default()); + strategy.runner.zksync_set_dual_compiled_contracts( + strategy.context.as_mut(), + dual_compiled_contracts.unwrap_or_default(), + ); let runner = MultiContractRunnerBuilder::new(config.clone()) .set_debug(should_debug) diff --git a/crates/forge/src/multi_runner.rs b/crates/forge/src/multi_runner.rs index 1cc93df19..1e2ca9854 100644 --- a/crates/forge/src/multi_runner.rs +++ b/crates/forge/src/multi_runner.rs @@ -73,7 +73,7 @@ pub struct MultiContractRunner { pub tcfg: TestRunnerConfig, /// Execution strategy. - pub strategy: Box, + pub strategy: ExecutorStrategy, } impl std::ops::Deref for MultiContractRunner { @@ -180,7 +180,7 @@ impl MultiContractRunner { trace!("running all tests"); // The DB backend that serves all the data. - let db = Backend::spawn(self.fork.take(), self.strategy.new_backend_strategy()); + let db = Backend::spawn(self.fork.take(), self.strategy.runner.new_backend_strategy()); let find_timer = Instant::now(); let contracts = self.matching_contracts(filter).collect::>(); @@ -244,10 +244,10 @@ impl MultiContractRunner { ) -> SuiteResult { let identifier = artifact_id.identifier(); let mut span_name = identifier.as_str(); + if !enabled!(tracing::Level::TRACE) { span_name = get_contract_name(&identifier); } - let span = debug_span!("suite", name = %span_name); let span_local = span.clone(); let _guard = span_local.enter(); @@ -261,7 +261,7 @@ impl MultiContractRunner { self.known_contracts.clone(), artifact_id, db.clone(), - self.strategy.new_cloned(), + self.strategy.clone(), ), progress, tokio_handle, @@ -353,7 +353,7 @@ impl TestRunnerConfig { known_contracts: ContractsByArtifact, artifact_id: &ArtifactId, db: Backend, - strategy: Box, + strategy: ExecutorStrategy, ) -> Executor { let cheats_config = Arc::new(CheatsConfig::new( &self.config, @@ -361,7 +361,7 @@ impl TestRunnerConfig { Some(known_contracts), Some(artifact_id.name.clone()), Some(artifact_id.version.clone()), - strategy.new_cheatcode_inspector_strategy(), + strategy.runner.new_cheatcode_inspector_strategy(strategy.context.as_ref()), )); ExecutorBuilder::new() @@ -486,7 +486,7 @@ impl MultiContractRunnerBuilder { zk_output: Option>, env: revm::primitives::Env, evm_opts: EvmOpts, - strategy: Box, + strategy: ExecutorStrategy, ) -> Result { let contracts = output .artifact_ids() diff --git a/crates/forge/src/runner.rs b/crates/forge/src/runner.rs index 3c0120797..496cda0ca 100644 --- a/crates/forge/src/runner.rs +++ b/crates/forge/src/runner.rs @@ -185,11 +185,7 @@ impl<'a> ContractRunner<'a> { // nonce. if let Some(cheatcodes) = &mut self.executor.inspector.cheatcodes { debug!("test contract deployed"); - cheatcodes - .strategy - .as_mut() - .expect("failed acquiring strategy") - .base_contract_deployed(); + cheatcodes.strategy.runner.base_contract_deployed(cheatcodes.strategy.context.as_mut()); } // Optionally call the `setUp` function diff --git a/crates/forge/tests/it/test_helpers.rs b/crates/forge/tests/it/test_helpers.rs index 2a01c5029..d0e8398a0 100644 --- a/crates/forge/tests/it/test_helpers.rs +++ b/crates/forge/tests/it/test_helpers.rs @@ -3,8 +3,8 @@ use alloy_chains::NamedChain; use alloy_primitives::U256; use forge::{ - executors::strategy::EvmExecutorStrategy, revm::primitives::SpecId, MultiContractRunner, - MultiContractRunnerBuilder, + executors::strategy::ExecutorStrategy, revm::primitives::SpecId, MultiContractRunner, + MultiContractRunnerBuilder, TestOptions, TestOptionsBuilder, }; use foundry_cli::utils; use foundry_compilers::{ @@ -334,7 +334,9 @@ impl ForgeTestData { let sender = zk_config.sender; let mut strategy = utils::get_executor_strategy(&zk_config); - strategy.zksync_set_dual_compiled_contracts(dual_compiled_contracts); + strategy + .runner + .zksync_set_dual_compiled_contracts(strategy.context.as_mut(), dual_compiled_contracts); let mut builder = self.base_runner(); builder.config = Arc::new(zk_config); builder @@ -355,7 +357,7 @@ impl ForgeTestData { None, opts.local_evm_env(), opts, - Box::new(EvmExecutorStrategy::default()), + ExecutorStrategy::new_evm(), ) .unwrap() } @@ -378,7 +380,7 @@ impl ForgeTestData { None, env, opts, - Box::new(EvmExecutorStrategy::default()), + ExecutorStrategy::new_evm(), ) .unwrap() } diff --git a/crates/script/src/broadcast.rs b/crates/script/src/broadcast.rs index 647a49c3e..77174e365 100644 --- a/crates/script/src/broadcast.rs +++ b/crates/script/src/broadcast.rs @@ -73,7 +73,7 @@ pub async fn send_transaction( ) -> Result { let zk_tx_meta = if let SendTransactionKind::Raw(tx, _) | SendTransactionKind::Unlocked(tx) = &mut kind { - foundry_strategy_zksync::get_zksync_transaction_metadata(&tx.other) + foundry_strategy_zksync::try_get_zksync_transaction_metadata(&tx.other) } else { None }; diff --git a/crates/script/src/lib.rs b/crates/script/src/lib.rs index 8ce1b815a..16dd31bc4 100644 --- a/crates/script/src/lib.rs +++ b/crates/script/src/lib.rs @@ -607,7 +607,7 @@ impl ScriptConfig { Some(db) => db.clone(), None => { let fork = self.evm_opts.get_fork(&self.config, env.clone()); - let backend = Backend::spawn(fork, strategy.new_backend_strategy()); + let backend = Backend::spawn(fork, strategy.runner.new_backend_strategy()); self.backends.insert(fork_url.clone(), backend.clone()); backend } @@ -616,7 +616,7 @@ impl ScriptConfig { // It's only really `None`, when we don't pass any `--fork-url`. And if so, there is // no need to cache it, since there won't be any onchain simulation that we'd need // to cache the backend for. - Backend::spawn(None, strategy.new_backend_strategy()) + Backend::spawn(None, strategy.runner.new_backend_strategy()) }; // We need to enable tracing to decode contract names: local or external. @@ -634,10 +634,13 @@ impl ScriptConfig { if let Some((known_contracts, script_wallets, target, dual_compiled_contracts)) = cheats_data { - strategy.zksync_set_dual_compiled_contracts(dual_compiled_contracts); + strategy.runner.zksync_set_dual_compiled_contracts( + strategy.context.as_mut(), + dual_compiled_contracts, + ); if let Some(fork_url) = &self.evm_opts.fork_url { - strategy.zksync_set_fork_env(fork_url, &env)?; + strategy.runner.zksync_set_fork_env(strategy.context.as_mut(), fork_url, &env)?; } builder = builder.inspectors(|stack| { @@ -649,7 +652,9 @@ impl ScriptConfig { Some(known_contracts), Some(target.name), Some(target.version), - strategy.new_cheatcode_inspector_strategy(), + strategy + .runner + .new_cheatcode_inspector_strategy(strategy.context.as_ref()), ) .into(), ) diff --git a/crates/script/src/runner.rs b/crates/script/src/runner.rs index 3c8676c33..6fd579f10 100644 --- a/crates/script/src/runner.rs +++ b/crates/script/src/runner.rs @@ -174,11 +174,7 @@ impl ScriptRunner { // nonce. if let Some(cheatcodes) = &mut self.executor.inspector.cheatcodes { debug!("script deployed"); - cheatcodes - .strategy - .as_mut() - .expect("failed acquiring strategy") - .base_contract_deployed(); + cheatcodes.strategy.runner.base_contract_deployed(cheatcodes.strategy.context.as_mut()); } // Optionally call the `setUp` function @@ -261,7 +257,10 @@ impl ScriptRunner { other_fields: Option, ) -> Result { if let Some(other_fields) = other_fields { - self.executor.set_transaction_other_fields(other_fields); + self.executor.strategy.runner.zksync_set_transaction_context( + self.executor.strategy.context.as_mut(), + other_fields, + ); } if let Some(to) = to { diff --git a/crates/strategy/zksync/src/backend.rs b/crates/strategy/zksync/src/backend.rs index ee75740a7..a0ec8ce09 100644 --- a/crates/strategy/zksync/src/backend.rs +++ b/crates/strategy/zksync/src/backend.rs @@ -1,59 +1,127 @@ -use std::collections::hash_map::Entry; +use std::{any::Any, collections::hash_map::Entry}; use alloy_primitives::{map::HashMap, Address, U256}; -use foundry_evm::backend::strategy::BackendStrategyExt; +use eyre::Result; +use foundry_evm::{ + backend::{ + strategy::{BackendStrategy, BackendStrategyContext, BackendStrategyRunnerExt}, + Backend, + }, + InspectorExt, +}; use foundry_evm_core::backend::{ strategy::{ - BackendStrategy, BackendStrategyForkInfo, EvmBackendMergeStrategy, EvmBackendStrategy, + BackendStrategyForkInfo, BackendStrategyRunner, EvmBackendMergeStrategy, + EvmBackendStrategyRunner, }, BackendInner, Fork, ForkDB, FoundryEvmInMemoryDB, }; use foundry_zksync_core::{ - convert::ConvertH160, PaymasterParams, ACCOUNT_CODE_STORAGE_ADDRESS, + convert::ConvertH160, vm::ZkEnv, PaymasterParams, ACCOUNT_CODE_STORAGE_ADDRESS, IMMUTABLE_SIMULATOR_STORAGE_ADDRESS, KNOWN_CODES_STORAGE_ADDRESS, L2_BASE_TOKEN_ADDRESS, NONCE_HOLDER_ADDRESS, }; -use revm::{db::CacheDB, primitives::HashSet, DatabaseRef, JournaledState}; +use revm::{ + db::CacheDB, + primitives::{EnvWithHandlerCfg, HashSet, ResultAndState}, + DatabaseRef, JournaledState, +}; use serde::{Deserialize, Serialize}; use tracing::trace; +use zksync_types::H256; -#[derive(Debug, Default, Clone, Serialize, Deserialize)] -pub struct ZksyncBackendStrategy { - evm: EvmBackendStrategy, +/// Represents additional data for ZK transactions. +#[derive(Clone, Debug, Default)] +pub struct ZksyncInspectContext { + /// Factory Deps for ZK transactions. + pub factory_deps: Vec>, + /// Paymaster data for ZK transactions. + pub paymaster_data: Option, + /// Zksync environment. + pub zk_env: ZkEnv, +} + +/// Context for [ZksyncBackendStrategyRunner]. +#[derive(Debug, Default, Clone)] +pub struct ZksyncBackendStrategyContext { /// Store storage keys per contract address for immutable variables. persistent_immutable_keys: HashMap>, + /// Store persisted factory dependencies. + persisted_factory_deps: HashMap>, } -#[derive(Debug, Default, Clone, Serialize, Deserialize)] -pub struct ZkBackendInspectData { - #[serde(skip_serializing_if = "Option::is_none")] - pub factory_deps: Option>>, +impl BackendStrategyContext for ZksyncBackendStrategyContext { + fn new_cloned(&self) -> Box { + Box::new(self.clone()) + } - #[serde(skip_serializing_if = "Option::is_none")] - pub paymaster_data: Option, + fn as_any_ref(&self) -> &dyn Any { + self + } - pub use_evm: bool, + fn as_any_mut(&mut self) -> &mut dyn Any { + self + } +} + +/// ZKsync implementation for [BackendStrategyRunner]. +#[derive(Debug, Default, Clone, Serialize, Deserialize)] +pub struct ZksyncBackendStrategyRunner { + evm: EvmBackendStrategyRunner, } -impl BackendStrategy for ZksyncBackendStrategy { +impl BackendStrategyRunner for ZksyncBackendStrategyRunner { fn name(&self) -> &'static str { "zk" } - fn new_cloned(&self) -> Box { + fn new_cloned(&self) -> Box { Box::new(self.clone()) } + fn inspect( + &self, + backend: &mut Backend, + env: &mut EnvWithHandlerCfg, + inspector: &mut dyn InspectorExt, + inspect_ctx: Box, + ) -> Result { + if !is_zksync_inspect_context(inspect_ctx.as_ref()) { + return self.evm.inspect(backend, env, inspector, inspect_ctx); + } + + let inspect_ctx = get_inspect_context(inspect_ctx); + let mut persisted_factory_deps = + get_context(backend.strategy.context.as_mut()).persisted_factory_deps.clone(); + + let result = foundry_zksync_core::vm::transact( + Some(&mut persisted_factory_deps), + Some(inspect_ctx.factory_deps), + inspect_ctx.paymaster_data, + env, + &inspect_ctx.zk_env, + backend, + ); + + let ctx = get_context(backend.strategy.context.as_mut()); + ctx.persisted_factory_deps = persisted_factory_deps; + + result + } + /// When creating or switching forks, we update the AccountInfo of the contract. fn update_fork_db( &self, + ctx: &mut dyn BackendStrategyContext, fork_info: BackendStrategyForkInfo<'_>, mem_db: &FoundryEvmInMemoryDB, backend_inner: &BackendInner, active_journaled_state: &mut JournaledState, target_fork: &mut Fork, ) { + let ctx = get_context(ctx); self.update_fork_db_contracts( + ctx, fork_info, mem_db, backend_inner, @@ -64,13 +132,20 @@ impl BackendStrategy for ZksyncBackendStrategy { fn merge_journaled_state_data( &self, + ctx: &mut dyn BackendStrategyContext, addr: Address, active_journaled_state: &JournaledState, fork_journaled_state: &mut JournaledState, ) { - self.evm.merge_journaled_state_data(addr, active_journaled_state, fork_journaled_state); + self.evm.merge_journaled_state_data( + ctx, + addr, + active_journaled_state, + fork_journaled_state, + ); + let ctx = get_context(ctx); let zk_state = - &ZksyncMergeState { persistent_immutable_keys: &self.persistent_immutable_keys }; + &ZksyncMergeState { persistent_immutable_keys: &ctx.persistent_immutable_keys }; ZksyncBackendMerge::merge_zk_journaled_state_data( addr, active_journaled_state, @@ -79,18 +154,26 @@ impl BackendStrategy for ZksyncBackendStrategy { ); } - fn merge_db_account_data(&self, addr: Address, active: &ForkDB, fork_db: &mut ForkDB) { - self.evm.merge_db_account_data(addr, active, fork_db); + fn merge_db_account_data( + &self, + ctx: &mut dyn BackendStrategyContext, + addr: Address, + active: &ForkDB, + fork_db: &mut ForkDB, + ) { + self.evm.merge_db_account_data(ctx, addr, active, fork_db); + let ctx = get_context(ctx); let zk_state = - &ZksyncMergeState { persistent_immutable_keys: &self.persistent_immutable_keys }; + &ZksyncMergeState { persistent_immutable_keys: &ctx.persistent_immutable_keys }; ZksyncBackendMerge::merge_zk_account_data(addr, active, fork_db, zk_state); } } -impl ZksyncBackendStrategy { +impl ZksyncBackendStrategyRunner { /// Merges the state of all `accounts` from the currently active db into the given `fork` pub(crate) fn update_fork_db_contracts( &self, + ctx: &mut ZksyncBackendStrategyContext, fork_info: BackendStrategyForkInfo<'_>, mem_db: &FoundryEvmInMemoryDB, backend_inner: &BackendInner, @@ -107,7 +190,7 @@ impl ZksyncBackendStrategy { let accounts = backend_inner.persistent_accounts.iter().copied(); let zk_state = - &ZksyncMergeState { persistent_immutable_keys: &self.persistent_immutable_keys }; + &ZksyncMergeState { persistent_immutable_keys: &ctx.persistent_immutable_keys }; if let Some(db) = fork_info.active_fork.map(|f| &f.db) { ZksyncBackendMerge::merge_account_data( accounts, @@ -128,15 +211,36 @@ impl ZksyncBackendStrategy { } } -impl BackendStrategyExt for ZksyncBackendStrategy { - fn zksync_save_immutable_storage(&mut self, addr: Address, keys: HashSet) { - self.persistent_immutable_keys +impl BackendStrategyRunnerExt for ZksyncBackendStrategyRunner { + fn zksync_save_immutable_storage( + &self, + ctx: &mut dyn BackendStrategyContext, + addr: Address, + keys: HashSet, + ) { + let ctx = get_context(ctx); + ctx.persistent_immutable_keys .entry(addr) .and_modify(|entry| entry.extend(&keys)) .or_insert(keys); } } +/// Create ZKsync strategy for [BackendStrategy]. +pub trait ZksyncBackendStrategyBuilder { + /// Create new zksync strategy. + fn new_zksync() -> Self; +} + +impl ZksyncBackendStrategyBuilder for BackendStrategy { + fn new_zksync() -> Self { + Self { + runner: Box::new(ZksyncBackendStrategyRunner::default()), + context: Box::new(ZksyncBackendStrategyContext::default()), + } + } +} + pub(crate) struct ZksyncBackendMerge; /// Defines the zksync specific state to help during merge. @@ -313,3 +417,15 @@ impl ZksyncBackendMerge { } } } + +fn get_context(ctx: &mut dyn BackendStrategyContext) -> &mut ZksyncBackendStrategyContext { + ctx.as_any_mut().downcast_mut().expect("expected ZksyncBackendStrategyContext") +} + +fn get_inspect_context(ctx: Box) -> Box { + ctx.downcast().expect("expected ZksyncInspectContext") +} + +fn is_zksync_inspect_context(ctx: &dyn Any) -> bool { + ctx.downcast_ref::().is_some() +} diff --git a/crates/strategy/zksync/src/cheatcode.rs b/crates/strategy/zksync/src/cheatcode.rs index d43172b9d..d7ebdf277 100644 --- a/crates/strategy/zksync/src/cheatcode.rs +++ b/crates/strategy/zksync/src/cheatcode.rs @@ -10,7 +10,9 @@ use alloy_sol_types::SolValue; use foundry_cheatcodes::{ journaled_account, make_acc_non_empty, strategy::{ - CheatcodeInspectorStrategy, CheatcodeInspectorStrategyExt, EvmCheatcodeInspectorStrategy, + CheatcodeInspectorStrategy, CheatcodeInspectorStrategyContext, + CheatcodeInspectorStrategyExt, CheatcodeInspectorStrategyRunner, + EvmCheatcodeInspectorStrategyRunner, }, Broadcast, BroadcastableTransaction, BroadcastableTransactions, Cheatcodes, CheatcodesExecutor, CheatsConfig, CheatsCtxt, CommonCreateInput, DealRecord, Ecx, Error, InnerEcx, Result, Vm, @@ -34,7 +36,7 @@ use foundry_zksync_core::{ vm::ZkEnv, PaymasterParams, ZkPaymasterData, ZkTransactionMetadata, ACCOUNT_CODE_STORAGE_ADDRESS, CONTRACT_DEPLOYER_ADDRESS, DEFAULT_CREATE2_DEPLOYER_ZKSYNC, H256, KNOWN_CODES_STORAGE_ADDRESS, - L2_BASE_TOKEN_ADDRESS, NONCE_HOLDER_ADDRESS, + L2_BASE_TOKEN_ADDRESS, NONCE_HOLDER_ADDRESS, ZKSYNC_TRANSACTION_OTHER_FIELDS_KEY, }; use itertools::Itertools; use revm::{ @@ -55,9 +57,6 @@ use zksync_types::{ CURRENT_VIRTUAL_BLOCK_INFO_POSITION, SYSTEM_CONTEXT_ADDRESS, }; -/// Key used to set transaction metadata in other fields. -pub const ZKSYNC_TRANSACTION_OTHER_FIELDS_KEY: &str = "zksync"; - macro_rules! fmt_err { ($msg:literal $(,)?) => { Error::fmt(::std::format_args!($msg)) @@ -82,10 +81,15 @@ macro_rules! bail { }; } +/// ZKsync implementation for [CheatcodeInspectorStrategyRunner]. #[derive(Debug, Default, Clone)] -pub struct ZksyncCheatcodeInspectorStrategy { - evm: EvmCheatcodeInspectorStrategy, +pub struct ZksyncCheatcodeInspectorStrategyRunner { + evm: EvmCheatcodeInspectorStrategyRunner, +} +/// Context for [ZksyncCheatcodeInspectorStrategyRunner]. +#[derive(Debug, Default, Clone)] +pub struct ZksyncCheatcodeInspectorStrategyContext { pub using_zk_vm: bool, /// When in zkEVM context, execute the next CALL or CREATE in the EVM instead. @@ -130,7 +134,7 @@ pub struct ZksyncCheatcodeInspectorStrategy { pub zk_env: ZkEnv, } -impl ZksyncCheatcodeInspectorStrategy { +impl ZksyncCheatcodeInspectorStrategyContext { pub fn new(dual_compiled_contracts: DualCompiledContracts, zk_env: ZkEnv) -> Self { // We add the empty bytecode manually so it is correctly translated in zk mode. // This is used in many places in foundry, e.g. in cheatcode contract's account code. @@ -170,7 +174,6 @@ impl ZksyncCheatcodeInspectorStrategy { persisted_factory_deps.insert(zk_bytecode_hash, zk_deployed_bytecode); Self { - evm: EvmCheatcodeInspectorStrategy::default(), using_zk_vm: false, // We need to migrate once on initialize_interp skip_zk_vm: false, skip_zk_vm_addresses: Default::default(), @@ -187,6 +190,20 @@ impl ZksyncCheatcodeInspectorStrategy { } } +impl CheatcodeInspectorStrategyContext for ZksyncCheatcodeInspectorStrategyContext { + fn new_cloned(&self) -> Box { + Box::new(self.clone()) + } + + fn as_any_mut(&mut self) -> &mut dyn std::any::Any { + self + } + + fn as_any_ref(&self) -> &dyn std::any::Any { + self + } +} + /// Allows overriding nonce update behavior for the tx caller in the zkEVM. /// /// Since each CREATE or CALL is executed as a separate transaction within zkEVM, we currently skip @@ -220,17 +237,19 @@ impl ZkPersistNonceUpdate { } } -impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { +impl CheatcodeInspectorStrategyRunner for ZksyncCheatcodeInspectorStrategyRunner { fn name(&self) -> &'static str { "zk" } - fn new_cloned(&self) -> Box { + fn new_cloned(&self) -> Box { Box::new(self.clone()) } - fn get_nonce(&mut self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, address: Address) -> Result { - if !self.using_zk_vm { + fn get_nonce(&self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, address: Address) -> Result { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.get_nonce(ccx, address); } @@ -238,19 +257,23 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { Ok(nonce) } - fn base_contract_deployed(&mut self) { + fn base_contract_deployed(&self, ctx: &mut dyn CheatcodeInspectorStrategyContext) { + let ctx = get_context(ctx); + debug!("allowing startup storage migration"); - self.zk_startup_migration.allow(); + ctx.zk_startup_migration.allow(); debug!("allowing persisting next nonce update"); - self.zk_persist_nonce_update.persist_next(); + ctx.zk_persist_nonce_update.persist_next(); } fn cheatcode_get_nonce( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, address: Address, ) -> foundry_cheatcodes::Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { let nonce = self.evm.get_nonce(ccx, address)?; return Ok(nonce.abi_encode()); } @@ -259,8 +282,10 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { Ok(nonce.abi_encode()) } - fn cheatcode_roll(&mut self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, new_height: U256) -> Result { - if !self.using_zk_vm { + fn cheatcode_roll(&self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, new_height: U256) -> Result { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_roll(ccx, new_height); } @@ -269,12 +294,10 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { Ok(Default::default()) } - fn cheatcode_warp( - &mut self, - ccx: &mut CheatsCtxt<'_, '_, '_, '_>, - new_timestamp: U256, - ) -> Result { - if !self.using_zk_vm { + fn cheatcode_warp(&self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, new_timestamp: U256) -> Result { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_warp(ccx, new_timestamp); } @@ -284,12 +307,14 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn cheatcode_deal( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, address: Address, new_balance: U256, ) -> Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_deal(ccx, address, new_balance); } @@ -300,12 +325,14 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn cheatcode_etch( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, target: Address, new_runtime_bytecode: &Bytes, ) -> Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_etch(ccx, target, new_runtime_bytecode); } @@ -314,11 +341,13 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn cheatcode_reset_nonce( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, account: Address, ) -> Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_reset_nonce(ccx, account); } @@ -327,12 +356,14 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn cheatcode_set_nonce( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, account: Address, new_nonce: u64, ) -> Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_set_nonce(ccx, account, new_nonce); } @@ -350,12 +381,14 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn cheatcode_set_nonce_unsafe( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, account: Address, new_nonce: u64, ) -> Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_set_nonce_unsafe(ccx, account, new_nonce); } @@ -364,13 +397,15 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn cheatcode_mock_call( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, callee: Address, data: &Bytes, return_data: &Bytes, ) -> Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_mock_call(ccx, callee, data, return_data); } @@ -388,13 +423,15 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn cheatcode_mock_call_revert( - &mut self, + &self, ccx: &mut CheatsCtxt<'_, '_, '_, '_>, callee: Address, data: &Bytes, revert_data: &Bytes, ) -> Result { - if !self.using_zk_vm { + let ctx = get_context(ccx.state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return self.evm.cheatcode_mock_call_revert(ccx, callee, data, revert_data); } @@ -413,9 +450,11 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn get_artifact_code(&self, state: &Cheatcodes, path: &str, deployed: bool) -> Result { + let ctx = get_context_ref(state.strategy.context.as_ref()); + Ok(get_artifact_code( - &self.dual_compiled_contracts, - self.using_zk_vm, + &ctx.dual_compiled_contracts, + ctx.using_zk_vm, &state.config, path, deployed, @@ -424,15 +463,18 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn record_broadcastable_create_transactions( - &mut self, + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, config: Arc, input: &dyn CommonCreateInput, ecx_inner: InnerEcx<'_, '_, '_>, broadcast: &Broadcast, broadcastable_transactions: &mut BroadcastableTransactions, ) { - if !self.using_zk_vm { + let ctx_zk = get_context(ctx); + if !ctx_zk.using_zk_vm { return self.evm.record_broadcastable_create_transactions( + ctx, config, input, ecx_inner, @@ -441,13 +483,15 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { ); } + let ctx = ctx_zk; + let is_fixed_gas_limit = foundry_cheatcodes::check_if_fixed_gas_limit(ecx_inner, input.gas_limit()); let init_code = input.init_code(); let to = Some(TxKind::Call(CONTRACT_DEPLOYER_ADDRESS.to_address())); let mut nonce = foundry_zksync_core::nonce(broadcast.new_origin, ecx_inner) as u64; - let find_contract = self + let find_contract = ctx .dual_compiled_contracts .find_bytecode(&init_code.0) .unwrap_or_else(|| panic!("failed finding contract for {init_code:?}")); @@ -455,7 +499,7 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { let constructor_args = find_contract.constructor_args(); let contract = find_contract.contract(); - let factory_deps = self.dual_compiled_contracts.fetch_all_factory_deps(contract); + let factory_deps = ctx.dual_compiled_contracts.fetch_all_factory_deps(contract); let create_input = foundry_zksync_core::encode_create_params( &input.scheme().unwrap_or(CreateScheme::Create), @@ -466,21 +510,20 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { let mut zk_tx_factory_deps = factory_deps; - let paymaster_params = - self.paymaster_params.clone().map(|paymaster_data| PaymasterParams { - paymaster: paymaster_data.address.to_h160(), - paymaster_input: paymaster_data.input.to_vec(), - }); + let paymaster_params = ctx.paymaster_params.clone().map(|paymaster_data| PaymasterParams { + paymaster: paymaster_data.address.to_h160(), + paymaster_input: paymaster_data.input.to_vec(), + }); let rpc = ecx_inner.db.active_fork_url(); - let injected_factory_deps = self + let injected_factory_deps = ctx .zk_use_factory_deps .iter() .map(|contract| { get_artifact_code( - &self.dual_compiled_contracts, - self.using_zk_vm, + &ctx.dual_compiled_contracts, + ctx.using_zk_vm, &config, contract, false, @@ -545,7 +588,8 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } fn record_broadcastable_call_transactions( - &mut self, + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, config: Arc, call: &CallInputs, ecx_inner: InnerEcx<'_, '_, '_>, @@ -553,8 +597,11 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { broadcastable_transactions: &mut BroadcastableTransactions, active_delegation: &mut Option, ) { - if !self.using_zk_vm { + let ctx_zk = get_context(ctx); + + if !ctx_zk.using_zk_vm { return self.evm.record_broadcastable_call_transactions( + ctx, config, call, ecx_inner, @@ -564,19 +611,21 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { ); } + let ctx = ctx_zk; + let is_fixed_gas_limit = foundry_cheatcodes::check_if_fixed_gas_limit(ecx_inner, call.gas_limit); let nonce = foundry_zksync_core::nonce(broadcast.new_origin, ecx_inner) as u64; - let factory_deps = &mut self.set_deployer_call_input_factory_deps; - let injected_factory_deps = self + let factory_deps = &mut ctx.set_deployer_call_input_factory_deps; + let injected_factory_deps = ctx .zk_use_factory_deps .iter() .flat_map(|contract| { let artifact_code = get_artifact_code( - &self.dual_compiled_contracts, - self.using_zk_vm, + &ctx.dual_compiled_contracts, + ctx.using_zk_vm, &config, contract, false, @@ -586,17 +635,16 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { panic!("failed to get bytecode for factory deps contract {contract}") }) .to_vec(); - let res = self.dual_compiled_contracts.find_bytecode(&artifact_code).unwrap(); - self.dual_compiled_contracts.fetch_all_factory_deps(res.contract()) + let res = ctx.dual_compiled_contracts.find_bytecode(&artifact_code).unwrap(); + ctx.dual_compiled_contracts.fetch_all_factory_deps(res.contract()) }) .collect_vec(); factory_deps.extend(injected_factory_deps.clone()); - let paymaster_params = - self.paymaster_params.clone().map(|paymaster_data| PaymasterParams { - paymaster: paymaster_data.address.to_h160(), - paymaster_input: paymaster_data.input.to_vec(), - }); + let paymaster_params = ctx.paymaster_params.clone().map(|paymaster_data| PaymasterParams { + paymaster: paymaster_data.address.to_h160(), + paymaster_input: paymaster_data.input.to_vec(), + }); let factory_deps = if call.target_address == DEFAULT_CREATE2_DEPLOYER_ZKSYNC { // We shouldn't need factory_deps for CALLs factory_deps.clone() @@ -636,18 +684,32 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { debug!(target: "cheatcodes", tx=?broadcastable_transactions.back().unwrap(), "broadcastable call"); } - fn post_initialize_interp(&mut self, _interpreter: &mut Interpreter, ecx: Ecx<'_, '_, '_>) { - if self.zk_startup_migration.is_allowed() && !self.using_zk_vm { - self.select_zk_vm(ecx, None); - self.zk_startup_migration.done(); + fn post_initialize_interp( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + _interpreter: &mut Interpreter, + ecx: Ecx<'_, '_, '_>, + ) { + let ctx = get_context(ctx); + + if ctx.zk_startup_migration.is_allowed() && !ctx.using_zk_vm { + self.select_zk_vm(ctx, ecx, None); + ctx.zk_startup_migration.done(); debug!("startup zkEVM storage migration completed"); } } /// Returns true if handled. - fn pre_step_end(&mut self, interpreter: &mut Interpreter, ecx: Ecx<'_, '_, '_>) -> bool { + fn pre_step_end( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + interpreter: &mut Interpreter, + ecx: Ecx<'_, '_, '_>, + ) -> bool { // override address(x).balance retrieval to make it consistent between EraVM and EVM - if !self.using_zk_vm { + let ctx = get_context(ctx); + + if !ctx.using_zk_vm { return false; } @@ -681,30 +743,45 @@ impl CheatcodeInspectorStrategy for ZksyncCheatcodeInspectorStrategy { } } -impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { - fn zksync_cheatcode_skip_zkvm(&mut self) -> Result { - self.skip_zk_vm = true; +impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategyRunner { + fn zksync_cheatcode_skip_zkvm( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + ) -> Result { + let ctx = get_context(ctx); + + ctx.skip_zk_vm = true; Ok(Default::default()) } fn zksync_cheatcode_set_paymaster( - &mut self, + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, paymaster_address: Address, paymaster_input: &Bytes, ) -> Result { - self.paymaster_params = + let ctx = get_context(ctx); + + ctx.paymaster_params = Some(ZkPaymasterData { address: paymaster_address, input: paymaster_input.clone() }); Ok(Default::default()) } - fn zksync_cheatcode_use_factory_deps(&mut self, name: String) -> foundry_cheatcodes::Result { + fn zksync_cheatcode_use_factory_deps( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + name: String, + ) -> foundry_cheatcodes::Result { + let ctx = get_context(ctx); + info!("Adding factory dependency: {:?}", name); - self.zk_use_factory_deps.push(name); + ctx.zk_use_factory_deps.push(name); Ok(Default::default()) } fn zksync_cheatcode_register_contract( - &mut self, + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, name: String, zk_bytecode_hash: FixedBytes<32>, zk_deployed_bytecode: Vec, @@ -713,6 +790,8 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { evm_deployed_bytecode: Vec, evm_bytecode: Vec, ) -> Result { + let ctx = get_context(ctx); + let new_contract = DualCompiledContract { name, zk_bytecode_hash: H256(zk_bytecode_hash.0), @@ -723,7 +802,7 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { evm_bytecode, }; - if let Some(existing) = self.dual_compiled_contracts.iter().find(|contract| { + if let Some(existing) = ctx.dual_compiled_contracts.iter().find(|contract| { contract.evm_bytecode_hash == new_contract.evm_bytecode_hash && contract.zk_bytecode_hash == new_contract.zk_bytecode_hash }) { @@ -731,33 +810,51 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { return Ok(Default::default()) } - self.dual_compiled_contracts.push(new_contract); + ctx.dual_compiled_contracts.push(new_contract); Ok(Default::default()) } - fn zksync_record_create_address(&mut self, outcome: &CreateOutcome) { - if self.record_next_create_address { - self.record_next_create_address = false; + fn zksync_record_create_address( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + outcome: &CreateOutcome, + ) { + let ctx = get_context(ctx); + + if ctx.record_next_create_address { + ctx.record_next_create_address = false; if let Some(address) = outcome.address { - self.skip_zk_vm_addresses.insert(address); + ctx.skip_zk_vm_addresses.insert(address); } } } - fn zksync_sync_nonce(&mut self, sender: Address, nonce: u64, ecx: Ecx<'_, '_, '_>) { + fn zksync_sync_nonce( + &self, + _ctx: &mut dyn CheatcodeInspectorStrategyContext, + sender: Address, + nonce: u64, + ecx: Ecx<'_, '_, '_>, + ) { // NOTE(zk): We sync with the nonce changes to ensure that the nonce matches foundry_zksync_core::cheatcodes::set_nonce(sender, U256::from(nonce), ecx); } - fn zksync_set_deployer_call_input(&mut self, call: &mut CallInputs) { - self.set_deployer_call_input_factory_deps.clear(); - if call.target_address == DEFAULT_CREATE2_DEPLOYER && self.using_zk_vm { + fn zksync_set_deployer_call_input( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + call: &mut CallInputs, + ) { + let ctx = get_context(ctx); + + ctx.set_deployer_call_input_factory_deps.clear(); + if call.target_address == DEFAULT_CREATE2_DEPLOYER && ctx.using_zk_vm { call.target_address = DEFAULT_CREATE2_DEPLOYER_ZKSYNC; call.bytecode_address = DEFAULT_CREATE2_DEPLOYER_ZKSYNC; let (salt, init_code) = call.input.split_at(32); - let find_contract = self + let find_contract = ctx .dual_compiled_contracts .find_bytecode(init_code) .unwrap_or_else(|| panic!("failed finding contract for {init_code:?}")); @@ -766,8 +863,8 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { let contract = find_contract.contract(); // store these for broadcast reasons - self.set_deployer_call_input_factory_deps = - self.dual_compiled_contracts.fetch_all_factory_deps(contract); + ctx.set_deployer_call_input_factory_deps = + ctx.dual_compiled_contracts.fetch_all_factory_deps(contract); let create_input = foundry_zksync_core::encode_create_params( &CreateScheme::Create2 { salt: U256::from_be_slice(salt) }, @@ -783,19 +880,21 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { /// If `Some` is returned then the result must be returned immediately, else the call must be /// handled in EVM. fn zksync_try_create( - &mut self, + &self, state: &mut Cheatcodes, ecx: Ecx<'_, '_, '_>, input: &dyn CommonCreateInput, executor: &mut dyn CheatcodesExecutor, ) -> Option { - if !self.using_zk_vm { + let ctx = get_context(state.strategy.context.as_mut()); + + if !ctx.using_zk_vm { return None; } - if self.skip_zk_vm { - self.skip_zk_vm = false; // handled the skip, reset flag - self.record_next_create_address = true; + if ctx.skip_zk_vm { + ctx.skip_zk_vm = false; // handled the skip, reset flag + ctx.record_next_create_address = true; info!("running create in EVM, instead of zkEVM (skipped)"); return None } @@ -824,7 +923,7 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { info!("running create in zkEVM"); - let find_contract = self + let find_contract = ctx .dual_compiled_contracts .find_bytecode(&init_code.0) .unwrap_or_else(|| panic!("failed finding contract for {init_code:?}")); @@ -838,14 +937,14 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { constructor_args.to_vec(), ); - let mut factory_deps = self.dual_compiled_contracts.fetch_all_factory_deps(contract); - let injected_factory_deps = self + let mut factory_deps = ctx.dual_compiled_contracts.fetch_all_factory_deps(contract); + let injected_factory_deps = ctx .zk_use_factory_deps .iter() .flat_map(|contract| { let artifact_code = get_artifact_code( - &self.dual_compiled_contracts, - self.using_zk_vm, + &ctx.dual_compiled_contracts, + ctx.using_zk_vm, &state.config, contract, false, @@ -855,25 +954,25 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { panic!("failed to get bytecode for injected factory deps contract {contract}") }) .to_vec(); - let res = self.dual_compiled_contracts.find_bytecode(&artifact_code).unwrap(); - self.dual_compiled_contracts.fetch_all_factory_deps(res.contract()) + let res = ctx.dual_compiled_contracts.find_bytecode(&artifact_code).unwrap(); + ctx.dual_compiled_contracts.fetch_all_factory_deps(res.contract()) }) .collect_vec(); factory_deps.extend(injected_factory_deps); // NOTE(zk): Clear injected factory deps so that they are not sent on further transactions - self.zk_use_factory_deps.clear(); + ctx.zk_use_factory_deps.clear(); tracing::debug!(contract = contract.name, "using dual compiled contract"); - let zk_persist_nonce_update = self.zk_persist_nonce_update.check(); + let zk_persist_nonce_update = ctx.zk_persist_nonce_update.check(); let ccx = foundry_zksync_core::vm::CheatcodeTracerContext { mocked_calls: state.mocked_calls.clone(), expected_calls: Some(&mut state.expected_calls), accesses: state.accesses.as_mut(), - persisted_factory_deps: Some(&mut self.persisted_factory_deps), - paymaster_data: self.paymaster_params.take(), + persisted_factory_deps: Some(&mut ctx.persisted_factory_deps), + paymaster_data: ctx.paymaster_params.take(), persist_nonce_update: state.broadcast.is_some() || zk_persist_nonce_update, - zk_env: self.zk_env.clone(), + zk_env: ctx.zk_env.clone(), }; let zk_create = foundry_zksync_core::vm::ZkCreateInputs { @@ -935,7 +1034,12 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { .to_ru256() }) .collect::>(); - ecx.db.get_strategy().zksync_save_immutable_storage(addr, keys); + let strategy = ecx.db.get_strategy(); + strategy.runner.zksync_save_immutable_storage( + strategy.context.as_mut(), + addr, + keys, + ); } } @@ -1002,24 +1106,25 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { /// If `Some` is returned then the result must be returned immediately, else the call must be /// handled in EVM. fn zksync_try_call( - &mut self, + &self, state: &mut Cheatcodes, ecx: Ecx<'_, '_, '_>, call: &CallInputs, executor: &mut dyn CheatcodesExecutor, ) -> Option { + let ctx = get_context(state.strategy.context.as_mut()); + // We need to clear them out for the next call. - let factory_deps = std::mem::take(&mut self.set_deployer_call_input_factory_deps); + let factory_deps = std::mem::take(&mut ctx.set_deployer_call_input_factory_deps); - if !self.using_zk_vm { + if !ctx.using_zk_vm { return None; } // also skip if the target was created during a zkEVM skip - self.skip_zk_vm = - self.skip_zk_vm || self.skip_zk_vm_addresses.contains(&call.target_address); - if self.skip_zk_vm { - self.skip_zk_vm = false; // handled the skip, reset flag + ctx.skip_zk_vm = ctx.skip_zk_vm || ctx.skip_zk_vm_addresses.contains(&call.target_address); + if ctx.skip_zk_vm { + ctx.skip_zk_vm = false; // handled the skip, reset flag info!("running create in EVM, instead of zkEVM (skipped) {:#?}", call); return None; } @@ -1038,20 +1143,20 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { } info!("running call in zkEVM {:#?}", call); - let zk_persist_nonce_update = self.zk_persist_nonce_update.check(); + let zk_persist_nonce_update = ctx.zk_persist_nonce_update.check(); // NOTE(zk): Clear injected factory deps here even though it's actually used in broadcast. // To be consistent with where we clear factory deps in try_create_in_zk. - self.zk_use_factory_deps.clear(); + ctx.zk_use_factory_deps.clear(); let ccx = foundry_zksync_core::vm::CheatcodeTracerContext { mocked_calls: state.mocked_calls.clone(), expected_calls: Some(&mut state.expected_calls), accesses: state.accesses.as_mut(), - persisted_factory_deps: Some(&mut self.persisted_factory_deps), - paymaster_data: self.paymaster_params.take(), + persisted_factory_deps: Some(&mut ctx.persisted_factory_deps), + paymaster_data: ctx.paymaster_params.take(), persist_nonce_update: state.broadcast.is_some() || zk_persist_nonce_update, - zk_env: self.zk_env.clone(), + zk_env: ctx.zk_env.clone(), }; let mut gas = Gas::new(call.gas_limit); @@ -1157,45 +1262,66 @@ impl CheatcodeInspectorStrategyExt for ZksyncCheatcodeInspectorStrategy { } } - fn zksync_select_fork_vm(&mut self, data: InnerEcx<'_, '_, '_>, fork_id: LocalForkId) { - self.select_fork_vm(data, fork_id); + fn zksync_select_fork_vm( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + data: InnerEcx<'_, '_, '_>, + fork_id: LocalForkId, + ) { + let ctx = get_context(ctx); + self.select_fork_vm(ctx, data, fork_id); } - fn zksync_cheatcode_select_zk_vm(&mut self, data: InnerEcx<'_, '_, '_>, enable: bool) { + fn zksync_cheatcode_select_zk_vm( + &self, + ctx: &mut dyn CheatcodeInspectorStrategyContext, + data: InnerEcx<'_, '_, '_>, + enable: bool, + ) { + let ctx = get_context(ctx); if enable { - self.select_zk_vm(data, None) + self.select_zk_vm(ctx, data, None) } else { - self.select_evm(data); + self.select_evm(ctx, data); } } } -impl ZksyncCheatcodeInspectorStrategy { +impl ZksyncCheatcodeInspectorStrategyRunner { /// Selects the appropriate VM for the fork. Options: EVM, ZK-VM. /// CALL and CREATE are handled by the selected VM. /// /// Additionally: /// * Translates block information /// * Translates all persisted addresses - pub fn select_fork_vm(&mut self, data: InnerEcx<'_, '_, '_>, fork_id: LocalForkId) { + pub fn select_fork_vm( + &self, + ctx: &mut ZksyncCheatcodeInspectorStrategyContext, + data: InnerEcx<'_, '_, '_>, + fork_id: LocalForkId, + ) { let fork_info = data.db.get_fork_info(fork_id).expect("failed getting fork info"); if fork_info.fork_type.is_evm() { - self.select_evm(data) + self.select_evm(ctx, data) } else { - self.select_zk_vm(data, Some(&fork_info.fork_env)) + self.select_zk_vm(ctx, data, Some(&fork_info.fork_env)) } } /// Switch to EVM and translate block info, balances, nonces and deployed codes for persistent /// accounts - pub fn select_evm(&mut self, data: InnerEcx<'_, '_, '_>) { - if !self.using_zk_vm { + pub fn select_evm( + &self, + ctx: &mut ZksyncCheatcodeInspectorStrategyContext, + data: InnerEcx<'_, '_, '_>, + ) { + if !ctx.using_zk_vm { tracing::info!("already in EVM"); return } tracing::info!("switching to EVM"); - self.using_zk_vm = false; + ctx.using_zk_vm = false; let system_account = SYSTEM_CONTEXT_ADDRESS.to_address(); journaled_account(data, system_account).expect("failed to load account"); @@ -1232,7 +1358,7 @@ impl ZksyncCheatcodeInspectorStrategy { .sload(account_code_account, account_code_key) .ok() .and_then(|zk_bytecode_hash| { - self.dual_compiled_contracts + ctx.dual_compiled_contracts .find_by_zk_bytecode_hash(zk_bytecode_hash.to_h256()) .map(|contract| { ( @@ -1260,14 +1386,19 @@ impl ZksyncCheatcodeInspectorStrategy { /// Switch to ZK-VM and translate block info, balances, nonces and deployed codes for persistent /// accounts - pub fn select_zk_vm(&mut self, data: InnerEcx<'_, '_, '_>, new_env: Option<&Env>) { - if self.using_zk_vm { + pub fn select_zk_vm( + &self, + ctx: &mut ZksyncCheatcodeInspectorStrategyContext, + data: InnerEcx<'_, '_, '_>, + new_env: Option<&Env>, + ) { + if ctx.using_zk_vm { tracing::info!("already in ZK-VM"); return } tracing::info!("switching to ZK-VM"); - self.using_zk_vm = true; + ctx.using_zk_vm = true; let env = new_env.unwrap_or(data.env.as_ref()); @@ -1305,7 +1436,7 @@ impl ZksyncCheatcodeInspectorStrategy { continue; } - if let Some(contract) = self.dual_compiled_contracts.iter().find(|contract| { + if let Some(contract) = ctx.dual_compiled_contracts.iter().find(|contract| { info.code_hash != KECCAK_EMPTY && info.code_hash == contract.evm_bytecode_hash }) { account_code_storage.insert( @@ -1365,6 +1496,58 @@ impl ZksyncCheatcodeInspectorStrategy { } } +/// Setting for migrating the database to zkEVM storage when starting in ZKsync mode. +/// The migration is performed on the DB via the inspector so must only be performed once. +#[derive(Debug, Default, Clone)] +pub enum ZkStartupMigration { + /// Defer database migration to a later execution point. + /// + /// This is required as we need to wait for some baseline deployments + /// to occur before the test/script execution is performed. + #[default] + Defer, + /// Allow database migration. + Allow, + /// Database migration has already been performed. + Done, +} + +impl ZkStartupMigration { + /// Check if startup migration is allowed. Migration is disallowed if it's to be deferred or has + /// already been performed. + pub fn is_allowed(&self) -> bool { + matches!(self, Self::Allow) + } + + /// Allow migrating the the DB to zkEVM storage. + pub fn allow(&mut self) { + *self = Self::Allow + } + + /// Mark the migration as completed. It must not be performed again. + pub fn done(&mut self) { + *self = Self::Done + } +} + +/// Create ZKsync strategy for [CheatcodeInspectorStrategy]. +pub trait ZksyncCheatcodeInspectorStrategyBuilder { + /// Create new ZKsync strategy. + fn new_zksync(dual_compiled_contracts: DualCompiledContracts, zk_env: ZkEnv) -> Self; +} + +impl ZksyncCheatcodeInspectorStrategyBuilder for CheatcodeInspectorStrategy { + fn new_zksync(dual_compiled_contracts: DualCompiledContracts, zk_env: ZkEnv) -> Self { + Self { + runner: Box::new(ZksyncCheatcodeInspectorStrategyRunner::default()), + context: Box::new(ZksyncCheatcodeInspectorStrategyContext::new( + dual_compiled_contracts, + zk_env, + )), + } + } +} + fn get_artifact_code( dual_compiled_contracts: &DualCompiledContracts, using_zk_vm: bool, @@ -1496,36 +1679,14 @@ fn get_artifact_code( maybe_bytecode.ok_or_else(|| fmt_err!("no bytecode for contract; is it abstract or unlinked?")) } -/// Setting for migrating the database to zkEVM storage when starting in ZKsync mode. -/// The migration is performed on the DB via the inspector so must only be performed once. -#[derive(Debug, Default, Clone)] -pub enum ZkStartupMigration { - /// Defer database migration to a later execution point. - /// - /// This is required as we need to wait for some baseline deployments - /// to occur before the test/script execution is performed. - #[default] - Defer, - /// Allow database migration. - Allow, - /// Database migration has already been performed. - Done, +fn get_context( + ctx: &mut dyn CheatcodeInspectorStrategyContext, +) -> &mut ZksyncCheatcodeInspectorStrategyContext { + ctx.as_any_mut().downcast_mut().expect("expected ZksyncCheatcodeInspectorStrategyContext") } -impl ZkStartupMigration { - /// Check if startup migration is allowed. Migration is disallowed if it's to be deferred or has - /// already been performed. - pub fn is_allowed(&self) -> bool { - matches!(self, Self::Allow) - } - - /// Allow migrating the the DB to zkEVM storage. - pub fn allow(&mut self) { - *self = Self::Allow - } - - /// Mark the migration as completed. It must not be performed again. - pub fn done(&mut self) { - *self = Self::Done - } +fn get_context_ref( + ctx: &dyn CheatcodeInspectorStrategyContext, +) -> &ZksyncCheatcodeInspectorStrategyContext { + ctx.as_any_ref().downcast_ref().expect("expected ZksyncCheatcodeInspectorStrategyContext") } diff --git a/crates/strategy/zksync/src/executor.rs b/crates/strategy/zksync/src/executor.rs index 789b757b0..715f7ca07 100644 --- a/crates/strategy/zksync/src/executor.rs +++ b/crates/strategy/zksync/src/executor.rs @@ -2,54 +2,77 @@ use alloy_primitives::{Address, U256}; use alloy_rpc_types::serde_helpers::OtherFields; use alloy_zksync::provider::{zksync_provider, ZksyncProvider}; use eyre::Result; -use foundry_cheatcodes::strategy::CheatcodeInspectorStrategy; use foundry_evm::{ - backend::{strategy::BackendStrategy, BackendResult, DatabaseExt}, + backend::{Backend, BackendResult, CowBackend}, executors::{ - strategy::{EvmExecutorStrategy, ExecutorStrategy, ExecutorStrategyExt}, + strategy::{ + EvmExecutorStrategyRunner, ExecutorStrategy, ExecutorStrategyContext, + ExecutorStrategyExt, ExecutorStrategyRunner, + }, Executor, }, - InspectorExt, + inspectors::InspectorStack, }; use foundry_zksync_compilers::dual_compiled_contracts::DualCompiledContracts; -use foundry_zksync_core::{vm::ZkEnv, ZkTransactionMetadata}; +use foundry_zksync_core::{vm::ZkEnv, ZkTransactionMetadata, ZKSYNC_TRANSACTION_OTHER_FIELDS_KEY}; use revm::{ - primitives::{Env, EnvWithHandlerCfg, HashMap, ResultAndState}, + primitives::{Env, EnvWithHandlerCfg, ResultAndState}, Database, }; -use zksync_types::H256; use crate::{ - cheatcode::ZKSYNC_TRANSACTION_OTHER_FIELDS_KEY, ZksyncBackendStrategy, - ZksyncCheatcodeInspectorStrategy, + backend::{ZksyncBackendStrategyBuilder, ZksyncInspectContext}, + cheatcode::ZksyncCheatcodeInspectorStrategyBuilder, }; +/// Defines the context for [ZksyncExecutorStrategyRunner]. #[derive(Debug, Default, Clone)] -pub struct ZksyncExecutorStrategy { - evm: EvmExecutorStrategy, - inspect_context: Option, - persisted_factory_deps: HashMap>, +pub struct ZksyncExecutorStrategyContext { + transaction_context: Option, dual_compiled_contracts: DualCompiledContracts, zk_env: ZkEnv, } -impl ExecutorStrategy for ZksyncExecutorStrategy { +impl ExecutorStrategyContext for ZksyncExecutorStrategyContext { + fn new_cloned(&self) -> Box { + Box::new(self.clone()) + } + + fn as_any_ref(&self) -> &dyn std::any::Any { + self + } + + fn as_any_mut(&mut self) -> &mut dyn std::any::Any { + self + } +} + +/// Defines the [ExecutorStrategyRunner] strategy for ZKsync. +#[derive(Debug, Default, Clone)] +pub struct ZksyncExecutorStrategyRunner { + evm: EvmExecutorStrategyRunner, +} + +fn get_context_ref(ctx: &dyn ExecutorStrategyContext) -> &ZksyncExecutorStrategyContext { + ctx.as_any_ref().downcast_ref().expect("expected ZksyncExecutorStrategyContext") +} + +fn get_context(ctx: &mut dyn ExecutorStrategyContext) -> &mut ZksyncExecutorStrategyContext { + ctx.as_any_mut().downcast_mut().expect("expected ZksyncExecutorStrategyContext") +} + +impl ExecutorStrategyRunner for ZksyncExecutorStrategyRunner { fn name(&self) -> &'static str { "zk" } - fn new_cloned(&self) -> Box { + fn new_cloned(&self) -> Box { Box::new(self.clone()) } - fn set_inspect_context(&mut self, other_fields: OtherFields) { - let maybe_context = get_zksync_transaction_metadata(&other_fields); - self.inspect_context = maybe_context; - } - fn set_balance( - &mut self, + &self, executor: &mut Executor, address: Address, amount: U256, @@ -63,7 +86,7 @@ impl ExecutorStrategy for ZksyncExecutorStrategy { } fn set_nonce( - &mut self, + &self, executor: &mut Executor, address: Address, nonce: u64, @@ -81,73 +104,95 @@ impl ExecutorStrategy for ZksyncExecutorStrategy { Ok(()) } - fn new_backend_strategy(&self) -> Box { - Box::new(ZksyncBackendStrategy::default()) + fn new_backend_strategy(&self) -> foundry_evm_core::backend::strategy::BackendStrategy { + foundry_evm_core::backend::strategy::BackendStrategy::new_zksync() } - fn new_cheatcode_inspector_strategy(&self) -> Box { - Box::new(ZksyncCheatcodeInspectorStrategy::new( - self.dual_compiled_contracts.clone(), - self.zk_env.clone(), - )) + fn new_cheatcode_inspector_strategy( + &self, + ctx: &dyn ExecutorStrategyContext, + ) -> foundry_cheatcodes::strategy::CheatcodeInspectorStrategy { + let ctx = get_context_ref(ctx); + foundry_cheatcodes::strategy::CheatcodeInspectorStrategy::new_zksync( + ctx.dual_compiled_contracts.clone(), + ctx.zk_env.clone(), + ) } - fn call_inspect( + fn call( &self, - db: &mut dyn DatabaseExt, + ctx: &dyn ExecutorStrategyContext, + backend: &mut CowBackend<'_>, env: &mut EnvWithHandlerCfg, - inspector: &mut dyn InspectorExt, + executor_env: &EnvWithHandlerCfg, + inspector: &mut InspectorStack, ) -> eyre::Result { - match self.inspect_context.as_ref() { - None => self.evm.call_inspect(db, env, inspector), - Some(zk_tx) => foundry_zksync_core::vm::transact( - Some(&mut self.persisted_factory_deps.clone()), - Some(zk_tx.factory_deps.clone()), - zk_tx.paymaster_data.clone(), + let ctx = get_context_ref(ctx); + + match ctx.transaction_context.as_ref() { + None => self.evm.call(ctx, backend, env, executor_env, inspector), + Some(zk_tx) => backend.inspect( env, - &self.zk_env, - db, + inspector, + Box::new(ZksyncInspectContext { + factory_deps: zk_tx.factory_deps.clone(), + paymaster_data: zk_tx.paymaster_data.clone(), + zk_env: ctx.zk_env.clone(), + }), ), } } - fn transact_inspect( - &mut self, - db: &mut dyn DatabaseExt, + fn transact( + &self, + ctx: &mut dyn ExecutorStrategyContext, + backend: &mut Backend, env: &mut EnvWithHandlerCfg, executor_env: &EnvWithHandlerCfg, - inspector: &mut dyn InspectorExt, + inspector: &mut InspectorStack, ) -> eyre::Result { - match self.inspect_context.take() { - None => self.evm.transact_inspect(db, env, executor_env, inspector), + let ctx = get_context(ctx); + + match ctx.transaction_context.take() { + None => self.evm.transact(ctx, backend, env, executor_env, inspector), Some(zk_tx) => { // apply fork-related env instead of cheatcode handler // since it won't be set by zkEVM env.block = executor_env.block.clone(); env.tx.gas_price = executor_env.tx.gas_price; - foundry_zksync_core::vm::transact( - Some(&mut self.persisted_factory_deps), - Some(zk_tx.factory_deps), - zk_tx.paymaster_data, + backend.inspect( env, - &self.zk_env, - db, + inspector, + Box::new(ZksyncInspectContext { + factory_deps: zk_tx.factory_deps, + paymaster_data: zk_tx.paymaster_data, + zk_env: ctx.zk_env.clone(), + }), ) } } } } -impl ExecutorStrategyExt for ZksyncExecutorStrategy { +impl ExecutorStrategyExt for ZksyncExecutorStrategyRunner { fn zksync_set_dual_compiled_contracts( - &mut self, + &self, + ctx: &mut dyn ExecutorStrategyContext, dual_compiled_contracts: DualCompiledContracts, ) { - self.dual_compiled_contracts = dual_compiled_contracts; + let ctx = get_context(ctx); + ctx.dual_compiled_contracts = dual_compiled_contracts; } - fn zksync_set_fork_env(&mut self, fork_url: &str, env: &Env) -> Result<()> { + fn zksync_set_fork_env( + &self, + ctx: &mut dyn ExecutorStrategyContext, + fork_url: &str, + env: &Env, + ) -> Result<()> { + let ctx = get_context(ctx); + let provider = zksync_provider().with_recommended_fillers().on_http(fork_url.parse()?); let block_number = env.block.number.try_into()?; // TODO(zk): switch to getFeeParams call when it is implemented for anvil-zksync @@ -158,7 +203,7 @@ impl ExecutorStrategyExt for ZksyncExecutorStrategy { .flatten(); if let Some(block_details) = maybe_block_details { - self.zk_env = ZkEnv { + ctx.zk_env = ZkEnv { l1_gas_price: block_details .l1_gas_price .try_into() @@ -180,10 +225,19 @@ impl ExecutorStrategyExt for ZksyncExecutorStrategy { Ok(()) } + + fn zksync_set_transaction_context( + &self, + ctx: &mut dyn ExecutorStrategyContext, + other_fields: OtherFields, + ) { + let ctx = get_context(ctx); + let transaction_context = try_get_zksync_transaction_metadata(&other_fields); + ctx.transaction_context = transaction_context; + } } -/// Retrieve metadata for zksync tx -pub fn get_zksync_transaction_metadata( +pub fn try_get_zksync_transaction_metadata( other_fields: &OtherFields, ) -> Option { other_fields @@ -192,3 +246,18 @@ pub fn get_zksync_transaction_metadata( .ok() .flatten() } + +/// Create ZKsync strategy for [ExecutorStrategy]. +pub trait ZksyncExecutorStrategyBuilder { + /// Create new zksync strategy. + fn new_zksync() -> Self; +} + +impl ZksyncExecutorStrategyBuilder for ExecutorStrategy { + fn new_zksync() -> Self { + Self { + runner: Box::new(ZksyncExecutorStrategyRunner::default()), + context: Box::new(ZksyncExecutorStrategyContext::default()), + } + } +} diff --git a/crates/strategy/zksync/src/lib.rs b/crates/strategy/zksync/src/lib.rs index 0d7e2493b..33c144293 100644 --- a/crates/strategy/zksync/src/lib.rs +++ b/crates/strategy/zksync/src/lib.rs @@ -9,6 +9,9 @@ mod backend; mod cheatcode; mod executor; -pub use backend::ZksyncBackendStrategy; -pub use cheatcode::ZksyncCheatcodeInspectorStrategy; -pub use executor::{get_zksync_transaction_metadata, ZksyncExecutorStrategy}; +pub use backend::ZksyncBackendStrategyRunner; +pub use cheatcode::ZksyncCheatcodeInspectorStrategyRunner; +pub use executor::{ + try_get_zksync_transaction_metadata, ZksyncExecutorStrategyBuilder, + ZksyncExecutorStrategyRunner, +}; diff --git a/crates/verify/src/bytecode.rs b/crates/verify/src/bytecode.rs index 24ceaa547..5a9592b4b 100644 --- a/crates/verify/src/bytecode.rs +++ b/crates/verify/src/bytecode.rs @@ -240,7 +240,7 @@ impl VerifyBytecodeArgs { gen_blk_num, etherscan_metadata.evm_version()?.unwrap_or(EvmVersion::default()), evm_opts, - strategy.new_cloned(), + strategy.clone(), ) .await?; @@ -444,7 +444,7 @@ impl VerifyBytecodeArgs { simulation_block - 1, // env.fork_block_number etherscan_metadata.evm_version()?.unwrap_or(EvmVersion::default()), evm_opts, - strategy.new_cloned(), + strategy.clone(), ) .await?; env.block.number = U256::from(simulation_block); diff --git a/crates/verify/src/utils.rs b/crates/verify/src/utils.rs index 9defef666..403f5be61 100644 --- a/crates/verify/src/utils.rs +++ b/crates/verify/src/utils.rs @@ -326,7 +326,7 @@ pub async fn get_tracing_executor( fork_blk_num: u64, evm_version: EvmVersion, evm_opts: EvmOpts, - strategy: Box, + strategy: ExecutorStrategy, ) -> Result<(Env, TracingExecutor)> { fork_config.fork_block_number = Some(fork_blk_num); fork_config.evm_version = evm_version; diff --git a/crates/zksync/compilers/src/dual_compiled_contracts.rs b/crates/zksync/compilers/src/dual_compiled_contracts.rs index ede4c2360..6830f1de9 100644 --- a/crates/zksync/compilers/src/dual_compiled_contracts.rs +++ b/crates/zksync/compilers/src/dual_compiled_contracts.rs @@ -300,4 +300,14 @@ impl DualCompiledContracts { pub fn push(&mut self, contract: DualCompiledContract) { self.contracts.push(contract); } + + /// Retrieves the length of the collection. + pub fn len(&self) -> usize { + self.contracts.len() + } + + /// Retrieves if the collection is empty. + pub fn is_empty(&self) -> bool { + self.contracts.is_empty() + } } diff --git a/crates/zksync/core/src/lib.rs b/crates/zksync/core/src/lib.rs index e6b15b428..3a1bbf881 100644 --- a/crates/zksync/core/src/lib.rs +++ b/crates/zksync/core/src/lib.rs @@ -84,6 +84,10 @@ pub struct ZkPaymasterData { pub input: Bytes, } +/// Key used to set transaction metadata in other fields of [WithOtherFields]. +/// This is used when broadcasting in a test, and when a script reads the broadcasted transactions. +pub const ZKSYNC_TRANSACTION_OTHER_FIELDS_KEY: &str = "zksync"; + /// Represents additional data for ZK transactions. #[derive(Clone, Debug, Default, Serialize, Deserialize)] #[serde(rename_all = "camelCase")] diff --git a/foundryup-zksync/foundryup-zksync b/foundryup-zksync/foundryup-zksync index a157c78d9..21893e061 100755 --- a/foundryup-zksync/foundryup-zksync +++ b/foundryup-zksync/foundryup-zksync @@ -207,7 +207,7 @@ EOF architecture="aarch64" ;; *) - err "Unsupported architecture detected!" + err "Unsupported architecture '$arch' detected!" ;; esac