More Info
Private Name Tags
ContractCreator
Latest 14 from a total of 14 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Set Gov | 218546 | 617 days ago | IN | 0 ETH | 0.0001901 | ||||
Set Token Config | 218486 | 617 days ago | IN | 0 ETH | 0.00137024 | ||||
Set Token Config | 218484 | 617 days ago | IN | 0 ETH | 0.00140914 | ||||
Set Token Config | 218480 | 617 days ago | IN | 0 ETH | 0.00140906 | ||||
Set Token Config | 218476 | 617 days ago | IN | 0 ETH | 0.00127932 | ||||
Set Token Config | 218473 | 617 days ago | IN | 0 ETH | 0.00144148 | ||||
Set Token Config | 218469 | 617 days ago | IN | 0 ETH | 0.00178373 | ||||
Set Vault Utils | 218463 | 617 days ago | IN | 0 ETH | 0.00020708 | ||||
Set Error Contro... | 218456 | 617 days ago | IN | 0 ETH | 0.00033016 | ||||
Set Fees | 218454 | 617 days ago | IN | 0 ETH | 0.00055263 | ||||
Set Manager | 218451 | 617 days ago | IN | 0 ETH | 0.00034684 | ||||
Set In Manager M... | 218448 | 617 days ago | IN | 0 ETH | 0.00020912 | ||||
Set Funding Rate | 218447 | 617 days ago | IN | 0 ETH | 0.00025419 | ||||
Initialize | 218446 | 617 days ago | IN | 0 ETH | 0.00118773 |
Loading...
Loading
Contract Name:
Vault
Compiler Version
v0.6.12+commit.27d51765
Optimization Enabled:
Yes with 1 runs
Other Settings:
default evmVersion
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/IUSDQ.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 USDQ_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 override vaultUtils; address public errorController; address public override router; address public override priceFeed; address public override usdq; 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; // usdqAmounts tracks the amount of USDQ debt for each whitelisted token mapping (address => uint256) public override usdqAmounts; // maxUsdqAmounts allows setting a max amount of USDQ debt for a token mapping (address => uint256) public override maxUsdqAmounts; // 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 USDQ // 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 BuyUSDQ(address account, address token, uint256 tokenAmount, uint256 usdqAmount, uint256 feeBasisPoints); event SellUSDQ(address account, address token, uint256 usdqAmount, 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 IncreaseUsdqAmount(address token, uint256 amount); event DecreaseUsdqAmount(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 _usdq, address _priceFeed, uint256 _liquidationFeeUsd, uint256 _fundingRateFactor, uint256 _stableFundingRateFactor ) external { _onlyGov(); _validate(!isInitialized, 1); isInitialized = true; router = _router; usdq = _usdq; 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 _maxUsdqAmount, 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; maxUsdqAmounts[_token] = _maxUsdqAmount; 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 maxUsdqAmounts[_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 setUsdqAmount(address _token, uint256 _amount) external override { _onlyGov(); uint256 usdqAmount = usdqAmounts[_token]; if (_amount > usdqAmount) { _increaseUsdqAmount(_token, _amount.sub(usdqAmount)); return; } _decreaseUsdqAmount(_token, usdqAmount.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 USDQ 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 buyUSDQ(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 usdqAmount = tokenAmount.mul(price).div(PRICE_PRECISION); usdqAmount = adjustForDecimals(usdqAmount, _token, usdq); _validate(usdqAmount > 0, 18); uint256 feeBasisPoints = vaultUtils.getBuyUsdqFeeBasisPoints(_token, usdqAmount); uint256 amountAfterFees = _collectSwapFees(_token, tokenAmount, feeBasisPoints); uint256 mintAmount = amountAfterFees.mul(price).div(PRICE_PRECISION); mintAmount = adjustForDecimals(mintAmount, _token, usdq); _increaseUsdqAmount(_token, mintAmount); _increasePoolAmount(_token, amountAfterFees); IUSDQ(usdq).mint(_receiver, mintAmount); emit BuyUSDQ(_receiver, _token, tokenAmount, mintAmount, feeBasisPoints); useSwapPricing = false; return mintAmount; } function sellUSDQ(address _token, address _receiver) external override nonReentrant returns (uint256) { _validateManager(); _validate(whitelistedTokens[_token], 19); useSwapPricing = true; uint256 usdqAmount = _transferIn(usdq); _validate(usdqAmount > 0, 20); updateCumulativeFundingRate(_token, _token); uint256 redemptionAmount = getRedemptionAmount(_token, usdqAmount); _validate(redemptionAmount > 0, 21); _decreaseUsdqAmount(_token, usdqAmount); _decreasePoolAmount(_token, redemptionAmount); IUSDQ(usdq).burn(address(this), usdqAmount); // the _transferIn call increased the value of tokenBalances[usdq] // usually decreases in token balances are synced by calling _transferOut // however, for usdq, the tokens are burnt, so _updateTokenBalance should // be manually called to record the decrease in tokens _updateTokenBalance(usdq); uint256 feeBasisPoints = vaultUtils.getSellUsdqFeeBasisPoints(_token, usdqAmount); uint256 amountOut = _collectSwapFees(_token, redemptionAmount, feeBasisPoints); _validate(amountOut > 0, 22); _transferOut(_token, amountOut, _receiver); emit SellUSDQ(_receiver, _token, usdqAmount, 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 usdqAmounts by the same usdqAmount as debt is shifted between the assets uint256 usdqAmount = amountIn.mul(priceIn).div(PRICE_PRECISION); usdqAmount = adjustForDecimals(usdqAmount, _tokenIn, usdq); uint256 feeBasisPoints = vaultUtils.getSwapFeeBasisPoints(_tokenIn, _tokenOut, usdqAmount); uint256 amountOutAfterFees = _collectSwapFees(_tokenOut, amountOut, feeBasisPoints); _increaseUsdqAmount(_tokenIn, usdqAmount); _decreaseUsdqAmount(_tokenOut, usdqAmount); _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 _usdqAmount) public override view returns (uint256) { uint256 price = getMaxPrice(_token); uint256 redemptionAmount = _usdqAmount.mul(PRICE_PRECISION).div(price); return adjustForDecimals(redemptionAmount, usdq, _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 == usdq ? USDQ_DECIMALS : tokenDecimals[_tokenDiv]; uint256 decimalsMul = _tokenMul == usdq ? USDQ_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 _usdqDelta, uint256 _feeBasisPoints, uint256 _taxBasisPoints, bool _increment) public override view returns (uint256) { return vaultUtils.getFeeBasisPoints(_token, _usdqDelta, _feeBasisPoints, _taxBasisPoints, _increment); } function getTargetUsdqAmount(address _token) public override view returns (uint256) { uint256 supply = IERC20(usdq).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 _increaseUsdqAmount(address _token, uint256 _amount) private { usdqAmounts[_token] = usdqAmounts[_token].add(_amount); uint256 maxUsdqAmount = maxUsdqAmounts[_token]; if (maxUsdqAmount != 0) { _validate(usdqAmounts[_token] <= maxUsdqAmount, 51); } emit IncreaseUsdqAmount(_token, _amount); } function _decreaseUsdqAmount(address _token, uint256 _amount) private { uint256 value = usdqAmounts[_token]; // since USDQ can be minted using multiple assets // it is possible for the USDQ debt for a single asset to be less than zero // the USDQ debt is capped to zero for this case if (value <= _amount) { usdqAmounts[_token] = 0; emit DecreaseUsdqAmount(_token, value); return; } usdqAmounts[_token] = value.sub(_amount); emit DecreaseUsdqAmount(_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 usdq() external view returns (address); function gov() external view returns (address); function vaultUtils() external view returns (IVaultUtils); 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 getTargetUsdqAmount(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 setUsdqAmount(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 _maxUsdqAmount, 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 buyUSDQ(address _token, address _receiver) external returns (uint256); function sellUSDQ(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 _usdqDelta, 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 usdqAmounts(address _token) external view returns (uint256); function maxUsdqAmounts(address _token) external view returns (uint256); function getRedemptionAmount(address _token, uint256 _usdqAmount) 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 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 setPriceSampleSpace(uint256 _priceSampleSpace) external; function setMaxStrictPriceDeviation(uint256 _maxStrictPriceDeviation) external; function getPrice( address _token, bool _maximise, bool _includeAmmPrice, bool _useSwapPricing ) external view returns (uint256); function getPrimaryPrice(address _token, bool _maximise) external view returns (uint256); function setTokenConfig( address _token, address _priceFeed, uint256 _priceDecimals, bool _isStrictStable ) external; function getLatestPrimaryPrice(address _token) external view returns (uint256); }
// 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 getBuyUsdqFeeBasisPoints(address _token, uint256 _usdqAmount) external view returns (uint256); function getSellUsdqFeeBasisPoints(address _token, uint256 _usdqAmount) external view returns (uint256); function getSwapFeeBasisPoints(address _tokenIn, address _tokenOut, uint256 _usdqAmount) external view returns (uint256); function getFeeBasisPoints(address _token, uint256 _usdqDelta, uint256 _feeBasisPoints, uint256 _taxBasisPoints, bool _increment) external view returns (uint256); function getMaxAmountIn( address _tokenIn, address _tokenOut ) 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.12; /** * @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 IUSDQ { function addVault(address _vault) external; function removeVault(address _vault) external; function mint(address _account, uint256 _amount) external; function burn(address _account, uint256 _amount) external; }
{ "metadata": { "bytecodeHash": "none", "useLiteralContent": true }, "optimizer": { "enabled": true, "runs": 1 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
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":"usdqAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeBasisPoints","type":"uint256"}],"name":"BuyUSDQ","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":"DecreaseUsdqAmount","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":"IncreaseUsdqAmount","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":"usdqAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeBasisPoints","type":"uint256"}],"name":"SellUSDQ","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":"USDQ_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":"buyUSDQ","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":"_usdqDelta","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":"_usdqAmount","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":"getTargetUsdqAmount","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":"_usdq","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":"maxUsdqAmounts","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":"sellUSDQ","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":"_maxUsdqAmount","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":"setUsdqAmount","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":"usdq","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"usdqAmounts","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
60806040526001805462ff00001961ff00199091166101001716620100001781556207a1206008556032600a9081556014600b55601e600c819055600d556004600e55600f556011805460ff199081169091556170806012556016805463ffffff0019921690921716905534801561007657600080fd5b506001600055600680546001600160a01b03191633179055615eb1806200009e6000396000f3fe608060405234801561001057600080fd5b50600436106105695760003560e01c806304fef1db1461056e57806307c58752146105a65780630842b076146105ae5780630a48d5a9146105b657806310eb56c2146105e2578063123645aa146105ea578063126082cf1461061057806312d43a5114610618578063134ca63b1461063c578063174d269414610644578063181e210e1461064c5780631ce9cb8f1461066857806324b0c04d1461068e57806324ca984e146106af57806328e67be5146106d557806329ff96151461074a5780632c668ec1146107705780632d4b05761461079c57806330455ede146107da578063318bc689146107f957806334c1557d14610610578063351a964d14610801578063379789b514610809578063392e53cd146108355780633c5a6e351461083d5780633de39c111461088b5780633e72a2621461089357806340eb38021461089b57806340f98237146108ec578063421528731461091257806342b60b03146109465780634453a3741461096c57806345a6f3701461099a57806348d91abf146109f357806348f35cbb14610a375780634a3f088d14610a3f5780634a993ee914610ac05780634bc798f814610ae65780634befe2ca14610aee5780634d47b30414610af6578063514ea4bf14610afe57806351723e8214610b53578063523fba7f14610b91578063529a356f14610bb757806352f55eed14610bdd5780635662318514610c035780635c07eaab14610c0b5780635f7bc11914610c4b5780636092219914610c7157806362287a3214610c9f5780636274980314610ca75780636abbe0c814610ccd5780636ae0b15414610cd55780636be6026b14610cfb57806371089f4d14610d03578063724e78da14610d29578063728cdbca14610d4f57806372912e9414610d97578063741bef1a14610dbd5780637a210a2b14610dc55780637c2eb9f714610dcd57806381a612d614610dec57806382a0849014610e125780638585f4d214610e6657806388b1fbdf14610e925780638a27d46814610eb85780638a39735a146106105780638a78daa814610ee15780638ee573ac14610f075780638f7b840414610f2d5780639060b1ca14610f535780639331621214610f5b57806395082d2514610f935780639698d25a14610f9b5780639849e41214610fc15780639899cd0214610fc95780639d7432ca14610ff55780639f392eb314611027578063a22f23921461102f578063a42ab3d214611037578063a5e90eee14611063578063a93acac214611091578063ab08c1c6146110b7578063ab2f3ad4146110bf578063ae3302c2146110e5578063b06423f3146110ed578063b136ca49146110f5578063b19a5b601461111b578063b1cc53ab14611149578063b364accb14611181578063b7334977146111a7578063c3c7b9e9146111d5578063c4f718bf146111fb578063c65bc7b114611203578063c7e074c314611229578063cea0c32814611269578063cfad57a21461129f578063d2fa635e146112c5578063d3127e63146112e2578063d54d5a9f146112ff578063d8f897c31461135e578063d9ac422514611384578063da76524c1461138c578063daf9c210146113d6578063db3555fb146113fc578063db97495f14611422578063dc8f5fac1461146e578063de2ea94814611476578063df73a267146114be578063e124e6d2146114c6578063e468baf0146114ec578063e67f59a714611509578063efa10a6e1461152f578063f07456ce1461155b578063f07bbf7714611581578063f2555278146115a0578063f3ae2415146115ce578063f887ea40146115f4578063fa12dbc0146115fc578063fbfded6d1461162e578063fce28c101461165c578063fdaf6ac314611664578063fed1a606146116a8575b600080fd5b6105946004803603602081101561058457600080fd5b50356001600160a01b031661173a565b60408051918252519081900360200190f35b61059461179d565b6105946117ae565b610594600480360360408110156105cc57600080fd5b506001600160a01b0381351690602001356117b4565b610594611807565b6105946004803603602081101561060057600080fd5b50356001600160a01b031661180d565b61059461181f565b610620611825565b604080516001600160a01b039092168252519081900360200190f35b610594611834565b61059461183a565b610654611840565b604080519115158252519081900360200190f35b6105946004803603602081101561067e57600080fd5b50356001600160a01b0316611850565b6106ad600480360360208110156106a457600080fd5b50351515611862565b005b6106ad600480360360208110156106c557600080fd5b50356001600160a01b0316611886565b6106ad600480360360408110156106eb57600080fd5b81359190810190604081016020820135600160201b81111561070c57600080fd5b82018360208201111561071e57600080fd5b803590602001918460018302840111600160201b8311171561073f57600080fd5b5090925090506118b7565b6105946004803603602081101561076057600080fd5b50356001600160a01b0316611935565b6105946004803603604081101561078657600080fd5b506001600160a01b038135169060200135611949565b610594600480360360808110156107b257600080fd5b506001600160a01b03813581169160208101358216916040820135169060600135151561198e565b6106ad600480360360208110156107f057600080fd5b503515156119e4565b610594611a06565b610654611a0c565b6106ad6004803603604081101561081f57600080fd5b506001600160a01b038135169060200135611a1a565b610654611a75565b6106ad600480360360e081101561085357600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611a7e565b610594611bc0565b610654611bc6565b6106ad60048036036101208110156108b257600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611bd5565b6105946004803603602081101561090257600080fd5b50356001600160a01b0316611c8f565b6105946004803603606081101561092857600080fd5b508035906001600160a01b0360208201358116916040013516611d4f565b6106546004803603602081101561095c57600080fd5b50356001600160a01b0316611df1565b6106ad6004803603604081101561098257600080fd5b506001600160a01b0381351690602001351515611e06565b6109d8600480360360808110156109b057600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611e39565b60408051921515835260208301919091528051918290030190f35b6106ad600480360360a0811015610a0957600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013590608001351515611ed7565b610620612304565b610a7d60048036036080811015610a5557600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612313565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b61059460048036036020811015610ad657600080fd5b50356001600160a01b0316612409565b61059461241b565b610594612420565b610594612426565b610b1b60048036036020811015610b1457600080fd5b503561242c565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b61059460048036036080811015610b6957600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612469565b61059460048036036020811015610ba757600080fd5b50356001600160a01b031661250f565b61065460048036036020811015610bcd57600080fd5b50356001600160a01b0316612521565b61059460048036036020811015610bf357600080fd5b50356001600160a01b0316612536565b610620612548565b6109d8600480360360a0811015610c2157600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060800135612557565b6106ad60048036036020811015610c6157600080fd5b50356001600160a01b031661264f565b61065460048036036040811015610c8757600080fd5b506001600160a01b0381358116916020013516612732565b610594612752565b61059460048036036020811015610cbd57600080fd5b50356001600160a01b0316612758565b61062061276a565b6106ad60048036036020811015610ceb57600080fd5b50356001600160a01b0316612780565b6105946127ae565b6106ad60048036036020811015610d1957600080fd5b50356001600160a01b03166127b5565b6106ad60048036036020811015610d3f57600080fd5b50356001600160a01b03166127e9565b6106ad600480360360c0811015610d6557600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a00135612813565b61059460048036036020811015610dad57600080fd5b50356001600160a01b031661288a565b61062061289c565b6105946128ab565b6106ad60048036036020811015610de357600080fd5b503515156128b1565b61059460048036036020811015610e0257600080fd5b50356001600160a01b03166128d5565b610594600480360360e0811015610e2857600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612980565b6106ad60048036036040811015610e7c57600080fd5b506001600160a01b0381351690602001356129fe565b61059460048036036020811015610ea857600080fd5b50356001600160a01b0316612a22565b6106ad60048036036060811015610ece57600080fd5b5080359060208101359060400135612a34565b61059460048036036020811015610ef757600080fd5b50356001600160a01b0316612a7a565b61059460048036036020811015610f1d57600080fd5b50356001600160a01b0316612a8c565b6106ad60048036036020811015610f4357600080fd5b50356001600160a01b0316612a9e565b610654612ac8565b61059460048036036060811015610f7157600080fd5b506001600160a01b038135811691602081013582169160409091013516612ad7565b610594612dbf565b61059460048036036020811015610fb157600080fd5b50356001600160a01b0316612dcf565b610594612de1565b61059460048036036040811015610fdf57600080fd5b506001600160a01b038135169060200135612de7565b6105946004803603606081101561100b57600080fd5b506001600160a01b038135169060208101359060400135612e09565b610654612eb2565b610594612ebb565b6105946004803603604081101561104d57600080fd5b506001600160a01b038135169060200135612ec1565b6106ad6004803603604081101561107957600080fd5b506001600160a01b0381351690602001351515612ede565b610594600480360360208110156110a757600080fd5b50356001600160a01b0316612f11565b61065461300f565b610594600480360360208110156110d557600080fd5b50356001600160a01b0316613018565b61059461302a565b610654613030565b6105946004803603602081101561110b57600080fd5b50356001600160a01b031661303e565b6105946004803603604081101561113157600080fd5b506001600160a01b03813581169160200135166130df565b6105946004803603606081101561115f57600080fd5b506001600160a01b038135811691602081013590911690604001351515613373565b6109d86004803603602081101561119757600080fd5b50356001600160a01b031661340e565b610594600480360360408110156111bd57600080fd5b506001600160a01b03813581169160200135166134a6565b610594600480360360208110156111eb57600080fd5b50356001600160a01b0316613776565b610594613788565b6105946004803603602081101561121957600080fd5b50356001600160a01b031661378e565b610594600480360360a081101561123f57600080fd5b506001600160a01b03813516906020810135906040810135906060810135906080013515156137a0565b6106ad6004803603606081101561127f57600080fd5b506001600160a01b0381358116916020810135909116906040013561384a565b6106ad600480360360208110156112b557600080fd5b50356001600160a01b0316613866565b6106ad600480360360208110156112db57600080fd5b5035613890565b6106ad600480360360208110156112f857600080fd5b503561389d565b611345600480360360a081101561131557600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101351515906080013515156138b9565b6040805192835260208301919091528051918290030190f35b6105946004803603602081101561137457600080fd5b50356001600160a01b0316613970565b610594613982565b610594600480360360c08110156113a257600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a00135613988565b610654600480360360208110156113ec57600080fd5b50356001600160a01b0316613a3c565b6106546004803603602081101561141257600080fd5b50356001600160a01b0316613a51565b610594600480360360e081101561143857600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c00135613a65565b610594613af6565b6106ad600480360360a081101561148c57600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613afc565b610594613ec2565b610594600480360360208110156114dc57600080fd5b50356001600160a01b0316613ec8565b6106206004803603602081101561150257600080fd5b5035613f40565b6106ad6004803603602081101561151f57600080fd5b50356001600160a01b0316613f67565b6106ad6004803603604081101561154557600080fd5b506001600160a01b03813516906020013561403a565b6105946004803603602081101561157157600080fd5b50356001600160a01b031661405e565b6106ad6004803603602081101561159757600080fd5b50351515614070565b610594600480360360408110156115b657600080fd5b506001600160a01b0381358116916020013516614096565b610654600480360360208110156115e457600080fd5b50356001600160a01b03166140ec565b610620614101565b6105946004803603606081101561161257600080fd5b506001600160a01b038135169060208101359060400135614110565b6106ad6004803603604081101561164457600080fd5b506001600160a01b038135811691602001351661414a565b610594614323565b610594600480360360a081101561167a57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060800135614329565b6116c5600480360360208110156116be57600080fd5b503561439f565b6040805160208082528351818301528351919283929083019185019080838360005b838110156116ff5781810151838201526020016116e7565b50505050905090810190601f16801561172c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b03811660009081526025602052604081205480611762576000915050611798565b6001600160a01b03831660009081526026602052604090205461179490829061178e90620f424061443a565b90614493565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117c357506000611801565b60006117ce846128d5565b6001600160a01b0385166000908152601d60205260409020549091506117fc600a82900a61178e868561443a565b925050505b92915050565b600b5481565b60246020526000908152604090205481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b602c6020526000908152604090205481565b61186a6144d2565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611916576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061192f908383615ce9565b50505050565b6000611801826119448461303e565b6117b4565b60008061195584613ec8565b905060006119738261178e8668327cb2734119d3b7a9601e1b61443a565b6005549091506117fc9082906001600160a01b031687611d4f565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b6119ec6144d2565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b611a226144d2565b6001600160a01b03821660009081526023602052604090205480821115611a5c57611a5683611a5184846144ed565b61452f565b50611a71565b611a6f83611a6a83856144ed565b6145ed565b505b5050565b60015460ff1681565b611a866144d2565b6001600160a01b0387166000908152601c602052604090205460ff16611b0357600754611ab49060016146ab565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611b2a9082906144ed565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420899055601f83528184208054821689151517905591805290912080549091168415151790559050611ba981876146ab565b601555611bb588613ec8565b505050505050505050565b60175481565b60015462010000900460ff1681565b611bdd6144d2565b611bed6101f48a11156003614703565b611bfd6101f48911156004614703565b611c0d6101f48811156005614703565b611c1d6101f48711156006614703565b611c2d6101f48611156007614703565b611c3d6101f48511156008614703565b611c586904ee2d6d415b85acef8160201b8411156009614703565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611ce057600080fd5b505afa158015611cf4573d6000803e3d6000fd5b505050506040513d6020811015611d0a57600080fd5b5051905080611d1d576000915050611798565b6001600160a01b038316600090815260226020526040902054601554611d479061178e838561443a565b949350505050565b60055460009081906001600160a01b03858116911614611d87576001600160a01b0384166000908152601d6020526040902054611d8a565b60125b6005549091506000906001600160a01b03858116911614611dc3576001600160a01b0384166000908152601d6020526040902054611dc6565b60125b9050611de582600a0a61178e83600a0a8961443a90919063ffffffff16565b925050505b9392505050565b601f6020526000908152604090205460ff1681565b611e0e6144d2565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611e4a8787878761198e565b9050611e54615d67565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c082018190529092611ec89289929091908990612557565b93509350505094509492505050565b60026000541415611f1d576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b6002600055600154611f399062010000900460ff16601c614703565b611f416147af565b611f4a856147cb565b611f5584848361482e565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b158015611fc957600080fd5b505afa158015611fdd573d6000803e3d6000fd5b50505050611feb848461414a565b6000611ff98686868561198e565b6000818152602b602052604081209192508361201d57612018866128d5565b612026565b61202686613ec8565b825490915061203757600282018190555b8154158015906120475750600085115b1561206d57612067868360000154846002015487858a8860060154613a65565b60028301555b6000612086898989888a88600001548960030154614943565b90506000612093896149ff565b905060006120a18a836117b4565b60018601549091506120b390826146ab565b600186018190556120c890841115601d614703565b60018501546120d790846144ed565b60018601556120e78a8a89613373565b600386015584546120f890896146ab565b80865542600687015561210e901515601e614703565b61212085600001548660010154614aac565b61212e8b8b8b8a60016138b9565b5050600061213c8b8a612ec1565b600487015490915061214e90826146ab565b600487015561215d8b82614ad0565b87156121a2576121768b6121718b876146ab565b614b70565b6121808b83614bef565b61218a8b84614c6e565b61219d8b6121988d87612de7565b614d86565b61220e565b6001600160a01b038a166000908152602d60205260409020546121df576001600160a01b038a166000908152602e60205260409020859055612204565b6121ea8a868b612e09565b6001600160a01b038b166000908152602e60205260409020555b61220e8a8a614e59565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820187905251600080516020615dda833981519152918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b600080600080600080600080600061232d8d8d8d8d61198e565b9050612337615d67565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a00151136123b7578160a001516000036123bd565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b601281565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b6000806124788686868661198e565b9050612482615d67565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c08301526124ec9015156025614703565b60208101518151612504919061178e9061271061443a565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6005546001600160a01b031681565b600080612568600086116026614703565b60008461257d5761257888613ec8565b612586565b612586886128d5565b905060008187116125a05761259b82886144ed565b6125aa565b6125aa87836144ed565b905060006125bc8861178e8b8561443a565b9050600087156125cf57508783116125d4565b508288115b60006125eb601054896146ab90919063ffffffff16565b421161260f576001600160a01b038c166000908152601e6020526040902054612612565b60005b905081801561263557506126268b8261443a565b6126328461271061443a565b11155b1561263f57600092505b509a909950975050505050505050565b60026000541415612695576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546126c09060ff16600e614703565b60006126cb826149ff565b90506126db60008211600f614703565b6126e58282614c6e565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b6127bd6144d2565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b6127f16144d2565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b61281b6144d2565b6001805461282f9160ff9091161590614703565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b60236020526000908152604090205481565b6004546001600160a01b031681565b600a5481565b6128b96144d2565b60018054911515620100000262ff000019909216919091179055565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b15801561294e57600080fd5b505afa158015612962573d6000803e3d6000fd5b505050506040513d602081101561297857600080fd5b505192915050565b6000600260005414156129c8576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b60026000556129d56147af565b6129de886147cb565b6129ed88888888888888614f11565b600160005598975050505050505050565b612a066144d2565b6001600160a01b03909116600090815260276020526040902055565b601e6020526000908152604090205481565b612a3c6144d2565b612a4c610e10841015600a614703565b612a5c612710831115600b614703565b612a6c612710821115600c614703565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612aa66144d2565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415612b1f576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b6002600055600154612b3a90610100900460ff166017614703565b6001600160a01b0384166000908152601c6020526040902054612b619060ff166018614703565b6001600160a01b0383166000908152601c6020526040902054612b889060ff166019614703565b612ba8836001600160a01b0316856001600160a01b03161415601a614703565b6016805461ff001916610100179055612bc1848061414a565b612bcb838461414a565b6000612bd6856149ff565b9050612be660008211601b614703565b6000612bf1866128d5565b90506000612bfe86613ec8565b90506000612c108261178e868661443a565b9050612c1d818989611d4f565b90506000612c3b68327cb2734119d3b7a9601e1b61178e878761443a565b600554909150612c579082908b906001600160a01b0316611d4f565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b158015612cc157600080fd5b505afa158015612cd5573d6000803e3d6000fd5b505050506040513d6020811015612ceb57600080fd5b505190506000612cfc8a858461539e565b9050612d088b8461452f565b612d128a846145ed565b612d1c8b88614c6e565b612d268a85614d86565b612d2f8a615466565b612d3a8a828b6154db565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b600081612df657506000611801565b611dea8383612e0486613ec8565b614110565b6001600160a01b0383166000908152602d6020908152604080832054602e90925282205482858211612e4457612e3f86836144ed565b612e4e565b612e4e82876144ed565b90506000612e608361178e868561443a565b90508683116000612e7186896146ab565b9050600082612e8957612e8482856146ab565b612e93565b612e9382856144ed565b9050612ea38161178e8c8561443a565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b600081612ed057506000611801565b611dea8383612e04866128d5565b612ee66144d2565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a602052604081205490914291612f3a916146ab565b1115612f4857506000611798565b6012546001600160a01b0383166000908152602a60205260408120549091612f759161178e9042906144ed565b6001600160a01b03841660009081526025602052604090205490915080612fa157600092505050611798565b6001600160a01b0384166000908152601f602052604081205460ff16612fc957601354612fcd565b6014545b6001600160a01b03861660009081526026602052604090205490915061300690839061178e90869061300090869061443a565b9061443a565b95945050505050565b60165460ff1681565b60226020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff161561307e57506001600160a01b038116600090815260256020526040902054611798565b6001600160a01b0382166000908152602860205260408120546130a2908490612de7565b6001600160a01b038416600090815260266020908152604080832054602590925290912054919250611794916130d99084906146ab565b906144ed565b600060026000541415613127576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b6002600055613134615582565b6001600160a01b0383166000908152601c602052604090205461315b9060ff166013614703565b6016805461ff001916610100179055600554600090613182906001600160a01b03166149ff565b9050613192600082116014614703565b61319c848561414a565b60006131a88583611949565b90506131b8600082116015614703565b6131c285836145ed565b6131cc8582614d86565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b15801561321f57600080fd5b505af1158015613233573d6000803e3d6000fd5b505060055461324d92506001600160a01b031690506155b1565b6001546040805163ed9c75d160e01b81526001600160a01b038881166004830152602482018690529151600093630100000090049092169163ed9c75d191604480820192602092909190829003018186803b1580156132ab57600080fd5b505afa1580156132bf573d6000803e3d6000fd5b505050506040513d60208110156132d557600080fd5b5051905060006132e687848461539e565b90506132f6600082116016614703565b6133018782886154db565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517f34672e19eab59907c4b4bf08d03e0f0810a7d84d6668418748a47cf956b477da9181900360a00190a16016805461ff001916905560016000559695505050505050565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b1580156133da57600080fd5b505afa1580156133ee573d6000803e3d6000fd5b505050506040513d602081101561340457600080fd5b5051949350505050565b6001600160a01b0381166000908152602d602052604081205481908061343b5760008092509250506134a1565b600061344685613ec8565b6001600160a01b0386166000908152602e60205260408120549192508282116134785761347383836144ed565b613482565b61348282846144ed565b905060006134948361178e878561443a565b9390921195509193505050505b915091565b6000600260005414156134ee576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b60026000556134fb615582565b6001600160a01b0383166000908152601c60205260409020546135229060ff166010614703565b6016805461ff001916610100179055600061353c846149ff565b905061354c600082116011614703565b613556848561414a565b6000613561856128d5565b9050600061357f68327cb2734119d3b7a9601e1b61178e858561443a565b60055490915061359b90829088906001600160a01b0316611d4f565b90506135ab600082116012614703565b60015460408051636b2d846960e01b81526001600160a01b0389811660048301526024820185905291516000936301000000900490921691636b2d846991604480820192602092909190829003018186803b15801561360957600080fd5b505afa15801561361d573d6000803e3d6000fd5b505050506040513d602081101561363357600080fd5b50519050600061364488868461539e565b9050600061366268327cb2734119d3b7a9601e1b61178e848861443a565b60055490915061367e9082908b906001600160a01b0316611d4f565b905061368a898261452f565b6136948983614c6e565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b1580156136e957600080fd5b505af11580156136fd573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517f6bd2f87ab92af01eeb44f35c462696c23993c6ce138f6742cf21716a456661a993509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b60266020526000908152604090205481565b60135481565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b15801561381457600080fd5b505afa158015613828573d6000803e3d6000fd5b505050506040513d602081101561383e57600080fd5b50519695505050505050565b6138526144d2565b611a6f6001600160a01b038316848361564c565b61386e6144d2565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b6138986144d2565b601755565b6138a56144d2565b6138b461271082116002614703565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b15801561392f57600080fd5b505afa158015613943573d6000803e3d6000fd5b505050506040513d604081101561395957600080fd5b508051602090910151909890975095505050505050565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b158015613a0557600080fd5b505afa158015613a19573d6000803e3d6000fd5b505050506040513d6020811015613a2f57600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b6000806000613a778a8a8a8a88612557565b90925090506000613a888a876146ab565b905060008815613ab75783613aa657613aa182846144ed565b613ab0565b613ab082846146ab565b9050613ad8565b83613acb57613ac682846146ab565b613ad5565b613ad582846144ed565b90505b613ae68161178e8a8561443a565b9c9b505050505050505050505050565b60155481565b60026000541415613b42576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613b775733600090815260196020526040902054613b779060ff166022614703565b6016805460ff19169055613b8b848461414a565b6000613b998686868661198e565b9050613ba3615d67565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613c0f9015156023614703565b600080613c208989898960006138b9565b91509150613c3382600014156024614703565b8160021415613c6857613c50898989600087600001518b8f614f11565b50506016805460ff1916600117905550613eb6915050565b6000613c748983612de7565b6001600160a01b038a166000908152602c6020526040902054909150613c9a90826146ab565b6001600160a01b038a166000818152602c60209081526040918290209390935580519182529181018490528082018390529051600080516020615e858339815191529181900360600190a1613cf389856080015161569e565b8615613d225760208401518451613d14918b91613d0f916144ed565b614bef565b613d22896121988b85612de7565b600087613d3757613d3289613ec8565b613d40565b613d40896128d5565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613dff5750846020015183105b15613e2d576020850151600090613e1690856144ed565b9050613e2b8b613e268d84612de7565b614c6e565b505b87613e4057613e40898660000151615753565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613e8c908b90612198908290612de7565b613ea28a613e9c8c600954612de7565b896154db565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b15801561294e57600080fd5b601b8181548110613f4d57fe5b6000918252602090912001546001600160a01b0316905081565b613f6f6144d2565b6001600160a01b0381166000908152601c6020526040902054613f969060ff16600d614703565b6001600160a01b038116600090815260226020526040902054601554613fbb916144ed565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f8352818420805482169055918052909120805490911690556007546140349060016144ed565b60075550565b6140426144d2565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b6140786144d2565b6016805491151563010000000263ff00000019909216919091179055565b60006140a06144d2565b6001600160a01b0383166000908152602c6020526040902054806140c8576000915050611801565b6001600160a01b0384166000908152602c6020526040812055611dea8482856154db565b601a6020526000908152604090205460ff1681565b6003546001600160a01b031681565b60008261411f57506000611dea565b6001600160a01b0384166000908152601d60205260409020546130068361178e86600a85900a61443a565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b1580156141aa57600080fd5b505af11580156141be573d6000803e3d6000fd5b505050506040513d60208110156141d457600080fd5b50519050806141e35750611a71565b6001600160a01b0383166000908152602a602052604090205461423057601254614211906130004282614493565b6001600160a01b0384166000908152602a602052604090205550611a71565b6012546001600160a01b0384166000908152602a6020526040902054429161425891906146ab565b11156142645750611a71565b600061426f84612f11565b6001600160a01b03851660009081526029602052604090205490915061429590826146ab565b6001600160a01b0385166000908152602960205260409020556012546142bf906130004282614493565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b15801561381457600080fd5b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144325780601f1061440757610100808354040283529160200191614432565b820191906000526020600020905b81548152906001019060200180831161441557829003601f168201915b505050505081565b60008261444957506000611801565b8282028284828161445657fe5b0414611dea5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e3a6021913960400191505060405180910390fd5b6000611dea83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b8152506157bb565b6006546144eb906001600160a01b031633146035614703565b565b6000611dea83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f77000081525061585d565b6001600160a01b03821660009081526023602052604090205461455290826146ab565b6001600160a01b03831660009081526023602090815260408083209390935560249052205480156145a4576001600160a01b0383166000908152602360205260409020546145a4908210156033614703565b604080516001600160a01b03851681526020810184905281517f657f7c1f83e78444355b029d7109b870c44b4273665a92ad602e12669be6e770929181900390910190a1505050565b6001600160a01b038216600090815260236020526040902054818111614656576001600160a01b038316600081815260236020908152604080832092909255815192835282018390528051600080516020615e1a8339815191529281900390910190a150611a71565b61466081836144ed565b6001600160a01b0384166000818152602360209081526040918290209390935580519182529181018490528151600080516020615e1a833981519152929181900390910190a1505050565b600082820183811015611dea576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b600081815260306020526040902082611a6f5760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156147a05780601f10614775576101008083540402835291602001916147a0565b820191906000526020600020905b81548152906001019060200180831161478357829003601f168201915b50509250505060405180910390fd5b6017546147bb576144eb565b6144eb6017543a11156037614703565b336001600160a01b03821614156147e15761482b565b6003546001600160a01b03163314156147f95761482b565b6001600160a01b038116600090815260186020908152604080832033845290915290205461482b9060ff166029614703565b50565b80156148a757614853826001600160a01b0316846001600160a01b031614602a614703565b6001600160a01b0383166000908152601c602052604090205461487a9060ff16602b614703565b6001600160a01b0383166000908152601f60205260409020546148a29060ff1615602c614703565b611a6f565b6001600160a01b0383166000908152601c60205260409020546148ce9060ff16602d614703565b6001600160a01b0383166000908152601f60205260409020546148f59060ff16602e614703565b6001600160a01b0382166000908152601f602052604090205461491d9060ff1615602f614703565b6001600160a01b0382166000908152602080526040902054611a6f9060ff166030614703565b6000806149538989898989614329565b905060006149658a8a8a8a8989613988565b905061497182826146ab565b9150600061497f8a84612de7565b6001600160a01b038b166000908152602c60205260409020549091506149a590826146ab565b6001600160a01b038b166000818152602c60209081526040918290209390935580519182529181018590528082018390529051600080516020615e858339815191529181900360600190a150909998505050505050505050565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b158015614a5957600080fd5b505afa158015614a6d573d6000803e3d6000fd5b505050506040513d6020811015614a8357600080fd5b50516001600160a01b03851660009081526021602052604090208190559050611d4781836144ed565b81614ac257614abd81156027614703565b611a71565b611a71818310156028614703565b6001600160a01b038216600090815260266020526040902054614af390826146ab565b6001600160a01b038316600090815260266020818152604080842085905560258252909220549152614b289111156034614703565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614b9390826146ab565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614c1290826144ed565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614c9190826146ab565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614ce757600080fd5b505afa158015614cfb573d6000803e3d6000fd5b505050506040513d6020811015614d1157600080fd5b50516001600160a01b038416600090815260256020526040902054909150614d3d908210156031614703565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b038516600090815260259091529190912054614dde91839061585d565b6001600160a01b03831660009081526025602090815260408083208490556026909152902054614e119110156032614703565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054614e7c90826146ab565b6001600160a01b0383166000908152602d6020908152604080832093909355602f905220548015611a6f576001600160a01b0383166000908152602d6020526040902054811015611a6f576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b158015614f9557600080fd5b505afa158015614fa9573d6000803e3d6000fd5b50505050614fb7878761414a565b6000614fc58989898761198e565b6000818152602b60205260409020805491925090614fe6901515601f614703565b614ff886826000015410156020614703565b61500a87826001015410156021614703565b6001810154815460048301546000916150279161178e908b61443a565b600484015490915061503990826144ed565b60048401556150488b8261569e565b5060008061505a8d8d8d8d8d8d6158b7565b8554919350915089146151da576150728c8c8a613373565b60038501558354615083908a6144ed565b80855560018501546150959190614aac565b6150a38d8d8d8b60016138b9565b505087156150d0576150c68c6121718660010154866144ed90919063ffffffff16565b6150d08c8a614bef565b6000886150e5576150e08c613ec8565b6150ee565b6150ee8c6128d5565b9050600080516020615dfa833981519152868f8f8f8f8f8f886151118c8c6144ed565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820183905251600080516020615dda833981519152918190036101000190a15061533c565b87156151f4576151ea8c84614b70565b6151f48c8a614bef565b600088615209576152048c613ec8565b615212565b6152128c6128d5565b9050600080516020615dfa833981519152868f8f8f8f8f8f886152358c8c6144ed565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b8761534b5761534b8b8a615753565b811561538b578715615365576153658c6121988e85612de7565b60006153718d83612de7565b905061537e8d828a6154db565b9550612504945050505050565b5060009c9b505050505050505050505050565b6000806153bb61271061178e6153b482876144ed565b879061443a565b905060006153c985836144ed565b6001600160a01b0387166000908152602c60205260409020549091506153ef90826146ab565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b8661543481846117b4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b6001600160a01b038116600090815260276020908152604080832054602590925290912054101561482b576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b6154ef6001600160a01b038416828461564c565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b15801561553557600080fd5b505afa158015615549573d6000803e3d6000fd5b505050506040513d602081101561555f57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b60165462010000900460ff16156144eb57336000908152601a60205260409020546144eb9060ff166036614703565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b15801561560057600080fd5b505afa158015615614573d6000803e3d6000fd5b505050506040513d602081101561562a57600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611a6f908490615ab1565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b0385166000908152602690915291909120546156f791839061585d565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d60205260409020548082111561579357506001600160a01b0382166000908152602d6020526040812055611a71565b61579d81836144ed565b6001600160a01b0384166000908152602d6020526040902055505050565b600081836158475760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561580c5781810151838201526020016157f4565b50505050905090810190601f1680156158395780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161585357fe5b0495945050505050565b600081848411156158af5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561580c5781810151838201526020016157f4565b505050900390565b60008060006158c88989898761198e565b6000818152602b602052604081208054600382015493945090926158f5918d918d918d918b918d91614943565b90506000806000806159168d876000015488600201548d8a60060154612557565b8754919550859350915061592e9061178e8d8461443a565b9250505060008280156159415750600082115b1561597157506005840180548201905580886159715760006159638e84612de7565b905061596f8e82614d86565b505b8215801561597f5750600082115b156159c357600185015461599390836144ed565b6001860155886159b75760006159a98e84612de7565b90506159b58e82614c6e565b505b60058501805483900390555b8a156159eb576159d3818c6146ab565b60018601549091506159e5908c6144ed565b60018601555b84548a1415615a0e576001850154615a049082906146ab565b6000600187015590505b8084811115615a2857615a2182866144ed565b9050615a5c565b6001860154615a3790866144ed565b60018701558915615a5c576000615a4e8f87612de7565b9050615a5a8f82614d86565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615b06826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615b629092919063ffffffff16565b805190915015611a6f57808060200190516020811015615b2557600080fd5b5051611a6f5760405162461bcd60e51b815260040180806020018281038252602a815260200180615e5b602a913960400191505060405180910390fd5b6060611d47848460008585615b7685615c7d565b615bc7576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615c065780518252601f199092019160209182019101615be7565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615c68576040519150601f19603f3d011682016040523d82523d6000602084013e615c6d565b606091505b5091509150612504828286615c83565b3b151590565b60608315615c92575081611dea565b825115615ca25782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561580c5781810151838201526020016157f4565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615d2a5782800160ff19823516178555615d57565b82800160010185558215615d57579182015b82811115615d57578235825591602001919060010190615d3c565b50615d63929150615da4565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615d635760008155600101615da556fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c0020853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b77393d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d309a06bc6ada7d180719cf7588e62ef5b07703f581a986715871a28f1db8a2ae16536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f775361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011aa164736f6c634300060c000a
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106105695760003560e01c806304fef1db1461056e57806307c58752146105a65780630842b076146105ae5780630a48d5a9146105b657806310eb56c2146105e2578063123645aa146105ea578063126082cf1461061057806312d43a5114610618578063134ca63b1461063c578063174d269414610644578063181e210e1461064c5780631ce9cb8f1461066857806324b0c04d1461068e57806324ca984e146106af57806328e67be5146106d557806329ff96151461074a5780632c668ec1146107705780632d4b05761461079c57806330455ede146107da578063318bc689146107f957806334c1557d14610610578063351a964d14610801578063379789b514610809578063392e53cd146108355780633c5a6e351461083d5780633de39c111461088b5780633e72a2621461089357806340eb38021461089b57806340f98237146108ec578063421528731461091257806342b60b03146109465780634453a3741461096c57806345a6f3701461099a57806348d91abf146109f357806348f35cbb14610a375780634a3f088d14610a3f5780634a993ee914610ac05780634bc798f814610ae65780634befe2ca14610aee5780634d47b30414610af6578063514ea4bf14610afe57806351723e8214610b53578063523fba7f14610b91578063529a356f14610bb757806352f55eed14610bdd5780635662318514610c035780635c07eaab14610c0b5780635f7bc11914610c4b5780636092219914610c7157806362287a3214610c9f5780636274980314610ca75780636abbe0c814610ccd5780636ae0b15414610cd55780636be6026b14610cfb57806371089f4d14610d03578063724e78da14610d29578063728cdbca14610d4f57806372912e9414610d97578063741bef1a14610dbd5780637a210a2b14610dc55780637c2eb9f714610dcd57806381a612d614610dec57806382a0849014610e125780638585f4d214610e6657806388b1fbdf14610e925780638a27d46814610eb85780638a39735a146106105780638a78daa814610ee15780638ee573ac14610f075780638f7b840414610f2d5780639060b1ca14610f535780639331621214610f5b57806395082d2514610f935780639698d25a14610f9b5780639849e41214610fc15780639899cd0214610fc95780639d7432ca14610ff55780639f392eb314611027578063a22f23921461102f578063a42ab3d214611037578063a5e90eee14611063578063a93acac214611091578063ab08c1c6146110b7578063ab2f3ad4146110bf578063ae3302c2146110e5578063b06423f3146110ed578063b136ca49146110f5578063b19a5b601461111b578063b1cc53ab14611149578063b364accb14611181578063b7334977146111a7578063c3c7b9e9146111d5578063c4f718bf146111fb578063c65bc7b114611203578063c7e074c314611229578063cea0c32814611269578063cfad57a21461129f578063d2fa635e146112c5578063d3127e63146112e2578063d54d5a9f146112ff578063d8f897c31461135e578063d9ac422514611384578063da76524c1461138c578063daf9c210146113d6578063db3555fb146113fc578063db97495f14611422578063dc8f5fac1461146e578063de2ea94814611476578063df73a267146114be578063e124e6d2146114c6578063e468baf0146114ec578063e67f59a714611509578063efa10a6e1461152f578063f07456ce1461155b578063f07bbf7714611581578063f2555278146115a0578063f3ae2415146115ce578063f887ea40146115f4578063fa12dbc0146115fc578063fbfded6d1461162e578063fce28c101461165c578063fdaf6ac314611664578063fed1a606146116a8575b600080fd5b6105946004803603602081101561058457600080fd5b50356001600160a01b031661173a565b60408051918252519081900360200190f35b61059461179d565b6105946117ae565b610594600480360360408110156105cc57600080fd5b506001600160a01b0381351690602001356117b4565b610594611807565b6105946004803603602081101561060057600080fd5b50356001600160a01b031661180d565b61059461181f565b610620611825565b604080516001600160a01b039092168252519081900360200190f35b610594611834565b61059461183a565b610654611840565b604080519115158252519081900360200190f35b6105946004803603602081101561067e57600080fd5b50356001600160a01b0316611850565b6106ad600480360360208110156106a457600080fd5b50351515611862565b005b6106ad600480360360208110156106c557600080fd5b50356001600160a01b0316611886565b6106ad600480360360408110156106eb57600080fd5b81359190810190604081016020820135600160201b81111561070c57600080fd5b82018360208201111561071e57600080fd5b803590602001918460018302840111600160201b8311171561073f57600080fd5b5090925090506118b7565b6105946004803603602081101561076057600080fd5b50356001600160a01b0316611935565b6105946004803603604081101561078657600080fd5b506001600160a01b038135169060200135611949565b610594600480360360808110156107b257600080fd5b506001600160a01b03813581169160208101358216916040820135169060600135151561198e565b6106ad600480360360208110156107f057600080fd5b503515156119e4565b610594611a06565b610654611a0c565b6106ad6004803603604081101561081f57600080fd5b506001600160a01b038135169060200135611a1a565b610654611a75565b6106ad600480360360e081101561085357600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611a7e565b610594611bc0565b610654611bc6565b6106ad60048036036101208110156108b257600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611bd5565b6105946004803603602081101561090257600080fd5b50356001600160a01b0316611c8f565b6105946004803603606081101561092857600080fd5b508035906001600160a01b0360208201358116916040013516611d4f565b6106546004803603602081101561095c57600080fd5b50356001600160a01b0316611df1565b6106ad6004803603604081101561098257600080fd5b506001600160a01b0381351690602001351515611e06565b6109d8600480360360808110156109b057600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611e39565b60408051921515835260208301919091528051918290030190f35b6106ad600480360360a0811015610a0957600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013590608001351515611ed7565b610620612304565b610a7d60048036036080811015610a5557600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612313565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b61059460048036036020811015610ad657600080fd5b50356001600160a01b0316612409565b61059461241b565b610594612420565b610594612426565b610b1b60048036036020811015610b1457600080fd5b503561242c565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b61059460048036036080811015610b6957600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612469565b61059460048036036020811015610ba757600080fd5b50356001600160a01b031661250f565b61065460048036036020811015610bcd57600080fd5b50356001600160a01b0316612521565b61059460048036036020811015610bf357600080fd5b50356001600160a01b0316612536565b610620612548565b6109d8600480360360a0811015610c2157600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060800135612557565b6106ad60048036036020811015610c6157600080fd5b50356001600160a01b031661264f565b61065460048036036040811015610c8757600080fd5b506001600160a01b0381358116916020013516612732565b610594612752565b61059460048036036020811015610cbd57600080fd5b50356001600160a01b0316612758565b61062061276a565b6106ad60048036036020811015610ceb57600080fd5b50356001600160a01b0316612780565b6105946127ae565b6106ad60048036036020811015610d1957600080fd5b50356001600160a01b03166127b5565b6106ad60048036036020811015610d3f57600080fd5b50356001600160a01b03166127e9565b6106ad600480360360c0811015610d6557600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a00135612813565b61059460048036036020811015610dad57600080fd5b50356001600160a01b031661288a565b61062061289c565b6105946128ab565b6106ad60048036036020811015610de357600080fd5b503515156128b1565b61059460048036036020811015610e0257600080fd5b50356001600160a01b03166128d5565b610594600480360360e0811015610e2857600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612980565b6106ad60048036036040811015610e7c57600080fd5b506001600160a01b0381351690602001356129fe565b61059460048036036020811015610ea857600080fd5b50356001600160a01b0316612a22565b6106ad60048036036060811015610ece57600080fd5b5080359060208101359060400135612a34565b61059460048036036020811015610ef757600080fd5b50356001600160a01b0316612a7a565b61059460048036036020811015610f1d57600080fd5b50356001600160a01b0316612a8c565b6106ad60048036036020811015610f4357600080fd5b50356001600160a01b0316612a9e565b610654612ac8565b61059460048036036060811015610f7157600080fd5b506001600160a01b038135811691602081013582169160409091013516612ad7565b610594612dbf565b61059460048036036020811015610fb157600080fd5b50356001600160a01b0316612dcf565b610594612de1565b61059460048036036040811015610fdf57600080fd5b506001600160a01b038135169060200135612de7565b6105946004803603606081101561100b57600080fd5b506001600160a01b038135169060208101359060400135612e09565b610654612eb2565b610594612ebb565b6105946004803603604081101561104d57600080fd5b506001600160a01b038135169060200135612ec1565b6106ad6004803603604081101561107957600080fd5b506001600160a01b0381351690602001351515612ede565b610594600480360360208110156110a757600080fd5b50356001600160a01b0316612f11565b61065461300f565b610594600480360360208110156110d557600080fd5b50356001600160a01b0316613018565b61059461302a565b610654613030565b6105946004803603602081101561110b57600080fd5b50356001600160a01b031661303e565b6105946004803603604081101561113157600080fd5b506001600160a01b03813581169160200135166130df565b6105946004803603606081101561115f57600080fd5b506001600160a01b038135811691602081013590911690604001351515613373565b6109d86004803603602081101561119757600080fd5b50356001600160a01b031661340e565b610594600480360360408110156111bd57600080fd5b506001600160a01b03813581169160200135166134a6565b610594600480360360208110156111eb57600080fd5b50356001600160a01b0316613776565b610594613788565b6105946004803603602081101561121957600080fd5b50356001600160a01b031661378e565b610594600480360360a081101561123f57600080fd5b506001600160a01b03813516906020810135906040810135906060810135906080013515156137a0565b6106ad6004803603606081101561127f57600080fd5b506001600160a01b0381358116916020810135909116906040013561384a565b6106ad600480360360208110156112b557600080fd5b50356001600160a01b0316613866565b6106ad600480360360208110156112db57600080fd5b5035613890565b6106ad600480360360208110156112f857600080fd5b503561389d565b611345600480360360a081101561131557600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101351515906080013515156138b9565b6040805192835260208301919091528051918290030190f35b6105946004803603602081101561137457600080fd5b50356001600160a01b0316613970565b610594613982565b610594600480360360c08110156113a257600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a00135613988565b610654600480360360208110156113ec57600080fd5b50356001600160a01b0316613a3c565b6106546004803603602081101561141257600080fd5b50356001600160a01b0316613a51565b610594600480360360e081101561143857600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c00135613a65565b610594613af6565b6106ad600480360360a081101561148c57600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613afc565b610594613ec2565b610594600480360360208110156114dc57600080fd5b50356001600160a01b0316613ec8565b6106206004803603602081101561150257600080fd5b5035613f40565b6106ad6004803603602081101561151f57600080fd5b50356001600160a01b0316613f67565b6106ad6004803603604081101561154557600080fd5b506001600160a01b03813516906020013561403a565b6105946004803603602081101561157157600080fd5b50356001600160a01b031661405e565b6106ad6004803603602081101561159757600080fd5b50351515614070565b610594600480360360408110156115b657600080fd5b506001600160a01b0381358116916020013516614096565b610654600480360360208110156115e457600080fd5b50356001600160a01b03166140ec565b610620614101565b6105946004803603606081101561161257600080fd5b506001600160a01b038135169060208101359060400135614110565b6106ad6004803603604081101561164457600080fd5b506001600160a01b038135811691602001351661414a565b610594614323565b610594600480360360a081101561167a57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060800135614329565b6116c5600480360360208110156116be57600080fd5b503561439f565b6040805160208082528351818301528351919283929083019185019080838360005b838110156116ff5781810151838201526020016116e7565b50505050905090810190601f16801561172c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b03811660009081526025602052604081205480611762576000915050611798565b6001600160a01b03831660009081526026602052604090205461179490829061178e90620f424061443a565b90614493565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117c357506000611801565b60006117ce846128d5565b6001600160a01b0385166000908152601d60205260409020549091506117fc600a82900a61178e868561443a565b925050505b92915050565b600b5481565b60246020526000908152604090205481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b602c6020526000908152604090205481565b61186a6144d2565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611916576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061192f908383615ce9565b50505050565b6000611801826119448461303e565b6117b4565b60008061195584613ec8565b905060006119738261178e8668327cb2734119d3b7a9601e1b61443a565b6005549091506117fc9082906001600160a01b031687611d4f565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b6119ec6144d2565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b611a226144d2565b6001600160a01b03821660009081526023602052604090205480821115611a5c57611a5683611a5184846144ed565b61452f565b50611a71565b611a6f83611a6a83856144ed565b6145ed565b505b5050565b60015460ff1681565b611a866144d2565b6001600160a01b0387166000908152601c602052604090205460ff16611b0357600754611ab49060016146ab565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611b2a9082906144ed565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420899055601f83528184208054821689151517905591805290912080549091168415151790559050611ba981876146ab565b601555611bb588613ec8565b505050505050505050565b60175481565b60015462010000900460ff1681565b611bdd6144d2565b611bed6101f48a11156003614703565b611bfd6101f48911156004614703565b611c0d6101f48811156005614703565b611c1d6101f48711156006614703565b611c2d6101f48611156007614703565b611c3d6101f48511156008614703565b611c586904ee2d6d415b85acef8160201b8411156009614703565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611ce057600080fd5b505afa158015611cf4573d6000803e3d6000fd5b505050506040513d6020811015611d0a57600080fd5b5051905080611d1d576000915050611798565b6001600160a01b038316600090815260226020526040902054601554611d479061178e838561443a565b949350505050565b60055460009081906001600160a01b03858116911614611d87576001600160a01b0384166000908152601d6020526040902054611d8a565b60125b6005549091506000906001600160a01b03858116911614611dc3576001600160a01b0384166000908152601d6020526040902054611dc6565b60125b9050611de582600a0a61178e83600a0a8961443a90919063ffffffff16565b925050505b9392505050565b601f6020526000908152604090205460ff1681565b611e0e6144d2565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611e4a8787878761198e565b9050611e54615d67565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c082018190529092611ec89289929091908990612557565b93509350505094509492505050565b60026000541415611f1d576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b6002600055600154611f399062010000900460ff16601c614703565b611f416147af565b611f4a856147cb565b611f5584848361482e565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b158015611fc957600080fd5b505afa158015611fdd573d6000803e3d6000fd5b50505050611feb848461414a565b6000611ff98686868561198e565b6000818152602b602052604081209192508361201d57612018866128d5565b612026565b61202686613ec8565b825490915061203757600282018190555b8154158015906120475750600085115b1561206d57612067868360000154846002015487858a8860060154613a65565b60028301555b6000612086898989888a88600001548960030154614943565b90506000612093896149ff565b905060006120a18a836117b4565b60018601549091506120b390826146ab565b600186018190556120c890841115601d614703565b60018501546120d790846144ed565b60018601556120e78a8a89613373565b600386015584546120f890896146ab565b80865542600687015561210e901515601e614703565b61212085600001548660010154614aac565b61212e8b8b8b8a60016138b9565b5050600061213c8b8a612ec1565b600487015490915061214e90826146ab565b600487015561215d8b82614ad0565b87156121a2576121768b6121718b876146ab565b614b70565b6121808b83614bef565b61218a8b84614c6e565b61219d8b6121988d87612de7565b614d86565b61220e565b6001600160a01b038a166000908152602d60205260409020546121df576001600160a01b038a166000908152602e60205260409020859055612204565b6121ea8a868b612e09565b6001600160a01b038b166000908152602e60205260409020555b61220e8a8a614e59565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820187905251600080516020615dda833981519152918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b600080600080600080600080600061232d8d8d8d8d61198e565b9050612337615d67565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a00151136123b7578160a001516000036123bd565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b601281565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b6000806124788686868661198e565b9050612482615d67565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c08301526124ec9015156025614703565b60208101518151612504919061178e9061271061443a565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6005546001600160a01b031681565b600080612568600086116026614703565b60008461257d5761257888613ec8565b612586565b612586886128d5565b905060008187116125a05761259b82886144ed565b6125aa565b6125aa87836144ed565b905060006125bc8861178e8b8561443a565b9050600087156125cf57508783116125d4565b508288115b60006125eb601054896146ab90919063ffffffff16565b421161260f576001600160a01b038c166000908152601e6020526040902054612612565b60005b905081801561263557506126268b8261443a565b6126328461271061443a565b11155b1561263f57600092505b509a909950975050505050505050565b60026000541415612695576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546126c09060ff16600e614703565b60006126cb826149ff565b90506126db60008211600f614703565b6126e58282614c6e565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b6127bd6144d2565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b6127f16144d2565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b61281b6144d2565b6001805461282f9160ff9091161590614703565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b60236020526000908152604090205481565b6004546001600160a01b031681565b600a5481565b6128b96144d2565b60018054911515620100000262ff000019909216919091179055565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b15801561294e57600080fd5b505afa158015612962573d6000803e3d6000fd5b505050506040513d602081101561297857600080fd5b505192915050565b6000600260005414156129c8576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b60026000556129d56147af565b6129de886147cb565b6129ed88888888888888614f11565b600160005598975050505050505050565b612a066144d2565b6001600160a01b03909116600090815260276020526040902055565b601e6020526000908152604090205481565b612a3c6144d2565b612a4c610e10841015600a614703565b612a5c612710831115600b614703565b612a6c612710821115600c614703565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612aa66144d2565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415612b1f576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b6002600055600154612b3a90610100900460ff166017614703565b6001600160a01b0384166000908152601c6020526040902054612b619060ff166018614703565b6001600160a01b0383166000908152601c6020526040902054612b889060ff166019614703565b612ba8836001600160a01b0316856001600160a01b03161415601a614703565b6016805461ff001916610100179055612bc1848061414a565b612bcb838461414a565b6000612bd6856149ff565b9050612be660008211601b614703565b6000612bf1866128d5565b90506000612bfe86613ec8565b90506000612c108261178e868661443a565b9050612c1d818989611d4f565b90506000612c3b68327cb2734119d3b7a9601e1b61178e878761443a565b600554909150612c579082908b906001600160a01b0316611d4f565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b158015612cc157600080fd5b505afa158015612cd5573d6000803e3d6000fd5b505050506040513d6020811015612ceb57600080fd5b505190506000612cfc8a858461539e565b9050612d088b8461452f565b612d128a846145ed565b612d1c8b88614c6e565b612d268a85614d86565b612d2f8a615466565b612d3a8a828b6154db565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b600081612df657506000611801565b611dea8383612e0486613ec8565b614110565b6001600160a01b0383166000908152602d6020908152604080832054602e90925282205482858211612e4457612e3f86836144ed565b612e4e565b612e4e82876144ed565b90506000612e608361178e868561443a565b90508683116000612e7186896146ab565b9050600082612e8957612e8482856146ab565b612e93565b612e9382856144ed565b9050612ea38161178e8c8561443a565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b600081612ed057506000611801565b611dea8383612e04866128d5565b612ee66144d2565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a602052604081205490914291612f3a916146ab565b1115612f4857506000611798565b6012546001600160a01b0383166000908152602a60205260408120549091612f759161178e9042906144ed565b6001600160a01b03841660009081526025602052604090205490915080612fa157600092505050611798565b6001600160a01b0384166000908152601f602052604081205460ff16612fc957601354612fcd565b6014545b6001600160a01b03861660009081526026602052604090205490915061300690839061178e90869061300090869061443a565b9061443a565b95945050505050565b60165460ff1681565b60226020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff161561307e57506001600160a01b038116600090815260256020526040902054611798565b6001600160a01b0382166000908152602860205260408120546130a2908490612de7565b6001600160a01b038416600090815260266020908152604080832054602590925290912054919250611794916130d99084906146ab565b906144ed565b600060026000541415613127576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b6002600055613134615582565b6001600160a01b0383166000908152601c602052604090205461315b9060ff166013614703565b6016805461ff001916610100179055600554600090613182906001600160a01b03166149ff565b9050613192600082116014614703565b61319c848561414a565b60006131a88583611949565b90506131b8600082116015614703565b6131c285836145ed565b6131cc8582614d86565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b15801561321f57600080fd5b505af1158015613233573d6000803e3d6000fd5b505060055461324d92506001600160a01b031690506155b1565b6001546040805163ed9c75d160e01b81526001600160a01b038881166004830152602482018690529151600093630100000090049092169163ed9c75d191604480820192602092909190829003018186803b1580156132ab57600080fd5b505afa1580156132bf573d6000803e3d6000fd5b505050506040513d60208110156132d557600080fd5b5051905060006132e687848461539e565b90506132f6600082116016614703565b6133018782886154db565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517f34672e19eab59907c4b4bf08d03e0f0810a7d84d6668418748a47cf956b477da9181900360a00190a16016805461ff001916905560016000559695505050505050565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b1580156133da57600080fd5b505afa1580156133ee573d6000803e3d6000fd5b505050506040513d602081101561340457600080fd5b5051949350505050565b6001600160a01b0381166000908152602d602052604081205481908061343b5760008092509250506134a1565b600061344685613ec8565b6001600160a01b0386166000908152602e60205260408120549192508282116134785761347383836144ed565b613482565b61348282846144ed565b905060006134948361178e878561443a565b9390921195509193505050505b915091565b6000600260005414156134ee576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b60026000556134fb615582565b6001600160a01b0383166000908152601c60205260409020546135229060ff166010614703565b6016805461ff001916610100179055600061353c846149ff565b905061354c600082116011614703565b613556848561414a565b6000613561856128d5565b9050600061357f68327cb2734119d3b7a9601e1b61178e858561443a565b60055490915061359b90829088906001600160a01b0316611d4f565b90506135ab600082116012614703565b60015460408051636b2d846960e01b81526001600160a01b0389811660048301526024820185905291516000936301000000900490921691636b2d846991604480820192602092909190829003018186803b15801561360957600080fd5b505afa15801561361d573d6000803e3d6000fd5b505050506040513d602081101561363357600080fd5b50519050600061364488868461539e565b9050600061366268327cb2734119d3b7a9601e1b61178e848861443a565b60055490915061367e9082908b906001600160a01b0316611d4f565b905061368a898261452f565b6136948983614c6e565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b1580156136e957600080fd5b505af11580156136fd573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517f6bd2f87ab92af01eeb44f35c462696c23993c6ce138f6742cf21716a456661a993509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b60266020526000908152604090205481565b60135481565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b15801561381457600080fd5b505afa158015613828573d6000803e3d6000fd5b505050506040513d602081101561383e57600080fd5b50519695505050505050565b6138526144d2565b611a6f6001600160a01b038316848361564c565b61386e6144d2565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b6138986144d2565b601755565b6138a56144d2565b6138b461271082116002614703565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b15801561392f57600080fd5b505afa158015613943573d6000803e3d6000fd5b505050506040513d604081101561395957600080fd5b508051602090910151909890975095505050505050565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b158015613a0557600080fd5b505afa158015613a19573d6000803e3d6000fd5b505050506040513d6020811015613a2f57600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b6000806000613a778a8a8a8a88612557565b90925090506000613a888a876146ab565b905060008815613ab75783613aa657613aa182846144ed565b613ab0565b613ab082846146ab565b9050613ad8565b83613acb57613ac682846146ab565b613ad5565b613ad582846144ed565b90505b613ae68161178e8a8561443a565b9c9b505050505050505050505050565b60155481565b60026000541415613b42576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dba833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613b775733600090815260196020526040902054613b779060ff166022614703565b6016805460ff19169055613b8b848461414a565b6000613b998686868661198e565b9050613ba3615d67565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613c0f9015156023614703565b600080613c208989898960006138b9565b91509150613c3382600014156024614703565b8160021415613c6857613c50898989600087600001518b8f614f11565b50506016805460ff1916600117905550613eb6915050565b6000613c748983612de7565b6001600160a01b038a166000908152602c6020526040902054909150613c9a90826146ab565b6001600160a01b038a166000818152602c60209081526040918290209390935580519182529181018490528082018390529051600080516020615e858339815191529181900360600190a1613cf389856080015161569e565b8615613d225760208401518451613d14918b91613d0f916144ed565b614bef565b613d22896121988b85612de7565b600087613d3757613d3289613ec8565b613d40565b613d40896128d5565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613dff5750846020015183105b15613e2d576020850151600090613e1690856144ed565b9050613e2b8b613e268d84612de7565b614c6e565b505b87613e4057613e40898660000151615753565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613e8c908b90612198908290612de7565b613ea28a613e9c8c600954612de7565b896154db565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b15801561294e57600080fd5b601b8181548110613f4d57fe5b6000918252602090912001546001600160a01b0316905081565b613f6f6144d2565b6001600160a01b0381166000908152601c6020526040902054613f969060ff16600d614703565b6001600160a01b038116600090815260226020526040902054601554613fbb916144ed565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f8352818420805482169055918052909120805490911690556007546140349060016144ed565b60075550565b6140426144d2565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b6140786144d2565b6016805491151563010000000263ff00000019909216919091179055565b60006140a06144d2565b6001600160a01b0383166000908152602c6020526040902054806140c8576000915050611801565b6001600160a01b0384166000908152602c6020526040812055611dea8482856154db565b601a6020526000908152604090205460ff1681565b6003546001600160a01b031681565b60008261411f57506000611dea565b6001600160a01b0384166000908152601d60205260409020546130068361178e86600a85900a61443a565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b1580156141aa57600080fd5b505af11580156141be573d6000803e3d6000fd5b505050506040513d60208110156141d457600080fd5b50519050806141e35750611a71565b6001600160a01b0383166000908152602a602052604090205461423057601254614211906130004282614493565b6001600160a01b0384166000908152602a602052604090205550611a71565b6012546001600160a01b0384166000908152602a6020526040902054429161425891906146ab565b11156142645750611a71565b600061426f84612f11565b6001600160a01b03851660009081526029602052604090205490915061429590826146ab565b6001600160a01b0385166000908152602960205260409020556012546142bf906130004282614493565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b15801561381457600080fd5b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144325780601f1061440757610100808354040283529160200191614432565b820191906000526020600020905b81548152906001019060200180831161441557829003601f168201915b505050505081565b60008261444957506000611801565b8282028284828161445657fe5b0414611dea5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e3a6021913960400191505060405180910390fd5b6000611dea83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b8152506157bb565b6006546144eb906001600160a01b031633146035614703565b565b6000611dea83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f77000081525061585d565b6001600160a01b03821660009081526023602052604090205461455290826146ab565b6001600160a01b03831660009081526023602090815260408083209390935560249052205480156145a4576001600160a01b0383166000908152602360205260409020546145a4908210156033614703565b604080516001600160a01b03851681526020810184905281517f657f7c1f83e78444355b029d7109b870c44b4273665a92ad602e12669be6e770929181900390910190a1505050565b6001600160a01b038216600090815260236020526040902054818111614656576001600160a01b038316600081815260236020908152604080832092909255815192835282018390528051600080516020615e1a8339815191529281900390910190a150611a71565b61466081836144ed565b6001600160a01b0384166000818152602360209081526040918290209390935580519182529181018490528151600080516020615e1a833981519152929181900390910190a1505050565b600082820183811015611dea576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b600081815260306020526040902082611a6f5760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156147a05780601f10614775576101008083540402835291602001916147a0565b820191906000526020600020905b81548152906001019060200180831161478357829003601f168201915b50509250505060405180910390fd5b6017546147bb576144eb565b6144eb6017543a11156037614703565b336001600160a01b03821614156147e15761482b565b6003546001600160a01b03163314156147f95761482b565b6001600160a01b038116600090815260186020908152604080832033845290915290205461482b9060ff166029614703565b50565b80156148a757614853826001600160a01b0316846001600160a01b031614602a614703565b6001600160a01b0383166000908152601c602052604090205461487a9060ff16602b614703565b6001600160a01b0383166000908152601f60205260409020546148a29060ff1615602c614703565b611a6f565b6001600160a01b0383166000908152601c60205260409020546148ce9060ff16602d614703565b6001600160a01b0383166000908152601f60205260409020546148f59060ff16602e614703565b6001600160a01b0382166000908152601f602052604090205461491d9060ff1615602f614703565b6001600160a01b0382166000908152602080526040902054611a6f9060ff166030614703565b6000806149538989898989614329565b905060006149658a8a8a8a8989613988565b905061497182826146ab565b9150600061497f8a84612de7565b6001600160a01b038b166000908152602c60205260409020549091506149a590826146ab565b6001600160a01b038b166000818152602c60209081526040918290209390935580519182529181018590528082018390529051600080516020615e858339815191529181900360600190a150909998505050505050505050565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b158015614a5957600080fd5b505afa158015614a6d573d6000803e3d6000fd5b505050506040513d6020811015614a8357600080fd5b50516001600160a01b03851660009081526021602052604090208190559050611d4781836144ed565b81614ac257614abd81156027614703565b611a71565b611a71818310156028614703565b6001600160a01b038216600090815260266020526040902054614af390826146ab565b6001600160a01b038316600090815260266020818152604080842085905560258252909220549152614b289111156034614703565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614b9390826146ab565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614c1290826144ed565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614c9190826146ab565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614ce757600080fd5b505afa158015614cfb573d6000803e3d6000fd5b505050506040513d6020811015614d1157600080fd5b50516001600160a01b038416600090815260256020526040902054909150614d3d908210156031614703565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b038516600090815260259091529190912054614dde91839061585d565b6001600160a01b03831660009081526025602090815260408083208490556026909152902054614e119110156032614703565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054614e7c90826146ab565b6001600160a01b0383166000908152602d6020908152604080832093909355602f905220548015611a6f576001600160a01b0383166000908152602d6020526040902054811015611a6f576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b158015614f9557600080fd5b505afa158015614fa9573d6000803e3d6000fd5b50505050614fb7878761414a565b6000614fc58989898761198e565b6000818152602b60205260409020805491925090614fe6901515601f614703565b614ff886826000015410156020614703565b61500a87826001015410156021614703565b6001810154815460048301546000916150279161178e908b61443a565b600484015490915061503990826144ed565b60048401556150488b8261569e565b5060008061505a8d8d8d8d8d8d6158b7565b8554919350915089146151da576150728c8c8a613373565b60038501558354615083908a6144ed565b80855560018501546150959190614aac565b6150a38d8d8d8b60016138b9565b505087156150d0576150c68c6121718660010154866144ed90919063ffffffff16565b6150d08c8a614bef565b6000886150e5576150e08c613ec8565b6150ee565b6150ee8c6128d5565b9050600080516020615dfa833981519152868f8f8f8f8f8f886151118c8c6144ed565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820183905251600080516020615dda833981519152918190036101000190a15061533c565b87156151f4576151ea8c84614b70565b6151f48c8a614bef565b600088615209576152048c613ec8565b615212565b6152128c6128d5565b9050600080516020615dfa833981519152868f8f8f8f8f8f886152358c8c6144ed565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b8761534b5761534b8b8a615753565b811561538b578715615365576153658c6121988e85612de7565b60006153718d83612de7565b905061537e8d828a6154db565b9550612504945050505050565b5060009c9b505050505050505050505050565b6000806153bb61271061178e6153b482876144ed565b879061443a565b905060006153c985836144ed565b6001600160a01b0387166000908152602c60205260409020549091506153ef90826146ab565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b8661543481846117b4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b6001600160a01b038116600090815260276020908152604080832054602590925290912054101561482b576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b6154ef6001600160a01b038416828461564c565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b15801561553557600080fd5b505afa158015615549573d6000803e3d6000fd5b505050506040513d602081101561555f57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b60165462010000900460ff16156144eb57336000908152601a60205260409020546144eb9060ff166036614703565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b15801561560057600080fd5b505afa158015615614573d6000803e3d6000fd5b505050506040513d602081101561562a57600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b179052611a6f908490615ab1565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b0385166000908152602690915291909120546156f791839061585d565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d60205260409020548082111561579357506001600160a01b0382166000908152602d6020526040812055611a71565b61579d81836144ed565b6001600160a01b0384166000908152602d6020526040902055505050565b600081836158475760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561580c5781810151838201526020016157f4565b50505050905090810190601f1680156158395780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161585357fe5b0495945050505050565b600081848411156158af5760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561580c5781810151838201526020016157f4565b505050900390565b60008060006158c88989898761198e565b6000818152602b602052604081208054600382015493945090926158f5918d918d918d918b918d91614943565b90506000806000806159168d876000015488600201548d8a60060154612557565b8754919550859350915061592e9061178e8d8461443a565b9250505060008280156159415750600082115b1561597157506005840180548201905580886159715760006159638e84612de7565b905061596f8e82614d86565b505b8215801561597f5750600082115b156159c357600185015461599390836144ed565b6001860155886159b75760006159a98e84612de7565b90506159b58e82614c6e565b505b60058501805483900390555b8a156159eb576159d3818c6146ab565b60018601549091506159e5908c6144ed565b60018601555b84548a1415615a0e576001850154615a049082906146ab565b6000600187015590505b8084811115615a2857615a2182866144ed565b9050615a5c565b6001860154615a3790866144ed565b60018701558915615a5c576000615a4e8f87612de7565b9050615a5a8f82614d86565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615b06826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615b629092919063ffffffff16565b805190915015611a6f57808060200190516020811015615b2557600080fd5b5051611a6f5760405162461bcd60e51b815260040180806020018281038252602a815260200180615e5b602a913960400191505060405180910390fd5b6060611d47848460008585615b7685615c7d565b615bc7576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615c065780518252601f199092019160209182019101615be7565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615c68576040519150601f19603f3d011682016040523d82523d6000602084013e615c6d565b606091505b5091509150612504828286615c83565b3b151590565b60608315615c92575081611dea565b825115615ca25782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561580c5781810151838201526020016157f4565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615d2a5782800160ff19823516178555615d57565b82800160010185558215615d57579182015b82811115615d57578235825591602001919060010190615d3c565b50615d63929150615da4565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615d635760008155600101615da556fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c0020853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b77393d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d309a06bc6ada7d180719cf7588e62ef5b07703f581a986715871a28f1db8a2ae16536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f775361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011aa164736f6c634300060c000a
Loading...
Loading
Loading...
Loading
[ 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.