β Invariant Testing - Part 1
Setup
Fuzzing vs. Invariant Testing

Writing an invariant test
Writing invariant test for WETH
Last updated

Last updated
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.18;
import "forge-std/Test.sol";
import "forge-std/console.sol";
// Topics
// - Invariant
// - Difference between fuzz and invariant
// - Failing invariant
// - Passing invariant
// - Stats - runs, calls, reverts
contract InvariantIntro {
bool public flag;
function func_1() external {}
function func_2() external {}
function func_3() external {}
function func_4() external {}
function func_5() external {
flag = true;
}
}
contract IntroInvariantTest is Test {
InvariantIntro private target;
function setUp() public {
target = new InvariantIntro();
}
function invariant_flag_is_always_false() public {
assertEq(target.flag(), false);
}
} function invariant_flag_is_always_false() public {
assertEq(target.flag(), false);
}pragma solidity 0.8.18;
contract WETH {
string public name = "Wrapped Ether";
string public symbol = "WETH";
uint8 public decimals = 18;
event Approval(address indexed src, address indexed guy, uint256 wad);
event Transfer(address indexed src, address indexed dst, uint256 wad);
event Deposit(address indexed dst, uint256 wad);
event Withdrawal(address indexed src, uint256 wad);
mapping(address => uint256) public balanceOf;
mapping(address => mapping(address => uint256)) public allowance;
receive() external payable {
deposit();
}
function deposit() public payable {
balanceOf[msg.sender] += msg.value;
emit Deposit(msg.sender, msg.value);
}
function withdraw(uint256 wad) public {
balanceOf[msg.sender] -= wad;
payable(msg.sender).transfer(wad);
emit Withdrawal(msg.sender, wad);
}
function totalSupply() public view returns (uint256) {
return address(this).balance;
}
function approve(address guy, uint256 wad) public returns (bool) {
allowance[msg.sender][guy] = wad;
emit Approval(msg.sender, guy, wad);
return true;
}
function transfer(address dst, uint256 wad) public returns (bool) {
return transferFrom(msg.sender, dst, wad);
}
function transferFrom(address src, address dst, uint256 wad)
public
returns (bool)
{
if (
src != msg.sender && allowance[src][msg.sender] != type(uint256).max
) {
allowance[src][msg.sender] -= wad;
}
balanceOf[src] -= wad;
balanceOf[dst] += wad;
emit Transfer(src, dst, wad);
return true;
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.18;
import "forge-std/Test.sol";
import "forge-std/console.sol";
import {WETH} from "../../src/WETH.sol";
// https://book.getfoundry.sh/forge/invariant-testing?highlight=targetSelector#invariant-targets
// https://mirror.xyz/horsefacts.eth/Jex2YVaO65dda6zEyfM_-DXlXhOWCAoSpOx5PLocYgw
// NOTE: open testing - randomly call all public functions
contract WETH_Open_Invariant_Tests is Test {
WETH public weth;
function setUp() public {
weth = new WETH();
}
receive() external payable {}
// NOTE: - calls = runs x depth, (runs, calls, reverts)
function invariant_totalSupply_is_always_zero() public {
assertEq(0, weth.totalSupply());
}
}