rln-contract/test/RLN.t.sol

224 lines
7.8 KiB
Solidity
Raw Normal View History

// SPDX-License-Identifier: Unlicense
pragma solidity ^0.8.15;
import "../contracts/PoseidonHasher.sol";
import "../contracts/Rln.sol";
2023-05-26 08:01:05 +00:00
import "./Verifier.sol";
import "forge-std/Test.sol";
2023-03-29 11:48:00 +00:00
import "forge-std/StdCheats.sol";
import "forge-std/console.sol";
2023-05-26 08:01:05 +00:00
contract RLNTest is Test {
2023-03-29 11:48:00 +00:00
using stdStorage for StdStorage;
RLN public rln;
2023-03-29 11:48:00 +00:00
PoseidonHasher public poseidon;
2023-05-26 08:01:05 +00:00
TrueVerifier public trueVerifier;
FalseVerifier public falseVerifier;
uint256 public constant MEMBERSHIP_DEPOSIT = 1000000000000000;
uint256 public constant DEPTH = 20;
uint256 public constant SET_SIZE = 1048576;
2023-05-26 08:01:05 +00:00
uint256[8] public zeroedProof = [0, 0, 0, 0, 0, 0, 0, 0];
/// @dev Setup the testing environment.
function setUp() public {
2023-03-29 11:48:00 +00:00
poseidon = new PoseidonHasher();
2023-05-26 08:01:05 +00:00
trueVerifier = new TrueVerifier();
falseVerifier = new FalseVerifier();
rln = new RLN(MEMBERSHIP_DEPOSIT, DEPTH, address(poseidon), address(trueVerifier));
}
/// @dev Ensure that you can hash a value.
function test__Constants() public {
assertEq(rln.MEMBERSHIP_DEPOSIT(), MEMBERSHIP_DEPOSIT);
assertEq(rln.DEPTH(), DEPTH);
assertEq(rln.SET_SIZE(), SET_SIZE);
}
function test__ValidRegistration(uint256 idCommitment) public {
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
assertEq(rln.members(idCommitment), 1);
}
function test__InvalidRegistration__DuplicateCommitment(
uint256 idCommitment
) public {
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
assertEq(rln.members(idCommitment), 1);
2023-03-30 12:00:54 +00:00
vm.expectRevert(DuplicateIdCommitment.selector);
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
}
function test__InvalidRegistration__InsufficientDeposit(
uint256 idCommitment
) public {
2023-03-30 12:00:54 +00:00
uint256 badDepositAmount = MEMBERSHIP_DEPOSIT - 1;
vm.expectRevert(
2023-03-30 12:00:54 +00:00
abi.encodeWithSelector(
InsufficientDeposit.selector,
MEMBERSHIP_DEPOSIT,
badDepositAmount
)
);
2023-03-30 12:00:54 +00:00
rln.register{value: badDepositAmount}(idCommitment);
}
function test__InvalidRegistration__FullSet(
uint256 idCommitmentSeed
) public {
vm.assume(idCommitmentSeed < 2 ** 255 - SET_SIZE);
RLN tempRln = new RLN(
MEMBERSHIP_DEPOSIT,
2,
2023-05-26 08:01:05 +00:00
address(rln.poseidonHasher()),
address(rln.verifier())
);
uint256 setSize = tempRln.SET_SIZE() - 1;
for (uint256 i = 0; i < setSize; i++) {
tempRln.register{value: MEMBERSHIP_DEPOSIT}(idCommitmentSeed + i);
}
assertEq(tempRln.idCommitmentIndex(), 4);
vm.expectRevert(FullTree.selector);
tempRln.register{value: MEMBERSHIP_DEPOSIT}(idCommitmentSeed + setSize);
}
2023-03-29 11:48:00 +00:00
2023-05-26 08:01:05 +00:00
function test__ValidSlash(uint256 idCommitment, address payable to) public {
2023-03-29 11:48:00 +00:00
// avoid precompiles, etc
2023-03-29 12:21:48 +00:00
// TODO: wrap both of these in a single function
2023-03-29 11:48:00 +00:00
assumePayable(to);
2023-03-29 12:21:48 +00:00
assumeNoPrecompiles(to);
2023-03-29 11:59:33 +00:00
vm.assume(to != address(0));
2023-03-29 11:48:00 +00:00
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
uint256 balanceBefore = to.balance;
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, to, zeroedProof);
2023-03-30 14:53:08 +00:00
vm.prank(to);
rln.withdraw();
2023-03-29 11:48:00 +00:00
assertEq(rln.stakedAmounts(idCommitment), 0);
assertEq(rln.members(idCommitment), 0);
2023-03-29 11:48:00 +00:00
assertEq(to.balance, balanceBefore + MEMBERSHIP_DEPOSIT);
}
2023-03-30 14:53:08 +00:00
function test__InvalidSlash__ToZeroAddress() public {
2023-05-26 08:01:05 +00:00
uint256 idCommitment = 9014214495641488759237505126948346942972912379615652741039992445865937985820;
2023-03-29 11:48:00 +00:00
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
2023-03-30 12:00:54 +00:00
vm.expectRevert(
2023-03-30 14:53:08 +00:00
abi.encodeWithSelector(InvalidReceiverAddress.selector, address(0))
2023-03-30 12:00:54 +00:00
);
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, payable(address(0)), zeroedProof);
2023-03-29 11:48:00 +00:00
}
2023-03-30 14:53:08 +00:00
function test__InvalidSlash__ToRlnAddress() public {
2023-05-26 08:01:05 +00:00
uint256 idCommitment = 19014214495641488759237505126948346942972912379615652741039992445865937985820;
2023-03-29 11:48:00 +00:00
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
2023-03-30 12:00:54 +00:00
vm.expectRevert(
abi.encodeWithSelector(
2023-03-30 14:53:08 +00:00
InvalidReceiverAddress.selector,
2023-03-30 12:00:54 +00:00
address(rln)
)
);
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, payable(address(rln)), zeroedProof);
2023-03-29 11:48:00 +00:00
}
2023-03-30 14:53:08 +00:00
function test__InvalidSlash__InvalidIdCommitment(
2023-05-26 08:01:05 +00:00
uint256 idCommitment
2023-03-29 11:48:00 +00:00
) public {
2023-03-30 12:00:54 +00:00
vm.expectRevert(
abi.encodeWithSelector(MemberNotRegistered.selector, idCommitment)
);
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, payable(address(this)), zeroedProof);
2023-03-29 11:48:00 +00:00
}
// this shouldn't be possible, but just in case
2023-03-30 14:53:08 +00:00
function test__InvalidSlash__NoStake(
2023-05-26 08:01:05 +00:00
uint256 idCommitment,
2023-03-29 11:48:00 +00:00
address payable to
) public {
// avoid precompiles, etc
assumePayable(to);
2023-03-29 12:21:48 +00:00
assumeNoPrecompiles(to);
2023-03-29 11:59:33 +00:00
vm.assume(to != address(0));
2023-03-29 11:48:00 +00:00
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, to, zeroedProof);
2023-03-29 11:48:00 +00:00
assertEq(rln.stakedAmounts(idCommitment), 0);
assertEq(rln.members(idCommitment), 0);
2023-03-29 11:48:00 +00:00
// manually set members[idCommitment] to true using vm
stdstore
.target(address(rln))
.sig("members(uint256)")
.with_key(idCommitment)
.depth(0)
.checked_write(true);
2023-03-30 12:00:54 +00:00
vm.expectRevert(
abi.encodeWithSelector(MemberHasNoStake.selector, idCommitment)
);
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, to, zeroedProof);
}
function test__InvalidSlash__InvalidProof() public {
uint256 idCommitment = 19014214495641488759237505126948346942972912379615652741039992445865937985820;
RLN tempRln = new RLN(
MEMBERSHIP_DEPOSIT,
2,
address(rln.poseidonHasher()),
address(falseVerifier)
);
tempRln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
vm.expectRevert(InvalidProof.selector);
tempRln.slash(idCommitment, payable(address(this)), zeroedProof);
2023-03-29 11:48:00 +00:00
}
2023-03-30 14:53:08 +00:00
function test__InvalidWithdraw__InsufficientWithdrawalBalance() public {
vm.expectRevert(InsufficientWithdrawalBalance.selector);
rln.withdraw();
}
function test__InvalidWithdraw__InsufficientContractBalance() public {
2023-05-26 08:01:05 +00:00
uint256 idCommitment = 19014214495641488759237505126948346942972912379615652741039992445865937985820;
2023-03-30 14:53:08 +00:00
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, payable(address(this)), zeroedProof);
2023-03-30 14:53:08 +00:00
assertEq(rln.stakedAmounts(idCommitment), 0);
assertEq(rln.members(idCommitment), 0);
2023-03-30 14:53:08 +00:00
vm.deal(address(rln), 0);
vm.expectRevert(InsufficientContractBalance.selector);
rln.withdraw();
}
function test__ValidWithdraw(address payable to) public {
assumePayable(to);
assumeNoPrecompiles(to);
2023-05-26 08:01:05 +00:00
uint256 idCommitment = 19014214495641488759237505126948346942972912379615652741039992445865937985820;
2023-03-30 14:53:08 +00:00
rln.register{value: MEMBERSHIP_DEPOSIT}(idCommitment);
assertEq(rln.stakedAmounts(idCommitment), MEMBERSHIP_DEPOSIT);
2023-05-26 08:01:05 +00:00
rln.slash(idCommitment, to, zeroedProof);
2023-03-30 14:53:08 +00:00
assertEq(rln.stakedAmounts(idCommitment), 0);
assertEq(rln.members(idCommitment), 0);
2023-03-30 14:53:08 +00:00
vm.prank(to);
rln.withdraw();
assertEq(rln.withdrawalBalance(to), 0);
}
}