编译器
0.8.20+commit.a1b79de6
文件 1 的 4: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;
}
}
文件 2 的 4:GasRewardsDistributor.sol
pragma solidity ^0.8.20;
import {Ownable} from "openzeppelin/access/Ownable.sol";
import {MerkleProofLib} from "solmate/utils/MerkleProofLib.sol";
contract GasRewardsDistributor is Ownable {
receive() external payable {}
mapping(bytes32 root => uint16 round) public rootsToRounds;
mapping(uint16 round => uint64 endTime) public roundEndTimes;
mapping(bytes32 packedRecipientAndRound => bool used) public claimedRewards;
event Claimed(address recipient, uint16 round, uint256 amount);
error RoundNotOpen();
error AlreadyClaimed();
error InvalidProof();
error TransferFailed();
error NotEnoughEtherInContract();
error InvalidRoundId();
constructor() {
if (msg.sender != tx.origin) {
transferOwnership(tx.origin);
}
}
function _packRecipientAndRound(address _address, uint16 _round) internal pure returns (bytes32) {
return (bytes32(uint256(uint160(_address))) << 96) | bytes32(uint256(_round));
}
function claim(bytes32 merkleRoot, uint256 amount, bytes32[] calldata proof) public payable {
if (amount > address(this).balance) revert NotEnoughEtherInContract();
uint16 round = rootsToRounds[merkleRoot];
if (round == 0) revert InvalidRoundId();
if (block.timestamp > roundEndTimes[round]) revert RoundNotOpen();
bytes32 packedRecipientAndRound = _packRecipientAndRound(msg.sender, round);
if (claimedRewards[packedRecipientAndRound]) revert AlreadyClaimed();
if (!MerkleProofLib.verify(proof, merkleRoot, keccak256(abi.encodePacked(msg.sender, amount, round)))) {
revert InvalidProof();
}
claimedRewards[packedRecipientAndRound] = true;
emit Claimed(msg.sender, round, amount);
(bool success,) = payable(msg.sender).call{value: amount}("");
if (!success) revert TransferFailed();
}
function setRoundEndTime(uint16 round, uint32 endTime) public onlyOwner {
if (round == 0) revert InvalidRoundId();
roundEndTimes[round] = endTime;
}
function setRoundMerkleRoot(uint16 round, bytes32 merkleRoot) public onlyOwner {
rootsToRounds[merkleRoot] = round;
}
function withdraw(uint256 amount) public onlyOwner {
(bool success,) = payable(msg.sender).call{value: amount}("");
if (!success) revert TransferFailed();
}
}
文件 3 的 4:MerkleProofLib.sol
pragma solidity >=0.8.0;
library MerkleProofLib {
function verify(
bytes32[] calldata proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool isValid) {
assembly {
if proof.length {
let end := add(proof.offset, shl(5, proof.length))
let offset := proof.offset
for {} 1 {} {
let leafSlot := shl(5, gt(leaf, calldataload(offset)))
mstore(leafSlot, leaf)
mstore(xor(leafSlot, 32), calldataload(offset))
leaf := keccak256(0, 64)
offset := add(offset, 32)
if iszero(lt(offset, end)) { break }
}
}
isValid := eq(leaf, root)
}
}
}
文件 4 的 4:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_transferOwnership(_msgSender());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
{
"compilationTarget": {
"src/Fundrop/GasRewardsDistributor.sol": "GasRewardsDistributor"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 250000
},
"remappings": [
":ERC721A/=lib/ERC721A/contracts/",
":ds-test/=lib/forge-std/lib/ds-test/src/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/",
":openzeppelin/=lib/openzeppelin-contracts/contracts/",
":solady/=lib/solady/src/",
":solmate/=lib/solmate/src/"
]
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AlreadyClaimed","type":"error"},{"inputs":[],"name":"InvalidProof","type":"error"},{"inputs":[],"name":"InvalidRoundId","type":"error"},{"inputs":[],"name":"NotEnoughEtherInContract","type":"error"},{"inputs":[],"name":"RoundNotOpen","type":"error"},{"inputs":[],"name":"TransferFailed","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"recipient","type":"address"},{"indexed":false,"internalType":"uint16","name":"round","type":"uint16"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Claimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"inputs":[{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"claim","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"packedRecipientAndRound","type":"bytes32"}],"name":"claimedRewards","outputs":[{"internalType":"bool","name":"used","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"root","type":"bytes32"}],"name":"rootsToRounds","outputs":[{"internalType":"uint16","name":"round","type":"uint16"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"round","type":"uint16"}],"name":"roundEndTimes","outputs":[{"internalType":"uint64","name":"endTime","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"round","type":"uint16"},{"internalType":"uint32","name":"endTime","type":"uint32"}],"name":"setRoundEndTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"round","type":"uint16"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"}],"name":"setRoundMerkleRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]