// SPDX-License-Identifier: Apache-2.0
pragma solidity ^0.7.0;
// File: contracts/Ownable.sol
/// @title Ownable
/// @author Brecht Devos - <brecht@loopring.org>
/// @dev The Ownable contract has an owner address, and provides basic
/// authorization control functions, this simplifies the implementation of
/// "user permissions".
contract Ownable
{
address public owner;
event OwnershipTransferred(
address indexed previousOwner,
address indexed newOwner
);
/// @dev The Ownable constructor sets the original `owner` of the contract
/// to the sender.
constructor()
{
owner = msg.sender;
}
/// @dev Throws if called by any account other than the owner.
modifier onlyOwner()
{
require(msg.sender == owner, "UNAUTHORIZED");
_;
}
/// @dev Allows the current owner to transfer control of the contract to a
/// new owner.
/// @param newOwner The address to transfer ownership to.
function transferOwnership(
address newOwner
)
public
virtual
onlyOwner
{
require(newOwner != address(0), "ZERO_ADDRESS");
emit OwnershipTransferred(owner, newOwner);
owner = newOwner;
}
function renounceOwnership()
public
onlyOwner
{
emit OwnershipTransferred(owner, address(0));
owner = address(0);
}
}
// File: contracts/Claimable.sol
/// @title Claimable
/// @author Brecht Devos - <brecht@loopring.org>
/// @dev Extension for the Ownable contract, where the ownership needs
/// to be claimed. This allows the new owner to accept the transfer.
contract Claimable is Ownable
{
address public pendingOwner;
/// @dev Modifier throws if called by any account other than the pendingOwner.
modifier onlyPendingOwner() {
require(msg.sender == pendingOwner, "UNAUTHORIZED");
_;
}
/// @dev Allows the current owner to set the pendingOwner address.
/// @param newOwner The address to transfer ownership to.
function transferOwnership(
address newOwner
)
public
override
onlyOwner
{
require(newOwner != address(0) && newOwner != owner, "INVALID_ADDRESS");
pendingOwner = newOwner;
}
/// @dev Allows the pendingOwner address to finalize the transfer.
function claimOwnership()
public
onlyPendingOwner
{
emit OwnershipTransferred(owner, pendingOwner);
owner = pendingOwner;
pendingOwner = address(0);
}
}
// File: contracts/ERC20.sol
/// @title ERC20 Token Interface
/// @dev see https://github.com/ethereum/EIPs/issues/20
/// @author Daniel Wang - <daniel@loopring.org>
abstract contract ERC20
{
function totalSupply()
public
view
virtual
returns (uint);
function balanceOf(
address who
)
public
view
virtual
returns (uint);
function allowance(
address owner,
address spender
)
public
view
virtual
returns (uint);
function transfer(
address to,
uint value
)
public
virtual
returns (bool);
function transferFrom(
address from,
address to,
uint value
)
public
virtual
returns (bool);
function approve(
address spender,
uint value
)
public
virtual
returns (bool);
}
// File: contracts/MathUint.sol
/// @title Utility Functions for uint
/// @author Daniel Wang - <daniel@loopring.org>
library MathUint
{
function mul(
uint a,
uint b
)
internal
pure
returns (uint c)
{
c = a * b;
require(a == 0 || c / a == b, "MUL_OVERFLOW");
}
function sub(
uint a,
uint b
)
internal
pure
returns (uint)
{
require(b <= a, "SUB_UNDERFLOW");
return a - b;
}
function add(
uint a,
uint b
)
internal
pure
returns (uint c)
{
c = a + b;
require(c >= a, "ADD_OVERFLOW");
}
}
// File: contracts/BaseTokenOwnershipPlan.sol
/// @title EmployeeTokenOwnershipPlan
/// @author Freeman Zhong - <kongliang@loopring.org>
/// added at 2021-02-19
abstract contract BaseTokenOwnershipPlan is Claimable
{
using MathUint for uint;
struct Record {
uint lastWithdrawTime;
uint rewarded;
uint withdrawn;
}
uint public constant vestPeriod = 2 * 365 days;
address public constant lrcAddress = 0xBBbbCA6A901c926F240b89EacB641d8Aec7AEafD;
uint public totalReward;
uint public vestStart;
mapping (address => Record) public records;
event Withdrawal(
address indexed transactor,
address indexed member,
uint amount
);
event MemberAddressChanged(
address oldAddress,
address newAddress
);
function withdrawFor(address recipient)
external
{
_withdraw(recipient);
}
function updateRecipient(address oldRecipient, address newRecipient)
external
{
require(canChangeAddressFor(oldRecipient), "UNAUTHORIZED");
require(newRecipient != address(0), "INVALID_ADDRESS");
require(records[newRecipient].rewarded == 0, "INVALID_NEW_RECIPIENT");
Record storage r = records[oldRecipient];
require(r.rewarded > 0, "INVALID_OLD_RECIPIENT");
records[newRecipient] = r;
delete records[oldRecipient];
emit MemberAddressChanged(oldRecipient, newRecipient);
}
function vested(address recipient)
public
view
returns(uint)
{
if (block.timestamp.sub(vestStart) < vestPeriod) {
return records[recipient].rewarded.mul(block.timestamp.sub(vestStart)) / vestPeriod;
} else {
return records[recipient].rewarded;
}
}
function withdrawable(address recipient)
public
view
returns(uint)
{
return vested(recipient).sub(records[recipient].withdrawn);
}
function _withdraw(address recipient)
internal
{
uint amount = withdrawable(recipient);
require(amount > 0, "INVALID_AMOUNT");
Record storage r = records[recipient];
r.lastWithdrawTime = block.timestamp;
r.withdrawn = r.withdrawn.add(amount);
require(ERC20(lrcAddress).transfer(recipient, amount), "transfer failed");
emit Withdrawal(msg.sender, recipient, amount);
}
receive() external payable {
require(msg.value == 0, "INVALID_VALUE");
_withdraw(msg.sender);
}
function collect()
external
onlyOwner
{
require(block.timestamp > vestStart + vestPeriod + 60 days, "TOO_EARLY");
uint amount = ERC20(lrcAddress).balanceOf(address(this));
require(ERC20(lrcAddress).transfer(msg.sender, amount), "transfer failed");
}
function canChangeAddressFor(address who)
internal
view
virtual
returns (bool);
}
// File: contracts/CancellableEmployeeTokenOwnershipPlan.sol
/// @title EmployeeTokenOwnershipPlan2
/// added at 2021-02-14
/// @author Freeman Zhong - <kongliang@loopring.org>
contract CancellableEmployeeTokenOwnershipPlan is BaseTokenOwnershipPlan
{
using MathUint for uint;
constructor()
{
owner = 0x96f16FdB8Cd37C02DEeb7025C1C7618E1bB34d97;
address payable[35] memory _members = [
0xFF6f7B2afdd33671503705098dd3c4c26a0F0705,
0xf493af7DFd0e47869Aac4770B2221a259CA77Ac8,
0xf21e66578372Ea62BCb0D1cDfC070f231CF56898,
0xEBE85822e75D2B4716e228818B54154E4AfFD202,
0xeB4c50dF06cEb2Ea700ea127eA589A99a3aAe1Ec,
0xe0807d8E14F2BCbF3Cc58637259CCF3fDd1D3ce5,
0xD984D096B4bF9DCF5fd75D9cBaf052D00EBe74c4,
0xd3725C997B580E36707f73880aC006B6757b5009,
0xBc5F996840118B580C4452440351b601862c5672,
0xad05c57e06a80b8EC92383b3e10Fea0E2b4e571D,
0xa26cFCeCb07e401547be07eEe26E6FD608f77d1a,
0x933650184994CFce9D64A9F3Ed14F1Fd017fF89A,
0x813C12326A0E8C2aC91d584f025E50072CDb4467,
0x7F81D533B2ea31BE2591d89394ADD9A12499ff17,
0x7F6Dd0c1BeB26CFf8ABA5B020E78D7C0Ed54B8Cc,
0x7b3B1F252169Ff83E3E91106230c36bE672aFdE3,
0x7809D08edBBBC401c430e5D3862a1Fdfcb8094A2,
0x7154a02BA6eEaB9300D056e25f3EEA3481680f87,
0x650EACf9AD1576680f1af6eC6cC598A484d796Ad,
0x5a092E52B9b3109372B9905Fa5c0655417F0f1a5,
0x5a03a928b332EC269f68684A8e9c1881b4Da5f3d,
0x55634e271BCa62dDFb9B5f7eae19f3Ae94Fb96b7,
0x4c381276F4847255C675Eab90c3409FA2fce68bC,
0x4bA63ac57b45087d03Abfd8E98987705Fa56B1ab,
0x474A2F53D11c73Ef2343322d69dCAE93cd63Dd9e,
0x41cDd7034AD6b2a5d24397189802048E97b6532D,
0x33CDbeB3e060bf6973e28492BE3D469C05D32786,
0x2a791a837D70E6D6e35073Dd61a9Af878Ac231A5,
0x2234C96681E9533FDfD122baCBBc634EfbafA0F0,
0x21870650F40Fe8249DECc96525249a43829E9A32,
0x1F28F10176F89F4E9985873B84d14e75751BB3D1,
0x11a8632b5089c6a061760F0b03285e2cC1388E36,
0x10Bd72a6AfbF8860ec90f7aeCdB8e937a758f351,
0x07A7191de1BA70dBe875F12e744B020416a5712b,
0x067eceAd820BC54805A2412B06946b184d11CB4b
];
uint80[35] memory _amounts = [
187520 ether,
256002 ether,
538180 ether,
340060 ether,
289314 ether,
176782 ether,
308310 ether,
398740 ether,
31254 ether,
82284 ether,
435961 ether,
459366 ether,
453078 ether,
150296 ether,
500972 ether,
375040 ether,
283528 ether,
155765 ether,
316840 ether,
38873 ether,
89970 ether,
549381 ether,
150834 ether,
501058 ether,
77746 ether,
145641 ether,
173213 ether,
573800 ether,
539572 ether,
110258 ether,
58022 ether,
398740 ether,
561054 ether,
221724 ether,
485991 ether
];
uint _totalReward = 10415169 ether;
vestStart = block.timestamp;
for (uint i = 0; i < _members.length; i++) {
require(records[_members[i]].rewarded == 0, "DUPLICATED_MEMBER");
Record memory record = Record(block.timestamp, _amounts[i], 0);
records[_members[i]] = record;
totalReward = totalReward.add(_amounts[i]);
}
require(_totalReward == totalReward, "VALUE_MISMATCH");
}
function canChangeAddressFor(address who)
internal
view
override
returns (bool) {
return msg.sender == who || msg.sender == owner;
}
}
{
"compilationTarget": {
"CancellableEmployeeTokenOwnershipPlan.sol": "CancellableEmployeeTokenOwnershipPlan"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 999999
},
"remappings": []
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"oldAddress","type":"address"},{"indexed":false,"internalType":"address","name":"newAddress","type":"address"}],"name":"MemberAddressChanged","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":true,"internalType":"address","name":"transactor","type":"address"},{"indexed":true,"internalType":"address","name":"member","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Withdrawal","type":"event"},{"inputs":[],"name":"claimOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"collect","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"lrcAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"records","outputs":[{"internalType":"uint256","name":"lastWithdrawTime","type":"uint256"},{"internalType":"uint256","name":"rewarded","type":"uint256"},{"internalType":"uint256","name":"withdrawn","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"totalReward","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":"oldRecipient","type":"address"},{"internalType":"address","name":"newRecipient","type":"address"}],"name":"updateRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"vestPeriod","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"vestStart","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"}],"name":"vested","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"}],"name":"withdrawFor","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"}],"name":"withdrawable","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"stateMutability":"payable","type":"receive"}]