initialise
This commit is contained in:
commit
168ba92f3b
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
*.sol linguist-language=Solidity
|
3
.gitmodules
vendored
Normal file
3
.gitmodules
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
[submodule "lib/forge-std"]
|
||||||
|
path = lib/forge-std
|
||||||
|
url = https://github.com/foundry-rs/forge-std
|
8
README.md
Normal file
8
README.md
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
## Requirements
|
||||||
|
|
||||||
|
* NodeJS
|
||||||
|
* Foundryup
|
||||||
|
|
||||||
|
## Testing
|
||||||
|
|
||||||
|
`npm run test`
|
7
foundry.toml
Normal file
7
foundry.toml
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
[profile.default]
|
||||||
|
src = 'src'
|
||||||
|
out = 'out'
|
||||||
|
libs = ["node_modules", "lib"]
|
||||||
|
chain_id = 1
|
||||||
|
|
||||||
|
# See more config options https://github.com/foundry-rs/foundry/tree/master/config
|
15
package.json
Normal file
15
package.json
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"name": "relayer-penalty-proposal",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "",
|
||||||
|
"main": "index.js",
|
||||||
|
"directories": {
|
||||||
|
"lib": "lib",
|
||||||
|
"test": "test"
|
||||||
|
},
|
||||||
|
"scripts": {
|
||||||
|
"test": "forge test -vvvvv --fork-url https://api.securerpc.com/v1 --block-number 16216695"
|
||||||
|
},
|
||||||
|
"author": "",
|
||||||
|
"license": "ISC"
|
||||||
|
}
|
9
src/Proposal.sol
Normal file
9
src/Proposal.sol
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
pragma solidity 0.8.1;
|
||||||
|
|
||||||
|
import "./interfaces/IERC20.sol";
|
||||||
|
|
||||||
|
contract Proposal {
|
||||||
|
|
||||||
|
function executeProposal() external { }
|
||||||
|
|
||||||
|
}
|
20
src/interfaces/IGovernance.sol
Normal file
20
src/interfaces/IGovernance.sol
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
pragma solidity ^0.8.1;
|
||||||
|
|
||||||
|
interface IGovernance {
|
||||||
|
|
||||||
|
function propose(address target, string memory description) external returns (uint256);
|
||||||
|
|
||||||
|
function castVote(uint256 proposalId, bool support) external;
|
||||||
|
|
||||||
|
function execute(uint256 proposalId) external payable;
|
||||||
|
|
||||||
|
function lock(
|
||||||
|
address owner,
|
||||||
|
uint256 amount,
|
||||||
|
uint256 deadline,
|
||||||
|
uint8 v,
|
||||||
|
bytes32 r,
|
||||||
|
bytes32 s
|
||||||
|
) external;
|
||||||
|
|
||||||
|
}
|
32
src/proprietary/Mock.sol
Normal file
32
src/proprietary/Mock.sol
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
pragma solidity ^0.8.1;
|
||||||
|
|
||||||
|
contract Mock {
|
||||||
|
|
||||||
|
uint256 constant TEST_PRIVATE_KEY_ONE = 0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80;
|
||||||
|
uint256 constant TEST_PRIVATE_KEY_TWO = 0x59c6995e998f97a5a0044966f0945389dc9e86dae88c7a8412f4603b6b78690d;
|
||||||
|
address constant TEST_ADDRESS_ONE = 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266;
|
||||||
|
address constant TEST_ADDRESS_TWO = 0x70997970C51812dc3A010C7d01b50e0d17dc79C8;
|
||||||
|
|
||||||
|
uint256 constant PROPOSAL_DURATION = 7 days;
|
||||||
|
uint256 constant PROPOSAL_THRESHOLD = 25000 ether;
|
||||||
|
string constant PROPOSAL_DESCRIPTION = "Proposal #15: Renumeration and accounting";
|
||||||
|
|
||||||
|
address constant VERIFIER_ADDRESS = 0x77777FeDdddFfC19Ff86DB637967013e6C6A116C;
|
||||||
|
|
||||||
|
bytes32 constant public PERMIT_TYPEHASH = keccak256(
|
||||||
|
"Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)"
|
||||||
|
);
|
||||||
|
|
||||||
|
bytes32 constant EIP712_DOMAIN = keccak256(
|
||||||
|
abi.encode(
|
||||||
|
keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"),
|
||||||
|
keccak256(bytes("TornadoCash")),
|
||||||
|
keccak256(bytes("1")),
|
||||||
|
1,
|
||||||
|
VERIFIER_ADDRESS
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
uint16 constant PERMIT_FUNC_SELECTOR = uint16(0x1901);
|
||||||
|
|
||||||
|
}
|
3
src/proprietary/Parameters.sol
Normal file
3
src/proprietary/Parameters.sol
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
pragma solidity 0.8.1;
|
||||||
|
|
||||||
|
contract Parameters { }
|
92
test/Proposal.t.sol
Normal file
92
test/Proposal.t.sol
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
pragma solidity ^0.8.1;
|
||||||
|
|
||||||
|
import "../src/interfaces/IGovernance.sol";
|
||||||
|
import "../src/interfaces/IERC20.sol";
|
||||||
|
|
||||||
|
import "../src/proprietary/Parameters.sol";
|
||||||
|
import "../src/proprietary/Mock.sol";
|
||||||
|
|
||||||
|
import "../src/Proposal.sol";
|
||||||
|
import "forge-std/Test.sol";
|
||||||
|
|
||||||
|
contract ProposalTest is Test, Mock, Parameters {
|
||||||
|
|
||||||
|
function testProposal() public {
|
||||||
|
checkParameters();
|
||||||
|
|
||||||
|
uint256 proposalId = voteAndCreateProposal(address(new Proposal()));
|
||||||
|
|
||||||
|
IGovernance(governanceAddress).execute(proposalId);
|
||||||
|
}
|
||||||
|
|
||||||
|
function voteAndCreateProposal(address proposalAddress) public returns (uint256) {
|
||||||
|
retrieveAndLockBalance(TEST_PRIVATE_KEY_ONE, TEST_ADDRESS_ONE, PROPOSAL_THRESHOLD);
|
||||||
|
retrieveAndLockBalance(TEST_PRIVATE_KEY_TWO, TEST_ADDRESS_TWO, 1 ether);
|
||||||
|
|
||||||
|
/* ----------PROPOSER------------ */
|
||||||
|
vm.startPrank(TEST_ADDRESS_ONE);
|
||||||
|
|
||||||
|
uint256 proposalId = IGovernance(governanceAddress).propose(
|
||||||
|
proposalAddress,
|
||||||
|
PROPOSAL_DESCRIPTION
|
||||||
|
);
|
||||||
|
|
||||||
|
// TIME-TRAVEL
|
||||||
|
vm.warp(block.timestamp + 6 hours);
|
||||||
|
|
||||||
|
IGovernance(governanceAddress).castVote(proposalId, true);
|
||||||
|
|
||||||
|
vm.stopPrank();
|
||||||
|
/* ------------------------------ */
|
||||||
|
|
||||||
|
/* -------------VOTER-------------*/
|
||||||
|
vm.startPrank(TEST_ADDRESS_TWO);
|
||||||
|
IGovernance(governanceAddress).castVote(proposalId, true);
|
||||||
|
vm.stopPrank();
|
||||||
|
/* ------------------------------ */
|
||||||
|
|
||||||
|
// TIME-TRAVEL
|
||||||
|
vm.warp(block.timestamp + PROPOSAL_DURATION);
|
||||||
|
|
||||||
|
return proposalId;
|
||||||
|
}
|
||||||
|
|
||||||
|
function retrieveAndLockBalance(uint256 privateKey, address voter, uint256 amount) internal {
|
||||||
|
uint256 lockTimestamp = block.timestamp + PROPOSAL_DURATION;
|
||||||
|
bytes32 messageHash = keccak256(
|
||||||
|
abi.encodePacked(
|
||||||
|
PERMIT_FUNC_SELECTOR,
|
||||||
|
EIP712_DOMAIN,
|
||||||
|
keccak256(
|
||||||
|
abi.encode(
|
||||||
|
PERMIT_TYPEHASH,
|
||||||
|
voter,
|
||||||
|
governanceAddress,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
lockTimestamp
|
||||||
|
)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
/* ----------GOVERNANCE------- */
|
||||||
|
vm.startPrank(governanceAddress);
|
||||||
|
IERC20(tokenAddress).transfer(voter, amount);
|
||||||
|
vm.stopPrank();
|
||||||
|
/* ----------------------------*/
|
||||||
|
|
||||||
|
(uint8 v, bytes32 r, bytes32 s) = vm.sign(privateKey, messageHash);
|
||||||
|
|
||||||
|
/* ----------VOTER------------ */
|
||||||
|
vm.startPrank(voter);
|
||||||
|
IGovernance(governanceAddress).lock(voter, amount, lockTimestamp, v, r, s);
|
||||||
|
vm.stopPrank();
|
||||||
|
/* ----------------------------*/
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkParameters() internal {}
|
||||||
|
|
||||||
|
function checkResults() internal {}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user