//////////DEFI product for staking Your YOP tokens and getting rewards
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20 {
/**
* @dev Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `recipient`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address recipient, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: Beware that changing an allowance with this method brings the risk
* that someone may use both the old and the new allowance by unfortunate
* transaction ordering. One possible solution to mitigate this race
* condition is to first reduce the spender's allowance to 0 and set the
* desired value afterwards:
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `sender` to `recipient` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/
event Transfer(address indexed from, address indexed to, uint256 value);
/**
* @dev Emitted when the allowance of a `spender` for an `owner` is set by
* a call to {approve}. `value` is the new allowance.
*/
event Approval(address indexed owner, address indexed spender, uint256 value);
}
///////////////////////////////////////////////////////////////////////////////////
pragma solidity ^0.8.0;
//Version 1.3
//THE smart contract for staking RAMP tokens and claiming the rewards
//Fully fair and simple. Enjoy ;)
contract HopOnYop
{
IERC20 public YOP;
uint256 public RewardPool;
uint256 public AllTimeStaked;
uint256 public TVL;
uint256 public RewardsOwed;
uint256 private constant minStake = 88 * (10 ** 8);
uint256 private constant maxStake = 33333 * (10 ** 8);
uint8 constant public reward1 = 6; uint256 constant public stakedFor1 = 30 days; //6% reward for 30 days lock
uint8 constant public reward2 = 15; uint256 constant public stakedFor2 = 60 days; //15% reward for 60 days lock
uint8 constant public reward3 = 33; uint256 constant public stakedFor3 = 90 days; //33% reward for 90 days lock
constructor (address addr)
{
YOP = IERC20(addr);
}
enum options {d30, d60, d90}
struct stake
{
uint256 amount;
uint256 stakingTime;
options option;
bool rewardTaken;
}
mapping(address => stake) private stakes;
/**
* @dev Adds more tokens to the pool, but first we needs to add allowance for this contract
*/
function feedRewardPool() public
{
uint256 tokenAmount = YOP.allowance(msg.sender, address(this));
RewardPool += tokenAmount;
require(YOP.transferFrom(msg.sender, address(this), tokenAmount)); //Transfers the tokens to smart contract
}
function stakeYOP(options option) public
{
require(stakes[msg.sender].stakingTime == 0, "Error: Only one staking per address!!!");
uint256 tokenAmount = YOP.allowance(msg.sender, address(this));
require(tokenAmount > 0, "Error: Need to increase allowance first");
require(tokenAmount >= minStake && tokenAmount <= maxStake ,"Error: You should stake from 33 to 88888 tokens.");
stakes[msg.sender].amount = tokenAmount;
stakes[msg.sender].option = option;
stakes[msg.sender].stakingTime = block.timestamp;
uint256 reward = calculateReward(msg.sender);
require(RewardPool >= reward + RewardsOwed, "Error: No enough rewards for You, shouldve thought about this before it went moon");
TVL += tokenAmount;
RewardsOwed += reward;
AllTimeStaked += tokenAmount;
require(YOP.transferFrom(msg.sender, address(this), tokenAmount)); //Transfers the tokens to smart contract
}
/**
* @dev claims the rewards and stake for the stake, can be only called by the user
* doesnt work if the campaign isnt finished yet
*/
function claimRewards() public
{
require(stakes[msg.sender].rewardTaken == false,"Error: You already took the reward");
uint256 stakedFor;
options option = stakes[msg.sender].option;
if(option == options.d30)
stakedFor = stakedFor1;
if(option == options.d60)
stakedFor = stakedFor2;
if(option == options.d90)
stakedFor = stakedFor3;
require(stakes[msg.sender].stakingTime + stakedFor <= block.timestamp, "Error: Too soon to unstake");
uint256 reward = calculateReward(msg.sender);
uint256 amount = stakes[msg.sender].amount;
TVL -= amount;
RewardsOwed -= reward;
RewardPool -= reward;
stakes[msg.sender].rewardTaken = true;
_withdraw(reward + amount);
}
/**
* @dev calculates the rewards+stake for the given staker
* @param staker is the staker we want the info for
*/
function calculateReward(address staker) public view returns(uint256)
{
uint256 reward;
options option = stakes[staker].option;
if(option == options.d30)
reward = reward1;
if(option == options.d60)
reward = reward2;
if(option == options.d90)
reward = reward3;
return ((stakes[staker].amount * reward) / 100);
}
function getStakerInfo(address addr) public view returns(uint256, uint256, options, bool)
{
return(stakes[addr].amount,stakes[addr].stakingTime,stakes[addr].option,stakes[addr].rewardTaken);
}
function _withdraw(uint256 amount) internal
{
require(YOP.transfer(msg.sender, amount));
emit withdrawHappened(msg.sender, amount);
}
event withdrawHappened(address indexed to, uint256 amount);
}
{
"compilationTarget": {
"HopOnYop.sol": "HopOnYop"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"addr","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"withdrawHappened","type":"event"},{"inputs":[],"name":"AllTimeStaked","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"RewardPool","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"RewardsOwed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"TVL","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"YOP","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"staker","type":"address"}],"name":"calculateReward","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimRewards","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"feedRewardPool","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"addr","type":"address"}],"name":"getStakerInfo","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"enum HopOnYop.options","name":"","type":"uint8"},{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"reward1","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"reward2","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"reward3","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"enum HopOnYop.options","name":"option","type":"uint8"}],"name":"stakeYOP","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"stakedFor1","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stakedFor2","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stakedFor3","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"}]