编译器
0.8.18+commit.87f61d96
文件 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 {
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address to, 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 from,
address to,
uint256 amount
) external returns (bool);
}
文件 3 的 5:Ownable.sol
pragma solidity ^0.8.0;
import "./Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_transferOwnership(_msgSender());
}
modifier onlyOwner() {
_checkOwner();
_;
}
function owner() public view virtual returns (address) {
return _owner;
}
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
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:staking.sol
pragma solidity 0.8.18;
import "./SafeMath.sol";
import "./IERC20.sol";
import "./Ownable.sol";
contract Staking is Ownable {
using SafeMath for uint256;
struct Staker {
address user;
uint256 amount;
STATUS status;
uint256 startDay;
}
enum STATUS {
ACTIVE,
COMPLETED
}
uint256 startStakingDate;
uint256 amountPerDay;
Staker[] internal stakers;
bool startStaking = false;
uint256 dayInSeconds = 24 * 60 * 60;
IERC20 token;
IERC20 xCrediToken;
uint256 periodInDays = 181;
uint256 daysForStaking = periodInDays * dayInSeconds;
uint256 minimumStakingAmount = 0;
mapping(uint256 => uint256) increase;
mapping(address => uint256[]) stakingsMap;
event Received(address, uint256);
event Staked(address indexed user, uint256 amount, uint256 startDay);
event Claimed(address indexed user, uint256 amount, uint256 date);
constructor(address _token, address _xCrediToken) {
require(_token != address(0), "Invalid address for token");
require(_xCrediToken != address(0), "Invalid address for xCrediToken");
token = IERC20(_token);
xCrediToken = IERC20(_xCrediToken);
}
modifier calculation(uint256 index) {
require(stakers.length > index, "Incorrect staker");
require(stakers[index].user == address(msg.sender), "Not owner");
_;
}
function params()
external
view
returns (
address,
address,
uint256,
uint256
)
{
return (address(token), address(xCrediToken), periodInDays, minimumStakingAmount);
}
receive() external payable {
emit Received(msg.sender, msg.value);
}
fallback() external payable {
emit Received(msg.sender, msg.value);
}
function getCurrentDate() external view returns (uint256) {
return block.timestamp;
}
function start() external onlyOwner {
require(startStaking == false, "Staking started");
require(xCrediToken.balanceOf(address(this)) > 0, "Token balance: 0");
startStaking = true;
startStakingDate = block.timestamp;
amountPerDay = xCrediToken.balanceOf(address(this)).div(periodInDays);
}
function withdraw() external onlyOwner {
uint256 balance = address(this).balance;
require(balance > 0, "Insufficient balance");
address payable owner = payable(owner());
owner.transfer(balance);
}
function withdrawToken(IERC20 _token) external onlyOwner {
uint256 balance = _token.balanceOf(address(this));
require(balance > 0, "Insufficient balance");
require(_token.transfer(address(owner()), balance), "Transfer failed");
}
function getBalance() external view returns (uint256) {
return address(this).balance;
}
function getAmountPerDay() external view returns (uint256) {
return amountPerDay;
}
function getEndDate() external view returns (uint256) {
return _getEndDate();
}
function _getEndDate() internal view returns (uint256) {
return (startStakingDate.add(daysForStaking));
}
function getStartStakingDate() external view returns (uint256) {
return startStakingDate;
}
function getTokenBalance() external view returns (uint256) {
return token.balanceOf(address(this));
}
function getXCrediBalance() external view returns (uint256) {
return xCrediToken.balanceOf(address(this));
}
function getStakers() external view returns (Staker[] memory) {
return stakers;
}
function stake(uint256 amount) external {
_stake(amount);
}
function weight(uint256 index) external view returns (uint256) {
require(index < periodInDays, "Period should be lower");
uint256 w = 0;
for (uint256 ti = 0; ti < index; ti++) {
w = w.add(increase[ti]);
}
return w;
}
function getCurrentDay() external view returns (uint256) {
uint256 startDate = (block.timestamp.sub(startStakingDate)).div(dayInSeconds);
return startDate;
}
function _stake(uint256 amount) internal {
require(startStaking == true, "Staking not started");
require(amount >= minimumStakingAmount, "Amount should be greater than 1000");
require(_getEndDate() > block.timestamp, "Staking period finished");
require(amount > 0, "Amount should be greater than 0");
require(amount <= token.balanceOf(address(msg.sender)), "Insufficient USDC balance");
require(token.transferFrom(address(msg.sender), address(this), amount), "Transfer failed");
uint256 startDate = (block.timestamp.sub(startStakingDate)).div(dayInSeconds);
stakers.push(Staker(address(msg.sender), amount, STATUS.ACTIVE, startDate));
stakingsMap[address(msg.sender)].push(stakers.length - 1);
increase[startDate] = increase[startDate].add(amount);
emit Staked(address(msg.sender), amount, startDate);
}
function userStakings() external view returns (uint256[] memory) {
return stakingsMap[address(msg.sender)];
}
function userStaking(uint256 index) external view returns (Staker memory) {
return stakers[index];
}
function calculateRewards(uint256 index) external view returns (uint256, bool) {
return _calculateRewards(index);
}
function _calculateRewards(uint256 index) internal view calculation(index) returns (uint256, bool) {
uint256 rewards = 0;
uint256 w = 0;
bool calimable = false;
(uint256 currentDay, uint256 startDay) = getDays(index);
for (uint256 ti = 0; ti < periodInDays; ti++) {
w = w.add(increase[ti]);
if (currentDay > ti && startDay <= ti) {
if (w > 0) {
uint256 amount = stakers[index].amount.mul(amountPerDay).div(w);
rewards = rewards.add(amount);
}
}
if (ti >= currentDay) {
break;
}
}
if (currentDay > periodInDays.sub(1)) {
calimable = true;
}
return (rewards, calimable);
}
function getDays(uint256 index) internal view returns (uint256, uint256) {
uint256 currentDay = (block.timestamp.sub(startStakingDate)).div(dayInSeconds);
uint256 startDay = stakers[index].startDay;
return (currentDay, startDay);
}
function withdrawPrincipal(uint256 index, uint256 requestAmount) external calculation(index) {
require(stakers[index].status == STATUS.ACTIVE, "Staking is completed");
require(stakers[index].amount >= requestAmount, "Request amount should be lower than staked amount");
(uint256 currentDay, uint256 startDay) = getDays(index);
require(currentDay < periodInDays, "Staking period is over, please request a reward claim");
if (requestAmount < stakers[index].amount) {
require(stakers[index].amount.sub(requestAmount) >= minimumStakingAmount, "Rest USDC amount should be greater than 1000");
uint256 restAmount = stakers[index].amount.sub(requestAmount);
stakers[index].amount = restAmount;
} else {
stakers[index].status = STATUS.COMPLETED;
stakers[index].amount = 0;
}
increase[startDay] = increase[startDay].sub(requestAmount);
require(token.transfer(stakers[index].user, requestAmount), "Transfer failed");
}
function claim(uint256 index) external calculation(index) {
require(stakers[index].status == STATUS.ACTIVE, "Staking is completed");
(uint256 currentDay, uint256 startDay) = getDays(index);
require(currentDay > periodInDays.sub(1), "Staking is not over");
(uint256 rewards, bool claimable) = _calculateRewards(index);
Staker memory staker = stakers[index];
uint256 amount = staker.amount;
require(token.transfer(staker.user, amount), "Transfer failed USDC");
require(xCrediToken.transfer(staker.user, rewards), "Transfer failed xCREDI");
stakers[index].status = STATUS.COMPLETED;
emit Claimed(staker.user, rewards, block.timestamp);
}
}
{
"compilationTarget": {
"staking.sol": "Staking"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_xCrediToken","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"date","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":"","type":"address"},{"indexed":false,"internalType":"uint256","name":"","type":"uint256"}],"name":"Received","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"startDay","type":"uint256"}],"name":"Staked","type":"event"},{"stateMutability":"payable","type":"fallback"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"calculateRewards","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"claim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getAmountPerDay","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getCurrentDate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getCurrentDay","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getEndDate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getStakers","outputs":[{"components":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"enum Staking.STATUS","name":"status","type":"uint8"},{"internalType":"uint256","name":"startDay","type":"uint256"}],"internalType":"struct Staking.Staker[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getStartStakingDate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTokenBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getXCrediBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"params","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"stake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"start","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"userStaking","outputs":[{"components":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"enum Staking.STATUS","name":"status","type":"uint8"},{"internalType":"uint256","name":"startDay","type":"uint256"}],"internalType":"struct Staking.Staker","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"userStakings","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"weight","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"},{"internalType":"uint256","name":"requestAmount","type":"uint256"}],"name":"withdrawPrincipal","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IERC20","name":"_token","type":"address"}],"name":"withdrawToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]