Foundry 2 Echidna

The purpose of this tool is to transform the foundry broadcast JSON file to an Etheno-like JSON file for seamless Foundry integration with Echidna.

Demo

Demo GIF

Installation

Make sure you have Rust installed.

Install from crates.io: cargo install foundry2echidna

Install from source:

shell git clone https://github.com/ChmielewskiKamil/foundry2echidna/foundry2echidna && cd foundry2echidna && cargo install --path .

How to use it

Once you have foundry2echidna installed, you are ready to transform broadcast files.

  1. In the root of your Foundry project, run the command foundry2echidna. By default, if no arguments were passed, the tool will look for the following:

    You can pass custom input and output paths like this:

    foundry2echidna --input-path path/to/broadcast.json --output-path path/to/init.json

    Or, for short: foundry2echidna -i path/to/broadcast.json -o path/to/init.json

  2. Seed Echidna with the generated init.json file. Add the following to your echidna_config.yaml:

  1. Update your EchidnaTest contract, just like you would be interacting with the contracts deployed on the blockchain.

```solidity // Get address of the Counter contract from the broadcast file counter = Counter(0x1234...);

// This works for contracts deployed by Factories and function calls as well counterFactory = CounterFactory(0x456...); anotherCounterDeployedByFactory = AnotherCounter(0x678...); ```

  1. Run Echidna.

Integrate foundry2echinda with your project

Here you can find dev documentation on docs.rs.

You can use the transform_broadcast function that takes two arguments:

to deserialize and serialize broadcast files.


Data Model (inner workings)

Etheno handles two main groups of events* (via EventSummaryPlugin):

ContractCreated event has the following fields:

FunctionCall event has the following fields:

*There is also block mined event, but it's not crucial for Echidna setup (?)

Foundry broadcast structure is more complicated than that, but we only care about a couple of fields. Since we want to transform the broadcast into this Etheno-like structure, the appropriate fields must be mapped together.

| Etheno field | Foundry field | | --- | --- | | event | transactions[i].transaction_type | | from | transactions[i].transaction.from| | to | transactions[i].transaction.to | | contract_address | transactions[i].contract_address| | gas_used | receipts[i].gas_used | | gas_price | receipts[i].effective_gas_price | | data | transactions[i].transaction.data | | value | transactions[i].transaction.value |