Skip to content

Commit

Permalink
Merge pull request #805 from matter-labs/popzxc-move-cheatcodes
Browse files Browse the repository at this point in the history
refactor: Move custom cheatcode logic into a generic handler
  • Loading branch information
popzxc authored Jan 6, 2025
2 parents ddf8b75 + b38a46c commit 2a052c6
Show file tree
Hide file tree
Showing 8 changed files with 274 additions and 517 deletions.
50 changes: 39 additions & 11 deletions crates/cheatcodes/src/evm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ use crate::{
};
use alloy_consensus::TxEnvelope;
use alloy_genesis::{Genesis, GenesisAccount};
use alloy_primitives::{Address, B256, U256};
use alloy_primitives::{Address, Bytes, B256, U256};
use alloy_rlp::Decodable;
use alloy_sol_types::SolValue;
use foundry_common::fs::{read_json_file, write_json_file};
Expand All @@ -18,7 +18,7 @@ use foundry_evm_core::{
};
use foundry_evm_traces::StackSnapshotType;
use rand::Rng;
use revm::primitives::{Account, SpecId};
use revm::primitives::{Account, Bytecode, SpecId, KECCAK_EMPTY};
use std::{collections::BTreeMap, path::Path};
mod record_debug_step;
use record_debug_step::{convert_call_trace_to_debug_step, flatten_call_trace};
Expand Down Expand Up @@ -63,8 +63,7 @@ impl Cheatcode for addrCall {
impl Cheatcode for getNonce_0Call {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { account } = self;

ccx.state.strategy.runner.clone().cheatcode_get_nonce(ccx, *account)
get_nonce(ccx, account)
}
}

Expand Down Expand Up @@ -350,7 +349,8 @@ impl Cheatcode for getBlobhashesCall {
impl Cheatcode for rollCall {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { newHeight } = self;
ccx.state.strategy.runner.clone().cheatcode_roll(ccx, *newHeight)
ccx.ecx.env.block.number = *newHeight;
Ok(Default::default())
}
}

Expand All @@ -372,7 +372,8 @@ impl Cheatcode for txGasPriceCall {
impl Cheatcode for warpCall {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { newTimestamp } = self;
ccx.state.strategy.runner.clone().cheatcode_warp(ccx, *newTimestamp)
ccx.ecx.env.block.timestamp = *newTimestamp;
Ok(Default::default())
}
}

Expand Down Expand Up @@ -407,38 +408,65 @@ impl Cheatcode for dealCall {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { account: address, newBalance: new_balance } = *self;

ccx.state.strategy.runner.clone().cheatcode_deal(ccx, address, new_balance)
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 };
ccx.state.eth_deals.push(record);
Ok(Default::default())
}
}

impl Cheatcode for etchCall {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { target, newRuntimeBytecode } = self;

ccx.state.strategy.runner.clone().cheatcode_etch(ccx, *target, newRuntimeBytecode)
ensure_not_precompile!(&target, ccx);
ccx.ecx.load_account(*target)?;
let bytecode = Bytecode::new_raw(Bytes::copy_from_slice(newRuntimeBytecode));
ccx.ecx.journaled_state.set_code(*target, bytecode);
Ok(Default::default())
}
}

impl Cheatcode for resetNonceCall {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { account } = self;
ccx.state.strategy.runner.clone().cheatcode_reset_nonce(ccx, *account)
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
// to avoid hitting the case where account's code is None.
let empty = account.info.code_hash == KECCAK_EMPTY;
let nonce = if empty { 0 } else { 1 };
account.info.nonce = nonce;
debug!(target: "cheatcodes", nonce, "reset");
Ok(Default::default())
}
}

impl Cheatcode for setNonceCall {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { account, newNonce } = *self;

ccx.state.strategy.runner.clone().cheatcode_set_nonce(ccx, account, newNonce)
let account = journaled_account(ccx.ecx, account)?;
// nonce must increment only
let current = account.info.nonce;
ensure!(
newNonce >= current,
"new nonce ({newNonce}) must be strictly equal to or higher than the \
account's current nonce ({current})"
);
account.info.nonce = newNonce;
Ok(Default::default())
}
}

impl Cheatcode for setNonceUnsafeCall {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { account, newNonce } = *self;

ccx.state.strategy.runner.clone().cheatcode_set_nonce_unsafe(ccx, account, newNonce)
let account = journaled_account(ccx.ecx, account)?;
account.info.nonce = newNonce;
Ok(Default::default())
}
}

Expand Down
9 changes: 7 additions & 2 deletions crates/cheatcodes/src/evm/mock.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,9 @@ impl Cheatcode for clearMockedCallsCall {
impl Cheatcode for mockCall_0Call {
fn apply_stateful(&self, ccx: &mut CheatsCtxt) -> Result {
let Self { callee, data, returnData } = self;
ccx.state.strategy.runner.clone().cheatcode_mock_call(ccx, *callee, data, returnData)
let _ = make_acc_non_empty(callee, ccx.ecx)?;
mock_call(ccx.state, callee, data, None, returnData, InstructionResult::Return);
Ok(Default::default())
}
}

Expand Down Expand Up @@ -83,7 +85,10 @@ 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.state.strategy.runner.clone().cheatcode_mock_call_revert(ccx, *callee, data, revertData)

let _ = make_acc_non_empty(callee, ccx.ecx)?;
mock_call(ccx.state, callee, data, None, revertData, InstructionResult::Revert);
Ok(Default::default())
}
}

Expand Down
2 changes: 1 addition & 1 deletion crates/cheatcodes/src/fs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -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.strategy.runner.get_artifact_code(state, path, false)
Ok(get_artifact_code(state, path, false)?.abi_encode())
}
}

Expand Down
3 changes: 2 additions & 1 deletion crates/cheatcodes/src/inspector.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2256,7 +2256,8 @@ fn apply_dispatch(
}

// Apply the cheatcode.
let mut result = cheat.dyn_apply(ccx, executor);
let runner = ccx.state.strategy.runner.new_cloned();
let mut result = runner.apply_full(cheat, ccx, executor);

// Format the error message to include the cheatcode name.
if let Err(e) = &mut result {
Expand Down
9 changes: 8 additions & 1 deletion crates/cheatcodes/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -96,12 +96,14 @@ pub(crate) trait Cheatcode: CheatcodeDef + DynCheatcode {
}
}

pub(crate) trait DynCheatcode: 'static {
pub trait DynCheatcode: 'static + std::any::Any {
fn cheatcode(&self) -> &'static spec::Cheatcode<'static>;

fn as_debug(&self) -> &dyn std::fmt::Debug;

fn dyn_apply(&self, ccx: &mut CheatsCtxt, executor: &mut dyn CheatcodesExecutor) -> Result;

fn as_any(&self) -> &dyn std::any::Any;
}

impl<T: Cheatcode> DynCheatcode for T {
Expand All @@ -119,6 +121,11 @@ impl<T: Cheatcode> DynCheatcode for T {
fn dyn_apply(&self, ccx: &mut CheatsCtxt, executor: &mut dyn CheatcodesExecutor) -> Result {
self.apply_full(ccx, executor)
}

#[inline]
fn as_any(&self) -> &dyn std::any::Any {
self
}
}

impl dyn DynCheatcode {
Expand Down
Loading

0 comments on commit 2a052c6

Please sign in to comment.