文件 1 的 6:Airdrop.sol
pragma solidity ^0.8.3;
import "../libraries/Authorizable.sol";
import "../libraries/MerkleRewards.sol";
contract Airdrop is MerkleRewards, Authorizable {
uint256 public immutable expiration;
constructor(
address _governance,
bytes32 _merkleRoot,
IERC20 _token,
uint256 _expiration,
ILockingVault _lockingVault
) MerkleRewards(_merkleRoot, _token, _lockingVault) {
expiration = _expiration;
setOwner(_governance);
}
function reclaim(address destination) external onlyOwner {
require(block.timestamp > expiration, "Not expired");
uint256 unclaimed = token.balanceOf(address(this));
token.transfer(destination, unclaimed);
}
function claim(
uint256 amount,
uint256 totalGrant,
bytes32[] calldata merkleProof,
address destination
) external virtual override {
revert("Not Allowed to claim");
}
}
文件 2 的 6:Authorizable.sol
pragma solidity >=0.7.0;
contract Authorizable {
address public owner;
mapping(address => bool) public authorized;
constructor() {
owner = msg.sender;
}
modifier onlyOwner() {
require(msg.sender == owner, "Sender not owner");
_;
}
modifier onlyAuthorized() {
require(isAuthorized(msg.sender), "Sender not Authorized");
_;
}
function isAuthorized(address who) public view returns (bool) {
return authorized[who];
}
function authorize(address who) external onlyOwner() {
_authorize(who);
}
function deauthorize(address who) external onlyOwner() {
authorized[who] = false;
}
function setOwner(address who) public onlyOwner() {
owner = who;
}
function _authorize(address who) internal {
authorized[who] = true;
}
}
文件 3 的 6:IERC20.sol
pragma solidity ^0.8.3;
interface IERC20 {
function symbol() external view returns (string memory);
function balanceOf(address account) external view returns (uint256);
function decimals() external view returns (uint8);
function transfer(address recipient, 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 sender,
address recipient,
uint256 amount
) external returns (bool);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
}
文件 4 的 6:ILockingVault.sol
pragma solidity ^0.8.3;
import "./IERC20.sol";
interface ILockingVault {
function deposit(
address fundedAccount,
uint256 amount,
address firstDelegation
) external;
function withdraw(uint256 amount) external;
function token() external returns (IERC20);
}
文件 5 的 6: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 processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
bytes32 proofElement = proof[i];
if (computedHash <= proofElement) {
computedHash = keccak256(abi.encodePacked(computedHash, proofElement));
} else {
computedHash = keccak256(abi.encodePacked(proofElement, computedHash));
}
}
return computedHash;
}
}
文件 6 的 6:MerkleRewards.sol
pragma solidity ^0.8.3;
import "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol";
import "../interfaces/IERC20.sol";
import "../interfaces/ILockingVault.sol";
abstract contract AbstractMerkleRewards {
bytes32 public rewardsRoot;
IERC20 public immutable token;
mapping(address => uint256) public claimed;
ILockingVault public lockingVault;
constructor(
bytes32 _rewardsRoot,
IERC20 _token,
ILockingVault _lockingVault
) {
rewardsRoot = _rewardsRoot;
token = _token;
lockingVault = _lockingVault;
_token.approve(address(lockingVault), type(uint256).max);
}
function claimAndDelegate(
uint256 amount,
address delegate,
uint256 totalGrant,
bytes32[] calldata merkleProof,
address destination
) external {
require(delegate != address(0), "Zero addr delegation");
_validateWithdraw(amount, totalGrant, merkleProof);
lockingVault.deposit(destination, amount, delegate);
}
function claim(
uint256 amount,
uint256 totalGrant,
bytes32[] calldata merkleProof,
address destination
) external virtual {
_validateWithdraw(amount, totalGrant, merkleProof);
token.transfer(destination, amount);
}
function _validateWithdraw(
uint256 amount,
uint256 totalGrant,
bytes32[] memory merkleProof
) internal {
bytes32 leafHash = keccak256(abi.encodePacked(msg.sender, totalGrant));
require(
MerkleProof.verify(merkleProof, rewardsRoot, leafHash),
"Invalid Proof"
);
require(claimed[msg.sender] + amount <= totalGrant, "Claimed too much");
claimed[msg.sender] += amount;
}
}
contract MerkleRewards is AbstractMerkleRewards {
constructor(
bytes32 _rewardsRoot,
IERC20 _token,
ILockingVault _lockingVault
) AbstractMerkleRewards(_rewardsRoot, _token, _lockingVault) {}
}
{
"compilationTarget": {
"contracts/council/features/Airdrop.sol": "Airdrop"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 10000
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_governance","type":"address"},{"internalType":"bytes32","name":"_merkleRoot","type":"bytes32"},{"internalType":"contract IERC20","name":"_token","type":"address"},{"internalType":"uint256","name":"_expiration","type":"uint256"},{"internalType":"contract ILockingVault","name":"_lockingVault","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"address","name":"who","type":"address"}],"name":"authorize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"authorized","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"totalGrant","type":"uint256"},{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"},{"internalType":"address","name":"destination","type":"address"}],"name":"claim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address","name":"delegate","type":"address"},{"internalType":"uint256","name":"totalGrant","type":"uint256"},{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"},{"internalType":"address","name":"destination","type":"address"}],"name":"claimAndDelegate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"claimed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"who","type":"address"}],"name":"deauthorize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"expiration","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"who","type":"address"}],"name":"isAuthorized","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lockingVault","outputs":[{"internalType":"contract ILockingVault","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"destination","type":"address"}],"name":"reclaim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"rewardsRoot","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"who","type":"address"}],"name":"setOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"token","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"}]