文件 1 的 5: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 的 5:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
function totalSupply() external view returns (uint256);
function balanceOf(address account) 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);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
}
文件 3 的 5: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() {
_setOwner(_msgSender());
}
function owner() public view virtual returns (address) {
return _owner;
}
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() public virtual onlyOwner {
_setOwner(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_setOwner(newOwner);
}
function _setOwner(address newOwner) private {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 4 的 5:SafeMath.sol
pragma solidity ^0.8.0;
library SafeMath {
function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
uint256 c = a + b;
if (c < a) return (false, 0);
return (true, c);
}
}
function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b > a) return (false, 0);
return (true, a - b);
}
}
function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (a == 0) return (true, 0);
uint256 c = a * b;
if (c / a != b) return (false, 0);
return (true, c);
}
}
function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b == 0) return (false, 0);
return (true, a / b);
}
}
function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b == 0) return (false, 0);
return (true, a % b);
}
}
function add(uint256 a, uint256 b) internal pure returns (uint256) {
return a + b;
}
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
return a - b;
}
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
return a * b;
}
function div(uint256 a, uint256 b) internal pure returns (uint256) {
return a / b;
}
function mod(uint256 a, uint256 b) internal pure returns (uint256) {
return a % b;
}
function sub(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
unchecked {
require(b <= a, errorMessage);
return a - b;
}
}
function div(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
unchecked {
require(b > 0, errorMessage);
return a / b;
}
}
function mod(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
unchecked {
require(b > 0, errorMessage);
return a % b;
}
}
}
文件 5 的 5:Vesting.sol
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/utils/math/SafeMath.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
contract VestingPeriod is Ownable {
using SafeMath for uint256;
struct Grant {
uint256 amount;
uint256 totalClaimed;
uint256 perSecond;
}
struct Pool {
uint256 startTime;
uint256 endTime;
uint256 vestingDuration;
uint256 amount;
uint256 totalClaimed;
uint256 grants;
}
uint256 constant internal SECONDS_PER_DAY = 86400;
IERC20 public token;
mapping (address => Grant) public tokenGrants;
Pool public pool;
mapping(address => address) public blacklist;
event GrantAdded(address indexed recipient, uint256 indexed amount);
event GrantTokensClaimed(address indexed recipient, uint256 indexed amountClaimed);
event ChangeInvestor(address indexed oldOwner, address indexed newOwner);
constructor(address _token, uint256 _startTime, uint256 _vestingDuration) {
require(_token != address(0), "VestingPeriod::constructor: _token must be valid token address");
require(_startTime > 0 && _vestingDuration > 0, "VestingPeriod::constructor: One of the time parameters is 0");
require(_startTime > block.timestamp, "VestingPeriod::constructor: Starting time shalll be in a future time");
require(_vestingDuration > 0, "VestingPeriod::constructor: Duration of the period must be > 0");
if (_vestingDuration < SECONDS_PER_DAY) {
require(_vestingDuration <= SECONDS_PER_DAY.mul(10).mul(365), "VestingPeriod::constructor: Duration should be less than 10 years");
}
token = IERC20(_token);
pool.startTime = _startTime;
pool.vestingDuration = _vestingDuration;
pool.endTime = _startTime.add(_vestingDuration);
}
function changeInvestor(address _oldAddress, address _newAddress) external onlyOwner {
require(blacklist[_oldAddress] == address(0), "VestingPeriod::changeInvestor: oldaddress already in the blacklist");
require(blacklist[_newAddress] == address(0), "VestingPeriod::changeInvestor: new address is a blacklisted address");
require(tokenGrants[_newAddress].amount == 0, "VestingPeriod::changeInvestor: requires a different address than existing granted");
require(tokenGrants[_oldAddress].amount > 0, "VestingPeriod::changeInvestor: oldAddress has no remaining balance");
tokenGrants[_newAddress] = Grant(tokenGrants[_oldAddress].amount, tokenGrants[_oldAddress].totalClaimed, tokenGrants[_oldAddress].perSecond);
delete tokenGrants[_oldAddress];
blacklist[_oldAddress] = _newAddress;
emit ChangeInvestor(_oldAddress, _newAddress);
}
function addTokenGrants(address[] memory _recipients, uint256[] memory _amounts) external onlyOwner {
require(_recipients.length > 0, "VestingPeriod::addTokenGrants: no recipients");
require(_recipients.length <= 100, "VestingPeriod::addTokenGrants: too many grants, it will probably fail");
require(_recipients.length == _amounts.length, "VestingPeriod::addTokenGrants: invalid parameters length (they should be same)");
uint256 amountSum = 0;
for (uint16 i = 0; i < _recipients.length; i++) {
require(_recipients[i] != address(0), "VestingPeriod:addTokenGrants: there is an address with value 0");
require(tokenGrants[_recipients[i]].amount == 0, "VestingPeriod::addTokenGrants: a grant already exists for one of the accounts");
require(blacklist[_recipients[i]] == address(0), "VestingPeriod:addTOkenGrants: Blacklisted address");
require(_amounts[i] > 0, "VestingPeriod::addTokenGrant: amount == 0");
amountSum = amountSum.add(_amounts[i]);
}
require(token.transferFrom(msg.sender, address(this), amountSum), "VestingPeriod::addTokenGrants: transfer failed");
for (uint16 i = 0; i < _recipients.length; i++) {
Grant memory grant = Grant({
amount: _amounts[i],
totalClaimed: 0,
perSecond: _amounts[i].div(pool.vestingDuration)
});
tokenGrants[_recipients[i]] = grant;
emit GrantAdded(_recipients[i], _amounts[i]);
}
pool.amount = pool.amount.add(amountSum);
}
function getTokenGrant(address _recipient) external view returns (Grant memory) {
return tokenGrants[_recipient];
}
function calculateGrantClaim(address _recipient) public view returns (uint256) {
if (block.timestamp < pool.startTime) {
return 0;
}
uint256 cap = block.timestamp;
if (cap > pool.endTime) {
cap = pool.endTime;
}
uint256 elapsedTime = cap.sub(pool.startTime);
if (elapsedTime >= pool.vestingDuration) {
uint256 remainingGrant = tokenGrants[_recipient].amount.sub(tokenGrants[_recipient].totalClaimed);
return remainingGrant;
} else {
uint256 amountVested = tokenGrants[_recipient].perSecond.mul(elapsedTime);
uint256 claimableAmount = amountVested.sub(tokenGrants[_recipient].totalClaimed);
return claimableAmount;
}
}
function vestedBalance(address _recipient) external view returns (uint256) {
if (block.timestamp < pool.startTime) {
return 0;
}
uint256 cap = block.timestamp;
if (cap > pool.endTime) {
cap = pool.endTime;
}
if (cap == pool.endTime) {
return tokenGrants[_recipient].amount;
} else {
uint256 elapsedTime = cap.sub(pool.startTime);
uint256 amountVested = tokenGrants[_recipient].perSecond.mul(elapsedTime);
return amountVested;
}
}
function claimedBalance(address _recipient) external view returns (uint256) {
return tokenGrants[_recipient].totalClaimed;
}
function claimVestedTokens(address _recipient) external {
uint256 amountVested = calculateGrantClaim(_recipient);
require(amountVested > 0, "VestingPeriod::claimVestedTokens: amountVested is 0");
require(token.transfer(_recipient, amountVested), "VestingPeriod::claimVestedTokens: transfer failed");
Grant storage tokenGrant = tokenGrants[_recipient];
tokenGrant.totalClaimed = uint256(tokenGrant.totalClaimed.add(amountVested));
pool.totalClaimed = pool.totalClaimed.add(amountVested);
emit GrantTokensClaimed(_recipient, amountVested);
}
function tokensVestedPerDay(address _recipient) external view returns(uint256) {
return tokenGrants[_recipient].amount.div(pool.vestingDuration.div(SECONDS_PER_DAY));
}
function tokensVestedPerDay(uint256 _amount) external view returns(uint256) {
return _amount.div(pool.vestingDuration.div(SECONDS_PER_DAY));
}
}
{
"compilationTarget": {
"Vesting.sol": "VestingPeriod"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_startTime","type":"uint256"},{"internalType":"uint256","name":"_vestingDuration","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"oldOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"ChangeInvestor","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"recipient","type":"address"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"GrantAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"recipient","type":"address"},{"indexed":true,"internalType":"uint256","name":"amountClaimed","type":"uint256"}],"name":"GrantTokensClaimed","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":"address[]","name":"_recipients","type":"address[]"},{"internalType":"uint256[]","name":"_amounts","type":"uint256[]"}],"name":"addTokenGrants","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"blacklist","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_recipient","type":"address"}],"name":"calculateGrantClaim","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_oldAddress","type":"address"},{"internalType":"address","name":"_newAddress","type":"address"}],"name":"changeInvestor","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_recipient","type":"address"}],"name":"claimVestedTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_recipient","type":"address"}],"name":"claimedBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_recipient","type":"address"}],"name":"getTokenGrant","outputs":[{"components":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"totalClaimed","type":"uint256"},{"internalType":"uint256","name":"perSecond","type":"uint256"}],"internalType":"struct VestingPeriod.Grant","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pool","outputs":[{"internalType":"uint256","name":"startTime","type":"uint256"},{"internalType":"uint256","name":"endTime","type":"uint256"},{"internalType":"uint256","name":"vestingDuration","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"totalClaimed","type":"uint256"},{"internalType":"uint256","name":"grants","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"token","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"tokenGrants","outputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"totalClaimed","type":"uint256"},{"internalType":"uint256","name":"perSecond","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_recipient","type":"address"}],"name":"tokensVestedPerDay","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"tokensVestedPerDay","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_recipient","type":"address"}],"name":"vestedBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"}]