Skip to content

Provides a way of reading a real Starknet State, so you can re-execute an existing transaction in any of the Starknet networks in an easy way

License

Notifications You must be signed in to change notification settings

lambdaclass/starknet-replay

Repository files navigation

starknet-replay

Provides a way of reading a real Starknet State, so you can re-execute an existing transaction in any of the Starknet networks in an easy way

Getting Started

Prerequisites

  • Linux or macOS (aarch64 included) only for now
  • LLVM 19 with MLIR
  • Rust 1.78.0 or later, since cairo-native makes use of the u128 abi change.
  • Git

Setup

Run the following make target to install dependencies:

make deps

It will automatically install LLVM 19 with MLIR on macos, if you are using linux you must do it manually. On debian, you can use apt.llvm.org, or build it from source.

This project is integrated with Cairo Native, see Cairo Native Setup to set it up correctly

Some environment variable are needed, you can automatically set them by sourcing env.sh. If the script doesn't adjust to your specific environment you can cp it into .env or .envrc and modify it.

# Cairo Native
export LLVM_SYS_191_PREFIX=/path/to/llvm-19
export MLIR_SYS_190_PREFIX=/path/to/llvm-19
export TABLEGEN_190_PREFIX=/path/to/llvm-19
export CAIRO_NATIVE_RUNTIME_LIBRARY=/path/to/cairo_native/target/release/libcairo_native_runtime.a
# RPC
export RPC_ENDPOINT_MAINNET=rpc.endpoint.mainnet.com
export RPC_ENDPOINT_TESTNET=rpc.endpoint.testnet.com

Once you have installed dependencies and set the needed environment variables, you can build the project and run the tests:

make build
make test

Cairo Native Setup

Starknet Replay is currenlty integrated with Cairo Native, which makes the execution of sierra programs possible through native machine code. To use it, the following needs to be setup:

  • On mac with brew, running make deps should have installed LLVM 19 with MLIR, otherwise, you must install it manually. On Debian, you can use apt.llvm.org, or build it from source.

  • The LLVM_SYS_191_PREFIX, MLIR_SYS_190_PREFIX and TABLEGEN_190_PREFIX environment variable needs to point to said installation. In macOS, run:

    export LLVM_SYS_190_PREFIX=/opt/homebrew/opt/llvm@19
    export MLIR_SYS_191_PREFIX=/opt/homebrew/opt/llvm@19
    export TABLEGEN_190_PREFIX=/opt/homebrew/opt/llvm@19
    

    and you're set.

Afterwards, compiling with the feature flag cairo-native will enable native execution. You can check out some example test code that uses it under tests/cairo_native.rs.

Using ahead of time compilation with Native.

Currently cairo-native with AOT needs a runtime library in a known place. For this you need to compile the cairo-native-runtime crate and point the following environment variable to a folder containing the dynamic library. The path must be an absolute path.

CAIRO_NATIVE_RUNTIME_LIBRARY=/absolute/path/to/cairo-native/target/release/libcairo_native_runtime.a

If you don't do this you will get a linker error when using AOT.

replay

You can use the replay crate to execute transactions or blocks via the CLI. For example:

* cargo run tx 0x04ba569a40a866fd1cbb2f3d3ba37ef68fb91267a4931a377d6acc6e5a854f9a mainnet 648461
* cargo run block mainnet 648655
* cargo run block-range 90000 90002 mainnet

Important

Compiled contracts are cached to disk at compiled_programs directory. This saves time when reexecuting transactions, but can also cause errors if you try to run a contract that was compiled with a different Cairo Native version.

Make sure to remove the directory every time you update the Cairo Native version. Running make clean will automatically remove it.

Benchmarks

To run benchmarks with the replay crate, you can use either bench-block-range or bench-tx commands. These make sure to cache all needed information (including cairo native compilation) before the actual execution. To use it you must compile the binary under the benchmark flag.

* cargo run --features benchmark bench-tx 0x04ba569a40a866fd1cbb2f3d3ba37ef68fb91267a4931a377d6acc6e5a854f9a mainnet 648461 1
* cargo run --features benchmark bench-block-range 90000 90002 mainnet 1

These commands are like tx and block-range commands, but with the number of runs to execute as their last argument.

Logging

This projects uses tracing with env-filter, so logging can be modified by the RUST_LOG environment variable. By default, only info events from the replay crate are shown.

As an example, to show only error messages from the replay crate, run:

RUST_LOG=replay=error cargo run block mainnet 648461

Comparing with VM

To compare Native execution with the VM, you can use the state_dump feature. It will save to disk the execution info and state diff of every contract executed.

  • If executing Native, the dumps will be saved at: state_dumps/native/block{block_number}/{tx_hash}.json
  • If paired with only_cairo_vm feature, the dumps will be saved at: state_dumps/vm/block{block_number}/{tx_hash}.json

To compare the outputs, you can use the following scripts. Some of them required delta (modern diff).

  • cmp_state_dumps.sh. Prints which transactions match with the VM and which differ.
    > ./scripts/cmp_state_dumps.sh
    diff:  0x636326f93a16be14b36b7e62c546370d81d285d1f5398e13d5348fa03a00d05.json
    match: 0x6902da2a7ef7f7ab2e984c0cdfa94c535dedd7cc081c91f04b9f87a9805411b.json
    diff:  0x75ae71b0aaba9454965d2077d53f056ffd426481bad709831e8d76d50f32dbe.json
    match: 0x7895207d7d46df77f5b0de6b647cd393b9fc7bb18c52b6333c6ea852cf767e.json
    match: 0x2335142d7b7938eeb4512fbf59be7ec2f2284e6533c14baf51460c8de427dc7.json
    match: 0x26f6d10918250f16cddaebb8b69c5cececf9387d4a152f4d9197e1c03c40626.json
    
    Finished comparison
    - Matching: 4
    - Diffing:  16
  • delta_state_dumps.sh. It opens delta to review the differences between VM and Native with each transaction.
    > ./scripts/delta_state_dumps.sh

Plotting

In the plotting directory, you can find python scripts to plot relevant information. Before using them, you must first execute the replay with the structured_logging feature, and redirect the output to a file. You should do it with both Native execution and VM execution.

Make sure to erase the compiled_programs directory, then run:

./scripts/benchmark_tx.sh <tx> <net> <block> <laps>

This generates four files:

  • {native,vm}-data-$tx-$net.json: Contains the execution time of each contract call
  • {native,vm}-logs-$tx-$net.json: Contains the output of running the benchmark

If you want to benchmark a full block, you could run:

./scripts/benchmark_block.sh <block-start> <block-end> <net> <laps>

Once you have done this, you can use the plotting scripts:

  • python ./plotting/plot_execution_time.py native-data vm-data: Plots the execution time of Native vs VM, by contract class.
  • python ./plotting/plot_compilation_memory.py native-logs: Size of the compiled native libraries, by contract class.
  • python ./plotting/plot_compilation_memory_corr.py native-logs vm-logs: Size of the compiled native libraries, by the associated Casm contract size.
  • python ./plotting/plot_compilation_memory_trend.py native-logs vm-logs: Size of the compiled native and casm contracts, by the sierra contract size.
  • python ./plotting/plot_compilation_time.py native-logs: Native compilation time, by contract class
  • python ./plotting/plot_compilation_time_trend.py native-logs vm-logs: Native and Casm compilation time, by the sierra contract size.
  • python ./plotting/plot_compilation_time_finer.py native-logs: Native compilation time, with fine-grained stage separation, by contract class.

About

Provides a way of reading a real Starknet State, so you can re-execute an existing transaction in any of the Starknet networks in an easy way

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published