编译器
0.8.26+commit.8a97fa7a
文件 1 的 11:Address.sol
pragma solidity ^0.8.1;
library Address {
function isContract(address account) internal view returns (bool) {
return account.code.length > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResultFromTarget(target, success, returndata, errorMessage);
}
function verifyCallResultFromTarget(
address target,
bool success,
bytes memory returndata,
string memory errorMessage
) internal view returns (bytes memory) {
if (success) {
if (returndata.length == 0) {
require(isContract(target), "Address: call to non-contract");
}
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
_revert(returndata, errorMessage);
}
}
function _revert(bytes memory returndata, string memory errorMessage) private pure {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
文件 2 的 11: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;
}
function _contextSuffixLength() internal view virtual returns (uint256) {
return 0;
}
}
文件 3 的 11:Eth4.sol
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
import "./PZENPZICPriceLib.sol";
interface IPulseLiquidityPool {
function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast);
function token0() external view returns (address);
function token1() external view returns (address);
}
contract Eth4 is Ownable, ReentrancyGuard {
using SafeERC20 for IERC20;
using PZENPZICPriceLib for PZENPZICPriceLib.Observation[24];
event TimeUnitUpdated(uint256 oldTimeUnit, uint256 newTimeUnit);
event RewardCalculationDetails(
uint256 indexed pid,
address indexed user,
uint256 pzenEquivalent,
uint256 periodsPassed,
uint256 reward,
uint256 pzenPzicRatio,
uint256 finalPzicReward
);
event ObservationUpdated(uint256 timestamp, uint256 reservePZEN, uint256 totalSupply, uint32 blockTimestampLast);
struct Notification {
uint256 poolID;
uint256 amount;
address user;
string typeOf;
uint256 timestamp;
}
struct UserInfo {
uint256 amount;
uint256 pzenEquivalent;
uint256 lastRewardAt;
uint256 lockUntil;
}
struct PoolInfo {
uint256 poolID;
IERC20 depositToken;
IERC20 rewardToken;
uint256 depositedAmount;
uint256 apy;
uint256 lockDays;
}
uint256 public timeUnit;
Notification[] public notifications;
address public pzenTokenAddress;
address public plpTokenAddress;
IPulseLiquidityPool public plpContract;
uint256 public poolCount;
PoolInfo[] public poolInfo;
mapping(uint256 => mapping(address => UserInfo)) public userInfo;
event Deposit(address indexed user, uint256 indexed pid, uint256 amount, uint256 pzenEquivalent);
event Withdraw(address indexed user, uint256 indexed pid, uint256 amount, uint256 pzenEquivalent);
event Claim(address indexed user, uint256 indexed pid, uint256 amount);
event InsufficientRewardTokens(uint256 indexed pid, uint256 requestedAmount, uint256 availableAmount);
event PendingRewardCalculated(uint256 indexed pid, uint256 reward, uint256 adjustedReward);
struct Observation {
uint256 timestamp;
uint256 reservePZEN;
uint256 totalSupply;
}
Observation[] public observations;
uint256 public constant OBSERVATION_PERIOD = 1 hours;
uint256 public constant MAX_OBSERVATIONS = 24;
IPZENPZICPool public immutable pzenPzicPool;
PZENPZICPriceLib.Observation[24] private priceObservations;
uint256 private observationIndex;
uint256 private observationCardinality;
constructor(
address _plpTokenAddress,
address _pzenTokenAddress,
uint256 _initialTimeUnit,
address _pzenPzicPoolAddress
)
{
require(_plpTokenAddress != address(0), "Invalid PLP token address");
require(_pzenTokenAddress != address(0), "Invalid PZEN token address");
require(_initialTimeUnit >= 60, "Initial time unit must be at least 60 seconds");
pzenPzicPool = IPZENPZICPool(_pzenPzicPoolAddress);
plpTokenAddress = _plpTokenAddress;
pzenTokenAddress = _pzenTokenAddress;
plpContract = IPulseLiquidityPool(_plpTokenAddress);
timeUnit = _initialTimeUnit;
(observationIndex, observationCardinality) = PZENPZICPriceLib.updateObservation(
pzenPzicPool,
pzenTokenAddress,
priceObservations,
observationIndex,
observationCardinality
);
}
function initializeObservation() public onlyOwner {
require(observations.length == 0, "Observations already initialized");
updateObservation();
}
function updateObservation() internal {
bool shouldUpdate;
if (observations.length > 0) {
shouldUpdate = block.timestamp - observations[observations.length - 1].timestamp >= OBSERVATION_PERIOD;
} else {
shouldUpdate = true;
}
if (shouldUpdate) {
(uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast) = plpContract.getReserves();
uint256 reservePZEN = plpContract.token0() == pzenTokenAddress ? uint256(reserve0) : uint256(reserve1);
uint256 totalSupply = IERC20(plpTokenAddress).totalSupply();
if (observations.length > 0) {
if (observations.length >= MAX_OBSERVATIONS) {
for (uint i = 0; i < observations.length - 1; i++) {
observations[i] = observations[i + 1];
}
observations[observations.length - 1] = Observation(block.timestamp, reservePZEN, totalSupply);
} else {
observations.push(Observation(block.timestamp, reservePZEN, totalSupply));
}
} else {
observations.push(Observation(block.timestamp, reservePZEN, totalSupply));
}
emit ObservationUpdated(block.timestamp, reservePZEN, totalSupply, blockTimestampLast); }
}
function getPzenPzicRatio() public view returns (uint256) {
return PZENPZICPriceLib.calculateTWAP(
pzenPzicPool,
pzenTokenAddress,
priceObservations,
observationIndex,
observationCardinality
);
}
function sweep(address token, uint256 amount) external onlyOwner {
uint256 tokenBalance = IERC20(token).balanceOf(address(this));
require(amount <= tokenBalance, "Amount exceeds balance");
bool isRewardToken = false;
uint256 poolInfoLength = poolInfo.length;
for (uint256 i = 0; i < poolInfoLength; i++) {
if (address(poolInfo[i].rewardToken) == token) {
isRewardToken = true;
break;
}
}
require(isRewardToken, "Token is not a reward token");
IERC20(token).safeTransfer(msg.sender, amount);
}
function calculatePzenEquivalent(uint256 _plpAmount) public view returns (uint256) {
require(observations.length > 0, "No price data available");
if (observations.length <= 1) {
uint256 reservePZEN = observations[0].reservePZEN;
uint256 totalSupply = observations[0].totalSupply;
uint256 calculatedPzenEquivalent = (_plpAmount * reservePZEN) / totalSupply;
return calculatedPzenEquivalent;
}
uint256 timeWeightedReservePZEN = 0;
uint256 timeWeightedTotalSupply = 0;
uint256 timeSum = 0;
uint256 startIndex = 0;
for (uint i = observations.length - 1; i > 0; i--) {
if (block.timestamp - observations[i].timestamp > OBSERVATION_PERIOD) {
startIndex = i + 1;
break;
}
}
if (startIndex >= observations.length - 1) {
startIndex = observations.length > 1 ? observations.length - 2 : 0;
}
uint256 observationsLength = observations.length;
for (uint i = startIndex; i < observationsLength; i++) {
uint256 timeInterval;
if (i >= observations.length - 1) {
timeInterval = block.timestamp - observations[i].timestamp;
} else {
timeInterval = observations[i + 1].timestamp - observations[i].timestamp;
}
timeWeightedReservePZEN += observations[i].reservePZEN * timeInterval;
timeWeightedTotalSupply += observations[i].totalSupply * timeInterval;
timeSum += timeInterval;
}
if (timeSum <= 0) {
timeWeightedReservePZEN = observations[observations.length - 1].reservePZEN;
timeWeightedTotalSupply = observations[observations.length - 1].totalSupply;
timeSum = 1;
}
uint256 pzenEquivalent = (_plpAmount * timeWeightedReservePZEN) / timeWeightedTotalSupply;
return pzenEquivalent;
}
function checkRewardTokenBalance(uint256 _pid, uint256 _amount) internal view returns (bool) {
PoolInfo storage pool = poolInfo[_pid];
return pool.rewardToken.balanceOf(address(this)) >= _amount;
}
function deposit(uint256 _pid, uint256 _amount) public nonReentrant {
require(_pid < poolInfo.length, "Invalid pool ID");
require(_amount > 0, "Amount should be greater than 0");
(observationIndex, observationCardinality) = PZENPZICPriceLib.updateObservation(
pzenPzicPool,
pzenTokenAddress,
priceObservations,
observationIndex,
observationCardinality
);
updateObservation();
require(observations.length > 0, "No price data available");
PoolInfo storage pool = poolInfo[_pid];
UserInfo storage user = userInfo[_pid][msg.sender];
uint256 pending = 0;
if (user.amount > 0) {
pending = _calcPendingReward(user, _pid);
}
if (pending > 0 && !checkRewardTokenBalance(_pid, pending)) {
emit InsufficientRewardTokens(_pid, pending, pool.rewardToken.balanceOf(address(this)));
pending = 0;
}
uint256 pzenEquivalent = calculatePzenEquivalent(_amount);
require(
pool.depositToken.balanceOf(msg.sender) >= _amount,
"Insufficient balance"
);
pool.depositedAmount += _amount;
user.amount += _amount;
user.pzenEquivalent += pzenEquivalent;
user.lastRewardAt = block.timestamp;
if (user.lockUntil <= block.timestamp) {
user.lockUntil = block.timestamp + (pool.lockDays * 86400 * timeUnit / 86400);
}
_createNotification(_pid, _amount, msg.sender, "Deposit");
emit Deposit(msg.sender, _pid, _amount, pzenEquivalent);
pool.depositToken.safeTransferFrom(msg.sender, address(this), _amount);
if (pending > 0) {
require(
pool.rewardToken.balanceOf(address(this)) >= pending,
"Insufficient reward token balance"
);
pool.rewardToken.safeTransfer(msg.sender, pending);
emit Claim(msg.sender, _pid, pending);
}
}
function setTimeUnit(uint256 _timeUnit) external onlyOwner {
uint256 oldTimeUnit = timeUnit;
timeUnit = _timeUnit;
emit TimeUnitUpdated(oldTimeUnit, _timeUnit);
}
function setPlpTokenAddress(address _plpTokenAddress) external onlyOwner {
require(_plpTokenAddress != address(0), "Invalid PLP token address");
plpTokenAddress = _plpTokenAddress;
plpContract = IPulseLiquidityPool(_plpTokenAddress);
}
function setPzenTokenAddress(address _pzenTokenAddress) external onlyOwner {
require(_pzenTokenAddress != address(0), "Invalid PZEN token address");
pzenTokenAddress = _pzenTokenAddress;
}
function _createNotification(uint256 _id, uint256 _amount, address _user, string memory _typeOf) internal {
notifications.push(Notification({
poolID: _id,
amount: _amount,
user: _user,
typeOf: _typeOf,
timestamp: block.timestamp
}));
}
function getNotifications() public view returns (Notification[] memory) {
return notifications;
}
function addPool(
IERC20 _depositToken,
IERC20 _rewardToken,
uint256 _apy,
uint256 _lockDays
) public onlyOwner {
require(address(_depositToken) == plpTokenAddress, "Deposit token must be PLP");
poolInfo.push(
PoolInfo({
poolID: poolCount,
depositToken: _depositToken,
rewardToken: _rewardToken,
depositedAmount: 0,
apy: _apy,
lockDays: _lockDays
})
);
poolCount++;
}
function canWithdraw(
uint256 _pid,
uint256 _amount,
address _user
) public view returns (bool canWithdrawResult, string memory reason) {
require(_pid < poolInfo.length, "Invalid pool ID");
UserInfo storage user = userInfo[_pid][_user];
if (user.amount < _amount) {
return (false, "Withdraw amount exceeds balance");
}
if (user.lockUntil > block.timestamp) {
return (false, "Lock period not ended");
}
return (true, "");
}
function withdraw(uint256 _pid, uint256 _amount) public nonReentrant {
require(_pid < poolInfo.length, "Invalid pool ID");
require(_amount > 0, "Amount should be greater than 0");
(observationIndex, observationCardinality) = PZENPZICPriceLib.updateObservation(
pzenPzicPool,
pzenTokenAddress,
priceObservations,
observationIndex,
observationCardinality
);
updateObservation();
PoolInfo storage pool = poolInfo[_pid];
UserInfo storage user = userInfo[_pid][msg.sender];
require(user.amount >= _amount, "Withdraw amount exceeds balance");
require(user.lockUntil <= block.timestamp, "Lock period not ended");
uint256 totalPzenEquivalent = calculatePzenEquivalent(user.amount);
uint256 pzenEquivalentToWithdraw = (totalPzenEquivalent * _amount) / user.amount;
uint256 pending = _calcPendingReward(user, _pid);
uint256 rewardBalance = pool.rewardToken.balanceOf(address(this));
if (rewardBalance < pending) {
emit InsufficientRewardTokens(_pid, pending, rewardBalance);
pending = rewardBalance;
}
user.amount -= _amount;
user.pzenEquivalent = totalPzenEquivalent - pzenEquivalentToWithdraw;
pool.depositedAmount -= _amount;
if (user.amount > 0) {
user.lockUntil = block.timestamp + (pool.lockDays * 86400 * timeUnit /86400 );
} else {
user.lockUntil = 0;
}
user.lastRewardAt = block.timestamp;
if (pending > 0) {
_createNotification(_pid, pending, msg.sender, "Claim");
}
if (_amount > 0) {
_createNotification(_pid, _amount, msg.sender, "Withdraw");
}
emit Withdraw(msg.sender, _pid, _amount, pzenEquivalentToWithdraw);
if (pending > 0) {
emit Claim(msg.sender, _pid, pending);
}
pool.depositToken.safeTransfer(msg.sender, _amount);
if (pending > 0) {
require(
pool.rewardToken.balanceOf(address(this)) >= pending,
"Insufficient reward token balance"
);
pool.rewardToken.safeTransfer(msg.sender, pending);
}
emit WithdrawDetails(_pid, _amount, pzenEquivalentToWithdraw, pending, rewardBalance);
}
event WithdrawDetails(uint256 indexed pid, uint256 amount, uint256 pzenEquivalentWithdrawn, uint256 pendingRewards, uint256 contractRewardBalance);
function claimReward(uint256 _pid) public nonReentrant {
require(_pid < poolInfo.length, "Invalid pool ID");
PoolInfo storage pool = poolInfo[_pid];
UserInfo storage user = userInfo[_pid][msg.sender];
require(user.amount > 0, "No tokens deposited in this pool");
require(user.lockUntil <= block.timestamp, "Lock is active");
updateObservation();
uint256 pending = _calcPendingReward(user, _pid);
require(pending > 0, "No rewards to claim");
if (!checkRewardTokenBalance(_pid, pending)) {
emit InsufficientRewardTokens(_pid, pending, pool.rewardToken.balanceOf(address(this)));
revert("Insufficient reward tokens in the contract");
}
user.lastRewardAt = block.timestamp;
user.lockUntil = block.timestamp + (pool.lockDays * 86400 * timeUnit /86400);
user.pzenEquivalent = calculatePzenEquivalent(user.amount);
_createNotification(_pid, pending, msg.sender, "Claim");
emit Claim(msg.sender, _pid, pending);
require(
pool.rewardToken.balanceOf(address(this)) >= pending,
"Insufficient reward token balance"
);
pool.rewardToken.safeTransfer(msg.sender, pending);
}
function pendingReward(uint256 _pid, address _user) public view returns (uint256) {
UserInfo storage user = userInfo[_pid][_user];
return _calcPendingReward(user, _pid);
}
function _calcPendingReward(UserInfo storage user, uint256 _pid)
internal
view
returns (uint256)
{
PoolInfo storage pool = poolInfo[_pid];
uint256 periodsPassed = (block.timestamp > user.lastRewardAt)
? (block.timestamp - user.lastRewardAt) / timeUnit
: 0;
uint256 maxPeriods = pool.lockDays *86400 /timeUnit;
if (periodsPassed > maxPeriods) {
periodsPassed = maxPeriods;
}
if (periodsPassed == 0 || user.pzenEquivalent == 0) {
return (0);
}
uint256 rewardNumerator = user.pzenEquivalent * pool.apy * periodsPassed;
uint256 rewardDenominator = 365 * 100;
uint256 pzenPzicRatio = PZENPZICPriceLib.calculateTWAP(
pzenPzicPool,
pzenTokenAddress,
priceObservations,
observationIndex,
observationCardinality
);
uint256 pzicReward = (rewardNumerator * pzenPzicRatio) / (rewardDenominator * 1e18);
return pzicReward;
}
}
文件 4 的 11: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);
}
文件 5 的 11:IERC20Metadata.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
interface IERC20Metadata is IERC20 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
}
文件 6 的 11:IERC20Permit.sol
pragma solidity ^0.8.0;
interface IERC20Permit {
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
function nonces(address owner) external view returns (uint256);
function DOMAIN_SEPARATOR() external view returns (bytes32);
}
文件 7 的 11: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() {
_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);
}
}
文件 8 的 11:PZENPZICPriceLib.sol
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/utils/math/SafeMath.sol";
interface IPZENPZICPool {
function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast);
function token0() external view returns (address);
function token1() external view returns (address);
}
library PZENPZICPriceLib {
using SafeMath for uint256;
struct Observation {
uint timestamp;
uint256 pzenReserve;
uint256 pzicReserve;
}
uint256 private constant OBSERVATION_PERIOD = 1 hours;
uint256 private constant MAX_OBSERVATIONS = 24;
event TWAPCalculated(uint256 twap);
event PZENPZICObservationUpdated(uint256 timestamp, uint256 pzenReserve, uint256 pzicReserve, uint32 blockTimestampLast);
function calculateTWAP(
IPZENPZICPool pool,
address pzenAddress,
Observation[MAX_OBSERVATIONS] storage observations,
uint256 observationIndex,
uint256 observationCardinality
) internal view returns (uint256) {
require(observationCardinality > 0, "No observations");
uint256 timeWeightedReserveRatio;
uint256 timeSum;
for (uint256 i = 0; i < observationCardinality - 1; i++) {
uint256 nextIndex = (observationIndex + 1 + i) % observationCardinality;
uint256 currentIndex = (observationIndex + i) % observationCardinality;
Observation memory current = observations[currentIndex];
Observation memory next = observations[nextIndex];
uint256 timeInterval = next.timestamp - current.timestamp;
uint256 numerator = current.pzicReserve.mul(1e18).mul(timeInterval);
timeWeightedReserveRatio = timeWeightedReserveRatio.add(numerator.div(current.pzenReserve));
timeSum = timeSum.add(timeInterval);
}
if (timeSum == 0) {
(uint112 reserve0, uint112 reserve1,uint32 blockTimestampLast) = pool.getReserves();
address token0 = pool.token0();
uint256 pzenReserve = token0 == pzenAddress ? uint256(reserve0) : uint256(reserve1);
uint256 pzicReserve = token0 == pzenAddress ? uint256(reserve1) : uint256(reserve0);
uint256 currentRatio = pzicReserve.mul(1e18).div(pzenReserve);
uint256 dummy = uint256(blockTimestampLast);
dummy;
return currentRatio;
}
uint256 twap = timeWeightedReserveRatio.div(timeSum);
return twap;
}
function updateObservation(
IPZENPZICPool pool,
address pzenAddress,
Observation[MAX_OBSERVATIONS] storage observations,
uint256 observationIndex,
uint256 observationCardinality
) internal returns (uint256, uint256) {
(uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast) = pool.getReserves();
address token0 = pool.token0();
uint256 pzenReserve = token0 == pzenAddress ? uint256(reserve0) : uint256(reserve1);
uint256 pzicReserve = token0 == pzenAddress ? uint256(reserve1) : uint256(reserve0);
observations[observationIndex] = Observation(block.timestamp, pzenReserve, pzicReserve);
emit PZENPZICObservationUpdated(block.timestamp, pzenReserve, pzicReserve, blockTimestampLast);
observationIndex = (observationIndex + 1) % MAX_OBSERVATIONS;
if (observationCardinality < MAX_OBSERVATIONS) {
observationCardinality++;
}
return (observationIndex, observationCardinality);
}
}
文件 9 的 11:ReentrancyGuard.sol
pragma solidity ^0.8.0;
abstract contract ReentrancyGuard {
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
modifier nonReentrant() {
_nonReentrantBefore();
_;
_nonReentrantAfter();
}
function _nonReentrantBefore() private {
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
_status = _ENTERED;
}
function _nonReentrantAfter() private {
_status = _NOT_ENTERED;
}
function _reentrancyGuardEntered() internal view returns (bool) {
return _status == _ENTERED;
}
}
文件 10 的 11:SafeERC20.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/IERC20Permit.sol";
import "../../../utils/Address.sol";
library SafeERC20 {
using Address for address;
function safeTransfer(IERC20 token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
function safeApprove(IERC20 token, address spender, uint256 value) internal {
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
uint256 oldAllowance = token.allowance(address(this), spender);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
}
function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
}
}
function forceApprove(IERC20 token, address spender, uint256 value) internal {
bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
if (!_callOptionalReturnBool(token, approvalCall)) {
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
_callOptionalReturn(token, approvalCall);
}
}
function safePermit(
IERC20Permit token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) {
(bool success, bytes memory returndata) = address(token).call(data);
return
success && (returndata.length == 0 || abi.decode(returndata, (bool))) && Address.isContract(address(token));
}
}
文件 11 的 11: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;
}
}
}
{
"compilationTarget": {
"contracts/SEP26/Eth4.sol": "Eth4"
},
"evmVersion": "cancun",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_plpTokenAddress","type":"address"},{"internalType":"address","name":"_pzenTokenAddress","type":"address"},{"internalType":"uint256","name":"_initialTimeUnit","type":"uint256"},{"internalType":"address","name":"_pzenPzicPoolAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Claim","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"pzenEquivalent","type":"uint256"}],"name":"Deposit","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"requestedAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"availableAmount","type":"uint256"}],"name":"InsufficientRewardTokens","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"timestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"reservePZEN","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"totalSupply","type":"uint256"},{"indexed":false,"internalType":"uint32","name":"blockTimestampLast","type":"uint32"}],"name":"ObservationUpdated","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":"uint256","name":"timestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"pzenReserve","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"pzicReserve","type":"uint256"},{"indexed":false,"internalType":"uint32","name":"blockTimestampLast","type":"uint32"}],"name":"PZENPZICObservationUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"reward","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"adjustedReward","type":"uint256"}],"name":"PendingRewardCalculated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"pzenEquivalent","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"periodsPassed","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"reward","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"pzenPzicRatio","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"finalPzicReward","type":"uint256"}],"name":"RewardCalculationDetails","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"oldTimeUnit","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newTimeUnit","type":"uint256"}],"name":"TimeUnitUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"pzenEquivalent","type":"uint256"}],"name":"Withdraw","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"pid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"pzenEquivalentWithdrawn","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"pendingRewards","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"contractRewardBalance","type":"uint256"}],"name":"WithdrawDetails","type":"event"},{"inputs":[],"name":"MAX_OBSERVATIONS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"OBSERVATION_PERIOD","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"contract IERC20","name":"_depositToken","type":"address"},{"internalType":"contract IERC20","name":"_rewardToken","type":"address"},{"internalType":"uint256","name":"_apy","type":"uint256"},{"internalType":"uint256","name":"_lockDays","type":"uint256"}],"name":"addPool","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_plpAmount","type":"uint256"}],"name":"calculatePzenEquivalent","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_user","type":"address"}],"name":"canWithdraw","outputs":[{"internalType":"bool","name":"canWithdrawResult","type":"bool"},{"internalType":"string","name":"reason","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"}],"name":"claimReward","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"deposit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getNotifications","outputs":[{"components":[{"internalType":"uint256","name":"poolID","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address","name":"user","type":"address"},{"internalType":"string","name":"typeOf","type":"string"},{"internalType":"uint256","name":"timestamp","type":"uint256"}],"internalType":"struct Eth4.Notification[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getPzenPzicRatio","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"initializeObservation","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"notifications","outputs":[{"internalType":"uint256","name":"poolID","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address","name":"user","type":"address"},{"internalType":"string","name":"typeOf","type":"string"},{"internalType":"uint256","name":"timestamp","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"observations","outputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"uint256","name":"reservePZEN","type":"uint256"},{"internalType":"uint256","name":"totalSupply","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"address","name":"_user","type":"address"}],"name":"pendingReward","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"plpContract","outputs":[{"internalType":"contract IPulseLiquidityPool","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"plpTokenAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"poolCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"poolInfo","outputs":[{"internalType":"uint256","name":"poolID","type":"uint256"},{"internalType":"contract IERC20","name":"depositToken","type":"address"},{"internalType":"contract IERC20","name":"rewardToken","type":"address"},{"internalType":"uint256","name":"depositedAmount","type":"uint256"},{"internalType":"uint256","name":"apy","type":"uint256"},{"internalType":"uint256","name":"lockDays","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pzenPzicPool","outputs":[{"internalType":"contract IPZENPZICPool","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pzenTokenAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_plpTokenAddress","type":"address"}],"name":"setPlpTokenAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_pzenTokenAddress","type":"address"}],"name":"setPzenTokenAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_timeUnit","type":"uint256"}],"name":"setTimeUnit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"sweep","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"timeUnit","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":"","type":"uint256"},{"internalType":"address","name":"","type":"address"}],"name":"userInfo","outputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"pzenEquivalent","type":"uint256"},{"internalType":"uint256","name":"lastRewardAt","type":"uint256"},{"internalType":"uint256","name":"lockUntil","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_pid","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]