文件 1 的 13:Address.sol
pragma solidity ^0.8.0;
library Address {
function isContract(address account) internal view returns (bool) {
uint256 size;
assembly { size := extcodesize(account) }
return size > 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 functionCall(target, data, "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");
require(isContract(target), "Address: call to non-contract");
(bool success, bytes memory returndata) = target.call{ value: value }(data);
return _verifyCallResult(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) {
require(isContract(target), "Address: static call to non-contract");
(bool success, bytes memory returndata) = target.staticcall(data);
return _verifyCallResult(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) {
require(isContract(target), "Address: delegate call to non-contract");
(bool success, bytes memory returndata) = target.delegatecall(data);
return _verifyCallResult(success, returndata, errorMessage);
}
function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) {
if (success) {
return returndata;
} else {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
文件 2 的 13:Harvester.sol
pragma solidity 0.8.9;
import "SafeERC20.sol";
import "IVaultRewardHandler.sol";
import "ICurvePool.sol";
import "ICurveTriCrypto.sol";
import "ICurveV2Pool.sol";
import "ICvxCrvDeposit.sol";
import "ICurveFactoryPool.sol";
import "StrategyBase.sol";
contract stkCvxPrismaHarvester is stkCvxPrismaStrategyBase {
using SafeERC20 for IERC20;
address public owner;
address public immutable strategy;
uint256 public allowedSlippage = 9700;
uint256 public constant DECIMALS = 10000;
address public pendingOwner;
bool public useOracle = true;
bool public forceLock;
constructor(address _strategy) {
strategy = _strategy;
owner = msg.sender;
}
function setApprovals() external {
IERC20(CVX_TOKEN).safeApprove(CURVE_CVX_ETH_POOL, 0);
IERC20(CVX_TOKEN).safeApprove(CURVE_CVX_ETH_POOL, type(uint256).max);
IERC20(MKUSD_TOKEN).safeApprove(CURVE_PRISMA_MKUSD_POOL, 0);
IERC20(MKUSD_TOKEN).safeApprove(
CURVE_PRISMA_MKUSD_POOL,
type(uint256).max
);
IERC20(PRISMA_TOKEN).safeApprove(PRISMA_DEPOSIT, 0);
IERC20(PRISMA_TOKEN).safeApprove(PRISMA_DEPOSIT, type(uint256).max);
IERC20(PRISMA_TOKEN).safeApprove(CURVE_CVXPRISMA_PRISMA_POOL, 0);
IERC20(PRISMA_TOKEN).safeApprove(
CURVE_CVXPRISMA_PRISMA_POOL,
type(uint256).max
);
}
function switchOracle() external onlyOwner {
useOracle = !useOracle;
}
function setPendingOwner(address _po) external onlyOwner {
pendingOwner = _po;
}
function acceptOwnership() external {
require(pendingOwner == msg.sender, "only new owner");
owner = pendingOwner;
pendingOwner = address(0);
}
function setForceLock() external onlyOwner {
forceLock = !forceLock;
}
function rescueToken(address _token, address _to) external onlyOwner {
require(
_token != MKUSD_TOKEN &&
_token != PRISMA_TOKEN &&
_token != CVX_TOKEN,
"not allowed"
);
IERC20 _t = IERC20(_token);
uint256 _balance = _t.balanceOf(address(this));
_t.safeTransfer(_to, _balance);
}
function setSlippage(uint256 _slippage) external onlyOwner {
allowedSlippage = _slippage;
}
function _calcMinAmountOutCvxPrisma(
uint256 _amount
) internal returns (uint256) {
uint256 _cvxEthPrice = cvxEthSwap.price_oracle();
uint256 _ethPrismaPrice = prismaEthSwap.price_oracle();
uint256 _amountEthPrice = (_amount * _cvxEthPrice) / 1e18;
uint256 _amountPrismaPrice = (_amountEthPrice * 1e18) / _ethPrismaPrice;
return ((_amountPrismaPrice * allowedSlippage) / DECIMALS);
}
function _calcMinAmountOutMkUsdPrisma(
uint256 _amount
) internal returns (uint256) {
uint256 _prismaMkUsdPrice = mkUsdPrismaSwap.price_oracle();
uint256 _amountPrismaPrice = (_amount * 1e18) / _prismaMkUsdPrice;
return ((_amountPrismaPrice * allowedSlippage) / DECIMALS);
}
function processRewards()
external
onlyStrategy
returns (uint256 _harvested)
{
uint256 _cvxBalance = IERC20(CVX_TOKEN).balanceOf(address(this));
if (_cvxBalance > 0) {
_swapEthCvx(_cvxBalance, 0, false);
}
uint256 _ethBalance = address(this).balance;
_harvested = 0;
if (_ethBalance > 0) {
_swapEthPrisma(
_ethBalance,
useOracle ? _calcMinAmountOutCvxPrisma(_cvxBalance) : 0,
true
);
}
uint256 _mkUsdBalance = IERC20(MKUSD_TOKEN).balanceOf(address(this));
if (_mkUsdBalance > 0) {
_swapMkUsdPrisma(
_mkUsdBalance,
useOracle ? _calcMinAmountOutMkUsdPrisma(_mkUsdBalance) : 0,
true
);
}
uint256 _prismaBalance = IERC20(PRISMA_TOKEN).balanceOf(address(this));
if (_prismaBalance > 0) {
uint256 _oraclePrice = cvxPrismaPrismaSwap.price_oracle();
if (_oraclePrice > 1 ether || forceLock) {
cvxPrismaDeposit.deposit(_prismaBalance, true);
_harvested = _prismaBalance;
}
else {
uint256 _minCvxPrismaAmountOut = 0;
if (useOracle) {
_minCvxPrismaAmountOut =
(_prismaBalance * 1e18) /
_oraclePrice;
_minCvxPrismaAmountOut = ((_minCvxPrismaAmountOut *
allowedSlippage) / DECIMALS);
}
_harvested = _swapCvxPrismaPrisma(
_prismaBalance,
_minCvxPrismaAmountOut,
false
);
}
IERC20(CVXPRISMA_TOKEN).safeTransfer(msg.sender, _harvested);
}
return _harvested;
}
modifier onlyOwner() {
require((msg.sender == owner), "owner only");
_;
}
modifier onlyStrategy() {
require((msg.sender == strategy), "strategy only");
_;
}
}
文件 3 的 13:IBasicRewards.sol
pragma solidity 0.8.9;
interface IBasicRewards {
function stakeFor(address, uint256) external returns (bool);
function balanceOf(address) external view returns (uint256);
function earned(address) external view returns (uint256);
function withdrawAll(bool) external returns (bool);
function withdraw(uint256, bool) external returns (bool);
function withdrawAndUnwrap(uint256 amount, bool claim)
external
returns (bool);
function getReward() external returns (bool);
function stake(uint256) external returns (bool);
function extraRewards(uint256) external view returns (address);
function exit() external returns (bool);
}
文件 4 的 13:ICurveFactoryPool.sol
pragma solidity 0.8.9;
interface ICurveFactoryPool {
function get_dy(
int128 i,
int128 j,
uint256 dx
) external view returns (uint256);
function get_balances() external view returns (uint256[2] memory);
function add_liquidity(
uint256[2] memory _amounts,
uint256 _min_mint_amount,
address _receiver
) external returns (uint256);
function exchange(
int128 i,
int128 j,
uint256 _dx,
uint256 _min_dy,
address _receiver
) external returns (uint256);
}
文件 5 的 13:ICurvePool.sol
pragma solidity 0.8.9;
interface ICurvePool {
function remove_liquidity_one_coin(
uint256 token_amount,
int128 i,
uint256 min_amount
) external;
function calc_withdraw_one_coin(uint256 _token_amount, int128 i)
external
view
returns (uint256);
function exchange(
int128 i,
int128 j,
uint256 dx,
uint256 min_dy
) external returns (uint256);
function exchange(
int128 i,
int128 j,
uint256 dx,
uint256 min_dy,
address receiver
) external returns (uint256);
function get_dy(
int128 i,
int128 j,
uint256 dx
) external view returns (uint256);
function get_virtual_price() external view returns (uint256);
function price_oracle() external view returns (uint256);
}
文件 6 的 13:ICurveTriCrypto.sol
pragma solidity 0.8.9;
interface ICurveTriCrypto {
function exchange(
uint256 i,
uint256 j,
uint256 dx,
uint256 min_dy,
bool use_eth
) external payable;
function get_dy(
uint256 i,
uint256 j,
uint256 dx
) external view returns (uint256);
function price_oracle(uint256 k) external view returns (uint256);
}
文件 7 的 13:ICurveV2Pool.sol
pragma solidity 0.8.9;
interface ICurveV2Pool {
function get_dy(
uint256 i,
uint256 j,
uint256 dx
) external view returns (uint256);
function calc_token_amount(uint256[2] calldata amounts)
external
view
returns (uint256);
function exchange_underlying(
uint256 i,
uint256 j,
uint256 dx,
uint256 min_dy
) external payable returns (uint256);
function add_liquidity(uint256[2] calldata amounts, uint256 min_mint_amount)
external
returns (uint256);
function lp_price() external view returns (uint256);
function exchange(
uint256 i,
uint256 j,
uint256 dx,
uint256 min_dy
) external payable returns (uint256);
function exchange(
uint256 i,
uint256 j,
uint256 dx,
uint256 min_dy,
bool use_eth
) external payable returns (uint256);
function exchange(
uint256 i,
uint256 j,
uint256 dx,
uint256 min_dy,
bool use_eth,
address receiver
) external payable returns (uint256);
function price_oracle() external view returns (uint256);
function remove_liquidity_one_coin(
uint256 token_amount,
uint256 i,
uint256 min_amount,
bool use_eth,
address receiver
) external returns (uint256);
}
文件 8 的 13:ICvxCrvDeposit.sol
pragma solidity 0.8.9;
interface ICvxCrvDeposit {
function deposit(uint256, bool) external;
}
文件 9 的 13:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address recipient, 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 sender, address recipient, uint256 amount) external returns (bool);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
}
文件 10 的 13:IVaultRewardHandler.sol
pragma solidity 0.8.9;
interface IVaultRewardHandler {
function sell(uint256 _amount) external;
function setPendingOwner(address _po) external;
function applyPendingOwner() external;
function rescueToken(address _token, address _to) external;
}
文件 11 的 13:IWETH.sol
pragma solidity 0.8.9;
interface IWETH {
function deposit() external payable;
function transfer(address to, uint256 value) external returns (bool);
function withdraw(uint256) external;
}
文件 12 的 13:SafeERC20.sol
pragma solidity ^0.8.0;
import "IERC20.sol";
import "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 newAllowance = token.allowance(address(this), spender) + value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
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");
uint256 newAllowance = oldAllowance - value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) {
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}
文件 13 的 13:StrategyBase.sol
pragma solidity 0.8.9;
import "ICurveV2Pool.sol";
import "ICurvePool.sol";
import "ICurveFactoryPool.sol";
import "IBasicRewards.sol";
import "IWETH.sol";
interface ICvxPrismaDeposit {
function deposit(uint256, bool) external;
}
contract stkCvxPrismaStrategyBase {
address public constant PRISMA_DEPOSIT =
0x61404F7c2d8b1F3373eb3c6e8C4b8d8332c2D5B8;
address public constant CURVE_CVX_ETH_POOL =
0xB576491F1E6e5E62f1d8F26062Ee822B40B0E0d4;
address public constant CURVE_PRISMA_ETH_POOL =
0x322135Dd9cBAE8Afa84727d9aE1434b5B3EBA44B;
address public constant CURVE_CVXPRISMA_PRISMA_POOL =
0x3b21C2868B6028CfB38Ff86127eF22E68d16d53B;
address public constant CURVE_PRISMA_MKUSD_POOL =
0x9D8108DDD8aD1Ee89d527C0C9e928Cb9D2BBa2d3;
address public constant CVXPRISMA_TOKEN =
0x34635280737b5BFe6c7DC2FC3065D60d66e78185;
address public constant PRISMA_TOKEN =
0xdA47862a83dac0c112BA89c6abC2159b95afd71C;
address public constant CVX_TOKEN =
0x4e3FBD56CD56c3e72c1403e103b45Db9da5B9D2B;
address public constant MKUSD_TOKEN =
0x4591DBfF62656E7859Afe5e45f6f47D3669fBB28;
uint256 public constant PRISMAETH_ETH_INDEX = 0;
uint256 public constant PRISMAETH_PRISMA_INDEX = 1;
uint256 public constant CVXETH_ETH_INDEX = 0;
uint256 public constant CVXETH_CVX_INDEX = 1;
uint256 public constant PRISMAMKUSD_MKUSD_INDEX = 0;
uint256 public constant PRISMAMKUSD_PRISMA_INDEX = 1;
int128 public constant PRISMACVXPRISMA_CVXPRISMA_INDEX = 1;
int128 public constant PRISMACVXPRISMA_PRISMA_INDEX = 0;
ICvxPrismaDeposit cvxPrismaDeposit = ICvxPrismaDeposit(PRISMA_DEPOSIT);
ICurveV2Pool cvxEthSwap = ICurveV2Pool(CURVE_CVX_ETH_POOL);
ICurveV2Pool prismaEthSwap = ICurveV2Pool(CURVE_PRISMA_ETH_POOL);
ICurvePool cvxPrismaPrismaSwap = ICurvePool(CURVE_CVXPRISMA_PRISMA_POOL);
ICurveV2Pool mkUsdPrismaSwap = ICurveV2Pool(CURVE_PRISMA_MKUSD_POOL);
function _swapEthCvx(
uint256 amount,
uint256 minAmountOut,
bool ethToCvx
) internal returns (uint256) {
return
cvxEthSwap.exchange_underlying{value: ethToCvx ? amount : 0}(
ethToCvx ? CVXETH_ETH_INDEX : CVXETH_CVX_INDEX,
ethToCvx ? CVXETH_CVX_INDEX : CVXETH_ETH_INDEX,
amount,
minAmountOut
);
}
function _swapEthPrisma(
uint256 amount,
uint256 minAmountOut,
bool ethToPrisma
) internal returns (uint256) {
return
prismaEthSwap.exchange_underlying{value: ethToPrisma ? amount : 0}(
ethToPrisma ? PRISMAETH_ETH_INDEX : PRISMAETH_PRISMA_INDEX,
ethToPrisma ? PRISMAETH_PRISMA_INDEX : PRISMAETH_ETH_INDEX,
amount,
minAmountOut
);
}
function _swapMkUsdPrisma(
uint256 amount,
uint256 minAmountOut,
bool mkUsdToPrisma
) internal returns (uint256) {
return
mkUsdPrismaSwap.exchange_underlying(
mkUsdToPrisma
? PRISMAMKUSD_MKUSD_INDEX
: PRISMAMKUSD_PRISMA_INDEX,
mkUsdToPrisma
? PRISMAMKUSD_PRISMA_INDEX
: PRISMAMKUSD_MKUSD_INDEX,
amount,
minAmountOut
);
}
function _swapCvxPrismaPrisma(
uint256 amount,
uint256 minAmountOut,
bool cvxPrismaToPrisma
) internal returns (uint256) {
return
cvxPrismaPrismaSwap.exchange(
cvxPrismaToPrisma
? PRISMACVXPRISMA_CVXPRISMA_INDEX
: PRISMACVXPRISMA_PRISMA_INDEX,
cvxPrismaToPrisma
? PRISMACVXPRISMA_PRISMA_INDEX
: PRISMACVXPRISMA_CVXPRISMA_INDEX,
amount,
minAmountOut
);
}
receive() external payable {}
}
{
"compilationTarget": {
"Harvester.sol": "stkCvxPrismaHarvester"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_strategy","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"CURVE_CVXPRISMA_PRISMA_POOL","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CURVE_CVX_ETH_POOL","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CURVE_PRISMA_ETH_POOL","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CURVE_PRISMA_MKUSD_POOL","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CVXETH_CVX_INDEX","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CVXETH_ETH_INDEX","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CVXPRISMA_TOKEN","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CVX_TOKEN","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"DECIMALS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MKUSD_TOKEN","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMACVXPRISMA_CVXPRISMA_INDEX","outputs":[{"internalType":"int128","name":"","type":"int128"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMACVXPRISMA_PRISMA_INDEX","outputs":[{"internalType":"int128","name":"","type":"int128"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMAETH_ETH_INDEX","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMAETH_PRISMA_INDEX","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMAMKUSD_MKUSD_INDEX","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMAMKUSD_PRISMA_INDEX","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMA_DEPOSIT","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRISMA_TOKEN","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"acceptOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"allowedSlippage","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"forceLock","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":[],"name":"processRewards","outputs":[{"internalType":"uint256","name":"_harvested","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_to","type":"address"}],"name":"rescueToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"setApprovals","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"setForceLock","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_po","type":"address"}],"name":"setPendingOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_slippage","type":"uint256"}],"name":"setSlippage","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"strategy","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"switchOracle","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"useOracle","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"stateMutability":"payable","type":"receive"}]