From 1d0c63627e0ccdc70467e08b272ba1437128f9d5 Mon Sep 17 00:00:00 2001 From: Elpacos Date: Wed, 30 Oct 2024 14:19:32 +0100 Subject: [PATCH] version: 1.0.0 --- README.md | 131 ++++++++++++++++++++++++++++++++++++++++++++++ echidna_parser.py | 70 +++++++++++++++++++++++++ 2 files changed, 201 insertions(+) create mode 100755 echidna_parser.py diff --git a/README.md b/README.md index 4c0a357..3aa81cd 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,133 @@ # echidna-trace-parser + A parser that converts echidna call traces into foundry PoC tests + +## Echidna Trace Parser + +**Echidna Trace Parser** is a Python script designed to parse Echidna traces and generate corresponding Solidity test functions. This tool simplifies the process of converting Echidna output into usable Solidity tests, making it easier to verify smart contracts. + +## Features + +- **Parse Echidna Call Traces**: Extracts function calls, parameters, delays, and 'from' addresses from Echidna traces. +- **Generate Solidity Test Functions**: Automatically creates Solidity test functions based on the parsed data. +- **User-Friendly Input**: Allows users to paste Echidna traces directly into the terminal. + +## Prerequisites + +- Python 3.x +- Basic familiarity with the command line + +## Installation + +1. **Clone the repository**: + ```bash + git clone https://github.com/yourusername/echidna-trace-parser.git + cd echidna-trace-parser + ``` +2. **Download the Script**: Alternatively, download the script file directly and save it in a directory of your choice. + +3. **Ensure Python is Installed**: You can check if Python is installed by running: + + ```bash + python3 --version + ``` + + If it’s not installed, download and install it from [python.org](https://www.python.org). + +## Running the Script from Anywhere + +To run the Echidna Trace Parser from any directory in your terminal, follow these steps: + +### 1. Make the Script Executable + +If you're using macOS or Linux, you can make the script executable: + +- Open your terminal. +- Navigate to the directory where the script is located: + ```bash + cd /path/to/your/directory + ``` +- Run the following command to make the script executable: + ```bash + chmod +x echidna_parser.py + ``` + +### 2. Add the Script to Your PATH + +To run the script from anywhere, you can add its directory to your system's PATH. + +#### On macOS/Linux + +- Open your terminal. +- Edit your shell configuration file (like `~/.bashrc`, `~/.bash_profile`, or `~/.zshrc`) and add the following line, replacing `/path/to/your/directory` with the actual path where your script is located: + ```bash + export PATH="$PATH:/path/to/your/directory" + ``` +- Save the file and run: + ```bash + source ~/.bashrc + ``` + (or the corresponding file for your shell). + +#### On Windows + +1. Search for "Environment Variables" in the Start menu. +2. Click on "Edit the system environment variables". +3. In the System Properties window, click on "Environment Variables". +4. In the "System variables" section, find and select the `Path` variable, then click "Edit". +5. Click "New" and add the full path to the directory where your script is saved. +6. Click OK to save your changes. + +## Usage + +1. Open your terminal. +2. Run the script from anywhere: + ```bash + echidna_parser.py + ``` + or if you haven’t made it executable: + ```bash + python3 echidna_parser.py + ``` + +3. **Input your Echidna call trace**: + - Paste your Echidna call trace directly into the terminal. + - Press Enter twice to finish input. + +4. **View the Generated Solidity Test Function**: The tool will output the generated Solidity test function to the terminal. + +## Example + +```plaintext +Paste your Echidna call trace below. Press Enter twice to finish: +Tester.function_name(param1, param2) from: 0x12345 Time delay: 5 +*wait* Time delay: 3 + +Generated Foundry Test Function: + +function test_replay() public { + _setUpActor(0x12345); + _delay(5); + Tester.function_name(param1, param2); + _delay(3); +} +``` + +## Contributing + +Contributions are welcome! If you would like to contribute to this project, please follow these steps: + +1. Fork the repository. +2. Create a new branch for your feature or bug fix. +3. Make your changes and commit them. +4. Push your branch and submit a pull request. + +## License + +This project is licensed under the MIT License. See the LICENSE file for details. + +## Contact + +For any inquiries or feedback, feel free to reach out on the organization's Discord server. + +``` diff --git a/echidna_parser.py b/echidna_parser.py new file mode 100755 index 0000000..e0bb0d9 --- /dev/null +++ b/echidna_parser.py @@ -0,0 +1,70 @@ +#!/usr/bin/env python3 +import re + +def parse_echidna_trace(trace): + """Parse Echidna trace and extract function calls, parameters, delays, and 'from' addresses.""" + calls = [] + last_address = None # Track the last 'from' address + + for line in trace.strip().splitlines(): + # Parse function calls, including those with underscores + func_call = re.match(r"Tester\.([a-zA-Z0-9_]+)\(([^)]*)\)", line) + if func_call: + func_name = func_call.group(1) + params = func_call.group(2) + + # Check for 'from' address + from_address = re.search(r"from: (0x[a-fA-F0-9]+)", line) + time_delay = re.search(r"Time delay: (\d+)", line) + + # If we have a 'from' address, we need to set it up + if from_address: + address = from_address.group(1) + if address != last_address: + # Only set up the actor if the address has changed + calls.append(f"_setUpActor({address});") + last_address = address + + # Add the delay if it exists + if time_delay: + delay_time = time_delay.group(1) + calls.append(f"_delay({delay_time});") + + # Add the function call + calls.append(f"Tester.{func_name}({params});") + + # Handle special case of "*wait*" for a delay + elif "*wait*" in line: + time_delay = re.search(r"Time delay: (\d+)", line) + if time_delay: + delay_time = time_delay.group(1) + calls.append(f"_delay({delay_time});") + + return calls + +def generate_foundry_test(calls, test_name="test_replay"): + """Generate the Solidity test function code.""" + test_code = [f"function {test_name}() public {{"] + test_code.extend(f" {call}" for call in calls) + test_code.append("}") + + return "\n".join(test_code) + +# Ask user to paste the Echidna trace +print("Paste your Echidna call trace below. Press Enter twice to finish:") +trace = [] +while True: + line = input() + if line: + trace.append(line.strip()) + else: + break +trace = "\n".join(trace) + +# Parse the trace and generate the test +parsed_calls = parse_echidna_trace(trace) +solidity_test = generate_foundry_test(parsed_calls) + +# Output the generated Solidity test +print("\nGenerated Foundry Test Function:\n") +print(solidity_test)