More Info
Private Name Tags
ContractCreator
Latest 14 from a total of 14 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Approve Plugin | 5354193 | 5 days ago | IN | 0 S | 0.00148489 | ||||
Approve Plugin | 5353523 | 5 days ago | IN | 0 S | 0.00154489 | ||||
Set Gov | 4128521 | 14 days ago | IN | 0 S | 0.00297473 | ||||
Set Token Config | 4128514 | 14 days ago | IN | 0 S | 0.01746789 | ||||
Set Token Config | 4128496 | 14 days ago | IN | 0 S | 0.01746789 | ||||
Set Token Config | 4128488 | 14 days ago | IN | 0 S | 0.01746789 | ||||
Set Token Config | 4128478 | 14 days ago | IN | 0 S | 0.02310957 | ||||
Set Vault Utils | 4127745 | 14 days ago | IN | 0 S | 0.00324203 | ||||
Set Error Contro... | 4127726 | 14 days ago | IN | 0 S | 0.00508794 | ||||
Set Fees | 4127718 | 14 days ago | IN | 0 S | 0.0110308 | ||||
Set Manager | 4127713 | 14 days ago | IN | 0 S | 0.0051392 | ||||
Set In Manager M... | 4127709 | 14 days ago | IN | 0 S | 0.00319099 | ||||
Set Funding Rate | 4127704 | 14 days ago | IN | 0 S | 0.00372823 | ||||
Initialize | 4127698 | 14 days ago | IN | 0 S | 0.01802295 |
Loading...
Loading
Contract Name:
Vault
Compiler Version
v0.6.12+commit.27d51765
Optimization Enabled:
Yes with 10 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/IUSDG.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 USDG_DECIMALS = 18; uint256 public constant MAX_FEE_BASIS_POINTS = 500; // 5% uint256 public constant MAX_LIQUIDATION_FEE_USD = 100 * PRICE_PRECISION; // 100 USD uint256 public constant MIN_FUNDING_RATE_INTERVAL = 1 hours; uint256 public constant MAX_FUNDING_RATE_FACTOR = 10000; // 1% bool public override isInitialized; bool public override isSwapEnabled = true; bool public override isLeverageEnabled = true; IVaultUtils public vaultUtils; address public errorController; address public override router; address public override priceFeed; address public override usdg; 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; // usdgAmounts tracks the amount of USDG debt for each whitelisted token mapping (address => uint256) public override usdgAmounts; // maxUsdgAmounts allows setting a max amount of USDG debt for a token mapping (address => uint256) public override maxUsdgAmounts; // 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 USDG // 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 BuyUSDG(address account, address token, uint256 tokenAmount, uint256 usdgAmount, uint256 feeBasisPoints); event SellUSDG(address account, address token, uint256 usdgAmount, 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 IncreaseUsdgAmount(address token, uint256 amount); event DecreaseUsdgAmount(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 _usdg, address _priceFeed, uint256 _liquidationFeeUsd, uint256 _fundingRateFactor, uint256 _stableFundingRateFactor ) external { _onlyGov(); _validate(!isInitialized, 1); isInitialized = true; router = _router; usdg = _usdg; 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 _maxUsdgAmount, 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; maxUsdgAmounts[_token] = _maxUsdgAmount * (10 ** USDG_DECIMALS); 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 maxUsdgAmounts[_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 setUsdgAmount(address _token, uint256 _amount) external override { _onlyGov(); uint256 usdgAmount = usdgAmounts[_token]; if (_amount > usdgAmount) { _increaseUsdgAmount(_token, _amount.sub(usdgAmount)); return; } _decreaseUsdgAmount(_token, usdgAmount.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 USDG 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 buyUSDG(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 usdgAmount = tokenAmount.mul(price).div(PRICE_PRECISION); usdgAmount = adjustForDecimals(usdgAmount, _token, usdg); _validate(usdgAmount > 0, 18); uint256 feeBasisPoints = vaultUtils.getBuyUsdgFeeBasisPoints(_token, usdgAmount); uint256 amountAfterFees = _collectSwapFees(_token, tokenAmount, feeBasisPoints); uint256 mintAmount = amountAfterFees.mul(price).div(PRICE_PRECISION); mintAmount = adjustForDecimals(mintAmount, _token, usdg); _increaseUsdgAmount(_token, mintAmount); _increasePoolAmount(_token, amountAfterFees); IUSDG(usdg).mint(_receiver, mintAmount); emit BuyUSDG(_receiver, _token, tokenAmount, mintAmount, feeBasisPoints); useSwapPricing = false; return mintAmount; } function sellUSDG(address _token, address _receiver) external override nonReentrant returns (uint256) { _validateManager(); _validate(whitelistedTokens[_token], 19); useSwapPricing = true; uint256 usdgAmount = _transferIn(usdg); _validate(usdgAmount > 0, 20); updateCumulativeFundingRate(_token, _token); uint256 redemptionAmount = getRedemptionAmount(_token, usdgAmount); _validate(redemptionAmount > 0, 21); _decreaseUsdgAmount(_token, usdgAmount); _decreasePoolAmount(_token, redemptionAmount); IUSDG(usdg).burn(address(this), usdgAmount); // the _transferIn call increased the value of tokenBalances[usdg] // usually decreases in token balances are synced by calling _transferOut // however, for usdg, the tokens are burnt, so _updateTokenBalance should // be manually called to record the decrease in tokens _updateTokenBalance(usdg); uint256 feeBasisPoints = vaultUtils.getSellUsdgFeeBasisPoints(_token, usdgAmount); uint256 amountOut = _collectSwapFees(_token, redemptionAmount, feeBasisPoints); _validate(amountOut > 0, 22); _transferOut(_token, amountOut, _receiver); emit SellUSDG(_receiver, _token, usdgAmount, 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 usdgAmounts by the same usdgAmount as debt is shifted between the assets uint256 usdgAmount = amountIn.mul(priceIn).div(PRICE_PRECISION); usdgAmount = adjustForDecimals(usdgAmount, _tokenIn, usdg); uint256 feeBasisPoints = vaultUtils.getSwapFeeBasisPoints(_tokenIn, _tokenOut, usdgAmount); uint256 amountOutAfterFees = _collectSwapFees(_tokenOut, amountOut, feeBasisPoints); _increaseUsdgAmount(_tokenIn, usdgAmount); _decreaseUsdgAmount(_tokenOut, usdgAmount); _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 _usdgAmount) public override view returns (uint256) { uint256 price = getMaxPrice(_token); uint256 redemptionAmount = _usdgAmount.mul(PRICE_PRECISION).div(price); return adjustForDecimals(redemptionAmount, usdg, _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 == usdg ? USDG_DECIMALS : tokenDecimals[_tokenDiv]; uint256 decimalsMul = _tokenMul == usdg ? USDG_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 _usdgDelta, uint256 _feeBasisPoints, uint256 _taxBasisPoints, bool _increment) public override view returns (uint256) { return vaultUtils.getFeeBasisPoints(_token, _usdgDelta, _feeBasisPoints, _taxBasisPoints, _increment); } function getTargetUsdgAmount(address _token) public override view returns (uint256) { uint256 supply = IERC20(usdg).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 _increaseUsdgAmount(address _token, uint256 _amount) private { usdgAmounts[_token] = usdgAmounts[_token].add(_amount); uint256 maxUsdgAmount = maxUsdgAmounts[_token]; if (maxUsdgAmount != 0) { _validate(usdgAmounts[_token] <= maxUsdgAmount, 51); } emit IncreaseUsdgAmount(_token, _amount); } function _decreaseUsdgAmount(address _token, uint256 _amount) private { uint256 value = usdgAmounts[_token]; // since USDG can be minted using multiple assets // it is possible for the USDG debt for a single asset to be less than zero // the USDG debt is capped to zero for this case if (value <= _amount) { usdgAmounts[_token] = 0; emit DecreaseUsdgAmount(_token, value); return; } usdgAmounts[_token] = value.sub(_amount); emit DecreaseUsdgAmount(_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 usdg() external view returns (address); function gov() external view returns (address); function whitelistedTokenCount() external view returns (uint256); function maxLeverage() external view returns (uint256); function minProfitTime() external view returns (uint256); function hasDynamicFees() external view returns (bool); function fundingInterval() external view returns (uint256); function totalTokenWeights() external view returns (uint256); function getTargetUsdgAmount(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 setUsdgAmount(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 _maxUsdgAmount, 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 buyUSDG(address _token, address _receiver) external returns (uint256); function sellUSDG(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 _usdgDelta, 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 usdgAmounts(address _token) external view returns (uint256); function maxUsdgAmounts(address _token) external view returns (uint256); function getRedemptionAmount(address _token, uint256 _usdgAmount) 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 setUseV2Pricing(bool _useV2Pricing) external; function setIsAmmEnabled(bool _isEnabled) 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 getAmmPrice(address _token) external view returns (uint256); function getLatestPrimaryPrice(address _token) 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; }
// 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 getBuyUsdgFeeBasisPoints(address _token, uint256 _usdgAmount) external view returns (uint256); function getSellUsdgFeeBasisPoints(address _token, uint256 _usdgAmount) external view returns (uint256); function getSwapFeeBasisPoints(address _tokenIn, address _tokenOut, uint256 _usdgAmount) external view returns (uint256); function getFeeBasisPoints(address _token, uint256 _usdgDelta, uint256 _feeBasisPoints, uint256 _taxBasisPoints, bool _increment) external view returns (uint256); }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; /** * @dev Wrappers over Solidity's arithmetic operations with added overflow * checks. * * Arithmetic operations in Solidity wrap on overflow. This can easily result * in bugs, because programmers usually assume that an overflow raises an * error, which is the standard behavior in high level programming languages. * `SafeMath` restores this intuition by reverting the transaction when an * operation overflows. * * Using this library instead of the unchecked operations eliminates an entire * class of bugs, so it's recommended to use it always. */ library SafeMath { /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { uint256 c = a + b; require(c >= a, "SafeMath: addition overflow"); return c; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return sub(a, b, "SafeMath: subtraction overflow"); } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b <= a, errorMessage); uint256 c = a - b; return c; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) { return 0; } uint256 c = a * b; require(c / a == b, "SafeMath: multiplication overflow"); return c; } /** * @dev Returns the integer division of two unsigned integers. Reverts on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return div(a, b, "SafeMath: division by zero"); } /** * @dev Returns the integer division of two unsigned integers. Reverts with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b > 0, errorMessage); uint256 c = a / b; // assert(a == b * c + a % b); // There is no case in which this doesn't hold return c; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return mod(a, b, "SafeMath: modulo by zero"); } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts with custom message when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b != 0, errorMessage); return a % b; } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `recipient`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address recipient, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `sender` to `recipient` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address sender, address recipient, uint256 amount) external returns (bool); /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; import "./IERC20.sol"; import "../math/SafeMath.sol"; import "../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using SafeMath for uint256; using Address for address; function safeTransfer(IERC20 token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove(IERC20 token, address spender, uint256 value) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' // solhint-disable-next-line max-line-length require((value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 newAllowance = token.allowance(address(this), spender).add(value); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero"); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); if (returndata.length > 0) { // Return data is optional // solhint-disable-next-line max-line-length require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } }
// SPDX-License-Identifier: MIT pragma solidity ^0.6.2; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize, which returns 0 for contracts in // construction, since the code is only stored at the end of the // constructor execution. uint256 size; // solhint-disable-next-line no-inline-assembly assembly { size := extcodesize(account) } return size > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); // solhint-disable-next-line avoid-low-level-calls, avoid-call-value (bool success, ) = recipient.call{ value: amount }(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain`call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); // solhint-disable-next-line avoid-low-level-calls (bool success, bytes memory returndata) = target.call{ value: value }(data); return _verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data, string memory errorMessage) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); // solhint-disable-next-line avoid-low-level-calls (bool success, bytes memory returndata) = target.staticcall(data); return _verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.3._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.3._ */ function functionDelegateCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); // solhint-disable-next-line avoid-low-level-calls (bool success, bytes memory returndata) = target.delegatecall(data); return _verifyCallResult(success, returndata, errorMessage); } function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly // solhint-disable-next-line no-inline-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; /** * @dev Contract module that helps prevent reentrant calls to a function. * * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier * available, which can be applied to functions to make sure there are no nested * (reentrant) calls to them. * * Note that because there is a single `nonReentrant` guard, functions marked as * `nonReentrant` may not call one another. This can be worked around by making * those functions `private`, and then adding `external` `nonReentrant` entry * points to them. * * TIP: If you would like to learn more about reentrancy and alternative ways * to protect against it, check out our blog post * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul]. */ contract ReentrancyGuard { // Booleans are more expensive than uint256 or any type that takes up a full // word because each write operation emits an extra SLOAD to first read the // slot's contents, replace the bits taken up by the boolean, and then write // back. This is the compiler's defense against contract upgrades and // pointer aliasing, and it cannot be disabled. // The values being non-zero value makes deployment a bit more expensive, // but in exchange the refund on every call to nonReentrant will be lower in // amount. Since refunds are capped to a percentage of the total // transaction's gas, it is best to keep them low in cases like this one, to // increase the likelihood of the full refund coming into effect. uint256 private constant _NOT_ENTERED = 1; uint256 private constant _ENTERED = 2; uint256 private _status; constructor () internal { _status = _NOT_ENTERED; } /** * @dev Prevents a contract from calling itself, directly or indirectly. * Calling a `nonReentrant` function from another `nonReentrant` * function is not supported. It is possible to prevent this from happening * by making the `nonReentrant` function external, and make it call a * `private` function that does the actual work. */ modifier nonReentrant() { // On the first call to nonReentrant, _notEntered will be true require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; _; // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) _status = _NOT_ENTERED; } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; interface IUSDG { function addVault(address _vault) external; function removeVault(address _vault) external; function mint(address _account, uint256 _amount) external; function burn(address _account, uint256 _amount) external; }
{ "optimizer": { "enabled": true, "runs": 10 }, "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":"usdgAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeBasisPoints","type":"uint256"}],"name":"BuyUSDG","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":"DecreaseUsdgAmount","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":"IncreaseUsdgAmount","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":"usdgAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokenAmount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"feeBasisPoints","type":"uint256"}],"name":"SellUSDG","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":"USDG_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":"buyUSDG","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":"_usdgDelta","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":"_usdgAmount","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":"getTargetUsdgAmount","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":"_usdg","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":"maxUsdgAmounts","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":"sellUSDG","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":"_maxUsdgAmount","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":"setUsdgAmount","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":"usdg","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"usdgAmounts","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
60806040526001805462ff00001961ff00199091166101001716620100001781556207a1206008556032600a9081556014600b55601e600c819055600d556004600e55600f556011805460ff199081169091556170806012556016805463ffffff0019921690921716905534801561007657600080fd5b506001600055600680546001600160a01b03191633179055615f0d806200009e6000396000f3fe608060405234801561001057600080fd5b50600436106105995760003560e01c80638585f4d2116102e9578063c7e074c31161018f578063c7e074c314611225578063cea0c32814611265578063cfad57a21461129b578063d2fa635e146112c1578063d3127e63146112de578063d54d5a9f146112fb578063d66b000d1461135a578063d8f897c314611386578063d9ac4225146113ac578063da76524c146113b4578063daf9c210146113fe578063db3555fb14611424578063db97495f1461144a578063dc8f5fac14611496578063de2ea9481461149e578063df73a267146114e6578063e124e6d2146114ee578063e468baf014611514578063e67f59a714611531578063efa10a6e14611557578063f07456ce14611583578063f07bbf77146115a9578063f2555278146115c8578063f3ae2415146115f6578063f5b91b7b1461161c578063f887ea4014611624578063fa12dbc01461162c578063fbfded6d1461165e578063fce28c101461168c578063fdaf6ac314611694578063fed1a606146116d857610599565b80638585f4d214610e90578063870d917c14610ebc57806388b1fbdf14610ec45780638a27d46814610eea5780638a39735a1461061a5780638a78daa814610f135780638ee573ac14610f395780638f7b840414610f5f5780639060b1ca14610f855780639331621214610f8d57806395082d2514610fc55780639698d25a14610fcd5780639849e41214610ff35780639899cd0214610ffb5780639d7432ca146110275780639f392eb314611059578063a22f239214611061578063a42ab3d214611069578063a5e90eee14611095578063a93acac2146110c3578063ab08c1c6146110e9578063ab2f3ad4146110f1578063ad1e4f8d14611117578063ae3302c21461113d578063b06423f314611145578063b136ca491461114d578063b1cc53ab14611173578063b364accb146111ab578063c3c7b9e9146111d1578063c4f718bf146111f7578063c65bc7b1146111ff57610599565b80634453a3741161044e5780634453a3741461097057806345a6f3701461099e57806348d91abf146109f757806348f35cbb14610a3b5780634a3f088d14610a435780634a993ee914610ac45780634befe2ca14610aea5780634d47b30414610af2578063514ea4bf14610afa57806351723e8214610b4f578063523fba7f14610b8d578063529a356f14610bb357806352f55eed14610bd95780635c07eaab14610bff5780635f7bc11914610c3f5780636092219914610c6557806362287a3214610c935780636274980314610c9b5780636abbe0c814610cc15780636ae0b15414610cc95780636be6026b14610cef57806371089f4d14610cf7578063711e619014610d1d578063724e78da14610d4b578063728cdbca14610d71578063741bef1a14610db95780637a210a2b14610dc15780637c2eb9f714610dc9578063817bb85714610de857806381a612d614610e1657806382a0849014610e3c57610599565b806304fef1db1461059e57806307c58752146105d65780630842b076146105de5780630a48d5a9146105e657806310eb56c214610612578063126082cf1461061a57806312d43a5114610622578063134ca63b14610646578063174d26941461064e578063181e210e146106565780631aa4ace5146106725780631ce9cb8f1461069857806324b0c04d146106be57806324ca984e146106df57806328e67be51461070557806329ff96151461077a5780632c668ec1146107a05780632d4b0576146107cc57806330455ede1461080a578063318bc6891461082957806334c1557d1461061a578063351a964d14610831578063392e53cd146108395780633a05dcc1146108415780633c5a6e35146108675780633de39c11146108b55780633e72a262146108bd57806340eb3802146108c5578063421528731461091657806342b60b031461094a575b600080fd5b6105c4600480360360208110156105b457600080fd5b50356001600160a01b031661176a565b60408051918252519081900360200190f35b6105c46117cd565b6105c46117de565b6105c4600480360360408110156105fc57600080fd5b506001600160a01b0381351690602001356117e4565b6105c4611837565b6105c461183d565b61062a611843565b604080516001600160a01b039092168252519081900360200190f35b6105c4611852565b6105c4611858565b61065e61185e565b604080519115158252519081900360200190f35b6105c46004803603602081101561068857600080fd5b50356001600160a01b031661186e565b6105c4600480360360208110156106ae57600080fd5b50356001600160a01b0316611880565b6106dd600480360360208110156106d457600080fd5b50351515611892565b005b6106dd600480360360208110156106f557600080fd5b50356001600160a01b03166118b6565b6106dd6004803603604081101561071b57600080fd5b81359190810190604081016020820135600160201b81111561073c57600080fd5b82018360208201111561074e57600080fd5b803590602001918460018302840111600160201b8311171561076f57600080fd5b5090925090506118e7565b6105c46004803603602081101561079057600080fd5b50356001600160a01b0316611965565b6105c4600480360360408110156107b657600080fd5b506001600160a01b038135169060200135611979565b6105c4600480360360808110156107e257600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156119be565b6106dd6004803603602081101561082057600080fd5b50351515611a14565b6105c4611a36565b61065e611a3c565b61065e611a4a565b6105c46004803603602081101561085757600080fd5b50356001600160a01b0316611a53565b6106dd600480360360e081101561087d57600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611b13565b6105c4611c55565b61065e611c5b565b6106dd60048036036101208110156108dc57600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611c6a565b6105c46004803603606081101561092c57600080fd5b508035906001600160a01b0360208201358116916040013516611d24565b61065e6004803603602081101561096057600080fd5b50356001600160a01b0316611dc6565b6106dd6004803603604081101561098657600080fd5b506001600160a01b0381351690602001351515611ddb565b6109dc600480360360808110156109b457600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611e0e565b60408051921515835260208301919091528051918290030190f35b6106dd600480360360a0811015610a0d57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013590608001351515611eac565b61062a6122d9565b610a8160048036036080811015610a5957600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156122e8565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b6105c460048036036020811015610ada57600080fd5b50356001600160a01b03166123de565b6105c46123f0565b6105c46123f6565b610b1760048036036020811015610b1057600080fd5b50356123fc565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b6105c460048036036080811015610b6557600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612439565b6105c460048036036020811015610ba357600080fd5b50356001600160a01b03166124df565b61065e60048036036020811015610bc957600080fd5b50356001600160a01b03166124f1565b6105c460048036036020811015610bef57600080fd5b50356001600160a01b0316612506565b6109dc600480360360a0811015610c1557600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060800135612518565b6106dd60048036036020811015610c5557600080fd5b50356001600160a01b0316612610565b61065e60048036036040811015610c7b57600080fd5b506001600160a01b03813581169160200135166126f3565b6105c4612713565b6105c460048036036020811015610cb157600080fd5b50356001600160a01b0316612719565b61062a61272b565b6106dd60048036036020811015610cdf57600080fd5b50356001600160a01b0316612741565b6105c461276f565b6106dd60048036036020811015610d0d57600080fd5b50356001600160a01b0316612776565b6105c460048036036040811015610d3357600080fd5b506001600160a01b03813581169160200135166127aa565b6106dd60048036036020811015610d6157600080fd5b50356001600160a01b0316612a3e565b6106dd600480360360c0811015610d8757600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a00135612a68565b61062a612adf565b6105c4612aee565b6106dd60048036036020811015610ddf57600080fd5b50351515612af4565b6105c460048036036040811015610dfe57600080fd5b506001600160a01b0381358116916020013516612b18565b6105c460048036036020811015610e2c57600080fd5b50356001600160a01b0316612de8565b6105c4600480360360e0811015610e5257600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612e93565b6106dd60048036036040811015610ea657600080fd5b506001600160a01b038135169060200135612f11565b6105c4612f35565b6105c460048036036020811015610eda57600080fd5b50356001600160a01b0316612f3a565b6106dd60048036036060811015610f0057600080fd5b5080359060208101359060400135612f4c565b6105c460048036036020811015610f2957600080fd5b50356001600160a01b0316612f92565b6105c460048036036020811015610f4f57600080fd5b50356001600160a01b0316612fa4565b6106dd60048036036020811015610f7557600080fd5b50356001600160a01b0316612fb6565b61065e612fe0565b6105c460048036036060811015610fa357600080fd5b506001600160a01b038135811691602081013582169160409091013516612fef565b6105c46132d7565b6105c460048036036020811015610fe357600080fd5b50356001600160a01b03166132e7565b6105c46132f9565b6105c46004803603604081101561101157600080fd5b506001600160a01b0381351690602001356132ff565b6105c46004803603606081101561103d57600080fd5b506001600160a01b038135169060208101359060400135613321565b61065e6133ca565b6105c46133d3565b6105c46004803603604081101561107f57600080fd5b506001600160a01b0381351690602001356133d9565b6106dd600480360360408110156110ab57600080fd5b506001600160a01b03813516906020013515156133f6565b6105c4600480360360208110156110d957600080fd5b50356001600160a01b0316613429565b61065e613527565b6105c46004803603602081101561110757600080fd5b50356001600160a01b0316613530565b6105c46004803603602081101561112d57600080fd5b50356001600160a01b0316613542565b6105c4613554565b61065e61355a565b6105c46004803603602081101561116357600080fd5b50356001600160a01b0316613568565b6105c46004803603606081101561118957600080fd5b506001600160a01b038135811691602081013590911690604001351515613609565b6109dc600480360360208110156111c157600080fd5b50356001600160a01b03166136a4565b6105c4600480360360208110156111e757600080fd5b50356001600160a01b031661373c565b6105c461374e565b6105c46004803603602081101561121557600080fd5b50356001600160a01b0316613754565b6105c4600480360360a081101561123b57600080fd5b506001600160a01b0381351690602081013590604081013590606081013590608001351515613766565b6106dd6004803603606081101561127b57600080fd5b506001600160a01b03813581169160208101359091169060400135613810565b6106dd600480360360208110156112b157600080fd5b50356001600160a01b0316613831565b6106dd600480360360208110156112d757600080fd5b503561385b565b6106dd600480360360208110156112f457600080fd5b5035613868565b611341600480360360a081101561131157600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001351515613884565b6040805192835260208301919091528051918290030190f35b6106dd6004803603604081101561137057600080fd5b506001600160a01b03813516906020013561393b565b6105c46004803603602081101561139c57600080fd5b50356001600160a01b0316613994565b6105c46139a6565b6105c4600480360360c08110156113ca57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a001356139ac565b61065e6004803603602081101561141457600080fd5b50356001600160a01b0316613a60565b61065e6004803603602081101561143a57600080fd5b50356001600160a01b0316613a75565b6105c4600480360360e081101561146057600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c00135613a89565b6105c4613b1a565b6106dd600480360360a08110156114b457600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613b20565b6105c4613ee6565b6105c46004803603602081101561150457600080fd5b50356001600160a01b0316613eec565b61062a6004803603602081101561152a57600080fd5b5035613f64565b6106dd6004803603602081101561154757600080fd5b50356001600160a01b0316613f8b565b6106dd6004803603604081101561156d57600080fd5b506001600160a01b03813516906020013561405e565b6105c46004803603602081101561159957600080fd5b50356001600160a01b0316614082565b6106dd600480360360208110156115bf57600080fd5b50351515614094565b6105c4600480360360408110156115de57600080fd5b506001600160a01b03813581169160200135166140ba565b61065e6004803603602081101561160c57600080fd5b50356001600160a01b0316614110565b61062a614125565b61062a614134565b6105c46004803603606081101561164257600080fd5b506001600160a01b038135169060208101359060400135614143565b6106dd6004803603604081101561167457600080fd5b506001600160a01b038135811691602001351661417d565b6105c4614356565b6105c4600480360360a08110156116aa57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101351515906080013561435c565b6116f5600480360360208110156116ee57600080fd5b50356143d2565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561172f578181015183820152602001611717565b50505050905090810190601f16801561175c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b038116600090815260256020526040812054806117925760009150506117c8565b6001600160a01b0383166000908152602660205260409020546117c49082906117be90620f424061446d565b906144c6565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117f357506000611831565b60006117fe84612de8565b6001600160a01b0385166000908152601d602052604090205490915061182c600a82900a6117be868561446d565b925050505b92915050565b600b5481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b60236020526000908152604090205481565b602c6020526000908152604090205481565b61189a614505565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611946576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061195f908383615d1c565b50505050565b60006118318261197484613568565b6117e4565b60008061198584613eec565b905060006119a3826117be8668327cb2734119d3b7a9601e1b61446d565b60055490915061182c9082906001600160a01b031687611d24565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b611a1c614505565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b60015460ff1681565b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611aa457600080fd5b505afa158015611ab8573d6000803e3d6000fd5b505050506040513d6020811015611ace57600080fd5b5051905080611ae15760009150506117c8565b6001600160a01b038316600090815260226020526040902054601554611b0b906117be838561446d565b949350505050565b611b1b614505565b6001600160a01b0387166000908152601c602052604090205460ff16611b9857600754611b49906001614520565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611bbf908290614578565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420670de0b6b3a76400008a029055601f83528184208054821689151517905591805290912080549091168415151790559050611c488187614520565b6015555050505050505050565b60175481565b60015462010000900460ff1681565b611c72614505565b611c826101f48a111560036145ba565b611c926101f489111560046145ba565b611ca26101f488111560056145ba565b611cb26101f487111560066145ba565b611cc26101f486111560076145ba565b611cd26101f485111560086145ba565b611ced6904ee2d6d415b85acef8160201b84111560096145ba565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b60055460009081906001600160a01b03858116911614611d5c576001600160a01b0384166000908152601d6020526040902054611d5f565b60125b6005549091506000906001600160a01b03858116911614611d98576001600160a01b0384166000908152601d6020526040902054611d9b565b60125b9050611dba82600a0a6117be83600a0a8961446d90919063ffffffff16565b925050505b9392505050565b601f6020526000908152604090205460ff1681565b611de3614505565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611e1f878787876119be565b9050611e29615d9a565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c082018190529092611e9d9289929091908990612518565b93509350505094509492505050565b60026000541415611ef2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b6002600055600154611f0e9062010000900460ff16601c6145ba565b611f16614666565b611f1f85614682565b611f2a8484836146e5565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b158015611f9e57600080fd5b505afa158015611fb2573d6000803e3d6000fd5b50505050611fc0848461417d565b6000611fce868686856119be565b6000818152602b6020526040812091925083611ff257611fed86612de8565b611ffb565b611ffb86613eec565b825490915061200c57600282018190555b81541580159061201c5750600085115b156120425761203c868360000154846002015487858a8860060154613a89565b60028301555b600061205b898989888a886000015489600301546147fa565b90506000612068896148b6565b905060006120768a836117e4565b60018601549091506120889082614520565b6001860181905561209d90841115601d6145ba565b60018501546120ac9084614578565b60018601556120bc8a8a89613609565b600386015584546120cd9089614520565b8086554260068701556120e3901515601e6145ba565b6120f585600001548660010154614963565b6121038b8b8b8a6001613884565b505060006121118b8a6133d9565b60048701549091506121239082614520565b60048701556121328b82614987565b87156121775761214b8b6121468b87614520565b614a27565b6121558b83614aa6565b61215f8b84614b25565b6121728b61216d8d876132ff565b614c3d565b6121e3565b6001600160a01b038a166000908152602d60205260409020546121b4576001600160a01b038a166000908152602e602052604090208590556121d9565b6121bf8a868b613321565b6001600160a01b038b166000908152602e60205260409020555b6121e38a8a614d10565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820187905251600080516020615e0d833981519152918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b60008060008060008060008060006123028d8d8d8d6119be565b905061230c615d9a565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a001511361238c578160a00151600003612392565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b600080612448868686866119be565b9050612452615d9a565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c08301526124bc90151560256145ba565b602081015181516124d491906117be9061271061446d565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6000806125296000861160266145ba565b60008461253e5761253988613eec565b612547565b61254788612de8565b905060008187116125615761255c8288614578565b61256b565b61256b8783614578565b9050600061257d886117be8b8561446d565b9050600087156125905750878311612595565b508288115b60006125ac6010548961452090919063ffffffff16565b42116125d0576001600160a01b038c166000908152601e60205260409020546125d3565b60005b90508180156125f657506125e78b8261446d565b6125f38461271061446d565b11155b1561260057600092505b509a909950975050505050505050565b60026000541415612656576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546126819060ff16600e6145ba565b600061268c826148b6565b905061269c60008211600f6145ba565b6126a68282614b25565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b61277e614505565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b6000600260005414156127f2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b60026000556127ff614dc8565b6001600160a01b0383166000908152601c60205260409020546128269060ff1660136145ba565b6016805461ff00191661010017905560055460009061284d906001600160a01b03166148b6565b905061285d6000821160146145ba565b612867848561417d565b60006128738583611979565b90506128836000821160156145ba565b61288d8583614df7565b6128978582614c3d565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b1580156128ea57600080fd5b505af11580156128fe573d6000803e3d6000fd5b505060055461291892506001600160a01b03169050614eb5565b6001546040805163eb0835bf60e01b81526001600160a01b038881166004830152602482018690529151600093630100000090049092169163eb0835bf91604480820192602092909190829003018186803b15801561297657600080fd5b505afa15801561298a573d6000803e3d6000fd5b505050506040513d60208110156129a057600080fd5b5051905060006129b1878484614f50565b90506129c16000821160166145ba565b6129cc878288615018565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517fd732b7828fa6cee72c285eac756fc66a7477e3dc22e22e7c432f1c265d40b4839181900360a00190a16016805461ff001916905560016000559695505050505050565b612a46614505565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b612a70614505565b60018054612a849160ff90911615906145ba565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b6004546001600160a01b031681565b600a5481565b612afc614505565b60018054911515620100000262ff000019909216919091179055565b600060026000541415612b60576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b6002600055612b6d614dc8565b6001600160a01b0383166000908152601c6020526040902054612b949060ff1660106145ba565b6016805461ff0019166101001790556000612bae846148b6565b9050612bbe6000821160116145ba565b612bc8848561417d565b6000612bd385612de8565b90506000612bf168327cb2734119d3b7a9601e1b6117be858561446d565b600554909150612c0d90829088906001600160a01b0316611d24565b9050612c1d6000821160126145ba565b6001546040805163256f6ee360e11b81526001600160a01b0389811660048301526024820185905291516000936301000000900490921691634adeddc691604480820192602092909190829003018186803b158015612c7b57600080fd5b505afa158015612c8f573d6000803e3d6000fd5b505050506040513d6020811015612ca557600080fd5b505190506000612cb6888684614f50565b90506000612cd468327cb2734119d3b7a9601e1b6117be848861446d565b600554909150612cf09082908b906001600160a01b0316611d24565b9050612cfc89826150bf565b612d068983614b25565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b158015612d5b57600080fd5b505af1158015612d6f573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517fab4c77c74cd32c85f35416cf03e7ce9e2d4387f7b7f2c1f4bf53daaecf8ea72d93509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b158015612e6157600080fd5b505afa158015612e75573d6000803e3d6000fd5b505050506040513d6020811015612e8b57600080fd5b505192915050565b600060026000541415612edb576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b6002600055612ee8614666565b612ef188614682565b612f008888888888888861517d565b600160005598975050505050505050565b612f19614505565b6001600160a01b03909116600090815260276020526040902055565b601281565b601e6020526000908152604090205481565b612f54614505565b612f64610e10841015600a6145ba565b612f74612710831115600b6145ba565b612f84612710821115600c6145ba565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612fbe614505565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415613037576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b600260005560015461305290610100900460ff1660176145ba565b6001600160a01b0384166000908152601c60205260409020546130799060ff1660186145ba565b6001600160a01b0383166000908152601c60205260409020546130a09060ff1660196145ba565b6130c0836001600160a01b0316856001600160a01b03161415601a6145ba565b6016805461ff0019166101001790556130d9848061417d565b6130e3838461417d565b60006130ee856148b6565b90506130fe60008211601b6145ba565b600061310986612de8565b9050600061311686613eec565b90506000613128826117be868661446d565b9050613135818989611d24565b9050600061315368327cb2734119d3b7a9601e1b6117be878761446d565b60055490915061316f9082908b906001600160a01b0316611d24565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b1580156131d957600080fd5b505afa1580156131ed573d6000803e3d6000fd5b505050506040513d602081101561320357600080fd5b5051905060006132148a8584614f50565b90506132208b846150bf565b61322a8a84614df7565b6132348b88614b25565b61323e8a85614c3d565b6132478a61560a565b6132528a828b615018565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b60008161330e57506000611831565b611dbf838361331c86613eec565b614143565b6001600160a01b0383166000908152602d6020908152604080832054602e9092528220548285821161335c576133578683614578565b613366565b6133668287614578565b90506000613378836117be868561446d565b905086831160006133898689614520565b90506000826133a15761339c8285614520565b6133ab565b6133ab8285614578565b90506133bb816117be8c8561446d565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b6000816133e857506000611831565b611dbf838361331c86612de8565b6133fe614505565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a60205260408120549091429161345291614520565b1115613460575060006117c8565b6012546001600160a01b0383166000908152602a6020526040812054909161348d916117be904290614578565b6001600160a01b038416600090815260256020526040902054909150806134b9576000925050506117c8565b6001600160a01b0384166000908152601f602052604081205460ff166134e1576013546134e5565b6014545b6001600160a01b03861660009081526026602052604090205490915061351e9083906117be90869061351890869061446d565b9061446d565b95945050505050565b60165460ff1681565b60226020526000908152604090205481565b60246020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff16156135a857506001600160a01b0381166000908152602560205260409020546117c8565b6001600160a01b0382166000908152602860205260408120546135cc9084906132ff565b6001600160a01b0384166000908152602660209081526040808320546025909252909120549192506117c491613603908490614520565b90614578565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b15801561367057600080fd5b505afa158015613684573d6000803e3d6000fd5b505050506040513d602081101561369a57600080fd5b5051949350505050565b6001600160a01b0381166000908152602d60205260408120548190806136d1576000809250925050613737565b60006136dc85613eec565b6001600160a01b0386166000908152602e602052604081205491925082821161370e576137098383614578565b613718565b6137188284614578565b9050600061372a836117be878561446d565b9390921195509193505050505b915091565b60266020526000908152604090205481565b60135481565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b1580156137da57600080fd5b505afa1580156137ee573d6000803e3d6000fd5b505050506040513d602081101561380457600080fd5b50519695505050505050565b613818614505565b61382c6001600160a01b038316848361567f565b505050565b613839614505565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b613863614505565b601755565b613870614505565b61387f612710821160026145ba565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b1580156138fa57600080fd5b505afa15801561390e573d6000803e3d6000fd5b505050506040513d604081101561392457600080fd5b508051602090910151909890975095505050505050565b613943614505565b6001600160a01b0382166000908152602360205260409020548082111561397d57613977836139728484614578565b6150bf565b50613990565b61382c8361398b8385614578565b614df7565b5050565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b158015613a2957600080fd5b505afa158015613a3d573d6000803e3d6000fd5b505050506040513d6020811015613a5357600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b6000806000613a9b8a8a8a8a88612518565b90925090506000613aac8a87614520565b905060008815613adb5783613aca57613ac58284614578565b613ad4565b613ad48284614520565b9050613afc565b83613aef57613aea8284614520565b613af9565b613af98284614578565b90505b613b0a816117be8a8561446d565b9c9b505050505050505050505050565b60155481565b60026000541415613b66576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613b9b5733600090815260196020526040902054613b9b9060ff1660226145ba565b6016805460ff19169055613baf848461417d565b6000613bbd868686866119be565b9050613bc7615d9a565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613c3390151560236145ba565b600080613c44898989896000613884565b91509150613c57826000141560246145ba565b8160021415613c8c57613c74898989600087600001518b8f61517d565b50506016805460ff1916600117905550613eda915050565b6000613c9889836132ff565b6001600160a01b038a166000908152602c6020526040902054909150613cbe9082614520565b6001600160a01b038a166000818152602c60209081526040918290209390935580519182529181018490528082018390529051600080516020615eb88339815191529181900360600190a1613d178985608001516156d1565b8615613d465760208401518451613d38918b91613d3391614578565b614aa6565b613d468961216d8b856132ff565b600087613d5b57613d5689613eec565b613d64565b613d6489612de8565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613e235750846020015183105b15613e51576020850151600090613e3a9085614578565b9050613e4f8b613e4a8d846132ff565b614b25565b505b87613e6457613e64898660000151615786565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613eb0908b9061216d9082906132ff565b613ec68a613ec08c6009546132ff565b89615018565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b158015612e6157600080fd5b601b8181548110613f7157fe5b6000918252602090912001546001600160a01b0316905081565b613f93614505565b6001600160a01b0381166000908152601c6020526040902054613fba9060ff16600d6145ba565b6001600160a01b038116600090815260226020526040902054601554613fdf91614578565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f835281842080548216905591805290912080549091169055600754614058906001614578565b60075550565b614066614505565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b61409c614505565b6016805491151563010000000263ff00000019909216919091179055565b60006140c4614505565b6001600160a01b0383166000908152602c6020526040902054806140ec576000915050611831565b6001600160a01b0384166000908152602c6020526040812055611dbf848285615018565b601a6020526000908152604090205460ff1681565b6005546001600160a01b031681565b6003546001600160a01b031681565b60008261415257506000611dbf565b6001600160a01b0384166000908152601d602052604090205461351e836117be86600a85900a61446d565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b1580156141dd57600080fd5b505af11580156141f1573d6000803e3d6000fd5b505050506040513d602081101561420757600080fd5b50519050806142165750613990565b6001600160a01b0383166000908152602a6020526040902054614263576012546142449061351842826144c6565b6001600160a01b0384166000908152602a602052604090205550613990565b6012546001600160a01b0384166000908152602a6020526040902054429161428b9190614520565b11156142975750613990565b60006142a284613429565b6001600160a01b0385166000908152602960205260409020549091506142c89082614520565b6001600160a01b0385166000908152602960205260409020556012546142f29061351842826144c6565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b1580156137da57600080fd5b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144655780601f1061443a57610100808354040283529160200191614465565b820191906000526020600020905b81548152906001019060200180831161444857829003601f168201915b505050505081565b60008261447c57506000611831565b8282028284828161448957fe5b0414611dbf5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e4d6021913960400191505060405180910390fd5b6000611dbf83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b8152506157ee565b60065461451e906001600160a01b0316331460356145ba565b565b600082820183811015611dbf576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b6000611dbf83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f770000815250615890565b60008181526030602052604090208261382c5760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156146575780601f1061462c57610100808354040283529160200191614657565b820191906000526020600020905b81548152906001019060200180831161463a57829003601f168201915b50509250505060405180910390fd5b6017546146725761451e565b61451e6017543a111560376145ba565b336001600160a01b0382161415614698576146e2565b6003546001600160a01b03163314156146b0576146e2565b6001600160a01b03811660009081526018602090815260408083203384529091529020546146e29060ff1660296145ba565b50565b801561475e5761470a826001600160a01b0316846001600160a01b031614602a6145ba565b6001600160a01b0383166000908152601c60205260409020546147319060ff16602b6145ba565b6001600160a01b0383166000908152601f60205260409020546147599060ff1615602c6145ba565b61382c565b6001600160a01b0383166000908152601c60205260409020546147859060ff16602d6145ba565b6001600160a01b0383166000908152601f60205260409020546147ac9060ff16602e6145ba565b6001600160a01b0382166000908152601f60205260409020546147d49060ff1615602f6145ba565b6001600160a01b038216600090815260208052604090205461382c9060ff1660306145ba565b60008061480a898989898961435c565b9050600061481c8a8a8a8a89896139ac565b90506148288282614520565b915060006148368a846132ff565b6001600160a01b038b166000908152602c602052604090205490915061485c9082614520565b6001600160a01b038b166000818152602c60209081526040918290209390935580519182529181018590528082018390529051600080516020615eb88339815191529181900360600190a150909998505050505050505050565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b15801561491057600080fd5b505afa158015614924573d6000803e3d6000fd5b505050506040513d602081101561493a57600080fd5b50516001600160a01b03851660009081526021602052604090208190559050611b0b8183614578565b8161497957614974811560276145ba565b613990565b6139908183101560286145ba565b6001600160a01b0382166000908152602660205260409020546149aa9082614520565b6001600160a01b0383166000908152602660208181526040808420859055602582529092205491526149df91111560346145ba565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614a4a9082614520565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614ac99082614578565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614b489082614520565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614b9e57600080fd5b505afa158015614bb2573d6000803e3d6000fd5b505050506040513d6020811015614bc857600080fd5b50516001600160a01b038416600090815260256020526040902054909150614bf49082101560316145ba565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b038516600090815260259091529190912054614c95918390615890565b6001600160a01b03831660009081526025602090815260408083208490556026909152902054614cc891101560326145ba565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054614d339082614520565b6001600160a01b0383166000908152602d6020908152604080832093909355602f90522054801561382c576001600160a01b0383166000908152602d602052604090205481101561382c576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b60165462010000900460ff161561451e57336000908152601a602052604090205461451e9060ff1660366145ba565b6001600160a01b038216600090815260236020526040902054818111614e60576001600160a01b038316600081815260236020908152604080832092909255815192835282018390528051600080516020615e6e8339815191529281900390910190a150613990565b614e6a8183614578565b6001600160a01b0384166000818152602360209081526040918290209390935580519182529181018490528151600080516020615e6e833981519152929181900390910190a1505050565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b158015614f0457600080fd5b505afa158015614f18573d6000803e3d6000fd5b505050506040513d6020811015614f2e57600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b600080614f6d6127106117be614f668287614578565b879061446d565b90506000614f7b8583614578565b6001600160a01b0387166000908152602c6020526040902054909150614fa19082614520565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b86614fe681846117e4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b61502c6001600160a01b038416828461567f565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b15801561507257600080fd5b505afa158015615086573d6000803e3d6000fd5b505050506040513d602081101561509c57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b6001600160a01b0382166000908152602360205260409020546150e29082614520565b6001600160a01b0383166000908152602360209081526040808320939093556024905220548015615134576001600160a01b0383166000908152602360205260409020546151349082101560336145ba565b604080516001600160a01b03851681526020810184905281517f64243679a443432e2293343b77d411ff6144370404618f00ca0d2025d9ca9882929181900390910190a1505050565b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b15801561520157600080fd5b505afa158015615215573d6000803e3d6000fd5b50505050615223878761417d565b6000615231898989876119be565b6000818152602b60205260409020805491925090615252901515601f6145ba565b615264868260000154101560206145ba565b615276878260010154101560216145ba565b600181015481546004830154600091615293916117be908b61446d565b60048401549091506152a59082614578565b60048401556152b48b826156d1565b506000806152c68d8d8d8d8d8d6158ea565b855491935091508914615446576152de8c8c8a613609565b600385015583546152ef908a614578565b80855560018501546153019190614963565b61530f8d8d8d8b6001613884565b5050871561533c576153328c61214686600101548661457890919063ffffffff16565b61533c8c8a614aa6565b6000886153515761534c8c613eec565b61535a565b61535a8c612de8565b9050600080516020615e2d833981519152868f8f8f8f8f8f8861537d8c8c614578565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820183905251600080516020615e0d833981519152918190036101000190a1506155a8565b8715615460576154568c84614a27565b6154608c8a614aa6565b600088615475576154708c613eec565b61547e565b61547e8c612de8565b9050600080516020615e2d833981519152868f8f8f8f8f8f886154a18c8c614578565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b876155b7576155b78b8a615786565b81156155f75787156155d1576155d18c61216d8e856132ff565b60006155dd8d836132ff565b90506155ea8d828a615018565b95506124d4945050505050565b5060009c9b505050505050505050505050565b6001600160a01b03811660009081526027602090815260408083205460259092529091205410156146e2576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b17905261382c908490615ae4565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b03851660009081526026909152919091205461572a918390615890565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054808211156157c657506001600160a01b0382166000908152602d6020526040812055613990565b6157d08183614578565b6001600160a01b0384166000908152602d6020526040902055505050565b6000818361587a5760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561583f578181015183820152602001615827565b50505050905090810190601f16801561586c5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161588657fe5b0495945050505050565b600081848411156158e25760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561583f578181015183820152602001615827565b505050900390565b60008060006158fb898989876119be565b6000818152602b60205260408120805460038201549394509092615928918d918d918d918b918d916147fa565b90506000806000806159498d876000015488600201548d8a60060154612518565b87549195508593509150615961906117be8d8461446d565b9250505060008280156159745750600082115b156159a457506005840180548201905580886159a45760006159968e846132ff565b90506159a28e82614c3d565b505b821580156159b25750600082115b156159f65760018501546159c69083614578565b6001860155886159ea5760006159dc8e846132ff565b90506159e88e82614b25565b505b60058501805483900390555b8a15615a1e57615a06818c614520565b6001860154909150615a18908c614578565b60018601555b84548a1415615a41576001850154615a37908290614520565b6000600187015590505b8084811115615a5b57615a548286614578565b9050615a8f565b6001860154615a6a9086614578565b60018701558915615a8f576000615a818f876132ff565b9050615a8d8f82614c3d565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615b39826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615b959092919063ffffffff16565b80519091501561382c57808060200190516020811015615b5857600080fd5b505161382c5760405162461bcd60e51b815260040180806020018281038252602a815260200180615e8e602a913960400191505060405180910390fd5b6060611b0b848460008585615ba985615cb0565b615bfa576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615c395780518252601f199092019160209182019101615c1a565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615c9b576040519150601f19603f3d011682016040523d82523d6000602084013e615ca0565b606091505b50915091506124d4828286615cb6565b3b151590565b60608315615cc5575081611dbf565b825115615cd55782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561583f578181015183820152602001615827565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615d5d5782800160ff19823516178555615d8a565b82800160010185558215615d8a579182015b82811115615d8a578235825591602001919060010190615d6f565b50615d96929150615dd7565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615d965760008155600101615dd856fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c0020853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b77393d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f77e1e812596aac93a06ecc4ca627014d18e30f5c33b825160cc9d5c0ba61e452275361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011aa264697066735822122090c5359baef33c62d122bd4f925cebe6b7c9c9103277455b0a68532b74ccbacd64736f6c634300060c0033
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106105995760003560e01c80638585f4d2116102e9578063c7e074c31161018f578063c7e074c314611225578063cea0c32814611265578063cfad57a21461129b578063d2fa635e146112c1578063d3127e63146112de578063d54d5a9f146112fb578063d66b000d1461135a578063d8f897c314611386578063d9ac4225146113ac578063da76524c146113b4578063daf9c210146113fe578063db3555fb14611424578063db97495f1461144a578063dc8f5fac14611496578063de2ea9481461149e578063df73a267146114e6578063e124e6d2146114ee578063e468baf014611514578063e67f59a714611531578063efa10a6e14611557578063f07456ce14611583578063f07bbf77146115a9578063f2555278146115c8578063f3ae2415146115f6578063f5b91b7b1461161c578063f887ea4014611624578063fa12dbc01461162c578063fbfded6d1461165e578063fce28c101461168c578063fdaf6ac314611694578063fed1a606146116d857610599565b80638585f4d214610e90578063870d917c14610ebc57806388b1fbdf14610ec45780638a27d46814610eea5780638a39735a1461061a5780638a78daa814610f135780638ee573ac14610f395780638f7b840414610f5f5780639060b1ca14610f855780639331621214610f8d57806395082d2514610fc55780639698d25a14610fcd5780639849e41214610ff35780639899cd0214610ffb5780639d7432ca146110275780639f392eb314611059578063a22f239214611061578063a42ab3d214611069578063a5e90eee14611095578063a93acac2146110c3578063ab08c1c6146110e9578063ab2f3ad4146110f1578063ad1e4f8d14611117578063ae3302c21461113d578063b06423f314611145578063b136ca491461114d578063b1cc53ab14611173578063b364accb146111ab578063c3c7b9e9146111d1578063c4f718bf146111f7578063c65bc7b1146111ff57610599565b80634453a3741161044e5780634453a3741461097057806345a6f3701461099e57806348d91abf146109f757806348f35cbb14610a3b5780634a3f088d14610a435780634a993ee914610ac45780634befe2ca14610aea5780634d47b30414610af2578063514ea4bf14610afa57806351723e8214610b4f578063523fba7f14610b8d578063529a356f14610bb357806352f55eed14610bd95780635c07eaab14610bff5780635f7bc11914610c3f5780636092219914610c6557806362287a3214610c935780636274980314610c9b5780636abbe0c814610cc15780636ae0b15414610cc95780636be6026b14610cef57806371089f4d14610cf7578063711e619014610d1d578063724e78da14610d4b578063728cdbca14610d71578063741bef1a14610db95780637a210a2b14610dc15780637c2eb9f714610dc9578063817bb85714610de857806381a612d614610e1657806382a0849014610e3c57610599565b806304fef1db1461059e57806307c58752146105d65780630842b076146105de5780630a48d5a9146105e657806310eb56c214610612578063126082cf1461061a57806312d43a5114610622578063134ca63b14610646578063174d26941461064e578063181e210e146106565780631aa4ace5146106725780631ce9cb8f1461069857806324b0c04d146106be57806324ca984e146106df57806328e67be51461070557806329ff96151461077a5780632c668ec1146107a05780632d4b0576146107cc57806330455ede1461080a578063318bc6891461082957806334c1557d1461061a578063351a964d14610831578063392e53cd146108395780633a05dcc1146108415780633c5a6e35146108675780633de39c11146108b55780633e72a262146108bd57806340eb3802146108c5578063421528731461091657806342b60b031461094a575b600080fd5b6105c4600480360360208110156105b457600080fd5b50356001600160a01b031661176a565b60408051918252519081900360200190f35b6105c46117cd565b6105c46117de565b6105c4600480360360408110156105fc57600080fd5b506001600160a01b0381351690602001356117e4565b6105c4611837565b6105c461183d565b61062a611843565b604080516001600160a01b039092168252519081900360200190f35b6105c4611852565b6105c4611858565b61065e61185e565b604080519115158252519081900360200190f35b6105c46004803603602081101561068857600080fd5b50356001600160a01b031661186e565b6105c4600480360360208110156106ae57600080fd5b50356001600160a01b0316611880565b6106dd600480360360208110156106d457600080fd5b50351515611892565b005b6106dd600480360360208110156106f557600080fd5b50356001600160a01b03166118b6565b6106dd6004803603604081101561071b57600080fd5b81359190810190604081016020820135600160201b81111561073c57600080fd5b82018360208201111561074e57600080fd5b803590602001918460018302840111600160201b8311171561076f57600080fd5b5090925090506118e7565b6105c46004803603602081101561079057600080fd5b50356001600160a01b0316611965565b6105c4600480360360408110156107b657600080fd5b506001600160a01b038135169060200135611979565b6105c4600480360360808110156107e257600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156119be565b6106dd6004803603602081101561082057600080fd5b50351515611a14565b6105c4611a36565b61065e611a3c565b61065e611a4a565b6105c46004803603602081101561085757600080fd5b50356001600160a01b0316611a53565b6106dd600480360360e081101561087d57600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611b13565b6105c4611c55565b61065e611c5b565b6106dd60048036036101208110156108dc57600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611c6a565b6105c46004803603606081101561092c57600080fd5b508035906001600160a01b0360208201358116916040013516611d24565b61065e6004803603602081101561096057600080fd5b50356001600160a01b0316611dc6565b6106dd6004803603604081101561098657600080fd5b506001600160a01b0381351690602001351515611ddb565b6109dc600480360360808110156109b457600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611e0e565b60408051921515835260208301919091528051918290030190f35b6106dd600480360360a0811015610a0d57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013590608001351515611eac565b61062a6122d9565b610a8160048036036080811015610a5957600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156122e8565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b6105c460048036036020811015610ada57600080fd5b50356001600160a01b03166123de565b6105c46123f0565b6105c46123f6565b610b1760048036036020811015610b1057600080fd5b50356123fc565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b6105c460048036036080811015610b6557600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612439565b6105c460048036036020811015610ba357600080fd5b50356001600160a01b03166124df565b61065e60048036036020811015610bc957600080fd5b50356001600160a01b03166124f1565b6105c460048036036020811015610bef57600080fd5b50356001600160a01b0316612506565b6109dc600480360360a0811015610c1557600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060800135612518565b6106dd60048036036020811015610c5557600080fd5b50356001600160a01b0316612610565b61065e60048036036040811015610c7b57600080fd5b506001600160a01b03813581169160200135166126f3565b6105c4612713565b6105c460048036036020811015610cb157600080fd5b50356001600160a01b0316612719565b61062a61272b565b6106dd60048036036020811015610cdf57600080fd5b50356001600160a01b0316612741565b6105c461276f565b6106dd60048036036020811015610d0d57600080fd5b50356001600160a01b0316612776565b6105c460048036036040811015610d3357600080fd5b506001600160a01b03813581169160200135166127aa565b6106dd60048036036020811015610d6157600080fd5b50356001600160a01b0316612a3e565b6106dd600480360360c0811015610d8757600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a00135612a68565b61062a612adf565b6105c4612aee565b6106dd60048036036020811015610ddf57600080fd5b50351515612af4565b6105c460048036036040811015610dfe57600080fd5b506001600160a01b0381358116916020013516612b18565b6105c460048036036020811015610e2c57600080fd5b50356001600160a01b0316612de8565b6105c4600480360360e0811015610e5257600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612e93565b6106dd60048036036040811015610ea657600080fd5b506001600160a01b038135169060200135612f11565b6105c4612f35565b6105c460048036036020811015610eda57600080fd5b50356001600160a01b0316612f3a565b6106dd60048036036060811015610f0057600080fd5b5080359060208101359060400135612f4c565b6105c460048036036020811015610f2957600080fd5b50356001600160a01b0316612f92565b6105c460048036036020811015610f4f57600080fd5b50356001600160a01b0316612fa4565b6106dd60048036036020811015610f7557600080fd5b50356001600160a01b0316612fb6565b61065e612fe0565b6105c460048036036060811015610fa357600080fd5b506001600160a01b038135811691602081013582169160409091013516612fef565b6105c46132d7565b6105c460048036036020811015610fe357600080fd5b50356001600160a01b03166132e7565b6105c46132f9565b6105c46004803603604081101561101157600080fd5b506001600160a01b0381351690602001356132ff565b6105c46004803603606081101561103d57600080fd5b506001600160a01b038135169060208101359060400135613321565b61065e6133ca565b6105c46133d3565b6105c46004803603604081101561107f57600080fd5b506001600160a01b0381351690602001356133d9565b6106dd600480360360408110156110ab57600080fd5b506001600160a01b03813516906020013515156133f6565b6105c4600480360360208110156110d957600080fd5b50356001600160a01b0316613429565b61065e613527565b6105c46004803603602081101561110757600080fd5b50356001600160a01b0316613530565b6105c46004803603602081101561112d57600080fd5b50356001600160a01b0316613542565b6105c4613554565b61065e61355a565b6105c46004803603602081101561116357600080fd5b50356001600160a01b0316613568565b6105c46004803603606081101561118957600080fd5b506001600160a01b038135811691602081013590911690604001351515613609565b6109dc600480360360208110156111c157600080fd5b50356001600160a01b03166136a4565b6105c4600480360360208110156111e757600080fd5b50356001600160a01b031661373c565b6105c461374e565b6105c46004803603602081101561121557600080fd5b50356001600160a01b0316613754565b6105c4600480360360a081101561123b57600080fd5b506001600160a01b0381351690602081013590604081013590606081013590608001351515613766565b6106dd6004803603606081101561127b57600080fd5b506001600160a01b03813581169160208101359091169060400135613810565b6106dd600480360360208110156112b157600080fd5b50356001600160a01b0316613831565b6106dd600480360360208110156112d757600080fd5b503561385b565b6106dd600480360360208110156112f457600080fd5b5035613868565b611341600480360360a081101561131157600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001351515613884565b6040805192835260208301919091528051918290030190f35b6106dd6004803603604081101561137057600080fd5b506001600160a01b03813516906020013561393b565b6105c46004803603602081101561139c57600080fd5b50356001600160a01b0316613994565b6105c46139a6565b6105c4600480360360c08110156113ca57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a001356139ac565b61065e6004803603602081101561141457600080fd5b50356001600160a01b0316613a60565b61065e6004803603602081101561143a57600080fd5b50356001600160a01b0316613a75565b6105c4600480360360e081101561146057600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c00135613a89565b6105c4613b1a565b6106dd600480360360a08110156114b457600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613b20565b6105c4613ee6565b6105c46004803603602081101561150457600080fd5b50356001600160a01b0316613eec565b61062a6004803603602081101561152a57600080fd5b5035613f64565b6106dd6004803603602081101561154757600080fd5b50356001600160a01b0316613f8b565b6106dd6004803603604081101561156d57600080fd5b506001600160a01b03813516906020013561405e565b6105c46004803603602081101561159957600080fd5b50356001600160a01b0316614082565b6106dd600480360360208110156115bf57600080fd5b50351515614094565b6105c4600480360360408110156115de57600080fd5b506001600160a01b03813581169160200135166140ba565b61065e6004803603602081101561160c57600080fd5b50356001600160a01b0316614110565b61062a614125565b61062a614134565b6105c46004803603606081101561164257600080fd5b506001600160a01b038135169060208101359060400135614143565b6106dd6004803603604081101561167457600080fd5b506001600160a01b038135811691602001351661417d565b6105c4614356565b6105c4600480360360a08110156116aa57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101351515906080013561435c565b6116f5600480360360208110156116ee57600080fd5b50356143d2565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561172f578181015183820152602001611717565b50505050905090810190601f16801561175c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b038116600090815260256020526040812054806117925760009150506117c8565b6001600160a01b0383166000908152602660205260409020546117c49082906117be90620f424061446d565b906144c6565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117f357506000611831565b60006117fe84612de8565b6001600160a01b0385166000908152601d602052604090205490915061182c600a82900a6117be868561446d565b925050505b92915050565b600b5481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b60236020526000908152604090205481565b602c6020526000908152604090205481565b61189a614505565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611946576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061195f908383615d1c565b50505050565b60006118318261197484613568565b6117e4565b60008061198584613eec565b905060006119a3826117be8668327cb2734119d3b7a9601e1b61446d565b60055490915061182c9082906001600160a01b031687611d24565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b611a1c614505565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b60015460ff1681565b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611aa457600080fd5b505afa158015611ab8573d6000803e3d6000fd5b505050506040513d6020811015611ace57600080fd5b5051905080611ae15760009150506117c8565b6001600160a01b038316600090815260226020526040902054601554611b0b906117be838561446d565b949350505050565b611b1b614505565b6001600160a01b0387166000908152601c602052604090205460ff16611b9857600754611b49906001614520565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611bbf908290614578565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420670de0b6b3a76400008a029055601f83528184208054821689151517905591805290912080549091168415151790559050611c488187614520565b6015555050505050505050565b60175481565b60015462010000900460ff1681565b611c72614505565b611c826101f48a111560036145ba565b611c926101f489111560046145ba565b611ca26101f488111560056145ba565b611cb26101f487111560066145ba565b611cc26101f486111560076145ba565b611cd26101f485111560086145ba565b611ced6904ee2d6d415b85acef8160201b84111560096145ba565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b60055460009081906001600160a01b03858116911614611d5c576001600160a01b0384166000908152601d6020526040902054611d5f565b60125b6005549091506000906001600160a01b03858116911614611d98576001600160a01b0384166000908152601d6020526040902054611d9b565b60125b9050611dba82600a0a6117be83600a0a8961446d90919063ffffffff16565b925050505b9392505050565b601f6020526000908152604090205460ff1681565b611de3614505565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611e1f878787876119be565b9050611e29615d9a565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c082018190529092611e9d9289929091908990612518565b93509350505094509492505050565b60026000541415611ef2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b6002600055600154611f0e9062010000900460ff16601c6145ba565b611f16614666565b611f1f85614682565b611f2a8484836146e5565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b158015611f9e57600080fd5b505afa158015611fb2573d6000803e3d6000fd5b50505050611fc0848461417d565b6000611fce868686856119be565b6000818152602b6020526040812091925083611ff257611fed86612de8565b611ffb565b611ffb86613eec565b825490915061200c57600282018190555b81541580159061201c5750600085115b156120425761203c868360000154846002015487858a8860060154613a89565b60028301555b600061205b898989888a886000015489600301546147fa565b90506000612068896148b6565b905060006120768a836117e4565b60018601549091506120889082614520565b6001860181905561209d90841115601d6145ba565b60018501546120ac9084614578565b60018601556120bc8a8a89613609565b600386015584546120cd9089614520565b8086554260068701556120e3901515601e6145ba565b6120f585600001548660010154614963565b6121038b8b8b8a6001613884565b505060006121118b8a6133d9565b60048701549091506121239082614520565b60048701556121328b82614987565b87156121775761214b8b6121468b87614520565b614a27565b6121558b83614aa6565b61215f8b84614b25565b6121728b61216d8d876132ff565b614c3d565b6121e3565b6001600160a01b038a166000908152602d60205260409020546121b4576001600160a01b038a166000908152602e602052604090208590556121d9565b6121bf8a868b613321565b6001600160a01b038b166000908152602e60205260409020555b6121e38a8a614d10565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820187905251600080516020615e0d833981519152918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b60008060008060008060008060006123028d8d8d8d6119be565b905061230c615d9a565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a001511361238c578160a00151600003612392565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b600080612448868686866119be565b9050612452615d9a565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c08301526124bc90151560256145ba565b602081015181516124d491906117be9061271061446d565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6000806125296000861160266145ba565b60008461253e5761253988613eec565b612547565b61254788612de8565b905060008187116125615761255c8288614578565b61256b565b61256b8783614578565b9050600061257d886117be8b8561446d565b9050600087156125905750878311612595565b508288115b60006125ac6010548961452090919063ffffffff16565b42116125d0576001600160a01b038c166000908152601e60205260409020546125d3565b60005b90508180156125f657506125e78b8261446d565b6125f38461271061446d565b11155b1561260057600092505b509a909950975050505050505050565b60026000541415612656576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546126819060ff16600e6145ba565b600061268c826148b6565b905061269c60008211600f6145ba565b6126a68282614b25565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b61277e614505565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b6000600260005414156127f2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b60026000556127ff614dc8565b6001600160a01b0383166000908152601c60205260409020546128269060ff1660136145ba565b6016805461ff00191661010017905560055460009061284d906001600160a01b03166148b6565b905061285d6000821160146145ba565b612867848561417d565b60006128738583611979565b90506128836000821160156145ba565b61288d8583614df7565b6128978582614c3d565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b1580156128ea57600080fd5b505af11580156128fe573d6000803e3d6000fd5b505060055461291892506001600160a01b03169050614eb5565b6001546040805163eb0835bf60e01b81526001600160a01b038881166004830152602482018690529151600093630100000090049092169163eb0835bf91604480820192602092909190829003018186803b15801561297657600080fd5b505afa15801561298a573d6000803e3d6000fd5b505050506040513d60208110156129a057600080fd5b5051905060006129b1878484614f50565b90506129c16000821160166145ba565b6129cc878288615018565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517fd732b7828fa6cee72c285eac756fc66a7477e3dc22e22e7c432f1c265d40b4839181900360a00190a16016805461ff001916905560016000559695505050505050565b612a46614505565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b612a70614505565b60018054612a849160ff90911615906145ba565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b6004546001600160a01b031681565b600a5481565b612afc614505565b60018054911515620100000262ff000019909216919091179055565b600060026000541415612b60576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b6002600055612b6d614dc8565b6001600160a01b0383166000908152601c6020526040902054612b949060ff1660106145ba565b6016805461ff0019166101001790556000612bae846148b6565b9050612bbe6000821160116145ba565b612bc8848561417d565b6000612bd385612de8565b90506000612bf168327cb2734119d3b7a9601e1b6117be858561446d565b600554909150612c0d90829088906001600160a01b0316611d24565b9050612c1d6000821160126145ba565b6001546040805163256f6ee360e11b81526001600160a01b0389811660048301526024820185905291516000936301000000900490921691634adeddc691604480820192602092909190829003018186803b158015612c7b57600080fd5b505afa158015612c8f573d6000803e3d6000fd5b505050506040513d6020811015612ca557600080fd5b505190506000612cb6888684614f50565b90506000612cd468327cb2734119d3b7a9601e1b6117be848861446d565b600554909150612cf09082908b906001600160a01b0316611d24565b9050612cfc89826150bf565b612d068983614b25565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b158015612d5b57600080fd5b505af1158015612d6f573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517fab4c77c74cd32c85f35416cf03e7ce9e2d4387f7b7f2c1f4bf53daaecf8ea72d93509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b158015612e6157600080fd5b505afa158015612e75573d6000803e3d6000fd5b505050506040513d6020811015612e8b57600080fd5b505192915050565b600060026000541415612edb576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b6002600055612ee8614666565b612ef188614682565b612f008888888888888861517d565b600160005598975050505050505050565b612f19614505565b6001600160a01b03909116600090815260276020526040902055565b601281565b601e6020526000908152604090205481565b612f54614505565b612f64610e10841015600a6145ba565b612f74612710831115600b6145ba565b612f84612710821115600c6145ba565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612fbe614505565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415613037576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b600260005560015461305290610100900460ff1660176145ba565b6001600160a01b0384166000908152601c60205260409020546130799060ff1660186145ba565b6001600160a01b0383166000908152601c60205260409020546130a09060ff1660196145ba565b6130c0836001600160a01b0316856001600160a01b03161415601a6145ba565b6016805461ff0019166101001790556130d9848061417d565b6130e3838461417d565b60006130ee856148b6565b90506130fe60008211601b6145ba565b600061310986612de8565b9050600061311686613eec565b90506000613128826117be868661446d565b9050613135818989611d24565b9050600061315368327cb2734119d3b7a9601e1b6117be878761446d565b60055490915061316f9082908b906001600160a01b0316611d24565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b1580156131d957600080fd5b505afa1580156131ed573d6000803e3d6000fd5b505050506040513d602081101561320357600080fd5b5051905060006132148a8584614f50565b90506132208b846150bf565b61322a8a84614df7565b6132348b88614b25565b61323e8a85614c3d565b6132478a61560a565b6132528a828b615018565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b60008161330e57506000611831565b611dbf838361331c86613eec565b614143565b6001600160a01b0383166000908152602d6020908152604080832054602e9092528220548285821161335c576133578683614578565b613366565b6133668287614578565b90506000613378836117be868561446d565b905086831160006133898689614520565b90506000826133a15761339c8285614520565b6133ab565b6133ab8285614578565b90506133bb816117be8c8561446d565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b6000816133e857506000611831565b611dbf838361331c86612de8565b6133fe614505565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a60205260408120549091429161345291614520565b1115613460575060006117c8565b6012546001600160a01b0383166000908152602a6020526040812054909161348d916117be904290614578565b6001600160a01b038416600090815260256020526040902054909150806134b9576000925050506117c8565b6001600160a01b0384166000908152601f602052604081205460ff166134e1576013546134e5565b6014545b6001600160a01b03861660009081526026602052604090205490915061351e9083906117be90869061351890869061446d565b9061446d565b95945050505050565b60165460ff1681565b60226020526000908152604090205481565b60246020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff16156135a857506001600160a01b0381166000908152602560205260409020546117c8565b6001600160a01b0382166000908152602860205260408120546135cc9084906132ff565b6001600160a01b0384166000908152602660209081526040808320546025909252909120549192506117c491613603908490614520565b90614578565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b15801561367057600080fd5b505afa158015613684573d6000803e3d6000fd5b505050506040513d602081101561369a57600080fd5b5051949350505050565b6001600160a01b0381166000908152602d60205260408120548190806136d1576000809250925050613737565b60006136dc85613eec565b6001600160a01b0386166000908152602e602052604081205491925082821161370e576137098383614578565b613718565b6137188284614578565b9050600061372a836117be878561446d565b9390921195509193505050505b915091565b60266020526000908152604090205481565b60135481565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b1580156137da57600080fd5b505afa1580156137ee573d6000803e3d6000fd5b505050506040513d602081101561380457600080fd5b50519695505050505050565b613818614505565b61382c6001600160a01b038316848361567f565b505050565b613839614505565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b613863614505565b601755565b613870614505565b61387f612710821160026145ba565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b1580156138fa57600080fd5b505afa15801561390e573d6000803e3d6000fd5b505050506040513d604081101561392457600080fd5b508051602090910151909890975095505050505050565b613943614505565b6001600160a01b0382166000908152602360205260409020548082111561397d57613977836139728484614578565b6150bf565b50613990565b61382c8361398b8385614578565b614df7565b5050565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b158015613a2957600080fd5b505afa158015613a3d573d6000803e3d6000fd5b505050506040513d6020811015613a5357600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b6000806000613a9b8a8a8a8a88612518565b90925090506000613aac8a87614520565b905060008815613adb5783613aca57613ac58284614578565b613ad4565b613ad48284614520565b9050613afc565b83613aef57613aea8284614520565b613af9565b613af98284614578565b90505b613b0a816117be8a8561446d565b9c9b505050505050505050505050565b60155481565b60026000541415613b66576040805162461bcd60e51b815260206004820152601f6024820152600080516020615ded833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613b9b5733600090815260196020526040902054613b9b9060ff1660226145ba565b6016805460ff19169055613baf848461417d565b6000613bbd868686866119be565b9050613bc7615d9a565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613c3390151560236145ba565b600080613c44898989896000613884565b91509150613c57826000141560246145ba565b8160021415613c8c57613c74898989600087600001518b8f61517d565b50506016805460ff1916600117905550613eda915050565b6000613c9889836132ff565b6001600160a01b038a166000908152602c6020526040902054909150613cbe9082614520565b6001600160a01b038a166000818152602c60209081526040918290209390935580519182529181018490528082018390529051600080516020615eb88339815191529181900360600190a1613d178985608001516156d1565b8615613d465760208401518451613d38918b91613d3391614578565b614aa6565b613d468961216d8b856132ff565b600087613d5b57613d5689613eec565b613d64565b613d6489612de8565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613e235750846020015183105b15613e51576020850151600090613e3a9085614578565b9050613e4f8b613e4a8d846132ff565b614b25565b505b87613e6457613e64898660000151615786565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613eb0908b9061216d9082906132ff565b613ec68a613ec08c6009546132ff565b89615018565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b158015612e6157600080fd5b601b8181548110613f7157fe5b6000918252602090912001546001600160a01b0316905081565b613f93614505565b6001600160a01b0381166000908152601c6020526040902054613fba9060ff16600d6145ba565b6001600160a01b038116600090815260226020526040902054601554613fdf91614578565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f835281842080548216905591805290912080549091169055600754614058906001614578565b60075550565b614066614505565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b61409c614505565b6016805491151563010000000263ff00000019909216919091179055565b60006140c4614505565b6001600160a01b0383166000908152602c6020526040902054806140ec576000915050611831565b6001600160a01b0384166000908152602c6020526040812055611dbf848285615018565b601a6020526000908152604090205460ff1681565b6005546001600160a01b031681565b6003546001600160a01b031681565b60008261415257506000611dbf565b6001600160a01b0384166000908152601d602052604090205461351e836117be86600a85900a61446d565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b1580156141dd57600080fd5b505af11580156141f1573d6000803e3d6000fd5b505050506040513d602081101561420757600080fd5b50519050806142165750613990565b6001600160a01b0383166000908152602a6020526040902054614263576012546142449061351842826144c6565b6001600160a01b0384166000908152602a602052604090205550613990565b6012546001600160a01b0384166000908152602a6020526040902054429161428b9190614520565b11156142975750613990565b60006142a284613429565b6001600160a01b0385166000908152602960205260409020549091506142c89082614520565b6001600160a01b0385166000908152602960205260409020556012546142f29061351842826144c6565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b1580156137da57600080fd5b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144655780601f1061443a57610100808354040283529160200191614465565b820191906000526020600020905b81548152906001019060200180831161444857829003601f168201915b505050505081565b60008261447c57506000611831565b8282028284828161448957fe5b0414611dbf5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e4d6021913960400191505060405180910390fd5b6000611dbf83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b8152506157ee565b60065461451e906001600160a01b0316331460356145ba565b565b600082820183811015611dbf576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b6000611dbf83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f770000815250615890565b60008181526030602052604090208261382c5760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156146575780601f1061462c57610100808354040283529160200191614657565b820191906000526020600020905b81548152906001019060200180831161463a57829003601f168201915b50509250505060405180910390fd5b6017546146725761451e565b61451e6017543a111560376145ba565b336001600160a01b0382161415614698576146e2565b6003546001600160a01b03163314156146b0576146e2565b6001600160a01b03811660009081526018602090815260408083203384529091529020546146e29060ff1660296145ba565b50565b801561475e5761470a826001600160a01b0316846001600160a01b031614602a6145ba565b6001600160a01b0383166000908152601c60205260409020546147319060ff16602b6145ba565b6001600160a01b0383166000908152601f60205260409020546147599060ff1615602c6145ba565b61382c565b6001600160a01b0383166000908152601c60205260409020546147859060ff16602d6145ba565b6001600160a01b0383166000908152601f60205260409020546147ac9060ff16602e6145ba565b6001600160a01b0382166000908152601f60205260409020546147d49060ff1615602f6145ba565b6001600160a01b038216600090815260208052604090205461382c9060ff1660306145ba565b60008061480a898989898961435c565b9050600061481c8a8a8a8a89896139ac565b90506148288282614520565b915060006148368a846132ff565b6001600160a01b038b166000908152602c602052604090205490915061485c9082614520565b6001600160a01b038b166000818152602c60209081526040918290209390935580519182529181018590528082018390529051600080516020615eb88339815191529181900360600190a150909998505050505050505050565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b15801561491057600080fd5b505afa158015614924573d6000803e3d6000fd5b505050506040513d602081101561493a57600080fd5b50516001600160a01b03851660009081526021602052604090208190559050611b0b8183614578565b8161497957614974811560276145ba565b613990565b6139908183101560286145ba565b6001600160a01b0382166000908152602660205260409020546149aa9082614520565b6001600160a01b0383166000908152602660208181526040808420859055602582529092205491526149df91111560346145ba565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614a4a9082614520565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614ac99082614578565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614b489082614520565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614b9e57600080fd5b505afa158015614bb2573d6000803e3d6000fd5b505050506040513d6020811015614bc857600080fd5b50516001600160a01b038416600090815260256020526040902054909150614bf49082101560316145ba565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b038516600090815260259091529190912054614c95918390615890565b6001600160a01b03831660009081526025602090815260408083208490556026909152902054614cc891101560326145ba565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054614d339082614520565b6001600160a01b0383166000908152602d6020908152604080832093909355602f90522054801561382c576001600160a01b0383166000908152602d602052604090205481101561382c576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b60165462010000900460ff161561451e57336000908152601a602052604090205461451e9060ff1660366145ba565b6001600160a01b038216600090815260236020526040902054818111614e60576001600160a01b038316600081815260236020908152604080832092909255815192835282018390528051600080516020615e6e8339815191529281900390910190a150613990565b614e6a8183614578565b6001600160a01b0384166000818152602360209081526040918290209390935580519182529181018490528151600080516020615e6e833981519152929181900390910190a1505050565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b158015614f0457600080fd5b505afa158015614f18573d6000803e3d6000fd5b505050506040513d6020811015614f2e57600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b600080614f6d6127106117be614f668287614578565b879061446d565b90506000614f7b8583614578565b6001600160a01b0387166000908152602c6020526040902054909150614fa19082614520565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b86614fe681846117e4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b61502c6001600160a01b038416828461567f565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b15801561507257600080fd5b505afa158015615086573d6000803e3d6000fd5b505050506040513d602081101561509c57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b6001600160a01b0382166000908152602360205260409020546150e29082614520565b6001600160a01b0383166000908152602360209081526040808320939093556024905220548015615134576001600160a01b0383166000908152602360205260409020546151349082101560336145ba565b604080516001600160a01b03851681526020810184905281517f64243679a443432e2293343b77d411ff6144370404618f00ca0d2025d9ca9882929181900390910190a1505050565b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b15801561520157600080fd5b505afa158015615215573d6000803e3d6000fd5b50505050615223878761417d565b6000615231898989876119be565b6000818152602b60205260409020805491925090615252901515601f6145ba565b615264868260000154101560206145ba565b615276878260010154101560216145ba565b600181015481546004830154600091615293916117be908b61446d565b60048401549091506152a59082614578565b60048401556152b48b826156d1565b506000806152c68d8d8d8d8d8d6158ea565b855491935091508914615446576152de8c8c8a613609565b600385015583546152ef908a614578565b80855560018501546153019190614963565b61530f8d8d8d8b6001613884565b5050871561533c576153328c61214686600101548661457890919063ffffffff16565b61533c8c8a614aa6565b6000886153515761534c8c613eec565b61535a565b61535a8c612de8565b9050600080516020615e2d833981519152868f8f8f8f8f8f8861537d8c8c614578565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820183905251600080516020615e0d833981519152918190036101000190a1506155a8565b8715615460576154568c84614a27565b6154608c8a614aa6565b600088615475576154708c613eec565b61547e565b61547e8c612de8565b9050600080516020615e2d833981519152868f8f8f8f8f8f886154a18c8c614578565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b876155b7576155b78b8a615786565b81156155f75787156155d1576155d18c61216d8e856132ff565b60006155dd8d836132ff565b90506155ea8d828a615018565b95506124d4945050505050565b5060009c9b505050505050505050505050565b6001600160a01b03811660009081526027602090815260408083205460259092529091205410156146e2576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b17905261382c908490615ae4565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b03851660009081526026909152919091205461572a918390615890565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054808211156157c657506001600160a01b0382166000908152602d6020526040812055613990565b6157d08183614578565b6001600160a01b0384166000908152602d6020526040902055505050565b6000818361587a5760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561583f578181015183820152602001615827565b50505050905090810190601f16801561586c5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161588657fe5b0495945050505050565b600081848411156158e25760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561583f578181015183820152602001615827565b505050900390565b60008060006158fb898989876119be565b6000818152602b60205260408120805460038201549394509092615928918d918d918d918b918d916147fa565b90506000806000806159498d876000015488600201548d8a60060154612518565b87549195508593509150615961906117be8d8461446d565b9250505060008280156159745750600082115b156159a457506005840180548201905580886159a45760006159968e846132ff565b90506159a28e82614c3d565b505b821580156159b25750600082115b156159f65760018501546159c69083614578565b6001860155886159ea5760006159dc8e846132ff565b90506159e88e82614b25565b505b60058501805483900390555b8a15615a1e57615a06818c614520565b6001860154909150615a18908c614578565b60018601555b84548a1415615a41576001850154615a37908290614520565b6000600187015590505b8084811115615a5b57615a548286614578565b9050615a8f565b6001860154615a6a9086614578565b60018701558915615a8f576000615a818f876132ff565b9050615a8d8f82614c3d565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615b39826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615b959092919063ffffffff16565b80519091501561382c57808060200190516020811015615b5857600080fd5b505161382c5760405162461bcd60e51b815260040180806020018281038252602a815260200180615e8e602a913960400191505060405180910390fd5b6060611b0b848460008585615ba985615cb0565b615bfa576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615c395780518252601f199092019160209182019101615c1a565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615c9b576040519150601f19603f3d011682016040523d82523d6000602084013e615ca0565b606091505b50915091506124d4828286615cb6565b3b151590565b60608315615cc5575081611dbf565b825115615cd55782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561583f578181015183820152602001615827565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615d5d5782800160ff19823516178555615d8a565b82800160010185558215615d8a579182015b82811115615d8a578235825591602001919060010190615d6f565b50615d96929150615dd7565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615d965760008155600101615dd856fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c0020853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b77393d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f77e1e812596aac93a06ecc4ca627014d18e30f5c33b825160cc9d5c0ba61e452275361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011aa264697066735822122090c5359baef33c62d122bd4f925cebe6b7c9c9103277455b0a68532b74ccbacd64736f6c634300060c0033
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.