编译器
0.8.17+commit.8df45f5f
文件 1 的 27: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 的 27: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 的 27:CurvePoolUtils.sol
pragma solidity 0.8.17;
import "../interfaces/vendor/ICurvePoolV2.sol";
import "../interfaces/vendor/ICurvePoolV1.sol";
import "./ScaledMath.sol";
library CurvePoolUtils {
using ScaledMath for uint256;
uint256 internal constant _DEFAULT_IMBALANCE_THRESHOLD = 0.02e18;
enum AssetType {
USD,
ETH,
BTC,
OTHER,
CRYPTO
}
struct PoolMeta {
address pool;
uint256 numberOfCoins;
AssetType assetType;
uint256[] decimals;
uint256[] prices;
uint256[] thresholds;
}
function ensurePoolBalanced(PoolMeta memory poolMeta) internal view {
uint256 fromDecimals = poolMeta.decimals[0];
uint256 fromBalance = 10**fromDecimals;
uint256 fromPrice = poolMeta.prices[0];
for (uint256 i = 1; i < poolMeta.numberOfCoins; i++) {
uint256 toDecimals = poolMeta.decimals[i];
uint256 toPrice = poolMeta.prices[i];
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(0, i, fromBalance);
} else {
toActual = ICurvePoolV1(poolMeta.pool).get_dy(0, int128(uint128(i)), fromBalance);
}
require(
_isWithinThreshold(toExpected, toActual, poolMeta.thresholds[i]),
"pool is not balanced"
);
}
}
function _isWithinThreshold(
uint256 a,
uint256 b,
uint256 imbalanceTreshold
) internal pure returns (bool) {
if (imbalanceTreshold == 0) imbalanceTreshold = _DEFAULT_IMBALANCE_THRESHOLD;
if (a > b) return (a - b).divDown(a) <= imbalanceTreshold;
return (b - a).divDown(b) <= imbalanceTreshold;
}
}
文件 4 的 27: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 的 27: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);
}
文件 6 的 27:ICNCLockerV2.sol
pragma solidity 0.8.17;
import "../../libraries/MerkleProof.sol";
interface ICNCLockerV2 {
event Locked(address indexed account, uint256 amount, uint256 unlockTime, bool relocked);
event UnlockExecuted(address indexed account, uint256 amount);
event Relocked(address indexed account, uint256 amount);
event KickExecuted(address indexed account, address indexed kicker, uint256 amount);
event FeesReceived(address indexed sender, uint256 crvAmount, uint256 cvxAmount);
event FeesClaimed(address indexed claimer, uint256 crvAmount, uint256 cvxAmount);
event AirdropBoostClaimed(address indexed claimer, uint256 amount);
event Shutdown();
event TokenRecovered(address indexed token);
struct VoteLock {
uint256 amount;
uint64 unlockTime;
uint128 boost;
uint64 id;
}
function lock(uint256 amount, uint64 lockTime) external;
function lock(
uint256 amount,
uint64 lockTime,
bool relock
) external;
function lockFor(
uint256 amount,
uint64 lockTime,
bool relock,
address account
) external;
function relock(uint64 lockId, uint64 lockTime) external;
function relock(uint64 lockTime) external;
function relockMultiple(uint64[] calldata lockIds, uint64 lockTime) external;
function totalBoosted() external view returns (uint256);
function shutDown() external;
function recoverToken(address token) external;
function executeAvailableUnlocks() external returns (uint256);
function executeAvailableUnlocksFor(address dst) external returns (uint256);
function executeUnlocks(address dst, uint64[] calldata lockIds) external returns (uint256);
function claimAirdropBoost(uint256 amount, MerkleProof.Proof calldata proof) external;
function balanceOf(address user) external view returns (uint256);
function unlockableBalance(address user) external view returns (uint256);
function unlockableBalanceBoosted(address user) external view returns (uint256);
function kick(address user, uint64 lockId) external;
function receiveFees(uint256 amountCrv, uint256 amountCvx) external;
function claimableFees(address account)
external
view
returns (uint256 claimableCrv, uint256 claimableCvx);
function claimFees() external returns (uint256 crvAmount, uint256 cvxAmount);
function computeBoost(uint128 lockTime) external view returns (uint128);
function airdropBoost(address account) external view returns (uint256);
function claimedAirdrop(address account) external view returns (bool);
function totalVoteBoost(address account) external view returns (uint256);
function totalRewardsBoost(address account) external view returns (uint256);
function userLocks(address account) external view returns (VoteLock[] memory);
}
文件 7 的 27:IConicPool.sol
pragma solidity 0.8.17;
import "./ILpToken.sol";
import "./IRewardManager.sol";
import "../IOracle.sol";
interface IConicPool {
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);
struct PoolWeight {
address poolAddress;
uint256 weight;
}
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 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 allCurvePools() external view returns (address[] memory);
function curvePoolsCount() external view returns (uint256);
function getCurvePoolAtIndex(uint256 _index) external view returns (address);
function unstakeAndWithdraw(uint256 _amount, uint256 _minAmount) external returns (uint256);
function withdraw(uint256 _amount, uint256 _minAmount) external returns (uint256);
function updateWeights(PoolWeight[] memory poolWeights) external;
function getWeight(address curvePool) external view returns (uint256);
function getWeights() external view returns (PoolWeight[] memory);
function getAllocatedUnderlying() external view returns (PoolWithAmount[] memory);
function removeCurvePool(address pool) external;
function addCurvePool(address pool) external;
function totalCurveLpBalance(address curvePool_) external view returns (uint256);
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 handleInvalidConvexPid(address pool) external;
function shutdownPool() external;
function isShutdown() external view returns (bool);
function handleDepeggedCurvePool(address curvePool_) external;
function isBalanced() external view returns (bool);
}
文件 8 的 27:IController.sol
pragma solidity 0.8.17;
import "./pools/IConicPool.sol";
import "./IOracle.sol";
import "./tokenomics/IInflationManager.sol";
import "./tokenomics/ILpTokenStaker.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 PriceOracleSet(address priceOracle);
event WeightUpdateMinDelaySet(uint256 weightUpdateMinDelay);
struct WeightUpdate {
address conicPoolAddress;
IConicPool.PoolWeight[] weights;
}
function inflationManager() external view returns (IInflationManager);
function setInflationManager(address manager) external;
function curveRegistryCache() external view returns (ICurveRegistryCache);
function setLpTokenStaker(address _lpTokenStaker) external;
function lpTokenStaker() external view returns (ILpTokenStaker);
function priceOracle() external view returns (IOracle);
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 emergencyMinter() external view returns (address);
function setWeightUpdateMinDelay(uint256 delay) external;
}
文件 9 的 27:IConvexHandler.sol
pragma solidity 0.8.17;
interface IConvexHandler {
function deposit(address _curvePool, uint256 _amount) external;
function claimBatchEarnings(address[] memory _curvePools, 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);
}
文件 10 的 27:ICurveHandler.sol
pragma solidity 0.8.17;
interface ICurveHandler {
function deposit(
address _curvePool,
address _token,
uint256 _amount
) external;
function withdraw(
address _curvePool,
address _token,
uint256 _amount
) external;
}
文件 11 的 27: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;
}
文件 12 的 27: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 的 27:ICurveRegistryCache.sol
pragma solidity 0.8.17;
import "./vendor/IBooster.sol";
import "../libraries/CurvePoolUtils.sol";
interface ICurveRegistryCache {
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);
}
文件 14 的 27: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 的 27: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 的 27: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 hasPoolRebalancingRewardHandlers(address poolAddress, address handler)
external
view
returns (bool);
}
文件 17 的 27:ILpToken.sol
pragma solidity 0.8.17;
import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
interface ILpToken is IERC20Metadata {
function mint(address account, uint256 amount) external returns (uint256);
function burn(address _owner, uint256 _amount) external returns (uint256);
}
文件 18 的 27: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);
}
文件 19 的 27: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);
}
文件 20 的 27: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
);
struct RewardMeta {
uint256 earnedIntegral;
uint256 lastEarned;
mapping(address => uint256) accountIntegral;
mapping(address => uint256) accountShare;
}
function accountCheckpoint(address account) external;
function poolCheckpoint() external returns (bool);
function addExtraReward(address reward) external returns (bool);
function addBatchExtraRewards(address[] memory rewards) external;
function pool() 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;
}
文件 21 的 27:MerkleProof.sol
pragma solidity 0.8.17;
library MerkleProof {
struct Proof {
uint16 nodeIndex;
bytes32[] hashes;
}
function isValid(
Proof memory proof,
bytes32 node,
bytes32 merkleRoot
) internal pure returns (bool) {
uint256 length = proof.hashes.length;
uint16 nodeIndex = proof.nodeIndex;
for (uint256 i = 0; i < length; i++) {
if (nodeIndex % 2 == 0) {
node = keccak256(abi.encodePacked(node, proof.hashes[i]));
} else {
node = keccak256(abi.encodePacked(proof.hashes[i], node));
}
nodeIndex /= 2;
}
return node == merkleRoot;
}
}
文件 22 的 27: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);
}
}
文件 23 的 27:RewardManager.sol
pragma solidity 0.8.17;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
import "../interfaces/pools/IConicPool.sol";
import "../interfaces/pools/ILpToken.sol";
import "../interfaces/pools/IRewardManager.sol";
import "../interfaces/IConvexHandler.sol";
import "../interfaces/ICurveHandler.sol";
import "../interfaces/IController.sol";
import "../interfaces/IOracle.sol";
import "../interfaces/tokenomics/IInflationManager.sol";
import "../interfaces/tokenomics/ILpTokenStaker.sol";
import "../interfaces/tokenomics/ICNCLockerV2.sol";
import "../interfaces/vendor/ICurvePoolV2.sol";
import "../interfaces/vendor/UniswapRouter02.sol";
import "../libraries/ScaledMath.sol";
contract RewardManager is IRewardManager, Ownable {
using ScaledMath for uint256;
using SafeERC20 for IERC20;
using EnumerableSet for EnumerableSet.AddressSet;
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 immutable override pool;
ILpToken public immutable lpToken;
IERC20 public immutable underlying;
IController public immutable controller;
ICNCLockerV2 public immutable locker;
uint256 internal _oldCrvBalance;
uint256 internal _oldCvxBalance;
EnumerableSet.AddressSet internal _extraRewards;
mapping(address => address) public extraRewardsCurvePool;
mapping(bytes32 => RewardMeta) internal _rewardsMeta;
bool public feesEnabled;
uint256 public feePercentage;
constructor(
address _controller,
address _pool,
address _lpToken,
address _underlying,
address cncLocker
) {
pool = _pool;
lpToken = ILpToken(_lpToken);
underlying = IERC20(_underlying);
controller = IController(_controller);
WETH.safeApprove(address(CNC_ETH_POOL), type(uint256).max);
locker = ICNCLockerV2(cncLocker);
}
function poolCheckpoint() public override returns (bool) {
IConvexHandler convexHandler = IConvexHandler(controller.convexHandler());
(uint256 crvEarned, uint256 cvxEarned, uint256 cncEarned) = _getEarnedRewards(
convexHandler
);
uint256 crvFee;
uint256 cvxFee;
if (feesEnabled) {
crvFee = crvEarned.mulDown(feePercentage);
cvxFee = cvxEarned.mulDown(feePercentage);
crvEarned = crvEarned - crvFee;
cvxEarned = cvxEarned - cvxFee;
}
uint256 _totalStaked = controller.lpTokenStaker().getBalanceForPool(pool);
if (_totalStaked > 0) {
_updateEarned(_CVX_KEY, cvxEarned, _totalStaked);
_updateEarned(_CRV_KEY, crvEarned, _totalStaked);
_updateEarned(_CNC_KEY, cncEarned, _totalStaked);
}
if (!feesEnabled) {
return false;
}
bool rewardsClaimed = false;
if (crvFee > CRV.balanceOf(pool) || cvxFee > CVX.balanceOf(pool)) {
_claimPoolEarningsAndSellRewardTokens();
rewardsClaimed = true;
}
CRV.safeTransferFrom(pool, address(this), crvFee);
CVX.safeTransferFrom(pool, address(this), cvxFee);
CRV.safeApprove(address(locker), crvFee);
CVX.safeApprove(address(locker), cvxFee);
locker.receiveFees(crvFee, cvxFee);
_oldCrvBalance = CRV.balanceOf(pool);
_oldCvxBalance = CVX.balanceOf(pool);
return rewardsClaimed;
}
function _updateEarned(
bytes32 key,
uint256 earned,
uint256 _totalSupply
) internal {
_rewardsMeta[key].earnedIntegral += earned.divDown(_totalSupply);
_rewardsMeta[key].lastEarned += earned;
}
function _getEarnedRewards()
internal
view
returns (
uint256 crvEarned,
uint256 cvxEarned,
uint256 cncEarned
)
{
IConvexHandler convexHandler = IConvexHandler(controller.convexHandler());
return _getEarnedRewards(convexHandler);
}
function _getEarnedRewards(IConvexHandler convexHandler)
internal
view
returns (
uint256 crvEarned,
uint256 cvxEarned,
uint256 cncEarned
)
{
address[] memory curvePools = IConicPool(pool).allCurvePools();
uint256 claimableCRV = convexHandler.getCrvEarnedBatch(pool, curvePools);
uint256 totalCRVEarned = CRV.balanceOf(pool) - _oldCrvBalance + claimableCRV;
uint256 claimableCVX = convexHandler.computeClaimableConvex(claimableCRV);
uint256 totalCVXEarned = CVX.balanceOf(pool) - _oldCvxBalance + claimableCVX;
uint256 totalCNCEarned = controller.lpTokenStaker().claimableCnc(pool);
if (totalCRVEarned > _rewardsMeta[_CRV_KEY].lastEarned) {
crvEarned = totalCRVEarned - _rewardsMeta[_CRV_KEY].lastEarned;
}
if (totalCVXEarned > _rewardsMeta[_CVX_KEY].lastEarned) {
cvxEarned = totalCVXEarned - _rewardsMeta[_CVX_KEY].lastEarned;
}
if (totalCNCEarned > _rewardsMeta[_CNC_KEY].lastEarned) {
cncEarned = totalCNCEarned - _rewardsMeta[_CNC_KEY].lastEarned;
}
}
function accountCheckpoint(address account) external {
_accountCheckpoint(account);
}
function _accountCheckpoint(address account) internal {
uint256 accountBalance = controller.lpTokenStaker().getUserBalanceForPool(pool, 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(pool) ||
cvxAmount > CVX.balanceOf(pool) ||
cncAmount > CNC.balanceOf(pool)
) {
_claimPoolEarningsAndSellRewardTokens();
}
_rewardsMeta[_CNC_KEY].accountShare[msg.sender] = 0;
_rewardsMeta[_CVX_KEY].accountShare[msg.sender] = 0;
_rewardsMeta[_CRV_KEY].accountShare[msg.sender] = 0;
CRV.safeTransferFrom(pool, msg.sender, crvAmount);
CVX.safeTransferFrom(pool, msg.sender, cvxAmount);
CNC.safeTransferFrom(pool, msg.sender, cncAmount);
_oldCrvBalance = CRV.balanceOf(pool);
_oldCvxBalance = CVX.balanceOf(pool);
emit EarningsClaimed(msg.sender, cncAmount, crvAmount, cvxAmount);
return (cncAmount, crvAmount, cvxAmount);
}
function claimPoolEarningsAndSellRewardTokens() external override {
if (!poolCheckpoint()) {
_claimPoolEarningsAndSellRewardTokens();
_oldCrvBalance = CRV.balanceOf(pool);
_oldCvxBalance = CVX.balanceOf(pool);
}
}
function _claimPoolEarningsAndSellRewardTokens() internal {
_rewardsMeta[_CRV_KEY].lastEarned = 0;
_rewardsMeta[_CVX_KEY].lastEarned = 0;
_rewardsMeta[_CNC_KEY].lastEarned = 0;
_claimPoolEarnings();
uint256 cncBalanceBefore_ = CNC.balanceOf(pool);
_sellRewardTokens();
uint256 receivedCnc_ = CNC.balanceOf(pool) - cncBalanceBefore_;
uint256 _totalStaked = controller.lpTokenStaker().getBalanceForPool(pool);
if (_totalStaked > 0)
_rewardsMeta[_CNC_KEY].earnedIntegral += receivedCnc_.divDown(_totalStaked);
emit SoldRewardTokens(receivedCnc_);
}
function _claimPoolEarnings() internal {
controller.lpTokenStaker().claimCNCRewardsForPool(pool);
uint256 cvxBalance = CVX.balanceOf(pool);
uint256 crvBalance = CRV.balanceOf(pool);
address convexHandler = controller.convexHandler();
IConvexHandler(convexHandler).claimBatchEarnings(IConicPool(pool).allCurvePools(), pool);
uint256 claimedCvx = CVX.balanceOf(pool) - cvxBalance;
uint256 claimedCrv = CRV.balanceOf(pool) - crvBalance;
emit ClaimedRewards(claimedCrv, claimedCvx);
}
function _sellRewardTokens() internal {
uint256 extraRewardsLength_ = _extraRewards.length();
if (extraRewardsLength_ == 0) return;
for (uint256 i; i < extraRewardsLength_; i++) {
_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(
reward != address(CVX) &&
reward != address(CRV) &&
reward != address(underlying) &&
reward != address(CNC),
"token not allowed"
);
address[] memory curvePools_ = IConicPool(pool).allCurvePools();
for (uint256 i; i < curvePools_.length; i++) {
address curveLpToken_ = controller.curveRegistryCache().lpToken(curvePools_[i]);
require(reward != curveLpToken_, "token not allowed");
}
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 {
_extraRewards.remove(tokenAddress);
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(locker.totalBoosted() > 0);
feePercentage = _feePercentage;
feesEnabled = true;
emit FeesSet(feePercentage);
}
function claimableRewards(address account)
external
view
returns (
uint256 cncRewards,
uint256 crvRewards,
uint256 cvxRewards
)
{
uint256 _totalStaked = controller.lpTokenStaker().getBalanceForPool(pool);
if (_totalStaked == 0) return (0, 0, 0);
(uint256 crvEarned, uint256 cvxEarned, uint256 cncEarned) = _getEarnedRewards();
uint256 userBalance = controller.lpTokenStaker().getUserBalanceForPool(pool, 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 (deductFee) {
integral += earned.divDown(_totalSupply).mulDown(ScaledMath.ONE - feePercentage);
} 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,
pool
);
}
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);
}
}
文件 24 的 27:SafeERC20.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/draft-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 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 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");
if (returndata.length > 0) {
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}
文件 25 的 27: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;
}
}
}
文件 26 的 27: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);
}
文件 27 的 27:draft-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);
}
{
"compilationTarget": {
"contracts/RewardManager.sol": "RewardManager"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":@chainlink/contracts/=node_modules/@chainlink/contracts/src/v0.8/",
":@openzeppelin/=node_modules/@openzeppelin/",
":ds-test/=lib/forge-std/lib/ds-test/src/",
":forge-std/=lib/forge-std/src/"
]
}
[{"inputs":[{"internalType":"address","name":"_controller","type":"address"},{"internalType":"address","name":"_pool","type":"address"},{"internalType":"address","name":"_lpToken","type":"address"},{"internalType":"address","name":"_underlying","type":"address"},{"internalType":"address","name":"cncLocker","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":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":"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":[],"name":"listExtraRewards","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"locker","outputs":[{"internalType":"contract ICNCLockerV2","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lpToken","outputs":[{"internalType":"contract ILpToken","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pool","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"}]