Forge is a fast and flexible Ethereum testing framework, inspired by Dapp
If you are looking into how to consume the software as an end user, check the CLI README.
For more context on how the package works under the hood, look in the code docs.
Writing tests in Javascript/Typescript while writing your smart contracts in Solidity can be confusing. Forge lets you write your tests in Solidity, so you can focus on what matters.
contract Foo {
uint256 public x = 1;
function set(uint256 _x) external {
x = _x;
}
function double() external {
x = 2 * x;
}
}
contract FooTest {
Foo foo;
// The state of the contract gets reset before each
// test is run, with the `setUp()` function being called
// each time after deployment.
function setUp() public {
foo = new Foo();
}
// A simple unit test
function testDouble() public {
require(foo.x() == 1);
foo.double();
require(foo.x() == 2);
}
}
When testing smart contracts, fuzzing can uncover edge cases which would be hard to manually detect with manual unit testing. We support fuzzing natively, where any test function that takes >0 arguments will be fuzzed, using the proptest crate.
An example of how a fuzzed test would look like can be seen below:
function testDoubleWithFuzzing(uint256 x) public {
foo.set(x);
require(foo.x() == x);
foo.double();
require(foo.x() == 2 * x);
}
- test
- Simple unit tests
- Gas costs
- DappTools style test output
- JSON test output
- Matching on regex
- DSTest-style assertions support
- Fuzzing
- Symbolic execution
- Coverage
- HEVM-style Solidity cheatcodes
- Structured tracing with abi decoding
- Per-line gas profiling
- Forking mode
- Automatic solc selection
- Simple unit tests
- build
- Can read DappTools-style .sol.json artifacts
- Manual remappings
- Automatic remappings
- Multiple compiler versions
- Incremental compilation
- Can read Hardhat-style artifacts
- Can read Truffle-style artifacts
- install
- update
- debug
- CLI Tracing with
RUST_LOG=forge=trace
The below is modified from Dapp's README
We allow modifying blockchain state with "cheat codes". These can be accessed by
calling into a contract at address 0x7109709ECfa91a80626fF3989D68f67F5b1DD12D
,
which implements the following methods:
-
function warp(uint x) public
Sets the block timestamp tox
. -
function roll(uint x) public
Sets the block number tox
. -
function store(address c, bytes32 loc, bytes32 val) public
Sets the slotloc
of contractc
toval
. -
function load(address c, bytes32 loc) public returns (bytes32 val)
Reads the slotloc
of contractc
. -
function sign(uint sk, bytes32 digest) public returns (uint8 v, bytes32 r, bytes32 s)
Signs thedigest
using the private keysk
. Note that signatures produced viahevm.sign
will leak the private key. -
function addr(uint sk) public returns (address addr)
Derives an ethereum address from the private keysk
. Note thathevm.addr(0)
will fail withBadCheatCode
as0
is an invalid ECDSA private key. -
function ffi(string[] calldata) external returns (bytes memory)
Executes the arguments as a command in the system shell and returns stdout. Note that this cheatcode means test authors can execute arbitrary code on user machines as part of a call todapp test
, for this reason all calls toffi
will fail unless the--ffi
flag is passed. -
function deal(address who, uint256 amount)
: Sets an account's balance -
function etch(address where, bytes memory what)
:` Sets the contract code at some address contract code -
function prank(address from, address to, bytes calldata) (bool success,bytes retdata)
: Performs a smart contract call as another address
The below example uses the warp
cheatcode to override the timestamp:
interface Vm {
function warp(uint256 x) external;
}
contract MyTest {
Vm vm = Vm(0x7109709ECfa91a80626fF3989D68f67F5b1DD12D);
function testWarp() public {
vm.warp(100);
require(block.timestamp == 100);
}
Over the next months, we intend to add the following features which are available in upstream dapptools:
- Stack Traces: Currently we do not provide any debug information when a call fails. We intend to add a structured printer (something like this which will show all the calls, logs and arguments passed across intermediate smart contract calls, which should help with debugging.
- Invariant Tests
- Interactive Debugger
- Code coverage
- Gas snapshots
- Symbolic EVM
We also intend to add features which are not available in dapptools:
- Even faster tests with parallel EVM execution that produces state diffs instead of modifying the state
- Improved UX for assertions:
- Check revert error or reason on a Solidity call
- Check that an event was emitted with expected arguments
- Support more EVM backends (revm, geth's evm, hevm etc.) & benchmark performance across them
- Declarative deployment system based on a config file
- Formatting & Linting (maybe powered by
Solang)
dapp fmt
, an automatic code formatter according to standard rules (likeprettier-plugin-solidity
)dapp lint
, a linter + static analyzer, like a combination ofsolhint
and slither
- Flamegraphs for gas profiling