编译器
0.8.17+commit.8df45f5f
文件 1 的 32: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 的 32: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;
}
}
文件 3 的 32:CurvePoolUtils.sol
pragma solidity 0.8.17;
import "ICurvePoolV2.sol";
import "ICurvePoolV1.sol";
import "ScaledMath.sol";
library CurvePoolUtils {
using ScaledMath for uint256;
error NotWithinThreshold(address pool, uint256 assetA, uint256 assetB);
uint256 internal constant _DEFAULT_IMBALANCE_BUFFER = 30e14;
uint8 internal constant _CURVE_POOL_FEE_DECIMALS = 10;
uint256 internal constant _FEE_IMBALANCE_MULTIPLIER = 3;
enum AssetType {
USD,
ETH,
BTC,
OTHER,
CRYPTO
}
struct PoolMeta {
address pool;
uint256 numberOfCoins;
AssetType assetType;
uint256[] decimals;
uint256[] prices;
uint256[] imbalanceBuffers;
}
function ensurePoolBalanced(PoolMeta memory poolMeta) internal view {
uint256 poolFee = ICurvePoolV1(poolMeta.pool).fee().convertScale(
_CURVE_POOL_FEE_DECIMALS,
18
);
for (uint256 i = 0; i < poolMeta.numberOfCoins - 1; i++) {
uint256 fromDecimals = poolMeta.decimals[i];
uint256 fromBalance = 10 ** fromDecimals;
uint256 fromPrice = poolMeta.prices[i];
for (uint256 j = i + 1; j < poolMeta.numberOfCoins; j++) {
uint256 toDecimals = poolMeta.decimals[j];
uint256 toPrice = poolMeta.prices[j];
uint256 toExpectedUnscaled = (fromBalance * fromPrice) / toPrice;
uint256 toExpected = toExpectedUnscaled.convertScale(
uint8(fromDecimals),
uint8(toDecimals)
);
uint256 toActual;
if (poolMeta.assetType == AssetType.CRYPTO) {
toActual = ICurvePoolV2(poolMeta.pool).get_dy(i, j, fromBalance);
} else {
toActual = ICurvePoolV1(poolMeta.pool).get_dy(
int128(uint128(i)),
int128(uint128(j)),
fromBalance
);
}
uint256 _maxImbalanceBuffer = poolMeta.imbalanceBuffers[i].max(
poolMeta.imbalanceBuffers[j]
);
if (!_isWithinThreshold(toExpected, toActual, poolFee, _maxImbalanceBuffer))
revert NotWithinThreshold(poolMeta.pool, i, j);
}
}
}
function _isWithinThreshold(
uint256 a,
uint256 b,
uint256 poolFee,
uint256 imbalanceBuffer
) internal pure returns (bool) {
if (imbalanceBuffer == 0) imbalanceBuffer = _DEFAULT_IMBALANCE_BUFFER;
uint256 imbalanceTreshold = imbalanceBuffer + poolFee * _FEE_IMBALANCE_MULTIPLIER;
if (a > b) return (a - b).divDown(a) <= imbalanceTreshold;
return (b - a).divDown(b) <= imbalanceTreshold;
}
}
文件 4 的 32:EnumerableSet.sol
pragma solidity ^0.8.0;
library EnumerableSet {
struct Set {
bytes32[] _values;
mapping(bytes32 => uint256) _indexes;
}
function _add(Set storage set, bytes32 value) private returns (bool) {
if (!_contains(set, value)) {
set._values.push(value);
set._indexes[value] = set._values.length;
return true;
} else {
return false;
}
}
function _remove(Set storage set, bytes32 value) private returns (bool) {
uint256 valueIndex = set._indexes[value];
if (valueIndex != 0) {
uint256 toDeleteIndex = valueIndex - 1;
uint256 lastIndex = set._values.length - 1;
if (lastIndex != toDeleteIndex) {
bytes32 lastValue = set._values[lastIndex];
set._values[toDeleteIndex] = lastValue;
set._indexes[lastValue] = valueIndex;
}
set._values.pop();
delete set._indexes[value];
return true;
} else {
return false;
}
}
function _contains(Set storage set, bytes32 value) private view returns (bool) {
return set._indexes[value] != 0;
}
function _length(Set storage set) private view returns (uint256) {
return set._values.length;
}
function _at(Set storage set, uint256 index) private view returns (bytes32) {
return set._values[index];
}
function _values(Set storage set) private view returns (bytes32[] memory) {
return set._values;
}
struct Bytes32Set {
Set _inner;
}
function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _add(set._inner, value);
}
function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
return _remove(set._inner, value);
}
function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
return _contains(set._inner, value);
}
function length(Bytes32Set storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
return _at(set._inner, index);
}
function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
bytes32[] memory store = _values(set._inner);
bytes32[] memory result;
assembly {
result := store
}
return result;
}
struct AddressSet {
Set _inner;
}
function add(AddressSet storage set, address value) internal returns (bool) {
return _add(set._inner, bytes32(uint256(uint160(value))));
}
function remove(AddressSet storage set, address value) internal returns (bool) {
return _remove(set._inner, bytes32(uint256(uint160(value))));
}
function contains(AddressSet storage set, address value) internal view returns (bool) {
return _contains(set._inner, bytes32(uint256(uint160(value))));
}
function length(AddressSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(AddressSet storage set, uint256 index) internal view returns (address) {
return address(uint160(uint256(_at(set._inner, index))));
}
function values(AddressSet storage set) internal view returns (address[] memory) {
bytes32[] memory store = _values(set._inner);
address[] memory result;
assembly {
result := store
}
return result;
}
struct UintSet {
Set _inner;
}
function add(UintSet storage set, uint256 value) internal returns (bool) {
return _add(set._inner, bytes32(value));
}
function remove(UintSet storage set, uint256 value) internal returns (bool) {
return _remove(set._inner, bytes32(value));
}
function contains(UintSet storage set, uint256 value) internal view returns (bool) {
return _contains(set._inner, bytes32(value));
}
function length(UintSet storage set) internal view returns (uint256) {
return _length(set._inner);
}
function at(UintSet storage set, uint256 index) internal view returns (uint256) {
return uint256(_at(set._inner, index));
}
function values(UintSet storage set) internal view returns (uint256[] memory) {
bytes32[] memory store = _values(set._inner);
uint256[] memory result;
assembly {
result := store
}
return result;
}
}
文件 5 的 32:IBonding.sol
pragma solidity 0.8.17;
interface IBonding {
event CncStartPriceSet(uint256 startPrice);
event PriceIncreaseFactorSet(uint256 factor);
event MinBondingAmountSet(uint256 amount);
event Bonded(
address indexed account,
address indexed recipient,
uint256 lpTokenAmount,
uint256 cncReceived,
uint256 lockTime
);
event DebtPoolSet(address indexed pool);
event DebtPoolFeesClaimed(uint256 crvAmount, uint256 cvxAmount, uint256 cncAmount);
event StreamClaimed(address indexed account, uint256 amount);
event BondingStarted(uint256 amount, uint256 epochs);
event RemainingCNCRecovered(uint256 amount);
function startBonding() external;
function setCncStartPrice(uint256 _cncStartPrice) external;
function setCncPriceIncreaseFactor(uint256 _priceIncreaseFactor) external;
function setMinBondingAmount(uint256 _minBondingAmount) external;
function setDebtPool(address _debtPool) external;
function bondCncCrvUsd(
uint256 lpTokenAmount,
uint256 minCncReceived,
uint64 cncLockTime
) external returns (uint256);
function recoverRemainingCNC() external;
function claimStream() external;
function claimFeesForDebtPool() external;
function streamCheckpoint() external;
function accountCheckpoint(address account) external;
function computeCurrentCncBondPrice() external view returns (uint256);
function cncAvailable() external view returns (uint256);
function cncBondPrice() external view returns (uint256);
function bondCncCrvUsdFor(
uint256 lpTokenAmount,
uint256 minCncReceived,
uint64 cncLockTime,
address recipient
) external returns (uint256);
}
文件 6 的 32:IBooster.sol
pragma solidity 0.8.17;
interface IBooster {
function poolInfo(
uint256 pid
)
external
view
returns (
address lpToken,
address token,
address gauge,
address crvRewards,
address stash,
bool shutdown
);
function poolLength() external view returns (uint256);
function deposit(uint256 _pid, uint256 _amount, bool _stake) external returns (bool);
function withdraw(uint256 _pid, uint256 _amount) external returns (bool);
function withdrawAll(uint256 _pid) external returns (bool);
function depositAll(uint256 _pid, bool _stake) external returns (bool);
function earmarkRewards(uint256 _pid) external returns (bool);
function isShutdown() external view returns (bool);
}
文件 7 的 32:IConicPool.sol
pragma solidity 0.8.17;
import "ILpToken.sol";
import "IRewardManager.sol";
import "IOracle.sol";
import "IController.sol";
import "IPausable.sol";
import "IConicPoolWeightManagement.sol";
interface IConicPool is IConicPoolWeightManagement, IPausable {
event Deposit(
address indexed sender,
address indexed receiver,
uint256 depositedAmount,
uint256 lpReceived
);
event Withdraw(address indexed account, uint256 amount);
event NewWeight(address indexed curvePool, uint256 newWeight);
event NewMaxIdleCurveLpRatio(uint256 newRatio);
event ClaimedRewards(uint256 claimedCrv, uint256 claimedCvx);
event HandledDepeggedCurvePool(address curvePool_);
event HandledInvalidConvexPid(address curvePool_, uint256 pid_);
event CurvePoolAdded(address curvePool_);
event CurvePoolRemoved(address curvePool_);
event Shutdown();
event DepegThresholdUpdated(uint256 newThreshold);
event MaxDeviationUpdated(uint256 newMaxDeviation);
event RebalancingRewardsEnabledSet(bool enabled);
event EmergencyRebalancingRewardFactorUpdated(uint256 factor);
struct PoolWithAmount {
address poolAddress;
uint256 amount;
}
function underlying() external view returns (IERC20Metadata);
function lpToken() external view returns (ILpToken);
function rewardManager() external view returns (IRewardManager);
function depegThreshold() external view returns (uint256);
function maxIdleCurveLpRatio() external view returns (uint256);
function setMaxIdleCurveLpRatio(uint256 value) external;
function setMaxDeviation(uint256 maxDeviation_) external;
function updateDepegThreshold(uint256 value) external;
function depositFor(
address _account,
uint256 _amount,
uint256 _minLpReceived,
bool stake
) external returns (uint256);
function deposit(uint256 _amount, uint256 _minLpReceived) external returns (uint256);
function deposit(
uint256 _amount,
uint256 _minLpReceived,
bool stake
) external returns (uint256);
function exchangeRate() external view returns (uint256);
function usdExchangeRate() external view returns (uint256);
function unstakeAndWithdraw(uint256 _amount, uint256 _minAmount) external returns (uint256);
function unstakeAndWithdraw(
uint256 _amount,
uint256 _minAmount,
address _to
) external returns (uint256);
function withdraw(uint256 _amount, uint256 _minAmount) external returns (uint256);
function withdraw(uint256 _amount, uint256 _minAmount, address _to) external returns (uint256);
function getAllocatedUnderlying() external view returns (PoolWithAmount[] memory);
function rebalancingRewardActive() external view returns (bool);
function totalDeviationAfterWeightUpdate() external view returns (uint256);
function computeTotalDeviation() external view returns (uint256);
function totalUnderlying() external view returns (uint256);
function getTotalAndPerPoolUnderlying()
external
view
returns (
uint256 totalUnderlying_,
uint256 totalAllocated_,
uint256[] memory perPoolUnderlying_
);
function cachedTotalUnderlying() external view returns (uint256);
function updateRewardSpendingApproval(address token, bool approved) external;
function shutdownPool() external;
function isShutdown() external view returns (bool);
function isBalanced() external view returns (bool);
function rebalancingRewardsEnabled() external view returns (bool);
function setRebalancingRewardsEnabled(bool enabled) external;
function getAllUnderlyingCoins() external view returns (address[] memory result);
function rebalancingRewardsFactor() external view returns (uint256);
function rebalancingRewardsActivatedAt() external view returns (uint64);
function getWeights() external view returns (PoolWeight[] memory);
function runSanityChecks() external;
}
文件 8 的 32:IConicPoolWeightManagement.sol
pragma solidity 0.8.17;
interface IConicPoolWeightManagement {
struct PoolWeight {
address poolAddress;
uint256 weight;
}
function addPool(address pool) external;
function removePool(address pool) external;
function updateWeights(PoolWeight[] memory poolWeights) external;
function handleDepeggedCurvePool(address curvePool_) external;
function handleInvalidConvexPid(address pool) external returns (uint256);
function allPools() external view returns (address[] memory);
function poolsCount() external view returns (uint256);
function getPoolAtIndex(uint256 _index) external view returns (address);
function getWeight(address curvePool) external view returns (uint256);
function getWeights() external view returns (PoolWeight[] memory);
function isRegisteredPool(address _pool) external view returns (bool);
}
文件 9 的 32:IController.sol
pragma solidity 0.8.17;
import "IConicPoolWeightManagement.sol";
import "IConicPool.sol";
import "IGenericOracle.sol";
import "IInflationManager.sol";
import "ILpTokenStaker.sol";
import "IBonding.sol";
import "IPoolAdapter.sol";
import "IFeeRecipient.sol";
import "ICurveRegistryCache.sol";
interface IController {
event PoolAdded(address indexed pool);
event PoolRemoved(address indexed pool);
event PoolShutdown(address indexed pool);
event ConvexBoosterSet(address convexBooster);
event CurveHandlerSet(address curveHandler);
event ConvexHandlerSet(address convexHandler);
event CurveRegistryCacheSet(address curveRegistryCache);
event InflationManagerSet(address inflationManager);
event BondingSet(address bonding);
event FeeRecipientSet(address feeRecipient);
event PriceOracleSet(address priceOracle);
event WeightUpdateMinDelaySet(uint256 weightUpdateMinDelay);
event PauseManagerSet(address indexed manager, bool isManager);
event MultiDepositsWithdrawsWhitelistSet(address pool, bool allowed);
event MinimumTaintedTransferAmountSet(address indexed token, uint256 amount);
event DefaultPoolAdapterSet(address poolAdapter);
event CustomPoolAdapterSet(address indexed pool, address poolAdapter);
struct WeightUpdate {
address conicPoolAddress;
IConicPoolWeightManagement.PoolWeight[] weights;
}
function initialize(address _lpTokenStaker) external;
function inflationManager() external view returns (IInflationManager);
function setInflationManager(address manager) external;
function curveRegistryCache() external view returns (ICurveRegistryCache);
function poolAdapterFor(address pool) external view returns (IPoolAdapter);
function defaultPoolAdapter() external view returns (IPoolAdapter);
function setDefaultPoolAdapter(address poolAdapter) external;
function setCustomPoolAdapter(address pool, address poolAdapter) external;
function switchLpTokenStaker(address _lpTokenStaker) external;
function lpTokenStaker() external view returns (ILpTokenStaker);
function bonding() external view returns (IBonding);
function setBonding(address _bonding) external;
function feeRecipient() external view returns (IFeeRecipient);
function setFeeRecipient(address _feeRecipient) external;
function priceOracle() external view returns (IGenericOracle);
function setPriceOracle(address oracle) external;
function listPools() external view returns (address[] memory);
function listActivePools() external view returns (address[] memory);
function isPool(address poolAddress) external view returns (bool);
function isActivePool(address poolAddress) external view returns (bool);
function addPool(address poolAddress) external;
function shutdownPool(address poolAddress) external;
function removePool(address poolAddress) external;
function cncToken() external view returns (address);
function lastWeightUpdate(address poolAddress) external view returns (uint256);
function updateWeights(WeightUpdate memory update) external;
function updateAllWeights(WeightUpdate[] memory weights) external;
function convexBooster() external view returns (address);
function curveHandler() external view returns (address);
function convexHandler() external view returns (address);
function setConvexBooster(address _convexBooster) external;
function setCurveHandler(address _curveHandler) external;
function setConvexHandler(address _convexHandler) external;
function setCurveRegistryCache(address curveRegistryCache_) external;
function setWeightUpdateMinDelay(uint256 delay) external;
function isPauseManager(address account) external view returns (bool);
function listPauseManagers() external view returns (address[] memory);
function setPauseManager(address account, bool isManager) external;
function isAllowedMultipleDepositsWithdraws(address poolAddress) external view returns (bool);
function setAllowedMultipleDepositsWithdraws(address account, bool allowed) external;
function getMultipleDepositsWithdrawsWhitelist() external view returns (address[] memory);
function setMinimumTaintedTransferAmount(address token, uint256 amount) external;
function getMinimumTaintedTransferAmount(address token) external view returns (uint256);
function MAX_WEIGHT_UPDATE_MIN_DELAY() external view returns (uint256);
function MIN_WEIGHT_UPDATE_MIN_DELAY() external view returns (uint256);
}
文件 10 的 32:IConvexHandler.sol
pragma solidity 0.8.17;
import "Types.sol";
interface IConvexHandler {
function deposit(address _curvePool, uint256 _amount) external;
function claimEarnings(address _curvePool, address _conicPool) external;
function getRewardPool(address _curvePool) external view returns (address);
function getCrvEarned(address _account, address _curvePool) external view returns (uint256);
function getCrvEarnedBatch(
address _account,
address[] memory _curvePools
) external view returns (uint256);
function computeClaimableConvex(uint256 crvAmount) external view returns (uint256);
function computeClaimableConvexWithCliffInfo(
uint256 crvAmount
) external view returns (uint256, Types.CliffInfo memory);
}
文件 11 的 32:ICurvePoolV1.sol
pragma solidity 0.8.17;
interface ICurvePoolV1 {
function get_virtual_price() external view returns (uint256);
function add_liquidity(uint256[8] calldata amounts, uint256 min_mint_amount) external;
function add_liquidity(uint256[7] calldata amounts, uint256 min_mint_amount) external;
function add_liquidity(uint256[6] calldata amounts, uint256 min_mint_amount) external;
function add_liquidity(uint256[5] calldata amounts, uint256 min_mint_amount) external;
function add_liquidity(uint256[4] calldata amounts, uint256 min_mint_amount) external;
function add_liquidity(uint256[3] calldata amounts, uint256 min_mint_amount) external;
function add_liquidity(uint256[2] calldata amounts, uint256 min_mint_amount) external;
function remove_liquidity_imbalance(
uint256[4] calldata amounts,
uint256 max_burn_amount
) external;
function remove_liquidity_imbalance(
uint256[3] calldata amounts,
uint256 max_burn_amount
) external;
function remove_liquidity_imbalance(
uint256[2] calldata amounts,
uint256 max_burn_amount
) external;
function lp_token() external view returns (address);
function A_PRECISION() external view returns (uint256);
function A_precise() external view returns (uint256);
function remove_liquidity(uint256 _amount, uint256[3] calldata min_amounts) external;
function exchange(
int128 from,
int128 to,
uint256 _from_amount,
uint256 _min_to_amount
) external;
function coins(uint256 i) external view returns (address);
function balances(uint256 i) external view returns (uint256);
function get_dy(int128 i, int128 j, uint256 _dx) external view returns (uint256);
function calc_token_amount(
uint256[4] calldata amounts,
bool deposit
) external view returns (uint256);
function calc_token_amount(
uint256[3] calldata amounts,
bool deposit
) external view returns (uint256);
function calc_token_amount(
uint256[2] calldata amounts,
bool deposit
) external view returns (uint256);
function calc_withdraw_one_coin(
uint256 _token_amount,
int128 i
) external view returns (uint256);
function remove_liquidity_one_coin(
uint256 _token_amount,
int128 i,
uint256 min_amount
) external;
function fee() external view returns (uint256);
}
文件 12 的 32:ICurvePoolV2.sol
pragma solidity 0.8.17;
interface ICurvePoolV2 {
function token() external view returns (address);
function coins(uint256 i) external view returns (address);
function factory() external view returns (address);
function exchange(
uint256 i,
uint256 j,
uint256 dx,
uint256 min_dy,
bool use_eth,
address receiver
) external returns (uint256);
function exchange_underlying(
uint256 i,
uint256 j,
uint256 dx,
uint256 min_dy,
address receiver
) external returns (uint256);
function add_liquidity(
uint256[2] memory amounts,
uint256 min_mint_amount,
bool use_eth,
address receiver
) external returns (uint256);
function add_liquidity(
uint256[2] memory amounts,
uint256 min_mint_amount
) external returns (uint256);
function add_liquidity(
uint256[3] memory amounts,
uint256 min_mint_amount,
bool use_eth,
address receiver
) external returns (uint256);
function add_liquidity(
uint256[3] memory amounts,
uint256 min_mint_amount
) external returns (uint256);
function remove_liquidity(
uint256 _amount,
uint256[2] memory min_amounts,
bool use_eth,
address receiver
) external;
function remove_liquidity(uint256 _amount, uint256[2] memory min_amounts) external;
function remove_liquidity(
uint256 _amount,
uint256[3] memory min_amounts,
bool use_eth,
address receiver
) external;
function remove_liquidity(uint256 _amount, uint256[3] memory min_amounts) external;
function remove_liquidity_one_coin(
uint256 token_amount,
uint256 i,
uint256 min_amount,
bool use_eth,
address receiver
) external returns (uint256);
function get_dy(uint256 i, uint256 j, uint256 dx) external view returns (uint256);
function calc_token_amount(uint256[] memory amounts) external view returns (uint256);
function calc_withdraw_one_coin(
uint256 token_amount,
uint256 i
) external view returns (uint256);
function get_virtual_price() external view returns (uint256);
}
文件 13 的 32:ICurveRegistryCache.sol
pragma solidity 0.8.17;
import "IBooster.sol";
import "CurvePoolUtils.sol";
interface ICurveRegistryCache {
event PoolInitialized(address indexed pool, uint256 indexed pid);
function BOOSTER() external view returns (IBooster);
function initPool(address pool_) external;
function initPool(address pool_, uint256 pid_) external;
function lpToken(address pool_) external view returns (address);
function assetType(address pool_) external view returns (CurvePoolUtils.AssetType);
function isRegistered(address pool_) external view returns (bool);
function hasCoinDirectly(address pool_, address coin_) external view returns (bool);
function hasCoinAnywhere(address pool_, address coin_) external view returns (bool);
function basePool(address pool_) external view returns (address);
function coinIndex(address pool_, address coin_) external view returns (int128);
function nCoins(address pool_) external view returns (uint256);
function coinIndices(
address pool_,
address from_,
address to_
) external view returns (int128, int128, bool);
function decimals(address pool_) external view returns (uint256[] memory);
function interfaceVersion(address pool_) external view returns (uint256);
function poolFromLpToken(address lpToken_) external view returns (address);
function coins(address pool_) external view returns (address[] memory);
function getPid(address _pool) external view returns (uint256);
function getRewardPool(address _pool) external view returns (address);
function isShutdownPid(uint256 pid_) external view returns (bool);
function getAllUnderlyingCoins(address pool) external view returns (address[] memory);
}
文件 14 的 32: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);
}
文件 15 的 32: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);
}
文件 16 的 32: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);
}
文件 17 的 32:IFeeRecipient.sol
pragma solidity 0.8.17;
interface IFeeRecipient {
event FeesReceived(address indexed sender, uint256 crvAmount, uint256 cvxAmount);
function receiveFees(uint256 amountCrv, uint256 amountCvx) external;
}
文件 18 的 32:IGenericOracle.sol
pragma solidity 0.8.17;
import "IOracle.sol";
interface IGenericOracle is IOracle {
function getOracle(address token) external view returns (IOracle);
function curveLpToUnderlying(
address curveLpToken,
address underlying,
uint256 curveLpAmount
) external view returns (uint256);
function curveLpToUnderlying(
address curveLpToken,
address underlying,
uint256 curveLpAmount,
uint256 underlyingPrice
) external view returns (uint256);
function underlyingToCurveLp(
address underlying,
address curveLpToken,
uint256 underlyingAmount
) external view returns (uint256);
}
文件 19 的 32:IInflationManager.sol
pragma solidity 0.8.17;
interface IInflationManager {
event TokensClaimed(address indexed pool, uint256 cncAmount);
event RebalancingRewardHandlerAdded(address indexed pool, address indexed handler);
event RebalancingRewardHandlerRemoved(address indexed pool, address indexed handler);
event PoolWeightsUpdated();
function executeInflationRateUpdate() external;
function updatePoolWeights() external;
function computePoolWeights()
external
view
returns (address[] memory _pools, uint256[] memory poolWeights, uint256 totalUSDValue);
function computePoolWeight(
address pool
) external view returns (uint256 poolWeight, uint256 totalUSDValue);
function currentInflationRate() external view returns (uint256);
function getCurrentPoolInflationRate(address pool) external view returns (uint256);
function handleRebalancingRewards(
address account,
uint256 deviationBefore,
uint256 deviationAfter
) external;
function addPoolRebalancingRewardHandler(
address poolAddress,
address rebalancingRewardHandler
) external;
function removePoolRebalancingRewardHandler(
address poolAddress,
address rebalancingRewardHandler
) external;
function rebalancingRewardHandlers(
address poolAddress
) external view returns (address[] memory);
function hasPoolRebalancingRewardHandler(
address poolAddress,
address handler
) external view returns (bool);
}
文件 20 的 32:ILpToken.sol
pragma solidity 0.8.17;
import "IERC20Metadata.sol";
interface ILpToken is IERC20Metadata {
function minter() external view returns (address);
function mint(address account, uint256 amount, address ubo) external returns (uint256);
function burn(address _owner, uint256 _amount, address ubo) external returns (uint256);
function taint(address from, address to, uint256 amount) external;
}
文件 21 的 32:ILpTokenStaker.sol
pragma solidity 0.8.17;
interface ILpTokenStaker {
event LpTokenStaked(address indexed account, uint256 amount);
event LpTokenUnstaked(address indexed account, uint256 amount);
event TokensClaimed(address indexed pool, uint256 cncAmount);
event Shutdown();
function stake(uint256 amount, address conicPool) external;
function unstake(uint256 amount, address conicPool) external;
function stakeFor(uint256 amount, address conicPool, address account) external;
function unstakeFor(uint256 amount, address conicPool, address account) external;
function unstakeFrom(uint256 amount, address account) external;
function getUserBalanceForPool(
address conicPool,
address account
) external view returns (uint256);
function getBalanceForPool(address conicPool) external view returns (uint256);
function updateBoost(address user) external;
function claimCNCRewardsForPool(address pool) external;
function claimableCnc(address pool) external view returns (uint256);
function checkpoint(address pool) external returns (uint256);
function shutdown() external;
function getBoost(address user) external view returns (uint256);
function isShutdown() external view returns (bool);
}
文件 22 的 32:IOracle.sol
pragma solidity 0.8.17;
interface IOracle {
event TokenUpdated(address indexed token, address feed, uint256 maxDelay, bool isEthPrice);
function getUSDPrice(address token) external view returns (uint256);
function isTokenSupported(address token) external view returns (bool);
}
文件 23 的 32:IPausable.sol
pragma solidity 0.8.17;
import "Ownable.sol";
import "IController.sol";
interface IPausable {
event Paused(uint256 pausedUntil);
event PauseDurationSet(uint256 pauseDuration);
function controller() external view returns (IController);
function pausedUntil() external view returns (uint256);
function pauseDuration() external view returns (uint256);
function isPaused() external view returns (bool);
function setPauseDuration(uint256 _pauseDuration) external;
function pause() external;
}
文件 24 的 32:IPoolAdapter.sol
pragma solidity 0.8.17;
interface IPoolAdapter {
enum PriceMode {
Latest,
Cached,
Minimum
}
function deposit(address pool, address underlying, uint256 underlyingAmount) external;
function withdraw(address pool, address underlying, uint256 underlyingAmount) external;
function computePoolValueInUSD(
address conicPool,
address pool
) external view returns (uint256 usdAmount);
function updatePriceCache(address pool) external;
function computePoolValueInUSD(
address conicPool,
address pool,
PriceMode priceMode
) external view returns (uint256 usdAmount);
function computePoolValueInUnderlying(
address conicPool,
address pool,
address underlying,
uint256 underlyingPrice
) external view returns (uint256 underlyingAmount);
function computePoolValueInUnderlying(
address conicPool,
address pool,
address underlying,
uint256 underlyingPrice,
PriceMode priceMode
) external view returns (uint256 underlyingAmount);
function claimEarnings(address conicPool, address pool) external;
function lpToken(address pool) external view returns (address);
function supportsAsset(address pool, address asset) external view returns (bool);
function getCRVEarnedOnConvex(
address account,
address curvePool
) external view returns (uint256);
function executeSanityCheck(address pool) external;
function getAllUnderlyingCoins(address pool) external view returns (address[] memory);
}
文件 25 的 32:IRewardManager.sol
pragma solidity 0.8.17;
interface IRewardManager {
event ClaimedRewards(uint256 claimedCrv, uint256 claimedCvx);
event SoldRewardTokens(uint256 targetTokenReceived);
event ExtraRewardAdded(address reward);
event ExtraRewardRemoved(address reward);
event ExtraRewardsCurvePoolSet(address extraReward, address curvePool);
event FeesSet(uint256 feePercentage);
event FeesEnabled(uint256 feePercentage);
event EarningsClaimed(
address indexed claimedBy,
uint256 cncEarned,
uint256 crvEarned,
uint256 cvxEarned
);
function accountCheckpoint(address account) external;
function poolCheckpoint() external returns (bool);
function addExtraReward(address reward) external returns (bool);
function addBatchExtraRewards(address[] memory rewards) external;
function conicPool() external view returns (address);
function setFeePercentage(uint256 _feePercentage) external;
function claimableRewards(
address account
) external view returns (uint256 cncRewards, uint256 crvRewards, uint256 cvxRewards);
function claimEarnings() external returns (uint256, uint256, uint256);
function claimPoolEarningsAndSellRewardTokens() external;
function feePercentage() external view returns (uint256);
function feesEnabled() external view returns (bool);
}
文件 26 的 32:Initializable.sol
pragma solidity ^0.8.2;
import "Address.sol";
abstract contract Initializable {
uint8 private _initialized;
bool private _initializing;
event Initialized(uint8 version);
modifier initializer() {
bool isTopLevelCall = !_initializing;
require(
(isTopLevelCall && _initialized < 1) || (!Address.isContract(address(this)) && _initialized == 1),
"Initializable: contract is already initialized"
);
_initialized = 1;
if (isTopLevelCall) {
_initializing = true;
}
_;
if (isTopLevelCall) {
_initializing = false;
emit Initialized(1);
}
}
modifier reinitializer(uint8 version) {
require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
_initialized = version;
_initializing = true;
_;
_initializing = false;
emit Initialized(version);
}
modifier onlyInitializing() {
require(_initializing, "Initializable: contract is not initializing");
_;
}
function _disableInitializers() internal virtual {
require(!_initializing, "Initializable: contract is initializing");
if (_initialized != type(uint8).max) {
_initialized = type(uint8).max;
emit Initialized(type(uint8).max);
}
}
function _getInitializedVersion() internal view returns (uint8) {
return _initialized;
}
function _isInitializing() internal view returns (bool) {
return _initializing;
}
}
文件 27 的 32: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);
}
}
文件 28 的 32:RewardManager.sol
pragma solidity 0.8.17;
import "Ownable.sol";
import "Initializable.sol";
import "EnumerableSet.sol";
import "SafeERC20.sol";
import "IERC20Metadata.sol";
import "IConicPool.sol";
import "ILpToken.sol";
import "IRewardManager.sol";
import "IConvexHandler.sol";
import "IController.sol";
import "IOracle.sol";
import "IInflationManager.sol";
import "ILpTokenStaker.sol";
import "ICurvePoolV2.sol";
import "UniswapRouter02.sol";
import "ScaledMath.sol";
import "Types.sol";
contract RewardManager is IRewardManager, Ownable, Initializable {
using ScaledMath for uint256;
using SafeERC20 for IERC20;
using EnumerableSet for EnumerableSet.AddressSet;
struct RewardMeta {
uint256 earnedIntegral;
uint256 lastHoldings;
mapping(address => uint256) accountIntegral;
mapping(address => uint256) accountShare;
}
IERC20 public constant CVX = IERC20(0x4e3FBD56CD56c3e72c1403e103b45Db9da5B9D2B);
IERC20 public constant CRV = IERC20(0xD533a949740bb3306d119CC777fa900bA034cd52);
IERC20 public constant WETH = IERC20(0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2);
IERC20 public constant CNC = IERC20(0x9aE380F0272E2162340a5bB646c354271c0F5cFC);
UniswapRouter02 public constant SUSHISWAP =
UniswapRouter02(0xd9e1cE17f2641f24aE83637ab66a2cca9C378B9F);
ICurvePoolV2 public constant CNC_ETH_POOL =
ICurvePoolV2(0x838af967537350D2C44ABB8c010E49E32673ab94);
uint256 public constant MAX_FEE_PERCENTAGE = 3e17;
uint256 public constant SLIPPAGE_THRESHOLD = 0.95e18;
bytes32 internal constant _CNC_KEY = "cnc";
bytes32 internal constant _CRV_KEY = "crv";
bytes32 internal constant _CVX_KEY = "cvx";
address public override conicPool;
IERC20 public immutable underlying;
IController public immutable controller;
bool internal _claimingCNC;
EnumerableSet.AddressSet internal _extraRewards;
mapping(address => address) public extraRewardsCurvePool;
mapping(bytes32 => RewardMeta) internal _rewardsMeta;
mapping(uint256 => bool) internal _claimedForCliff;
bool public feesEnabled;
uint256 public feePercentage;
constructor(address _controller, address _underlying) {
underlying = IERC20(_underlying);
controller = IController(_controller);
WETH.safeApprove(address(CNC_ETH_POOL), type(uint256).max);
}
function initialize(address _pool) external onlyOwner initializer {
conicPool = _pool;
}
function poolCheckpoint() public override returns (bool) {
IConvexHandler convexHandler = IConvexHandler(controller.convexHandler());
(
uint256 crvHoldings,
uint256 cvxHoldings,
uint256 cncHoldings,
Types.CliffInfo memory cliffInfo
) = _getHoldingsWithCliffInfo(convexHandler);
if (cliffInfo.withinThreshold && !_claimedForCliff[cliffInfo.currentCliff]) {
_claimPoolEarningsForCliff(cliffInfo.currentCliff);
}
uint256 crvEarned = crvHoldings - _rewardsMeta[_CRV_KEY].lastHoldings;
uint256 cvxEarned;
if (cvxHoldings > _rewardsMeta[_CVX_KEY].lastHoldings) {
cvxEarned = cvxHoldings - _rewardsMeta[_CVX_KEY].lastHoldings;
}
uint256 cncEarned = cncHoldings - _rewardsMeta[_CNC_KEY].lastHoldings;
uint256 crvFee;
uint256 cvxFee;
if (feesEnabled) {
crvFee = crvEarned.mulDown(feePercentage);
cvxFee = cvxEarned.mulDown(feePercentage);
crvEarned -= crvFee;
cvxEarned -= cvxFee;
crvHoldings -= crvFee;
cvxHoldings -= cvxFee;
}
uint256 _totalStaked = controller.lpTokenStaker().getBalanceForPool(conicPool);
if (_totalStaked > 0) {
_updateEarned(_CVX_KEY, cvxHoldings, cvxEarned, _totalStaked);
_updateEarned(_CRV_KEY, crvHoldings, crvEarned, _totalStaked);
_updateEarned(_CNC_KEY, cncHoldings, cncEarned, _totalStaked);
}
if (!feesEnabled) {
return false;
}
bool rewardsClaimed = false;
if (crvFee > CRV.balanceOf(conicPool) || cvxFee > CVX.balanceOf(conicPool)) {
_claimPoolEarningsAndSellRewardTokens();
rewardsClaimed = true;
}
CRV.safeTransferFrom(conicPool, address(this), crvFee);
CVX.safeTransferFrom(conicPool, address(this), cvxFee);
IFeeRecipient _feeRecipient = controller.feeRecipient();
CRV.forceApprove(address(_feeRecipient), crvFee);
CVX.forceApprove(address(_feeRecipient), cvxFee);
_feeRecipient.receiveFees(crvFee, cvxFee);
return rewardsClaimed;
}
function _updateEarned(
bytes32 key,
uint256 holdings,
uint256 earned,
uint256 _totalSupply
) internal {
_rewardsMeta[key].earnedIntegral += earned.divDown(_totalSupply);
_rewardsMeta[key].lastHoldings = holdings;
}
function _getEarnedRewards()
internal
view
returns (uint256 crvEarned, uint256 cvxEarned, uint256 cncEarned)
{
IConvexHandler convexHandler = IConvexHandler(controller.convexHandler());
return _getEarnedRewards(convexHandler);
}
function _getHoldingsWithCliffInfo(
IConvexHandler convexHandler
)
internal
view
returns (
uint256 crvHoldings,
uint256 cvxHoldings,
uint256 cncHoldings,
Types.CliffInfo memory cliffInfo
)
{
address[] memory curvePools = IConicPool(conicPool).allPools();
uint256 claimableCRV;
for (uint256 i; i < curvePools.length; i++) {
claimableCRV += controller.poolAdapterFor(curvePools[i]).getCRVEarnedOnConvex(
conicPool,
curvePools[i]
);
}
crvHoldings = CRV.balanceOf(conicPool) + claimableCRV;
uint256 claimableCVX;
(claimableCVX, cliffInfo) = convexHandler.computeClaimableConvexWithCliffInfo(claimableCRV);
cvxHoldings = CVX.balanceOf(conicPool) + claimableCVX;
cncHoldings = CNC.balanceOf(conicPool);
if (!_claimingCNC) {
cncHoldings += controller.lpTokenStaker().claimableCnc(conicPool);
}
}
function _getHoldings(
IConvexHandler convexHandler
) internal view returns (uint256 crvHoldings, uint256 cvxHoldings, uint256 cncHoldings) {
(crvHoldings, cvxHoldings, cncHoldings, ) = _getHoldingsWithCliffInfo(convexHandler);
}
function _getEarnedRewards(
IConvexHandler convexHandler
) internal view returns (uint256 crvEarned, uint256 cvxEarned, uint256 cncEarned) {
(
uint256 currentHoldingsCRV,
uint256 currentHoldingsCVX,
uint256 currentHoldingsCNC
) = _getHoldings(convexHandler);
crvEarned = currentHoldingsCRV - _rewardsMeta[_CRV_KEY].lastHoldings;
cvxEarned = currentHoldingsCVX - _rewardsMeta[_CVX_KEY].lastHoldings;
cncEarned = currentHoldingsCNC - _rewardsMeta[_CNC_KEY].lastHoldings;
}
function accountCheckpoint(address account) external {
_accountCheckpoint(account);
}
function _accountCheckpoint(address account) internal {
uint256 accountBalance = controller.lpTokenStaker().getUserBalanceForPool(
conicPool,
account
);
poolCheckpoint();
_updateAccountRewardsMeta(_CNC_KEY, account, accountBalance);
_updateAccountRewardsMeta(_CRV_KEY, account, accountBalance);
_updateAccountRewardsMeta(_CVX_KEY, account, accountBalance);
}
function _updateAccountRewardsMeta(bytes32 key, address account, uint256 balance) internal {
RewardMeta storage meta = _rewardsMeta[key];
uint256 share = balance.mulDown(meta.earnedIntegral - meta.accountIntegral[account]);
meta.accountShare[account] += share;
meta.accountIntegral[account] = meta.earnedIntegral;
}
function claimEarnings() public override returns (uint256, uint256, uint256) {
_accountCheckpoint(msg.sender);
uint256 crvAmount = _rewardsMeta[_CRV_KEY].accountShare[msg.sender];
uint256 cvxAmount = _rewardsMeta[_CVX_KEY].accountShare[msg.sender];
uint256 cncAmount = _rewardsMeta[_CNC_KEY].accountShare[msg.sender];
if (
crvAmount > CRV.balanceOf(conicPool) ||
cvxAmount > CVX.balanceOf(conicPool) ||
cncAmount > CNC.balanceOf(conicPool)
) {
_claimPoolEarningsAndSellRewardTokens();
if (_extraRewards.length() > 0) {
_accountCheckpoint(msg.sender);
cncAmount = _rewardsMeta[_CNC_KEY].accountShare[msg.sender];
}
}
_rewardsMeta[_CNC_KEY].accountShare[msg.sender] = 0;
_rewardsMeta[_CVX_KEY].accountShare[msg.sender] = 0;
_rewardsMeta[_CRV_KEY].accountShare[msg.sender] = 0;
CRV.safeTransferFrom(conicPool, msg.sender, crvAmount);
CVX.safeTransferFrom(conicPool, msg.sender, cvxAmount);
if (CNC.balanceOf(conicPool) >= cncAmount) {
CNC.safeTransferFrom(conicPool, msg.sender, cncAmount);
}
(
uint256 currentHoldingsCRV,
uint256 currentHoldingsCVX,
uint256 currentHoldingsCNC
) = _getHoldings(IConvexHandler(controller.convexHandler()));
_rewardsMeta[_CRV_KEY].lastHoldings = currentHoldingsCRV;
_rewardsMeta[_CVX_KEY].lastHoldings = currentHoldingsCVX;
_rewardsMeta[_CNC_KEY].lastHoldings = currentHoldingsCNC;
emit EarningsClaimed(msg.sender, cncAmount, crvAmount, cvxAmount);
return (cncAmount, crvAmount, cvxAmount);
}
function claimPoolEarningsAndSellRewardTokens() external override {
if (!poolCheckpoint()) {
_claimPoolEarningsAndSellRewardTokens();
}
}
function _claimPoolEarningsForCliff(uint256 cliff) internal {
_claimPoolEarningsAndSellRewardTokens();
_claimedForCliff[cliff] = true;
}
function _claimPoolEarningsAndSellRewardTokens() internal {
_claimPoolEarnings();
uint256 cncBalanceBefore_ = CNC.balanceOf(conicPool);
_sellRewardTokens();
uint256 receivedCnc_ = CNC.balanceOf(conicPool) - cncBalanceBefore_;
uint256 _totalStaked = controller.lpTokenStaker().getBalanceForPool(conicPool);
if (_totalStaked > 0 && receivedCnc_ > 0) {
_rewardsMeta[_CNC_KEY].earnedIntegral += receivedCnc_.divDown(_totalStaked);
_rewardsMeta[_CNC_KEY].lastHoldings += receivedCnc_;
emit SoldRewardTokens(receivedCnc_);
}
}
function _claimPoolEarnings() internal {
_claimingCNC = true;
controller.lpTokenStaker().claimCNCRewardsForPool(conicPool);
_claimingCNC = false;
uint256 cvxBalance = CVX.balanceOf(conicPool);
uint256 crvBalance = CRV.balanceOf(conicPool);
address[] memory pools = IConicPool(conicPool).allPools();
for (uint256 i; i < pools.length; i++) {
controller.poolAdapterFor(pools[i]).claimEarnings(conicPool, pools[i]);
}
uint256 claimedCvx = CVX.balanceOf(conicPool) - cvxBalance;
uint256 claimedCrv = CRV.balanceOf(conicPool) - crvBalance;
emit ClaimedRewards(claimedCrv, claimedCvx);
}
function _sellRewardTokens() internal {
uint256 extraRewardsLength_ = _extraRewards.length();
if (extraRewardsLength_ == 0) return;
for (uint256 i; i < extraRewardsLength_; i++) {
IERC20(_extraRewards.at(i)).safeTransferFrom(
conicPool,
address(this),
IERC20(_extraRewards.at(i)).balanceOf(conicPool)
);
_swapRewardTokenForWeth(_extraRewards.at(i));
}
_swapWethForCNC();
}
function listExtraRewards() external view returns (address[] memory) {
return _extraRewards.values();
}
function addExtraReward(address reward) public override onlyOwner returns (bool) {
require(reward != address(0), "invalid address");
require(!_extraRewards.contains(reward), "already added");
require(
reward != address(CVX) &&
reward != address(CRV) &&
reward != address(underlying) &&
reward != address(CNC),
"token not allowed"
);
address[] memory pools = IConicPool(conicPool).allPools();
for (uint256 i; i < pools.length; i++) {
address curveLpToken_ = controller.poolAdapterFor(pools[i]).lpToken(pools[i]);
require(reward != curveLpToken_, "token not allowed");
}
IConicPool(conicPool).updateRewardSpendingApproval(reward, true);
IERC20(reward).safeApprove(address(SUSHISWAP), 0);
IERC20(reward).safeApprove(address(SUSHISWAP), type(uint256).max);
emit ExtraRewardAdded(reward);
return _extraRewards.add(reward);
}
function addBatchExtraRewards(address[] memory _rewards) external override onlyOwner {
for (uint256 i; i < _rewards.length; i++) {
addExtraReward(_rewards[i]);
}
}
function removeExtraReward(address tokenAddress) external onlyOwner {
require(_extraRewards.contains(tokenAddress), "not added");
_extraRewards.remove(tokenAddress);
IConicPool(conicPool).updateRewardSpendingApproval(tokenAddress, false);
emit ExtraRewardRemoved(tokenAddress);
}
function setExtraRewardsCurvePool(address extraReward_, address curvePool_) external onlyOwner {
require(curvePool_ != extraRewardsCurvePool[extraReward_], "must be different to current");
if (curvePool_ != address(0)) {
IERC20(extraReward_).safeApprove(curvePool_, 0);
IERC20(extraReward_).safeApprove(curvePool_, type(uint256).max);
}
extraRewardsCurvePool[extraReward_] = curvePool_;
emit ExtraRewardsCurvePoolSet(extraReward_, curvePool_);
}
function setFeePercentage(uint256 _feePercentage) external override onlyOwner {
require(_feePercentage < MAX_FEE_PERCENTAGE, "cannot set fee percentage to more than 30%");
require(_feePercentage != feePercentage, "must be different to current");
require(address(controller.feeRecipient()) != address(0), "fee recipient not set");
feePercentage = _feePercentage;
feesEnabled = _feePercentage > 0;
emit FeesSet(feePercentage);
}
function claimableRewards(
address account
) external view returns (uint256 cncRewards, uint256 crvRewards, uint256 cvxRewards) {
uint256 _totalStaked = controller.lpTokenStaker().getBalanceForPool(conicPool);
(uint256 crvEarned, uint256 cvxEarned, uint256 cncEarned) = _getEarnedRewards();
uint256 userBalance = controller.lpTokenStaker().getUserBalanceForPool(conicPool, account);
cncRewards = _getClaimableReward(
account,
_CNC_KEY,
cncEarned,
userBalance,
_totalStaked,
false
);
crvRewards = _getClaimableReward(
account,
_CRV_KEY,
crvEarned,
userBalance,
_totalStaked,
feesEnabled
);
cvxRewards = _getClaimableReward(
account,
_CVX_KEY,
cvxEarned,
userBalance,
_totalStaked,
feesEnabled
);
}
function _getClaimableReward(
address account,
bytes32 key,
uint256 earned,
uint256 userBalance,
uint256 _totalSupply,
bool deductFee
) internal view returns (uint256) {
RewardMeta storage meta = _rewardsMeta[key];
uint256 integral = meta.earnedIntegral;
if (_totalSupply > 0) {
if (deductFee) {
integral += earned.mulDown(ScaledMath.ONE - feePercentage).divDown(_totalSupply);
} else {
integral += earned.divDown(_totalSupply);
}
}
return
meta.accountShare[account] +
userBalance.mulDown(integral - meta.accountIntegral[account]);
}
function _swapRewardTokenForWeth(address rewardToken_) internal {
uint256 tokenBalance_ = IERC20(rewardToken_).balanceOf(address(this));
if (tokenBalance_ == 0) return;
ICurvePoolV2 curvePool_ = ICurvePoolV2(extraRewardsCurvePool[rewardToken_]);
if (address(curvePool_) != address(0)) {
(int128 i, int128 j, ) = controller.curveRegistryCache().coinIndices(
address(curvePool_),
rewardToken_,
address(WETH)
);
(uint256 from_, uint256 to_) = (uint256(uint128(i)), uint256(uint128(j)));
curvePool_.exchange(
from_,
to_,
tokenBalance_,
_minAmountOut(address(rewardToken_), address(WETH), tokenBalance_),
false,
address(this)
);
return;
}
address[] memory path_ = new address[](2);
path_[0] = rewardToken_;
path_[1] = address(WETH);
SUSHISWAP.swapExactTokensForTokens(
tokenBalance_,
_minAmountOut(address(rewardToken_), address(WETH), tokenBalance_),
path_,
address(this),
block.timestamp
);
}
function _swapWethForCNC() internal {
uint256 wethBalance_ = WETH.balanceOf(address(this));
if (wethBalance_ == 0) return;
CNC_ETH_POOL.exchange(
0,
1,
wethBalance_,
_minAmountOut(address(WETH), address(CNC), wethBalance_),
false,
conicPool
);
}
function _minAmountOut(
address tokenIn_,
address tokenOut_,
uint256 amountIn_
) internal view returns (uint256) {
IOracle oracle_ = controller.priceOracle();
if (tokenIn_ == tokenOut_) {
return amountIn_;
}
if (!oracle_.isTokenSupported(tokenIn_) || !oracle_.isTokenSupported(tokenOut_)) {
return 0;
}
return
amountIn_
.mulDown(oracle_.getUSDPrice(tokenIn_))
.divDown(oracle_.getUSDPrice(tokenOut_))
.convertScale(
IERC20Metadata(tokenIn_).decimals(),
IERC20Metadata(tokenOut_).decimals()
)
.mulDown(SLIPPAGE_THRESHOLD);
}
}
文件 29 的 32:SafeERC20.sol
pragma solidity ^0.8.0;
import "IERC20.sol";
import "IERC20Permit.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 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));
}
}
文件 30 的 32:ScaledMath.sol
pragma solidity 0.8.17;
library ScaledMath {
uint256 internal constant DECIMALS = 18;
uint256 internal constant ONE = 10 ** DECIMALS;
function mulDown(uint256 a, uint256 b) internal pure returns (uint256) {
return (a * b) / ONE;
}
function mulDown(uint256 a, uint256 b, uint256 decimals) internal pure returns (uint256) {
return (a * b) / (10 ** decimals);
}
function divDown(uint256 a, uint256 b) internal pure returns (uint256) {
return (a * ONE) / b;
}
function divDown(uint256 a, uint256 b, uint256 decimals) internal pure returns (uint256) {
return (a * 10 ** decimals) / b;
}
function divUp(uint256 a, uint256 b) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
return ((a * ONE) - 1) / b + 1;
}
function mulDown(int256 a, int256 b) internal pure returns (int256) {
return (a * b) / int256(ONE);
}
function mulDownUint128(uint128 a, uint128 b) internal pure returns (uint128) {
return (a * b) / uint128(ONE);
}
function mulDown(int256 a, int256 b, uint256 decimals) internal pure returns (int256) {
return (a * b) / int256(10 ** decimals);
}
function divDown(int256 a, int256 b) internal pure returns (int256) {
return (a * int256(ONE)) / b;
}
function divDownUint128(uint128 a, uint128 b) internal pure returns (uint128) {
return (a * uint128(ONE)) / b;
}
function divDown(int256 a, int256 b, uint256 decimals) internal pure returns (int256) {
return (a * int256(10 ** decimals)) / b;
}
function convertScale(
uint256 a,
uint8 fromDecimals,
uint8 toDecimals
) internal pure returns (uint256) {
if (fromDecimals == toDecimals) return a;
if (fromDecimals > toDecimals) return downscale(a, fromDecimals, toDecimals);
return upscale(a, fromDecimals, toDecimals);
}
function convertScale(
int256 a,
uint8 fromDecimals,
uint8 toDecimals
) internal pure returns (int256) {
if (fromDecimals == toDecimals) return a;
if (fromDecimals > toDecimals) return downscale(a, fromDecimals, toDecimals);
return upscale(a, fromDecimals, toDecimals);
}
function upscale(
uint256 a,
uint8 fromDecimals,
uint8 toDecimals
) internal pure returns (uint256) {
return a * (10 ** (toDecimals - fromDecimals));
}
function downscale(
uint256 a,
uint8 fromDecimals,
uint8 toDecimals
) internal pure returns (uint256) {
return a / (10 ** (fromDecimals - toDecimals));
}
function upscale(
int256 a,
uint8 fromDecimals,
uint8 toDecimals
) internal pure returns (int256) {
return a * int256(10 ** (toDecimals - fromDecimals));
}
function downscale(
int256 a,
uint8 fromDecimals,
uint8 toDecimals
) internal pure returns (int256) {
return a / int256(10 ** (fromDecimals - toDecimals));
}
function intPow(uint256 a, uint256 n) internal pure returns (uint256) {
uint256 result = ONE;
for (uint256 i; i < n; ) {
result = mulDown(result, a);
unchecked {
++i;
}
}
return result;
}
function absSub(uint256 a, uint256 b) internal pure returns (uint256) {
unchecked {
return a >= b ? a - b : b - a;
}
}
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a >= b ? a : b;
}
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a <= b ? a : b;
}
}
文件 31 的 32:Types.sol
pragma solidity 0.8.17;
library Types {
struct Coin {
address coinAddress;
uint8 decimals;
}
struct CliffInfo {
uint256 currentCliff;
bool withinThreshold;
}
}
文件 32 的 32:UniswapRouter02.sol
pragma solidity 0.8.17;
interface UniswapRouter02 {
function swapTokensForExactTokens(
uint256 amountOut,
uint256 amountInMax,
address[] calldata path,
address to,
uint256 deadline
) external returns (uint256[] memory amounts);
function swapExactTokensForTokens(
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path,
address to,
uint256 deadline
) external returns (uint256[] memory amounts);
function swapETHForExactTokens(
uint256 amountOut,
address[] calldata path,
address to,
uint256 deadline
) external payable returns (uint256[] memory amounts);
function swapExactTokensForETH(
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path,
address to,
uint256 deadline
) external returns (uint256[] memory amounts);
function getAmountIn(
uint256 amountOut,
uint256 reserveIn,
uint256 reserveOut
) external returns (uint256 amountIn);
function getAmountOut(
uint256 amountIn,
uint256 reserveIn,
uint256 reserveOut
) external view returns (uint256 amountOut);
function getAmountsIn(
uint256 amountOut,
address[] calldata path
) external view returns (uint256[] memory amounts);
function getAmountsOut(
uint256 amountIn,
address[] memory path
) external view returns (uint256[] memory amounts);
function getReserves(
address factory,
address tokenA,
address tokenB
) external view returns (uint256 reserveA, uint256 reserveB);
function WETH() external pure returns (address);
}
interface UniswapV2Pair {
function getReserves()
external
view
returns (uint112 _reserve0, uint112 _reserve1, uint32 _blockTimestampLast);
}
interface UniswapV2Factory {
function getPair(address tokenA, address tokenB) external view returns (address pair);
}
{
"compilationTarget": {
"RewardManager.sol": "RewardManager"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_controller","type":"address"},{"internalType":"address","name":"_underlying","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"claimedCrv","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"claimedCvx","type":"uint256"}],"name":"ClaimedRewards","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"claimedBy","type":"address"},{"indexed":false,"internalType":"uint256","name":"cncEarned","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"crvEarned","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"cvxEarned","type":"uint256"}],"name":"EarningsClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"reward","type":"address"}],"name":"ExtraRewardAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"reward","type":"address"}],"name":"ExtraRewardRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"extraReward","type":"address"},{"indexed":false,"internalType":"address","name":"curvePool","type":"address"}],"name":"ExtraRewardsCurvePoolSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"feePercentage","type":"uint256"}],"name":"FeesEnabled","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"feePercentage","type":"uint256"}],"name":"FeesSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","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":"targetTokenReceived","type":"uint256"}],"name":"SoldRewardTokens","type":"event"},{"inputs":[],"name":"CNC","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CNC_ETH_POOL","outputs":[{"internalType":"contract ICurvePoolV2","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CRV","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"CVX","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_FEE_PERCENTAGE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"SLIPPAGE_THRESHOLD","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"SUSHISWAP","outputs":[{"internalType":"contract UniswapRouter02","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"WETH","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"accountCheckpoint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"_rewards","type":"address[]"}],"name":"addBatchExtraRewards","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"reward","type":"address"}],"name":"addExtraReward","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"claimEarnings","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"claimPoolEarningsAndSellRewardTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"claimableRewards","outputs":[{"internalType":"uint256","name":"cncRewards","type":"uint256"},{"internalType":"uint256","name":"crvRewards","type":"uint256"},{"internalType":"uint256","name":"cvxRewards","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"conicPool","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"controller","outputs":[{"internalType":"contract IController","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"extraRewardsCurvePool","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"feePercentage","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"feesEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_pool","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"listExtraRewards","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"poolCheckpoint","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"tokenAddress","type":"address"}],"name":"removeExtraReward","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"extraReward_","type":"address"},{"internalType":"address","name":"curvePool_","type":"address"}],"name":"setExtraRewardsCurvePool","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_feePercentage","type":"uint256"}],"name":"setFeePercentage","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"underlying","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"}]