文件 1 的 7:Address.sol
pragma solidity ^0.8.0;
library Address {
function isContract(address account) internal view returns (bool) {
bytes32 codehash;
bytes32 accountHash = 0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470;
assembly { codehash := extcodehash(account) }
return (codehash != accountHash && codehash != 0x0);
}
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 functionCall(target, data, "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");
return _functionCallWithValue(target, data, value, errorMessage);
}
function _functionCallWithValue(address target, bytes memory data, uint256 weiValue, string memory errorMessage) private returns (bytes memory) {
require(isContract(target), "Address: call to non-contract");
(bool success, bytes memory returndata) = target.call{ value: weiValue }(data);
if (success) {
return returndata;
} else {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
文件 2 的 7: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 balanceOf(address account) external view returns (uint256);
function totalSupply() external view returns (uint256);
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);
function increaseAllowance(address spender, uint256 addedValue) external returns (bool);
function decreaseAllowance(address spender, uint256 subtractedValue) external returns (bool);
}
文件 3 的 7:IRewardsAirdropWithLock.sol
pragma solidity ^0.8.0;
interface IRewardsAirdropWithLock {
event Claimed(uint256 roundInd, uint256 merkleInd, address account, uint256 claimedAmount, uint256 amount);
event UpdatedPenaltyReceiver(address old, address _new);
event UpdatedRoundStatus(uint256 roundInd, bool oldDisabled, bool _newDisabled);
event AddedAirdrop(uint256 roundInd, address token, uint256 total);
struct AirdropRound {
address token;
bytes32 merkleRoot;
bool disabled;
uint256 startTime;
uint256 lockWindow;
uint256 lockRate;
uint256 total;
uint256 totalClaimed;
}
function penaltyReceiver() external view returns (address);
function claimWindow() external view returns (uint256);
function isClaimed(uint256 _roundsIndex, uint256 index) external view returns (bool);
function getAllAirdropRounds() external returns (AirdropRound[] memory);
function getAirdropRounds(uint256 _startInd, uint256 _endInd) external returns (AirdropRound[] memory);
function getAirdropRoundsLength() external returns (uint256);
function claim(
uint256 _roundsIndex,
uint256 _merkleIndex,
address account,
uint256 amount,
bytes32[] calldata merkleProof
) external;
function updatePaneltyReceiver(address _new) external;
function addAirdrop(
address _token,
bytes32 _merkleRoot,
uint256 _lockWindow,
uint256 _lockRate,
uint256 _total
) external returns (uint256);
function updateRoundStatus(uint256 _roundInd, bool _disabled) external;
}
文件 4 的 7:MerkleProof.sol
pragma solidity ^0.8.0;
library MerkleProof {
function verify(bytes32[] memory proof, bytes32 root, bytes32 leaf) internal pure returns (bool) {
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 == root;
}
}
文件 5 的 7:Ownable.sol
pragma solidity ^0.8.0;
contract Ownable {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor () {
_owner = msg.sender;
emit OwnershipTransferred(address(0), _owner);
}
function owner() public view returns (address) {
return _owner;
}
modifier onlyOwner() {
require(_owner == msg.sender, "Ownable: caller is not the owner");
_;
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
emit OwnershipTransferred(_owner, newOwner);
_owner = newOwner;
}
}
文件 6 的 7:RewardsAirdropWithLock.sol
pragma solidity ^0.8.0;
import "./utils/MerkleProof.sol";
import "./utils/Ownable.sol";
import "./utils/SafeERC20.sol";
import "./interfaces/IERC20.sol";
import "./interfaces/IRewardsAirdropWithLock.sol";
contract RewardsAirdropWithLock is IRewardsAirdropWithLock, Ownable {
using SafeERC20 for IERC20;
address public override penaltyReceiver;
uint256 public constant override claimWindow = 120 days;
uint256 public constant BASE = 1e18;
AirdropRound[] private airdropRounds;
mapping(uint256 => mapping(uint256 => uint256)) private claimedBitMaps;
modifier onlyNotDisabled(uint256 _roundInd) {
require(!airdropRounds[_roundInd].disabled, "RWL: Round disabled");
_;
}
constructor(address _penaltyReceiver) {
penaltyReceiver = _penaltyReceiver;
}
function updatePaneltyReceiver(address _new) external override onlyOwner {
require(_new != address(0), "RWL: penaltyReceiver is 0");
emit UpdatedPenaltyReceiver(penaltyReceiver, _new);
penaltyReceiver = _new;
}
function addAirdrop(
address _token,
bytes32 _merkleRoot,
uint256 _lockWindow,
uint256 _lockRate,
uint256 _total
) external override onlyOwner returns (uint256) {
require(_token != address(0), "RWL: token is 0");
require(_total > 0, "RWL: total is 0");
require(_merkleRoot.length > 0, "RWL: empty merkle");
IERC20(_token).safeTransferFrom(msg.sender, address(this), _total);
airdropRounds.push(AirdropRound(
_token,
_merkleRoot,
false,
block.timestamp,
_lockWindow,
_lockRate,
_total,
0
));
uint256 index = airdropRounds.length - 1;
emit AddedAirdrop(index, _token, _total);
return index;
}
function updateRoundStatus(uint256 _roundInd, bool _disabled) external override onlyOwner {
emit UpdatedRoundStatus(_roundInd, airdropRounds[_roundInd].disabled, _disabled);
airdropRounds[_roundInd].disabled = _disabled;
}
function claim(
uint256 _roundInd,
uint256 _merkleInd,
address account,
uint256 amount,
bytes32[] calldata merkleProof
) external override onlyNotDisabled(_roundInd) {
require(!isClaimed(_roundInd, _merkleInd), "RWL: Already claimed");
AirdropRound memory airdropRound = airdropRounds[_roundInd];
require(block.timestamp <= airdropRound.startTime + claimWindow, "RWL: Too late");
bytes32 node = keccak256(abi.encodePacked(_merkleInd, account, amount));
require(MerkleProof.verify(merkleProof, airdropRound.merkleRoot, node), "RWL: Invalid proof");
airdropRounds[_roundInd].totalClaimed = airdropRound.totalClaimed + amount;
_setClaimed(_roundInd, _merkleInd);
uint256 claimableAmount = amount;
if (block.timestamp < airdropRound.startTime + airdropRound.lockWindow) {
uint256 penalty = airdropRound.lockRate * amount / BASE;
IERC20(airdropRound.token).safeTransfer(penaltyReceiver, penalty);
claimableAmount -= penalty;
}
IERC20(airdropRound.token).safeTransfer(account, claimableAmount);
emit Claimed(_roundInd, _merkleInd, account, claimableAmount, amount);
}
function collectDust(uint256[] calldata _roundInds) external onlyOwner {
for (uint256 i = 0; i < _roundInds.length; i ++) {
AirdropRound memory airdropRound = airdropRounds[_roundInds[i]];
require(block.timestamp > airdropRound.startTime + claimWindow || airdropRound.disabled, "RWL: Not ready");
airdropRounds[_roundInds[i]].disabled = true;
uint256 toCollect = airdropRound.total - airdropRound.totalClaimed;
IERC20(airdropRound.token).safeTransfer(owner(), toCollect);
}
}
function isClaimed(uint256 _roundInd, uint256 _merkleInd) public view override returns (bool) {
uint256 claimedWordIndex = _merkleInd / 256;
uint256 claimedBitIndex = _merkleInd % 256;
uint256 claimedWord = claimedBitMaps[_roundInd][claimedWordIndex];
uint256 mask = (1 << claimedBitIndex);
return claimedWord & mask == mask;
}
function getAllAirdropRounds() external view override returns (AirdropRound[] memory) {
return airdropRounds;
}
function getAirdropRoundsLength() external view override returns (uint256) {
return airdropRounds.length;
}
function getAirdropRounds(uint256 _startInd, uint256 _endInd) external view override returns (AirdropRound[] memory) {
AirdropRound[] memory roundsResults = new AirdropRound[](_endInd - _startInd);
AirdropRound[] memory roundsCopy = airdropRounds;
uint256 resultInd;
for (uint256 i = _startInd; i < _endInd; i++) {
roundsResults[resultInd] = roundsCopy[i];
resultInd++;
}
return roundsResults;
}
function _setClaimed(uint256 _roundInd, uint256 _merkleInd) private {
uint256 claimedWordIndex = _merkleInd / 256;
uint256 claimedBitIndex = _merkleInd % 256;
claimedBitMaps[_roundInd][claimedWordIndex] = claimedBitMaps[_roundInd][claimedWordIndex] | (1 << claimedBitIndex);
}
}
文件 7 的 7:SafeERC20.sol
pragma solidity ^0.8.0;
import "../interfaces/IERC20.sol";
import "./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 {
uint256 newAllowance = token.allowance(address(this), spender) - value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
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");
}
}
}
{
"compilationTarget": {
"contracts/RewardsAirdropWithLock.sol": "RewardsAirdropWithLock"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_penaltyReceiver","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"roundInd","type":"uint256"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"total","type":"uint256"}],"name":"AddedAirdrop","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"roundInd","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"merkleInd","type":"uint256"},{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"uint256","name":"claimedAmount","type":"uint256"},{"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"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"old","type":"address"},{"indexed":false,"internalType":"address","name":"_new","type":"address"}],"name":"UpdatedPenaltyReceiver","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"roundInd","type":"uint256"},{"indexed":false,"internalType":"bool","name":"oldDisabled","type":"bool"},{"indexed":false,"internalType":"bool","name":"_newDisabled","type":"bool"}],"name":"UpdatedRoundStatus","type":"event"},{"inputs":[],"name":"BASE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"bytes32","name":"_merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"_lockWindow","type":"uint256"},{"internalType":"uint256","name":"_lockRate","type":"uint256"},{"internalType":"uint256","name":"_total","type":"uint256"}],"name":"addAirdrop","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_roundInd","type":"uint256"},{"internalType":"uint256","name":"_merkleInd","type":"uint256"},{"internalType":"address","name":"account","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"merkleProof","type":"bytes32[]"}],"name":"claim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"claimWindow","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_roundInds","type":"uint256[]"}],"name":"collectDust","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_startInd","type":"uint256"},{"internalType":"uint256","name":"_endInd","type":"uint256"}],"name":"getAirdropRounds","outputs":[{"components":[{"internalType":"address","name":"token","type":"address"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"bool","name":"disabled","type":"bool"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"lockWindow","type":"uint256"},{"internalType":"uint256","name":"lockRate","type":"uint256"},{"internalType":"uint256","name":"total","type":"uint256"},{"internalType":"uint256","name":"totalClaimed","type":"uint256"}],"internalType":"struct IRewardsAirdropWithLock.AirdropRound[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getAirdropRoundsLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getAllAirdropRounds","outputs":[{"components":[{"internalType":"address","name":"token","type":"address"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"bool","name":"disabled","type":"bool"},{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"lockWindow","type":"uint256"},{"internalType":"uint256","name":"lockRate","type":"uint256"},{"internalType":"uint256","name":"total","type":"uint256"},{"internalType":"uint256","name":"totalClaimed","type":"uint256"}],"internalType":"struct IRewardsAirdropWithLock.AirdropRound[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_roundInd","type":"uint256"},{"internalType":"uint256","name":"_merkleInd","type":"uint256"}],"name":"isClaimed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"penaltyReceiver","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_new","type":"address"}],"name":"updatePaneltyReceiver","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_roundInd","type":"uint256"},{"internalType":"bool","name":"_disabled","type":"bool"}],"name":"updateRoundStatus","outputs":[],"stateMutability":"nonpayable","type":"function"}]