More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 28 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Set Gov | 6478723 | 298 days ago | IN | 0 GLMR | 0.01253 | ||||
Set Max Leverage | 6478722 | 298 days ago | IN | 0 GLMR | 0.012604 | ||||
Set Gov | 5757848 | 400 days ago | IN | 0 GLMR | 0.01367493 | ||||
Set Token Config | 5757808 | 400 days ago | IN | 0 GLMR | 0.083302 | ||||
Set Token Config | 5757806 | 400 days ago | IN | 0 GLMR | 0.083534 | ||||
Set Token Config | 5757803 | 400 days ago | IN | 0 GLMR | 0.083534 | ||||
Set Token Config | 5757801 | 400 days ago | IN | 0 GLMR | 0.083534 | ||||
Set Token Config | 5757799 | 400 days ago | IN | 0 GLMR | 0.083534 | ||||
Set Token Config | 5757797 | 400 days ago | IN | 0 GLMR | 0.083534 | ||||
Set Gov | 4018946 | 645 days ago | IN | 0 GLMR | 0.00368592 | ||||
Set Max Leverage | 4018945 | 645 days ago | IN | 0 GLMR | 0.00392518 | ||||
Set Gov | 4013744 | 645 days ago | IN | 0 GLMR | 0.00447382 | ||||
Set Max Leverage | 4013743 | 645 days ago | IN | 0 GLMR | 0.0047648 | ||||
Set Gov | 3813082 | 674 days ago | IN | 0 GLMR | 0.00368592 | ||||
Set Token Config | 3715323 | 688 days ago | IN | 0 GLMR | 0.03554385 | ||||
Set Token Config | 3715321 | 688 days ago | IN | 0 GLMR | 0.03550875 | ||||
Set Token Config | 3715319 | 688 days ago | IN | 0 GLMR | 0.03554385 | ||||
Set Token Config | 3715317 | 688 days ago | IN | 0 GLMR | 0.03554385 | ||||
Set Token Config | 3715315 | 688 days ago | IN | 0 GLMR | 0.04323705 | ||||
Set Liquidator | 3715178 | 688 days ago | IN | 0 GLMR | 0.00704085 | ||||
Set Vault Utils | 3715097 | 688 days ago | IN | 0 GLMR | 0.0045132 | ||||
Set Error Contro... | 3715028 | 688 days ago | IN | 0 GLMR | 0.00608428 | ||||
Set Fees | 3715010 | 688 days ago | IN | 0 GLMR | 0.01203791 | ||||
Set Manager | 3714998 | 688 days ago | IN | 0 GLMR | 0.00610475 | ||||
Set In Manager M... | 3714996 | 688 days ago | IN | 0 GLMR | 0.00367937 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Contract Name:
Vault
Compiler Version
v0.6.12+commit.27d51765
Optimization Enabled:
Yes with 1 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; import "../libraries/math/SafeMath.sol"; import "../libraries/token/IERC20.sol"; import "../libraries/token/SafeERC20.sol"; import "../libraries/utils/ReentrancyGuard.sol"; import "../tokens/interfaces/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 = 5 * 10000; // 5x uint256 public override liquidationFeeUsd; uint256 public override taxBasisPoints = 50; // 0.5% uint256 public override stableTaxBasisPoints = 5; // 0.05% uint256 public override mintBurnFeeBasisPoints = 25; // 0.25% uint256 public override swapFeeBasisPoints = 25; // 0.25% uint256 public override stableSwapFeeBasisPoints = 1; // 0.01% 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 view override 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; 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 ) public view override returns (uint256, uint256) { return vaultUtils.validateLiquidation( _account, _collateralToken, _indexToken, _isLong, _raise ); } function getMaxPrice( address _token ) public view override returns (uint256) { return IVaultPriceFeed(priceFeed).getPrice( _token, true, includeAmmPrice, useSwapPricing ); } function getMinPrice( address _token ) public view override returns (uint256) { return IVaultPriceFeed(priceFeed).getPrice( _token, false, includeAmmPrice, useSwapPricing ); } function getRedemptionAmount( address _token, uint256 _usdgAmount ) public view override 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 view override 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 view override 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 view override 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 view override 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 view override returns (uint256) { return vaultUtils.getFeeBasisPoints( _token, _usdgDelta, _feeBasisPoints, _taxBasisPoints, _increment ); } function getTargetUsdgAmount( address _token ) public view override 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; /** * @dev Wrappers over Solidity's arithmetic operations with added overflow * checks. * * Arithmetic operations in Solidity wrap on overflow. This can easily result * in bugs, because programmers usually assume that an overflow raises an * error, which is the standard behavior in high level programming languages. * `SafeMath` restores this intuition by reverting the transaction when an * operation overflows. * * Using this library instead of the unchecked operations eliminates an entire * class of bugs, so it's recommended to use it always. */ library SafeMath { /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { uint256 c = a + b; require(c >= a, "SafeMath: addition overflow"); return c; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return sub(a, b, "SafeMath: subtraction overflow"); } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b <= a, errorMessage); uint256 c = a - b; return c; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) { return 0; } uint256 c = a * b; require(c / a == b, "SafeMath: multiplication overflow"); return c; } /** * @dev Returns the integer division of two unsigned integers. Reverts on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return div(a, b, "SafeMath: division by zero"); } /** * @dev Returns the integer division of two unsigned integers. Reverts with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b > 0, errorMessage); uint256 c = a / b; // assert(a == b * c + a % b); // There is no case in which this doesn't hold return c; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return mod(a, b, "SafeMath: modulo by zero"); } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * Reverts with custom message when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { require(b != 0, errorMessage); return a % b; } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `recipient`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address recipient, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `sender` to `recipient` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address sender, address recipient, uint256 amount) external returns (bool); /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; import "./IERC20.sol"; import "../math/SafeMath.sol"; import "../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using SafeMath for uint256; using Address for address; function safeTransfer(IERC20 token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove(IERC20 token, address spender, uint256 value) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' // solhint-disable-next-line max-line-length require((value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 newAllowance = token.allowance(address(this), spender).add(value); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero"); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); if (returndata.length > 0) { // Return data is optional // solhint-disable-next-line max-line-length require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } }
// SPDX-License-Identifier: MIT pragma solidity 0.6.12; 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; }
// 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 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; 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.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); } } } }
{ "optimizer": { "enabled": true, "runs": 1 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"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
60806040526001805462ff00001961ff001990911661010017166201000017815561c3506008556032600a9081556005600b556019600c819055600d55600e829055600f556011805460ff199081169091556170806012556016805463ffffff0019921690921716905534801561007557600080fd5b506001600055600680546001600160a01b03191633179055615edd806200009d6000396000f3fe608060405234801561001057600080fd5b50600436106105695760003560e01c806304fef1db1461056e57806307c58752146105a65780630842b076146105ae5780630a48d5a9146105b657806310eb56c2146105e2578063126082cf146105ea57806312d43a51146105f2578063134ca63b14610616578063174d26941461061e578063181e210e146106265780631aa4ace5146106425780631ce9cb8f1461066857806324b0c04d1461068e57806324ca984e146106af57806328e67be5146106d557806329ff96151461074a5780632c668ec1146107705780632d4b05761461079c57806330455ede146107da578063318bc689146107f957806334c1557d146105ea578063351a964d14610801578063392e53cd146108095780633a05dcc1146108115780633c5a6e35146108375780633de39c11146108855780633e72a2621461088d57806340eb38021461089557806342152873146108e657806342b60b031461091a5780634453a3741461094057806345a6f3701461096e57806348d91abf146109c757806348f35cbb14610a0b5780634a3f088d14610a135780634a993ee914610a945780634befe2ca14610aba5780634d47b30414610ac2578063514ea4bf14610aca57806351723e8214610b1f578063523fba7f14610b5d578063529a356f14610b8357806352f55eed14610ba95780635c07eaab14610bcf5780635f7bc11914610c0f5780636092219914610c3557806362287a3214610c635780636274980314610c6b5780636abbe0c814610c915780636ae0b15414610c995780636be6026b14610cbf57806371089f4d14610cc7578063711e619014610ced578063724e78da14610d1b578063728cdbca14610d41578063741bef1a14610d895780637a210a2b14610d915780637c2eb9f714610d99578063817bb85714610db857806381a612d614610de657806382a0849014610e0c5780638585f4d214610e60578063870d917c14610e8c57806388b1fbdf14610e945780638a27d46814610eba5780638a39735a146105ea5780638a78daa814610ee35780638ee573ac14610f095780638f7b840414610f2f5780639060b1ca14610f555780639331621214610f5d57806395082d2514610f955780639698d25a14610f9d5780639849e41214610fc35780639899cd0214610fcb5780639d7432ca14610ff75780639f392eb314611029578063a22f239214611031578063a42ab3d214611039578063a5e90eee14611065578063a93acac214611093578063ab08c1c6146110b9578063ab2f3ad4146110c1578063ad1e4f8d146110e7578063ae3302c21461110d578063b06423f314611115578063b136ca491461111d578063b1cc53ab14611143578063b364accb1461117b578063c3c7b9e9146111a1578063c4f718bf146111c7578063c65bc7b1146111cf578063c7e074c3146111f5578063cea0c32814611235578063cfad57a21461126b578063d2fa635e14611291578063d3127e63146112ae578063d54d5a9f146112cb578063d66b000d1461132a578063d8f897c314611356578063d9ac42251461137c578063da76524c14611384578063daf9c210146113ce578063db3555fb146113f4578063db97495f1461141a578063dc8f5fac14611466578063de2ea9481461146e578063df73a267146114b6578063e124e6d2146114be578063e468baf0146114e4578063e67f59a714611501578063efa10a6e14611527578063f07456ce14611553578063f07bbf7714611579578063f255527814611598578063f3ae2415146115c6578063f5b91b7b146115ec578063f887ea40146115f4578063fa12dbc0146115fc578063fbfded6d1461162e578063fce28c101461165c578063fdaf6ac314611664578063fed1a606146116a8575b600080fd5b6105946004803603602081101561058457600080fd5b50356001600160a01b031661173a565b60408051918252519081900360200190f35b61059461179d565b6105946117ae565b610594600480360360408110156105cc57600080fd5b506001600160a01b0381351690602001356117b4565b610594611807565b61059461180d565b6105fa611813565b604080516001600160a01b039092168252519081900360200190f35b610594611822565b610594611828565b61062e61182e565b604080519115158252519081900360200190f35b6105946004803603602081101561065857600080fd5b50356001600160a01b031661183e565b6105946004803603602081101561067e57600080fd5b50356001600160a01b0316611850565b6106ad600480360360208110156106a457600080fd5b50351515611862565b005b6106ad600480360360208110156106c557600080fd5b50356001600160a01b0316611886565b6106ad600480360360408110156106eb57600080fd5b81359190810190604081016020820135600160201b81111561070c57600080fd5b82018360208201111561071e57600080fd5b803590602001918460018302840111600160201b8311171561073f57600080fd5b5090925090506118b7565b6105946004803603602081101561076057600080fd5b50356001600160a01b0316611935565b6105946004803603604081101561078657600080fd5b506001600160a01b038135169060200135611949565b610594600480360360808110156107b257600080fd5b506001600160a01b03813581169160208101358216916040820135169060600135151561198e565b6106ad600480360360208110156107f057600080fd5b503515156119e4565b610594611a06565b61062e611a0c565b61062e611a1a565b6105946004803603602081101561082757600080fd5b50356001600160a01b0316611a23565b6106ad600480360360e081101561084d57600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611ae3565b610594611c25565b61062e611c2b565b6106ad60048036036101208110156108ac57600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611c3a565b610594600480360360608110156108fc57600080fd5b508035906001600160a01b0360208201358116916040013516611cf4565b61062e6004803603602081101561093057600080fd5b50356001600160a01b0316611d96565b6106ad6004803603604081101561095657600080fd5b506001600160a01b0381351690602001351515611dab565b6109ac6004803603608081101561098457600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611dde565b60408051921515835260208301919091528051918290030190f35b6106ad600480360360a08110156109dd57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013590608001351515611e7c565b6105fa6122a9565b610a5160048036036080811015610a2957600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156122b8565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b61059460048036036020811015610aaa57600080fd5b50356001600160a01b03166123ae565b6105946123c0565b6105946123c6565b610ae760048036036020811015610ae057600080fd5b50356123cc565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b61059460048036036080811015610b3557600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612409565b61059460048036036020811015610b7357600080fd5b50356001600160a01b03166124af565b61062e60048036036020811015610b9957600080fd5b50356001600160a01b03166124c1565b61059460048036036020811015610bbf57600080fd5b50356001600160a01b03166124d6565b6109ac600480360360a0811015610be557600080fd5b506001600160a01b03813516906020810135906040810135906060810135151590608001356124e8565b6106ad60048036036020811015610c2557600080fd5b50356001600160a01b03166125e0565b61062e60048036036040811015610c4b57600080fd5b506001600160a01b03813581169160200135166126c3565b6105946126e3565b61059460048036036020811015610c8157600080fd5b50356001600160a01b03166126e9565b6105fa6126fb565b6106ad60048036036020811015610caf57600080fd5b50356001600160a01b0316612711565b61059461273f565b6106ad60048036036020811015610cdd57600080fd5b50356001600160a01b0316612746565b61059460048036036040811015610d0357600080fd5b506001600160a01b038135811691602001351661277a565b6106ad60048036036020811015610d3157600080fd5b50356001600160a01b0316612a0e565b6106ad600480360360c0811015610d5757600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a00135612a38565b6105fa612aaf565b610594612abe565b6106ad60048036036020811015610daf57600080fd5b50351515612ac4565b61059460048036036040811015610dce57600080fd5b506001600160a01b0381358116916020013516612ae8565b61059460048036036020811015610dfc57600080fd5b50356001600160a01b0316612db8565b610594600480360360e0811015610e2257600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612e63565b6106ad60048036036040811015610e7657600080fd5b506001600160a01b038135169060200135612ee1565b610594612f05565b61059460048036036020811015610eaa57600080fd5b50356001600160a01b0316612f0a565b6106ad60048036036060811015610ed057600080fd5b5080359060208101359060400135612f1c565b61059460048036036020811015610ef957600080fd5b50356001600160a01b0316612f62565b61059460048036036020811015610f1f57600080fd5b50356001600160a01b0316612f74565b6106ad60048036036020811015610f4557600080fd5b50356001600160a01b0316612f86565b61062e612fb0565b61059460048036036060811015610f7357600080fd5b506001600160a01b038135811691602081013582169160409091013516612fbf565b6105946132a7565b61059460048036036020811015610fb357600080fd5b50356001600160a01b03166132b7565b6105946132c9565b61059460048036036040811015610fe157600080fd5b506001600160a01b0381351690602001356132cf565b6105946004803603606081101561100d57600080fd5b506001600160a01b0381351690602081013590604001356132f1565b61062e61339a565b6105946133a3565b6105946004803603604081101561104f57600080fd5b506001600160a01b0381351690602001356133a9565b6106ad6004803603604081101561107b57600080fd5b506001600160a01b03813516906020013515156133c6565b610594600480360360208110156110a957600080fd5b50356001600160a01b03166133f9565b61062e6134f7565b610594600480360360208110156110d757600080fd5b50356001600160a01b0316613500565b610594600480360360208110156110fd57600080fd5b50356001600160a01b0316613512565b610594613524565b61062e61352a565b6105946004803603602081101561113357600080fd5b50356001600160a01b0316613538565b6105946004803603606081101561115957600080fd5b506001600160a01b0381358116916020810135909116906040013515156135d9565b6109ac6004803603602081101561119157600080fd5b50356001600160a01b0316613674565b610594600480360360208110156111b757600080fd5b50356001600160a01b031661370c565b61059461371e565b610594600480360360208110156111e557600080fd5b50356001600160a01b0316613724565b610594600480360360a081101561120b57600080fd5b506001600160a01b0381351690602081013590604081013590606081013590608001351515613736565b6106ad6004803603606081101561124b57600080fd5b506001600160a01b038135811691602081013590911690604001356137e0565b6106ad6004803603602081101561128157600080fd5b50356001600160a01b0316613801565b6106ad600480360360208110156112a757600080fd5b503561382b565b6106ad600480360360208110156112c457600080fd5b5035613838565b611311600480360360a08110156112e157600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001351515613854565b6040805192835260208301919091528051918290030190f35b6106ad6004803603604081101561134057600080fd5b506001600160a01b03813516906020013561390b565b6105946004803603602081101561136c57600080fd5b50356001600160a01b0316613964565b610594613976565b610594600480360360c081101561139a57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a0013561397c565b61062e600480360360208110156113e457600080fd5b50356001600160a01b0316613a30565b61062e6004803603602081101561140a57600080fd5b50356001600160a01b0316613a45565b610594600480360360e081101561143057600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c00135613a59565b610594613aea565b6106ad600480360360a081101561148457600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613af0565b610594613eb6565b610594600480360360208110156114d457600080fd5b50356001600160a01b0316613ebc565b6105fa600480360360208110156114fa57600080fd5b5035613f34565b6106ad6004803603602081101561151757600080fd5b50356001600160a01b0316613f5b565b6106ad6004803603604081101561153d57600080fd5b506001600160a01b03813516906020013561402e565b6105946004803603602081101561156957600080fd5b50356001600160a01b0316614052565b6106ad6004803603602081101561158f57600080fd5b50351515614064565b610594600480360360408110156115ae57600080fd5b506001600160a01b038135811691602001351661408a565b61062e600480360360208110156115dc57600080fd5b50356001600160a01b03166140e0565b6105fa6140f5565b6105fa614104565b6105946004803603606081101561161257600080fd5b506001600160a01b038135169060208101359060400135614113565b6106ad6004803603604081101561164457600080fd5b506001600160a01b038135811691602001351661414d565b610594614326565b610594600480360360a081101561167a57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101351515906080013561432c565b6116c5600480360360208110156116be57600080fd5b50356143a2565b6040805160208082528351818301528351919283929083019185019080838360005b838110156116ff5781810151838201526020016116e7565b50505050905090810190601f16801561172c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b03811660009081526025602052604081205480611762576000915050611798565b6001600160a01b03831660009081526026602052604090205461179490829061178e90620f424061443d565b90614496565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117c357506000611801565b60006117ce84612db8565b6001600160a01b0385166000908152601d60205260409020549091506117fc600a82900a61178e868561443d565b925050505b92915050565b600b5481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b60236020526000908152604090205481565b602c6020526000908152604090205481565b61186a6144d5565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611916576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061192f908383615cec565b50505050565b60006118018261194484613538565b6117b4565b60008061195584613ebc565b905060006119738261178e8668327cb2734119d3b7a9601e1b61443d565b6005549091506117fc9082906001600160a01b031687611cf4565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b6119ec6144d5565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b60015460ff1681565b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611a7457600080fd5b505afa158015611a88573d6000803e3d6000fd5b505050506040513d6020811015611a9e57600080fd5b5051905080611ab1576000915050611798565b6001600160a01b038316600090815260226020526040902054601554611adb9061178e838561443d565b949350505050565b611aeb6144d5565b6001600160a01b0387166000908152601c602052604090205460ff16611b6857600754611b199060016144f0565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611b8f908290614548565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420899055601f83528184208054821689151517905591805290912080549091168415151790559050611c0e81876144f0565b601555611c1a88613ebc565b505050505050505050565b60175481565b60015462010000900460ff1681565b611c426144d5565b611c526101f48a1115600361458a565b611c626101f4891115600461458a565b611c726101f4881115600561458a565b611c826101f4871115600661458a565b611c926101f4861115600761458a565b611ca26101f4851115600861458a565b611cbd6904ee2d6d415b85acef8160201b841115600961458a565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b60055460009081906001600160a01b03858116911614611d2c576001600160a01b0384166000908152601d6020526040902054611d2f565b60125b6005549091506000906001600160a01b03858116911614611d68576001600160a01b0384166000908152601d6020526040902054611d6b565b60125b9050611d8a82600a0a61178e83600a0a8961443d90919063ffffffff16565b925050505b9392505050565b601f6020526000908152604090205460ff1681565b611db36144d5565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611def8787878761198e565b9050611df9615d6a565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c082018190529092611e6d92899290919089906124e8565b93509350505094509492505050565b60026000541415611ec2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b6002600055600154611ede9062010000900460ff16601c61458a565b611ee6614636565b611eef85614652565b611efa8484836146b5565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b158015611f6e57600080fd5b505afa158015611f82573d6000803e3d6000fd5b50505050611f90848461414d565b6000611f9e8686868561198e565b6000818152602b6020526040812091925083611fc257611fbd86612db8565b611fcb565b611fcb86613ebc565b8254909150611fdc57600282018190555b815415801590611fec5750600085115b156120125761200c868360000154846002015487858a8860060154613a59565b60028301555b600061202b898989888a886000015489600301546147ca565b9050600061203889614886565b905060006120468a836117b4565b600186015490915061205890826144f0565b6001860181905561206d90841115601d61458a565b600185015461207c9084614548565b600186015561208c8a8a896135d9565b6003860155845461209d90896144f0565b8086554260068701556120b3901515601e61458a565b6120c585600001548660010154614933565b6120d38b8b8b8a6001613854565b505060006120e18b8a6133a9565b60048701549091506120f390826144f0565b60048701556121028b82614957565b87156121475761211b8b6121168b876144f0565b6149f7565b6121258b83614a76565b61212f8b84614af5565b6121428b61213d8d876132cf565b614c0d565b6121b3565b6001600160a01b038a166000908152602d6020526040902054612184576001600160a01b038a166000908152602e602052604090208590556121a9565b61218f8a868b6132f1565b6001600160a01b038b166000908152602e60205260409020555b6121b38a8a614ce0565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820187905251600080516020615ddd833981519152918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b60008060008060008060008060006122d28d8d8d8d61198e565b90506122dc615d6a565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a001511361235c578160a00151600003612362565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b6000806124188686868661198e565b9050612422615d6a565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c083015261248c901515602561458a565b602081015181516124a4919061178e9061271061443d565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6000806124f960008611602661458a565b60008461250e5761250988613ebc565b612517565b61251788612db8565b905060008187116125315761252c8288614548565b61253b565b61253b8783614548565b9050600061254d8861178e8b8561443d565b9050600087156125605750878311612565565b508288115b600061257c601054896144f090919063ffffffff16565b42116125a0576001600160a01b038c166000908152601e60205260409020546125a3565b60005b90508180156125c657506125b78b8261443d565b6125c38461271061443d565b11155b156125d057600092505b509a909950975050505050505050565b60026000541415612626576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546126519060ff16600e61458a565b600061265c82614886565b905061266c60008211600f61458a565b6126768282614af5565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b61274e6144d5565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b6000600260005414156127c2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b60026000556127cf614d98565b6001600160a01b0383166000908152601c60205260409020546127f69060ff16601361458a565b6016805461ff00191661010017905560055460009061281d906001600160a01b0316614886565b905061282d60008211601461458a565b612837848561414d565b60006128438583611949565b905061285360008211601561458a565b61285d8583614dc7565b6128678582614c0d565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b1580156128ba57600080fd5b505af11580156128ce573d6000803e3d6000fd5b50506005546128e892506001600160a01b03169050614e85565b6001546040805163eb0835bf60e01b81526001600160a01b038881166004830152602482018690529151600093630100000090049092169163eb0835bf91604480820192602092909190829003018186803b15801561294657600080fd5b505afa15801561295a573d6000803e3d6000fd5b505050506040513d602081101561297057600080fd5b505190506000612981878484614f20565b905061299160008211601661458a565b61299c878288614fe8565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517fd732b7828fa6cee72c285eac756fc66a7477e3dc22e22e7c432f1c265d40b4839181900360a00190a16016805461ff001916905560016000559695505050505050565b612a166144d5565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b612a406144d5565b60018054612a549160ff909116159061458a565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b6004546001600160a01b031681565b600a5481565b612acc6144d5565b60018054911515620100000262ff000019909216919091179055565b600060026000541415612b30576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b6002600055612b3d614d98565b6001600160a01b0383166000908152601c6020526040902054612b649060ff16601061458a565b6016805461ff0019166101001790556000612b7e84614886565b9050612b8e60008211601161458a565b612b98848561414d565b6000612ba385612db8565b90506000612bc168327cb2734119d3b7a9601e1b61178e858561443d565b600554909150612bdd90829088906001600160a01b0316611cf4565b9050612bed60008211601261458a565b6001546040805163256f6ee360e11b81526001600160a01b0389811660048301526024820185905291516000936301000000900490921691634adeddc691604480820192602092909190829003018186803b158015612c4b57600080fd5b505afa158015612c5f573d6000803e3d6000fd5b505050506040513d6020811015612c7557600080fd5b505190506000612c86888684614f20565b90506000612ca468327cb2734119d3b7a9601e1b61178e848861443d565b600554909150612cc09082908b906001600160a01b0316611cf4565b9050612ccc898261508f565b612cd68983614af5565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b158015612d2b57600080fd5b505af1158015612d3f573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517fab4c77c74cd32c85f35416cf03e7ce9e2d4387f7b7f2c1f4bf53daaecf8ea72d93509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b158015612e3157600080fd5b505afa158015612e45573d6000803e3d6000fd5b505050506040513d6020811015612e5b57600080fd5b505192915050565b600060026000541415612eab576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b6002600055612eb8614636565b612ec188614652565b612ed08888888888888861514d565b600160005598975050505050505050565b612ee96144d5565b6001600160a01b03909116600090815260276020526040902055565b601281565b601e6020526000908152604090205481565b612f246144d5565b612f34610e10841015600a61458a565b612f44612710831115600b61458a565b612f54612710821115600c61458a565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612f8e6144d5565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415613007576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b600260005560015461302290610100900460ff16601761458a565b6001600160a01b0384166000908152601c60205260409020546130499060ff16601861458a565b6001600160a01b0383166000908152601c60205260409020546130709060ff16601961458a565b613090836001600160a01b0316856001600160a01b03161415601a61458a565b6016805461ff0019166101001790556130a9848061414d565b6130b3838461414d565b60006130be85614886565b90506130ce60008211601b61458a565b60006130d986612db8565b905060006130e686613ebc565b905060006130f88261178e868661443d565b9050613105818989611cf4565b9050600061312368327cb2734119d3b7a9601e1b61178e878761443d565b60055490915061313f9082908b906001600160a01b0316611cf4565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b1580156131a957600080fd5b505afa1580156131bd573d6000803e3d6000fd5b505050506040513d60208110156131d357600080fd5b5051905060006131e48a8584614f20565b90506131f08b8461508f565b6131fa8a84614dc7565b6132048b88614af5565b61320e8a85614c0d565b6132178a6155da565b6132228a828b614fe8565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b6000816132de57506000611801565b611d8f83836132ec86613ebc565b614113565b6001600160a01b0383166000908152602d6020908152604080832054602e9092528220548285821161332c576133278683614548565b613336565b6133368287614548565b905060006133488361178e868561443d565b9050868311600061335986896144f0565b90506000826133715761336c82856144f0565b61337b565b61337b8285614548565b905061338b8161178e8c8561443d565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b6000816133b857506000611801565b611d8f83836132ec86612db8565b6133ce6144d5565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a602052604081205490914291613422916144f0565b111561343057506000611798565b6012546001600160a01b0383166000908152602a6020526040812054909161345d9161178e904290614548565b6001600160a01b0384166000908152602560205260409020549091508061348957600092505050611798565b6001600160a01b0384166000908152601f602052604081205460ff166134b1576013546134b5565b6014545b6001600160a01b0386166000908152602660205260409020549091506134ee90839061178e9086906134e890869061443d565b9061443d565b95945050505050565b60165460ff1681565b60226020526000908152604090205481565b60246020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff161561357857506001600160a01b038116600090815260256020526040902054611798565b6001600160a01b03821660009081526028602052604081205461359c9084906132cf565b6001600160a01b038416600090815260266020908152604080832054602590925290912054919250611794916135d39084906144f0565b90614548565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b15801561364057600080fd5b505afa158015613654573d6000803e3d6000fd5b505050506040513d602081101561366a57600080fd5b5051949350505050565b6001600160a01b0381166000908152602d60205260408120548190806136a1576000809250925050613707565b60006136ac85613ebc565b6001600160a01b0386166000908152602e60205260408120549192508282116136de576136d98383614548565b6136e8565b6136e88284614548565b905060006136fa8361178e878561443d565b9390921195509193505050505b915091565b60266020526000908152604090205481565b60135481565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b1580156137aa57600080fd5b505afa1580156137be573d6000803e3d6000fd5b505050506040513d60208110156137d457600080fd5b50519695505050505050565b6137e86144d5565b6137fc6001600160a01b038316848361564f565b505050565b6138096144d5565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b6138336144d5565b601755565b6138406144d5565b61384f6127108211600261458a565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b1580156138ca57600080fd5b505afa1580156138de573d6000803e3d6000fd5b505050506040513d60408110156138f457600080fd5b508051602090910151909890975095505050505050565b6139136144d5565b6001600160a01b0382166000908152602360205260409020548082111561394d57613947836139428484614548565b61508f565b50613960565b6137fc8361395b8385614548565b614dc7565b5050565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b1580156139f957600080fd5b505afa158015613a0d573d6000803e3d6000fd5b505050506040513d6020811015613a2357600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b6000806000613a6b8a8a8a8a886124e8565b90925090506000613a7c8a876144f0565b905060008815613aab5783613a9a57613a958284614548565b613aa4565b613aa482846144f0565b9050613acc565b83613abf57613aba82846144f0565b613ac9565b613ac98284614548565b90505b613ada8161178e8a8561443d565b9c9b505050505050505050505050565b60155481565b60026000541415613b36576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613b6b5733600090815260196020526040902054613b6b9060ff16602261458a565b6016805460ff19169055613b7f848461414d565b6000613b8d8686868661198e565b9050613b97615d6a565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613c03901515602361458a565b600080613c14898989896000613854565b91509150613c278260001415602461458a565b8160021415613c5c57613c44898989600087600001518b8f61514d565b50506016805460ff1916600117905550613eaa915050565b6000613c6889836132cf565b6001600160a01b038a166000908152602c6020526040902054909150613c8e90826144f0565b6001600160a01b038a166000818152602c60209081526040918290209390935580519182529181018490528082018390529051600080516020615e888339815191529181900360600190a1613ce78985608001516156a1565b8615613d165760208401518451613d08918b91613d0391614548565b614a76565b613d168961213d8b856132cf565b600087613d2b57613d2689613ebc565b613d34565b613d3489612db8565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613df35750846020015183105b15613e21576020850151600090613e0a9085614548565b9050613e1f8b613e1a8d846132cf565b614af5565b505b87613e3457613e34898660000151615756565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613e80908b9061213d9082906132cf565b613e968a613e908c6009546132cf565b89614fe8565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b158015612e3157600080fd5b601b8181548110613f4157fe5b6000918252602090912001546001600160a01b0316905081565b613f636144d5565b6001600160a01b0381166000908152601c6020526040902054613f8a9060ff16600d61458a565b6001600160a01b038116600090815260226020526040902054601554613faf91614548565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f835281842080548216905591805290912080549091169055600754614028906001614548565b60075550565b6140366144d5565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b61406c6144d5565b6016805491151563010000000263ff00000019909216919091179055565b60006140946144d5565b6001600160a01b0383166000908152602c6020526040902054806140bc576000915050611801565b6001600160a01b0384166000908152602c6020526040812055611d8f848285614fe8565b601a6020526000908152604090205460ff1681565b6005546001600160a01b031681565b6003546001600160a01b031681565b60008261412257506000611d8f565b6001600160a01b0384166000908152601d60205260409020546134ee8361178e86600a85900a61443d565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b1580156141ad57600080fd5b505af11580156141c1573d6000803e3d6000fd5b505050506040513d60208110156141d757600080fd5b50519050806141e65750613960565b6001600160a01b0383166000908152602a602052604090205461423357601254614214906134e84282614496565b6001600160a01b0384166000908152602a602052604090205550613960565b6012546001600160a01b0384166000908152602a6020526040902054429161425b91906144f0565b11156142675750613960565b6000614272846133f9565b6001600160a01b03851660009081526029602052604090205490915061429890826144f0565b6001600160a01b0385166000908152602960205260409020556012546142c2906134e84282614496565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b1580156137aa57600080fd5b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144355780601f1061440a57610100808354040283529160200191614435565b820191906000526020600020905b81548152906001019060200180831161441857829003601f168201915b505050505081565b60008261444c57506000611801565b8282028284828161445957fe5b0414611d8f5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e1d6021913960400191505060405180910390fd5b6000611d8f83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b8152506157be565b6006546144ee906001600160a01b03163314603561458a565b565b600082820183811015611d8f576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b6000611d8f83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f770000815250615860565b6000818152603060205260409020826137fc5760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156146275780601f106145fc57610100808354040283529160200191614627565b820191906000526020600020905b81548152906001019060200180831161460a57829003601f168201915b50509250505060405180910390fd5b601754614642576144ee565b6144ee6017543a1115603761458a565b336001600160a01b0382161415614668576146b2565b6003546001600160a01b0316331415614680576146b2565b6001600160a01b03811660009081526018602090815260408083203384529091529020546146b29060ff16602961458a565b50565b801561472e576146da826001600160a01b0316846001600160a01b031614602a61458a565b6001600160a01b0383166000908152601c60205260409020546147019060ff16602b61458a565b6001600160a01b0383166000908152601f60205260409020546147299060ff1615602c61458a565b6137fc565b6001600160a01b0383166000908152601c60205260409020546147559060ff16602d61458a565b6001600160a01b0383166000908152601f602052604090205461477c9060ff16602e61458a565b6001600160a01b0382166000908152601f60205260409020546147a49060ff1615602f61458a565b6001600160a01b03821660009081526020805260409020546137fc9060ff16603061458a565b6000806147da898989898961432c565b905060006147ec8a8a8a8a898961397c565b90506147f882826144f0565b915060006148068a846132cf565b6001600160a01b038b166000908152602c602052604090205490915061482c90826144f0565b6001600160a01b038b166000818152602c60209081526040918290209390935580519182529181018590528082018390529051600080516020615e888339815191529181900360600190a150909998505050505050505050565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b1580156148e057600080fd5b505afa1580156148f4573d6000803e3d6000fd5b505050506040513d602081101561490a57600080fd5b50516001600160a01b03851660009081526021602052604090208190559050611adb8183614548565b81614949576149448115602761458a565b613960565b61396081831015602861458a565b6001600160a01b03821660009081526026602052604090205461497a90826144f0565b6001600160a01b0383166000908152602660208181526040808420859055602582529092205491526149af911115603461458a565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614a1a90826144f0565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614a999082614548565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614b1890826144f0565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614b6e57600080fd5b505afa158015614b82573d6000803e3d6000fd5b505050506040513d6020811015614b9857600080fd5b50516001600160a01b038416600090815260256020526040902054909150614bc490821015603161458a565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b038516600090815260259091529190912054614c65918390615860565b6001600160a01b03831660009081526025602090815260408083208490556026909152902054614c98911015603261458a565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054614d0390826144f0565b6001600160a01b0383166000908152602d6020908152604080832093909355602f9052205480156137fc576001600160a01b0383166000908152602d60205260409020548110156137fc576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b60165462010000900460ff16156144ee57336000908152601a60205260409020546144ee9060ff16603661458a565b6001600160a01b038216600090815260236020526040902054818111614e30576001600160a01b038316600081815260236020908152604080832092909255815192835282018390528051600080516020615e3e8339815191529281900390910190a150613960565b614e3a8183614548565b6001600160a01b0384166000818152602360209081526040918290209390935580519182529181018490528151600080516020615e3e833981519152929181900390910190a1505050565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b158015614ed457600080fd5b505afa158015614ee8573d6000803e3d6000fd5b505050506040513d6020811015614efe57600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b600080614f3d61271061178e614f368287614548565b879061443d565b90506000614f4b8583614548565b6001600160a01b0387166000908152602c6020526040902054909150614f7190826144f0565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b86614fb681846117b4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b614ffc6001600160a01b038416828461564f565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b15801561504257600080fd5b505afa158015615056573d6000803e3d6000fd5b505050506040513d602081101561506c57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b6001600160a01b0382166000908152602360205260409020546150b290826144f0565b6001600160a01b0383166000908152602360209081526040808320939093556024905220548015615104576001600160a01b03831660009081526023602052604090205461510490821015603361458a565b604080516001600160a01b03851681526020810184905281517f64243679a443432e2293343b77d411ff6144370404618f00ca0d2025d9ca9882929181900390910190a1505050565b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b1580156151d157600080fd5b505afa1580156151e5573d6000803e3d6000fd5b505050506151f3878761414d565b60006152018989898761198e565b6000818152602b60205260409020805491925090615222901515601f61458a565b6152348682600001541015602061458a565b6152468782600101541015602161458a565b6001810154815460048301546000916152639161178e908b61443d565b60048401549091506152759082614548565b60048401556152848b826156a1565b506000806152968d8d8d8d8d8d6158ba565b855491935091508914615416576152ae8c8c8a6135d9565b600385015583546152bf908a614548565b80855560018501546152d19190614933565b6152df8d8d8d8b6001613854565b5050871561530c576153028c61211686600101548661454890919063ffffffff16565b61530c8c8a614a76565b6000886153215761531c8c613ebc565b61532a565b61532a8c612db8565b9050600080516020615dfd833981519152868f8f8f8f8f8f8861534d8c8c614548565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820183905251600080516020615ddd833981519152918190036101000190a150615578565b8715615430576154268c846149f7565b6154308c8a614a76565b600088615445576154408c613ebc565b61544e565b61544e8c612db8565b9050600080516020615dfd833981519152868f8f8f8f8f8f886154718c8c614548565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b87615587576155878b8a615756565b81156155c75787156155a1576155a18c61213d8e856132cf565b60006155ad8d836132cf565b90506155ba8d828a614fe8565b95506124a4945050505050565b5060009c9b505050505050505050505050565b6001600160a01b03811660009081526027602090815260408083205460259092529091205410156146b2576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b1790526137fc908490615ab4565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b0385166000908152602690915291909120546156fa918390615860565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d60205260409020548082111561579657506001600160a01b0382166000908152602d6020526040812055613960565b6157a08183614548565b6001600160a01b0384166000908152602d6020526040902055505050565b6000818361584a5760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561580f5781810151838201526020016157f7565b50505050905090810190601f16801561583c5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161585657fe5b0495945050505050565b600081848411156158b25760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561580f5781810151838201526020016157f7565b505050900390565b60008060006158cb8989898761198e565b6000818152602b602052604081208054600382015493945090926158f8918d918d918d918b918d916147ca565b90506000806000806159198d876000015488600201548d8a600601546124e8565b875491955085935091506159319061178e8d8461443d565b9250505060008280156159445750600082115b1561597457506005840180548201905580886159745760006159668e846132cf565b90506159728e82614c0d565b505b821580156159825750600082115b156159c65760018501546159969083614548565b6001860155886159ba5760006159ac8e846132cf565b90506159b88e82614af5565b505b60058501805483900390555b8a156159ee576159d6818c6144f0565b60018601549091506159e8908c614548565b60018601555b84548a1415615a11576001850154615a079082906144f0565b6000600187015590505b8084811115615a2b57615a248286614548565b9050615a5f565b6001860154615a3a9086614548565b60018701558915615a5f576000615a518f876132cf565b9050615a5d8f82614c0d565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615b09826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615b659092919063ffffffff16565b8051909150156137fc57808060200190516020811015615b2857600080fd5b50516137fc5760405162461bcd60e51b815260040180806020018281038252602a815260200180615e5e602a913960400191505060405180910390fd5b6060611adb848460008585615b7985615c80565b615bca576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615c095780518252601f199092019160209182019101615bea565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615c6b576040519150601f19603f3d011682016040523d82523d6000602084013e615c70565b606091505b50915091506124a4828286615c86565b3b151590565b60608315615c95575081611d8f565b825115615ca55782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561580f5781810151838201526020016157f7565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615d2d5782800160ff19823516178555615d5a565b82800160010185558215615d5a579182015b82811115615d5a578235825591602001919060010190615d3f565b50615d66929150615da7565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615d665760008155600101615da856fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c0020853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b77393d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f77e1e812596aac93a06ecc4ca627014d18e30f5c33b825160cc9d5c0ba61e452275361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011aa2646970667358221220278b7f556d929c862af1a8bac6e804ea6cff611f6b4579f83344dae30cd2030664736f6c634300060c0033
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106105695760003560e01c806304fef1db1461056e57806307c58752146105a65780630842b076146105ae5780630a48d5a9146105b657806310eb56c2146105e2578063126082cf146105ea57806312d43a51146105f2578063134ca63b14610616578063174d26941461061e578063181e210e146106265780631aa4ace5146106425780631ce9cb8f1461066857806324b0c04d1461068e57806324ca984e146106af57806328e67be5146106d557806329ff96151461074a5780632c668ec1146107705780632d4b05761461079c57806330455ede146107da578063318bc689146107f957806334c1557d146105ea578063351a964d14610801578063392e53cd146108095780633a05dcc1146108115780633c5a6e35146108375780633de39c11146108855780633e72a2621461088d57806340eb38021461089557806342152873146108e657806342b60b031461091a5780634453a3741461094057806345a6f3701461096e57806348d91abf146109c757806348f35cbb14610a0b5780634a3f088d14610a135780634a993ee914610a945780634befe2ca14610aba5780634d47b30414610ac2578063514ea4bf14610aca57806351723e8214610b1f578063523fba7f14610b5d578063529a356f14610b8357806352f55eed14610ba95780635c07eaab14610bcf5780635f7bc11914610c0f5780636092219914610c3557806362287a3214610c635780636274980314610c6b5780636abbe0c814610c915780636ae0b15414610c995780636be6026b14610cbf57806371089f4d14610cc7578063711e619014610ced578063724e78da14610d1b578063728cdbca14610d41578063741bef1a14610d895780637a210a2b14610d915780637c2eb9f714610d99578063817bb85714610db857806381a612d614610de657806382a0849014610e0c5780638585f4d214610e60578063870d917c14610e8c57806388b1fbdf14610e945780638a27d46814610eba5780638a39735a146105ea5780638a78daa814610ee35780638ee573ac14610f095780638f7b840414610f2f5780639060b1ca14610f555780639331621214610f5d57806395082d2514610f955780639698d25a14610f9d5780639849e41214610fc35780639899cd0214610fcb5780639d7432ca14610ff75780639f392eb314611029578063a22f239214611031578063a42ab3d214611039578063a5e90eee14611065578063a93acac214611093578063ab08c1c6146110b9578063ab2f3ad4146110c1578063ad1e4f8d146110e7578063ae3302c21461110d578063b06423f314611115578063b136ca491461111d578063b1cc53ab14611143578063b364accb1461117b578063c3c7b9e9146111a1578063c4f718bf146111c7578063c65bc7b1146111cf578063c7e074c3146111f5578063cea0c32814611235578063cfad57a21461126b578063d2fa635e14611291578063d3127e63146112ae578063d54d5a9f146112cb578063d66b000d1461132a578063d8f897c314611356578063d9ac42251461137c578063da76524c14611384578063daf9c210146113ce578063db3555fb146113f4578063db97495f1461141a578063dc8f5fac14611466578063de2ea9481461146e578063df73a267146114b6578063e124e6d2146114be578063e468baf0146114e4578063e67f59a714611501578063efa10a6e14611527578063f07456ce14611553578063f07bbf7714611579578063f255527814611598578063f3ae2415146115c6578063f5b91b7b146115ec578063f887ea40146115f4578063fa12dbc0146115fc578063fbfded6d1461162e578063fce28c101461165c578063fdaf6ac314611664578063fed1a606146116a8575b600080fd5b6105946004803603602081101561058457600080fd5b50356001600160a01b031661173a565b60408051918252519081900360200190f35b61059461179d565b6105946117ae565b610594600480360360408110156105cc57600080fd5b506001600160a01b0381351690602001356117b4565b610594611807565b61059461180d565b6105fa611813565b604080516001600160a01b039092168252519081900360200190f35b610594611822565b610594611828565b61062e61182e565b604080519115158252519081900360200190f35b6105946004803603602081101561065857600080fd5b50356001600160a01b031661183e565b6105946004803603602081101561067e57600080fd5b50356001600160a01b0316611850565b6106ad600480360360208110156106a457600080fd5b50351515611862565b005b6106ad600480360360208110156106c557600080fd5b50356001600160a01b0316611886565b6106ad600480360360408110156106eb57600080fd5b81359190810190604081016020820135600160201b81111561070c57600080fd5b82018360208201111561071e57600080fd5b803590602001918460018302840111600160201b8311171561073f57600080fd5b5090925090506118b7565b6105946004803603602081101561076057600080fd5b50356001600160a01b0316611935565b6105946004803603604081101561078657600080fd5b506001600160a01b038135169060200135611949565b610594600480360360808110156107b257600080fd5b506001600160a01b03813581169160208101358216916040820135169060600135151561198e565b6106ad600480360360208110156107f057600080fd5b503515156119e4565b610594611a06565b61062e611a0c565b61062e611a1a565b6105946004803603602081101561082757600080fd5b50356001600160a01b0316611a23565b6106ad600480360360e081101561084d57600080fd5b506001600160a01b038135169060208101359060408101359060608101359060808101359060a081013515159060c001351515611ae3565b610594611c25565b61062e611c2b565b6106ad60048036036101208110156108ac57600080fd5b5080359060208101359060408101359060608101359060808101359060a08101359060c08101359060e08101359061010001351515611c3a565b610594600480360360608110156108fc57600080fd5b508035906001600160a01b0360208201358116916040013516611cf4565b61062e6004803603602081101561093057600080fd5b50356001600160a01b0316611d96565b6106ad6004803603604081101561095657600080fd5b506001600160a01b0381351690602001351515611dab565b6109ac6004803603608081101561098457600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515611dde565b60408051921515835260208301919091528051918290030190f35b6106ad600480360360a08110156109dd57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013590608001351515611e7c565b6105fa6122a9565b610a5160048036036080811015610a2957600080fd5b506001600160a01b0381358116916020810135821691604082013516906060013515156122b8565b604080519889526020890197909752878701959095526060870193909352608086019190915260a0850152151560c084015260e083015251908190036101000190f35b61059460048036036020811015610aaa57600080fd5b50356001600160a01b03166123ae565b6105946123c0565b6105946123c6565b610ae760048036036020811015610ae057600080fd5b50356123cc565b604080519788526020880196909652868601949094526060860192909252608085015260a084015260c0830152519081900360e00190f35b61059460048036036080811015610b3557600080fd5b506001600160a01b038135811691602081013582169160408201351690606001351515612409565b61059460048036036020811015610b7357600080fd5b50356001600160a01b03166124af565b61062e60048036036020811015610b9957600080fd5b50356001600160a01b03166124c1565b61059460048036036020811015610bbf57600080fd5b50356001600160a01b03166124d6565b6109ac600480360360a0811015610be557600080fd5b506001600160a01b03813516906020810135906040810135906060810135151590608001356124e8565b6106ad60048036036020811015610c2557600080fd5b50356001600160a01b03166125e0565b61062e60048036036040811015610c4b57600080fd5b506001600160a01b03813581169160200135166126c3565b6105946126e3565b61059460048036036020811015610c8157600080fd5b50356001600160a01b03166126e9565b6105fa6126fb565b6106ad60048036036020811015610caf57600080fd5b50356001600160a01b0316612711565b61059461273f565b6106ad60048036036020811015610cdd57600080fd5b50356001600160a01b0316612746565b61059460048036036040811015610d0357600080fd5b506001600160a01b038135811691602001351661277a565b6106ad60048036036020811015610d3157600080fd5b50356001600160a01b0316612a0e565b6106ad600480360360c0811015610d5757600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101359060808101359060a00135612a38565b6105fa612aaf565b610594612abe565b6106ad60048036036020811015610daf57600080fd5b50351515612ac4565b61059460048036036040811015610dce57600080fd5b506001600160a01b0381358116916020013516612ae8565b61059460048036036020811015610dfc57600080fd5b50356001600160a01b0316612db8565b610594600480360360e0811015610e2257600080fd5b506001600160a01b0381358116916020810135821691604082013581169160608101359160808201359160a081013515159160c09091013516612e63565b6106ad60048036036040811015610e7657600080fd5b506001600160a01b038135169060200135612ee1565b610594612f05565b61059460048036036020811015610eaa57600080fd5b50356001600160a01b0316612f0a565b6106ad60048036036060811015610ed057600080fd5b5080359060208101359060400135612f1c565b61059460048036036020811015610ef957600080fd5b50356001600160a01b0316612f62565b61059460048036036020811015610f1f57600080fd5b50356001600160a01b0316612f74565b6106ad60048036036020811015610f4557600080fd5b50356001600160a01b0316612f86565b61062e612fb0565b61059460048036036060811015610f7357600080fd5b506001600160a01b038135811691602081013582169160409091013516612fbf565b6105946132a7565b61059460048036036020811015610fb357600080fd5b50356001600160a01b03166132b7565b6105946132c9565b61059460048036036040811015610fe157600080fd5b506001600160a01b0381351690602001356132cf565b6105946004803603606081101561100d57600080fd5b506001600160a01b0381351690602081013590604001356132f1565b61062e61339a565b6105946133a3565b6105946004803603604081101561104f57600080fd5b506001600160a01b0381351690602001356133a9565b6106ad6004803603604081101561107b57600080fd5b506001600160a01b03813516906020013515156133c6565b610594600480360360208110156110a957600080fd5b50356001600160a01b03166133f9565b61062e6134f7565b610594600480360360208110156110d757600080fd5b50356001600160a01b0316613500565b610594600480360360208110156110fd57600080fd5b50356001600160a01b0316613512565b610594613524565b61062e61352a565b6105946004803603602081101561113357600080fd5b50356001600160a01b0316613538565b6105946004803603606081101561115957600080fd5b506001600160a01b0381358116916020810135909116906040013515156135d9565b6109ac6004803603602081101561119157600080fd5b50356001600160a01b0316613674565b610594600480360360208110156111b757600080fd5b50356001600160a01b031661370c565b61059461371e565b610594600480360360208110156111e557600080fd5b50356001600160a01b0316613724565b610594600480360360a081101561120b57600080fd5b506001600160a01b0381351690602081013590604081013590606081013590608001351515613736565b6106ad6004803603606081101561124b57600080fd5b506001600160a01b038135811691602081013590911690604001356137e0565b6106ad6004803603602081101561128157600080fd5b50356001600160a01b0316613801565b6106ad600480360360208110156112a757600080fd5b503561382b565b6106ad600480360360208110156112c457600080fd5b5035613838565b611311600480360360a08110156112e157600080fd5b506001600160a01b0381358116916020810135821691604082013516906060810135151590608001351515613854565b6040805192835260208301919091528051918290030190f35b6106ad6004803603604081101561134057600080fd5b506001600160a01b03813516906020013561390b565b6105946004803603602081101561136c57600080fd5b50356001600160a01b0316613964565b610594613976565b610594600480360360c081101561139a57600080fd5b506001600160a01b038135811691602081013582169160408201351690606081013515159060808101359060a0013561397c565b61062e600480360360208110156113e457600080fd5b50356001600160a01b0316613a30565b61062e6004803603602081101561140a57600080fd5b50356001600160a01b0316613a45565b610594600480360360e081101561143057600080fd5b506001600160a01b0381351690602081013590604081013590606081013515159060808101359060a08101359060c00135613a59565b610594613aea565b6106ad600480360360a081101561148457600080fd5b506001600160a01b03813581169160208101358216916040820135811691606081013515159160809091013516613af0565b610594613eb6565b610594600480360360208110156114d457600080fd5b50356001600160a01b0316613ebc565b6105fa600480360360208110156114fa57600080fd5b5035613f34565b6106ad6004803603602081101561151757600080fd5b50356001600160a01b0316613f5b565b6106ad6004803603604081101561153d57600080fd5b506001600160a01b03813516906020013561402e565b6105946004803603602081101561156957600080fd5b50356001600160a01b0316614052565b6106ad6004803603602081101561158f57600080fd5b50351515614064565b610594600480360360408110156115ae57600080fd5b506001600160a01b038135811691602001351661408a565b61062e600480360360208110156115dc57600080fd5b50356001600160a01b03166140e0565b6105fa6140f5565b6105fa614104565b6105946004803603606081101561161257600080fd5b506001600160a01b038135169060208101359060400135614113565b6106ad6004803603604081101561164457600080fd5b506001600160a01b038135811691602001351661414d565b610594614326565b610594600480360360a081101561167a57600080fd5b506001600160a01b03813581169160208101358216916040820135169060608101351515906080013561432c565b6116c5600480360360208110156116be57600080fd5b50356143a2565b6040805160208082528351818301528351919283929083019185019080838360005b838110156116ff5781810151838201526020016116e7565b50505050905090810190601f16801561172c5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6001600160a01b03811660009081526025602052604081205480611762576000915050611798565b6001600160a01b03831660009081526026602052604090205461179490829061178e90620f424061443d565b90614496565b9150505b919050565b6904ee2d6d415b85acef8160201b81565b601b5490565b6000816117c357506000611801565b60006117ce84612db8565b6001600160a01b0385166000908152601d60205260409020549091506117fc600a82900a61178e868561443d565b925050505b92915050565b600b5481565b61271081565b6006546001600160a01b031681565b60145481565b60095481565b6016546301000000900460ff1681565b60236020526000908152604090205481565b602c6020526000908152604090205481565b61186a6144d5565b60168054911515620100000262ff000019909216919091179055565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19166001179055565b6002546001600160a01b03163314611916576040805162461bcd60e51b815260206004820152601e60248201527f5661756c743a20696e76616c6964206572726f72436f6e74726f6c6c65720000604482015290519081900360640190fd5b600083815260306020526040902061192f908383615cec565b50505050565b60006118018261194484613538565b6117b4565b60008061195584613ebc565b905060006119738261178e8668327cb2734119d3b7a9601e1b61443d565b6005549091506117fc9082906001600160a01b031687611cf4565b604080516001600160601b0319606096871b811660208084019190915295871b811660348301529390951b9092166048850152151560f81b605c8401528051603d818503018152605d9093019052815191012090565b6119ec6144d5565b600180549115156101000261ff0019909216919091179055565b600f5481565b600154610100900460ff1681565b60015460ff1681565b600080600560009054906101000a90046001600160a01b03166001600160a01b03166318160ddd6040518163ffffffff1660e01b815260040160206040518083038186803b158015611a7457600080fd5b505afa158015611a88573d6000803e3d6000fd5b505050506040513d6020811015611a9e57600080fd5b5051905080611ab1576000915050611798565b6001600160a01b038316600090815260226020526040902054601554611adb9061178e838561443d565b949350505050565b611aeb6144d5565b6001600160a01b0387166000908152601c602052604090205460ff16611b6857600754611b199060016144f0565b600755601b80546001810182556000919091527f3ad8aa4f87544323a9d1e5dd902f40c356527a7955687113db5f9a85ad579dc10180546001600160a01b0319166001600160a01b0389161790555b6015546001600160a01b038816600090815260226020526040902054611b8f908290614548565b6001600160a01b0389166000908152601c602090815260408083208054600160ff1991821617909155601d83528184208c9055602283528184208b9055601e83528184208a905560248352818420899055601f83528184208054821689151517905591805290912080549091168415151790559050611c0e81876144f0565b601555611c1a88613ebc565b505050505050505050565b60175481565b60015462010000900460ff1681565b611c426144d5565b611c526101f48a1115600361458a565b611c626101f4891115600461458a565b611c726101f4881115600561458a565b611c826101f4871115600661458a565b611c926101f4861115600761458a565b611ca26101f4851115600861458a565b611cbd6904ee2d6d415b85acef8160201b841115600961458a565b600a98909855600b96909655600c94909455600d92909255600e55600f556009556010556011805460ff1916911515919091179055565b60055460009081906001600160a01b03858116911614611d2c576001600160a01b0384166000908152601d6020526040902054611d2f565b60125b6005549091506000906001600160a01b03858116911614611d68576001600160a01b0384166000908152601d6020526040902054611d6b565b60125b9050611d8a82600a0a61178e83600a0a8961443d90919063ffffffff16565b925050505b9392505050565b601f6020526000908152604090205460ff1681565b611db36144d5565b6001600160a01b03919091166000908152601960205260409020805460ff1916911515919091179055565b6000806000611def8787878761198e565b9050611df9615d6a565b506000818152602b6020908152604091829020825160e081018452815480825260018301549382019390935260028201549381018490526003820154606082015260048201546080820152600582015460a082015260069091015460c082018190529092611e6d92899290919089906124e8565b93509350505094509492505050565b60026000541415611ec2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b6002600055600154611ede9062010000900460ff16601c61458a565b611ee6614636565b611eef85614652565b611efa8484836146b5565b60015460408051634eae147d60e11b81526001600160a01b038881166004830152878116602483015286811660448301526064820186905284151560848301529151630100000090930490911691639d5c28fa9160a480820192600092909190829003018186803b158015611f6e57600080fd5b505afa158015611f82573d6000803e3d6000fd5b50505050611f90848461414d565b6000611f9e8686868561198e565b6000818152602b6020526040812091925083611fc257611fbd86612db8565b611fcb565b611fcb86613ebc565b8254909150611fdc57600282018190555b815415801590611fec5750600085115b156120125761200c868360000154846002015487858a8860060154613a59565b60028301555b600061202b898989888a886000015489600301546147ca565b9050600061203889614886565b905060006120468a836117b4565b600186015490915061205890826144f0565b6001860181905561206d90841115601d61458a565b600185015461207c9084614548565b600186015561208c8a8a896135d9565b6003860155845461209d90896144f0565b8086554260068701556120b3901515601e61458a565b6120c585600001548660010154614933565b6120d38b8b8b8a6001613854565b505060006120e18b8a6133a9565b60048701549091506120f390826144f0565b60048701556121028b82614957565b87156121475761211b8b6121168b876144f0565b6149f7565b6121258b83614a76565b61212f8b84614af5565b6121428b61213d8d876132cf565b614c0d565b6121b3565b6001600160a01b038a166000908152602d6020526040902054612184576001600160a01b038a166000908152602e602052604090208590556121a9565b61218f8a868b6132f1565b6001600160a01b038b166000908152602e60205260409020555b6121b38a8a614ce0565b604080518881526001600160a01b03808f166020830152808e16828401528c1660608201526080810184905260a081018b905289151560c082015260e08101879052610100810186905290517f2fe68525253654c21998f35787a8d0f361905ef647c854092430ab65f2f15022918190036101200190a1855460018701546002880154600389015460048a015460058b0154604080518e81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820187905251600080516020615ddd833981519152918190036101000190a15050600160005550505050505050505050565b6002546001600160a01b031681565b60008060008060008060008060006122d28d8d8d8d61198e565b90506122dc615d6a565b602b60008381526020019081526020016000206040518060e001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481526020016005820154815260200160068201548152505090506000808260a001511361235c578160a00151600003612362565b8160a001515b9050816000015182602001518360400151846060015185608001518560008860a0015112158860c001519a509a509a509a509a509a509a509a5050505094995094995094999196509450565b60276020526000908152604090205481565b6101f481565b600c5481565b602b602052600090815260409020805460018201546002830154600384015460048501546005860154600690960154949593949293919290919087565b6000806124188686868661198e565b9050612422615d6a565b506000818152602b6020908152604091829020825160e0810184528154815260018201549281018390526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c083015261248c901515602561458a565b602081015181516124a4919061178e9061271061443d565b979650505050505050565b60216020526000908152604090205481565b60196020526000908152604090205460ff1681565b60256020526000908152604090205481565b6000806124f960008611602661458a565b60008461250e5761250988613ebc565b612517565b61251788612db8565b905060008187116125315761252c8288614548565b61253b565b61253b8783614548565b9050600061254d8861178e8b8561443d565b9050600087156125605750878311612565565b508288115b600061257c601054896144f090919063ffffffff16565b42116125a0576001600160a01b038c166000908152601e60205260409020546125a3565b60005b90508180156125c657506125b78b8261443d565b6125c38461271061443d565b11155b156125d057600092505b509a909950975050505050505050565b60026000541415612626576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b600260009081556001600160a01b0382168152601c60205260409020546126519060ff16600e61458a565b600061265c82614886565b905061266c60008211600f61458a565b6126768282614af5565b604080516001600160a01b03841681526020810183905281517fa5a389190ebf6170a133bda5c769b77f4d6715b8aa172ec0ddf8473d0b4944bd929181900390910190a150506001600055565b601860209081526000928352604080842090915290825290205460ff1681565b60075481565b602e6020526000908152604090205481565b600154630100000090046001600160a01b031681565b3360009081526018602090815260408083206001600160a01b0394909416835292905220805460ff19169055565b620f424081565b61274e6144d5565b600180546001600160a01b039092166301000000026301000000600160b81b0319909216919091179055565b6000600260005414156127c2576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b60026000556127cf614d98565b6001600160a01b0383166000908152601c60205260409020546127f69060ff16601361458a565b6016805461ff00191661010017905560055460009061281d906001600160a01b0316614886565b905061282d60008211601461458a565b612837848561414d565b60006128438583611949565b905061285360008211601561458a565b61285d8583614dc7565b6128678582614c0d565b60055460408051632770a7eb60e21b81523060048201526024810185905290516001600160a01b0390921691639dc29fac9160448082019260009290919082900301818387803b1580156128ba57600080fd5b505af11580156128ce573d6000803e3d6000fd5b50506005546128e892506001600160a01b03169050614e85565b6001546040805163eb0835bf60e01b81526001600160a01b038881166004830152602482018690529151600093630100000090049092169163eb0835bf91604480820192602092909190829003018186803b15801561294657600080fd5b505afa15801561295a573d6000803e3d6000fd5b505050506040513d602081101561297057600080fd5b505190506000612981878484614f20565b905061299160008211601661458a565b61299c878288614fe8565b604080516001600160a01b03808916825289166020820152808201869052606081018390526080810184905290517fd732b7828fa6cee72c285eac756fc66a7477e3dc22e22e7c432f1c265d40b4839181900360a00190a16016805461ff001916905560016000559695505050505050565b612a166144d5565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b612a406144d5565b60018054612a549160ff909116159061458a565b6001805460ff191681179055600380546001600160a01b03199081166001600160a01b039889161790915560058054821696881696909617909555600480549095169390951692909217909255600991909155601355601455565b6004546001600160a01b031681565b600a5481565b612acc6144d5565b60018054911515620100000262ff000019909216919091179055565b600060026000541415612b30576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b6002600055612b3d614d98565b6001600160a01b0383166000908152601c6020526040902054612b649060ff16601061458a565b6016805461ff0019166101001790556000612b7e84614886565b9050612b8e60008211601161458a565b612b98848561414d565b6000612ba385612db8565b90506000612bc168327cb2734119d3b7a9601e1b61178e858561443d565b600554909150612bdd90829088906001600160a01b0316611cf4565b9050612bed60008211601261458a565b6001546040805163256f6ee360e11b81526001600160a01b0389811660048301526024820185905291516000936301000000900490921691634adeddc691604480820192602092909190829003018186803b158015612c4b57600080fd5b505afa158015612c5f573d6000803e3d6000fd5b505050506040513d6020811015612c7557600080fd5b505190506000612c86888684614f20565b90506000612ca468327cb2734119d3b7a9601e1b61178e848861443d565b600554909150612cc09082908b906001600160a01b0316611cf4565b9050612ccc898261508f565b612cd68983614af5565b600554604080516340c10f1960e01b81526001600160a01b038b8116600483015260248201859052915191909216916340c10f1991604480830192600092919082900301818387803b158015612d2b57600080fd5b505af1158015612d3f573d6000803e3d6000fd5b5050604080516001600160a01b03808d1682528d1660208201528082018a9052606081018590526080810187905290517fab4c77c74cd32c85f35416cf03e7ce9e2d4387f7b7f2c1f4bf53daaecf8ea72d93509081900360a0019150a16016805461ff0019169055600160005598975050505050505050565b60048054601654604080516317e1d38560e11b81526001600160a01b038681169582019590955260006024820181905260ff80851615156044840152610100909404909316151560648201529051919390921691632fc3a70a916084808301926020929190829003018186803b158015612e3157600080fd5b505afa158015612e45573d6000803e3d6000fd5b505050506040513d6020811015612e5b57600080fd5b505192915050565b600060026000541415612eab576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b6002600055612eb8614636565b612ec188614652565b612ed08888888888888861514d565b600160005598975050505050505050565b612ee96144d5565b6001600160a01b03909116600090815260276020526040902055565b601281565b601e6020526000908152604090205481565b612f246144d5565b612f34610e10841015600a61458a565b612f44612710831115600b61458a565b612f54612710821115600c61458a565b601292909255601355601455565b602d6020526000908152604090205481565b601d6020526000908152604090205481565b612f8e6144d5565b600280546001600160a01b0319166001600160a01b0392909216919091179055565b60165462010000900460ff1681565b600060026000541415613007576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b600260005560015461302290610100900460ff16601761458a565b6001600160a01b0384166000908152601c60205260409020546130499060ff16601861458a565b6001600160a01b0383166000908152601c60205260409020546130709060ff16601961458a565b613090836001600160a01b0316856001600160a01b03161415601a61458a565b6016805461ff0019166101001790556130a9848061414d565b6130b3838461414d565b60006130be85614886565b90506130ce60008211601b61458a565b60006130d986612db8565b905060006130e686613ebc565b905060006130f88261178e868661443d565b9050613105818989611cf4565b9050600061312368327cb2734119d3b7a9601e1b61178e878761443d565b60055490915061313f9082908b906001600160a01b0316611cf4565b60015460408051636d099c0b60e11b81526001600160a01b038d811660048301528c8116602483015260448201859052915193945060009363010000009093049091169163da13381691606480820192602092909190829003018186803b1580156131a957600080fd5b505afa1580156131bd573d6000803e3d6000fd5b505050506040513d60208110156131d357600080fd5b5051905060006131e48a8584614f20565b90506131f08b8461508f565b6131fa8a84614dc7565b6132048b88614af5565b61320e8a85614c0d565b6132178a6155da565b6132228a828b614fe8565b604080516001600160a01b03808c168252808e1660208301528c1681830152606081018990526080810186905260a0810183905260c0810184905290517f0874b2d545cb271cdbda4e093020c452328b24af12382ed62c4d00f5c26709db9181900360e00190a16016805461ff001916905560016000559a9950505050505050505050565b68327cb2734119d3b7a9601e1b81565b602f6020526000908152604090205481565b60125481565b6000816132de57506000611801565b611d8f83836132ec86613ebc565b614113565b6001600160a01b0383166000908152602d6020908152604080832054602e9092528220548285821161332c576133278683614548565b613336565b6133368287614548565b905060006133488361178e868561443d565b9050868311600061335986896144f0565b90506000826133715761336c82856144f0565b61337b565b61337b8285614548565b905061338b8161178e8c8561443d565b9b9a5050505050505050505050565b60115460ff1681565b600d5481565b6000816133b857506000611801565b611d8f83836132ec86612db8565b6133ce6144d5565b6001600160a01b03919091166000908152601a60205260409020805460ff1916911515919091179055565b6012546001600160a01b0382166000908152602a602052604081205490914291613422916144f0565b111561343057506000611798565b6012546001600160a01b0383166000908152602a6020526040812054909161345d9161178e904290614548565b6001600160a01b0384166000908152602560205260409020549091508061348957600092505050611798565b6001600160a01b0384166000908152601f602052604081205460ff166134b1576013546134b5565b6014545b6001600160a01b0386166000908152602660205260409020549091506134ee90839061178e9086906134e890869061443d565b9061443d565b95945050505050565b60165460ff1681565b60226020526000908152604090205481565b60246020526000908152604090205481565b60085481565b601654610100900460ff1681565b6001600160a01b0381166000908152601f602052604081205460ff161561357857506001600160a01b038116600090815260256020526040902054611798565b6001600160a01b03821660009081526028602052604081205461359c9084906132cf565b6001600160a01b038416600090815260266020908152604080832054602590925290912054919250611794916135d39084906144f0565b90614548565b6001546040805163b1cc53ab60e01b81526001600160a01b038681166004830152858116602483015284151560448301529151600093630100000090049092169163b1cc53ab91606480820192602092909190829003018186803b15801561364057600080fd5b505afa158015613654573d6000803e3d6000fd5b505050506040513d602081101561366a57600080fd5b5051949350505050565b6001600160a01b0381166000908152602d60205260408120548190806136a1576000809250925050613707565b60006136ac85613ebc565b6001600160a01b0386166000908152602e60205260408120549192508282116136de576136d98383614548565b6136e8565b6136e88284614548565b905060006136fa8361178e878561443d565b9390921195509193505050505b915091565b60266020526000908152604090205481565b60135481565b60296020526000908152604090205481565b6001546040805163c7e074c360e01b81526001600160a01b03888116600483015260248201889052604482018790526064820186905284151560848301529151600093630100000090049092169163c7e074c39160a480820192602092909190829003018186803b1580156137aa57600080fd5b505afa1580156137be573d6000803e3d6000fd5b505050506040513d60208110156137d457600080fd5b50519695505050505050565b6137e86144d5565b6137fc6001600160a01b038316848361564f565b505050565b6138096144d5565b600680546001600160a01b0319166001600160a01b0392909216919091179055565b6138336144d5565b601755565b6138406144d5565b61384f6127108211600261458a565b600855565b6001546040805163d54d5a9f60e01b81526001600160a01b03888116600483015287811660248301528681166044830152851515606483015284151560848301528251600094859463010000009091049092169263d54d5a9f9260a4808301939192829003018186803b1580156138ca57600080fd5b505afa1580156138de573d6000803e3d6000fd5b505050506040513d60408110156138f457600080fd5b508051602090910151909890975095505050505050565b6139136144d5565b6001600160a01b0382166000908152602360205260409020548082111561394d57613947836139428484614548565b61508f565b50613960565b6137fc8361395b8385614548565b614dc7565b5050565b602a6020526000908152604090205481565b60105481565b6001546040805163369d949360e21b81526001600160a01b0389811660048301528881166024830152878116604483015286151560648301526084820186905260a482018590529151600093630100000090049092169163da76524c9160c480820192602092909190829003018186803b1580156139f957600080fd5b505afa158015613a0d573d6000803e3d6000fd5b505050506040513d6020811015613a2357600080fd5b5051979650505050505050565b601c6020526000908152604090205460ff1681565b602080526000908152604090205460ff1681565b6000806000613a6b8a8a8a8a886124e8565b90925090506000613a7c8a876144f0565b905060008815613aab5783613a9a57613a958284614548565b613aa4565b613aa482846144f0565b9050613acc565b83613abf57613aba82846144f0565b613ac9565b613ac98284614548565b90505b613ada8161178e8a8561443d565b9c9b505050505050505050505050565b60155481565b60026000541415613b36576040805162461bcd60e51b815260206004820152601f6024820152600080516020615dbd833981519152604482015290519081900360640190fd5b60026000556016546301000000900460ff1615613b6b5733600090815260196020526040902054613b6b9060ff16602261458a565b6016805460ff19169055613b7f848461414d565b6000613b8d8686868661198e565b9050613b97615d6a565b506000818152602b6020908152604091829020825160e08101845281548082526001830154938201939093526002820154938101939093526003810154606084015260048101546080840152600581015460a08401526006015460c0830152613c03901515602361458a565b600080613c14898989896000613854565b91509150613c278260001415602461458a565b8160021415613c5c57613c44898989600087600001518b8f61514d565b50506016805460ff1916600117905550613eaa915050565b6000613c6889836132cf565b6001600160a01b038a166000908152602c6020526040902054909150613c8e90826144f0565b6001600160a01b038a166000818152602c60209081526040918290209390935580519182529181018490528082018390529051600080516020615e888339815191529181900360600190a1613ce78985608001516156a1565b8615613d165760208401518451613d08918b91613d0391614548565b614a76565b613d168961213d8b856132cf565b600087613d2b57613d2689613ebc565b613d34565b613d3489612db8565b90507f2e1f85a64a2f22cf2f0c42584e7c919ed4abe8d53675cff0f62bf1e95a1c676f868c8c8c8c8a600001518b602001518c608001518d60a001518a604051808b81526020018a6001600160a01b03168152602001896001600160a01b03168152602001886001600160a01b0316815260200187151581526020018681526020018581526020018481526020018381526020018281526020019a505050505050505050505060405180910390a187158015613df35750846020015183105b15613e21576020850151600090613e0a9085614548565b9050613e1f8b613e1a8d846132cf565b614af5565b505b87613e3457613e34898660000151615756565b6000868152602b60205260408120818155600181018290556002810182905560038101829055600481018290556005810182905560060155600954613e80908b9061213d9082906132cf565b613e968a613e908c6009546132cf565b89614fe8565b50506016805460ff19166001179055505050505b50506001600055505050565b600e5481565b60048054601654604080516317e1d38560e11b81526001600160a01b03868116958201959095526001602482015260ff80841615156044830152610100909304909216151560648301525160009390921691632fc3a70a91608480820192602092909190829003018186803b158015612e3157600080fd5b601b8181548110613f4157fe5b6000918252602090912001546001600160a01b0316905081565b613f636144d5565b6001600160a01b0381166000908152601c6020526040902054613f8a9060ff16600d61458a565b6001600160a01b038116600090815260226020526040902054601554613faf91614548565b6015556001600160a01b0381166000908152601c60209081526040808320805460ff19908116909155601d835281842084905560228352818420849055601e835281842084905560248352818420849055601f835281842080548216905591805290912080549091169055600754614028906001614548565b60075550565b6140366144d5565b6001600160a01b039091166000908152602f6020526040902055565b60286020526000908152604090205481565b61406c6144d5565b6016805491151563010000000263ff00000019909216919091179055565b60006140946144d5565b6001600160a01b0383166000908152602c6020526040902054806140bc576000915050611801565b6001600160a01b0384166000908152602c6020526040812055611d8f848285614fe8565b601a6020526000908152604090205460ff1681565b6005546001600160a01b031681565b6003546001600160a01b031681565b60008261412257506000611d8f565b6001600160a01b0384166000908152601d60205260409020546134ee8361178e86600a85900a61443d565b6001546040805163fbfded6d60e01b81526001600160a01b03858116600483015284811660248301529151600093630100000090049092169163fbfded6d9160448082019260209290919082900301818787803b1580156141ad57600080fd5b505af11580156141c1573d6000803e3d6000fd5b505050506040513d60208110156141d757600080fd5b50519050806141e65750613960565b6001600160a01b0383166000908152602a602052604090205461423357601254614214906134e84282614496565b6001600160a01b0384166000908152602a602052604090205550613960565b6012546001600160a01b0384166000908152602a6020526040902054429161425b91906144f0565b11156142675750613960565b6000614272846133f9565b6001600160a01b03851660009081526029602052604090205490915061429890826144f0565b6001600160a01b0385166000908152602960205260409020556012546142c2906134e84282614496565b6001600160a01b0385166000818152602a602090815260408083209490945560298152908390205483519283529082015281517fa146fc154e1913322e9817d49f0d5c37466c24326e15de10e739a948be815eab929181900390910190a150505050565b610e1081565b6001546040805163fdaf6ac360e01b81526001600160a01b038881166004830152878116602483015286811660448301528515156064830152608482018590529151600093630100000090049092169163fdaf6ac39160a480820192602092909190829003018186803b1580156137aa57600080fd5b60306020908152600091825260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156144355780601f1061440a57610100808354040283529160200191614435565b820191906000526020600020905b81548152906001019060200180831161441857829003601f168201915b505050505081565b60008261444c57506000611801565b8282028284828161445957fe5b0414611d8f5760405162461bcd60e51b8152600401808060200182810382526021815260200180615e1d6021913960400191505060405180910390fd5b6000611d8f83836040518060400160405280601a815260200179536166654d6174683a206469766973696f6e206279207a65726f60301b8152506157be565b6006546144ee906001600160a01b03163314603561458a565b565b600082820183811015611d8f576040805162461bcd60e51b815260206004820152601b60248201527a536166654d6174683a206164646974696f6e206f766572666c6f7760281b604482015290519081900360640190fd5b6000611d8f83836040518060400160405280601e81526020017f536166654d6174683a207375627472616374696f6e206f766572666c6f770000815250615860565b6000818152603060205260409020826137fc5760405162461bcd60e51b81526020600482019081528254600260001961010060018416150201909116046024830181905290918291604490910190849080156146275780601f106145fc57610100808354040283529160200191614627565b820191906000526020600020905b81548152906001019060200180831161460a57829003601f168201915b50509250505060405180910390fd5b601754614642576144ee565b6144ee6017543a1115603761458a565b336001600160a01b0382161415614668576146b2565b6003546001600160a01b0316331415614680576146b2565b6001600160a01b03811660009081526018602090815260408083203384529091529020546146b29060ff16602961458a565b50565b801561472e576146da826001600160a01b0316846001600160a01b031614602a61458a565b6001600160a01b0383166000908152601c60205260409020546147019060ff16602b61458a565b6001600160a01b0383166000908152601f60205260409020546147299060ff1615602c61458a565b6137fc565b6001600160a01b0383166000908152601c60205260409020546147559060ff16602d61458a565b6001600160a01b0383166000908152601f602052604090205461477c9060ff16602e61458a565b6001600160a01b0382166000908152601f60205260409020546147a49060ff1615602f61458a565b6001600160a01b03821660009081526020805260409020546137fc9060ff16603061458a565b6000806147da898989898961432c565b905060006147ec8a8a8a8a898961397c565b90506147f882826144f0565b915060006148068a846132cf565b6001600160a01b038b166000908152602c602052604090205490915061482c90826144f0565b6001600160a01b038b166000818152602c60209081526040918290209390935580519182529181018590528082018390529051600080516020615e888339815191529181900360600190a150909998505050505050505050565b6001600160a01b03811660008181526021602090815260408083205481516370a0823160e01b8152306004820152915193949093859391926370a08231926024808301939192829003018186803b1580156148e057600080fd5b505afa1580156148f4573d6000803e3d6000fd5b505050506040513d602081101561490a57600080fd5b50516001600160a01b03851660009081526021602052604090208190559050611adb8183614548565b81614949576149448115602761458a565b613960565b61396081831015602861458a565b6001600160a01b03821660009081526026602052604090205461497a90826144f0565b6001600160a01b0383166000908152602660208181526040808420859055602582529092205491526149af911115603461458a565b604080516001600160a01b03841681526020810183905281517faa5649d82f5462be9d19b0f2b31a59b2259950a6076550bac9f3a1c07db9f66d929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614a1a90826144f0565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517fd9d4761f75e0d0103b5cbeab941eeb443d7a56a35b5baf2a0787c03f03f4e474929181900390910190a15050565b6001600160a01b038216600090815260286020526040902054614a999082614548565b6001600160a01b03831660008181526028602090815260409182902093909355805191825291810183905281517f34e07158b9db50df5613e591c44ea2ebc82834eff4a4dc3a46e000e608261d68929181900390910190a15050565b6001600160a01b038216600090815260256020526040902054614b1890826144f0565b6001600160a01b03831660008181526025602090815260408083209490945583516370a0823160e01b8152306004820152935191936370a082319260248083019392829003018186803b158015614b6e57600080fd5b505afa158015614b82573d6000803e3d6000fd5b505050506040513d6020811015614b9857600080fd5b50516001600160a01b038416600090815260256020526040902054909150614bc490821015603161458a565b604080516001600160a01b03851681526020810184905281517f976177fbe09a15e5e43f848844963a42b41ef919ef17ff21a17a5421de8f4737929181900390910190a1505050565b604080518082018252601a81527915985d5b1d0e881c1bdbdb105b5bdd5b9d08195e18d95959195960321b6020808301919091526001600160a01b038516600090815260259091529190912054614c65918390615860565b6001600160a01b03831660009081526025602090815260408083208490556026909152902054614c98911015603261458a565b604080516001600160a01b03841681526020810183905281517f112726233fbeaeed0f5b1dba5cb0b2b81883dee49fb35ff99fd98ed9f6d31eb0929181900390910190a15050565b6001600160a01b0382166000908152602d6020526040902054614d0390826144f0565b6001600160a01b0383166000908152602d6020908152604080832093909355602f9052205480156137fc576001600160a01b0383166000908152602d60205260409020548110156137fc576040805162461bcd60e51b815260206004820152601a60248201527915985d5b1d0e881b585e081cda1bdc9d1cc8195e18d95959195960321b604482015290519081900360640190fd5b60165462010000900460ff16156144ee57336000908152601a60205260409020546144ee9060ff16603661458a565b6001600160a01b038216600090815260236020526040902054818111614e30576001600160a01b038316600081815260236020908152604080832092909255815192835282018390528051600080516020615e3e8339815191529281900390910190a150613960565b614e3a8183614548565b6001600160a01b0384166000818152602360209081526040918290209390935580519182529181018490528151600080516020615e3e833981519152929181900390910190a1505050565b6000816001600160a01b03166370a08231306040518263ffffffff1660e01b815260040180826001600160a01b0316815260200191505060206040518083038186803b158015614ed457600080fd5b505afa158015614ee8573d6000803e3d6000fd5b505050506040513d6020811015614efe57600080fd5b50516001600160a01b0390921660009081526021602052604090209190915550565b600080614f3d61271061178e614f368287614548565b879061443d565b90506000614f4b8583614548565b6001600160a01b0387166000908152602c6020526040902054909150614f7190826144f0565b6001600160a01b0387166000908152602c60205260409020557f47cd9dda0e50ce30bcaaacd0488452b596221c07ac402a581cfae4d3933cac2b86614fb681846117b4565b604080516001600160a01b0390931683526020830191909152818101849052519081900360600190a150949350505050565b614ffc6001600160a01b038416828461564f565b604080516370a0823160e01b815230600482015290516001600160a01b038516916370a08231916024808301926020929190829003018186803b15801561504257600080fd5b505afa158015615056573d6000803e3d6000fd5b505050506040513d602081101561506c57600080fd5b50516001600160a01b039093166000908152602160205260409020929092555050565b6001600160a01b0382166000908152602360205260409020546150b290826144f0565b6001600160a01b0383166000908152602360209081526040808320939093556024905220548015615104576001600160a01b03831660009081526023602052604090205461510490821015603361458a565b604080516001600160a01b03851681526020810184905281517f64243679a443432e2293343b77d411ff6144370404618f00ca0d2025d9ca9882929181900390910190a1505050565b600154604080516381d11a2360e01b81526001600160a01b038a8116600483015289811660248301528881166044830152606482018890526084820187905285151560a483015284811660c4830152915160009363010000009004909216916381d11a239160e4808201928692909190829003018186803b1580156151d157600080fd5b505afa1580156151e5573d6000803e3d6000fd5b505050506151f3878761414d565b60006152018989898761198e565b6000818152602b60205260409020805491925090615222901515601f61458a565b6152348682600001541015602061458a565b6152468782600101541015602161458a565b6001810154815460048301546000916152639161178e908b61443d565b60048401549091506152759082614548565b60048401556152848b826156a1565b506000806152968d8d8d8d8d8d6158ba565b855491935091508914615416576152ae8c8c8a6135d9565b600385015583546152bf908a614548565b80855560018501546152d19190614933565b6152df8d8d8d8b6001613854565b5050871561530c576153028c61211686600101548661454890919063ffffffff16565b61530c8c8a614a76565b6000886153215761531c8c613ebc565b61532a565b61532a8c612db8565b9050600080516020615dfd833981519152868f8f8f8f8f8f8861534d8c8c614548565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c083015260e0820183905251600080516020615ddd833981519152918190036101000190a150615578565b8715615430576154268c846149f7565b6154308c8a614a76565b600088615445576154408c613ebc565b61544e565b61544e8c612db8565b9050600080516020615dfd833981519152868f8f8f8f8f8f886154718c8c614548565b60408051998a526001600160a01b0398891660208b015296881689880152949096166060880152608087019290925260a0860152151560c085015260e084019290925261010083019190915251908190036101200190a18454600186015460028701546003880154600489015460058a0154604080518d81526020810197909752868101959095526060860193909352608085019190915260a084015260c0830152517f73af1d417d82c240fdb6d319b34ad884487c6bf2845d98980cc52ad9171cb4559181900360e00190a1506000858152602b602052604081208181556001810182905560028101829055600381018290556004810182905560058101829055600601555b87615587576155878b8a615756565b81156155c75787156155a1576155a18c61213d8e856132cf565b60006155ad8d836132cf565b90506155ba8d828a614fe8565b95506124a4945050505050565b5060009c9b505050505050505050505050565b6001600160a01b03811660009081526027602090815260408083205460259092529091205410156146b2576040805162461bcd60e51b815260206004820152601a6024820152792b30bab63a1d103837b7b620b6b7bab73a101e10313ab33332b960311b604482015290519081900360640190fd5b604080516001600160a01b038416602482015260448082018490528251808303909101815260649091019091526020810180516001600160e01b031663a9059cbb60e01b1790526137fc908490615ab4565b604080518082018252601b81527a5661756c743a20696e73756666696369656e74207265736572766560281b6020808301919091526001600160a01b0385166000908152602690915291909120546156fa918390615860565b6001600160a01b03831660008181526026602090815260409182902093909355805191825291810183905281517f533cb5ed32be6a90284e96b5747a1bfc2d38fdb5768a6b5f67ff7d62144ed67b929181900390910190a15050565b6001600160a01b0382166000908152602d60205260409020548082111561579657506001600160a01b0382166000908152602d6020526040812055613960565b6157a08183614548565b6001600160a01b0384166000908152602d6020526040902055505050565b6000818361584a5760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b8381101561580f5781810151838201526020016157f7565b50505050905090810190601f16801561583c5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b50600083858161585657fe5b0495945050505050565b600081848411156158b25760405162461bcd60e51b815260206004820181815283516024840152835190928392604490910191908501908083836000831561580f5781810151838201526020016157f7565b505050900390565b60008060006158cb8989898761198e565b6000818152602b602052604081208054600382015493945090926158f8918d918d918d918b918d916147ca565b90506000806000806159198d876000015488600201548d8a600601546124e8565b875491955085935091506159319061178e8d8461443d565b9250505060008280156159445750600082115b1561597457506005840180548201905580886159745760006159668e846132cf565b90506159728e82614c0d565b505b821580156159825750600082115b156159c65760018501546159969083614548565b6001860155886159ba5760006159ac8e846132cf565b90506159b88e82614af5565b505b60058501805483900390555b8a156159ee576159d6818c6144f0565b60018601549091506159e8908c614548565b60018601555b84548a1415615a11576001850154615a079082906144f0565b6000600187015590505b8084811115615a2b57615a248286614548565b9050615a5f565b6001860154615a3a9086614548565b60018701558915615a5f576000615a518f876132cf565b9050615a5d8f82614c0d565b505b60408051888152851515602082015280820185905290517f3ff41bdde87755b687ae83d0221a232b6be51a803330ed9661c1b5d0105e0d8a9181900360600190a1909e909d509b505050505050505050505050565b6060615b09826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b0316615b659092919063ffffffff16565b8051909150156137fc57808060200190516020811015615b2857600080fd5b50516137fc5760405162461bcd60e51b815260040180806020018281038252602a815260200180615e5e602a913960400191505060405180910390fd5b6060611adb848460008585615b7985615c80565b615bca576040805162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604482015290519081900360640190fd5b60006060866001600160a01b031685876040518082805190602001908083835b60208310615c095780518252601f199092019160209182019101615bea565b6001836020036101000a03801982511681845116808217855250505050505090500191505060006040518083038185875af1925050503d8060008114615c6b576040519150601f19603f3d011682016040523d82523d6000602084013e615c70565b606091505b50915091506124a4828286615c86565b3b151590565b60608315615c95575081611d8f565b825115615ca55782518084602001fd5b60405162461bcd60e51b815260206004820181815284516024840152845185939192839260440191908501908083836000831561580f5781810151838201526020016157f7565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f10615d2d5782800160ff19823516178555615d5a565b82800160010185558215615d5a579182015b82811115615d5a578235825591602001919060010190615d3f565b50615d66929150615da7565b5090565b6040518060e00160405280600081526020016000815260200160008152602001600081526020016000815260200160008152602001600081525090565b5b80821115615d665760008155600101615da856fe5265656e7472616e637947756172643a207265656e7472616e742063616c6c0020853733b590dce729d9f4628682ebd9a34d2354e72679e66f43a008fc03b77393d75d64d1f84fc6f430a64fc578bdd4c1e090e90ea2d51773e626d19de56d30536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f77e1e812596aac93a06ecc4ca627014d18e30f5c33b825160cc9d5c0ba61e452275361666545524332303a204552433230206f7065726174696f6e20646964206e6f7420737563636565645d0c0019d3d45fadeb74eff9d2c9924d146d000ac6bcf3c28bf0ac3c9baa011aa2646970667358221220278b7f556d929c862af1a8bac6e804ea6cff611f6b4579f83344dae30cd2030664736f6c634300060c0033
Loading...
Loading
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 34 Chains
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.