编译器
0.8.21+commit.d9974bed
文件 1 的 13:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value
) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 2 的 13:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 3 的 13:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address to, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(
address from,
address to,
uint256 amount
) external returns (bool);
}
文件 4 的 13:MerkleProof.sol
pragma solidity ^0.8.0;
library MerkleProof {
function verify(
bytes32[] memory proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool) {
return processProof(proof, leaf) == root;
}
function verifyCalldata(
bytes32[] calldata proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool) {
return processProofCalldata(proof, leaf) == root;
}
function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
computedHash = _hashPair(computedHash, proof[i]);
}
return computedHash;
}
function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
computedHash = _hashPair(computedHash, proof[i]);
}
return computedHash;
}
function multiProofVerify(
bytes32[] memory proof,
bool[] memory proofFlags,
bytes32 root,
bytes32[] memory leaves
) internal pure returns (bool) {
return processMultiProof(proof, proofFlags, leaves) == root;
}
function multiProofVerifyCalldata(
bytes32[] calldata proof,
bool[] calldata proofFlags,
bytes32 root,
bytes32[] memory leaves
) internal pure returns (bool) {
return processMultiProofCalldata(proof, proofFlags, leaves) == root;
}
function processMultiProof(
bytes32[] memory proof,
bool[] memory proofFlags,
bytes32[] memory leaves
) internal pure returns (bytes32 merkleRoot) {
uint256 leavesLen = leaves.length;
uint256 totalHashes = proofFlags.length;
require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");
bytes32[] memory hashes = new bytes32[](totalHashes);
uint256 leafPos = 0;
uint256 hashPos = 0;
uint256 proofPos = 0;
for (uint256 i = 0; i < totalHashes; i++) {
bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
hashes[i] = _hashPair(a, b);
}
if (totalHashes > 0) {
return hashes[totalHashes - 1];
} else if (leavesLen > 0) {
return leaves[0];
} else {
return proof[0];
}
}
function processMultiProofCalldata(
bytes32[] calldata proof,
bool[] calldata proofFlags,
bytes32[] memory leaves
) internal pure returns (bytes32 merkleRoot) {
uint256 leavesLen = leaves.length;
uint256 totalHashes = proofFlags.length;
require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");
bytes32[] memory hashes = new bytes32[](totalHashes);
uint256 leafPos = 0;
uint256 hashPos = 0;
uint256 proofPos = 0;
for (uint256 i = 0; i < totalHashes; i++) {
bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
hashes[i] = _hashPair(a, b);
}
if (totalHashes > 0) {
return hashes[totalHashes - 1];
} else if (leavesLen > 0) {
return leaves[0];
} else {
return proof[0];
}
}
function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) {
return a < b ? _efficientHash(a, b) : _efficientHash(b, a);
}
function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) {
assembly {
mstore(0x00, a)
mstore(0x20, b)
value := keccak256(0x00, 0x40)
}
}
}
文件 5 的 13:Owned.sol
pragma solidity >=0.8.0;
abstract contract Owned {
event OwnershipTransferred(address indexed user, address indexed newOwner);
address public owner;
modifier onlyOwner() virtual {
require(msg.sender == owner, "UNAUTHORIZED");
_;
}
constructor(address _owner) {
owner = _owner;
emit OwnershipTransferred(address(0), _owner);
}
function transferOwnership(address newOwner) public virtual onlyOwner {
owner = newOwner;
emit OwnershipTransferred(msg.sender, newOwner);
}
}
文件 6 的 13:Pausable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Pausable is Context {
event Paused(address account);
event Unpaused(address account);
bool private _paused;
constructor() {
_paused = false;
}
modifier whenNotPaused() {
_requireNotPaused();
_;
}
modifier whenPaused() {
_requirePaused();
_;
}
function paused() public view virtual returns (bool) {
return _paused;
}
function _requireNotPaused() internal view virtual {
require(!paused(), "Pausable: paused");
}
function _requirePaused() internal view virtual {
require(paused(), "Pausable: not paused");
}
function _pause() internal virtual whenNotPaused {
_paused = true;
emit Paused(_msgSender());
}
function _unpause() internal virtual whenPaused {
_paused = false;
emit Unpaused(_msgSender());
}
}
文件 7 的 13:SafeERC20.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/draft-IERC20Permit.sol";
import "../../../utils/Address.sol";
library SafeERC20 {
using Address for address;
function safeTransfer(
IERC20 token,
address to,
uint256 value
) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(
IERC20 token,
address from,
address to,
uint256 value
) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
function safeApprove(
IERC20 token,
address spender,
uint256 value
) internal {
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(
IERC20 token,
address spender,
uint256 value
) internal {
uint256 newAllowance = token.allowance(address(this), spender) + value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
function safeDecreaseAllowance(
IERC20 token,
address spender,
uint256 value
) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
uint256 newAllowance = oldAllowance - value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
}
function safePermit(
IERC20Permit token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) {
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}
文件 8 的 13:draft-IERC20Permit.sol
pragma solidity ^0.8.0;
interface IERC20Permit {
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
function nonces(address owner) external view returns (uint256);
function DOMAIN_SEPARATOR() external view returns (bytes32);
}
文件 9 的 13:errors.sol
pragma solidity 0.8.21;
abstract contract Errors {
error Unauthorized();
error InvalidParams();
error InvalidCycle();
error InvalidProof();
error NothingToClaim();
error MsgSenderNotRecipient();
}
文件 10 的 13:events.sol
pragma solidity 0.8.21;
abstract contract Events {
event LogUpdateProposer(address proposer, bool isProposer);
event LogUpdateApprover(address approver, bool isApprover);
event LogRootProposed(uint256 cycle, bytes32 root, bytes32 contentHash, uint256 timestamp, uint256 blockNumber);
event LogRootUpdated(uint256 cycle, bytes32 root, bytes32 contentHash, uint256 timestamp, uint256 blockNumber);
event LogClaimed(
address user,
uint256 amount,
uint256 cycle,
uint8 positionType,
bytes32 positionId,
uint256 timestamp,
uint256 blockNumber
);
}
文件 11 的 13:main.sol
pragma solidity 0.8.21;
import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import { MerkleProof } from "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol";
import { Address } from "@openzeppelin/contracts/utils/Address.sol";
import { Structs } from "./structs.sol";
import { Variables } from "./variables.sol";
import { Events } from "./events.sol";
import { Errors } from "./errors.sol";
abstract contract FluidMerkleDistributorCore is Structs, Variables, Events, Errors {
modifier validAddress(address value_) {
if (value_ == address(0)) {
revert InvalidParams();
}
_;
}
}
abstract contract FluidMerkleDistributorAdmin is FluidMerkleDistributorCore {
function updateProposer(address proposer_, bool isProposer_) public onlyOwner validAddress(proposer_) {
_proposers[proposer_] = isProposer_;
emit LogUpdateProposer(proposer_, isProposer_);
}
function updateApprover(address approver_, bool isApprover_) public onlyOwner validAddress(approver_) {
_approvers[approver_] = isApprover_;
emit LogUpdateApprover(approver_, isApprover_);
}
function spell(address[] memory targets_, bytes[] memory calldatas_) public onlyOwner {
for (uint256 i = 0; i < targets_.length; i++) {
Address.functionDelegateCall(targets_[i], calldatas_[i]);
}
}
function pause() external onlyOwner {
_pause();
}
function unpause() external onlyOwner {
_unpause();
}
}
abstract contract FluidMerkleDistributorApprover is FluidMerkleDistributorCore {
modifier onlyApprover() {
if (!isApprover(msg.sender)) {
revert Unauthorized();
}
_;
}
function isApprover(address approver_) public view returns (bool) {
return (_approvers[approver_] || owner == approver_);
}
function approveRoot(
bytes32 root_,
bytes32 contentHash_,
uint40 cycle_,
uint40 startBlock_,
uint40 endBlock_
) external onlyApprover {
MerkleCycle memory merkleCycle_ = _pendingMerkleCycle;
if (
root_ != merkleCycle_.merkleRoot ||
contentHash_ != merkleCycle_.merkleContentHash ||
cycle_ != merkleCycle_.cycle ||
startBlock_ != merkleCycle_.startBlock ||
endBlock_ != merkleCycle_.endBlock
) {
revert InvalidParams();
}
previousMerkleRoot = _currentMerkleCycle.merkleRoot;
merkleCycle_.timestamp = uint40(block.timestamp);
merkleCycle_.publishBlock = uint40(block.number);
_currentMerkleCycle = merkleCycle_;
emit LogRootUpdated(cycle_, root_, contentHash_, block.timestamp, block.number);
}
}
abstract contract FluidMerkleDistributorProposer is FluidMerkleDistributorCore {
modifier onlyProposer() {
if (!isProposer(msg.sender)) {
revert Unauthorized();
}
_;
}
function isProposer(address proposer_) public view returns (bool) {
return (_proposers[proposer_] || owner == proposer_);
}
function proposeRoot(
bytes32 root_,
bytes32 contentHash_,
uint40 cycle_,
uint40 startBlock_,
uint40 endBlock_
) external whenNotPaused onlyProposer {
if (cycle_ != _currentMerkleCycle.cycle + 1 || startBlock_ > endBlock_) {
revert InvalidParams();
}
_pendingMerkleCycle = MerkleCycle({
merkleRoot: root_,
merkleContentHash: contentHash_,
cycle: cycle_,
startBlock: startBlock_,
endBlock: endBlock_,
timestamp: uint40(block.timestamp),
publishBlock: uint40(block.number)
});
emit LogRootProposed(cycle_, root_, contentHash_, block.timestamp, block.number);
}
}
contract FluidMerkleDistributor is
FluidMerkleDistributorCore,
FluidMerkleDistributorAdmin,
FluidMerkleDistributorApprover,
FluidMerkleDistributorProposer
{
constructor(
string memory name_,
address owner_,
address proposer_,
address approver_,
address rewardToken_
)
validAddress(owner_)
validAddress(proposer_)
validAddress(approver_)
validAddress(rewardToken_)
Variables(owner_, rewardToken_)
{
name = name_;
_proposers[proposer_] = true;
emit LogUpdateProposer(proposer_, true);
_approvers[approver_] = true;
emit LogUpdateApprover(approver_, true);
}
function hasPendingRoot() external view returns (bool) {
return _pendingMerkleCycle.cycle == _currentMerkleCycle.cycle + 1;
}
function currentMerkleCycle() public view returns (MerkleCycle memory) {
return _currentMerkleCycle;
}
function pendingMerkleCycle() public view returns (MerkleCycle memory) {
return _pendingMerkleCycle;
}
function encodeClaim(
address recipient_,
uint256 cumulativeAmount_,
uint8 positionType_,
bytes32 positionId_,
uint256 cycle_,
bytes memory metadata_
) public pure returns (bytes memory encoded_, bytes32 hash_) {
encoded_ = abi.encode(positionType_, positionId_, recipient_, cycle_, cumulativeAmount_, metadata_);
hash_ = keccak256(bytes.concat(keccak256(encoded_)));
}
function claim(
address recipient_,
uint256 cumulativeAmount_,
uint8 positionType_,
bytes32 positionId_,
uint256 cycle_,
bytes32[] calldata merkleProof_,
bytes memory metadata_
) external whenNotPaused {
if(msg.sender != recipient_) revert MsgSenderNotRecipient();
uint256 currentCycle_ = uint256(_currentMerkleCycle.cycle);
if (!(cycle_ == currentCycle_ || (currentCycle_ > 0 && cycle_ == currentCycle_ - 1))) {
revert InvalidCycle();
}
bytes32 node_ = keccak256(
bytes.concat(keccak256(abi.encode(positionType_, positionId_, recipient_, cycle_, cumulativeAmount_, metadata_)))
);
if (
!MerkleProof.verify(
merkleProof_,
cycle_ == currentCycle_ ? _currentMerkleCycle.merkleRoot : previousMerkleRoot,
node_
)
) {
revert InvalidProof();
}
uint256 claimable_ = cumulativeAmount_ - claimed[recipient_][positionId_];
if (claimable_ == 0) {
revert NothingToClaim();
}
claimed[recipient_][positionId_] = cumulativeAmount_;
SafeERC20.safeTransfer(TOKEN, recipient_, claimable_);
emit LogClaimed(recipient_, claimable_, cycle_, positionType_, positionId_, block.timestamp, block.number);
}
}
文件 12 的 13:structs.sol
pragma solidity 0.8.21;
abstract contract Structs {
struct MerkleCycle {
bytes32 merkleRoot;
bytes32 merkleContentHash;
uint40 cycle;
uint40 timestamp;
uint40 publishBlock;
uint40 startBlock;
uint40 endBlock;
}
}
文件 13 的 13:variables.sol
pragma solidity 0.8.21;
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { Owned } from "solmate/src/auth/Owned.sol";
import { Pausable } from "@openzeppelin/contracts/security/Pausable.sol";
import { Structs } from "./structs.sol";
abstract contract Constants {
IERC20 public immutable TOKEN;
constructor(address rewardToken_) {
TOKEN = IERC20(rewardToken_);
}
}
abstract contract Variables is Owned, Pausable, Constants, Structs {
string public name;
mapping(address => bool) internal _proposers;
mapping(address => bool) internal _approvers;
MerkleCycle internal _currentMerkleCycle;
MerkleCycle internal _pendingMerkleCycle;
bytes32 public previousMerkleRoot;
mapping(address => mapping(bytes32 => uint256)) public claimed;
constructor(address owner_, address rewardToken_) Constants(rewardToken_) Owned(owner_) {}
}
{
"compilationTarget": {
"contracts/protocols/lending/merkleDistributor/main.sol": "FluidMerkleDistributor"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs",
"useLiteralContent": true
},
"optimizer": {
"enabled": true,
"runs": 10000000
},
"remappings": []
}
[{"inputs":[{"internalType":"string","name":"name_","type":"string"},{"internalType":"address","name":"owner_","type":"address"},{"internalType":"address","name":"proposer_","type":"address"},{"internalType":"address","name":"approver_","type":"address"},{"internalType":"address","name":"rewardToken_","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"InvalidCycle","type":"error"},{"inputs":[],"name":"InvalidParams","type":"error"},{"inputs":[],"name":"InvalidProof","type":"error"},{"inputs":[],"name":"MsgSenderNotRecipient","type":"error"},{"inputs":[],"name":"NothingToClaim","type":"error"},{"inputs":[],"name":"Unauthorized","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"cycle","type":"uint256"},{"indexed":false,"internalType":"uint8","name":"positionType","type":"uint8"},{"indexed":false,"internalType":"bytes32","name":"positionId","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"timestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"LogClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"cycle","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"root","type":"bytes32"},{"indexed":false,"internalType":"bytes32","name":"contentHash","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"timestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"LogRootProposed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"cycle","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"root","type":"bytes32"},{"indexed":false,"internalType":"bytes32","name":"contentHash","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"timestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"LogRootUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"approver","type":"address"},{"indexed":false,"internalType":"bool","name":"isApprover","type":"bool"}],"name":"LogUpdateApprover","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"proposer","type":"address"},{"indexed":false,"internalType":"bool","name":"isProposer","type":"bool"}],"name":"LogUpdateProposer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"inputs":[],"name":"TOKEN","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"root_","type":"bytes32"},{"internalType":"bytes32","name":"contentHash_","type":"bytes32"},{"internalType":"uint40","name":"cycle_","type":"uint40"},{"internalType":"uint40","name":"startBlock_","type":"uint40"},{"internalType":"uint40","name":"endBlock_","type":"uint40"}],"name":"approveRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient_","type":"address"},{"internalType":"uint256","name":"cumulativeAmount_","type":"uint256"},{"internalType":"uint8","name":"positionType_","type":"uint8"},{"internalType":"bytes32","name":"positionId_","type":"bytes32"},{"internalType":"uint256","name":"cycle_","type":"uint256"},{"internalType":"bytes32[]","name":"merkleProof_","type":"bytes32[]"},{"internalType":"bytes","name":"metadata_","type":"bytes"}],"name":"claim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"bytes32","name":"","type":"bytes32"}],"name":"claimed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentMerkleCycle","outputs":[{"components":[{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"bytes32","name":"merkleContentHash","type":"bytes32"},{"internalType":"uint40","name":"cycle","type":"uint40"},{"internalType":"uint40","name":"timestamp","type":"uint40"},{"internalType":"uint40","name":"publishBlock","type":"uint40"},{"internalType":"uint40","name":"startBlock","type":"uint40"},{"internalType":"uint40","name":"endBlock","type":"uint40"}],"internalType":"struct Structs.MerkleCycle","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient_","type":"address"},{"internalType":"uint256","name":"cumulativeAmount_","type":"uint256"},{"internalType":"uint8","name":"positionType_","type":"uint8"},{"internalType":"bytes32","name":"positionId_","type":"bytes32"},{"internalType":"uint256","name":"cycle_","type":"uint256"},{"internalType":"bytes","name":"metadata_","type":"bytes"}],"name":"encodeClaim","outputs":[{"internalType":"bytes","name":"encoded_","type":"bytes"},{"internalType":"bytes32","name":"hash_","type":"bytes32"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"hasPendingRoot","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"approver_","type":"address"}],"name":"isApprover","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"proposer_","type":"address"}],"name":"isProposer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingMerkleCycle","outputs":[{"components":[{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"bytes32","name":"merkleContentHash","type":"bytes32"},{"internalType":"uint40","name":"cycle","type":"uint40"},{"internalType":"uint40","name":"timestamp","type":"uint40"},{"internalType":"uint40","name":"publishBlock","type":"uint40"},{"internalType":"uint40","name":"startBlock","type":"uint40"},{"internalType":"uint40","name":"endBlock","type":"uint40"}],"internalType":"struct Structs.MerkleCycle","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"previousMerkleRoot","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"root_","type":"bytes32"},{"internalType":"bytes32","name":"contentHash_","type":"bytes32"},{"internalType":"uint40","name":"cycle_","type":"uint40"},{"internalType":"uint40","name":"startBlock_","type":"uint40"},{"internalType":"uint40","name":"endBlock_","type":"uint40"}],"name":"proposeRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"targets_","type":"address[]"},{"internalType":"bytes[]","name":"calldatas_","type":"bytes[]"}],"name":"spell","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"approver_","type":"address"},{"internalType":"bool","name":"isApprover_","type":"bool"}],"name":"updateApprover","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"proposer_","type":"address"},{"internalType":"bool","name":"isProposer_","type":"bool"}],"name":"updateProposer","outputs":[],"stateMutability":"nonpayable","type":"function"}]