Overview
S Balance
0 S
S Value
-More Info
Private Name Tags
ContractCreator
Latest 13 from a total of 13 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Set Gov | 673681 | 5 days ago | IN | 0 S | 0.00002974 | ||||
Set Token Config | 665331 | 5 days ago | IN | 0 S | 0.00025354 | ||||
Set Token Config | 665330 | 5 days ago | IN | 0 S | 0.00029511 | ||||
Set Liquidator | 664413 | 5 days ago | IN | 0 S | 0.00005166 | ||||
Set Manager | 664367 | 5 days ago | IN | 0 S | 0.00005132 | ||||
Set Max Leverage | 664346 | 5 days ago | IN | 0 S | 0.00003182 | ||||
Set Vault Utils | 664335 | 5 days ago | IN | 0 S | 0.00003239 | ||||
Set Error Contro... | 664329 | 5 days ago | IN | 0 S | 0.0000508 | ||||
Set Fees | 664327 | 5 days ago | IN | 0 S | 0.0001041 | ||||
Set In Private L... | 664326 | 5 days ago | IN | 0 S | 0.00003212 | ||||
Set In Manager M... | 664324 | 5 days ago | IN | 0 S | 0.00003188 | ||||
Set Funding Rate | 664322 | 5 days ago | IN | 0 S | 0.00003721 | ||||
Initialize | 664318 | 5 days ago | IN | 0 S | 0.00018018 |
Loading...
Loading
Contract Name:
Vault
Compiler Version
v0.6.12+commit.27d51765
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; import "../libraries/math/SafeMath.sol"; import "../libraries/token/IERC20.sol"; import "../libraries/token/SafeERC20.sol"; import "../libraries/utils/ReentrancyGuard.sol"; import "../tokens/interfaces/IUSDN.sol"; import "./interfaces/IVault.sol"; import "./interfaces/IVaultUtils.sol"; import "./interfaces/IVaultPriceFeed.sol"; contract Vault is ReentrancyGuard, IVault { using SafeMath for uint256; using SafeERC20 for IERC20; struct Position { uint256 size; uint256 collateral; uint256 averagePrice; uint256 entryFundingRate; uint256 reserveAmount; int256 realisedPnl; uint256 lastIncreasedTime; } uint256 public constant BASIS_POINTS_DIVISOR = 10000; uint256 public constant FUNDING_RATE_PRECISION = 1000000; uint256 public constant PRICE_PRECISION = 10 ** 30; uint256 public constant MIN_LEVERAGE = 10000; // 1x uint256 public constant USDN_DECIMALS = 18; uint256 public constant MAX_FEE_BASIS_POINTS = 500; // 5% uint256 public constant MAX_LIQUIDATION_FEE_USD = 100 * PRICE_PRECISION; // 100 USD uint256 public constant MIN_FUNDING_RATE_INTERVAL = 1 hours; uint256 public constant MAX_FUNDING_RATE_FACTOR = 10000; // 1% bool public override isInitialized; bool public override isSwapEnabled = true; bool public override isLeverageEnabled = true; IVaultUtils public vaultUtils; address public errorController; address public override router; address public override priceFeed; address public override usdn; address public override gov; uint256 public override whitelistedTokenCount; uint256 public override maxLeverage = 50 * 10000; // 50x uint256 public override liquidationFeeUsd; uint256 public override taxBasisPoints = 50; // 0.5% uint256 public override stableTaxBasisPoints = 20; // 0.2% uint256 public override mintBurnFeeBasisPoints = 30; // 0.3% uint256 public override swapFeeBasisPoints = 30; // 0.3% uint256 public override stableSwapFeeBasisPoints = 4; // 0.04% uint256 public override marginFeeBasisPoints = 10; // 0.1% uint256 public override minProfitTime; bool public override hasDynamicFees = false; uint256 public override fundingInterval = 8 hours; uint256 public override fundingRateFactor; uint256 public override stableFundingRateFactor; uint256 public override totalTokenWeights; bool public includeAmmPrice = true; bool public useSwapPricing = false; bool public override inManagerMode = false; bool public override inPrivateLiquidationMode = false; uint256 public override maxGasPrice; mapping (address => mapping (address => bool)) public override approvedRouters; mapping (address => bool) public override isLiquidator; mapping (address => bool) public override isManager; address[] public override allWhitelistedTokens; mapping (address => bool) public override whitelistedTokens; mapping (address => uint256) public override tokenDecimals; mapping (address => uint256) public override minProfitBasisPoints; mapping (address => bool) public override stableTokens; mapping (address => bool) public override shortableTokens; // tokenBalances is used only to determine _transferIn values mapping (address => uint256) public override tokenBalances; // tokenWeights allows customisation of index composition mapping (address => uint256) public override tokenWeights; // usdnAmounts tracks the amount of USDN debt for each whitelisted token mapping (address => uint256) public override usdnAmounts; // maxUsdnAmounts allows setting a max amount of USDN debt for a token mapping (address => uint256) public override maxUsdnAmounts; // poolAmounts tracks the number of received tokens that can be used for leverage // this is tracked separately from tokenBalances to exclude funds that are deposited as margin collateral mapping (address => uint256) public override poolAmounts; // reservedAmounts tracks the number of tokens reserved for open leverage positions mapping (address => uint256) public override reservedAmounts; // bufferAmounts allows specification of an amount to exclude from swaps // this can be used to ensure a certain amount of liquidity is available for leverage positions mapping (address => uint256) public override bufferAmounts; // guaranteedUsd tracks the amount of USD that is "guaranteed" by opened leverage positions // this value is used to calculate the redemption values for selling of USDN // this is an estimated amount, it is possible for the actual guaranteed value to be lower // in the case of sudden price decreases, the guaranteed value should be corrected // after liquidations are carried out mapping (address => uint256) public override guaranteedUsd; // cumulativeFundingRates tracks the funding rates based on utilization mapping (address => uint256) public override cumulativeFundingRates; // lastFundingTimes tracks the last time funding was updated for a token mapping (address => uint256) public override lastFundingTimes; // positions tracks all open positions mapping (bytes32 => Position) public positions; // feeReserves tracks the amount of fees per token mapping (address => uint256) public override feeReserves; mapping (address => uint256) public override globalShortSizes; mapping (address => uint256) public override globalShortAveragePrices; mapping (address => uint256) public override maxGlobalShortSizes; mapping (uint256 => string) public errors; event BuyUSDN(address account, address token, uint256 tokenAmount, uint256 usdnAmount, uint256 feeBasisPoints); event SellUSDN(address account, address token, uint256 usdnAmount, uint256 tokenAmount, uint256 feeBasisPoints); event Swap(address account, address tokenIn, address tokenOut, uint256 amountIn, uint256 amountOut, uint256 amountOutAfterFees, uint256 feeBasisPoints); event IncreasePosition( bytes32 key, address account, address collateralToken, address indexToken, uint256 collateralDelta, uint256 sizeDelta, bool isLong, uint256 price, uint256 fee ); event DecreasePosition( bytes32 key, address account, address collateralToken, address indexToken, uint256 collateralDelta, uint256 sizeDelta, bool isLong, uint256 price, uint256 fee ); event LiquidatePosition( bytes32 key, address account, address collateralToken, address indexToken, bool isLong, uint256 size, uint256 collateral, uint256 reserveAmount, int256 realisedPnl, uint256 markPrice ); event UpdatePosition( bytes32 key, uint256 size, uint256 collateral, uint256 averagePrice, uint256 entryFundingRate, uint256 reserveAmount, int256 realisedPnl, uint256 markPrice ); event ClosePosition( bytes32 key, uint256 size, uint256 collateral, uint256 averagePrice, uint256 entryFundingRate, uint256 reserveAmount, int256 realisedPnl ); event UpdateFundingRate(address token, uint256 fundingRate); event UpdatePnl(bytes32 key, bool hasProfit, uint256 delta); event CollectSwapFees(address token, uint256 feeUsd, uint256 feeTokens); event CollectMarginFees(address token, uint256 feeUsd, uint256 feeTokens); event DirectPoolDeposit(address token, uint256 amount); event IncreasePoolAmount(address token, uint256 amount); event DecreasePoolAmount(address token, uint256 amount); event IncreaseUsdnAmount(address token, uint256 amount); event DecreaseUsdnAmount(address token, uint256 amount); event IncreaseReservedAmount(address token, uint256 amount); event DecreaseReservedAmount(address token, uint256 amount); event IncreaseGuaranteedUsd(address token, uint256 amount); event DecreaseGuaranteedUsd(address token, uint256 amount); // once the parameters are verified to be working correctly, // gov should be set to a timelock contract or a governance contract constructor() public { gov = msg.sender; } function initialize( address _router, address _usdn, address _priceFeed, uint256 _liquidationFeeUsd, uint256 _fundingRateFactor, uint256 _stableFundingRateFactor ) external { _onlyGov(); _validate(!isInitialized, 1); isInitialized = true; router = _router; usdn = _usdn; priceFeed = _priceFeed; liquidationFeeUsd = _liquidationFeeUsd; fundingRateFactor = _fundingRateFactor; stableFundingRateFactor = _stableFundingRateFactor; } function setVaultUtils(IVaultUtils _vaultUtils) external override { _onlyGov(); vaultUtils = _vaultUtils; } function setErrorController(address _errorController) external { _onlyGov(); errorController = _errorController; } function setError(uint256 _errorCode, string calldata _error) external override { require(msg.sender == errorController, "Vault: invalid errorController"); errors[_errorCode] = _error; } function allWhitelistedTokensLength() external override view returns (uint256) { return allWhitelistedTokens.length; } function setInManagerMode(bool _inManagerMode) external override { _onlyGov(); inManagerMode = _inManagerMode; } function setManager(address _manager, bool _isManager) external override { _onlyGov(); isManager[_manager] = _isManager; } function setInPrivateLiquidationMode(bool _inPrivateLiquidationMode) external override { _onlyGov(); inPrivateLiquidationMode = _inPrivateLiquidationMode; } function setLiquidator(address _liquidator, bool _isActive) external override { _onlyGov(); isLiquidator[_liquidator] = _isActive; } function setIsSwapEnabled(bool _isSwapEnabled) external override { _onlyGov(); isSwapEnabled = _isSwapEnabled; } function setIsLeverageEnabled(bool _isLeverageEnabled) external override { _onlyGov(); isLeverageEnabled = _isLeverageEnabled; } function setMaxGasPrice(uint256 _maxGasPrice) external override { _onlyGov(); maxGasPrice = _maxGasPrice; } function setGov(address _gov) external { _onlyGov(); gov = _gov; } function setPriceFeed(address _priceFeed) external override { _onlyGov(); priceFeed = _priceFeed; } function setMaxLeverage(uint256 _maxLeverage) external override { _onlyGov(); _validate(_maxLeverage > MIN_LEVERAGE, 2); maxLeverage = _maxLeverage; } function setBufferAmount(address _token, uint256 _amount) external override { _onlyGov(); bufferAmounts[_token] = _amount; } function setMaxGlobalShortSize(address _token, uint256 _amount) external override { _onlyGov(); maxGlobalShortSizes[_token] = _amount; } function setFees( uint256 _taxBasisPoints, uint256 _stableTaxBasisPoints, uint256 _mintBurnFeeBasisPoints, uint256 _swapFeeBasisPoints, uint256 _stableSwapFeeBasisPoints, uint256 _marginFeeBasisPoints, uint256 _liquidationFeeUsd, uint256 _minProfitTime, bool _hasDynamicFees ) external override { _onlyGov(); _validate(_taxBasisPoints <= MAX_FEE_BASIS_POINTS, 3); _validate(_stableTaxBasisPoints <= MAX_FEE_BASIS_POINTS, 4); _validate(_mintBurnFeeBasisPoints <= MAX_FEE_BASIS_POINTS, 5); _validate(_swapFeeBasisPoints <= MAX_FEE_BASIS_POINTS, 6); _validate(_stableSwapFeeBasisPoints <= MAX_FEE_BASIS_POINTS, 7); _validate(_marginFeeBasisPoints <= MAX_FEE_BASIS_POINTS, 8); _validate(_liquidationFeeUsd <= MAX_LIQUIDATION_FEE_USD, 9); taxBasisPoints = _taxBasisPoints; stableTaxBasisPoints = _stableTaxBasisPoints; mintBurnFeeBasisPoints = _mintBurnFeeBasisPoints; swapFeeBasisPoints = _swapFeeBasisPoints; stableSwapFeeBasisPoints = _stableSwapFeeBasisPoints; marginFeeBasisPoints = _marginFeeBasisPoints; liquidationFeeUsd = _liquidationFeeUsd; minProfitTime = _minProfitTime; hasDynamicFees = _hasDynamicFees; } function setFundingRate(uint256 _fundingInterval, uint256 _fundingRateFactor, uint256 _stableFundingRateFactor) external override { _onlyGov(); _validate(_fundingInterval >= MIN_FUNDING_RATE_INTERVAL, 10); _validate(_fundingRateFactor <= MAX_FUNDING_RATE_FACTOR, 11); _validate(_stableFundingRateFactor <= MAX_FUNDING_RATE_FACTOR, 12); fundingInterval = _fundingInterval; fundingRateFactor = _fundingRateFactor; stableFundingRateFactor = _stableFundingRateFactor; } function setTokenConfig( address _token, uint256 _tokenDecimals, uint256 _tokenWeight, uint256 _minProfitBps, uint256 _maxUsdnAmount, bool _isStable, bool _isShortable ) external override { _onlyGov(); // increment token count for the first time if (!whitelistedTokens[_token]) { whitelistedTokenCount = whitelistedTokenCount.add(1); allWhitelistedTokens.push(_token); } uint256 _totalTokenWeights = totalTokenWeights; _totalTokenWeights = _totalTokenWeights.sub(tokenWeights[_token]); whitelistedTokens[_token] = true; tokenDecimals[_token] = _tokenDecimals; tokenWeights[_token] = _tokenWeight; minProfitBasisPoints[_token] = _minProfitBps; maxUsdnAmounts[_token] = _maxUsdnAmount; stableTokens[_token] = _isStable; shortableTokens[_token] = _isShortable; totalTokenWeights = _totalTokenWeights.add(_tokenWeight); // validate price feed getMaxPrice(_token); } function clearTokenConfig(address _token) external { _onlyGov(); _validate(whitelistedTokens[_token], 13); totalTokenWeights = totalTokenWeights.sub(tokenWeights[_token]); delete whitelistedTokens[_token]; delete tokenDecimals[_token]; delete tokenWeights[_token]; delete minProfitBasisPoints[_token]; delete maxUsdnAmounts[_token]; delete stableTokens[_token]; delete shortableTokens[_token]; whitelistedTokenCount = whitelistedTokenCount.sub(1); } function withdrawFees(address _token, address _receiver) external override returns (uint256) { _onlyGov(); uint256 amount = feeReserves[_token]; if(amount == 0) { return 0; } feeReserves[_token] = 0; _transferOut(_token, amount, _receiver); return amount; } function addRouter(address _router) external { approvedRouters[msg.sender][_router] = true; } function removeRouter(address _router) external { approvedRouters[msg.sender][_router] = false; } function setUsdnAmount(address _token, uint256 _amount) external override { _onlyGov(); uint256 usdnAmount = usdnAmounts[_token]; if (_amount > usdnAmount) { _increaseUsdnAmount(_token, _amount.sub(usdnAmount)); return; } _decreaseUsdnAmount(_token, usdnAmount.sub(_amount)); } // the governance controlling this function should have a timelock function upgradeVault(address _newVault, address _token, uint256 _amount) external { _onlyGov(); IERC20(_token).safeTransfer(_newVault, _amount); } // deposit into the pool without minting USDN tokens // useful in allowing the pool to become over-collaterised function directPoolDeposit(address _token) external override nonReentrant { _validate(whitelistedTokens[_token], 14); uint256 tokenAmount = _transferIn(_token); _validate(tokenAmount > 0, 15); _increasePoolAmount(_token, tokenAmount); emit DirectPoolDeposit(_token, tokenAmount); } function buyUSDN(address _token, address _receiver) external override nonReentrant returns (uint256) { _validateManager(); _validate(whitelistedTokens[_token], 16); useSwapPricing = true; uint256 tokenAmount = _transferIn(_token); _validate(tokenAmount > 0, 17); updateCumulativeFundingRate(_token, _token); uint256 price = getMinPrice(_token); uint256 usdnAmount = tokenAmount.mul(price).div(PRICE_PRECISION); usdnAmount = adjustForDecimals(usdnAmount, _token, usdn); _validate(usdnAmount > 0, 18); uint256 feeBasisPoints = vaultUtils.getBuyUsdnFeeBasisPoints(_token, usdnAmount); uint256 amountAfterFees = _collectSwapFees(_token, tokenAmount, feeBasisPoints); uint256 mintAmount = amountAfterFees.mul(price).div(PRICE_PRECISION); mintAmount = adjustForDecimals(mintAmount, _token, usdn); _increaseUsdnAmount(_token, mintAmount); _increasePoolAmount(_token, amountAfterFees); IUSDN(usdn).mint(_receiver, mintAmount); emit BuyUSDN(_receiver, _token, tokenAmount, mintAmount, feeBasisPoints); useSwapPricing = false; return mintAmount; } function sellUSDN(address _token, address _receiver) external override nonReentrant returns (uint256) { _validateManager(); _validate(whitelistedTokens[_token], 19); useSwapPricing = true; uint256 usdnAmount = _transferIn(usdn); _validate(usdnAmount > 0, 20); updateCumulativeFundingRate(_token, _token); uint256 redemptionAmount = getRedemptionAmount(_token, usdnAmount); _validate(redemptionAmount > 0, 21); _decreaseUsdnAmount(_token, usdnAmount); _decreasePoolAmount(_token, redemptionAmount); IUSDN(usdn).burn(address(this), usdnAmount); // the _transferIn call increased the value of tokenBalances[usdn] // usually decreases in token balances are synced by calling _transferOut // however, for usdn, the tokens are burnt, so _updateTokenBalance should // be manually called to record the decrease in tokens _updateTokenBalance(usdn); uint256 feeBasisPoints = vaultUtils.getSellUsdnFeeBasisPoints(_token, usdnAmount); uint256 amountOut = _collectSwapFees(_token, redemptionAmount, feeBasisPoints); _validate(amountOut > 0, 22); _transferOut(_token, amountOut, _receiver); emit SellUSDN(_receiver, _token, usdnAmount, amountOut, feeBasisPoints); useSwapPricing = false; return amountOut; } function swap(address _tokenIn, address _tokenOut, address _receiver) external override nonReentrant returns (uint256) { _validate(isSwapEnabled, 23); _validate(whitelistedTokens[_tokenIn], 24); _validate(whitelistedTokens[_tokenOut], 25); _validate(_tokenIn != _tokenOut, 26); useSwapPricing = true; updateCumulativeFundingRate(_tokenIn, _tokenIn); updateCumulativeFundingRate(_tokenOut, _tokenOut); uint256 amountIn = _transferIn(_tokenIn); _validate(amountIn > 0, 27); uint256 priceIn = getMinPrice(_tokenIn); uint256 priceOut = getMaxPrice(_tokenOut); uint256 amountOut = amountIn.mul(priceIn).div(priceOut); amountOut = adjustForDecimals(amountOut, _tokenIn, _tokenOut); // adjust usdnAmounts by the same usdnAmount as debt is shifted between the assets uint256 usdnAmount = amountIn.mul(priceIn).div(PRICE_PRECISION); usdnAmount = adjustForDecimals(usdnAmount, _tokenIn, usdn); uint256 feeBasisPoints = vaultUtils.getSwapFeeBasisPoints(_tokenIn, _tokenOut, usdnAmount); uint256 amountOutAfterFees = _collectSwapFees(_tokenOut, amountOut, feeBasisPoints); _increaseUsdnAmount(_tokenIn, usdnAmount); _decreaseUsdnAmount(_tokenOut, usdnAmount); _increasePoolAmount(_tokenIn, amountIn); _decreasePoolAmount(_tokenOut, amountOut); _validateBufferAmount(_tokenOut); _transferOut(_tokenOut, amountOutAfterFees, _receiver); emit Swap(_receiver, _tokenIn, _tokenOut, amountIn, amountOut, amountOutAfterFees, feeBasisPoints); useSwapPricing = false; return amountOutAfterFees; } function increasePosition(address _account, address _collateralToken, address _indexToken, uint256 _sizeDelta, bool _isLong) external override nonReentrant { _validate(isLeverageEnabled, 28); _validateGasPrice(); _validateRouter(_account); _validateTokens(_collateralToken, _indexToken, _isLong); vaultUtils.validateIncreasePosition(_account, _collateralToken, _indexToken, _sizeDelta, _isLong); updateCumulativeFundingRate(_collateralToken, _indexToken); bytes32 key = getPositionKey(_account, _collateralToken, _indexToken, _isLong); Position storage position = positions[key]; uint256 price = _isLong ? getMaxPrice(_indexToken) : getMinPrice(_indexToken); if (position.size == 0) { position.averagePrice = price; } if (position.size > 0 && _sizeDelta > 0) { position.averagePrice = getNextAveragePrice(_indexToken, position.size, position.averagePrice, _isLong, price, _sizeDelta, position.lastIncreasedTime); } uint256 fee = _collectMarginFees(_account, _collateralToken, _indexToken, _isLong, _sizeDelta, position.size, position.entryFundingRate); uint256 collateralDelta = _transferIn(_collateralToken); uint256 collateralDeltaUsd = tokenToUsdMin(_collateralToken, collateralDelta); position.collateral = position.collateral.add(collateralDeltaUsd); _validate(position.collateral >= fee, 29); position.collateral = position.collateral.sub(fee); position.entryFundingRate = getEntryFundingRate(_collateralToken, _indexToken, _isLong); position.size = position.size.add(_sizeDelta); position.lastIncreasedTime = block.timestamp; _validate(position.size > 0, 30); _validatePosition(position.size, position.collateral); validateLiquidation(_account, _collateralToken, _indexToken, _isLong, true); // reserve tokens to pay profits on the position uint256 reserveDelta = usdToTokenMax(_collateralToken, _sizeDelta); position.reserveAmount = position.reserveAmount.add(reserveDelta); _increaseReservedAmount(_collateralToken, reserveDelta); if (_isLong) { // guaranteedUsd stores the sum of (position.size - position.collateral) for all positions // if a fee is charged on the collateral then guaranteedUsd should be increased by that fee amount // since (position.size - position.collateral) would have increased by `fee` _increaseGuaranteedUsd(_collateralToken, _sizeDelta.add(fee)); _decreaseGuaranteedUsd(_collateralToken, collateralDeltaUsd); // treat the deposited collateral as part of the pool _increasePoolAmount(_collateralToken, collateralDelta); // fees need to be deducted from the pool since fees are deducted from position.collateral // and collateral is treated as part of the pool _decreasePoolAmount(_collateralToken, usdToTokenMin(_collateralToken, fee)); } else { if (globalShortSizes[_indexToken] == 0) { globalShortAveragePrices[_indexToken] = price; } else { globalShortAveragePrices[_indexToken] = getNextGlobalShortAveragePrice(_indexToken, price, _sizeDelta); } _increaseGlobalShortSize(_indexToken, _sizeDelta); } emit IncreasePosition(key, _account, _collateralToken, _indexToken, collateralDeltaUsd, _sizeDelta, _isLong, price, fee); emit UpdatePosition(key, position.size, position.collateral, position.averagePrice, position.entryFundingRate, position.reserveAmount, position.realisedPnl, price); } function decreasePosition(address _account, address _collateralToken, address _indexToken, uint256 _collateralDelta, uint256 _sizeDelta, bool _isLong, address _receiver) external override nonReentrant returns (uint256) { _validateGasPrice(); _validateRouter(_account); return _decreasePosition(_account, _collateralToken, _indexToken, _collateralDelta, _sizeDelta, _isLong, _receiver); } function _decreasePosition(address _account, address _collateralToken, address _indexToken, uint256 _collateralDelta, uint256 _sizeDelta, bool _isLong, address _receiver) private returns (uint256) { vaultUtils.validateDecreasePosition(_account, _collateralToken, _indexToken, _collateralDelta, _sizeDelta, _isLong, _receiver); updateCumulativeFundingRate(_collateralToken, _indexToken); bytes32 key = getPositionKey(_account, _collateralToken, _indexToken, _isLong); Position storage position = positions[key]; _validate(position.size > 0, 31); _validate(position.size >= _sizeDelta, 32); _validate(position.collateral >= _collateralDelta, 33); uint256 collateral = position.collateral; // scrop variables to avoid stack too deep errors { uint256 reserveDelta = position.reserveAmount.mul(_sizeDelta).div(position.size); position.reserveAmount = position.reserveAmount.sub(reserveDelta); _decreaseReservedAmount(_collateralToken, reserveDelta); } (uint256 usdOut, uint256 usdOutAfterFee) = _reduceCollateral(_account, _collateralToken, _indexToken, _collateralDelta, _sizeDelta, _isLong); if (position.size != _sizeDelta) { position.entryFundingRate = getEntryFundingRate(_collateralToken, _indexToken, _isLong); position.size = position.size.sub(_sizeDelta); _validatePosition(position.size, position.collateral); validateLiquidation(_account, _collateralToken, _indexToken, _isLong, true); if (_isLong) { _increaseGuaranteedUsd(_collateralToken, collateral.sub(position.collateral)); _decreaseGuaranteedUsd(_collateralToken, _sizeDelta); } uint256 price = _isLong ? getMinPrice(_indexToken) : getMaxPrice(_indexToken); emit DecreasePosition(key, _account, _collateralToken, _indexToken, _collateralDelta, _sizeDelta, _isLong, price, usdOut.sub(usdOutAfterFee)); emit UpdatePosition(key, position.size, position.collateral, position.averagePrice, position.entryFundingRate, position.reserveAmount, position.realisedPnl, price); } else { if (_isLong) { _increaseGuaranteedUsd(_collateralToken, collateral); _decreaseGuaranteedUsd(_collateralToken, _sizeDelta); } uint256 price = _isLong ? getMinPrice(_indexToken) : getMaxPrice(_indexToken); emit DecreasePosition(key, _account, _collateralToken, _indexToken, _collateralDelta, _sizeDelta, _isLong, price, usdOut.sub(usdOutAfterFee)); emit ClosePosition(key, position.size, position.collateral, position.averagePrice, position.entryFundingRate, position.reserveAmount, position.realisedPnl); delete positions[key]; } if (!_isLong) { _decreaseGlobalShortSize(_indexToken, _sizeDelta); } if (usdOut > 0) { if (_isLong) { _decreasePoolAmount(_collateralToken, usdToTokenMin(_collateralToken, usdOut)); } uint256 amountOutAfterFees = usdToTokenMin(_collateralToken, usdOutAfterFee); _transferOut(_collateralToken, amountOutAfterFees, _receiver); return amountOutAfterFees; } return 0; } function liquidatePosition(address _account, address _collateralToken, address _indexToken, bool _isLong, address _feeReceiver) external override nonReentrant { if (inPrivateLiquidationMode) { _validate(isLiquidator[msg.sender], 34); } // set includeAmmPrice to false to prevent manipulated liquidations includeAmmPrice = false; updateCumulativeFundingRate(_collateralToken, _indexToken); bytes32 key = getPositionKey(_account, _collateralToken, _indexToken, _isLong); Position memory position = positions[key]; _validate(position.size > 0, 35); (uint256 liquidationState, uint256 marginFees) = validateLiquidation(_account, _collateralToken, _indexToken, _isLong, false); _validate(liquidationState != 0, 36); if (liquidationState == 2) { // max leverage exceeded but there is collateral remaining after deducting losses so decreasePosition instead _decreasePosition(_account, _collateralToken, _indexToken, 0, position.size, _isLong, _account); includeAmmPrice = true; return; } uint256 feeTokens = usdToTokenMin(_collateralToken, marginFees); feeReserves[_collateralToken] = feeReserves[_collateralToken].add(feeTokens); emit CollectMarginFees(_collateralToken, marginFees, feeTokens); _decreaseReservedAmount(_collateralToken, position.reserveAmount); if (_isLong) { _decreaseGuaranteedUsd(_collateralToken, position.size.sub(position.collateral)); _decreasePoolAmount(_collateralToken, usdToTokenMin(_collateralToken, marginFees)); } uint256 markPrice = _isLong ? getMinPrice(_indexToken) : getMaxPrice(_indexToken); emit LiquidatePosition(key, _account, _collateralToken, _indexToken, _isLong, position.size, position.collateral, position.reserveAmount, position.realisedPnl, markPrice); if (!_isLong && marginFees < position.collateral) { uint256 remainingCollateral = position.collateral.sub(marginFees); _increasePoolAmount(_collateralToken, usdToTokenMin(_collateralToken, remainingCollateral)); } if (!_isLong) { _decreaseGlobalShortSize(_indexToken, position.size); } delete positions[key]; // pay the fee receiver using the pool, we assume that in general the liquidated amount should be sufficient to cover // the liquidation fees _decreasePoolAmount(_collateralToken, usdToTokenMin(_collateralToken, liquidationFeeUsd)); _transferOut(_collateralToken, usdToTokenMin(_collateralToken, liquidationFeeUsd), _feeReceiver); includeAmmPrice = true; } // validateLiquidation returns (state, fees) function validateLiquidation(address _account, address _collateralToken, address _indexToken, bool _isLong, bool _raise) override public view returns (uint256, uint256) { return vaultUtils.validateLiquidation(_account, _collateralToken, _indexToken, _isLong, _raise); } function getMaxPrice(address _token) public override view returns (uint256) { return IVaultPriceFeed(priceFeed).getPrice(_token, true, includeAmmPrice, useSwapPricing); } function getMinPrice(address _token) public override view returns (uint256) { return IVaultPriceFeed(priceFeed).getPrice(_token, false, includeAmmPrice, useSwapPricing); } function getRedemptionAmount(address _token, uint256 _usdnAmount) public override view returns (uint256) { uint256 price = getMaxPrice(_token); uint256 redemptionAmount = _usdnAmount.mul(PRICE_PRECISION).div(price); return adjustForDecimals(redemptionAmount, usdn, _token); } function getRedemptionCollateral(address _token) public view returns (uint256) { if (stableTokens[_token]) { return poolAmounts[_token]; } uint256 collateral = usdToTokenMin(_token, guaranteedUsd[_token]); return collateral.add(poolAmounts[_token]).sub(reservedAmounts[_token]); } function getRedemptionCollateralUsd(address _token) public view returns (uint256) { return tokenToUsdMin(_token, getRedemptionCollateral(_token)); } function adjustForDecimals(uint256 _amount, address _tokenDiv, address _tokenMul) public view returns (uint256) { uint256 decimalsDiv = _tokenDiv == usdn ? USDN_DECIMALS : tokenDecimals[_tokenDiv]; uint256 decimalsMul = _tokenMul == usdn ? USDN_DECIMALS : tokenDecimals[_tokenMul]; return _amount.mul(10 ** decimalsMul).div(10 ** decimalsDiv); } function tokenToUsdMin(address _token, uint256 _tokenAmount) public override view returns (uint256) { if (_tokenAmount == 0) { return 0; } uint256 price = getMinPrice(_token); uint256 decimals = tokenDecimals[_token]; return _tokenAmount.mul(price).div(10 ** decimals); } function usdToTokenMax(address _token, uint256 _usdAmount) public view returns (uint256) { if (_usdAmount == 0) { return 0; } return usdToToken(_token, _usdAmount, getMinPrice(_token)); } function usdToTokenMin(address _token, uint256 _usdAmount) public view returns (uint256) { if (_usdAmount == 0) { return 0; } return usdToToken(_token, _usdAmount, getMaxPrice(_token)); } function usdToToken(address _token, uint256 _usdAmount, uint256 _price) public view returns (uint256) { if (_usdAmount == 0) { return 0; } uint256 decimals = tokenDecimals[_token]; return _usdAmount.mul(10 ** decimals).div(_price); } function getPosition(address _account, address _collateralToken, address _indexToken, bool _isLong) public override view returns (uint256, uint256, uint256, uint256, uint256, uint256, bool, uint256) { bytes32 key = getPositionKey(_account, _collateralToken, _indexToken, _isLong); Position memory position = positions[key]; uint256 realisedPnl = position.realisedPnl > 0 ? uint256(position.realisedPnl) : uint256(-position.realisedPnl); return ( position.size, // 0 position.collateral, // 1 position.averagePrice, // 2 position.entryFundingRate, // 3 position.reserveAmount, // 4 realisedPnl, // 5 position.realisedPnl >= 0, // 6 position.lastIncreasedTime // 7 ); } function getPositionKey(address _account, address _collateralToken, address _indexToken, bool _isLong) public pure returns (bytes32) { return keccak256(abi.encodePacked( _account, _collateralToken, _indexToken, _isLong )); } function updateCumulativeFundingRate(address _collateralToken, address _indexToken) public { bool shouldUpdate = vaultUtils.updateCumulativeFundingRate(_collateralToken, _indexToken); if (!shouldUpdate) { return; } if (lastFundingTimes[_collateralToken] == 0) { lastFundingTimes[_collateralToken] = block.timestamp.div(fundingInterval).mul(fundingInterval); return; } if (lastFundingTimes[_collateralToken].add(fundingInterval) > block.timestamp) { return; } uint256 fundingRate = getNextFundingRate(_collateralToken); cumulativeFundingRates[_collateralToken] = cumulativeFundingRates[_collateralToken].add(fundingRate); lastFundingTimes[_collateralToken] = block.timestamp.div(fundingInterval).mul(fundingInterval); emit UpdateFundingRate(_collateralToken, cumulativeFundingRates[_collateralToken]); } function getNextFundingRate(address _token) public override view returns (uint256) { if (lastFundingTimes[_token].add(fundingInterval) > block.timestamp) { return 0; } uint256 intervals = block.timestamp.sub(lastFundingTimes[_token]).div(fundingInterval); uint256 poolAmount = poolAmounts[_token]; if (poolAmount == 0) { return 0; } uint256 _fundingRateFactor = stableTokens[_token] ? stableFundingRateFactor : fundingRateFactor; return _fundingRateFactor.mul(reservedAmounts[_token]).mul(intervals).div(poolAmount); } function getUtilisation(address _token) public view returns (uint256) { uint256 poolAmount = poolAmounts[_token]; if (poolAmount == 0) { return 0; } return reservedAmounts[_token].mul(FUNDING_RATE_PRECISION).div(poolAmount); } function getPositionLeverage(address _account, address _collateralToken, address _indexToken, bool _isLong) public view returns (uint256) { bytes32 key = getPositionKey(_account, _collateralToken, _indexToken, _isLong); Position memory position = positions[key]; _validate(position.collateral > 0, 37); return position.size.mul(BASIS_POINTS_DIVISOR).div(position.collateral); } // for longs: nextAveragePrice = (nextPrice * nextSize)/ (nextSize + delta) // for shorts: nextAveragePrice = (nextPrice * nextSize) / (nextSize - delta) function getNextAveragePrice(address _indexToken, uint256 _size, uint256 _averagePrice, bool _isLong, uint256 _nextPrice, uint256 _sizeDelta, uint256 _lastIncreasedTime) public view returns (uint256) { (bool hasProfit, uint256 delta) = getDelta(_indexToken, _size, _averagePrice, _isLong, _lastIncreasedTime); uint256 nextSize = _size.add(_sizeDelta); uint256 divisor; if (_isLong) { divisor = hasProfit ? nextSize.add(delta) : nextSize.sub(delta); } else { divisor = hasProfit ? nextSize.sub(delta) : nextSize.add(delta); } return _nextPrice.mul(nextSize).div(divisor); } // for longs: nextAveragePrice = (nextPrice * nextSize)/ (nextSize + delta) // for shorts: nextAveragePrice = (nextPrice * nextSize) / (nextSize - delta) function getNextGlobalShortAveragePrice(address _indexToken, uint256 _nextPrice, uint256 _sizeDelta) public view returns (uint256) { uint256 size = globalShortSizes[_indexToken]; uint256 averagePrice = globalShortAveragePrices[_indexToken]; uint256 priceDelta = averagePrice > _nextPrice ? averagePrice.sub(_nextPrice) : _nextPrice.sub(averagePrice); uint256 delta = size.mul(priceDelta).div(averagePrice); bool hasProfit = averagePrice > _nextPrice; uint256 nextSize = size.add(_sizeDelta); uint256 divisor = hasProfit ? nextSize.sub(delta) : nextSize.add(delta); return _nextPrice.mul(nextSize).div(divisor); } function getGlobalShortDelta(address _token) public view returns (bool, uint256) { uint256 size = globalShortSizes[_token]; if (size == 0) { return (false, 0); } uint256 nextPrice = getMaxPrice(_token); uint256 averagePrice = globalShortAveragePrices[_token]; uint256 priceDelta = averagePrice > nextPrice ? averagePrice.sub(nextPrice) : nextPrice.sub(averagePrice); uint256 delta = size.mul(priceDelta).div(averagePrice); bool hasProfit = averagePrice > nextPrice; return (hasProfit, delta); } function getPositionDelta(address _account, address _collateralToken, address _indexToken, bool _isLong) public view returns (bool, uint256) { bytes32 key = getPositionKey(_account, _collateralToken, _indexToken, _isLong); Position memory position = positions[key]; return getDelta(_indexToken, position.size, position.averagePrice, _isLong, position.lastIncreasedTime); } function getDelta(address _indexToken, uint256 _size, uint256 _averagePrice, bool _isLong, uint256 _lastIncreasedTime) public override view returns (bool, uint256) { _validate(_averagePrice > 0, 38); uint256 price = _isLong ? getMinPrice(_indexToken) : getMaxPrice(_indexToken); uint256 priceDelta = _averagePrice > price ? _averagePrice.sub(price) : price.sub(_averagePrice); uint256 delta = _size.mul(priceDelta).div(_averagePrice); bool hasProfit; if (_isLong) { hasProfit = price > _averagePrice; } else { hasProfit = _averagePrice > price; } // if the minProfitTime has passed then there will be no min profit threshold // the min profit threshold helps to prevent front-running issues uint256 minBps = block.timestamp > _lastIncreasedTime.add(minProfitTime) ? 0 : minProfitBasisPoints[_indexToken]; if (hasProfit && delta.mul(BASIS_POINTS_DIVISOR) <= _size.mul(minBps)) { delta = 0; } return (hasProfit, delta); } function getEntryFundingRate(address _collateralToken, address _indexToken, bool _isLong) public view returns (uint256) { return vaultUtils.getEntryFundingRate(_collateralToken, _indexToken, _isLong); } function getFundingFee(address _account, address _collateralToken, address _indexToken, bool _isLong, uint256 _size, uint256 _entryFundingRate) public view returns (uint256) { return vaultUtils.getFundingFee(_account, _collateralToken, _indexToken, _isLong, _size, _entryFundingRate); } function getPositionFee(address _account, address _collateralToken, address _indexToken, bool _isLong, uint256 _sizeDelta) public view returns (uint256) { return vaultUtils.getPositionFee(_account, _collateralToken, _indexToken, _isLong, _sizeDelta); } // cases to consider // 1. initialAmount is far from targetAmount, action increases balance slightly => high rebate // 2. initialAmount is far from targetAmount, action increases balance largely => high rebate // 3. initialAmount is close to targetAmount, action increases balance slightly => low rebate // 4. initialAmount is far from targetAmount, action reduces balance slightly => high tax // 5. initialAmount is far from targetAmount, action reduces balance largely => high tax // 6. initialAmount is close to targetAmount, action reduces balance largely => low tax // 7. initialAmount is above targetAmount, nextAmount is below targetAmount and vice versa // 8. a large swap should have similar fees as the same trade split into multiple smaller swaps function getFeeBasisPoints(address _token, uint256 _usdnDelta, uint256 _feeBasisPoints, uint256 _taxBasisPoints, bool _increment) public override view returns (uint256) { return vaultUtils.getFeeBasisPoints(_token, _usdnDelta, _feeBasisPoints, _taxBasisPoints, _increment); } function getTargetUsdnAmount(address _token) public override view returns (uint256) { uint256 supply = IERC20(usdn).totalSupply(); if (supply == 0) { return 0; } uint256 weight = tokenWeights[_token]; return weight.mul(supply).div(totalTokenWeights); } function _reduceCollateral(address _account, address _collateralToken, address _indexToken, uint256 _collateralDelta, uint256 _sizeDelta, bool _isLong) private returns (uint256, uint256) { bytes32 key = getPositionKey(_account, _collateralToken, _indexToken, _isLong); Position storage position = positions[key]; uint256 fee = _collectMarginFees(_account, _collateralToken, _indexToken, _isLong, _sizeDelta, position.size, position.entryFundingRate); bool hasProfit; uint256 adjustedDelta; // scope variables to avoid stack too deep errors { (bool _hasProfit, uint256 delta) = getDelta(_indexToken, position.size, position.averagePrice, _isLong, position.lastIncreasedTime); hasProfit = _hasProfit; // get the proportional change in pnl adjustedDelta = _sizeDelta.mul(delta).div(position.size); } uint256 usdOut; // transfer profits out if (hasProfit && adjustedDelta > 0) { usdOut = adjustedDelta; position.realisedPnl = position.realisedPnl + int256(adjustedDelta); // pay out realised profits from the pool amount for short positions if (!_isLong) { uint256 tokenAmount = usdToTokenMin(_collateralToken, adjustedDelta); _decreasePoolAmount(_collateralToken, tokenAmount); } } if (!hasProfit && adjustedDelta > 0) { position.collateral = position.collateral.sub(adjustedDelta); // transfer realised losses to the pool for short positions // realised losses for long positions are not transferred here as // _increasePoolAmount was already called in increasePosition for longs if (!_isLong) { uint256 tokenAmount = usdToTokenMin(_collateralToken, adjustedDelta); _increasePoolAmount(_collateralToken, tokenAmount); } position.realisedPnl = position.realisedPnl - int256(adjustedDelta); } // reduce the position's collateral by _collateralDelta // transfer _collateralDelta out if (_collateralDelta > 0) { usdOut = usdOut.add(_collateralDelta); position.collateral = position.collateral.sub(_collateralDelta); } // if the position will be closed, then transfer the remaining collateral out if (position.size == _sizeDelta) { usdOut = usdOut.add(position.collateral); position.collateral = 0; } // if the usdOut is more than the fee then deduct the fee from the usdOut directly // else deduct the fee from the position's collateral uint256 usdOutAfterFee = usdOut; if (usdOut > fee) { usdOutAfterFee = usdOut.sub(fee); } else { position.collateral = position.collateral.sub(fee); if (_isLong) { uint256 feeTokens = usdToTokenMin(_collateralToken, fee); _decreasePoolAmount(_collateralToken, feeTokens); } } emit UpdatePnl(key, hasProfit, adjustedDelta); return (usdOut, usdOutAfterFee); } function _validatePosition(uint256 _size, uint256 _collateral) private view { if (_size == 0) { _validate(_collateral == 0, 39); return; } _validate(_size >= _collateral, 40); } function _validateRouter(address _account) private view { if (msg.sender == _account) { return; } if (msg.sender == router) { return; } _validate(approvedRouters[_account][msg.sender], 41); } function _validateTokens(address _collateralToken, address _indexToken, bool _isLong) private view { if (_isLong) { _validate(_collateralToken == _indexToken, 42); _validate(whitelistedTokens[_collateralToken], 43); _validate(!stableTokens[_collateralToken], 44); return; } _validate(whitelistedTokens[_collateralToken], 45); _validate(stableTokens[_collateralToken], 46); _validate(!stableTokens[_indexToken], 47); _validate(shortableTokens[_indexToken], 48); } function _collectSwapFees(address _token, uint256 _amount, uint256 _feeBasisPoints) private returns (uint256) { uint256 afterFeeAmount = _amount.mul(BASIS_POINTS_DIVISOR.sub(_feeBasisPoints)).div(BASIS_POINTS_DIVISOR); uint256 feeAmount = _amount.sub(afterFeeAmount); feeReserves[_token] = feeReserves[_token].add(feeAmount); emit CollectSwapFees(_token, tokenToUsdMin(_token, feeAmount), feeAmount); return afterFeeAmount; } function _collectMarginFees(address _account, address _collateralToken, address _indexToken, bool _isLong, uint256 _sizeDelta, uint256 _size, uint256 _entryFundingRate) private returns (uint256) { uint256 feeUsd = getPositionFee(_account, _collateralToken, _indexToken, _isLong, _sizeDelta); uint256 fundingFee = getFundingFee(_account, _collateralToken, _indexToken, _isLong, _size, _entryFundingRate); feeUsd = feeUsd.add(fundingFee); uint256 feeTokens = usdToTokenMin(_collateralToken, feeUsd); feeReserves[_collateralToken] = feeReserves[_collateralToken].add(feeTokens); emit CollectMarginFees(_collateralToken, feeUsd, feeTokens); return feeUsd; } function _transferIn(address _token) private returns (uint256) { uint256 prevBalance = tokenBalances[_token]; uint256 nextBalance = IERC20(_token).balanceOf(address(this)); tokenBalances[_token] = nextBalance; return nextBalance.sub(prevBalance); } function _transferOut(address _token, uint256 _amount, address _receiver) private { IERC20(_token).safeTransfer(_receiver, _amount); tokenBalances[_token] = IERC20(_token).balanceOf(address(this)); } function _updateTokenBalance(address _token) private { uint256 nextBalance = IERC20(_token).balanceOf(address(this)); tokenBalances[_token] = nextBalance; } function _increasePoolAmount(address _token, uint256 _amount) private { poolAmounts[_token] = poolAmounts[_token].add(_amount); uint256 balance = IERC20(_token).balanceOf(address(this)); _validate(poolAmounts[_token] <= balance, 49); emit IncreasePoolAmount(_token, _amount); } function _decreasePoolAmount(address _token, uint256 _amount) private { poolAmounts[_token] = poolAmounts[_token].sub(_amount, "Vault: poolAmount exceeded"); _validate(reservedAmounts[_token] <= poolAmounts[_token], 50); emit DecreasePoolAmount(_token, _amount); } function _validateBufferAmount(address _token) private view { if (poolAmounts[_token] < bufferAmounts[_token]) { revert("Vault: poolAmount < buffer"); } } function _increaseUsdnAmount(address _token, uint256 _amount) private { usdnAmounts[_token] = usdnAmounts[_token].add(_amount); uint256 maxUsdnAmount = maxUsdnAmounts[_token]; if (maxUsdnAmount != 0) { _validate(usdnAmounts[_token] <= maxUsdnAmount, 51); } emit IncreaseUsdnAmount(_token, _amount); } function _decreaseUsdnAmount(address _token, uint256 _amount) private { uint256 value = usdnAmounts[_token]; // since USDN can be minted using multiple assets // it is possible for the USDN debt for a single asset to be less than zero // the USDN debt is capped to zero for this case if (value <= _amount) { usdnAmounts[_token] = 0; emit DecreaseUsdnAmount(_token, value); return; } usdnAmounts[_token] = value.sub(_amount); emit DecreaseUsdnAmount(_token, _amount); } function _increaseReservedAmount(address _token, uint256 _amount) private { reservedAmounts[_token] = reservedAmounts[_token].add(_amount); _validate(reservedAmounts[_token] <= poolAmounts[_token], 52); emit IncreaseReservedAmount(_token, _amount); } function _decreaseReservedAmount(address _token, uint256 _amount) private { reservedAmounts[_token] = reservedAmounts[_token].sub(_amount, "Vault: insufficient reserve"); emit DecreaseReservedAmount(_token, _amount); } function _increaseGuaranteedUsd(address _token, uint256 _usdAmount) private { guaranteedUsd[_token] = guaranteedUsd[_token].add(_usdAmount); emit IncreaseGuaranteedUsd(_token, _usdAmount); } function _decreaseGuaranteedUsd(address _token, uint256 _usdAmount) private { guaranteedUsd[_token] = guaranteedUsd[_token].sub(_usdAmount); emit DecreaseGuaranteedUsd(_token, _usdAmount); } function _increaseGlobalShortSize(address _token, uint256 _amount) internal { globalShortSizes[_token] = globalShortSizes[_token].add(_amount); uint256 maxSize = maxGlobalShortSizes[_token]; if (maxSize != 0) { require(globalShortSizes[_token] <= maxSize, "Vault: max shorts exceeded"); } } function _decreaseGlobalShortSize(address _token, uint256 _amount) private { uint256 size = globalShortSizes[_token]; if (_amount > size) { globalShortSizes[_token] = 0; return; } globalShortSizes[_token] = size.sub(_amount); } // we have this validation as a function instead of a modifier to reduce contract size function _onlyGov() private view { _validate(msg.sender == gov, 53); } // we have this validation as a function instead of a modifier to reduce contract size function _validateManager() private view { if (inManagerMode) { _validate(isManager[msg.sender], 54); } } // we have this validation as a function instead of a modifier to reduce contract size function _validateGasPrice() private view { if (maxGasPrice == 0) { return; } _validate(tx.gasprice <= maxGasPrice, 55); } function _validate(bool _condition, uint256 _errorCode) private view { require(_condition, errors[_errorCode]); } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; import "./IVaultUtils.sol"; interface IVault { function isInitialized() external view returns (bool); function isSwapEnabled() external view returns (bool); function isLeverageEnabled() external view returns (bool); function setVaultUtils(IVaultUtils _vaultUtils) external; function setError(uint256 _errorCode, string calldata _error) external; function router() external view returns (address); function usdn() external view returns (address); function gov() external view returns (address); function whitelistedTokenCount() external view returns (uint256); function maxLeverage() external view returns (uint256); function minProfitTime() external view returns (uint256); function hasDynamicFees() external view returns (bool); function fundingInterval() external view returns (uint256); function totalTokenWeights() external view returns (uint256); function getTargetUsdnAmount(address _token) external view returns (uint256); function inManagerMode() external view returns (bool); function inPrivateLiquidationMode() external view returns (bool); function maxGasPrice() external view returns (uint256); function approvedRouters(address _account, address _router) external view returns (bool); function isLiquidator(address _account) external view returns (bool); function isManager(address _account) external view returns (bool); function minProfitBasisPoints(address _token) external view returns (uint256); function tokenBalances(address _token) external view returns (uint256); function lastFundingTimes(address _token) external view returns (uint256); function setMaxLeverage(uint256 _maxLeverage) external; function setInManagerMode(bool _inManagerMode) external; function setManager(address _manager, bool _isManager) external; function setIsSwapEnabled(bool _isSwapEnabled) external; function setIsLeverageEnabled(bool _isLeverageEnabled) external; function setMaxGasPrice(uint256 _maxGasPrice) external; function setUsdnAmount(address _token, uint256 _amount) external; function setBufferAmount(address _token, uint256 _amount) external; function setMaxGlobalShortSize(address _token, uint256 _amount) external; function setInPrivateLiquidationMode(bool _inPrivateLiquidationMode) external; function setLiquidator(address _liquidator, bool _isActive) external; function setFundingRate(uint256 _fundingInterval, uint256 _fundingRateFactor, uint256 _stableFundingRateFactor) external; function setFees( uint256 _taxBasisPoints, uint256 _stableTaxBasisPoints, uint256 _mintBurnFeeBasisPoints, uint256 _swapFeeBasisPoints, uint256 _stableSwapFeeBasisPoints, uint256 _marginFeeBasisPoints, uint256 _liquidationFeeUsd, uint256 _minProfitTime, bool _hasDynamicFees ) external; function setTokenConfig( address _token, uint256 _tokenDecimals, uint256 _redemptionBps, uint256 _minProfitBps, uint256 _maxUsdnAmount, bool _isStable, bool _isShortable ) external; function setPriceFeed(address _priceFeed) external; function withdrawFees(address _token, address _receiver) external returns (uint256); function directPoolDeposit(address _token) external; function buyUSDN(address _token, address _receiver) external returns (uint256); function sellUSDN(address _token, address _receiver) external returns (uint256); function swap(address _tokenIn, address _tokenOut, address _receiver) external returns (uint256); function increasePosition(address _account, address _collateralToken, address _indexToken, uint256 _sizeDelta, bool _isLong) external; function decreasePosition(address _account, address _collateralToken, address _indexToken, uint256 _collateralDelta, uint256 _sizeDelta, bool _isLong, address _receiver) external returns (uint256); function validateLiquidation(address _account, address _collateralToken, address _indexToken, bool _isLong, bool _raise) external view returns (uint256, uint256); function liquidatePosition(address _account, address _collateralToken, address _indexToken, bool _isLong, address _feeReceiver) external; function tokenToUsdMin(address _token, uint256 _tokenAmount) external view returns (uint256); function priceFeed() external view returns (address); function fundingRateFactor() external view returns (uint256); function stableFundingRateFactor() external view returns (uint256); function cumulativeFundingRates(address _token) external view returns (uint256); function getNextFundingRate(address _token) external view returns (uint256); function getFeeBasisPoints(address _token, uint256 _usdnDelta, uint256 _feeBasisPoints, uint256 _taxBasisPoints, bool _increment) external view returns (uint256); function liquidationFeeUsd() external view returns (uint256); function taxBasisPoints() external view returns (uint256); function stableTaxBasisPoints() external view returns (uint256); function mintBurnFeeBasisPoints() external view returns (uint256); function swapFeeBasisPoints() external view returns (uint256); function stableSwapFeeBasisPoints() external view returns (uint256); function marginFeeBasisPoints() external view returns (uint256); function allWhitelistedTokensLength() external view returns (uint256); function allWhitelistedTokens(uint256) external view returns (address); function whitelistedTokens(address _token) external view returns (bool); function stableTokens(address _token) external view returns (bool); function shortableTokens(address _token) external view returns (bool); function feeReserves(address _token) external view returns (uint256); function globalShortSizes(address _token) external view returns (uint256); function globalShortAveragePrices(address _token) external view returns (uint256); function maxGlobalShortSizes(address _token) external view returns (uint256); function tokenDecimals(address _token) external view returns (uint256); function tokenWeights(address _token) external view returns (uint256); function guaranteedUsd(address _token) external view returns (uint256); function poolAmounts(address _token) external view returns (uint256); function bufferAmounts(address _token) external view returns (uint256); function reservedAmounts(address _token) external view returns (uint256); function usdnAmounts(address _token) external view returns (uint256); function maxUsdnAmounts(address _token) external view returns (uint256); function getRedemptionAmount(address _token, uint256 _usdnAmount) external view returns (uint256); function getMaxPrice(address _token) external view returns (uint256); function getMinPrice(address _token) external view returns (uint256); function getDelta(address _indexToken, uint256 _size, uint256 _averagePrice, bool _isLong, uint256 _lastIncreasedTime) external view returns (bool, uint256); function getPosition(address _account, address _collateralToken, address _indexToken, bool _isLong) external view returns (uint256, uint256, uint256, uint256, uint256, uint256, bool, uint256); }
// SPDX-License-Identifier: MIT pragma solidity >=0.6.12; interface IVaultPriceFeed { function adjustmentBasisPoints(address _token) external view returns (uint256); function isAdjustmentAdditive(address _token) external view returns (bool); function pythNetwork() external view returns (address); function setAdjustment(address _token, bool _isAdditive, uint256 _adjustmentBps) external; function setIsSecondaryPriceEnabled(bool _isEnabled) external; function setSpreadBasisPoints(address _token, uint256 _spreadBasisPoints) external; function setSpreadThresholdBasisPoints(uint256 _spreadThresholdBasisPoints) external; function setFavorPrimaryPrice(bool _favorPrimaryPrice) external; function setMaxStrictPriceDeviation(uint256 _maxStrictPriceDeviation) external; function getPrice(address _token, bool _maximise, bool _includeAmmPrice, bool _useSwapPricing) external view returns (uint256); function getLatestPrimaryPrice(address _token) external view returns (uint256); function getPrimaryPrice(address _token, bool _maximise) external view returns (uint256); function setPythNetwork(address _pythNetwork) external; function setTokenConfig( address _token, bytes32 _priceFeed, uint256 _allowedStaleness, bool _isStrictStable ) external; function setSecondaryPriceFeed(address _secondaryPriceFeed) external; }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; interface IVaultUtils { function updateCumulativeFundingRate(address _collateralToken, address _indexToken) external returns (bool); function validateIncreasePosition(address _account, address _collateralToken, address _indexToken, uint256 _sizeDelta, bool _isLong) external view; function validateDecreasePosition(address _account, address _collateralToken, address _indexToken, uint256 _collateralDelta, uint256 _sizeDelta, bool _isLong, address _receiver) external view; function validateLiquidation(address _account, address _collateralToken, address _indexToken, bool _isLong, bool _raise) external view returns (uint256, uint256); function getEntryFundingRate(address _collateralToken, address _indexToken, bool _isLong) external view returns (uint256); function getPositionFee(address _account, address _collateralToken, address _indexToken, bool _isLong, uint256 _sizeDelta) external view returns (uint256); function getFundingFee(address _account, address _collateralToken, address _indexToken, bool _isLong, uint256 _size, uint256 _entryFundingRate) external view returns (uint256); function getBuyUsdnFeeBasisPoints(address _token, uint256 _usdnAmount) external view returns (uint256); function getSellUsdnFeeBasisPoints(address _token, uint256 _usdnAmount) external view returns (uint256); function getSwapFeeBasisPoints(address _tokenIn, address _tokenOut, uint256 _usdnAmount) external view returns (uint256); function getFeeBasisPoints(address _token, uint256 _usdnDelta, uint256 _feeBasisPoints, uint256 _taxBasisPoints, bool _increment) external view returns (uint256); }
// SPDX-License-Identifier: MIT pragma solidity >=0.6.12; /** * @dev Wrappers over Solidity's arithmetic operations with added overflow * checks. * * Arithmetic operations in Solidity wrap on overflow. This can easily result * in bugs, because programmers usually assume that an overflow raises an * error, which is the standard behavior in high level programming languages. * `SafeMath` restores this intuition by reverting the transaction when an * operation overflows. * * Using this library instead of the unchecked operations eliminates an entire * class of bugs, so it's recommended to use it always. */ library SafeMath { /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { uint256 c = a + b; require(c >= a, "SafeMath: addition overflow"); return c; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return sub(a, b, "SafeMath: subtraction overflow"); } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b <= a, errorMessage); uint256 c = a - b; return c; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) { return 0; } uint256 c = a * b; require(c / a == b, "SafeMath: multiplication overflow"); return c; } /** * @dev Returns the integer division of two unsigned integers. Reverts on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return div(a, b, "SafeMath: division by zero"); } /** * @dev Returns the integer division of two unsigned integers. Reverts with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b > 0, errorMessage); uint256 c = a / b; // assert(a == b * c + a % b); // There is no case in which this doesn't hold return c; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return mod(a, b, "SafeMath: modulo by zero"); } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts with custom message when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b != 0, errorMessage); return a % b; } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `recipient`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address recipient, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `sender` to `recipient` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address sender, address recipient, uint256 amount) external returns (bool); /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; import "./IERC20.sol"; import "../math/SafeMath.sol"; import "../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using SafeMath for uint256; 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)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove(IERC20 token, address spender, uint256 value) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' // solhint-disable-next-line max-line-length 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).add(value); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero"); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); if (returndata.length > 0) { // Return data is optional // solhint-disable-next-line max-line-length require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.6.2; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize, which returns 0 for contracts in // construction, since the code is only stored at the end of the // constructor execution. uint256 size; // solhint-disable-next-line no-inline-assembly assembly { size := extcodesize(account) } return size > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); // solhint-disable-next-line avoid-low-level-calls, avoid-call-value (bool success, ) = recipient.call{ value: amount }(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain`call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ 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"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); // solhint-disable-next-line avoid-low-level-calls (bool success, bytes memory returndata) = target.call{ value: value }(data); return _verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data, string memory errorMessage) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); // solhint-disable-next-line avoid-low-level-calls (bool success, bytes memory returndata) = target.staticcall(data); return _verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.3._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.3._ */ function functionDelegateCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); // solhint-disable-next-line avoid-low-level-calls (bool success, bytes memory returndata) = target.delegatecall(data); return _verifyCallResult(success, returndata, errorMessage); } function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly // solhint-disable-next-line no-inline-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; /** * @dev Contract module that helps prevent reentrant calls to a function. * * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier * available, which can be applied to functions to make sure there are no nested * (reentrant) calls to them. * * Note that because there is a single `nonReentrant` guard, functions marked as * `nonReentrant` may not call one another. This can be worked around by making * those functions `private`, and then adding `external` `nonReentrant` entry * points to them. * * TIP: If you would like to learn more about reentrancy and alternative ways * to protect against it, check out our blog post * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul]. */ contract ReentrancyGuard { // Booleans are more expensive than uint256 or any type that takes up a full // word because each write operation emits an extra SLOAD to first read the // slot's contents, replace the bits taken up by the boolean, and then write // back. This is the compiler's defense against contract upgrades and // pointer aliasing, and it cannot be disabled. // The values being non-zero value makes deployment a bit more expensive, // but in exchange the refund on every call to nonReentrant will be lower in // amount. Since refunds are capped to a percentage of the total // transaction's gas, it is best to keep them low in cases like this one, to // increase the likelihood of the full refund coming into effect. uint256 private constant _NOT_ENTERED = 1; uint256 private constant _ENTERED = 2; uint256 private _status; constructor () internal { _status = _NOT_ENTERED; } /** * @dev Prevents a contract from calling itself, directly or indirectly. * Calling a `nonReentrant` function from another `nonReentrant` * function is not supported. It is possible to prevent this from happening * by making the `nonReentrant` function external, and make it call a * `private` function that does the actual work. */ modifier nonReentrant() { // On the first call to nonReentrant, _notEntered will be true require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; _; // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) _status = _NOT_ENTERED; } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; interface IUSDN { function addVault(address _vault) external; function removeVault(address _vault) external; function mint(address _account, uint256 _amount) external; function burn(address _account, uint256 _amount) external; }
{ "evmVersion": "istanbul", "libraries": {}, "metadata": { "bytecodeHash": "ipfs", "useLiteralContent": true }, "optimizer": { "enabled": true, "runs": 14 }, "remappings": [], "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } } }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"tokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"usdnAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeBasisPoints","type":"uint256"}],"name":"BuyUSDN","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"key","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"size","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"collateral","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"averagePrice","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"entryFundingRate","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"reserveAmount","type":"uint256"},{"indexed":false,"internalType":"int256","name":"realisedPnl","type":"int256"}],"name":"ClosePosition","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"feeUsd","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeTokens","type":"uint256"}],"name":"CollectMarginFees","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"feeUsd","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeTokens","type":"uint256"}],"name":"CollectSwapFees","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"DecreaseGuaranteedUsd","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"DecreasePoolAmount","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"key","type":"bytes32"},{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"collateralToken","type":"address"},{"indexed":false,"internalType":"address","name":"indexToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"collateralDelta","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"sizeDelta","type":"uint256"},{"indexed":false,"internalType":"bool","name":"isLong","type":"bool"},{"indexed":false,"internalType":"uint256","name":"price","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"fee","type":"uint256"}],"name":"DecreasePosition","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"DecreaseReservedAmount","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"DecreaseUsdnAmount","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"DirectPoolDeposit","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"IncreaseGuaranteedUsd","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"IncreasePoolAmount","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"key","type":"bytes32"},{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"collateralToken","type":"address"},{"indexed":false,"internalType":"address","name":"indexToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"collateralDelta","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"sizeDelta","type":"uint256"},{"indexed":false,"internalType":"bool","name":"isLong","type":"bool"},{"indexed":false,"internalType":"uint256","name":"price","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"fee","type":"uint256"}],"name":"IncreasePosition","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"IncreaseReservedAmount","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"IncreaseUsdnAmount","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"key","type":"bytes32"},{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"collateralToken","type":"address"},{"indexed":false,"internalType":"address","name":"indexToken","type":"address"},{"indexed":false,"internalType":"bool","name":"isLong","type":"bool"},{"indexed":false,"internalType":"uint256","name":"size","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"collateral","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"reserveAmount","type":"uint256"},{"indexed":false,"internalType":"int256","name":"realisedPnl","type":"int256"},{"indexed":false,"internalType":"uint256","name":"markPrice","type":"uint256"}],"name":"LiquidatePosition","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"usdnAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeBasisPoints","type":"uint256"}],"name":"SellUSDN","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"tokenIn","type":"address"},{"indexed":false,"internalType":"address","name":"tokenOut","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountIn","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amountOut","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amountOutAfterFees","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeBasisPoints","type":"uint256"}],"name":"Swap","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"fundingRate","type":"uint256"}],"name":"UpdateFundingRate","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"key","type":"bytes32"},{"indexed":false,"internalType":"bool","name":"hasProfit","type":"bool"},{"indexed":false,"internalType":"uint256","name":"delta","type":"uint256"}],"name":"UpdatePnl","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"key","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"size","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"collateral","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"averagePrice","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"entryFundingRate","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"reserveAmount","type":"uint256"},{"indexed":false,"internalType":"int256","name":"realisedPnl","type":"int256"},{"indexed":false,"internalType":"uint256","name":"markPrice","type":"uint256"}],"name":"UpdatePosition","type":"event"},{"inputs":[],"name":"BASIS_POINTS_DIVISOR","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"FUNDING_RATE_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_FEE_BASIS_POINTS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_FUNDING_RATE_FACTOR","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_LIQUIDATION_FEE_USD","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MIN_FUNDING_RATE_INTERVAL","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MIN_LEVERAGE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRICE_PRECISION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"USDN_DECIMALS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_router","type":"address"}],"name":"addRouter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_tokenDiv","type":"address"},{"internalType":"address","name":"_tokenMul","type":"address"}],"name":"adjustForDecimals","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"allWhitelistedTokens","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"allWhitelistedTokensLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"approvedRouters","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"bufferAmounts","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"buyUSDN","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"clearTokenConfig","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"cumulativeFundingRates","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"uint256","name":"_collateralDelta","type":"uint256"},{"internalType":"uint256","name":"_sizeDelta","type":"uint256"},{"internalType":"bool","name":"_isLong","type":"bool"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"decreasePosition","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"directPoolDeposit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"errorController","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"errors","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"feeReserves","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"fundingInterval","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"fundingRateFactor","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"uint256","name":"_size","type":"uint256"},{"internalType":"uint256","name":"_averagePrice","type":"uint256"},{"internalType":"bool","name":"_isLong","type":"bool"},{"internalType":"uint256","name":"_lastIncreasedTime","type":"uint256"}],"name":"getDelta","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"}],"name":"getEntryFundingRate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_usdnDelta","type":"uint256"},{"internalType":"uint256","name":"_feeBasisPoints","type":"uint256"},{"internalType":"uint256","name":"_taxBasisPoints","type":"uint256"},{"internalType":"bool","name":"_increment","type":"bool"}],"name":"getFeeBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"},{"internalType":"uint256","name":"_size","type":"uint256"},{"internalType":"uint256","name":"_entryFundingRate","type":"uint256"}],"name":"getFundingFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getGlobalShortDelta","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getMaxPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getMinPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"uint256","name":"_size","type":"uint256"},{"internalType":"uint256","name":"_averagePrice","type":"uint256"},{"internalType":"bool","name":"_isLong","type":"bool"},{"internalType":"uint256","name":"_nextPrice","type":"uint256"},{"internalType":"uint256","name":"_sizeDelta","type":"uint256"},{"internalType":"uint256","name":"_lastIncreasedTime","type":"uint256"}],"name":"getNextAveragePrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getNextFundingRate","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"uint256","name":"_nextPrice","type":"uint256"},{"internalType":"uint256","name":"_sizeDelta","type":"uint256"}],"name":"getNextGlobalShortAveragePrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"}],"name":"getPosition","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bool","name":"","type":"bool"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"}],"name":"getPositionDelta","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"},{"internalType":"uint256","name":"_sizeDelta","type":"uint256"}],"name":"getPositionFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"}],"name":"getPositionKey","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"}],"name":"getPositionLeverage","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_usdnAmount","type":"uint256"}],"name":"getRedemptionAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getRedemptionCollateral","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getRedemptionCollateralUsd","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getTargetUsdnAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"getUtilisation","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"globalShortAveragePrices","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"globalShortSizes","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"gov","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"guaranteedUsd","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"hasDynamicFees","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"inManagerMode","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"inPrivateLiquidationMode","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"includeAmmPrice","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"uint256","name":"_sizeDelta","type":"uint256"},{"internalType":"bool","name":"_isLong","type":"bool"}],"name":"increasePosition","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_router","type":"address"},{"internalType":"address","name":"_usdn","type":"address"},{"internalType":"address","name":"_priceFeed","type":"address"},{"internalType":"uint256","name":"_liquidationFeeUsd","type":"uint256"},{"internalType":"uint256","name":"_fundingRateFactor","type":"uint256"},{"internalType":"uint256","name":"_stableFundingRateFactor","type":"uint256"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"isInitialized","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isLeverageEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"isLiquidator","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"isManager","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isSwapEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"lastFundingTimes","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"},{"internalType":"address","name":"_feeReceiver","type":"address"}],"name":"liquidatePosition","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"liquidationFeeUsd","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"marginFeeBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxGasPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"maxGlobalShortSizes","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxLeverage","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"maxUsdnAmounts","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"minProfitBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"minProfitTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mintBurnFeeBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"poolAmounts","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"name":"positions","outputs":[{"internalType":"uint256","name":"size","type":"uint256"},{"internalType":"uint256","name":"collateral","type":"uint256"},{"internalType":"uint256","name":"averagePrice","type":"uint256"},{"internalType":"uint256","name":"entryFundingRate","type":"uint256"},{"internalType":"uint256","name":"reserveAmount","type":"uint256"},{"internalType":"int256","name":"realisedPnl","type":"int256"},{"internalType":"uint256","name":"lastIncreasedTime","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"priceFeed","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_router","type":"address"}],"name":"removeRouter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"reservedAmounts","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"router","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"sellUSDN","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"setBufferAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_errorCode","type":"uint256"},{"internalType":"string","name":"_error","type":"string"}],"name":"setError","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_errorController","type":"address"}],"name":"setErrorController","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_taxBasisPoints","type":"uint256"},{"internalType":"uint256","name":"_stableTaxBasisPoints","type":"uint256"},{"internalType":"uint256","name":"_mintBurnFeeBasisPoints","type":"uint256"},{"internalType":"uint256","name":"_swapFeeBasisPoints","type":"uint256"},{"internalType":"uint256","name":"_stableSwapFeeBasisPoints","type":"uint256"},{"internalType":"uint256","name":"_marginFeeBasisPoints","type":"uint256"},{"internalType":"uint256","name":"_liquidationFeeUsd","type":"uint256"},{"internalType":"uint256","name":"_minProfitTime","type":"uint256"},{"internalType":"bool","name":"_hasDynamicFees","type":"bool"}],"name":"setFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_fundingInterval","type":"uint256"},{"internalType":"uint256","name":"_fundingRateFactor","type":"uint256"},{"internalType":"uint256","name":"_stableFundingRateFactor","type":"uint256"}],"name":"setFundingRate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_gov","type":"address"}],"name":"setGov","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_inManagerMode","type":"bool"}],"name":"setInManagerMode","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_inPrivateLiquidationMode","type":"bool"}],"name":"setInPrivateLiquidationMode","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_isLeverageEnabled","type":"bool"}],"name":"setIsLeverageEnabled","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_isSwapEnabled","type":"bool"}],"name":"setIsSwapEnabled","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_liquidator","type":"address"},{"internalType":"bool","name":"_isActive","type":"bool"}],"name":"setLiquidator","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_manager","type":"address"},{"internalType":"bool","name":"_isManager","type":"bool"}],"name":"setManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_maxGasPrice","type":"uint256"}],"name":"setMaxGasPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"setMaxGlobalShortSize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_maxLeverage","type":"uint256"}],"name":"setMaxLeverage","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_priceFeed","type":"address"}],"name":"setPriceFeed","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_tokenDecimals","type":"uint256"},{"internalType":"uint256","name":"_tokenWeight","type":"uint256"},{"internalType":"uint256","name":"_minProfitBps","type":"uint256"},{"internalType":"uint256","name":"_maxUsdnAmount","type":"uint256"},{"internalType":"bool","name":"_isStable","type":"bool"},{"internalType":"bool","name":"_isShortable","type":"bool"}],"name":"setTokenConfig","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"setUsdnAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IVaultUtils","name":"_vaultUtils","type":"address"}],"name":"setVaultUtils","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"shortableTokens","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stableFundingRateFactor","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stableSwapFeeBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stableTaxBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"stableTokens","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_tokenIn","type":"address"},{"internalType":"address","name":"_tokenOut","type":"address"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"swap","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"swapFeeBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"taxBasisPoints","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"tokenBalances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"tokenDecimals","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_tokenAmount","type":"uint256"}],"name":"tokenToUsdMin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"tokenWeights","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalTokenWeights","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"}],"name":"updateCumulativeFundingRate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_newVault","type":"address"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"upgradeVault","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_usdAmount","type":"uint256"},{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"usdToToken","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_usdAmount","type":"uint256"}],"name":"usdToTokenMax","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_usdAmount","type":"uint256"}],"name":"usdToTokenMin","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"usdn","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"usdnAmounts","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"useSwapPricing","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_account","type":"address"},{"internalType":"address","name":"_collateralToken","type":"address"},{"internalType":"address","name":"_indexToken","type":"address"},{"internalType":"bool","name":"_isLong","type":"bool"},{"internalType":"bool","name":"_raise","type":"bool"}],"name":"validateLiquidation","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"vaultUtils","outputs":[{"internalType":"contract IVaultUtils","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"whitelistedTokenCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"whitelistedTokens","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_receiver","type":"address"}],"name":"withdrawFees","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
60806040526001805462ff00001961ff00199091166101001716620100001781556207a1206008556032600a9081556014600b55601e600c819055600d556004600e55600f556011805460ff199081169091556170806012556016805463ffffff0019921690921716905534801561007657600080fd5b506001600055600680546001600160a01b03191633179055615f1a806200009e6000396000f3fe608060405234801561001057600080fd5b50600436106105995760003560e01c80638a27d468116102e9578063cea0c3281161018f578063cea0c3281461123d578063cfa2c54314611273578063cfad57a2146112a1578063d2fa635e146112c7578063d3127e63146112e4578063d54d5a9f14611301578063d8d4093314611360578063d8f897c314611368578063d9ac42251461138e578063da76524c14611396578063daf9c210146113e0578063db3555fb14611406578063db97495f1461142c578063dc8f5fac14611478578063de2ea94814611480578063df73a267146114c8578063e124e6d2146114d0578063e468baf0146114f6578063e67f59a714611513578063efa10a6e14611539578063f07456ce14611565578063f07bbf771461158b578063f2555278146115aa578063f3ae2415146115d8578063f887ea40146115fe578063fa12dbc014611606578063fbfded6d14611638578063fce28c1014611666578063fdaf6ac31461166e578063fec27765146116b2578063fed1a606146116d857610599565b80638a27d46814610e8e5780638a39735a1461061a5780638a78daa814610eb75780638ee573ac14610edd5780638f7b840414610f035780639060b1ca14610f295780639331621214610f3157806395082d2514610f695780639698d25a14610f715780639849e41214610f975780639899cd0214610f9f5780639d7432ca14610fcb5780639f392eb314610ffd578063a22f239214611005578063a42ab3d21461100d578063a5e90eee14611039578063a93acac214611067578063aa43eeb61461108d578063ab08c1c6146110b9578063ab2f3ad4146110c1578063ae3302c2146110e7578063b06423f3146110ef578063b136ca49146110f7578063b1cc53ab1461111d578063b364accb14611155578063bd996b4e1461117b578063c3c7b9e9146111a1578063c4f718bf146111c7578063c596d83c146111cf578063c65bc7b1146111d7578063c7e074c3146111fd57610599565b806345a6f3701161044e57806345a6f3701461098057806348d91abf146109d957806348f35cbb14610a1d5780634a3f088d14610a255780634a993ee914610aa65780634befe2ca14610acc5780634d47b30414610ad4578063514ea4bf14610adc57806351723e8214610b31578063523fba7f14610b6f578063529a356f14610b9557806352f55eed14610bbb5780635c07eaab14610be15780635f7bc11914610c215780636092219914610c4757806362287a3214610c755780636274980314610c7d5780636abbe0c814610ca35780636ae0b15414610cab5780636be6026b14610cd157806371089f4d14610cd9578063724e78da14610cff578063728cdbca14610d25578063741bef1a14610d6d5780637a210a2b14610d755780637c2eb9f714610d7d57806381a612d614610d9c57806382a0849014610dc257806384748c9014610e165780638585f4d214610e3c57806388b1fbdf14610e6857610599565b806304fef1db1461059e57806307c58752146105d65780630842b076146105de5780630a48d5a9146105e657806310eb56c214610612578063126082cf1461061a57806312d43a5114610622578063134ca63b14610646578063174d26941461064e578063181e210e146106565780631ce9cb8f1461067257806324b0c04d1461069857806324ca984e146106b957806328e67be5146106df57806329ff9615146107545780632c668ec11461077a5780632d4b0576146107a657806330455ede146107e4578063318bc6891461080357806334c1557d1461061a578063351a964d1461080b578063392e53cd146108135780633c5a6e351461081b5780633de39c11146108695780633e72a2621461087157806340eb38021461087957806342152873146108ca57806342820a10146108fe57806342b60b031461092c5780634453a37414610952575b600080fd5b6105c4600480360360208110156105b457600080fd5b50356001600160a01b031661176a565b60408051918252519081900360200190f35b6105c46117cd565b6105c46117de565b6105c4600480360360408110156105fc57600080fd5b506001600160a01b0381351690602001356117e4565b6105c4611837565b6105c461183d565b61062a611843565b604080516001600160a01b039092168252519081900360200190f35b6105c4611852565b6105c4611858565b61065e61185e565b604080519115158252519081900360200190f35b6105c46004803603602081101561068857600080fd5b50356001600160a01b031661186e565b6106b7600480360360208110156106ae57600080fd5b50351515611880565b005b6106b7600480360360208110156106cf57600080fd5b50356001600160a01b03166118a4565b6106b7600480360360408110156106f557600080fd5b81359190810190604081016020820135600160201b81111561071657600080fd5b82018360208201111561072857600080fd5b803590602001918460018302840111600160201b8311171561074957600080fd5b5090925090506118d5565b6105c46004803603602081101561076a57600080fd5b50356001600160a01b0316611953565b6105c46004803603604081101561079057600080fd5b506001600160a01b038135169060200135611967565b6105c4600480360360808110156107bc57600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156119ac565b6106b7600480360360208110156107fa57600080fd5b50351515611a02565b6105c4611a24565b61065e611a2a565b61065e611a38565b6106b7600480360360e081101561083157600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611a41565b6105c4611b83565b61065e611b89565b6106b7600480360361012081101561089057600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611b98565b6105c4600480360360608110156108e057600080fd5b508035906001600160a01b0360208201358116916040013516611c52565b6105c46004803603604081101561091457600080fd5b506001600160a01b0381358116916020013516611cf4565b61065e6004803603602081101561094257600080fd5b50356001600160a01b0316611f88565b6106b76004803603604081101561096857600080fd5b506001600160a01b0381351690602001351515611f9d565b6109be6004803603608081101561099657600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611fd0565b60408051921515835260208301919091528051918290030190f35b6106b7600480360360a08110156109ef57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060800135151561206e565b61062a6124ad565b610a6360048036036080811015610a3b57600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156124bc565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b6105c460048036036020811015610abc57600080fd5b50356001600160a01b03166125b2565b6105c46125c4565b6105c46125ca565b610af960048036036020811015610af257600080fd5b50356125d0565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b6105c460048036036080811015610b4757600080fd5b506001600160a01b03813581169160208101358216916040820135169060600135151561260d565b6105c460048036036020811015610b8557600080fd5b50356001600160a01b03166126b3565b61065e60048036036020811015610bab57600080fd5b50356001600160a01b03166126c5565b6105c460048036036020811015610bd157600080fd5b50356001600160a01b03166126da565b6109be600480360360a0811015610bf757600080fd5b506001600160a01b03813516906020810135906040810135906060810135151590608001356126ec565b6106b760048036036020811015610c3757600080fd5b50356001600160a01b03166127e4565b61065e60048036036040811015610c5d57600080fd5b506001600160a01b03813581169160200135166128c7565b6105c46128e7565b6105c460048036036020811015610c9357600080fd5b50356001600160a01b03166128ed565b61062a6128ff565b6106b760048036036020811015610cc157600080fd5b50356001600160a01b0316612915565b6105c4612943565b6106b760048036036020811015610cef57600080fd5b50356001600160a01b031661294a565b6106b760048036036020811015610d1557600080fd5b50356001600160a01b031661297e565b6106b7600480360360c0811015610d3b57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a001356129a8565b61062a612a1f565b6105c4612a2e565b6106b760048036036020811015610d9357600080fd5b50351515612a34565b6105c460048036036020811015610db257600080fd5b50356001600160a01b0316612a58565b6105c4600480360360e0811015610dd857600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612b03565b6105c460048036036020811015610e2c57600080fd5b50356001600160a01b0316612b81565b6106b760048036036040811015610e5257600080fd5b506001600160a01b038135169060200135612b93565b6105c460048036036020811015610e7e57600080fd5b50356001600160a01b0316612bb7565b6106b760048036036060811015610ea457600080fd5b5080359060208101359060400135612bc9565b6105c460048036036020811015610ecd57600080fd5b50356001600160a01b0316612c0f565b6105c460048036036020811015610ef357600080fd5b50356001600160a01b0316612c21565b6106b760048036036020811015610f1957600080fd5b50356001600160a01b0316612c33565b61065e612c5d565b6105c460048036036060811015610f4757600080fd5b506001600160a01b038135811691602081013582169160409091013516612c6c565b6105c4612f54565b6105c460048036036020811015610f8757600080fd5b50356001600160a01b0316612f64565b6105c4612f76565b6105c460048036036040811015610fb557600080fd5b506001600160a01b038135169060200135612f7c565b6105c460048036036060811015610fe157600080fd5b506001600160a01b038135169060208101359060400135612f9e565b61065e613047565b6105c4613050565b6105c46004803603604081101561102357600080fd5b506001600160a01b038135169060200135613056565b6106b76004803603604081101561104f57600080fd5b506001600160a01b0381351690602001351515613073565b6105c46004803603602081101561107d57600080fd5b50356001600160a01b03166130a6565b6106b7600480360360408110156110a357600080fd5b506001600160a01b0381351690602001356131a4565b61065e6131ff565b6105c4600480360360208110156110d757600080fd5b50356001600160a01b0316613208565b6105c461321a565b61065e613220565b6105c46004803603602081101561110d57600080fd5b50356001600160a01b031661322e565b6105c46004803603606081101561113357600080fd5b506001600160a01b0381358116916020810135909116906040013515156132cf565b6109be6004803603602081101561116b57600080fd5b50356001600160a01b031661336a565b6105c46004803603602081101561119157600080fd5b50356001600160a01b0316613402565b6105c4600480360360208110156111b757600080fd5b50356001600160a01b0316613414565b6105c4613426565b61062a61342c565b6105c4600480360360208110156111ed57600080fd5b50356001600160a01b031661343b565b6105c4600480360360a081101561121357600080fd5b506001600160a01b038135169060208101359060408101359060608101359060800135151561344d565b6106b76004803603606081101561125357600080fd5b506001600160a01b038135811691602081013590911690604001356134f7565b6105c46004803603604081101561128957600080fd5b506001600160a01b0381358116916020013516613513565b6106b7600480360360208110156112b757600080fd5b50356001600160a01b03166137e3565b6106b7600480360360208110156112dd57600080fd5b503561380d565b6106b7600480360360208110156112fa57600080fd5b503561381a565b611347600480360360a081101561131757600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001351515613836565b6040805192835260208301919091528051918290030190f35b6105c46138ed565b6105c46004803603602081101561137e57600080fd5b50356001600160a01b03166138f2565b6105c4613904565b6105c4600480360360c08110156113ac57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a0013561390a565b61065e600480360360208110156113f657600080fd5b50356001600160a01b03166139be565b61065e6004803603602081101561141c57600080fd5b50356001600160a01b03166139d3565b6105c4600480360360e081101561144257600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c001356139e7565b6105c4613a78565b6106b7600480360360a081101561149657600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613a7e565b6105c4613e56565b6105c4600480360360208110156114e657600080fd5b50356001600160a01b0316613e5c565b61062a6004803603602081101561150c57600080fd5b5035613ed4565b6106b76004803603602081101561152957600080fd5b50356001600160a01b0316613efb565b6106b76004803603604081101561154f57600080fd5b506001600160a01b038135169060200135613fce565b6105c46004803603602081101561157b57600080fd5b50356001600160a01b0316613ff2565b6106b7600480360360208110156115a157600080fd5b50351515614004565b6105c4600480360360408110156115c057600080fd5b506001600160a01b038135811691602001351661402a565b61065e600480360360208110156115ee57600080fd5b50356001600160a01b0316614080565b61062a614095565b6105c46004803603606081101561161c57600080fd5b506001600160a01b0381351690602081013590604001356140a4565b6106b76004803603604081101561164e57600080fd5b506001600160a01b03813581169160200135166140de565b6105c46142b7565b6105c4600480360360a081101561168457600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001356142bd565b6105c4600480360360208110156116c857600080fd5b50356001600160a01b0316614333565b6116f5600480360360208110156116ee57600080fd5b50356143f3565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561172f578181015183820152602001611717565b50505050905090810190601f16801561175c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b038116600090815260256020526040812054806117925760009150506117c8565b6001600160a01b0383166000908152602660205260409020546117c49082906117be90620f424061448e565b906144e7565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117f357506000611831565b60006117fe84612a58565b6001600160a01b0385166000908152601d602052604090205490915061182c600a82900a6117be868561448e565b925050505b92915050565b600b5481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b602c6020526000908152604090205481565b611888614526565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611934576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061194d908383615da9565b50505050565b6000611831826119628461322e565b6117e4565b60008061197384613e5c565b90506000611991826117be8668327cb2734119d3b7a9601e1b61448e565b60055490915061182c9082906001600160a01b031687611c52565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b611a0a614526565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b60015460ff1681565b611a49614526565b6001600160a01b0387166000908152601c602052604090205460ff16611ac657600754611a77906001614541565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611aed908290614599565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420899055601f83528184208054821689151517905591805290912080549091168415151790559050611b6c8187614541565b601555611b7888613e5c565b505050505050505050565b60175481565b60015462010000900460ff1681565b611ba0614526565b611bb06101f48a111560036145db565b611bc06101f489111560046145db565b611bd06101f488111560056145db565b611be06101f487111560066145db565b611bf06101f486111560076145db565b611c006101f485111560086145db565b611c1b6904ee2d6d415b85acef8160201b84111560096145db565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b60055460009081906001600160a01b03858116911614611c8a576001600160a01b0384166000908152601d6020526040902054611c8d565b60125b6005549091506000906001600160a01b03858116911614611cc6576001600160a01b0384166000908152601d6020526040902054611cc9565b60125b9050611ce882600a0a6117be83600a0a8961448e90919063ffffffff16565b925050505b9392505050565b600060026000541415611d3c576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055611d49614687565b6001600160a01b0383166000908152601c6020526040902054611d709060ff1660136145db565b6016805461ff001916610100179055600554600090611d97906001600160a01b03166146b6565b9050611da76000821160146145db565b611db184856140de565b6000611dbd8583611967565b9050611dcd6000821160156145db565b611dd78583614763565b611de18582614845565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b158015611e3457600080fd5b505af1158015611e48573d6000803e3d6000fd5b5050600554611e6292506001600160a01b03169050614918565b60015460408051634869c99560e11b81526001600160a01b03888116600483015260248201869052915160009363010000009004909216916390d3932a91604480820192602092909190829003018186803b158015611ec057600080fd5b505afa158015611ed4573d6000803e3d6000fd5b505050506040513d6020811015611eea57600080fd5b505190506000611efb8784846149b3565b9050611f0b6000821160166145db565b611f16878288614a7b565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517f33bd65d7d58f0e061076c61dff4ed1e2fe1209ea7b1be46bb4977e10f6df696a9181900360a00190a16016805461ff001916905560016000559695505050505050565b601f6020526000908152604090205460ff1681565b611fa5614526565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611fe1878787876119ac565b9050611feb615e27565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c08201819052909261205f92899290919089906126ec565b93509350505094509492505050565b600260005414156120b4576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b60026000556001546120d09062010000900460ff16601c6145db565b6120d8614b22565b6120e185614b3e565b6120ec848483614ba1565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b15801561216057600080fd5b505afa158015612174573d6000803e3d6000fd5b5050505061218284846140de565b6000612190868686856119ac565b6000818152602b60205260408120919250836121b4576121af86612a58565b6121bd565b6121bd86613e5c565b82549091506121ce57600282018190555b8154158015906121de5750600085115b15612204576121fe868360000154846002015487858a88600601546139e7565b60028301555b600061221d898989888a88600001548960030154614cb6565b9050600061222a896146b6565b905060006122388a836117e4565b600186015490915061224a9082614541565b6001860181905561225f90841115601d6145db565b600185015461226e9084614599565b600186015561227e8a8a896132cf565b6003860155845461228f9089614541565b8086554260068701556122a5901515601e6145db565b6122b785600001548660010154614d84565b6122c58b8b8b8a6001613836565b505060006122d38b8a613056565b60048701549091506122e59082614541565b60048701556122f48b82614da8565b87156123395761230d8b6123088b87614541565b614e48565b6123178b83614ec7565b6123218b84614f46565b6123348b61232f8d87612f7c565b614845565b6123a5565b6001600160a01b038a166000908152602d6020526040902054612376576001600160a01b038a166000908152602e6020526040902085905561239b565b6123818a868b612f9e565b6001600160a01b038b166000908152602e60205260409020555b6123a58a8a61505e565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e08201879052517f20853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b773918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b60008060008060008060008060006124d68d8d8d8d6119ac565b90506124e0615e27565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a0015113612560578160a00151600003612566565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b60008061261c868686866119ac565b9050612626615e27565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c083015261269090151560256145db565b602081015181516126a891906117be9061271061448e565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6000806126fd6000861160266145db565b6000846127125761270d88613e5c565b61271b565b61271b88612a58565b90506000818711612735576127308288614599565b61273f565b61273f8783614599565b90506000612751886117be8b8561448e565b9050600087156127645750878311612769565b508288115b60006127806010548961454190919063ffffffff16565b42116127a4576001600160a01b038c166000908152601e60205260409020546127a7565b60005b90508180156127ca57506127bb8b8261448e565b6127c78461271061448e565b11155b156127d457600092505b509a909950975050505050505050565b6002600054141561282a576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546128559060ff16600e6145db565b6000612860826146b6565b905061287060008211600f6145db565b61287a8282614f46565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b612952614526565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b612986614526565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b6129b0614526565b600180546129c49160ff90911615906145db565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b6004546001600160a01b031681565b600a5481565b612a3c614526565b60018054911515620100000262ff000019909216919091179055565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b158015612ad157600080fd5b505afa158015612ae5573d6000803e3d6000fd5b505050506040513d6020811015612afb57600080fd5b505192915050565b600060026000541415612b4b576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055612b58614b22565b612b6188614b3e565b612b7088888888888888615116565b600160005598975050505050505050565b60236020526000908152604090205481565b612b9b614526565b6001600160a01b03909116600090815260276020526040902055565b601e6020526000908152604090205481565b612bd1614526565b612be1610e10841015600a6145db565b612bf1612710831115600b6145db565b612c01612710821115600c6145db565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612c3b614526565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415612cb4576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055600154612ccf90610100900460ff1660176145db565b6001600160a01b0384166000908152601c6020526040902054612cf69060ff1660186145db565b6001600160a01b0383166000908152601c6020526040902054612d1d9060ff1660196145db565b612d3d836001600160a01b0316856001600160a01b03161415601a6145db565b6016805461ff001916610100179055612d5684806140de565b612d6083846140de565b6000612d6b856146b6565b9050612d7b60008211601b6145db565b6000612d8686612a58565b90506000612d9386613e5c565b90506000612da5826117be868661448e565b9050612db2818989611c52565b90506000612dd068327cb2734119d3b7a9601e1b6117be878761448e565b600554909150612dec9082908b906001600160a01b0316611c52565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b158015612e5657600080fd5b505afa158015612e6a573d6000803e3d6000fd5b505050506040513d6020811015612e8057600080fd5b505190506000612e918a85846149b3565b9050612e9d8b846155d9565b612ea78a84614763565b612eb18b88614f46565b612ebb8a85614845565b612ec48a615697565b612ecf8a828b614a7b565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b600081612f8b57506000611831565b611ced8383612f9986613e5c565b6140a4565b6001600160a01b0383166000908152602d6020908152604080832054602e90925282205482858211612fd957612fd48683614599565b612fe3565b612fe38287614599565b90506000612ff5836117be868561448e565b905086831160006130068689614541565b905060008261301e576130198285614541565b613028565b6130288285614599565b9050613038816117be8c8561448e565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b60008161306557506000611831565b611ced8383612f9986612a58565b61307b614526565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a6020526040812054909142916130cf91614541565b11156130dd575060006117c8565b6012546001600160a01b0383166000908152602a6020526040812054909161310a916117be904290614599565b6001600160a01b03841660009081526025602052604090205490915080613136576000925050506117c8565b6001600160a01b0384166000908152601f602052604081205460ff1661315e57601354613162565b6014545b6001600160a01b03861660009081526026602052604090205490915061319b9083906117be90869061319590869061448e565b9061448e565b95945050505050565b6131ac614526565b6001600160a01b038216600090815260236020526040902054808211156131e6576131e0836131db8484614599565b6155d9565b506131fb565b6131f9836131f48385614599565b614763565b505b5050565b60165460ff1681565b60226020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff161561326e57506001600160a01b0381166000908152602560205260409020546117c8565b6001600160a01b038216600090815260286020526040812054613292908490612f7c565b6001600160a01b0384166000908152602660209081526040808320546025909252909120549192506117c4916132c9908490614541565b90614599565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b15801561333657600080fd5b505afa15801561334a573d6000803e3d6000fd5b505050506040513d602081101561336057600080fd5b5051949350505050565b6001600160a01b0381166000908152602d60205260408120548190806133975760008092509250506133fd565b60006133a285613e5c565b6001600160a01b0386166000908152602e60205260408120549192508282116133d4576133cf8383614599565b6133de565b6133de8284614599565b905060006133f0836117be878561448e565b9390921195509193505050505b915091565b60246020526000908152604090205481565b60266020526000908152604090205481565b60135481565b6005546001600160a01b031681565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b1580156134c157600080fd5b505afa1580156134d5573d6000803e3d6000fd5b505050506040513d60208110156134eb57600080fd5b50519695505050505050565b6134ff614526565b6131f96001600160a01b038316848361570c565b60006002600054141561355b576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055613568614687565b6001600160a01b0383166000908152601c602052604090205461358f9060ff1660106145db565b6016805461ff00191661010017905560006135a9846146b6565b90506135b96000821160116145db565b6135c384856140de565b60006135ce85612a58565b905060006135ec68327cb2734119d3b7a9601e1b6117be858561448e565b60055490915061360890829088906001600160a01b0316611c52565b90506136186000821160126145db565b6001546040805163d54b31f160e01b81526001600160a01b038981166004830152602482018590529151600093630100000090049092169163d54b31f191604480820192602092909190829003018186803b15801561367657600080fd5b505afa15801561368a573d6000803e3d6000fd5b505050506040513d60208110156136a057600080fd5b5051905060006136b18886846149b3565b905060006136cf68327cb2734119d3b7a9601e1b6117be848861448e565b6005549091506136eb9082908b906001600160a01b0316611c52565b90506136f789826155d9565b6137018983614f46565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b15801561375657600080fd5b505af115801561376a573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517ff69f1ee12180eb7d5ff48e7f458d08949a51bd7689b2d2995392b40eea45fbac93509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b6137eb614526565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b613815614526565b601755565b613822614526565b613831612710821160026145db565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b1580156138ac57600080fd5b505afa1580156138c0573d6000803e3d6000fd5b505050506040513d60408110156138d657600080fd5b508051602090910151909890975095505050505050565b601281565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b15801561398757600080fd5b505afa15801561399b573d6000803e3d6000fd5b505050506040513d60208110156139b157600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b60008060006139f98a8a8a8a886126ec565b90925090506000613a0a8a87614541565b905060008815613a395783613a2857613a238284614599565b613a32565b613a328284614541565b9050613a5a565b83613a4d57613a488284614541565b613a57565b613a578284614599565b90505b613a68816117be8a8561448e565b9c9b505050505050505050505050565b60155481565b60026000541415613ac4576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613af95733600090815260196020526040902054613af99060ff1660226145db565b6016805460ff19169055613b0d84846140de565b6000613b1b868686866119ac565b9050613b25615e27565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613b9190151560236145db565b600080613ba2898989896000613836565b91509150613bb5826000141560246145db565b8160021415613bea57613bd2898989600087600001518b8f615116565b50506016805460ff1916600117905550613e4a915050565b6000613bf68983612f7c565b6001600160a01b038a166000908152602c6020526040902054909150613c1c9082614541565b6001600160a01b038a166000818152602c602090815260409182902093909355805191825291810184905280820183905290517f5d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011a9181900360600190a1613c8789856080015161575e565b8615613cb65760208401518451613ca8918b91613ca391614599565b614ec7565b613cb68961232f8b85612f7c565b600087613ccb57613cc689613e5c565b613cd4565b613cd489612a58565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613d935750846020015183105b15613dc1576020850151600090613daa9085614599565b9050613dbf8b613dba8d84612f7c565b614f46565b505b87613dd457613dd4898660000151615813565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613e20908b9061232f908290612f7c565b613e368a613e308c600954612f7c565b89614a7b565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b158015612ad157600080fd5b601b8181548110613ee157fe5b6000918252602090912001546001600160a01b0316905081565b613f03614526565b6001600160a01b0381166000908152601c6020526040902054613f2a9060ff16600d6145db565b6001600160a01b038116600090815260226020526040902054601554613f4f91614599565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f835281842080548216905591805290912080549091169055600754613fc8906001614599565b60075550565b613fd6614526565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b61400c614526565b6016805491151563010000000263ff00000019909216919091179055565b6000614034614526565b6001600160a01b0383166000908152602c60205260409020548061405c576000915050611831565b6001600160a01b0384166000908152602c6020526040812055611ced848285614a7b565b601a6020526000908152604090205460ff1681565b6003546001600160a01b031681565b6000826140b357506000611ced565b6001600160a01b0384166000908152601d602052604090205461319b836117be86600a85900a61448e565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b15801561413e57600080fd5b505af1158015614152573d6000803e3d6000fd5b505050506040513d602081101561416857600080fd5b505190508061417757506131fb565b6001600160a01b0383166000908152602a60205260409020546141c4576012546141a59061319542826144e7565b6001600160a01b0384166000908152602a6020526040902055506131fb565b6012546001600160a01b0384166000908152602a602052604090205442916141ec9190614541565b11156141f857506131fb565b6000614203846130a6565b6001600160a01b0385166000908152602960205260409020549091506142299082614541565b6001600160a01b0385166000908152602960205260409020556012546142539061319542826144e7565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b1580156134c157600080fd5b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561438457600080fd5b505afa158015614398573d6000803e3d6000fd5b505050506040513d60208110156143ae57600080fd5b50519050806143c15760009150506117c8565b6001600160a01b0383166000908152602260205260409020546015546143eb906117be838561448e565b949350505050565b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144865780601f1061445b57610100808354040283529160200191614486565b820191906000526020600020905b81548152906001019060200180831161446957829003601f168201915b505050505081565b60008261449d57506000611831565b828202828482816144aa57fe5b0414611ced5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e9a6021913960400191505060405180910390fd5b6000611ced83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b81525061587b565b60065461453f906001600160a01b0316331460356145db565b565b600082820183811015611ced576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b6000611ced83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f77000081525061591d565b6000818152603060205260409020826131f95760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156146785780601f1061464d57610100808354040283529160200191614678565b820191906000526020600020905b81548152906001019060200180831161465b57829003601f168201915b50509250505060405180910390fd5b60165462010000900460ff161561453f57336000908152601a602052604090205461453f9060ff1660366145db565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b15801561471057600080fd5b505afa158015614724573d6000803e3d6000fd5b505050506040513d602081101561473a57600080fd5b50516001600160a01b038516600090815260216020526040902081905590506143eb8183614599565b6001600160a01b0382166000908152602360205260409020548181116147de576001600160a01b0383166000818152602360209081526040808320929092558151928352820183905280517f9b334cd33ac6a46d0c2f047a99c238a41ac13f4dc0d74daf8e9cc81a7837b3119281900390910190a1506131fb565b6147e88183614599565b6001600160a01b03841660008181526023602090815260409182902093909355805191825291810184905281517f9b334cd33ac6a46d0c2f047a99c238a41ac13f4dc0d74daf8e9cc81a7837b311929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b03851660009081526025909152919091205461489d91839061591d565b6001600160a01b038316600090815260256020908152604080832084905560269091529020546148d091101560326145db565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b15801561496757600080fd5b505afa15801561497b573d6000803e3d6000fd5b505050506040513d602081101561499157600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b6000806149d06127106117be6149c98287614599565b879061448e565b905060006149de8583614599565b6001600160a01b0387166000908152602c6020526040902054909150614a049082614541565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b86614a4981846117e4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b614a8f6001600160a01b038416828461570c565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b158015614ad557600080fd5b505afa158015614ae9573d6000803e3d6000fd5b505050506040513d6020811015614aff57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b601754614b2e5761453f565b61453f6017543a111560376145db565b336001600160a01b0382161415614b5457614b9e565b6003546001600160a01b0316331415614b6c57614b9e565b6001600160a01b0381166000908152601860209081526040808320338452909152902054614b9e9060ff1660296145db565b50565b8015614c1a57614bc6826001600160a01b0316846001600160a01b031614602a6145db565b6001600160a01b0383166000908152601c6020526040902054614bed9060ff16602b6145db565b6001600160a01b0383166000908152601f6020526040902054614c159060ff1615602c6145db565b6131f9565b6001600160a01b0383166000908152601c6020526040902054614c419060ff16602d6145db565b6001600160a01b0383166000908152601f6020526040902054614c689060ff16602e6145db565b6001600160a01b0382166000908152601f6020526040902054614c909060ff1615602f6145db565b6001600160a01b03821660009081526020805260409020546131f99060ff1660306145db565b600080614cc689898989896142bd565b90506000614cd88a8a8a8a898961390a565b9050614ce48282614541565b91506000614cf28a84612f7c565b6001600160a01b038b166000908152602c6020526040902054909150614d189082614541565b6001600160a01b038b166000818152602c602090815260409182902093909355805191825291810185905280820183905290517f5d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011a9181900360600190a150909998505050505050505050565b81614d9a57614d95811560276145db565b6131fb565b6131fb8183101560286145db565b6001600160a01b038216600090815260266020526040902054614dcb9082614541565b6001600160a01b038316600090815260266020818152604080842085905560258252909220549152614e0091111560346145db565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614e6b9082614541565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614eea9082614599565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614f699082614541565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614fbf57600080fd5b505afa158015614fd3573d6000803e3d6000fd5b505050506040513d6020811015614fe957600080fd5b50516001600160a01b0384166000908152602560205260409020549091506150159082101560316145db565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b6001600160a01b0382166000908152602d60205260409020546150819082614541565b6001600160a01b0383166000908152602d6020908152604080832093909355602f9052205480156131f9576001600160a01b0383166000908152602d60205260409020548110156131f9576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b15801561519a57600080fd5b505afa1580156151ae573d6000803e3d6000fd5b505050506151bc87876140de565b60006151ca898989876119ac565b6000818152602b602052604090208054919250906151eb901515601f6145db565b6151fd868260000154101560206145db565b61520f878260010154101560216145db565b60018101548154600483015460009161522c916117be908b61448e565b600484015490915061523e9082614599565b600484015561524d8b8261575e565b5060008061525f8d8d8d8d8d8d615977565b855491935091508914615403576152778c8c8a6132cf565b60038501558354615288908a614599565b808555600185015461529a9190614d84565b6152a88d8d8d8b6001613836565b505087156152d5576152cb8c61230886600101548661459990919063ffffffff16565b6152d58c8a614ec7565b6000886152ea576152e58c613e5c565b6152f3565b6152f38c612a58565b90507f93d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30868f8f8f8f8f8f886153288c8c614599565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e08201839052517f20853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b773918190036101000190a150615577565b871561541d576154138c84614e48565b61541d8c8a614ec7565b6000886154325761542d8c613e5c565b61543b565b61543b8c612a58565b90507f93d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30868f8f8f8f8f8f886154708c8c614599565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b87615586576155868b8a615813565b81156155c65787156155a0576155a08c61232f8e85612f7c565b60006155ac8d83612f7c565b90506155b98d828a614a7b565b95506126a8945050505050565b5060009c9b505050505050505050505050565b6001600160a01b0382166000908152602360205260409020546155fc9082614541565b6001600160a01b038316600090815260236020908152604080832093909355602490522054801561564e576001600160a01b03831660009081526023602052604090205461564e9082101560336145db565b604080516001600160a01b03851681526020810184905281517ffb9f1ce4a8a927fbc3811e4804d87f82ebd0ad0b0a9b6bf89f261cfd2cdfd605929181900390910190a1505050565b6001600160a01b0381166000908152602760209081526040808320546025909252909120541015614b9e576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b1790526131f9908490615b71565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b0385166000908152602690915291909120546157b791839061591d565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d60205260409020548082111561585357506001600160a01b0382166000908152602d60205260408120556131fb565b61585d8183614599565b6001600160a01b0384166000908152602d6020526040902055505050565b600081836159075760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b838110156158cc5781810151838201526020016158b4565b50505050905090810190601f1680156158f95780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161591357fe5b0495945050505050565b6000818484111561596f5760405162461bcd60e51b81526020600482018181528351602484015283519092839260449091019190850190808383600083156158cc5781810151838201526020016158b4565b505050900390565b6000806000615988898989876119ac565b6000818152602b602052604081208054600382015493945090926159b5918d918d918d918b918d91614cb6565b90506000806000806159d68d876000015488600201548d8a600601546126ec565b875491955085935091506159ee906117be8d8461448e565b925050506000828015615a015750600082115b15615a315750600584018054820190558088615a31576000615a238e84612f7c565b9050615a2f8e82614845565b505b82158015615a3f5750600082115b15615a83576001850154615a539083614599565b600186015588615a77576000615a698e84612f7c565b9050615a758e82614f46565b505b60058501805483900390555b8a15615aab57615a93818c614541565b6001860154909150615aa5908c614599565b60018601555b84548a1415615ace576001850154615ac4908290614541565b6000600187015590505b8084811115615ae857615ae18286614599565b9050615b1c565b6001860154615af79086614599565b60018701558915615b1c576000615b0e8f87612f7c565b9050615b1a8f82614845565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615bc6826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615c229092919063ffffffff16565b8051909150156131f957808060200190516020811015615be557600080fd5b50516131f95760405162461bcd60e51b815260040180806020018281038252602a815260200180615ebb602a913960400191505060405180910390fd5b60606143eb848460008585615c3685615d3d565b615c87576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615cc65780518252601f199092019160209182019101615ca7565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615d28576040519150601f19603f3d011682016040523d82523d6000602084013e615d2d565b606091505b50915091506126a8828286615d43565b3b151590565b60608315615d52575081611ced565b825115615d625782518084602001fd5b60405162461bcd60e51b81526020600482018181528451602484015284518593919283926044019190850190808383600083156158cc5781810151838201526020016158b4565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615dea5782800160ff19823516178555615e17565b82800160010185558215615e17579182015b82811115615e17578235825591602001919060010190615dfc565b50615e23929150615e64565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615e235760008155600101615e6556fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c00536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f775361666545524332303a204552433230206f7065726174696f6e20646964206e6f742073756363656564a26469706673582212204054febd56ce178902e27acd8e42627811492947af19ff9bc106ace5fcd4f96864736f6c634300060c0033
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106105995760003560e01c80638a27d468116102e9578063cea0c3281161018f578063cea0c3281461123d578063cfa2c54314611273578063cfad57a2146112a1578063d2fa635e146112c7578063d3127e63146112e4578063d54d5a9f14611301578063d8d4093314611360578063d8f897c314611368578063d9ac42251461138e578063da76524c14611396578063daf9c210146113e0578063db3555fb14611406578063db97495f1461142c578063dc8f5fac14611478578063de2ea94814611480578063df73a267146114c8578063e124e6d2146114d0578063e468baf0146114f6578063e67f59a714611513578063efa10a6e14611539578063f07456ce14611565578063f07bbf771461158b578063f2555278146115aa578063f3ae2415146115d8578063f887ea40146115fe578063fa12dbc014611606578063fbfded6d14611638578063fce28c1014611666578063fdaf6ac31461166e578063fec27765146116b2578063fed1a606146116d857610599565b80638a27d46814610e8e5780638a39735a1461061a5780638a78daa814610eb75780638ee573ac14610edd5780638f7b840414610f035780639060b1ca14610f295780639331621214610f3157806395082d2514610f695780639698d25a14610f715780639849e41214610f975780639899cd0214610f9f5780639d7432ca14610fcb5780639f392eb314610ffd578063a22f239214611005578063a42ab3d21461100d578063a5e90eee14611039578063a93acac214611067578063aa43eeb61461108d578063ab08c1c6146110b9578063ab2f3ad4146110c1578063ae3302c2146110e7578063b06423f3146110ef578063b136ca49146110f7578063b1cc53ab1461111d578063b364accb14611155578063bd996b4e1461117b578063c3c7b9e9146111a1578063c4f718bf146111c7578063c596d83c146111cf578063c65bc7b1146111d7578063c7e074c3146111fd57610599565b806345a6f3701161044e57806345a6f3701461098057806348d91abf146109d957806348f35cbb14610a1d5780634a3f088d14610a255780634a993ee914610aa65780634befe2ca14610acc5780634d47b30414610ad4578063514ea4bf14610adc57806351723e8214610b31578063523fba7f14610b6f578063529a356f14610b9557806352f55eed14610bbb5780635c07eaab14610be15780635f7bc11914610c215780636092219914610c4757806362287a3214610c755780636274980314610c7d5780636abbe0c814610ca35780636ae0b15414610cab5780636be6026b14610cd157806371089f4d14610cd9578063724e78da14610cff578063728cdbca14610d25578063741bef1a14610d6d5780637a210a2b14610d755780637c2eb9f714610d7d57806381a612d614610d9c57806382a0849014610dc257806384748c9014610e165780638585f4d214610e3c57806388b1fbdf14610e6857610599565b806304fef1db1461059e57806307c58752146105d65780630842b076146105de5780630a48d5a9146105e657806310eb56c214610612578063126082cf1461061a57806312d43a5114610622578063134ca63b14610646578063174d26941461064e578063181e210e146106565780631ce9cb8f1461067257806324b0c04d1461069857806324ca984e146106b957806328e67be5146106df57806329ff9615146107545780632c668ec11461077a5780632d4b0576146107a657806330455ede146107e4578063318bc6891461080357806334c1557d1461061a578063351a964d1461080b578063392e53cd146108135780633c5a6e351461081b5780633de39c11146108695780633e72a2621461087157806340eb38021461087957806342152873146108ca57806342820a10146108fe57806342b60b031461092c5780634453a37414610952575b600080fd5b6105c4600480360360208110156105b457600080fd5b50356001600160a01b031661176a565b60408051918252519081900360200190f35b6105c46117cd565b6105c46117de565b6105c4600480360360408110156105fc57600080fd5b506001600160a01b0381351690602001356117e4565b6105c4611837565b6105c461183d565b61062a611843565b604080516001600160a01b039092168252519081900360200190f35b6105c4611852565b6105c4611858565b61065e61185e565b604080519115158252519081900360200190f35b6105c46004803603602081101561068857600080fd5b50356001600160a01b031661186e565b6106b7600480360360208110156106ae57600080fd5b50351515611880565b005b6106b7600480360360208110156106cf57600080fd5b50356001600160a01b03166118a4565b6106b7600480360360408110156106f557600080fd5b81359190810190604081016020820135600160201b81111561071657600080fd5b82018360208201111561072857600080fd5b803590602001918460018302840111600160201b8311171561074957600080fd5b5090925090506118d5565b6105c46004803603602081101561076a57600080fd5b50356001600160a01b0316611953565b6105c46004803603604081101561079057600080fd5b506001600160a01b038135169060200135611967565b6105c4600480360360808110156107bc57600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156119ac565b6106b7600480360360208110156107fa57600080fd5b50351515611a02565b6105c4611a24565b61065e611a2a565b61065e611a38565b6106b7600480360360e081101561083157600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611a41565b6105c4611b83565b61065e611b89565b6106b7600480360361012081101561089057600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611b98565b6105c4600480360360608110156108e057600080fd5b508035906001600160a01b0360208201358116916040013516611c52565b6105c46004803603604081101561091457600080fd5b506001600160a01b0381358116916020013516611cf4565b61065e6004803603602081101561094257600080fd5b50356001600160a01b0316611f88565b6106b76004803603604081101561096857600080fd5b506001600160a01b0381351690602001351515611f9d565b6109be6004803603608081101561099657600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611fd0565b60408051921515835260208301919091528051918290030190f35b6106b7600480360360a08110156109ef57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060800135151561206e565b61062a6124ad565b610a6360048036036080811015610a3b57600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156124bc565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b6105c460048036036020811015610abc57600080fd5b50356001600160a01b03166125b2565b6105c46125c4565b6105c46125ca565b610af960048036036020811015610af257600080fd5b50356125d0565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b6105c460048036036080811015610b4757600080fd5b506001600160a01b03813581169160208101358216916040820135169060600135151561260d565b6105c460048036036020811015610b8557600080fd5b50356001600160a01b03166126b3565b61065e60048036036020811015610bab57600080fd5b50356001600160a01b03166126c5565b6105c460048036036020811015610bd157600080fd5b50356001600160a01b03166126da565b6109be600480360360a0811015610bf757600080fd5b506001600160a01b03813516906020810135906040810135906060810135151590608001356126ec565b6106b760048036036020811015610c3757600080fd5b50356001600160a01b03166127e4565b61065e60048036036040811015610c5d57600080fd5b506001600160a01b03813581169160200135166128c7565b6105c46128e7565b6105c460048036036020811015610c9357600080fd5b50356001600160a01b03166128ed565b61062a6128ff565b6106b760048036036020811015610cc157600080fd5b50356001600160a01b0316612915565b6105c4612943565b6106b760048036036020811015610cef57600080fd5b50356001600160a01b031661294a565b6106b760048036036020811015610d1557600080fd5b50356001600160a01b031661297e565b6106b7600480360360c0811015610d3b57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a001356129a8565b61062a612a1f565b6105c4612a2e565b6106b760048036036020811015610d9357600080fd5b50351515612a34565b6105c460048036036020811015610db257600080fd5b50356001600160a01b0316612a58565b6105c4600480360360e0811015610dd857600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612b03565b6105c460048036036020811015610e2c57600080fd5b50356001600160a01b0316612b81565b6106b760048036036040811015610e5257600080fd5b506001600160a01b038135169060200135612b93565b6105c460048036036020811015610e7e57600080fd5b50356001600160a01b0316612bb7565b6106b760048036036060811015610ea457600080fd5b5080359060208101359060400135612bc9565b6105c460048036036020811015610ecd57600080fd5b50356001600160a01b0316612c0f565b6105c460048036036020811015610ef357600080fd5b50356001600160a01b0316612c21565b6106b760048036036020811015610f1957600080fd5b50356001600160a01b0316612c33565b61065e612c5d565b6105c460048036036060811015610f4757600080fd5b506001600160a01b038135811691602081013582169160409091013516612c6c565b6105c4612f54565b6105c460048036036020811015610f8757600080fd5b50356001600160a01b0316612f64565b6105c4612f76565b6105c460048036036040811015610fb557600080fd5b506001600160a01b038135169060200135612f7c565b6105c460048036036060811015610fe157600080fd5b506001600160a01b038135169060208101359060400135612f9e565b61065e613047565b6105c4613050565b6105c46004803603604081101561102357600080fd5b506001600160a01b038135169060200135613056565b6106b76004803603604081101561104f57600080fd5b506001600160a01b0381351690602001351515613073565b6105c46004803603602081101561107d57600080fd5b50356001600160a01b03166130a6565b6106b7600480360360408110156110a357600080fd5b506001600160a01b0381351690602001356131a4565b61065e6131ff565b6105c4600480360360208110156110d757600080fd5b50356001600160a01b0316613208565b6105c461321a565b61065e613220565b6105c46004803603602081101561110d57600080fd5b50356001600160a01b031661322e565b6105c46004803603606081101561113357600080fd5b506001600160a01b0381358116916020810135909116906040013515156132cf565b6109be6004803603602081101561116b57600080fd5b50356001600160a01b031661336a565b6105c46004803603602081101561119157600080fd5b50356001600160a01b0316613402565b6105c4600480360360208110156111b757600080fd5b50356001600160a01b0316613414565b6105c4613426565b61062a61342c565b6105c4600480360360208110156111ed57600080fd5b50356001600160a01b031661343b565b6105c4600480360360a081101561121357600080fd5b506001600160a01b038135169060208101359060408101359060608101359060800135151561344d565b6106b76004803603606081101561125357600080fd5b506001600160a01b038135811691602081013590911690604001356134f7565b6105c46004803603604081101561128957600080fd5b506001600160a01b0381358116916020013516613513565b6106b7600480360360208110156112b757600080fd5b50356001600160a01b03166137e3565b6106b7600480360360208110156112dd57600080fd5b503561380d565b6106b7600480360360208110156112fa57600080fd5b503561381a565b611347600480360360a081101561131757600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001351515613836565b6040805192835260208301919091528051918290030190f35b6105c46138ed565b6105c46004803603602081101561137e57600080fd5b50356001600160a01b03166138f2565b6105c4613904565b6105c4600480360360c08110156113ac57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a0013561390a565b61065e600480360360208110156113f657600080fd5b50356001600160a01b03166139be565b61065e6004803603602081101561141c57600080fd5b50356001600160a01b03166139d3565b6105c4600480360360e081101561144257600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c001356139e7565b6105c4613a78565b6106b7600480360360a081101561149657600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613a7e565b6105c4613e56565b6105c4600480360360208110156114e657600080fd5b50356001600160a01b0316613e5c565b61062a6004803603602081101561150c57600080fd5b5035613ed4565b6106b76004803603602081101561152957600080fd5b50356001600160a01b0316613efb565b6106b76004803603604081101561154f57600080fd5b506001600160a01b038135169060200135613fce565b6105c46004803603602081101561157b57600080fd5b50356001600160a01b0316613ff2565b6106b7600480360360208110156115a157600080fd5b50351515614004565b6105c4600480360360408110156115c057600080fd5b506001600160a01b038135811691602001351661402a565b61065e600480360360208110156115ee57600080fd5b50356001600160a01b0316614080565b61062a614095565b6105c46004803603606081101561161c57600080fd5b506001600160a01b0381351690602081013590604001356140a4565b6106b76004803603604081101561164e57600080fd5b506001600160a01b03813581169160200135166140de565b6105c46142b7565b6105c4600480360360a081101561168457600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001356142bd565b6105c4600480360360208110156116c857600080fd5b50356001600160a01b0316614333565b6116f5600480360360208110156116ee57600080fd5b50356143f3565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561172f578181015183820152602001611717565b50505050905090810190601f16801561175c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b038116600090815260256020526040812054806117925760009150506117c8565b6001600160a01b0383166000908152602660205260409020546117c49082906117be90620f424061448e565b906144e7565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117f357506000611831565b60006117fe84612a58565b6001600160a01b0385166000908152601d602052604090205490915061182c600a82900a6117be868561448e565b925050505b92915050565b600b5481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b602c6020526000908152604090205481565b611888614526565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611934576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061194d908383615da9565b50505050565b6000611831826119628461322e565b6117e4565b60008061197384613e5c565b90506000611991826117be8668327cb2734119d3b7a9601e1b61448e565b60055490915061182c9082906001600160a01b031687611c52565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b611a0a614526565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b60015460ff1681565b611a49614526565b6001600160a01b0387166000908152601c602052604090205460ff16611ac657600754611a77906001614541565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611aed908290614599565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420899055601f83528184208054821689151517905591805290912080549091168415151790559050611b6c8187614541565b601555611b7888613e5c565b505050505050505050565b60175481565b60015462010000900460ff1681565b611ba0614526565b611bb06101f48a111560036145db565b611bc06101f489111560046145db565b611bd06101f488111560056145db565b611be06101f487111560066145db565b611bf06101f486111560076145db565b611c006101f485111560086145db565b611c1b6904ee2d6d415b85acef8160201b84111560096145db565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b60055460009081906001600160a01b03858116911614611c8a576001600160a01b0384166000908152601d6020526040902054611c8d565b60125b6005549091506000906001600160a01b03858116911614611cc6576001600160a01b0384166000908152601d6020526040902054611cc9565b60125b9050611ce882600a0a6117be83600a0a8961448e90919063ffffffff16565b925050505b9392505050565b600060026000541415611d3c576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055611d49614687565b6001600160a01b0383166000908152601c6020526040902054611d709060ff1660136145db565b6016805461ff001916610100179055600554600090611d97906001600160a01b03166146b6565b9050611da76000821160146145db565b611db184856140de565b6000611dbd8583611967565b9050611dcd6000821160156145db565b611dd78583614763565b611de18582614845565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b158015611e3457600080fd5b505af1158015611e48573d6000803e3d6000fd5b5050600554611e6292506001600160a01b03169050614918565b60015460408051634869c99560e11b81526001600160a01b03888116600483015260248201869052915160009363010000009004909216916390d3932a91604480820192602092909190829003018186803b158015611ec057600080fd5b505afa158015611ed4573d6000803e3d6000fd5b505050506040513d6020811015611eea57600080fd5b505190506000611efb8784846149b3565b9050611f0b6000821160166145db565b611f16878288614a7b565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517f33bd65d7d58f0e061076c61dff4ed1e2fe1209ea7b1be46bb4977e10f6df696a9181900360a00190a16016805461ff001916905560016000559695505050505050565b601f6020526000908152604090205460ff1681565b611fa5614526565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611fe1878787876119ac565b9050611feb615e27565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c08201819052909261205f92899290919089906126ec565b93509350505094509492505050565b600260005414156120b4576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b60026000556001546120d09062010000900460ff16601c6145db565b6120d8614b22565b6120e185614b3e565b6120ec848483614ba1565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b15801561216057600080fd5b505afa158015612174573d6000803e3d6000fd5b5050505061218284846140de565b6000612190868686856119ac565b6000818152602b60205260408120919250836121b4576121af86612a58565b6121bd565b6121bd86613e5c565b82549091506121ce57600282018190555b8154158015906121de5750600085115b15612204576121fe868360000154846002015487858a88600601546139e7565b60028301555b600061221d898989888a88600001548960030154614cb6565b9050600061222a896146b6565b905060006122388a836117e4565b600186015490915061224a9082614541565b6001860181905561225f90841115601d6145db565b600185015461226e9084614599565b600186015561227e8a8a896132cf565b6003860155845461228f9089614541565b8086554260068701556122a5901515601e6145db565b6122b785600001548660010154614d84565b6122c58b8b8b8a6001613836565b505060006122d38b8a613056565b60048701549091506122e59082614541565b60048701556122f48b82614da8565b87156123395761230d8b6123088b87614541565b614e48565b6123178b83614ec7565b6123218b84614f46565b6123348b61232f8d87612f7c565b614845565b6123a5565b6001600160a01b038a166000908152602d6020526040902054612376576001600160a01b038a166000908152602e6020526040902085905561239b565b6123818a868b612f9e565b6001600160a01b038b166000908152602e60205260409020555b6123a58a8a61505e565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e08201879052517f20853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b773918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b60008060008060008060008060006124d68d8d8d8d6119ac565b90506124e0615e27565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a0015113612560578160a00151600003612566565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b60008061261c868686866119ac565b9050612626615e27565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c083015261269090151560256145db565b602081015181516126a891906117be9061271061448e565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6000806126fd6000861160266145db565b6000846127125761270d88613e5c565b61271b565b61271b88612a58565b90506000818711612735576127308288614599565b61273f565b61273f8783614599565b90506000612751886117be8b8561448e565b9050600087156127645750878311612769565b508288115b60006127806010548961454190919063ffffffff16565b42116127a4576001600160a01b038c166000908152601e60205260409020546127a7565b60005b90508180156127ca57506127bb8b8261448e565b6127c78461271061448e565b11155b156127d457600092505b509a909950975050505050505050565b6002600054141561282a576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546128559060ff16600e6145db565b6000612860826146b6565b905061287060008211600f6145db565b61287a8282614f46565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b612952614526565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b612986614526565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b6129b0614526565b600180546129c49160ff90911615906145db565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b6004546001600160a01b031681565b600a5481565b612a3c614526565b60018054911515620100000262ff000019909216919091179055565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b158015612ad157600080fd5b505afa158015612ae5573d6000803e3d6000fd5b505050506040513d6020811015612afb57600080fd5b505192915050565b600060026000541415612b4b576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055612b58614b22565b612b6188614b3e565b612b7088888888888888615116565b600160005598975050505050505050565b60236020526000908152604090205481565b612b9b614526565b6001600160a01b03909116600090815260276020526040902055565b601e6020526000908152604090205481565b612bd1614526565b612be1610e10841015600a6145db565b612bf1612710831115600b6145db565b612c01612710821115600c6145db565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612c3b614526565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415612cb4576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055600154612ccf90610100900460ff1660176145db565b6001600160a01b0384166000908152601c6020526040902054612cf69060ff1660186145db565b6001600160a01b0383166000908152601c6020526040902054612d1d9060ff1660196145db565b612d3d836001600160a01b0316856001600160a01b03161415601a6145db565b6016805461ff001916610100179055612d5684806140de565b612d6083846140de565b6000612d6b856146b6565b9050612d7b60008211601b6145db565b6000612d8686612a58565b90506000612d9386613e5c565b90506000612da5826117be868661448e565b9050612db2818989611c52565b90506000612dd068327cb2734119d3b7a9601e1b6117be878761448e565b600554909150612dec9082908b906001600160a01b0316611c52565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b158015612e5657600080fd5b505afa158015612e6a573d6000803e3d6000fd5b505050506040513d6020811015612e8057600080fd5b505190506000612e918a85846149b3565b9050612e9d8b846155d9565b612ea78a84614763565b612eb18b88614f46565b612ebb8a85614845565b612ec48a615697565b612ecf8a828b614a7b565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b600081612f8b57506000611831565b611ced8383612f9986613e5c565b6140a4565b6001600160a01b0383166000908152602d6020908152604080832054602e90925282205482858211612fd957612fd48683614599565b612fe3565b612fe38287614599565b90506000612ff5836117be868561448e565b905086831160006130068689614541565b905060008261301e576130198285614541565b613028565b6130288285614599565b9050613038816117be8c8561448e565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b60008161306557506000611831565b611ced8383612f9986612a58565b61307b614526565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a6020526040812054909142916130cf91614541565b11156130dd575060006117c8565b6012546001600160a01b0383166000908152602a6020526040812054909161310a916117be904290614599565b6001600160a01b03841660009081526025602052604090205490915080613136576000925050506117c8565b6001600160a01b0384166000908152601f602052604081205460ff1661315e57601354613162565b6014545b6001600160a01b03861660009081526026602052604090205490915061319b9083906117be90869061319590869061448e565b9061448e565b95945050505050565b6131ac614526565b6001600160a01b038216600090815260236020526040902054808211156131e6576131e0836131db8484614599565b6155d9565b506131fb565b6131f9836131f48385614599565b614763565b505b5050565b60165460ff1681565b60226020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff161561326e57506001600160a01b0381166000908152602560205260409020546117c8565b6001600160a01b038216600090815260286020526040812054613292908490612f7c565b6001600160a01b0384166000908152602660209081526040808320546025909252909120549192506117c4916132c9908490614541565b90614599565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b15801561333657600080fd5b505afa15801561334a573d6000803e3d6000fd5b505050506040513d602081101561336057600080fd5b5051949350505050565b6001600160a01b0381166000908152602d60205260408120548190806133975760008092509250506133fd565b60006133a285613e5c565b6001600160a01b0386166000908152602e60205260408120549192508282116133d4576133cf8383614599565b6133de565b6133de8284614599565b905060006133f0836117be878561448e565b9390921195509193505050505b915091565b60246020526000908152604090205481565b60266020526000908152604090205481565b60135481565b6005546001600160a01b031681565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b1580156134c157600080fd5b505afa1580156134d5573d6000803e3d6000fd5b505050506040513d60208110156134eb57600080fd5b50519695505050505050565b6134ff614526565b6131f96001600160a01b038316848361570c565b60006002600054141561355b576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b6002600055613568614687565b6001600160a01b0383166000908152601c602052604090205461358f9060ff1660106145db565b6016805461ff00191661010017905560006135a9846146b6565b90506135b96000821160116145db565b6135c384856140de565b60006135ce85612a58565b905060006135ec68327cb2734119d3b7a9601e1b6117be858561448e565b60055490915061360890829088906001600160a01b0316611c52565b90506136186000821160126145db565b6001546040805163d54b31f160e01b81526001600160a01b038981166004830152602482018590529151600093630100000090049092169163d54b31f191604480820192602092909190829003018186803b15801561367657600080fd5b505afa15801561368a573d6000803e3d6000fd5b505050506040513d60208110156136a057600080fd5b5051905060006136b18886846149b3565b905060006136cf68327cb2734119d3b7a9601e1b6117be848861448e565b6005549091506136eb9082908b906001600160a01b0316611c52565b90506136f789826155d9565b6137018983614f46565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b15801561375657600080fd5b505af115801561376a573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517ff69f1ee12180eb7d5ff48e7f458d08949a51bd7689b2d2995392b40eea45fbac93509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b6137eb614526565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b613815614526565b601755565b613822614526565b613831612710821160026145db565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b1580156138ac57600080fd5b505afa1580156138c0573d6000803e3d6000fd5b505050506040513d60408110156138d657600080fd5b508051602090910151909890975095505050505050565b601281565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b15801561398757600080fd5b505afa15801561399b573d6000803e3d6000fd5b505050506040513d60208110156139b157600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b60008060006139f98a8a8a8a886126ec565b90925090506000613a0a8a87614541565b905060008815613a395783613a2857613a238284614599565b613a32565b613a328284614541565b9050613a5a565b83613a4d57613a488284614541565b613a57565b613a578284614599565b90505b613a68816117be8a8561448e565b9c9b505050505050505050505050565b60155481565b60026000541415613ac4576040805162461bcd60e51b815260206004820152601f6024820152600080516020615e7a833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613af95733600090815260196020526040902054613af99060ff1660226145db565b6016805460ff19169055613b0d84846140de565b6000613b1b868686866119ac565b9050613b25615e27565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613b9190151560236145db565b600080613ba2898989896000613836565b91509150613bb5826000141560246145db565b8160021415613bea57613bd2898989600087600001518b8f615116565b50506016805460ff1916600117905550613e4a915050565b6000613bf68983612f7c565b6001600160a01b038a166000908152602c6020526040902054909150613c1c9082614541565b6001600160a01b038a166000818152602c602090815260409182902093909355805191825291810184905280820183905290517f5d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011a9181900360600190a1613c8789856080015161575e565b8615613cb65760208401518451613ca8918b91613ca391614599565b614ec7565b613cb68961232f8b85612f7c565b600087613ccb57613cc689613e5c565b613cd4565b613cd489612a58565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613d935750846020015183105b15613dc1576020850151600090613daa9085614599565b9050613dbf8b613dba8d84612f7c565b614f46565b505b87613dd457613dd4898660000151615813565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613e20908b9061232f908290612f7c565b613e368a613e308c600954612f7c565b89614a7b565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b158015612ad157600080fd5b601b8181548110613ee157fe5b6000918252602090912001546001600160a01b0316905081565b613f03614526565b6001600160a01b0381166000908152601c6020526040902054613f2a9060ff16600d6145db565b6001600160a01b038116600090815260226020526040902054601554613f4f91614599565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f835281842080548216905591805290912080549091169055600754613fc8906001614599565b60075550565b613fd6614526565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b61400c614526565b6016805491151563010000000263ff00000019909216919091179055565b6000614034614526565b6001600160a01b0383166000908152602c60205260409020548061405c576000915050611831565b6001600160a01b0384166000908152602c6020526040812055611ced848285614a7b565b601a6020526000908152604090205460ff1681565b6003546001600160a01b031681565b6000826140b357506000611ced565b6001600160a01b0384166000908152601d602052604090205461319b836117be86600a85900a61448e565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b15801561413e57600080fd5b505af1158015614152573d6000803e3d6000fd5b505050506040513d602081101561416857600080fd5b505190508061417757506131fb565b6001600160a01b0383166000908152602a60205260409020546141c4576012546141a59061319542826144e7565b6001600160a01b0384166000908152602a6020526040902055506131fb565b6012546001600160a01b0384166000908152602a602052604090205442916141ec9190614541565b11156141f857506131fb565b6000614203846130a6565b6001600160a01b0385166000908152602960205260409020549091506142299082614541565b6001600160a01b0385166000908152602960205260409020556012546142539061319542826144e7565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b1580156134c157600080fd5b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b15801561438457600080fd5b505afa158015614398573d6000803e3d6000fd5b505050506040513d60208110156143ae57600080fd5b50519050806143c15760009150506117c8565b6001600160a01b0383166000908152602260205260409020546015546143eb906117be838561448e565b949350505050565b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144865780601f1061445b57610100808354040283529160200191614486565b820191906000526020600020905b81548152906001019060200180831161446957829003601f168201915b505050505081565b60008261449d57506000611831565b828202828482816144aa57fe5b0414611ced5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e9a6021913960400191505060405180910390fd5b6000611ced83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b81525061587b565b60065461453f906001600160a01b0316331460356145db565b565b600082820183811015611ced576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b6000611ced83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f77000081525061591d565b6000818152603060205260409020826131f95760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156146785780601f1061464d57610100808354040283529160200191614678565b820191906000526020600020905b81548152906001019060200180831161465b57829003601f168201915b50509250505060405180910390fd5b60165462010000900460ff161561453f57336000908152601a602052604090205461453f9060ff1660366145db565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b15801561471057600080fd5b505afa158015614724573d6000803e3d6000fd5b505050506040513d602081101561473a57600080fd5b50516001600160a01b038516600090815260216020526040902081905590506143eb8183614599565b6001600160a01b0382166000908152602360205260409020548181116147de576001600160a01b0383166000818152602360209081526040808320929092558151928352820183905280517f9b334cd33ac6a46d0c2f047a99c238a41ac13f4dc0d74daf8e9cc81a7837b3119281900390910190a1506131fb565b6147e88183614599565b6001600160a01b03841660008181526023602090815260409182902093909355805191825291810184905281517f9b334cd33ac6a46d0c2f047a99c238a41ac13f4dc0d74daf8e9cc81a7837b311929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b03851660009081526025909152919091205461489d91839061591d565b6001600160a01b038316600090815260256020908152604080832084905560269091529020546148d091101560326145db565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b15801561496757600080fd5b505afa15801561497b573d6000803e3d6000fd5b505050506040513d602081101561499157600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b6000806149d06127106117be6149c98287614599565b879061448e565b905060006149de8583614599565b6001600160a01b0387166000908152602c6020526040902054909150614a049082614541565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b86614a4981846117e4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b614a8f6001600160a01b038416828461570c565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b158015614ad557600080fd5b505afa158015614ae9573d6000803e3d6000fd5b505050506040513d6020811015614aff57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b601754614b2e5761453f565b61453f6017543a111560376145db565b336001600160a01b0382161415614b5457614b9e565b6003546001600160a01b0316331415614b6c57614b9e565b6001600160a01b0381166000908152601860209081526040808320338452909152902054614b9e9060ff1660296145db565b50565b8015614c1a57614bc6826001600160a01b0316846001600160a01b031614602a6145db565b6001600160a01b0383166000908152601c6020526040902054614bed9060ff16602b6145db565b6001600160a01b0383166000908152601f6020526040902054614c159060ff1615602c6145db565b6131f9565b6001600160a01b0383166000908152601c6020526040902054614c419060ff16602d6145db565b6001600160a01b0383166000908152601f6020526040902054614c689060ff16602e6145db565b6001600160a01b0382166000908152601f6020526040902054614c909060ff1615602f6145db565b6001600160a01b03821660009081526020805260409020546131f99060ff1660306145db565b600080614cc689898989896142bd565b90506000614cd88a8a8a8a898961390a565b9050614ce48282614541565b91506000614cf28a84612f7c565b6001600160a01b038b166000908152602c6020526040902054909150614d189082614541565b6001600160a01b038b166000818152602c602090815260409182902093909355805191825291810185905280820183905290517f5d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011a9181900360600190a150909998505050505050505050565b81614d9a57614d95811560276145db565b6131fb565b6131fb8183101560286145db565b6001600160a01b038216600090815260266020526040902054614dcb9082614541565b6001600160a01b038316600090815260266020818152604080842085905560258252909220549152614e0091111560346145db565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614e6b9082614541565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614eea9082614599565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614f699082614541565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614fbf57600080fd5b505afa158015614fd3573d6000803e3d6000fd5b505050506040513d6020811015614fe957600080fd5b50516001600160a01b0384166000908152602560205260409020549091506150159082101560316145db565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b6001600160a01b0382166000908152602d60205260409020546150819082614541565b6001600160a01b0383166000908152602d6020908152604080832093909355602f9052205480156131f9576001600160a01b0383166000908152602d60205260409020548110156131f9576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b15801561519a57600080fd5b505afa1580156151ae573d6000803e3d6000fd5b505050506151bc87876140de565b60006151ca898989876119ac565b6000818152602b602052604090208054919250906151eb901515601f6145db565b6151fd868260000154101560206145db565b61520f878260010154101560216145db565b60018101548154600483015460009161522c916117be908b61448e565b600484015490915061523e9082614599565b600484015561524d8b8261575e565b5060008061525f8d8d8d8d8d8d615977565b855491935091508914615403576152778c8c8a6132cf565b60038501558354615288908a614599565b808555600185015461529a9190614d84565b6152a88d8d8d8b6001613836565b505087156152d5576152cb8c61230886600101548661459990919063ffffffff16565b6152d58c8a614ec7565b6000886152ea576152e58c613e5c565b6152f3565b6152f38c612a58565b90507f93d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30868f8f8f8f8f8f886153288c8c614599565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e08201839052517f20853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b773918190036101000190a150615577565b871561541d576154138c84614e48565b61541d8c8a614ec7565b6000886154325761542d8c613e5c565b61543b565b61543b8c612a58565b90507f93d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30868f8f8f8f8f8f886154708c8c614599565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b87615586576155868b8a615813565b81156155c65787156155a0576155a08c61232f8e85612f7c565b60006155ac8d83612f7c565b90506155b98d828a614a7b565b95506126a8945050505050565b5060009c9b505050505050505050505050565b6001600160a01b0382166000908152602360205260409020546155fc9082614541565b6001600160a01b038316600090815260236020908152604080832093909355602490522054801561564e576001600160a01b03831660009081526023602052604090205461564e9082101560336145db565b604080516001600160a01b03851681526020810184905281517ffb9f1ce4a8a927fbc3811e4804d87f82ebd0ad0b0a9b6bf89f261cfd2cdfd605929181900390910190a1505050565b6001600160a01b0381166000908152602760209081526040808320546025909252909120541015614b9e576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b1790526131f9908490615b71565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b0385166000908152602690915291909120546157b791839061591d565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d60205260409020548082111561585357506001600160a01b0382166000908152602d60205260408120556131fb565b61585d8183614599565b6001600160a01b0384166000908152602d6020526040902055505050565b600081836159075760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b838110156158cc5781810151838201526020016158b4565b50505050905090810190601f1680156158f95780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161591357fe5b0495945050505050565b6000818484111561596f5760405162461bcd60e51b81526020600482018181528351602484015283519092839260449091019190850190808383600083156158cc5781810151838201526020016158b4565b505050900390565b6000806000615988898989876119ac565b6000818152602b602052604081208054600382015493945090926159b5918d918d918d918b918d91614cb6565b90506000806000806159d68d876000015488600201548d8a600601546126ec565b875491955085935091506159ee906117be8d8461448e565b925050506000828015615a015750600082115b15615a315750600584018054820190558088615a31576000615a238e84612f7c565b9050615a2f8e82614845565b505b82158015615a3f5750600082115b15615a83576001850154615a539083614599565b600186015588615a77576000615a698e84612f7c565b9050615a758e82614f46565b505b60058501805483900390555b8a15615aab57615a93818c614541565b6001860154909150615aa5908c614599565b60018601555b84548a1415615ace576001850154615ac4908290614541565b6000600187015590505b8084811115615ae857615ae18286614599565b9050615b1c565b6001860154615af79086614599565b60018701558915615b1c576000615b0e8f87612f7c565b9050615b1a8f82614845565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615bc6826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615c229092919063ffffffff16565b8051909150156131f957808060200190516020811015615be557600080fd5b50516131f95760405162461bcd60e51b815260040180806020018281038252602a815260200180615ebb602a913960400191505060405180910390fd5b60606143eb848460008585615c3685615d3d565b615c87576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615cc65780518252601f199092019160209182019101615ca7565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615d28576040519150601f19603f3d011682016040523d82523d6000602084013e615d2d565b606091505b50915091506126a8828286615d43565b3b151590565b60608315615d52575081611ced565b825115615d625782518084602001fd5b60405162461bcd60e51b81526020600482018181528451602484015284518593919283926044019190850190808383600083156158cc5781810151838201526020016158b4565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615dea5782800160ff19823516178555615e17565b82800160010185558215615e17579182015b82811115615e17578235825591602001919060010190615dfc565b50615e23929150615e64565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615e235760008155600101615e6556fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c00536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f775361666545524332303a204552433230206f7065726174696f6e20646964206e6f742073756363656564a26469706673582212204054febd56ce178902e27acd8e42627811492947af19ff9bc106ace5fcd4f96864736f6c634300060c0033
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.