// SPDX-License-Identifier: MIT
pragma solidity 0.8.4;
interface IERC20 {
function balanceOf(address account) external view returns (uint256);
function transfer(address recipient, uint256 amount) external returns (bool);
function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
}
contract Staking {
address public owner;
IERC20 public TKN;
uint256[5] public periods = [30 days, 60 days, 90 days, 180 days, 360 days];
uint8[5] public rates = [101, 102, 103, 106, 112];
uint256[5] public amounts = [10000e18, 20000e18, 30000e18, 50000e18, 100000e18];
uint256 public rewardsPool;
uint256 public MAX_STAKES = 100;
struct Stake {
uint8 class;
uint8 cycle;
uint256 initialAmount;
uint256 finalAmount;
uint256 timestamp;
bool unstaked;
}
Stake[] public stakes;
mapping(address => uint256[]) public stakesOf;
mapping(uint256 => address) public ownerOf;
event Staked(address indexed sender, uint8 indexed class, uint256 amount, uint256 finalAmount);
event Prolonged(address indexed sender, uint8 indexed class, uint8 cycle, uint256 newAmount, uint256 newFinalAmount);
event Unstaked(address indexed sender, uint8 indexed class, uint8 cycle, uint256 amount);
event TransferOwnership(address indexed previousOwner, address indexed newOwner);
event IncreaseRewardsPool(address indexed adder, uint256 added, uint256 newSize);
modifier restricted {
require(msg.sender == owner, 'This function is restricted to owner');
_;
}
function stakesInfo(uint256 _from, uint256 _to) public view returns (Stake[] memory s) {
s = new Stake[](_to - _from);
for (uint256 i = _from; i <= _to; i++) s[i - _from] = stakes[i];
}
function stakesInfoAll() public view returns (Stake[] memory s) {
s = new Stake[](stakes.length);
for (uint256 i = 0; i < stakes.length; i++) s[i] = stakes[i];
}
function stakesLength() public view returns (uint256) {
return stakes.length;
}
function myStakes(address _me) public view returns (Stake[] memory s, uint256[] memory indexes) {
s = new Stake[](stakesOf[_me].length);
indexes = new uint256[](stakesOf[_me].length);
for (uint256 i = 0; i < stakesOf[_me].length; i++) {
indexes[i] = stakesOf[_me][i];
s[i] = stakes[indexes[i]];
}
}
function myActiveStakesCount(address _me) public view returns (uint256 l) {
uint256[] storage _s = stakesOf[_me];
for (uint256 i = 0; i < _s.length; i++) if (!stakes[_s[i]].unstaked) l++;
}
function stake(uint8 _class) public {
require(_class < 5, "Wrong class"); // data valid
uint256 _amount = amounts[_class];
require(myActiveStakesCount(msg.sender) < MAX_STAKES, "MAX_STAKES overflow"); // has space for new active stake
uint256 _finalAmount = (_amount * rates[_class]) / 100;
require(rewardsPool >= _finalAmount - _amount, "Rewards pool is empty for now");
rewardsPool -= _finalAmount - _amount;
require(TKN.transferFrom(msg.sender, address(this), _amount));
uint256 _index = stakes.length;
stakesOf[msg.sender].push(_index);
stakes.push(Stake({
class: _class,
cycle: 1,
initialAmount: _amount,
finalAmount: _finalAmount,
timestamp: block.timestamp,
unstaked: false
}));
ownerOf[_index] = msg.sender;
emit Staked(msg.sender, _class, _amount, _finalAmount);
}
function prolong(uint256 _index) public {
require(msg.sender == ownerOf[_index]);
Stake storage _s = stakes[_index];
require(!_s.unstaked); // not unstaked yet
require(block.timestamp >= _s.timestamp + periods[_s.class]); // staking period finished
uint256 _newFinalAmount = (_s.finalAmount * rates[_s.class]) / 100;
require(rewardsPool >= _newFinalAmount - _s.finalAmount, "Rewards pool is empty for now");
rewardsPool -= _newFinalAmount - _s.finalAmount;
_s.timestamp = block.timestamp;
_s.cycle++;
require(_s.cycle * periods[_s.class] <= 360 days, "total staking time exceeds 360 days");
emit Prolonged(msg.sender, _s.class, _s.cycle, _s.finalAmount, _newFinalAmount);
_s.finalAmount = _newFinalAmount;
}
function unstake(uint256 _index) public {
require(msg.sender == ownerOf[_index]);
Stake storage _s = stakes[_index];
require(!_s.unstaked); // not unstaked yet
require(block.timestamp >= _s.timestamp + periods[_s.class]); // staking period finished
require(TKN.transfer(msg.sender, _s.finalAmount));
_s.unstaked = true;
emit Unstaked(msg.sender, _s.class, _s.cycle, _s.finalAmount);
}
function transferOwnership(address _newOwner) public restricted {
require(_newOwner != address(0), 'Invalid address: should not be 0x0');
emit TransferOwnership(owner, _newOwner);
owner = _newOwner;
}
function returnAccidentallySent(IERC20 _TKN) public restricted {
require(address(_TKN) != address(TKN));
uint256 _amount = _TKN.balanceOf(address(this));
require(TKN.transfer(msg.sender, _amount));
}
function increaseRewardsPool(uint256 _amount) public {
TKN.transferFrom(msg.sender, address(this), _amount);
rewardsPool += _amount;
emit IncreaseRewardsPool(msg.sender, _amount, rewardsPool);
}
function updateMax(uint256 _max) public restricted {
MAX_STAKES = _max;
}
constructor(IERC20 _TKN) {
owner = msg.sender;
TKN = _TKN;
}
}
{
"compilationTarget": {
"Staking.sol": "Staking"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"contract IERC20","name":"_TKN","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"adder","type":"address"},{"indexed":false,"internalType":"uint256","name":"added","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newSize","type":"uint256"}],"name":"IncreaseRewardsPool","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"uint8","name":"class","type":"uint8"},{"indexed":false,"internalType":"uint8","name":"cycle","type":"uint8"},{"indexed":false,"internalType":"uint256","name":"newAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newFinalAmount","type":"uint256"}],"name":"Prolonged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"uint8","name":"class","type":"uint8"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"finalAmount","type":"uint256"}],"name":"Staked","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"TransferOwnership","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"uint8","name":"class","type":"uint8"},{"indexed":false,"internalType":"uint8","name":"cycle","type":"uint8"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Unstaked","type":"event"},{"inputs":[],"name":"MAX_STAKES","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"TKN","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"amounts","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"increaseRewardsPool","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_me","type":"address"}],"name":"myActiveStakesCount","outputs":[{"internalType":"uint256","name":"l","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_me","type":"address"}],"name":"myStakes","outputs":[{"components":[{"internalType":"uint8","name":"class","type":"uint8"},{"internalType":"uint8","name":"cycle","type":"uint8"},{"internalType":"uint256","name":"initialAmount","type":"uint256"},{"internalType":"uint256","name":"finalAmount","type":"uint256"},{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"bool","name":"unstaked","type":"bool"}],"internalType":"struct Staking.Stake[]","name":"s","type":"tuple[]"},{"internalType":"uint256[]","name":"indexes","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"periods","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_index","type":"uint256"}],"name":"prolong","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"rates","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"contract IERC20","name":"_TKN","type":"address"}],"name":"returnAccidentallySent","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"rewardsPool","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint8","name":"_class","type":"uint8"}],"name":"stake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"stakes","outputs":[{"internalType":"uint8","name":"class","type":"uint8"},{"internalType":"uint8","name":"cycle","type":"uint8"},{"internalType":"uint256","name":"initialAmount","type":"uint256"},{"internalType":"uint256","name":"finalAmount","type":"uint256"},{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"bool","name":"unstaked","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_from","type":"uint256"},{"internalType":"uint256","name":"_to","type":"uint256"}],"name":"stakesInfo","outputs":[{"components":[{"internalType":"uint8","name":"class","type":"uint8"},{"internalType":"uint8","name":"cycle","type":"uint8"},{"internalType":"uint256","name":"initialAmount","type":"uint256"},{"internalType":"uint256","name":"finalAmount","type":"uint256"},{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"bool","name":"unstaked","type":"bool"}],"internalType":"struct Staking.Stake[]","name":"s","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stakesInfoAll","outputs":[{"components":[{"internalType":"uint8","name":"class","type":"uint8"},{"internalType":"uint8","name":"cycle","type":"uint8"},{"internalType":"uint256","name":"initialAmount","type":"uint256"},{"internalType":"uint256","name":"finalAmount","type":"uint256"},{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"bool","name":"unstaked","type":"bool"}],"internalType":"struct Staking.Stake[]","name":"s","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stakesLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"stakesOf","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":"uint256","name":"_index","type":"uint256"}],"name":"unstake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_max","type":"uint256"}],"name":"updateMax","outputs":[],"stateMutability":"nonpayable","type":"function"}]