Overview
ETH Balance
0 ETH
ETH Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 262,195 transactions (+25 Pending)
Latest 25 internal transactions (View All)
Parent Transaction Hash | Block | From | To | |||
---|---|---|---|---|---|---|
9328304 | 30 secs ago | 0.00135796 ETH | ||||
9328304 | 30 secs ago | 0.00135796 ETH | ||||
9328173 | 7 mins ago | 0.00011457 ETH | ||||
9328173 | 7 mins ago | 0.02291478 ETH | ||||
9328161 | 7 mins ago | 0.02291478 ETH | ||||
9328150 | 8 mins ago | 0.023 ETH | ||||
9328115 | 9 mins ago | 0.00013239 ETH | ||||
9328115 | 9 mins ago | 0.02647803 ETH | ||||
9328010 | 15 mins ago | 0 ETH | ||||
9328010 | 15 mins ago | 0.00009999 ETH | ||||
9327994 | 16 mins ago | 0.00005922 ETH | ||||
9327994 | 16 mins ago | 0.02300589 ETH | ||||
9327917 | 19 mins ago | 0.00481969 ETH | ||||
9327917 | 19 mins ago | 0.00481969 ETH | ||||
9327893 | 21 mins ago | 0.0366996 ETH | ||||
9327893 | 21 mins ago | 0.0366996 ETH | ||||
9327872 | 22 mins ago | 0.01314573 ETH | ||||
9327872 | 22 mins ago | 0.01314573 ETH | ||||
9327871 | 22 mins ago | 0 ETH | ||||
9327871 | 22 mins ago | 0.03695849 ETH | ||||
9327870 | 22 mins ago | 0.0533995 ETH | ||||
9327870 | 22 mins ago | 0.0533995 ETH | ||||
9327864 | 22 mins ago | 0.00414858 ETH | ||||
9327864 | 22 mins ago | 0.00414858 ETH | ||||
9327792 | 26 mins ago | 0.03847073 ETH |
Loading...
Loading
Contract Name:
UniversalSwapRouter
Compiler Version
v0.8.4+commit.c7e474f2
Contract Source Code (Solidity)
/** *Submitted for verification at scrollscan.com on 2024-06-28 */ // Sources flattened with hardhat v2.16.1 https://hardhat.org // File @openzeppelin/contracts/utils/[email protected] // SPDX-License-Identifier: BUSL-1.1 // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } } // File @openzeppelin/contracts/access/[email protected] // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol) pragma solidity ^0.8.0; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby disabling any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } } // File contracts/libraries/BytesLib.sol /* * @title Solidity Bytes Arrays Utils * @author Gonçalo Sá <[email protected]> * * @dev Bytes tightly packed arrays utility library for ethereum contracts written in Solidity. * The library lets you concatenate, slice and type cast bytes arrays both in memory and storage. */ pragma solidity ^0.8.4; library BytesLib { function slice( bytes memory _bytes, uint256 _start, uint256 _length ) internal pure returns (bytes memory) { require(_length + 31 >= _length, 'slice_overflow'); require(_start + _length >= _start, 'slice_overflow'); require(_bytes.length >= _start + _length, 'slice_outOfBounds'); bytes memory tempBytes; assembly { switch iszero(_length) case 0 { // Get a location of some free memory and store it in tempBytes as // Solidity does for memory variables. tempBytes := mload(0x40) // The first word of the slice result is potentially a partial // word read from the original array. To read it, we calculate // the length of that partial word and start copying that many // bytes into the array. The first word we copy will start with // data we don't care about, but the last `lengthmod` bytes will // land at the beginning of the contents of the new array. When // we're done copying, we overwrite the full first word with // the actual length of the slice. let lengthmod := and(_length, 31) // The multiplication in the next line is necessary // because when slicing multiples of 32 bytes (lengthmod == 0) // the following copy loop was copying the origin's length // and then ending prematurely not copying everything it should. let mc := add(add(tempBytes, lengthmod), mul(0x20, iszero(lengthmod))) let end := add(mc, _length) for { // The multiplication in the next line has the same exact purpose // as the one above. let cc := add(add(add(_bytes, lengthmod), mul(0x20, iszero(lengthmod))), _start) } lt(mc, end) { mc := add(mc, 0x20) cc := add(cc, 0x20) } { mstore(mc, mload(cc)) } mstore(tempBytes, _length) //update free-memory pointer //allocating the array padded to 32 bytes like the compiler does now mstore(0x40, and(add(mc, 31), not(31))) } //if we want a zero-length slice let's just return a zero-length array default { tempBytes := mload(0x40) //zero out the 32 bytes slice we are about to return //we need to do it because Solidity does not garbage collect mstore(tempBytes, 0) mstore(0x40, add(tempBytes, 0x20)) } } return tempBytes; } function toAddress(bytes memory _bytes, uint256 _start) internal pure returns (address) { require(_start + 20 >= _start, 'toAddress_overflow'); require(_bytes.length >= _start + 20, 'toAddress_outOfBounds'); address tempAddress; assembly { tempAddress := div(mload(add(add(_bytes, 0x20), _start)), 0x1000000000000000000000000) } return tempAddress; } function toUint24(bytes memory _bytes, uint256 _start) internal pure returns (uint24) { require(_start + 3 >= _start, 'toUint24_overflow'); require(_bytes.length >= _start + 3, 'toUint24_outOfBounds'); uint24 tempUint; assembly { tempUint := mload(add(add(_bytes, 0x3), _start)) } return tempUint; } } // File contracts/libraries/Path.sol pragma solidity ^0.8.4; /// @title Functions for manipulating path data for multihop swaps library Path { using BytesLib for bytes; /// @dev The length of the bytes encoded address uint256 private constant ADDR_SIZE = 20; /// @dev The length of the bytes encoded fee uint256 private constant FEE_SIZE = 3; /// @dev The offset of a single token address and pool fee uint256 private constant NEXT_OFFSET = ADDR_SIZE + FEE_SIZE; /// @dev The offset of an encoded pool key uint256 private constant POP_OFFSET = NEXT_OFFSET + ADDR_SIZE; /// @dev The minimum length of an encoding that contains 2 or more pools uint256 private constant MULTIPLE_POOLS_MIN_LENGTH = POP_OFFSET + NEXT_OFFSET; /// @notice Returns true iff the path contains two or more pools /// @param path The encoded swap path /// @return True if path contains two or more pools, otherwise false function hasMultiplePools(bytes memory path) internal pure returns (bool) { return path.length >= MULTIPLE_POOLS_MIN_LENGTH; } /// @notice Returns the number of pools in the path /// @param path The encoded swap path /// @return The number of pools in the path function numPools(bytes memory path) internal pure returns (uint256) { // Ignore the first token address. From then on every fee and token offset indicates a pool. return ((path.length - ADDR_SIZE) / NEXT_OFFSET); } /// @notice Decodes the first pool in path /// @param path The bytes encoded swap path /// @return tokenA The first token of the given pool /// @return tokenB The second token of the given pool /// @return fee The fee level of the pool function decodeFirstPool(bytes memory path) internal pure returns ( address tokenA, address tokenB, uint24 fee ) { tokenA = path.toAddress(0); fee = path.toUint24(ADDR_SIZE); tokenB = path.toAddress(NEXT_OFFSET); } /// @notice Gets the segment corresponding to the first pool in the path /// @param path The bytes encoded swap path /// @return The segment containing all data necessary to target the first pool in the path function getFirstPool(bytes memory path) internal pure returns (bytes memory) { return path.slice(0, POP_OFFSET); } /// @notice Skips a token + fee element from the buffer and returns the remainder /// @param path The swap path /// @return The remaining token + fee elements in the path function skipToken(bytes memory path) internal pure returns (bytes memory) { return path.slice(NEXT_OFFSET, path.length - NEXT_OFFSET); } } // File @openzeppelin/contracts/token/ERC20/[email protected] // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @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); /** * @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 `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, 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 `from` to `to` 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 from, address to, uint256 amount) external returns (bool); } // File contracts/core/interfaces/IiZiSwapCallback.sol pragma solidity ^0.8.4; interface IiZiSwapMintCallback { /// @notice Called to msg.sender in iZiSwapPool#mint call /// @param x Amount of tokenX need to pay from miner /// @param y Amount of tokenY need to pay from miner /// @param data Any data passed through by the msg.sender via the iZiSwapPool#mint call function mintDepositCallback( uint256 x, uint256 y, bytes calldata data ) external; } interface IiZiSwapCallback { /// @notice Called to msg.sender in iZiSwapPool#swapY2X(DesireX) call /// @param x Amount of tokenX trader will acquire /// @param y Amount of tokenY trader will pay /// @param data Any dadta passed though by the msg.sender via the iZiSwapPool#swapY2X(DesireX) call function swapY2XCallback( uint256 x, uint256 y, bytes calldata data ) external; /// @notice Called to msg.sender in iZiSwapPool#swapX2Y(DesireY) call /// @param x Amount of tokenX trader will pay /// @param y Amount of tokenY trader will require /// @param data Any dadta passed though by the msg.sender via the iZiSwapPool#swapX2Y(DesireY) call function swapX2YCallback( uint256 x, uint256 y, bytes calldata data ) external; } interface IiZiSwapAddLimOrderCallback { /// @notice Called to msg.sender in iZiSwapPool#addLimOrderWithX(Y) call /// @param x Amount of tokenX seller will pay /// @param y Amount of tokenY seller will pay /// @param data Any dadta passed though by the msg.sender via the iZiSwapPool#addLimOrderWithX(Y) call function payCallback( uint256 x, uint256 y, bytes calldata data ) external; } // File contracts/core/interfaces/IiZiSwapFactory.sol pragma solidity ^0.8.4; interface IiZiSwapFactory { /// @notice emit when successfuly create a new pool (calling iZiSwapFactory#newPool) /// @param tokenX address of erc-20 tokenX /// @param tokenY address of erc-20 tokenY /// @param fee fee amount of swap (3000 means 0.3%) /// @param pointDelta minimum number of distance between initialized or limitorder points /// @param pool address of swap pool event NewPool( address indexed tokenX, address indexed tokenY, uint24 indexed fee, uint24 pointDelta, address pool ); /// @notice module to support swap from tokenX to tokenY /// @return swapX2YModule address function swapX2YModule() external returns (address); /// @notice module to support swap from tokenY to tokenX /// @return swapY2XModule address function swapY2XModule() external returns (address); /// @notice module to support mint/burn/collect function of pool /// @return liquidityModule address function liquidityModule() external returns (address); /// @notice address of module for user to manage limit orders /// @return limitOrderModule address function limitOrderModule() external returns (address); /// @notice address of module for flash loan /// @return flashModule address function flashModule() external returns (address); /// @notice default fee rate from miner's fee gain /// @return defaultFeeChargePercent default fee rate * 100 function defaultFeeChargePercent() external returns (uint24); /// @notice Enables a fee amount with the given pointDelta /// @dev Fee amounts may never be removed once enabled /// @param fee fee amount (3000 means 0.3%) /// @param pointDelta The spacing between points to be enforced for all pools created with the given fee amount function enableFeeAmount(uint24 fee, uint24 pointDelta) external; /// @notice Create a new pool which not exists. /// @param tokenX address of tokenX /// @param tokenY address of tokenY /// @param fee fee amount /// @param currentPoint initial point (log 1.0001 of price) /// @return address of newly created pool function newPool( address tokenX, address tokenY, uint24 fee, int24 currentPoint ) external returns (address); /// @notice Charge receiver of all pools. /// @return address of charge receiver function chargeReceiver() external view returns(address); /// @notice Get pool of (tokenX, tokenY, fee), address(0) for not exists. /// @param tokenX address of tokenX /// @param tokenY address of tokenY /// @param fee fee amount /// @return address of pool function pool( address tokenX, address tokenY, uint24 fee ) external view returns(address); /// @notice Get point delta of a given fee amount. /// @param fee fee amount /// @return pointDelta the point delta function fee2pointDelta(uint24 fee) external view returns (int24 pointDelta); /// @notice Change charge receiver, only owner of factory can call. /// @param _chargeReceiver address of new receiver function modifyChargeReceiver(address _chargeReceiver) external; /// @notice Change defaultFeeChargePercent /// @param _defaultFeeChargePercent new charge percent function modifyDefaultFeeChargePercent(uint24 _defaultFeeChargePercent) external; function deployPoolParams() external view returns( address tokenX, address tokenY, uint24 fee, int24 currentPoint, int24 pointDelta, uint24 feeChargePercent ); } // File contracts/core/interfaces/IiZiSwapPool.sol pragma solidity ^0.8.4; interface IiZiSwapPool { /// @notice Emitted when miner successfully add liquidity (mint). /// @param sender the address that minted the liquidity /// @param owner the owner who will benefit from this liquidity /// @param leftPoint left endpoint of the liquidity /// @param rightPoint right endpoint of the liquidity /// @param liquidity the amount of liquidity minted to the range [leftPoint, rightPoint) /// @param amountX amount of tokenX deposit /// @param amountY amount of tokenY deposit event Mint( address sender, address indexed owner, int24 indexed leftPoint, int24 indexed rightPoint, uint128 liquidity, uint256 amountX, uint256 amountY ); /// @notice Emitted when miner successfully decrease liquidity (withdraw). /// @param owner owner address of liquidity /// @param leftPoint left endpoint of liquidity /// @param rightPoint right endpoint of liquidity /// @param liquidity amount of liquidity decreased /// @param amountX amount of tokenX withdrawed /// @param amountY amount of tokenY withdrawed event Burn( address indexed owner, int24 indexed leftPoint, int24 indexed rightPoint, uint128 liquidity, uint256 amountX, uint256 amountY ); /// @notice Emitted when fees and withdrawed liquidity are collected /// @param owner The owner of the Liquidity /// @param recipient recipient of those token /// @param leftPoint The left point of the liquidity /// @param rightPoint The right point of the liquidity /// @param amountX The amount of tokenX (fees and withdrawed tokenX from liquidity) /// @param amountY The amount of tokenY (fees and withdrawed tokenY from liquidity) event CollectLiquidity( address indexed owner, address recipient, int24 indexed leftPoint, int24 indexed rightPoint, uint256 amountX, uint256 amountY ); /// @notice Emitted when a trader successfully exchange. /// @param tokenX tokenX of pool /// @param tokenY tokenY of pool /// @param fee fee amount of pool /// @param sellXEarnY true for selling tokenX, false for buying tokenX /// @param amountX amount of tokenX in this exchange /// @param amountY amount of tokenY in this exchange event Swap( address indexed tokenX, address indexed tokenY, uint24 indexed fee, bool sellXEarnY, uint256 amountX, uint256 amountY ); /// @notice Emitted by the pool for any flashes of tokenX/tokenY. /// @param sender the address that initiated the swap call, and that received the callback /// @param recipient the address that received the tokens from flash /// @param amountX the amount of tokenX that was flashed /// @param amountY the amount of tokenY that was flashed /// @param paidX the amount of tokenX paid for the flash, which can exceed the amountX plus the fee /// @param paidY the amount of tokenY paid for the flash, which can exceed the amountY plus the fee event Flash( address indexed sender, address indexed recipient, uint256 amountX, uint256 amountY, uint256 paidX, uint256 paidY ); /// @notice Emitted when a seller successfully add a limit order. /// @param owner owner of limit order /// @param addAmount amount of token to sell the seller added /// @param acquireAmount amount of earn-token acquired, if there exists some opposite order before /// @param point point of limit order /// @param claimSold claimed sold sell-token, if this owner has order with same direction on this point before /// @param claimEarn claimed earned earn-token, if this owner has order with same direction on this point before /// @param sellXEarnY direction of limit order, etc. sell tokenX or sell tokenY event AddLimitOrder( address indexed owner, uint128 addAmount, uint128 acquireAmount, int24 indexed point, uint128 claimSold, uint128 claimEarn, bool sellXEarnY ); /// @notice Emitted when a seller successfully decrease a limit order. /// @param owner owner of limit order /// @param decreaseAmount amount of token to sell the seller decreased /// @param point point of limit order /// @param claimSold claimed sold sell-token /// @param claimEarn claimed earned earn-token /// @param sellXEarnY direction of limit order, etc. sell tokenX or sell tokenY event DecLimitOrder( address indexed owner, uint128 decreaseAmount, int24 indexed point, uint128 claimSold, uint128 claimEarn, bool sellXEarnY ); /// @notice Emitted when collect from a limit order /// @param owner The owner of the Liquidity /// @param recipient recipient of those token /// @param point The point of the limit order /// @param collectDec The amount of decreased sell token collected /// @param collectEarn The amount of earn token collected /// @param sellXEarnY direction of limit order, etc. sell tokenX or sell tokenY event CollectLimitOrder( address indexed owner, address recipient, int24 indexed point, uint128 collectDec, uint128 collectEarn, bool sellXEarnY ); /// @notice Returns the information about a liquidity by the liquidity's key. /// @param key the liquidity's key is a hash of a preimage composed by the miner(owner), pointLeft and pointRight /// @return liquidity the amount of liquidity, /// @return lastFeeScaleX_128 fee growth of tokenX inside the range as of the last mint/burn/collect, /// @return lastFeeScaleY_128 fee growth of tokenY inside the range as of the last mint/burn/collect, /// @return tokenOwedX the computed amount of tokenX miner can collect as of the last mint/burn/collect, /// @return tokenOwedY the computed amount of tokenY miner can collect as of the last mint/burn/collect function liquidity(bytes32 key) external view returns ( uint128 liquidity, uint256 lastFeeScaleX_128, uint256 lastFeeScaleY_128, uint256 tokenOwedX, uint256 tokenOwedY ); /// @notice Returns the information about a user's limit order (sell tokenY and earn tokenX). /// @param key the limit order's key is a hash of a preimage composed by the seller, point /// @return lastAccEarn total amount of tokenX earned by all users at this point as of the last add/dec/collect /// @return sellingRemain amount of tokenY not selled in this limit order /// @return sellingDec amount of tokenY decreased by seller from this limit order /// @return earn amount of unlegacy earned tokenX in this limit order not assigned /// @return legacyEarn amount of legacy earned tokenX in this limit order not assgined /// @return earnAssign assigned amount of tokenX earned (both legacy and unlegacy) in this limit order function userEarnX(bytes32 key) external view returns ( uint256 lastAccEarn, uint128 sellingRemain, uint128 sellingDec, uint128 earn, uint128 legacyEarn, uint128 earnAssign ); /// @notice Returns the information about a user's limit order (sell tokenX and earn tokenY). /// @param key the limit order's key is a hash of a preimage composed by the seller, point /// @return lastAccEarn total amount of tokenY earned by all users at this point as of the last add/dec/collect /// @return sellingRemain amount of tokenX not selled in this limit order /// @return sellingDec amount of tokenX decreased by seller from this limit order /// @return earn amount of unlegacy earned tokenY in this limit order not assigned /// @return legacyEarn amount of legacy earned tokenY in this limit order not assgined /// @return earnAssign assigned amount of tokenY earned (both legacy and unlegacy) in this limit order function userEarnY(bytes32 key) external view returns ( uint256 lastAccEarn, uint128 sellingRemain, uint128 sellingDec, uint128 earn, uint128 legacyEarn, uint128 earnAssign ); /// @notice Mark a given amount of tokenY in a limitorder(sellx and earn y) as assigned. /// @param point point (log Price) of seller's limit order,be sure to be times of pointDelta /// @param assignY max amount of tokenY to mark assigned /// @param fromLegacy true for assigning earned token from legacyEarnY /// @return actualAssignY actual amount of tokenY marked function assignLimOrderEarnY( int24 point, uint128 assignY, bool fromLegacy ) external returns(uint128 actualAssignY); /// @notice Mark a given amount of tokenX in a limitorder(selly and earn x) as assigned. /// @param point point (log Price) of seller's limit order,be sure to be times of pointDelta /// @param assignX max amount of tokenX to mark assigned /// @param fromLegacy true for assigning earned token from legacyEarnX /// @return actualAssignX actual amount of tokenX marked function assignLimOrderEarnX( int24 point, uint128 assignX, bool fromLegacy ) external returns(uint128 actualAssignX); /// @notice Decrease limitorder of selling X. /// @param point point of seller's limit order, be sure to be times of pointDelta /// @param deltaX max amount of tokenX seller wants to decrease /// @return actualDeltaX actual amount of tokenX decreased /// @return legacyAccEarn legacyAccEarnY of pointOrder at point when calling this interface function decLimOrderWithX( int24 point, uint128 deltaX ) external returns (uint128 actualDeltaX, uint256 legacyAccEarn); /// @notice Decrease limitorder of selling Y. /// @param point point of seller's limit order, be sure to be times of pointDelta /// @param deltaY max amount of tokenY seller wants to decrease /// @return actualDeltaY actual amount of tokenY decreased /// @return legacyAccEarn legacyAccEarnX of pointOrder at point when calling this interface function decLimOrderWithY( int24 point, uint128 deltaY ) external returns (uint128 actualDeltaY, uint256 legacyAccEarn); /// @notice Add a limit order (selling x) in the pool. /// @param recipient owner of the limit order /// @param point point of the order, be sure to be times of pointDelta /// @param amountX amount of tokenX to sell /// @param data any data that should be passed through to the callback /// @return orderX actual added amount of tokenX /// @return acquireY amount of tokenY acquired if there is a limit order to sell y before adding function addLimOrderWithX( address recipient, int24 point, uint128 amountX, bytes calldata data ) external returns (uint128 orderX, uint128 acquireY); /// @notice Add a limit order (selling y) in the pool. /// @param recipient owner of the limit order /// @param point point of the order, be sure to be times of pointDelta /// @param amountY amount of tokenY to sell /// @param data any data that should be passed through to the callback /// @return orderY actual added amount of tokenY /// @return acquireX amount of tokenX acquired if there exists a limit order to sell x before adding function addLimOrderWithY( address recipient, int24 point, uint128 amountY, bytes calldata data ) external returns (uint128 orderY, uint128 acquireX); /// @notice Collect earned or decreased token from limit order. /// @param recipient address to benefit /// @param point point of limit order, be sure to be times of pointDelta /// @param collectDec max amount of decreased selling token to collect /// @param collectEarn max amount of earned token to collect /// @param isEarnY direction of this limit order, true for sell y, false for sell x /// @return actualCollectDec actual amount of decresed selling token collected /// @return actualCollectEarn actual amount of earned token collected function collectLimOrder( address recipient, int24 point, uint128 collectDec, uint128 collectEarn, bool isEarnY ) external returns(uint128 actualCollectDec, uint128 actualCollectEarn); /// @notice Add liquidity to the pool. /// @param recipient newly created liquidity will belong to this address /// @param leftPt left endpoint of the liquidity, be sure to be times of pointDelta /// @param rightPt right endpoint of the liquidity, be sure to be times of pointDelta /// @param liquidDelta amount of liquidity to add /// @param data any data that should be passed through to the callback /// @return amountX The amount of tokenX that was paid for the liquidity. Matches the value in the callback /// @return amountY The amount of tokenY that was paid for the liquidity. Matches the value in the callback function mint( address recipient, int24 leftPt, int24 rightPt, uint128 liquidDelta, bytes calldata data ) external returns (uint256 amountX, uint256 amountY); /// @notice Decrease a given amount of liquidity from msg.sender's liquidities. /// @param leftPt left endpoint of the liquidity /// @param rightPt right endpoint of the liquidity /// @param liquidDelta amount of liquidity to burn /// @return amountX The amount of tokenX should be refund after burn /// @return amountY The amount of tokenY should be refund after burn function burn( int24 leftPt, int24 rightPt, uint128 liquidDelta ) external returns (uint256 amountX, uint256 amountY); /// @notice Collect tokens (fee or refunded after burn) from a liquidity. /// @param recipient the address which should receive the collected tokens /// @param leftPt left endpoint of the liquidity /// @param rightPt right endpoint of the liquidity /// @param amountXLim max amount of tokenX the owner wants to collect /// @param amountYLim max amount of tokenY the owner wants to collect /// @return actualAmountX the amount tokenX collected /// @return actualAmountY the amount tokenY collected function collect( address recipient, int24 leftPt, int24 rightPt, uint256 amountXLim, uint256 amountYLim ) external returns (uint256 actualAmountX, uint256 actualAmountY); /// @notice Swap tokenY for tokenX, given max amount of tokenY user willing to pay. /// @param recipient the address to receive tokenX /// @param amount the max amount of tokenY user willing to pay /// @param highPt the highest point(price) of x/y during swap /// @param data any data to be passed through to the callback /// @return amountX amount of tokenX payed /// @return amountY amount of tokenY acquired function swapY2X( address recipient, uint128 amount, int24 highPt, bytes calldata data ) external returns (uint256 amountX, uint256 amountY); /// @notice Swap tokenY for tokenX, given amount of tokenX user desires. /// @param recipient the address to receive tokenX /// @param desireX the amount of tokenX user desires /// @param highPt the highest point(price) of x/y during swap /// @param data any data to be passed through to the callback /// @return amountX amount of tokenX payed /// @return amountY amount of tokenY acquired function swapY2XDesireX( address recipient, uint128 desireX, int24 highPt, bytes calldata data ) external returns (uint256 amountX, uint256 amountY); /// @notice Swap tokenX for tokenY, given max amount of tokenX user willing to pay. /// @param recipient the address to receive tokenY /// @param amount the max amount of tokenX user willing to pay /// @param lowPt the lowest point(price) of x/y during swap /// @param data any data to be passed through to the callback /// @return amountX amount of tokenX acquired /// @return amountY amount of tokenY payed function swapX2Y( address recipient, uint128 amount, int24 lowPt, bytes calldata data ) external returns (uint256 amountX, uint256 amountY); /// @notice Swap tokenX for tokenY, given amount of tokenY user desires. /// @param recipient the address to receive tokenY /// @param desireY the amount of tokenY user desires /// @param lowPt the lowest point(price) of x/y during swap /// @param data any data to be passed through to the callback /// @return amountX amount of tokenX acquired /// @return amountY amount of tokenY payed function swapX2YDesireY( address recipient, uint128 desireY, int24 lowPt, bytes calldata data ) external returns (uint256 amountX, uint256 amountY); /// @notice Returns sqrt(1.0001), in 96 bit fixpoint number. function sqrtRate_96() external view returns(uint160); /// @notice State values of pool. /// @return sqrtPrice_96 a 96 fixpoing number describe the sqrt value of current price(tokenX/tokenY) /// @return currentPoint the current point of the pool, 1.0001 ^ currentPoint = price /// @return observationCurrentIndex the index of the last oracle observation that was written, /// @return observationQueueLen the current maximum number of observations stored in the pool, /// @return observationNextQueueLen the next maximum number of observations, to be updated when the observation. /// @return locked whether the pool is locked (only used for checking reentrance) /// @return liquidity liquidity on the currentPoint (currX * sqrtPrice + currY / sqrtPrice) /// @return liquidityX liquidity of tokenX function state() external view returns( uint160 sqrtPrice_96, int24 currentPoint, uint16 observationCurrentIndex, uint16 observationQueueLen, uint16 observationNextQueueLen, bool locked, uint128 liquidity, uint128 liquidityX ); /// @notice LimitOrder info on a given point. /// @param point the given point /// @return sellingX total amount of tokenX selling on the point /// @return earnY total amount of unclaimed earned tokenY for unlegacy sellingX /// @return accEarnY total amount of earned tokenY(via selling tokenX) by all users at this point as of the last swap /// @return legacyAccEarnY latest recorded 'accEarnY' value when sellingX is clear (legacy) /// @return legacyEarnY total amount of unclaimed earned tokenY for legacy (cleared during swap) sellingX /// @return sellingY total amount of tokenYselling on the point /// @return earnX total amount of unclaimed earned tokenX for unlegacy sellingY /// @return legacyEarnX total amount of unclaimed earned tokenX for legacy (cleared during swap) sellingY /// @return accEarnX total amount of earned tokenX(via selling tokenY) by all users at this point as of the last swap /// @return legacyAccEarnX latest recorded 'accEarnX' value when sellingY is clear (legacy) function limitOrderData(int24 point) external view returns( uint128 sellingX, uint128 earnY, uint256 accEarnY, uint256 legacyAccEarnY, uint128 legacyEarnY, uint128 sellingY, uint128 earnX, uint128 legacyEarnX, uint256 accEarnX, uint256 legacyAccEarnX ); /// @notice Query infomation about a point whether has limit order or is an liquidity's endpoint. /// @param point point to query /// @return val endpoint for val&1>0 and has limit order for val&2 > 0 function orderOrEndpoint(int24 point) external returns(int24 val); /// @notice Returns observation data about a specific index. /// @param index the index of observation array /// @return timestamp the timestamp of the observation, /// @return accPoint the point multiplied by seconds elapsed for the life of the pool as of the observation timestamp, /// @return init whether the observation has been initialized and the above values are safe to use function observations(uint256 index) external view returns ( uint32 timestamp, int56 accPoint, bool init ); /// @notice Point status in the pool. /// @param point the point /// @return liquidSum the total amount of liquidity that uses the point either as left endpoint or right endpoint /// @return liquidDelta how much liquidity changes when the pool price crosses the point from left to right /// @return accFeeXOut_128 the fee growth on the other side of the point from the current point in tokenX /// @return accFeeYOut_128 the fee growth on the other side of the point from the current point in tokenY /// @return isEndpt whether the point is an endpoint of a some miner's liquidity, true if liquidSum > 0 function points(int24 point) external view returns ( uint128 liquidSum, int128 liquidDelta, uint256 accFeeXOut_128, uint256 accFeeYOut_128, bool isEndpt ); /// @notice Returns 256 packed point (statusVal>0) boolean values. See PointBitmap for more information. function pointBitmap(int16 wordPosition) external view returns (uint256); /// @notice Returns the integral value of point(time) and integral value of 1/liquidity(time) /// at some target timestamps (block.timestamp - secondsAgo[i]) /// @dev Reverts if target timestamp is early than oldest observation in the queue /// @dev If you call this method with secondsAgos = [3600, 0]. the average point of this pool during recent hour is /// (accPoints[1] - accPoints[0]) / 3600 /// @param secondsAgos describe the target timestamp , targetTimestimp[i] = block.timestamp - secondsAgo[i] /// @return accPoints integral value of point(time) from 0 to each target timestamp function observe(uint32[] calldata secondsAgos) external view returns (int56[] memory accPoints); /// @notice Expand max-length of observation queue. /// @param newNextQueueLen new value of observationNextQueueLen, which should be greater than current observationNextQueueLen function expandObservationQueue(uint16 newNextQueueLen) external; /// @notice Borrow tokenX and/or tokenY and pay it back within a block. /// @dev The caller needs to implement a IiZiSwapPool#flashCallback callback function /// @param recipient the address which will receive the tokenY and/or tokenX /// @param amountX the amount of tokenX to borrow /// @param amountY the amount of tokenY to borrow /// @param data Any data to be passed through to the callback function flash( address recipient, uint256 amountX, uint256 amountY, bytes calldata data ) external; /// @notice Returns a snapshot infomation of Liquidity in [leftPoint, rightPoint). /// @param leftPoint left endpoint of range, should be times of pointDelta /// @param rightPoint right endpoint of range, should be times of pointDelta /// @return deltaLiquidities an array of delta liquidity for points in the range /// note 1. delta liquidity here is amount of liquidity changed when cross a point from left to right /// note 2. deltaLiquidities only contains points which are times of pointDelta /// note 3. this function may cost a ENORMOUS amount of gas, be careful to call function liquiditySnapshot(int24 leftPoint, int24 rightPoint) external view returns(int128[] memory deltaLiquidities); struct LimitOrderStruct { uint128 sellingX; uint128 earnY; uint256 accEarnY; uint128 sellingY; uint128 earnX; uint256 accEarnX; } /// @notice Returns a snapshot infomation of Limit Order in [leftPoint, rightPoint). /// @param leftPoint left endpoint of range, should be times of pointDelta /// @param rightPoint right endpoint of range, should be times of pointDelta /// @return limitOrders an array of Limit Orders for points in the range /// note 1. this function may cost a HUGE amount of gas, be careful to call function limitOrderSnapshot(int24 leftPoint, int24 rightPoint) external view returns(LimitOrderStruct[] memory limitOrders); /// @notice Amount of charged fee on tokenX. function totalFeeXCharged() external view returns(uint256); /// @notice Amount of charged fee on tokenY. function totalFeeYCharged() external view returns(uint256); /// @notice Percent to charge from miner's fee. function feeChargePercent() external view returns(uint24); /// @notice Collect charged fee, only factory's chargeReceiver can call. function collectFeeCharged() external; /// @notice modify 'feeChargePercent', only owner has authority. /// @param newFeeChargePercent new value of feeChargePercent, a nature number range in [0, 100], function modifyFeeChargePercent(uint24 newFeeChargePercent) external; } // File contracts/universal/interfaces/IiZiSwapClassicFactory.sol pragma solidity >=0.5.0; interface IiZiSwapClassicFactory { function getPair(address tokenA, address tokenB) external view returns (address pair); } // File contracts/universal/interfaces/IiZiSwapClassicPair.sol pragma solidity >=0.5.0; interface IiZiSwapClassicPair { function getPairState() external view returns (uint112 reserve0, uint112 reserve1, uint16 fee, uint32 blockTimestampLast); function token0() external view returns (address); function token1() external view returns (address); function swap(uint amount0Out, uint amount1Out, address to, bytes calldata data) external; } // File contracts/universal/types.sol pragma solidity ^0.8.4; struct SwapCallbackData { bytes path; address payer; } struct SwapSingleParams { address tokenIn; address tokenOut; uint24 fee; uint256 amount; address recipient; } // File contracts/universal/ClassicSwapRouter.sol pragma solidity ^0.8.4; abstract contract ClassicSwapRouter { using Path for bytes; address public iZiClassicFactory; /// @notice constructor to create this contract /// @param _iZiClassicFactory address of iZiSwap classic factory constructor(address _iZiClassicFactory) { iZiClassicFactory = _iZiClassicFactory; } function iZiClassicPair(address tokenA, address tokenB) public view returns(address) { return IiZiSwapClassicFactory(iZiClassicFactory).getPair(tokenA, tokenB); } function getPairState(address tokenA, address tokenB) public view returns(uint256 reserveA, uint256 reserveB, uint16 fee, address pair) { pair = iZiClassicPair(tokenA, tokenB); uint112 reserve0; uint112 reserve1; (reserve0, reserve1, fee, ) = IiZiSwapClassicPair(pair).getPairState(); address token0 = IiZiSwapClassicPair(pair).token0(); (reserveA, reserveB) = (tokenA == token0) ? (reserve0, reserve1) : (reserve1, reserve0); } // given an input amount of an asset and pair reserves, returns the maximum output amount of the other asset function getAmountOut(uint256 amountIn, uint256 reserveIn, uint256 reserveOut, uint16 fee) internal pure returns (uint amountOut) { require(amountIn > 0, 'iZiSwapClassicLibrary: INSUFFICIENT_INPUT_AMOUNT'); require(reserveIn > 0 && reserveOut > 0, 'iZiSwapClassicLibrary: INSUFFICIENT_LIQUIDITY'); uint256 amountInWithFee = amountIn * (10000 - fee); uint256 numerator = amountInWithFee * reserveOut; uint256 denominator = reserveIn * 10000 + amountInWithFee; amountOut = numerator / denominator; } // given an output amount of an asset and pair reserves, returns a required input amount of the other asset function getAmountIn(uint256 amountOut, uint256 reserveIn, uint256 reserveOut, uint16 fee) internal pure returns (uint amountIn) { require(amountOut > 0, 'iZiSwapClassicLibrary: INSUFFICIENT_OUTPUT_AMOUNT'); require(reserveIn > 0 && reserveOut > 0, 'iZiSwapClassicLibrary: INSUFFICIENT_LIQUIDITY'); uint256 numerator = reserveIn * amountOut * 10000; uint256 denominator = (reserveOut - amountOut) * (10000 - fee); amountIn = numerator / denominator + 1; } function classicGetAmountOut( address tokenIn, address tokenOut, uint256 amountIn ) internal view returns (uint256 amountOut, address pair) { (uint256 reserveIn, uint256 reserveOut, uint16 fee, address _pair) = getPairState(tokenIn, tokenOut); amountOut = getAmountOut(amountIn, reserveIn, reserveOut, fee); pair = _pair; } function classicGetAmountIn( address tokenIn, address tokenOut, uint256 amountOut ) internal view returns (uint256 amountIn, address pair) { (uint256 reserveIn, uint256 reserveOut, uint16 fee, address _pair) = getPairState(tokenIn, tokenOut); amountIn = getAmountIn(amountOut, reserveIn, reserveOut, fee); pair = _pair; } function _classicSwap(address pair, address tokenIn, address tokenOut, uint256 amountOut, address recipient) internal virtual returns(uint256 acquire){ (uint256 amount0Out, uint256 amount1Out) = tokenIn < tokenOut ? (uint256(0), amountOut) : (amountOut, uint256(0)); uint256 amountBefore = IERC20(tokenOut).balanceOf(recipient); IiZiSwapClassicPair(pair).swap( amount0Out, amount1Out, recipient, new bytes(0) ); uint256 amountAfter = IERC20(tokenOut).balanceOf(recipient); acquire = amountAfter - amountBefore; } function classicAmountSingleInternal( SwapSingleParams memory params, address payer ) internal returns (uint256 cost, uint256 acquire) { (uint256 _acquire, address pair) = classicGetAmountOut(params.tokenIn, params.tokenOut, params.amount); // pay from payer pay(params.tokenIn, payer, pair, params.amount); cost = params.amount; acquire = _classicSwap(pair, params.tokenIn, params.tokenOut, _acquire, params.recipient); } function classicDesireSingleInternal( SwapSingleParams memory params, SwapCallbackData memory data ) internal returns (uint256 acquire) { (uint256 cost, address pair) = classicGetAmountIn(params.tokenIn, params.tokenOut, params.amount); if (data.path.hasMultiplePools()) { // call classicDesireInternalCallback(...) data.path = data.path.skipToken(); swapDesireInternal(cost, pair, data); } else { // pay from payer pay(params.tokenIn, data.payer, pair, cost); } acquire = _classicSwap(pair, params.tokenIn, params.tokenOut, params.amount, params.recipient); } function swapDesireInternal( uint256 desire, address recipient, SwapCallbackData memory data ) internal virtual returns (uint256 acquire, address tokenOut) {} function pay( address token, address payer, address recipient, uint256 amount ) internal virtual {} } // File contracts/universal/iZiSwapRouter.sol pragma solidity ^0.8.4; abstract contract iZiSwapRouter is IiZiSwapCallback { using Path for bytes; function iZiSwapPool(address tokenX, address tokenY, uint24 fee) public view returns(address) { return IiZiSwapFactory(iZiSwapFactory).pool(tokenX, tokenY, fee); } function iZiSwapVerify(address tokenX, address tokenY, uint24 fee) internal view { require (msg.sender == iZiSwapPool(tokenX, tokenY, fee), "sp"); } address public iZiSwapFactory; /// @notice constructor to create this contract /// @param _iZiSwapFactory address of iZiSwap factory constructor(address _iZiSwapFactory) { iZiSwapFactory = _iZiSwapFactory; } /// @notice Callback for swapY2X and swapY2XDesireX, in order to pay tokenY from trader. /// @param x amount of tokenX trader acquired /// @param y amount of tokenY need to pay from trader /// @param data encoded SwapCallbackData function swapY2XCallback( uint256 x, uint256 y, bytes calldata data ) external override { SwapCallbackData memory dt = abi.decode(data, (SwapCallbackData)); (address token0, address token1, uint24 fee) = dt.path.decodeFirstPool(); iZiSwapVerify(token0, token1, fee); if (token0 < token1) { // token1 is y, amount of token1 is calculated // called from swapY2XDesireX(...) if (dt.path.hasMultiplePools()) { dt.path = dt.path.skipToken(); swapDesireInternal(y, msg.sender, dt); } else { pay(token1, dt.payer, msg.sender, y); } } else { // token0 is y, amount of token0 is input param // called from swapY2X(...) pay(token0, dt.payer, msg.sender, y); } } /// @notice Callback for swapX2Y and swapX2YDesireY, in order to pay tokenX from trader. /// @param x amount of tokenX need to pay from trader /// @param y amount of tokenY trader acquired /// @param data encoded SwapCallbackData function swapX2YCallback( uint256 x, uint256 y, bytes calldata data ) external override { SwapCallbackData memory dt = abi.decode(data, (SwapCallbackData)); (address token0, address token1, uint24 fee) = dt.path.decodeFirstPool(); iZiSwapVerify(token0, token1, fee); if (token0 < token1) { // token0 is x, amount of token0 is input param // called from swapX2Y(...) pay(token0, dt.payer, msg.sender, x); } else { // token1 is x, amount of token1 is calculated param // called from swapX2YDesireY(...) if (dt.path.hasMultiplePools()) { dt.path = dt.path.skipToken(); swapDesireInternal(x, msg.sender, dt); } else { pay(token1, dt.payer, msg.sender, x); } } } function iZiSwapDesireSingleInternal( SwapSingleParams memory params, SwapCallbackData memory data ) internal returns (uint256 acquire) { address poolAddr = iZiSwapPool(params.tokenOut, params.tokenIn, params.fee); if (params.tokenOut < params.tokenIn) { // tokenOut is tokenX, tokenIn is tokenY // we should call y2XDesireX (acquire, ) = IiZiSwapPool(poolAddr).swapY2XDesireX( params.recipient, uint128(params.amount), 799999, abi.encode(data) ); } else { // tokenOut is tokenY // tokenIn is tokenX (, acquire) = IiZiSwapPool(poolAddr).swapX2YDesireY( params.recipient, uint128(params.amount), -799999, abi.encode(data) ); } } function iZiSwapAmountSingleInternal( SwapSingleParams memory params, address payer ) internal returns (uint256 cost, uint256 acquire) { address poolAddr = iZiSwapPool(params.tokenOut, params.tokenIn, params.fee); if (params.tokenIn < params.tokenOut) { // swapX2Y (cost, acquire) = IiZiSwapPool(poolAddr).swapX2Y( params.recipient, uint128(params.amount), -799999, abi.encode(SwapCallbackData({path: abi.encodePacked(params.tokenIn, params.fee, params.tokenOut), payer: payer})) ); } else { // swapY2X uint256 costY; (acquire, costY) = IiZiSwapPool(poolAddr).swapY2X( params.recipient, uint128(params.amount), 799999, abi.encode(SwapCallbackData({path: abi.encodePacked(params.tokenIn, params.fee, params.tokenOut), payer: payer})) ); } } function swapDesireInternal( uint256 desire, address recipient, SwapCallbackData memory data ) internal virtual returns (uint256 acquire, address tokenOut) {} function pay( address token, address payer, address recipient, uint256 amount ) internal virtual {} } // File contracts/universal/UniversalBase.sol pragma solidity ^0.8.4; /// @title Interface for WETH9 interface IWETH9 is IERC20 { /// @notice Deposit ether to get wrapped ether function deposit() external payable; /// @notice Withdraw wrapped ether to get ether function withdraw(uint256) external; } abstract contract UniversalBase { address public WETH9; modifier checkDeadline(uint256 deadline) { require(block.timestamp <= deadline, 'Out of time'); _; } constructor(address weth) { WETH9 = weth; } receive() external payable {} /// @notice Make multiple function calls in this contract in a single transaction /// and return the data for each function call, revert if any function call fails /// @param data The encoded function data for each function call /// @return results result of each function call function multicall(bytes[] calldata data) external payable returns (bytes[] memory results) { results = new bytes[](data.length); for (uint256 i = 0; i < data.length; i++) { (bool success, bytes memory result) = address(this).delegatecall(data[i]); if (!success) { if (result.length < 68) revert(); assembly { result := add(result, 0x04) } revert(abi.decode(result, (string))); } results[i] = result; } } /// @notice Transfer tokens from the targeted address to the given destination /// @notice Errors with 'STF' if transfer fails /// @param token The contract address of the token to be transferred /// @param from The originating address from which the tokens will be transferred /// @param to The destination address of the transfer /// @param value The amount to be transferred function safeTransferFrom( address token, address from, address to, uint256 value ) internal { (bool success, bytes memory data) = token.call(abi.encodeWithSelector(IERC20.transferFrom.selector, from, to, value)); require(success && (data.length == 0 || abi.decode(data, (bool))), 'STF'); } /// @notice Transfer tokens from msg.sender to a recipient /// @dev Errors with ST if transfer fails /// @param token The contract address of the token which will be transferred /// @param to The recipient of the transfer /// @param value The value of the transfer function safeTransfer( address token, address to, uint256 value ) internal { (bool success, bytes memory data) = token.call(abi.encodeWithSelector(IERC20.transfer.selector, to, value)); require(success && (data.length == 0 || abi.decode(data, (bool))), 'ST'); } /// @notice Approve the stipulated contract to spend the given allowance in the given token /// @dev Errors with 'SA' if transfer fails /// @param token The contract address of the token to be approved /// @param to The target of the approval /// @param value The amount of the given token the target will be allowed to spend function safeApprove( address token, address to, uint256 value ) internal { (bool success, bytes memory data) = token.call(abi.encodeWithSelector(IERC20.approve.selector, to, value)); require(success && (data.length == 0 || abi.decode(data, (bool))), 'SA'); } /// @notice Transfer ETH to the recipient address /// @dev Fails with `STE` /// @param to The destination of the transfer /// @param value The value to be transferred function safeTransferETH(address to, uint256 value) internal { (bool success, ) = to.call{value: value}(new bytes(0)); require(success, 'STE'); } /// @notice Withdraw all weth9 token of this contract and send the withdrawed eth to recipient /// usually used in multicall when mint/swap/update limitorder with eth /// normally this contract has no any erc20 token or eth after or before a transaction /// we donot need to worry that some one can steal eth from this contract /// @param minAmount The minimum amount of WETH9 to withdraw /// @param recipient The address to receive all withdrawed eth from this contract function unwrapWETH9(uint256 minAmount, address recipient) external payable { uint256 all = IWETH9(WETH9).balanceOf(address(this)); require(all >= minAmount, 'WETH9 Not Enough'); if (all > 0) { IWETH9(WETH9).withdraw(all); safeTransferETH(recipient, all); } } /// @notice Send all balance of specified token in this contract to recipient /// usually used in multicall when mint/swap/update limitorder with eth /// normally this contract has no any erc20 token or eth after or before a transaction /// we donot need to worry that some one can steal some token from this contract /// @param token address of the token /// @param minAmount balance should >= minAmount /// @param recipient the address to receive specified token from this contract function sweepToken( address token, uint256 minAmount, address recipient ) external payable { uint256 all = IERC20(token).balanceOf(address(this)); require(all >= minAmount, 'WETH9 Not Enough'); if (all > 0) { safeTransfer(token, recipient, all); } } /// @notice Send all balance of eth in this contract to msg.sender /// usually used in multicall when mint/swap/update limitorder with eth /// normally this contract has no any erc20 token or eth after or before a transaction /// we donot need to worry that some one can steal some token from this contract function refundETH() external payable { if (address(this).balance > 0) safeTransferETH(msg.sender, address(this).balance); } } // File contracts/universal/UniversalSwapRouter.sol pragma solidity ^0.8.4; contract UniversalSwapRouter is Ownable, UniversalBase, iZiSwapRouter, ClassicSwapRouter { using Path for bytes; uint256 private constant DEFAULT_PAYED_CACHED = type(uint256).max; uint256 private payedCached = DEFAULT_PAYED_CACHED; address public charger; /// @notice constructor to create this contract /// @param _iZiSwapFactory address of iZiSwap factory /// @param _iZiClassicFactory address of iZiSwap classic factory /// @param _weth address of weth token /// @param _charger address of fee charger constructor(address _iZiSwapFactory, address _iZiClassicFactory, address _weth, address _charger) UniversalBase(_weth) iZiSwapRouter(_iZiSwapFactory) ClassicSwapRouter(_iZiClassicFactory) { charger = _charger; } /// @param token The token to pay /// @param payer The entity that must pay /// @param recipient The entity that will receive payment /// @param value The amount to pay function pay( address token, address payer, address recipient, uint256 value ) internal override(ClassicSwapRouter, iZiSwapRouter) { if (token == WETH9 && address(this).balance >= value) { // pay with WETH9 IWETH9(WETH9).deposit{value: value}(); // wrap only what is needed to pay IWETH9(WETH9).transfer(recipient, value); payedCached = value; } else if (payer == address(this)) { // pay with tokens already in the contract (for the exact input multihop case) safeTransfer(token, recipient, value); } else { // pull payment safeTransferFrom(token, payer, recipient, value); payedCached = value; } } function swapDesireInternal( uint256 desire, address recipient, SwapCallbackData memory data ) internal override(iZiSwapRouter, ClassicSwapRouter) returns (uint256 acquire, address tokenOut) { address tokenIn; uint24 poolFee; (tokenOut, tokenIn, poolFee) = data.path.decodeFirstPool(); if (poolFee != 0) { // iZiSwap acquire = iZiSwapDesireSingleInternal( SwapSingleParams({ tokenIn: tokenIn, tokenOut: tokenOut, fee: poolFee, amount: desire, recipient: recipient }), data ); } else { // iZiSwap classic require(iZiClassicFactory != address(0), "classic not supported!"); acquire = classicDesireSingleInternal( SwapSingleParams({ tokenIn: tokenIn, tokenOut: tokenOut, fee: poolFee, amount: desire, recipient: recipient }), data ); } } function swapAmountInternal( uint128 amount, SwapCallbackData memory data ) private returns (uint256 cost, uint256 acquire, address tokenOut) { address payer = msg.sender; // msg.sender pays for the first hop bool firstHop = true; while (true) { bool hasMultiplePools = data.path.hasMultiplePools(); address tokenIn; uint24 poolFee; (tokenIn, tokenOut, poolFee) = data.path.decodeFirstPool(); SwapSingleParams memory params = SwapSingleParams({ tokenIn: tokenIn, tokenOut: tokenOut, fee: poolFee, amount: amount, recipient: address(this) }); uint256 _cost; if (poolFee != 0) { // iZiSwap (_cost, acquire) = iZiSwapAmountSingleInternal( params, payer ); } else { // iZiSwap classic require(iZiClassicFactory != address(0), "classic not supported!"); (_cost, acquire) = classicAmountSingleInternal( params, payer ); } if (firstHop) { cost = _cost; } firstHop = false; // decide whether to continue or terminate if (hasMultiplePools) { payer = address(this); // at this point, the caller has paid data.path = data.path.skipToken(); amount = uint128(acquire); } else { break; } } } function desireAddFee(uint256 originDesire, uint16 outFee) internal view returns(uint256 desire) { if (charger == address(0) || outFee == 0) { return originDesire; } desire = originDesire * 10000 / (10000 - outFee) + 1; return desire; } function chargeFee(address token, uint256 originAcquire, uint16 outFee) internal returns(uint256 acquire) { if (charger == address(0)) { return originAcquire; } uint256 tokenFee = originAcquire * outFee / 10000; acquire = originAcquire - tokenFee; if (tokenFee > 0) { safeTransfer(token, charger, tokenFee); } return acquire; } struct SwapDesireParams { bytes path; address recipient; uint128 desire; uint256 maxPayed; // outFee / 10000 is feeTier // outFee must <= 500 uint16 outFee; uint256 deadline; } /// @notice Swap given amount of target token, usually used in multi-hop case. function swapDesire(SwapDesireParams calldata params) external payable checkDeadline(params.deadline) returns (uint256 cost, uint256 acquire) { require(params.outFee <= 500, "outFee too much!"); address recipient = params.recipient; if (recipient == address(0)) { recipient = address(this); } address tokenOut; uint256 desire = params.desire; if (params.outFee > 0) { desire = desireAddFee(params.desire, params.outFee); } (acquire, tokenOut) = swapDesireInternal( desire, address(this), SwapCallbackData({path: params.path, payer: msg.sender}) ); if (params.outFee > 0) { acquire = chargeFee(tokenOut, acquire, params.outFee); } if (recipient != address(this)) { safeTransfer(tokenOut, recipient, acquire); } cost = payedCached; require(cost <= params.maxPayed, 'Too much payed in swapDesire'); require(acquire >= params.desire, 'Too much requested in swapDesire'); payedCached = DEFAULT_PAYED_CACHED; } struct SwapAmountParams { bytes path; address recipient; uint128 amount; uint256 minAcquired; // outFee / 10000 is feeTier // outFee must <= 500 uint16 outFee; uint256 deadline; } /// @notice Swap given amount of input token, usually used in multi-hop case. function swapAmount(SwapAmountParams calldata params) external payable checkDeadline(params.deadline) returns (uint256 cost, uint256 acquire) { require(params.outFee <= 500, "outFee too much!"); address recipient = params.recipient; if (recipient == address(0)) { recipient = address(this); } address tokenOut; (cost, acquire, tokenOut) = swapAmountInternal( params.amount, SwapCallbackData({path: params.path, payer: msg.sender}) ); if (params.outFee > 0) { acquire = chargeFee(tokenOut, acquire, params.outFee); } if (recipient != address(this)) { safeTransfer(tokenOut, recipient, acquire); } require(acquire >= params.minAcquired, 'Too much requested in swapAmount'); } function setCharger(address _charger) external onlyOwner { charger = _charger; } }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_iZiSwapFactory","type":"address"},{"internalType":"address","name":"_iZiClassicFactory","type":"address"},{"internalType":"address","name":"_weth","type":"address"},{"internalType":"address","name":"_charger","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"inputs":[],"name":"WETH9","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"charger","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenA","type":"address"},{"internalType":"address","name":"tokenB","type":"address"}],"name":"getPairState","outputs":[{"internalType":"uint256","name":"reserveA","type":"uint256"},{"internalType":"uint256","name":"reserveB","type":"uint256"},{"internalType":"uint16","name":"fee","type":"uint16"},{"internalType":"address","name":"pair","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"iZiClassicFactory","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenA","type":"address"},{"internalType":"address","name":"tokenB","type":"address"}],"name":"iZiClassicPair","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"iZiSwapFactory","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenX","type":"address"},{"internalType":"address","name":"tokenY","type":"address"},{"internalType":"uint24","name":"fee","type":"uint24"}],"name":"iZiSwapPool","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes[]","name":"data","type":"bytes[]"}],"name":"multicall","outputs":[{"internalType":"bytes[]","name":"results","type":"bytes[]"}],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"refundETH","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_charger","type":"address"}],"name":"setCharger","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"bytes","name":"path","type":"bytes"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint128","name":"amount","type":"uint128"},{"internalType":"uint256","name":"minAcquired","type":"uint256"},{"internalType":"uint16","name":"outFee","type":"uint16"},{"internalType":"uint256","name":"deadline","type":"uint256"}],"internalType":"struct UniversalSwapRouter.SwapAmountParams","name":"params","type":"tuple"}],"name":"swapAmount","outputs":[{"internalType":"uint256","name":"cost","type":"uint256"},{"internalType":"uint256","name":"acquire","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"components":[{"internalType":"bytes","name":"path","type":"bytes"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint128","name":"desire","type":"uint128"},{"internalType":"uint256","name":"maxPayed","type":"uint256"},{"internalType":"uint16","name":"outFee","type":"uint16"},{"internalType":"uint256","name":"deadline","type":"uint256"}],"internalType":"struct UniversalSwapRouter.SwapDesireParams","name":"params","type":"tuple"}],"name":"swapDesire","outputs":[{"internalType":"uint256","name":"cost","type":"uint256"},{"internalType":"uint256","name":"acquire","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"x","type":"uint256"},{"internalType":"uint256","name":"y","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"swapX2YCallback","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"x","type":"uint256"},{"internalType":"uint256","name":"y","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"swapY2XCallback","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"token","type":"address"},{"internalType":"uint256","name":"minAmount","type":"uint256"},{"internalType":"address","name":"recipient","type":"address"}],"name":"sweepToken","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"minAmount","type":"uint256"},{"internalType":"address","name":"recipient","type":"address"}],"name":"unwrapWETH9","outputs":[],"stateMutability":"payable","type":"function"},{"stateMutability":"payable","type":"receive"}]
Contract Creation Code
60806040526000196004553480156200001757600080fd5b5060405162002e4038038062002e408339810160408190526200003a916200010f565b8284836200004833620000a2565b600180546001600160a01b03199081166001600160a01b039384161790915560028054821693831693909317909255600380548316938216939093179092556005805490911692909116919091179055506200016b915050565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b80516001600160a01b03811681146200010a57600080fd5b919050565b6000806000806080858703121562000125578384fd5b6200013085620000f2565b93506200014060208601620000f2565b92506200015060408601620000f2565b91506200016060608601620000f2565b905092959194509250565b612cc5806200017b6000396000f3fe6080604052600436106101085760003560e01c80638da5cb5b116100a0578063d3e1c28411610064578063d3e1c284146102f5578063d5525c8014610315578063df2ab5bb14610335578063e58c749c14610348578063f2fde38b1461035b57600080fd5b80638da5cb5b14610257578063ac9650d814610275578063afddaa1114610295578063b8c69d74146102b5578063c02ee7b7146102d557600080fd5b80631017459a1461011457806312210e8a1461014a578063187806841461015457806349404b7c146101745780634aa4a4fc146101875780635e862391146101a7578063715018a6146101fa5780637752020b1461020f57806388d527dc1461022f57600080fd5b3661010f57005b600080fd5b34801561012057600080fd5b50600254610134906001600160a01b031681565b6040516101419190612837565b60405180910390f35b61015261037b565b005b34801561016057600080fd5b5061015261016f36600461272a565b61038d565b6101526101823660046126e3565b610438565b34801561019357600080fd5b50600154610134906001600160a01b031681565b3480156101b357600080fd5b506101c76101c2366004612380565b610557565b6040516101419493929190938452602084019290925261ffff1660408301526001600160a01b0316606082015260800190565b34801561020657600080fd5b506101526106a0565b34801561021b57600080fd5b50600554610134906001600160a01b031681565b61024261023d36600461253c565b6106b2565b60408051928352602083019190915201610141565b34801561026357600080fd5b506000546001600160a01b0316610134565b61028861028336600461243f565b610858565b6040516101419190612892565b3480156102a157600080fd5b50600354610134906001600160a01b031681565b3480156102c157600080fd5b506101346102d0366004612380565b6109d9565b3480156102e157600080fd5b506101346102f03660046123b8565b610a68565b34801561030157600080fd5b5061015261031036600461272a565b610b03565b34801561032157600080fd5b50610152610330366004612348565b610b98565b610152610343366004612409565b610bc2565b61024261035636600461253c565b610c7a565b34801561036757600080fd5b50610152610376366004612348565b610ef4565b471561038b5761038b3347610f6d565b565b600061039b82840184612576565b905060008060006103af8460000151611010565b9250925092506103c083838361104c565b816001600160a01b0316836001600160a01b031610156103ef576103ea838560200151338b61109c565b61042e565b83516103fa906111ee565b1561041e57835161040a90611228565b8452610417883386611259565b505061042e565b61042e828560200151338b61109c565b5050505050505050565b6001546040516370a0823160e01b81526000916001600160a01b0316906370a0823190610469903090600401612837565b60206040518083038186803b15801561048157600080fd5b505afa158015610495573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906104b991906126cb565b9050828110156104e45760405162461bcd60e51b81526004016104db9061295e565b60405180910390fd5b801561055257600154604051632e1a7d4d60e01b8152600481018390526001600160a01b0390911690632e1a7d4d90602401600060405180830381600087803b15801561053057600080fd5b505af1158015610544573d6000803e3d6000fd5b505050506105528282610f6d565b505050565b60008060008061056786866109d9565b9050600080826001600160a01b031663caee3ee66040518163ffffffff1660e01b815260040160806040518083038186803b1580156105a557600080fd5b505afa1580156105b9573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105dd9190612628565b508096508193508294505050506000836001600160a01b0316630dfe16816040518163ffffffff1660e01b815260040160206040518083038186803b15801561062557600080fd5b505afa158015610639573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061065d9190612364565b9050806001600160a01b0316896001600160a01b03161461067f578183610682565b82825b6001600160701b039182169b91169950949750929550929350505050565b6106a861135a565b61038b60006113b4565b6000808260a00135804211156106da5760405162461bcd60e51b81526004016104db90612988565b6101f46106ed60a08601608087016126af565b61ffff16111561070f5760405162461bcd60e51b81526004016104db906129e8565b60006107216040860160208701612348565b90506001600160a01b0381166107345750305b60006107a16107496060880160408901612688565b604080518082019091528061075e8a80612a7c565b8080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525050509082525033602090910152611404565b9196509450905060006107ba60a08801608089016126af565b61ffff1611156107e1576107de81856107d960a08a0160808b016126af565b611512565b93505b6001600160a01b03821630146107fc576107fc81838661157d565b85606001358410156108505760405162461bcd60e51b815260206004820181905260248201527f546f6f206d7563682072657175657374656420696e2073776170416d6f756e7460448201526064016104db565b505050915091565b6060816001600160401b0381111561088057634e487b7160e01b600052604160045260246000fd5b6040519080825280602002602001820160405280156108b357816020015b606081526020019060019003908161089e5790505b50905060005b828110156109d257600080308686858181106108e557634e487b7160e01b600052603260045260246000fd5b90506020028101906108f79190612a7c565b60405161090592919061280b565b600060405180830381855af49150503d8060008114610940576040519150601f19603f3d011682016040523d82523d6000602084013e610945565b606091505b5091509150816109915760448151101561095e57600080fd5b6004810190508080602001905181019061097891906124ce565b60405162461bcd60e51b81526004016104db91906128f3565b808484815181106109b257634e487b7160e01b600052603260045260246000fd5b6020026020010181905250505080806109ca90612c03565b9150506108b9565b5092915050565b60035460405163e6a4390560e01b81526001600160a01b0384811660048301528381166024830152600092169063e6a439059060440160206040518083038186803b158015610a2757600080fd5b505afa158015610a3b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a5f9190612364565b90505b92915050565b600254604051635f65e63560e11b81526001600160a01b038581166004830152848116602483015262ffffff84166044830152600092169063becbcc6a9060640160206040518083038186803b158015610ac157600080fd5b505afa158015610ad5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610af99190612364565b90505b9392505050565b6000610b1182840184612576565b90506000806000610b258460000151611010565b925092509250610b3683838361104c565b816001600160a01b0316836001600160a01b03161015610b88578351610b5b906111ee565b15610b78578351610b6b90611228565b8452610417873386611259565b6103ea828560200151338a61109c565b61042e838560200151338a61109c565b610ba061135a565b600580546001600160a01b0319166001600160a01b0392909216919091179055565b6040516370a0823160e01b81526000906001600160a01b038516906370a0823190610bf1903090600401612837565b60206040518083038186803b158015610c0957600080fd5b505afa158015610c1d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c4191906126cb565b905082811015610c635760405162461bcd60e51b81526004016104db9061295e565b8015610c7457610c7484838361157d565b50505050565b6000808260a0013580421115610ca25760405162461bcd60e51b81526004016104db90612988565b6101f4610cb560a08601608087016126af565b61ffff161115610cd75760405162461bcd60e51b81526004016104db906129e8565b6000610ce96040860160208701612348565b90506001600160a01b038116610cfc5750305b600080610d0f6060880160408901612688565b6001600160801b031690506000610d2c60a0890160808a016126af565b61ffff161115610d6a57610d67610d496060890160408a01612688565b6001600160801b0316610d6260a08a0160808b016126af565b61167d565b90505b610dcd813060405180604001604052808b8060000190610d8a9190612a7c565b8080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525050509082525033602090910152611259565b90955091506000610de460a0890160808a016126af565b61ffff161115610e0657610e0382866107d960a08b0160808c016126af565b94505b6001600160a01b0383163014610e2157610e2182848761157d565b60045495508660600135861115610e7a5760405162461bcd60e51b815260206004820152601c60248201527f546f6f206d75636820706179656420696e20737761704465736972650000000060448201526064016104db565b610e8a6060880160408901612688565b6001600160801b0316851015610ee25760405162461bcd60e51b815260206004820181905260248201527f546f6f206d7563682072657175657374656420696e207377617044657369726560448201526064016104db565b60001960048190555050505050915091565b610efc61135a565b6001600160a01b038116610f615760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016104db565b610f6a816113b4565b50565b604080516000808252602082019092526001600160a01b038416908390604051610f97919061281b565b60006040518083038185875af1925050503d8060008114610fd4576040519150601f19603f3d011682016040523d82523d6000602084013e610fd9565b606091505b50509050806105525760405162461bcd60e51b815260206004820152600360248201526253544560e81b60448201526064016104db565b6000808061101e84826116d7565b925061102b84601461178b565b905061104361103c60036014612b46565b85906116d7565b91509193909250565b611057838383610a68565b6001600160a01b0316336001600160a01b0316146105525760405162461bcd60e51b8152602060048201526002602482015261073760f41b60448201526064016104db565b6001546001600160a01b0385811691161480156110b95750804710155b156111b957600160009054906101000a90046001600160a01b03166001600160a01b031663d0e30db0826040518263ffffffff1660e01b81526004016000604051808303818588803b15801561110e57600080fd5b505af1158015611122573d6000803e3d6000fd5b505060015460405163a9059cbb60e01b81526001600160a01b03878116600483015260248201879052909116935063a9059cbb92506044019050602060405180830381600087803b15801561117657600080fd5b505af115801561118a573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111ae91906124ae565b506004819055610c74565b6001600160a01b0383163014156111da576111d584838361157d565b610c74565b6111e684848484611836565b600455505050565b60006111fc60036014612b46565b6014611209600382612b46565b6112139190612b46565b61121d9190612b46565b825110159050919050565b6060610a6261123960036014612b46565b61124560036014612b46565b84516112519190612bc0565b849190611940565b60008060008061126c8560000151611010565b9194509250905062ffffff8116156112d5576112ce6040518060a00160405280846001600160a01b03168152602001856001600160a01b031681526020018362ffffff168152602001898152602001886001600160a01b031681525086611a55565b9350611350565b6003546001600160a01b03166112fd5760405162461bcd60e51b81526004016104db90612906565b61134d6040518060a00160405280846001600160a01b03168152602001856001600160a01b031681526020018362ffffff168152602001898152602001886001600160a01b031681525086611bff565b93505b5050935093915050565b6000546001600160a01b0316331461038b5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016104db565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b600080803360015b600061141b87600001516111ee565b905060008061142d8960000151611010565b6040805160a0810182526001600160a01b0380861682528416602082015262ffffff83169181018290526001600160801b038f16606082015230608082015292995092945092509060009015611490576114878288611c81565b995090506114c7565b6003546001600160a01b03166114b85760405162461bcd60e51b81526004016104db90612906565b6114c28288611ecd565b995090505b85156114d1578099505b6000955084156114f5578a513097506114e990611228565b8b52979a508a976114ff565b5050505050611509565b505050505061140c565b50509250925092565b6005546000906001600160a01b031661152c575081610afc565b600061271061153f61ffff851686612b7e565b6115499190612b5e565b90506115558185612bc0565b91508015611575576005546115759086906001600160a01b03168361157d565b509392505050565b604080516001600160a01b038481166024830152604480830185905283518084039091018152606490920183526020820180516001600160e01b031663a9059cbb60e01b17905291516000928392908716916115d9919061281b565b6000604051808303816000865af19150503d8060008114611616576040519150601f19603f3d011682016040523d82523d6000602084013e61161b565b606091505b509150915081801561164557508051158061164557508080602001905181019061164591906124ae565b6116765760405162461bcd60e51b815260206004820152600260248201526114d560f21b60448201526064016104db565b5050505050565b6005546000906001600160a01b0316158061169a575061ffff8216155b156116a6575081610a62565b6116b282612710612b9d565b61ffff166116c284612710612b7e565b6116cc9190612b5e565b610a5f906001612b46565b6000816116e5816014612b46565b10156117285760405162461bcd60e51b8152602060048201526012602482015271746f416464726573735f6f766572666c6f7760701b60448201526064016104db565b611733826014612b46565b8351101561177b5760405162461bcd60e51b8152602060048201526015602482015274746f416464726573735f6f75744f66426f756e647360581b60448201526064016104db565b500160200151600160601b900490565b600081611799816003612b46565b10156117db5760405162461bcd60e51b8152602060048201526011602482015270746f55696e7432345f6f766572666c6f7760781b60448201526064016104db565b6117e6826003612b46565b8351101561182d5760405162461bcd60e51b8152602060048201526014602482015273746f55696e7432345f6f75744f66426f756e647360601b60448201526064016104db565b50016003015190565b604080516001600160a01b0385811660248301528481166044830152606480830185905283518084039091018152608490920183526020820180516001600160e01b03166323b872dd60e01b179052915160009283929088169161189a919061281b565b6000604051808303816000865af19150503d80600081146118d7576040519150601f19603f3d011682016040523d82523d6000602084013e6118dc565b606091505b509150915081801561190657508051158061190657508080602001905181019061190691906124ae565b6119385760405162461bcd60e51b815260206004820152600360248201526229aa2360e91b60448201526064016104db565b505050505050565b60608161194e81601f612b46565b101561196c5760405162461bcd60e51b81526004016104db90612936565b826119778382612b46565b10156119955760405162461bcd60e51b81526004016104db90612936565b61199f8284612b46565b845110156119e35760405162461bcd60e51b8152602060048201526011602482015270736c6963655f6f75744f66426f756e647360781b60448201526064016104db565b606082158015611a025760405191506000825260208201604052611a4c565b6040519150601f8416801560200281840101858101878315602002848b0101015b81831015611a3b578051835260209283019201611a23565b5050858452601f01601f1916604052505b50949350505050565b600080611a6f846020015185600001518660400151610a68565b905083600001516001600160a01b031684602001516001600160a01b03161015611b4857806001600160a01b031663f094685a85608001518660600151620c34ff87604051602001611ac19190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611aef949392919061284b565b6040805180830381600087803b158015611b0857600080fd5b505af1158015611b1c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b409190612707565b5091506109d2565b806001600160a01b03166359dd143685608001518660600151620c34fe1987604051602001611b779190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611ba5949392919061284b565b6040805180830381600087803b158015611bbe57600080fd5b505af1158015611bd2573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bf69190612707565b95945050505050565b6000806000611c1b856000015186602001518760600151611f2d565b91509150611c2c84600001516111ee565b15611c50578351611c3c90611228565b8452611c49828286611259565b5050611c64565b611c6485600001518560200151838561109c565b611bf6818660000151876020015188606001518960800151611f62565b6000806000611c9d856020015186600001518760400151610a68565b905084602001516001600160a01b031685600001516001600160a01b03161015611dc557806001600160a01b031663857f812f86608001518760600151620c34fe1960405180604001604052808b600001518c604001518d60200151604051602001611d0b939291906127d0565b60405160208183030381529060405281526020018a6001600160a01b0316815250604051602001611d3c9190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611d6a949392919061284b565b6040805180830381600087803b158015611d8357600080fd5b505af1158015611d97573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dbb9190612707565b9093509150611ec5565b6000816001600160a01b0316632c48125287608001518860600151620c34ff60405180604001604052808c600001518d604001518e60200151604051602001611e10939291906127d0565b60405160208183030381529060405281526020018b6001600160a01b0316815250604051602001611e419190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611e6f949392919061284b565b6040805180830381600087803b158015611e8857600080fd5b505af1158015611e9c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611ec09190612707565b509250505b509250929050565b600080600080611eea86600001518760200151886060015161211b565b91509150611f0286600001518683896060015161109c565b85606001519350611f228187600001518860200151858a60800151611f62565b925050509250929050565b600080600080600080611f408989610557565b9350935093509350611f5487858585612142565b999098509650505050505050565b6000806000856001600160a01b0316876001600160a01b031610611f8857846000611f8c565b6000855b915091506000866001600160a01b03166370a08231866040518263ffffffff1660e01b8152600401611fbe9190612837565b60206040518083038186803b158015611fd657600080fd5b505afa158015611fea573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061200e91906126cb565b6040805160008152602081019182905263022c0d9f60e01b9091529091506001600160a01b038a169063022c0d9f9061205090869086908a9060248101612a4f565b600060405180830381600087803b15801561206a57600080fd5b505af115801561207e573d6000803e3d6000fd5b50506040516370a0823160e01b8152600092506001600160a01b038a1691506370a08231906120b1908990600401612837565b60206040518083038186803b1580156120c957600080fd5b505afa1580156120dd573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061210191906126cb565b905061210d8282612bc0565b9a9950505050505050505050565b60008060008060008061212e8989610557565b9350935093509350611f5487858585612229565b600080851161219b5760405162461bcd60e51b81526020600482015260316024820152600080516020612c7083398151915260448201527011539517d3d55514155517d05353d55395607a1b60648201526084016104db565b6000841180156121ab5750600083115b6121c75760405162461bcd60e51b81526004016104db906129ad565b60006121d38686612b7e565b6121df90612710612b7e565b905060006121ef84612710612b9d565b61ffff166121fd8887612bc0565b6122079190612b7e565b90506122138183612b5e565b61221e906001612b46565b979650505050505050565b60008085116122815760405162461bcd60e51b81526020600482015260306024820152600080516020612c7083398151915260448201526f11539517d25394155517d05353d5539560821b60648201526084016104db565b6000841180156122915750600083115b6122ad5760405162461bcd60e51b81526004016104db906129ad565b60006122bb83612710612b9d565b6122c99061ffff1687612b7e565b905060006122d78583612b7e565b90506000826122e888612710612b7e565b6122f29190612b46565b90506122fe8183612b5e565b98975050505050505050565b803561231581612c4a565b919050565b600060c0828403121561232b578081fd5b50919050565b80516001600160701b038116811461231557600080fd5b600060208284031215612359578081fd5b8135610afc81612c4a565b600060208284031215612375578081fd5b8151610afc81612c4a565b60008060408385031215612392578081fd5b823561239d81612c4a565b915060208301356123ad81612c4a565b809150509250929050565b6000806000606084860312156123cc578081fd5b83356123d781612c4a565b925060208401356123e781612c4a565b9150604084013562ffffff811681146123fe578182fd5b809150509250925092565b60008060006060848603121561241d578283fd5b833561242881612c4a565b92506020840135915060408401356123fe81612c4a565b60008060208385031215612451578182fd5b82356001600160401b0380821115612467578384fd5b818501915085601f83011261247a578384fd5b813581811115612488578485fd5b8660208260051b850101111561249c578485fd5b60209290920196919550909350505050565b6000602082840312156124bf578081fd5b81518015158114610afc578182fd5b6000602082840312156124df578081fd5b81516001600160401b038111156124f4578182fd5b8201601f81018413612504578182fd5b805161251761251282612b1f565b612aef565b81815285602083850101111561252b578384fd5b611bf6826020830160208601612bd7565b60006020828403121561254d578081fd5b81356001600160401b03811115612562578182fd5b61256e8482850161231a565b949350505050565b60006020808385031215612588578182fd5b82356001600160401b038082111561259e578384fd5b90840190604082870312156125b1578384fd5b6125b9612ac7565b8235828111156125c7578586fd5b83019150601f820187136125d9578485fd5b81356125e761251282612b1f565b81815288868386010111156125fa578687fd5b81868501878301379081018501869052815261261783850161230a565b848201528094505050505092915050565b6000806000806080858703121561263d578182fd5b61264685612331565b935061265460208601612331565b9250604085015161266481612c5f565b606086015190925063ffffffff8116811461267d578182fd5b939692955090935050565b600060208284031215612699578081fd5b81356001600160801b0381168114610afc578182fd5b6000602082840312156126c0578081fd5b8135610afc81612c5f565b6000602082840312156126dc578081fd5b5051919050565b600080604083850312156126f5578182fd5b8235915060208301356123ad81612c4a565b60008060408385031215612719578182fd5b505080516020909101519092909150565b6000806000806060858703121561273f578182fd5b843593506020850135925060408501356001600160401b0380821115612763578384fd5b818701915087601f830112612776578384fd5b813581811115612784578485fd5b886020828501011115612795578485fd5b95989497505060200194505050565b600081518084526127bc816020860160208601612bd7565b601f01601f19169290920160200192915050565b606093841b6bffffffffffffffffffffffff19908116825260e89390931b6001600160e81b0319166014820152921b166017820152602b0190565b8183823760009101908152919050565b6000825161282d818460208701612bd7565b9190910192915050565b6001600160a01b0391909116815260200190565b6001600160a01b03851681526001600160801b0384166020820152600283900b6040820152608060608201819052600090612888908301846127a4565b9695505050505050565b6000602080830181845280855180835260408601915060408160051b8701019250838701855b828110156128e657603f198886030184526128d48583516127a4565b945092850192908501906001016128b8565b5092979650505050505050565b602081526000610a5f60208301846127a4565b602080825260169082015275636c6173736963206e6f7420737570706f727465642160501b604082015260600190565b6020808252600e908201526d736c6963655f6f766572666c6f7760901b604082015260600190565b60208082526010908201526f0ae8aa89072409cdee8408adcdeeaced60831b604082015260600190565b6020808252600b908201526a4f7574206f662074696d6560a81b604082015260600190565b6020808252602d90820152600080516020612c7083398151915260408201526c454e545f4c495155494449545960981b606082015260800190565b60208082526010908201526f6f757446656520746f6f206d7563682160801b604082015260600190565b602081526000825160406020840152612a2e60608401826127a4565b602094909401516001600160a01b0316604093909301929092525090919050565b84815283602082015260018060a01b038316604082015260806060820152600061288860808301846127a4565b6000808335601e19843603018112612a92578283fd5b8301803591506001600160401b03821115612aab578283fd5b602001915036819003821315612ac057600080fd5b9250929050565b604080519081016001600160401b0381118282101715612ae957612ae9612c34565b60405290565b604051601f8201601f191681016001600160401b0381118282101715612b1757612b17612c34565b604052919050565b60006001600160401b03821115612b3857612b38612c34565b50601f01601f191660200190565b60008219821115612b5957612b59612c1e565b500190565b600082612b7957634e487b7160e01b81526012600452602481fd5b500490565b6000816000190483118215151615612b9857612b98612c1e565b500290565b600061ffff83811690831681811015612bb857612bb8612c1e565b039392505050565b600082821015612bd257612bd2612c1e565b500390565b60005b83811015612bf2578181015183820152602001612bda565b83811115610c745750506000910152565b6000600019821415612c1757612c17612c1e565b5060010190565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160a01b0381168114610f6a57600080fd5b61ffff81168114610f6a57600080fdfe695a6953776170436c61737369634c6962726172793a20494e53554646494349a2646970667358221220eb6b31921860c20ba862671739b3d360a9481afb6f969a43e8ad47f7713f05b664736f6c634300080400330000000000000000000000008c7d3063579bdb0b90997e18a770eae32e1ebb0800000000000000000000000025c030116feb2e7bba054b9de0915e5f51b03e3100000000000000000000000053000000000000000000000000000000000000040000000000000000000000000481b236f191877619523ee309c82b3574214597
Deployed Bytecode
0x6080604052600436106101085760003560e01c80638da5cb5b116100a0578063d3e1c28411610064578063d3e1c284146102f5578063d5525c8014610315578063df2ab5bb14610335578063e58c749c14610348578063f2fde38b1461035b57600080fd5b80638da5cb5b14610257578063ac9650d814610275578063afddaa1114610295578063b8c69d74146102b5578063c02ee7b7146102d557600080fd5b80631017459a1461011457806312210e8a1461014a578063187806841461015457806349404b7c146101745780634aa4a4fc146101875780635e862391146101a7578063715018a6146101fa5780637752020b1461020f57806388d527dc1461022f57600080fd5b3661010f57005b600080fd5b34801561012057600080fd5b50600254610134906001600160a01b031681565b6040516101419190612837565b60405180910390f35b61015261037b565b005b34801561016057600080fd5b5061015261016f36600461272a565b61038d565b6101526101823660046126e3565b610438565b34801561019357600080fd5b50600154610134906001600160a01b031681565b3480156101b357600080fd5b506101c76101c2366004612380565b610557565b6040516101419493929190938452602084019290925261ffff1660408301526001600160a01b0316606082015260800190565b34801561020657600080fd5b506101526106a0565b34801561021b57600080fd5b50600554610134906001600160a01b031681565b61024261023d36600461253c565b6106b2565b60408051928352602083019190915201610141565b34801561026357600080fd5b506000546001600160a01b0316610134565b61028861028336600461243f565b610858565b6040516101419190612892565b3480156102a157600080fd5b50600354610134906001600160a01b031681565b3480156102c157600080fd5b506101346102d0366004612380565b6109d9565b3480156102e157600080fd5b506101346102f03660046123b8565b610a68565b34801561030157600080fd5b5061015261031036600461272a565b610b03565b34801561032157600080fd5b50610152610330366004612348565b610b98565b610152610343366004612409565b610bc2565b61024261035636600461253c565b610c7a565b34801561036757600080fd5b50610152610376366004612348565b610ef4565b471561038b5761038b3347610f6d565b565b600061039b82840184612576565b905060008060006103af8460000151611010565b9250925092506103c083838361104c565b816001600160a01b0316836001600160a01b031610156103ef576103ea838560200151338b61109c565b61042e565b83516103fa906111ee565b1561041e57835161040a90611228565b8452610417883386611259565b505061042e565b61042e828560200151338b61109c565b5050505050505050565b6001546040516370a0823160e01b81526000916001600160a01b0316906370a0823190610469903090600401612837565b60206040518083038186803b15801561048157600080fd5b505afa158015610495573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906104b991906126cb565b9050828110156104e45760405162461bcd60e51b81526004016104db9061295e565b60405180910390fd5b801561055257600154604051632e1a7d4d60e01b8152600481018390526001600160a01b0390911690632e1a7d4d90602401600060405180830381600087803b15801561053057600080fd5b505af1158015610544573d6000803e3d6000fd5b505050506105528282610f6d565b505050565b60008060008061056786866109d9565b9050600080826001600160a01b031663caee3ee66040518163ffffffff1660e01b815260040160806040518083038186803b1580156105a557600080fd5b505afa1580156105b9573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906105dd9190612628565b508096508193508294505050506000836001600160a01b0316630dfe16816040518163ffffffff1660e01b815260040160206040518083038186803b15801561062557600080fd5b505afa158015610639573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061065d9190612364565b9050806001600160a01b0316896001600160a01b03161461067f578183610682565b82825b6001600160701b039182169b91169950949750929550929350505050565b6106a861135a565b61038b60006113b4565b6000808260a00135804211156106da5760405162461bcd60e51b81526004016104db90612988565b6101f46106ed60a08601608087016126af565b61ffff16111561070f5760405162461bcd60e51b81526004016104db906129e8565b60006107216040860160208701612348565b90506001600160a01b0381166107345750305b60006107a16107496060880160408901612688565b604080518082019091528061075e8a80612a7c565b8080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525050509082525033602090910152611404565b9196509450905060006107ba60a08801608089016126af565b61ffff1611156107e1576107de81856107d960a08a0160808b016126af565b611512565b93505b6001600160a01b03821630146107fc576107fc81838661157d565b85606001358410156108505760405162461bcd60e51b815260206004820181905260248201527f546f6f206d7563682072657175657374656420696e2073776170416d6f756e7460448201526064016104db565b505050915091565b6060816001600160401b0381111561088057634e487b7160e01b600052604160045260246000fd5b6040519080825280602002602001820160405280156108b357816020015b606081526020019060019003908161089e5790505b50905060005b828110156109d257600080308686858181106108e557634e487b7160e01b600052603260045260246000fd5b90506020028101906108f79190612a7c565b60405161090592919061280b565b600060405180830381855af49150503d8060008114610940576040519150601f19603f3d011682016040523d82523d6000602084013e610945565b606091505b5091509150816109915760448151101561095e57600080fd5b6004810190508080602001905181019061097891906124ce565b60405162461bcd60e51b81526004016104db91906128f3565b808484815181106109b257634e487b7160e01b600052603260045260246000fd5b6020026020010181905250505080806109ca90612c03565b9150506108b9565b5092915050565b60035460405163e6a4390560e01b81526001600160a01b0384811660048301528381166024830152600092169063e6a439059060440160206040518083038186803b158015610a2757600080fd5b505afa158015610a3b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610a5f9190612364565b90505b92915050565b600254604051635f65e63560e11b81526001600160a01b038581166004830152848116602483015262ffffff84166044830152600092169063becbcc6a9060640160206040518083038186803b158015610ac157600080fd5b505afa158015610ad5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610af99190612364565b90505b9392505050565b6000610b1182840184612576565b90506000806000610b258460000151611010565b925092509250610b3683838361104c565b816001600160a01b0316836001600160a01b03161015610b88578351610b5b906111ee565b15610b78578351610b6b90611228565b8452610417873386611259565b6103ea828560200151338a61109c565b61042e838560200151338a61109c565b610ba061135a565b600580546001600160a01b0319166001600160a01b0392909216919091179055565b6040516370a0823160e01b81526000906001600160a01b038516906370a0823190610bf1903090600401612837565b60206040518083038186803b158015610c0957600080fd5b505afa158015610c1d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c4191906126cb565b905082811015610c635760405162461bcd60e51b81526004016104db9061295e565b8015610c7457610c7484838361157d565b50505050565b6000808260a0013580421115610ca25760405162461bcd60e51b81526004016104db90612988565b6101f4610cb560a08601608087016126af565b61ffff161115610cd75760405162461bcd60e51b81526004016104db906129e8565b6000610ce96040860160208701612348565b90506001600160a01b038116610cfc5750305b600080610d0f6060880160408901612688565b6001600160801b031690506000610d2c60a0890160808a016126af565b61ffff161115610d6a57610d67610d496060890160408a01612688565b6001600160801b0316610d6260a08a0160808b016126af565b61167d565b90505b610dcd813060405180604001604052808b8060000190610d8a9190612a7c565b8080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525050509082525033602090910152611259565b90955091506000610de460a0890160808a016126af565b61ffff161115610e0657610e0382866107d960a08b0160808c016126af565b94505b6001600160a01b0383163014610e2157610e2182848761157d565b60045495508660600135861115610e7a5760405162461bcd60e51b815260206004820152601c60248201527f546f6f206d75636820706179656420696e20737761704465736972650000000060448201526064016104db565b610e8a6060880160408901612688565b6001600160801b0316851015610ee25760405162461bcd60e51b815260206004820181905260248201527f546f6f206d7563682072657175657374656420696e207377617044657369726560448201526064016104db565b60001960048190555050505050915091565b610efc61135a565b6001600160a01b038116610f615760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016104db565b610f6a816113b4565b50565b604080516000808252602082019092526001600160a01b038416908390604051610f97919061281b565b60006040518083038185875af1925050503d8060008114610fd4576040519150601f19603f3d011682016040523d82523d6000602084013e610fd9565b606091505b50509050806105525760405162461bcd60e51b815260206004820152600360248201526253544560e81b60448201526064016104db565b6000808061101e84826116d7565b925061102b84601461178b565b905061104361103c60036014612b46565b85906116d7565b91509193909250565b611057838383610a68565b6001600160a01b0316336001600160a01b0316146105525760405162461bcd60e51b8152602060048201526002602482015261073760f41b60448201526064016104db565b6001546001600160a01b0385811691161480156110b95750804710155b156111b957600160009054906101000a90046001600160a01b03166001600160a01b031663d0e30db0826040518263ffffffff1660e01b81526004016000604051808303818588803b15801561110e57600080fd5b505af1158015611122573d6000803e3d6000fd5b505060015460405163a9059cbb60e01b81526001600160a01b03878116600483015260248201879052909116935063a9059cbb92506044019050602060405180830381600087803b15801561117657600080fd5b505af115801561118a573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111ae91906124ae565b506004819055610c74565b6001600160a01b0383163014156111da576111d584838361157d565b610c74565b6111e684848484611836565b600455505050565b60006111fc60036014612b46565b6014611209600382612b46565b6112139190612b46565b61121d9190612b46565b825110159050919050565b6060610a6261123960036014612b46565b61124560036014612b46565b84516112519190612bc0565b849190611940565b60008060008061126c8560000151611010565b9194509250905062ffffff8116156112d5576112ce6040518060a00160405280846001600160a01b03168152602001856001600160a01b031681526020018362ffffff168152602001898152602001886001600160a01b031681525086611a55565b9350611350565b6003546001600160a01b03166112fd5760405162461bcd60e51b81526004016104db90612906565b61134d6040518060a00160405280846001600160a01b03168152602001856001600160a01b031681526020018362ffffff168152602001898152602001886001600160a01b031681525086611bff565b93505b5050935093915050565b6000546001600160a01b0316331461038b5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016104db565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b600080803360015b600061141b87600001516111ee565b905060008061142d8960000151611010565b6040805160a0810182526001600160a01b0380861682528416602082015262ffffff83169181018290526001600160801b038f16606082015230608082015292995092945092509060009015611490576114878288611c81565b995090506114c7565b6003546001600160a01b03166114b85760405162461bcd60e51b81526004016104db90612906565b6114c28288611ecd565b995090505b85156114d1578099505b6000955084156114f5578a513097506114e990611228565b8b52979a508a976114ff565b5050505050611509565b505050505061140c565b50509250925092565b6005546000906001600160a01b031661152c575081610afc565b600061271061153f61ffff851686612b7e565b6115499190612b5e565b90506115558185612bc0565b91508015611575576005546115759086906001600160a01b03168361157d565b509392505050565b604080516001600160a01b038481166024830152604480830185905283518084039091018152606490920183526020820180516001600160e01b031663a9059cbb60e01b17905291516000928392908716916115d9919061281b565b6000604051808303816000865af19150503d8060008114611616576040519150601f19603f3d011682016040523d82523d6000602084013e61161b565b606091505b509150915081801561164557508051158061164557508080602001905181019061164591906124ae565b6116765760405162461bcd60e51b815260206004820152600260248201526114d560f21b60448201526064016104db565b5050505050565b6005546000906001600160a01b0316158061169a575061ffff8216155b156116a6575081610a62565b6116b282612710612b9d565b61ffff166116c284612710612b7e565b6116cc9190612b5e565b610a5f906001612b46565b6000816116e5816014612b46565b10156117285760405162461bcd60e51b8152602060048201526012602482015271746f416464726573735f6f766572666c6f7760701b60448201526064016104db565b611733826014612b46565b8351101561177b5760405162461bcd60e51b8152602060048201526015602482015274746f416464726573735f6f75744f66426f756e647360581b60448201526064016104db565b500160200151600160601b900490565b600081611799816003612b46565b10156117db5760405162461bcd60e51b8152602060048201526011602482015270746f55696e7432345f6f766572666c6f7760781b60448201526064016104db565b6117e6826003612b46565b8351101561182d5760405162461bcd60e51b8152602060048201526014602482015273746f55696e7432345f6f75744f66426f756e647360601b60448201526064016104db565b50016003015190565b604080516001600160a01b0385811660248301528481166044830152606480830185905283518084039091018152608490920183526020820180516001600160e01b03166323b872dd60e01b179052915160009283929088169161189a919061281b565b6000604051808303816000865af19150503d80600081146118d7576040519150601f19603f3d011682016040523d82523d6000602084013e6118dc565b606091505b509150915081801561190657508051158061190657508080602001905181019061190691906124ae565b6119385760405162461bcd60e51b815260206004820152600360248201526229aa2360e91b60448201526064016104db565b505050505050565b60608161194e81601f612b46565b101561196c5760405162461bcd60e51b81526004016104db90612936565b826119778382612b46565b10156119955760405162461bcd60e51b81526004016104db90612936565b61199f8284612b46565b845110156119e35760405162461bcd60e51b8152602060048201526011602482015270736c6963655f6f75744f66426f756e647360781b60448201526064016104db565b606082158015611a025760405191506000825260208201604052611a4c565b6040519150601f8416801560200281840101858101878315602002848b0101015b81831015611a3b578051835260209283019201611a23565b5050858452601f01601f1916604052505b50949350505050565b600080611a6f846020015185600001518660400151610a68565b905083600001516001600160a01b031684602001516001600160a01b03161015611b4857806001600160a01b031663f094685a85608001518660600151620c34ff87604051602001611ac19190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611aef949392919061284b565b6040805180830381600087803b158015611b0857600080fd5b505af1158015611b1c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b409190612707565b5091506109d2565b806001600160a01b03166359dd143685608001518660600151620c34fe1987604051602001611b779190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611ba5949392919061284b565b6040805180830381600087803b158015611bbe57600080fd5b505af1158015611bd2573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bf69190612707565b95945050505050565b6000806000611c1b856000015186602001518760600151611f2d565b91509150611c2c84600001516111ee565b15611c50578351611c3c90611228565b8452611c49828286611259565b5050611c64565b611c6485600001518560200151838561109c565b611bf6818660000151876020015188606001518960800151611f62565b6000806000611c9d856020015186600001518760400151610a68565b905084602001516001600160a01b031685600001516001600160a01b03161015611dc557806001600160a01b031663857f812f86608001518760600151620c34fe1960405180604001604052808b600001518c604001518d60200151604051602001611d0b939291906127d0565b60405160208183030381529060405281526020018a6001600160a01b0316815250604051602001611d3c9190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611d6a949392919061284b565b6040805180830381600087803b158015611d8357600080fd5b505af1158015611d97573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611dbb9190612707565b9093509150611ec5565b6000816001600160a01b0316632c48125287608001518860600151620c34ff60405180604001604052808c600001518d604001518e60200151604051602001611e10939291906127d0565b60405160208183030381529060405281526020018b6001600160a01b0316815250604051602001611e419190612a12565b6040516020818303038152906040526040518563ffffffff1660e01b8152600401611e6f949392919061284b565b6040805180830381600087803b158015611e8857600080fd5b505af1158015611e9c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611ec09190612707565b509250505b509250929050565b600080600080611eea86600001518760200151886060015161211b565b91509150611f0286600001518683896060015161109c565b85606001519350611f228187600001518860200151858a60800151611f62565b925050509250929050565b600080600080600080611f408989610557565b9350935093509350611f5487858585612142565b999098509650505050505050565b6000806000856001600160a01b0316876001600160a01b031610611f8857846000611f8c565b6000855b915091506000866001600160a01b03166370a08231866040518263ffffffff1660e01b8152600401611fbe9190612837565b60206040518083038186803b158015611fd657600080fd5b505afa158015611fea573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061200e91906126cb565b6040805160008152602081019182905263022c0d9f60e01b9091529091506001600160a01b038a169063022c0d9f9061205090869086908a9060248101612a4f565b600060405180830381600087803b15801561206a57600080fd5b505af115801561207e573d6000803e3d6000fd5b50506040516370a0823160e01b8152600092506001600160a01b038a1691506370a08231906120b1908990600401612837565b60206040518083038186803b1580156120c957600080fd5b505afa1580156120dd573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061210191906126cb565b905061210d8282612bc0565b9a9950505050505050505050565b60008060008060008061212e8989610557565b9350935093509350611f5487858585612229565b600080851161219b5760405162461bcd60e51b81526020600482015260316024820152600080516020612c7083398151915260448201527011539517d3d55514155517d05353d55395607a1b60648201526084016104db565b6000841180156121ab5750600083115b6121c75760405162461bcd60e51b81526004016104db906129ad565b60006121d38686612b7e565b6121df90612710612b7e565b905060006121ef84612710612b9d565b61ffff166121fd8887612bc0565b6122079190612b7e565b90506122138183612b5e565b61221e906001612b46565b979650505050505050565b60008085116122815760405162461bcd60e51b81526020600482015260306024820152600080516020612c7083398151915260448201526f11539517d25394155517d05353d5539560821b60648201526084016104db565b6000841180156122915750600083115b6122ad5760405162461bcd60e51b81526004016104db906129ad565b60006122bb83612710612b9d565b6122c99061ffff1687612b7e565b905060006122d78583612b7e565b90506000826122e888612710612b7e565b6122f29190612b46565b90506122fe8183612b5e565b98975050505050505050565b803561231581612c4a565b919050565b600060c0828403121561232b578081fd5b50919050565b80516001600160701b038116811461231557600080fd5b600060208284031215612359578081fd5b8135610afc81612c4a565b600060208284031215612375578081fd5b8151610afc81612c4a565b60008060408385031215612392578081fd5b823561239d81612c4a565b915060208301356123ad81612c4a565b809150509250929050565b6000806000606084860312156123cc578081fd5b83356123d781612c4a565b925060208401356123e781612c4a565b9150604084013562ffffff811681146123fe578182fd5b809150509250925092565b60008060006060848603121561241d578283fd5b833561242881612c4a565b92506020840135915060408401356123fe81612c4a565b60008060208385031215612451578182fd5b82356001600160401b0380821115612467578384fd5b818501915085601f83011261247a578384fd5b813581811115612488578485fd5b8660208260051b850101111561249c578485fd5b60209290920196919550909350505050565b6000602082840312156124bf578081fd5b81518015158114610afc578182fd5b6000602082840312156124df578081fd5b81516001600160401b038111156124f4578182fd5b8201601f81018413612504578182fd5b805161251761251282612b1f565b612aef565b81815285602083850101111561252b578384fd5b611bf6826020830160208601612bd7565b60006020828403121561254d578081fd5b81356001600160401b03811115612562578182fd5b61256e8482850161231a565b949350505050565b60006020808385031215612588578182fd5b82356001600160401b038082111561259e578384fd5b90840190604082870312156125b1578384fd5b6125b9612ac7565b8235828111156125c7578586fd5b83019150601f820187136125d9578485fd5b81356125e761251282612b1f565b81815288868386010111156125fa578687fd5b81868501878301379081018501869052815261261783850161230a565b848201528094505050505092915050565b6000806000806080858703121561263d578182fd5b61264685612331565b935061265460208601612331565b9250604085015161266481612c5f565b606086015190925063ffffffff8116811461267d578182fd5b939692955090935050565b600060208284031215612699578081fd5b81356001600160801b0381168114610afc578182fd5b6000602082840312156126c0578081fd5b8135610afc81612c5f565b6000602082840312156126dc578081fd5b5051919050565b600080604083850312156126f5578182fd5b8235915060208301356123ad81612c4a565b60008060408385031215612719578182fd5b505080516020909101519092909150565b6000806000806060858703121561273f578182fd5b843593506020850135925060408501356001600160401b0380821115612763578384fd5b818701915087601f830112612776578384fd5b813581811115612784578485fd5b886020828501011115612795578485fd5b95989497505060200194505050565b600081518084526127bc816020860160208601612bd7565b601f01601f19169290920160200192915050565b606093841b6bffffffffffffffffffffffff19908116825260e89390931b6001600160e81b0319166014820152921b166017820152602b0190565b8183823760009101908152919050565b6000825161282d818460208701612bd7565b9190910192915050565b6001600160a01b0391909116815260200190565b6001600160a01b03851681526001600160801b0384166020820152600283900b6040820152608060608201819052600090612888908301846127a4565b9695505050505050565b6000602080830181845280855180835260408601915060408160051b8701019250838701855b828110156128e657603f198886030184526128d48583516127a4565b945092850192908501906001016128b8565b5092979650505050505050565b602081526000610a5f60208301846127a4565b602080825260169082015275636c6173736963206e6f7420737570706f727465642160501b604082015260600190565b6020808252600e908201526d736c6963655f6f766572666c6f7760901b604082015260600190565b60208082526010908201526f0ae8aa89072409cdee8408adcdeeaced60831b604082015260600190565b6020808252600b908201526a4f7574206f662074696d6560a81b604082015260600190565b6020808252602d90820152600080516020612c7083398151915260408201526c454e545f4c495155494449545960981b606082015260800190565b60208082526010908201526f6f757446656520746f6f206d7563682160801b604082015260600190565b602081526000825160406020840152612a2e60608401826127a4565b602094909401516001600160a01b0316604093909301929092525090919050565b84815283602082015260018060a01b038316604082015260806060820152600061288860808301846127a4565b6000808335601e19843603018112612a92578283fd5b8301803591506001600160401b03821115612aab578283fd5b602001915036819003821315612ac057600080fd5b9250929050565b604080519081016001600160401b0381118282101715612ae957612ae9612c34565b60405290565b604051601f8201601f191681016001600160401b0381118282101715612b1757612b17612c34565b604052919050565b60006001600160401b03821115612b3857612b38612c34565b50601f01601f191660200190565b60008219821115612b5957612b59612c1e565b500190565b600082612b7957634e487b7160e01b81526012600452602481fd5b500490565b6000816000190483118215151615612b9857612b98612c1e565b500290565b600061ffff83811690831681811015612bb857612bb8612c1e565b039392505050565b600082821015612bd257612bd2612c1e565b500390565b60005b83811015612bf2578181015183820152602001612bda565b83811115610c745750506000910152565b6000600019821415612c1757612c17612c1e565b5060010190565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160a01b0381168114610f6a57600080fd5b61ffff81168114610f6a57600080fdfe695a6953776170436c61737369634c6962726172793a20494e53554646494349a2646970667358221220eb6b31921860c20ba862671739b3d360a9481afb6f969a43e8ad47f7713f05b664736f6c63430008040033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
0000000000000000000000008c7d3063579bdb0b90997e18a770eae32e1ebb0800000000000000000000000025c030116feb2e7bba054b9de0915e5f51b03e3100000000000000000000000053000000000000000000000000000000000000040000000000000000000000000481b236f191877619523ee309c82b3574214597
-----Decoded View---------------
Arg [0] : _iZiSwapFactory (address): 0x8c7d3063579BdB0b90997e18A770eaE32E1eBb08
Arg [1] : _iZiClassicFactory (address): 0x25C030116Feb2E7BbA054b9de0915E5F51b03e31
Arg [2] : _weth (address): 0x5300000000000000000000000000000000000004
Arg [3] : _charger (address): 0x0481B236F191877619523ee309c82B3574214597
-----Encoded View---------------
4 Constructor Arguments found :
Arg [0] : 0000000000000000000000008c7d3063579bdb0b90997e18a770eae32e1ebb08
Arg [1] : 00000000000000000000000025c030116feb2e7bba054b9de0915e5f51b03e31
Arg [2] : 0000000000000000000000005300000000000000000000000000000000000004
Arg [3] : 0000000000000000000000000481b236f191877619523ee309c82b3574214597
Deployed Bytecode Sourcemap
63605:8504:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;52559:29;;;;;;;;;;-1:-1:-1;52559:29:0;;;;-1:-1:-1;;;;;52559:29:0;;;;;;;;;;:::i;:::-;;;;;;;;63356:138;;;:::i;:::-;;54212:903;;;;;;;;;;-1:-1:-1;54212:903:0;;;;;:::i;:::-;;:::i;61810:328::-;;;;;;:::i;:::-;;:::i;57714:20::-;;;;;;;;;;-1:-1:-1;57714:20:0;;;;-1:-1:-1;;;;;57714:20:0;;;47297:487;;;;;;;;;;-1:-1:-1;47297:487:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;;;23210:25:1;;;23266:2;23251:18;;23244:34;;;;23326:6;23314:19;23309:2;23294:18;;23287:47;-1:-1:-1;;;;;23370:32:1;23365:2;23350:18;;23343:60;23197:3;23182:19;;23164:245;2882:103:0;;;;;;;;;;;;;:::i;63888:22::-;;;;;;;;;;-1:-1:-1;63888:22:0;;;;-1:-1:-1;;;;;63888:22:0;;;71109:893;;;;;;:::i;:::-;;:::i;:::-;;;;22439:25:1;;;22495:2;22480:18;;22473:34;;;;22412:18;71109:893:0;22394:119:1;2241:87:0;;;;;;;;;;-1:-1:-1;2287:7:0;2314:6;-1:-1:-1;;;;;2314:6:0;2241:87;;58278:581;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;46844:32::-;;;;;;;;;;-1:-1:-1;46844:32:0;;;;-1:-1:-1;;;;;46844:32:0;;;47113:176;;;;;;;;;;-1:-1:-1;47113:176:0;;;;;:::i;:::-;;:::i;52206:177::-;;;;;;;;;;-1:-1:-1;52206:177:0;;;;;:::i;:::-;;:::i;53055:899::-;;;;;;;;;;-1:-1:-1;53055:899:0;;;;;:::i;:::-;;:::i;72010:94::-;;;;;;;;;;-1:-1:-1;72010:94:0;;;;;:::i;:::-;;:::i;62674:338::-;;;;;;:::i;:::-;;:::i;69551:1203::-;;;;;;:::i;:::-;;:::i;3140:201::-;;;;;;;;;;-1:-1:-1;3140:201:0;;;;;:::i;:::-;;:::i;63356:138::-;63409:21;:25;63405:81;;63436:50;63452:10;63464:21;63436:15;:50::i;:::-;63356:138::o;54212:903::-;54343:26;54372:36;;;;54383:4;54372:36;:::i;:::-;54343:65;;54420:14;54436;54452:10;54466:25;:2;:7;;;:23;:25::i;:::-;54419:72;;;;;;54502:34;54516:6;54524;54532:3;54502:13;:34::i;:::-;54560:6;-1:-1:-1;;;;;54551:15:0;:6;-1:-1:-1;;;;;54551:15:0;;54547:561;;;54685:36;54689:6;54697:2;:8;;;54707:10;54719:1;54685:3;:36::i;:::-;54547:561;;;54872:7;;:26;;:24;:26::i;:::-;54868:229;;;54929:7;;:19;;:17;:19::i;:::-;54919:29;;54967:37;54986:1;54989:10;54919:2;54967:18;:37::i;:::-;;;54868:229;;;55045:36;55049:6;55057:2;:8;;;55067:10;55079:1;55045:3;:36::i;:::-;54212:903;;;;;;;;:::o;61810:328::-;61918:5;;61911:38;;-1:-1:-1;;;61911:38:0;;61897:11;;-1:-1:-1;;;;;61918:5:0;;61911:23;;:38;;61943:4;;61911:38;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;61897:52;;61975:9;61968:3;:16;;61960:45;;;;-1:-1:-1;;;61960:45:0;;;;;;;:::i;:::-;;;;;;;;;62022:7;;62018:113;;62053:5;;62046:27;;-1:-1:-1;;;62046:27:0;;;;;22229:25:1;;;-1:-1:-1;;;;;62053:5:0;;;;62046:22;;22202:18:1;;62046:27:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;62088:31;62104:9;62115:3;62088:15;:31::i;:::-;61810:328;;;:::o;47297:487::-;47371:16;47389;47407:10;47419:12;47451:30;47466:6;47474;47451:14;:30::i;:::-;47444:37;;47492:16;47519;47596:4;-1:-1:-1;;;;;47576:38:0;;:40;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;47546:70;;;;;;;;;;;;;47627:14;47664:4;-1:-1:-1;;;;;47644:32:0;;:34;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;47627:51;;47723:6;-1:-1:-1;;;;;47713:16:0;:6;-1:-1:-1;;;;;47713:16:0;;47712:64;;47757:8;47767;47712:64;;;47734:8;47744;47712:64;-1:-1:-1;;;;;47689:87:0;;;;;;;-1:-1:-1;47297:487:0;;-1:-1:-1;47297:487:0;;-1:-1:-1;47297:487:0;;-1:-1:-1;;;;47297:487:0:o;2882:103::-;2127:13;:11;:13::i;:::-;2947:30:::1;2974:1;2947:18;:30::i;71109:893::-:0;71256:12;71270:15;71221:6;:15;;;57822:8;57803:15;:27;;57795:51;;;;-1:-1:-1;;;57795:51:0;;;;;;;:::i;:::-;71329:3:::1;71312:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;:20;;;;71304:49;;;;-1:-1:-1::0;;;71304:49:0::1;;;;;;;:::i;:::-;71364:17;71384:16;::::0;;;::::1;::::0;::::1;;:::i;:::-;71364:36:::0;-1:-1:-1;;;;;;71415:23:0;::::1;71411:81;;-1:-1:-1::0;71475:4:0::1;71411:81;71502:16;71557:128;71590:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;71618:56;::::0;;;;::::1;::::0;;;;71642:11:::1;:6:::0;;:11:::1;:::i;:::-;71618:56;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;::::0;;;;-1:-1:-1;;;71618:56:0;;;-1:-1:-1;71662:10:0::1;71618:56;::::0;;::::1;::::0;71557:18:::1;:128::i;:::-;71529:156:::0;;-1:-1:-1;71529:156:0;-1:-1:-1;71529:156:0;-1:-1:-1;71716:1:0::1;71700:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;:17;;;71696:103;;;71744:43;71754:8:::0;71764:7;71773:13:::1;::::0;;;::::1;::::0;::::1;;:::i;:::-;71744:9;:43::i;:::-;71734:53;;71696:103;-1:-1:-1::0;;;;;71813:26:0;::::1;71834:4;71813:26;71809:101;;71856:42;71869:8;71879:9;71890:7;71856:12;:42::i;:::-;71939:6;:18;;;71928:7;:29;;71920:74;;;::::0;-1:-1:-1;;;71920:74:0;;20341:2:1;71920:74:0::1;::::0;::::1;20323:21:1::0;;;20360:18;;;20353:30;20419:34;20399:18;;;20392:62;20471:18;;71920:74:0::1;20313:182:1::0;71920:74:0::1;57857:1;;71109:893:::0;;;;:::o;58278:581::-;58346:22;58403:4;-1:-1:-1;;;;;58391:24:0;;;;;-1:-1:-1;;;58391:24:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;58381:34;;58431:9;58426:426;58446:15;;;58426:426;;;58484:12;;58529:4;58548;;58553:1;58548:7;;;;;-1:-1:-1;;;58548:7:0;;;;;;;;;;;;;;;;;;;;:::i;:::-;58521:35;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;58483:73;;;;58578:7;58573:232;;58626:2;58610:6;:13;:18;58606:32;;;58630:8;;;58606:32;58711:4;58703:6;58699:17;58689:27;;58771:6;58760:28;;;;;;;;;;;;:::i;:::-;58753:36;;-1:-1:-1;;;58753:36:0;;;;;;;;:::i;58573:232::-;58834:6;58821:7;58829:1;58821:10;;;;;;-1:-1:-1;;;58821:10:0;;;;;;;;;;;;;;:19;;;;58426:426;;58463:3;;;;;:::i;:::-;;;;58426:426;;;;58278:581;;;;:::o;47113:176::-;47239:17;;47216:65;;-1:-1:-1;;;47216:65:0;;-1:-1:-1;;;;;10527:15:1;;;47216:65:0;;;10509:34:1;10579:15;;;10559:18;;;10552:43;47189:7:0;;47239:17;;47216:49;;10444:18:1;;47216:65:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;47209:72;;47113:176;;;;;:::o;52206:177::-;52334:14;;52318:57;;-1:-1:-1;;;52318:57:0;;-1:-1:-1;;;;;10862:15:1;;;52318:57:0;;;10844:34:1;10914:15;;;10894:18;;;10887:43;10978:8;10966:21;;10946:18;;;10939:49;52291:7:0;;52334:14;;52318:36;;10779:18:1;;52318:57:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;52311:64;;52206:177;;;;;;:::o;53055:899::-;53186:26;53215:36;;;;53226:4;53215:36;:::i;:::-;53186:65;;53265:14;53281;53297:10;53311:25;:2;:7;;;:23;:25::i;:::-;53264:72;;;;;;53347:34;53361:6;53369;53377:3;53347:13;:34::i;:::-;53405:6;-1:-1:-1;;;;;53396:15:0;:6;-1:-1:-1;;;;;53396:15:0;;53392:555;;;53540:7;;:26;;:24;:26::i;:::-;53536:229;;;53597:7;;:19;;:17;:19::i;:::-;53587:29;;53635:37;53654:1;53657:10;53587:2;53635:18;:37::i;53536:229::-;53713:36;53717:6;53725:2;:8;;;53735:10;53747:1;53713:3;:36::i;53392:555::-;53899:36;53903:6;53911:2;:8;;;53921:10;53933:1;53899:3;:36::i;72010:94::-;2127:13;:11;:13::i;:::-;72078:7:::1;:18:::0;;-1:-1:-1;;;;;;72078:18:0::1;-1:-1:-1::0;;;;;72078:18:0;;;::::1;::::0;;;::::1;::::0;;72010:94::o;62674:338::-;62823:38;;-1:-1:-1;;;62823:38:0;;62809:11;;-1:-1:-1;;;;;62823:23:0;;;;;:38;;62855:4;;62823:38;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;62809:52;;62887:9;62880:3;:16;;62872:45;;;;-1:-1:-1;;;62872:45:0;;;;;;;:::i;:::-;62934:7;;62930:75;;62958:35;62971:5;62978:9;62989:3;62958:12;:35::i;:::-;62674:338;;;;:::o;69551:1203::-;69698:12;69712:15;69663:6;:15;;;57822:8;57803:15;:27;;57795:51;;;;-1:-1:-1;;;57795:51:0;;;;;;;:::i;:::-;69770:3:::1;69753:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;:20;;;;69745:49;;;;-1:-1:-1::0;;;69745:49:0::1;;;;;;;:::i;:::-;69805:17;69825:16;::::0;;;::::1;::::0;::::1;;:::i;:::-;69805:36:::0;-1:-1:-1;;;;;;69856:23:0;::::1;69852:81;;-1:-1:-1::0;69916:4:0::1;69852:81;69943:16;::::0;69987:13:::1;::::0;;;::::1;::::0;::::1;;:::i;:::-;-1:-1:-1::0;;;;;69970:30:0::1;::::0;-1:-1:-1;70031:1:0::1;70015:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;:17;;;70011:101;;;70058:42;70071:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;-1:-1:-1::0;;;;;70058:42:0::1;70086:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;70058:12;:42::i;:::-;70049:51;;70011:101;70144:149;70177:6;70206:4;70226:56;;;;;;;;70250:6;:11;;;;;;;;:::i;:::-;70226:56;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;::::0;;;;-1:-1:-1;;;70226:56:0;;;-1:-1:-1;70270:10:0::1;70226:56;::::0;;::::1;::::0;70144:18:::1;:149::i;:::-;70122:171:::0;;-1:-1:-1;70122:171:0;-1:-1:-1;70324:1:0::1;70308:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;:17;;;70304:103;;;70352:43;70362:8:::0;70372:7;70381:13:::1;::::0;;;::::1;::::0;::::1;;:::i;70352:43::-;70342:53;;70304:103;-1:-1:-1::0;;;;;70421:26:0;::::1;70442:4;70421:26;70417:101;;70464:42;70477:8;70487:9;70498:7;70464:12;:42::i;:::-;70535:11;;70528:18;;70573:6;:15;;;70565:4;:23;;70557:64;;;::::0;-1:-1:-1;;;70557:64:0;;15430:2:1;70557:64:0::1;::::0;::::1;15412:21:1::0;15469:2;15449:18;;;15442:30;15508;15488:18;;;15481:58;15556:18;;70557:64:0::1;15402:178:1::0;70557:64:0::1;70651:13;::::0;;;::::1;::::0;::::1;;:::i;:::-;-1:-1:-1::0;;;;;70640:24:0::1;:7;:24;;70632:69;;;::::0;-1:-1:-1;;;70632:69:0;;17228:2:1;70632:69:0::1;::::0;::::1;17210:21:1::0;;;17247:18;;;17240:30;17306:34;17286:18;;;17279:62;17358:18;;70632:69:0::1;17200:182:1::0;70632:69:0::1;-1:-1:-1::0;;70712:11:0::1;:34;;;;57857:1;;;69551:1203:::0;;;;:::o;3140:201::-;2127:13;:11;:13::i;:::-;-1:-1:-1;;;;;3229:22:0;::::1;3221:73;;;::::0;-1:-1:-1;;;3221:73:0;;13928:2:1;3221:73:0::1;::::0;::::1;13910:21:1::0;13967:2;13947:18;;;13940:30;14006:34;13986:18;;;13979:62;-1:-1:-1;;;14057:18:1;;;14050:36;14103:19;;3221:73:0::1;13900:228:1::0;3221:73:0::1;3305:28;3324:8;3305:18;:28::i;:::-;3140:201:::0;:::o;61124:168::-;61237:12;;;61197;61237;;;;;;;;;-1:-1:-1;;;;;61215:7:0;;;61230:5;;61215:35;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;61196:54;;;61269:7;61261:23;;;;-1:-1:-1;;;61261:23:0;;15099:2:1;61261:23:0;;;15081:21:1;15138:1;15118:18;;;15111:29;-1:-1:-1;;;15156:18:1;;;15149:33;15199:18;;61261:23:0;15071:152:1;9797:326:0;9905:14;;;10010:17;:4;9905:14;10010;:17::i;:::-;10001:26;-1:-1:-1;10044:24:0;:4;8291:2;10044:13;:24::i;:::-;10038:30;-1:-1:-1;10088:27:0;8499:20;8386:1;8291:2;8499:20;:::i;:::-;10088:4;;:14;:27::i;:::-;10079:36;;9797:326;;;;;:::o;52389:162::-;52504:32;52516:6;52524;52532:3;52504:11;:32::i;:::-;-1:-1:-1;;;;;52490:46:0;:10;-1:-1:-1;;;;;52490:46:0;;52481:62;;;;-1:-1:-1;;;52481:62:0;;16481:2:1;52481:62:0;;;16463:21:1;16520:1;16500:18;;;16493:29;-1:-1:-1;;;16538:18:1;;;16531:32;16580:18;;52481:62:0;16453:151:1;64647:797:0;64843:5;;-1:-1:-1;;;;;64834:14:0;;;64843:5;;64834:14;:48;;;;;64877:5;64852:21;:30;;64834:48;64830:607;;;64937:5;;;;;;;;;-1:-1:-1;;;;;64937:5:0;-1:-1:-1;;;;;64930:21:0;;64959:5;64930:37;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;65024:5:0;;65017:40;;-1:-1:-1;;;65017:40:0;;-1:-1:-1;;;;;12607:32:1;;;65017:40:0;;;12589:51:1;12656:18;;;12649:34;;;65024:5:0;;;;-1:-1:-1;65017:22:0;;-1:-1:-1;12562:18:1;;;-1:-1:-1;65017:40:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;-1:-1:-1;65072:11:0;:19;;;64830:607;;;-1:-1:-1;;;;;65113:22:0;;65130:4;65113:22;65109:328;;;65244:37;65257:5;65264:9;65275:5;65244:12;:37::i;:::-;65109:328;;;65343:48;65360:5;65367;65374:9;65385:5;65343:16;:48::i;:::-;65406:11;:19;-1:-1:-1;;;64647:797:0:o;8993:140::-;9061:4;8499:20;8386:1;8291:2;8499:20;:::i;:::-;8291:2;8499:20;8386:1;8291:2;8499:20;:::i;:::-;8612:23;;;;:::i;:::-;8773:24;;;;:::i;:::-;9085:4;:11;:40;;9078:47;;8993:140;;;:::o;10677:151::-;10738:12;10770:50;8499:20;8386:1;8291:2;8499:20;:::i;:::-;;8386:1;8291:2;8499:20;:::i;:::-;10794:4;:11;:25;;;;:::i;:::-;10770:4;;:50;:10;:50::i;65452:1236::-;65640:15;65657:16;65688:15;65714:14;65772:27;:4;:9;;;:25;:27::i;:::-;65741:58;;-1:-1:-1;65741:58:0;-1:-1:-1;65741:58:0;-1:-1:-1;65816:12:0;;;;65812:867;;65879:316;65925:232;;;;;;;;65974:7;-1:-1:-1;;;;;65925:232:0;;;;;66014:8;-1:-1:-1;;;;;65925:232:0;;;;;66050:7;65925:232;;;;;;66088:6;65925:232;;;;66128:9;-1:-1:-1;;;;;65925:232:0;;;;66176:4;65879:27;:316::i;:::-;65869:326;;65812:867;;;66268:17;;-1:-1:-1;;;;;66268:17:0;66260:66;;;;-1:-1:-1;;;66260:66:0;;;;;;;:::i;:::-;66351:316;66397:232;;;;;;;;66446:7;-1:-1:-1;;;;;66397:232:0;;;;;66486:8;-1:-1:-1;;;;;66397:232:0;;;;;66522:7;66397:232;;;;;;66560:6;66397:232;;;;66600:9;-1:-1:-1;;;;;66397:232:0;;;;66648:4;66351:27;:316::i;:::-;66341:326;;65812:867;65452:1236;;;;;;;;:::o;2406:132::-;2287:7;2314:6;-1:-1:-1;;;;;2314:6:0;866:10;2470:23;2462:68;;;;-1:-1:-1;;;2462:68:0;;18613:2:1;2462:68:0;;;18595:21:1;;;18632:18;;;18625:30;18691:34;18671:18;;;18664:62;18743:18;;2462:68:0;18585:182:1;3501:191:0;3575:16;3594:6;;-1:-1:-1;;;;;3611:17:0;;;-1:-1:-1;;;;;;3611:17:0;;;;;;3644:40;;3594:6;;;;;;;3644:40;;3575:16;3644:40;3501:191;;:::o;66696:1773::-;66812:12;;;66890:10;66966:4;66983:1479;67011:21;67035:28;:4;:9;;;:26;:28::i;:::-;67011:52;;67078:15;67108:14;67182:27;:4;:9;;;:25;:27::i;:::-;67257:212;;;;;;;;-1:-1:-1;;;;;67257:212:0;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;67257:212:0;;;;;;67448:4;67257:212;;;;67151:58;;-1:-1:-1;67151:58:0;;-1:-1:-1;67151:58:0;-1:-1:-1;67257:212:0;67224:30;;67532:12;67528:487;;67612:103;67662:6;67691:5;67612:27;:103::i;:::-;67593:122;-1:-1:-1;67593:122:0;-1:-1:-1;67528:487:0;;;67800:17;;-1:-1:-1;;;;;67800:17:0;67792:66;;;;-1:-1:-1;;;67792:66:0;;;;;;;:::i;:::-;67896:103;67946:6;67975:5;67896:27;:103::i;:::-;67877:122;-1:-1:-1;67877:122:0;-1:-1:-1;67528:487:0;68033:8;68029:61;;;68069:5;68062:12;;68029:61;68115:5;68104:16;;68197;68193:258;;;68324:9;;68250:4;;-1:-1:-1;68324:21:0;;:19;:21::i;:::-;68312:33;;68381:7;;-1:-1:-1;68381:7:0;;68193:258;;;68430:5;;;;;;;68193:258;66983:1479;;;;;;;;66696:1773;;;;;;;:::o;68775:421::-;68896:7;;68864:15;;-1:-1:-1;;;;;68896:7:0;68892:74;;-1:-1:-1;68941:13:0;68934:20;;68892:74;68976:16;69020:5;68995:22;;;;:13;:22;:::i;:::-;:30;;;;:::i;:::-;68976:49;-1:-1:-1;69046:24:0;68976:49;69046:13;:24;:::i;:::-;69036:34;-1:-1:-1;69085:12:0;;69081:83;;69134:7;;69114:38;;69127:5;;-1:-1:-1;;;;;69134:7:0;69143:8;69114:12;:38::i;:::-;69174:14;68775:421;;;;;:::o;59940:316::-;60105:59;;;-1:-1:-1;;;;;12607:32:1;;;60105:59:0;;;12589:51:1;12656:18;;;;12649:34;;;60105:59:0;;;;;;;;;;12562:18:1;;;;60105:59:0;;;;;;;-1:-1:-1;;;;;60105:59:0;-1:-1:-1;;;60105:59:0;;;60094:71;;-1:-1:-1;;;;60094:10:0;;;;:71;;60105:59;60094:71;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60058:107;;;;60184:7;:57;;;;-1:-1:-1;60196:11:0;;:16;;:44;;;60227:4;60216:24;;;;;;;;;;;;:::i;:::-;60176:72;;;;-1:-1:-1;;;60176:72:0;;17589:2:1;60176:72:0;;;17571:21:1;17628:1;17608:18;;;17601:29;-1:-1:-1;;;17646:18:1;;;17639:32;17688:18;;60176:72:0;17561:151:1;60176:72:0;59940:316;;;;;:::o;68477:290::-;68589:7;;68558:14;;-1:-1:-1;;;;;68589:7:0;:21;;:36;;-1:-1:-1;68614:11:0;;;;68589:36;68585:88;;;-1:-1:-1;68649:12:0;68642:19;;68585:88;68716:14;68724:6;68716:5;:14;:::i;:::-;68692:39;;:20;:12;68707:5;68692:20;:::i;:::-;:39;;;;:::i;:::-;:43;;68734:1;68692:43;:::i;7184:426::-;7263:7;7306:6;7291:11;7306:6;7300:2;7291:11;:::i;:::-;:21;;7283:52;;;;-1:-1:-1;;;7283:52:0;;21116:2:1;7283:52:0;;;21098:21:1;21155:2;21135:18;;;21128:30;-1:-1:-1;;;21174:18:1;;;21167:48;21232:18;;7283:52:0;21088:168:1;7283:52:0;7371:11;:6;7380:2;7371:11;:::i;:::-;7354:6;:13;:28;;7346:62;;;;-1:-1:-1;;;7346:62:0;;18974:2:1;7346:62:0;;;18956:21:1;19013:2;18993:18;;;18986:30;-1:-1:-1;;;19032:18:1;;;19025:51;19093:18;;7346:62:0;18946:171:1;7346:62:0;-1:-1:-1;7500:30:0;7516:4;7500:30;7494:37;-1:-1:-1;;;7490:71:0;;;7184:426::o;7618:375::-;7696:6;7737;7723:10;7737:6;7732:1;7723:10;:::i;:::-;:20;;7715:50;;;;-1:-1:-1;;;7715:50:0;;14753:2:1;7715:50:0;;;14735:21:1;14792:2;14772:18;;;14765:30;-1:-1:-1;;;14811:18:1;;;14804:47;14868:18;;7715:50:0;14725:167:1;7715:50:0;7801:10;:6;7810:1;7801:10;:::i;:::-;7784:6;:13;:27;;7776:60;;;;-1:-1:-1;;;7776:60:0;;18264:2:1;7776:60:0;;;18246:21:1;18303:2;18283:18;;;18276:30;-1:-1:-1;;;18322:18:1;;;18315:50;18382:18;;7776:60:0;18236:170:1;7776:60:0;-1:-1:-1;7917:29:0;7933:3;7917:29;7911:36;;7618:375::o;59275:367::-;59480:69;;;-1:-1:-1;;;;;11257:15:1;;;59480:69:0;;;11239:34:1;11309:15;;;11289:18;;;11282:43;11341:18;;;;11334:34;;;59480:69:0;;;;;;;;;;11174:18:1;;;;59480:69:0;;;;;;;-1:-1:-1;;;;;59480:69:0;-1:-1:-1;;;59480:69:0;;;59469:81;;-1:-1:-1;;;;59469:10:0;;;;:81;;59480:69;59469:81;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;59420:130;;;;59569:7;:57;;;;-1:-1:-1;59581:11:0;;:16;;:44;;;59612:4;59601:24;;;;;;;;;;;;:::i;:::-;59561:73;;;;-1:-1:-1;;;59561:73:0;;19324:2:1;59561:73:0;;;19306:21:1;19363:1;19343:18;;;19336:29;-1:-1:-1;;;19381:18:1;;;19374:33;19424:18;;59561:73:0;19296:152:1;59561:73:0;59275:367;;;;;;:::o;4105:3071::-;4231:12;4280:7;4264:12;4280:7;4274:2;4264:12;:::i;:::-;:23;;4256:50;;;;-1:-1:-1;;;4256:50:0;;;;;;;:::i;:::-;4345:6;4325:16;4334:7;4345:6;4325:16;:::i;:::-;:26;;4317:53;;;;-1:-1:-1;;;4317:53:0;;;;;;;:::i;:::-;4406:16;4415:7;4406:6;:16;:::i;:::-;4389:6;:13;:33;;4381:63;;;;-1:-1:-1;;;4381:63:0;;19995:2:1;4381:63:0;;;19977:21:1;20034:2;20014:18;;;20007:30;-1:-1:-1;;;20053:18:1;;;20046:47;20110:18;;4381:63:0;19967:167:1;4381:63:0;4457:22;4523:15;;4556:2137;;;;6849:4;6843:11;6830:24;;7050:1;7039:9;7032:20;7104:4;7093:9;7089:20;7083:4;7076:34;4516:2613;;4556:2137;4753:4;4747:11;4734:24;;5458:2;5449:7;5445:16;5866:9;5859:17;5853:4;5849:28;5837:9;5826;5822:25;5818:60;5919:7;5915:2;5911:16;6192:6;6178:9;6171:17;6165:4;6161:28;6149:9;6141:6;6137:22;6133:57;6129:70;5951:470;6230:3;6226:2;6223:11;5951:470;;;6388:9;;6377:21;;6276:4;6268:13;;;;6313;5951:470;;;-1:-1:-1;;6445:26:0;;;6669:2;6652:11;-1:-1:-1;;6648:25:0;6642:4;6635:39;-1:-1:-1;4516:2613:0;-1:-1:-1;7159:9:0;4105:3071;-1:-1:-1;;;;4105:3071:0:o;55125:875::-;55267:15;55305:16;55324:56;55336:6;:15;;;55353:6;:14;;;55369:6;:10;;;55324:11;:56::i;:::-;55305:75;;55413:6;:14;;;-1:-1:-1;;;;;55395:32:0;:6;:15;;;-1:-1:-1;;;;;55395:32:0;;55391:602;;;55569:8;-1:-1:-1;;;;;55556:37:0;;55612:6;:16;;;55638:6;:13;;;55654:6;55690:4;55679:16;;;;;;;;:::i;:::-;;;;;;;;;;;;;55556:154;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;-1:-1:-1;55542:168:0;-1:-1:-1;55391:602:0;;;55839:8;-1:-1:-1;;;;;55826:37:0;;55882:6;:16;;;55908:6;:13;;;-1:-1:-1;;55961:4:0;55950:16;;;;;;;;:::i;:::-;;;;;;;;;;;;;55826:155;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;55812:169;55125:875;-1:-1:-1;;;;;55125:875:0:o;50979:698::-;51121:15;51150:12;51164;51180:66;51199:6;:14;;;51215:6;:15;;;51232:6;:13;;;51180:18;:66::i;:::-;51149:97;;;;51261:28;:4;:9;;;:26;:28::i;:::-;51257:308;;;51374:9;;:21;;:19;:21::i;:::-;51362:33;;51410:36;51429:4;51435;51362;51410:18;:36::i;:::-;;;51257:308;;;51510:43;51514:6;:14;;;51530:4;:10;;;51542:4;51548;51510:3;:43::i;:::-;51585:84;51598:4;51604:6;:14;;;51620:6;:15;;;51637:6;:13;;;51652:6;:16;;;51585:12;:84::i;56008:972::-;56135:12;56149:15;56187:16;56206:56;56218:6;:15;;;56235:6;:14;;;56251:6;:10;;;56206:11;:56::i;:::-;56187:75;;56294:6;:15;;;-1:-1:-1;;;;;56277:32:0;:6;:14;;;-1:-1:-1;;;;;56277:32:0;;56273:700;;;56381:8;-1:-1:-1;;;;;56368:30:0;;56417:6;:16;;;56443:6;:13;;;-1:-1:-1;;56496:101:0;;;;;;;;56537:6;:14;;;56553:6;:10;;;56565:6;:15;;;56520:61;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;56496:101;;;;56590:5;-1:-1:-1;;;;;56496:101:0;;;;56485:113;;;;;;;;:::i;:::-;;;;;;;;;;;;;56368:245;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;56350:263;;-1:-1:-1;56350:263:0;-1:-1:-1;56273:700:0;;;56670:13;56730:8;-1:-1:-1;;;;;56717:30:0;;56766:6;:16;;;56792:6;:13;;;56808:6;56844:101;;;;;;;;56885:6;:14;;;56901:6;:10;;;56913:6;:15;;;56868:61;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;56844:101;;;;56938:5;-1:-1:-1;;;;;56844:101:0;;;;56833:113;;;;;;;;:::i;:::-;;;;;;;;;;;;;56717:244;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;-1:-1:-1;56698:263:0;-1:-1:-1;;56273:700:0;56008:972;;;;;;:::o;50476:495::-;50603:12;50617:15;50646:16;50664:12;50680:67;50700:6;:14;;;50716:6;:15;;;50733:6;:13;;;50680:19;:67::i;:::-;50645:102;;;;50785:47;50789:6;:14;;;50805:5;50812:4;50818:6;:13;;;50785:3;:47::i;:::-;50850:6;:13;;;50843:20;;50884:79;50897:4;50903:6;:14;;;50919:6;:15;;;50936:8;50946:6;:16;;;50884:12;:79::i;:::-;50874:89;;50476:495;;;;;;;:::o;49491:385::-;49630:16;49648:12;49674:17;49693:18;49713:10;49725:13;49742:31;49755:7;49764:8;49742:12;:31::i;:::-;49673:100;;;;;;;;49795:50;49807:9;49818;49829:10;49841:3;49795:11;:50::i;:::-;49784:61;49863:5;;-1:-1:-1;49491:385:0;-1:-1:-1;;;;;;;49491:385:0:o;49884:584::-;50018:15;50046:18;50066;50098:8;-1:-1:-1;;;;;50088:18:0;:7;-1:-1:-1;;;;;50088:18:0;;:70;;50136:9;50155:1;50088:70;;;50118:1;50122:9;50088:70;50045:113;;;;50169:20;50199:8;-1:-1:-1;;;;;50192:26:0;;50219:9;50192:37;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;50320:12;;;50330:1;50320:12;;;;;;;;;-1:-1:-1;;;50240:103:0;;;50169:60;;-1:-1:-1;;;;;;50240:30:0;;;;;:103;;50285:10;;50297;;50309:9;;50240:103;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;50376:37:0;;-1:-1:-1;;;50376:37:0;;50354:19;;-1:-1:-1;;;;;;50376:26:0;;;-1:-1:-1;50376:26:0;;:37;;50403:9;;50376:37;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;50354:59;-1:-1:-1;50434:26:0;50448:12;50354:59;50434:26;:::i;:::-;50424:36;49884:584;-1:-1:-1;;;;;;;;;;49884:584:0:o;49096:387::-;49235:17;49254:12;49280:17;49299:18;49319:10;49331:13;49348:31;49361:7;49370:8;49348:12;:31::i;:::-;49279:100;;;;;;;;49402:50;49415:8;49425:9;49436:10;49448:3;49402:12;:50::i;48583:505::-;48697:13;48743:1;48731:9;:13;48723:75;;;;-1:-1:-1;;;48723:75:0;;14335:2:1;48723:75:0;;;14317:21:1;14374:2;14354:18;;;14347:30;-1:-1:-1;;;;;;;;;;;14393:18:1;;;14386:62;-1:-1:-1;;;14464:18:1;;;14457:47;14521:19;;48723:75:0;14307:239:1;48723:75:0;48829:1;48817:9;:13;:31;;;;;48847:1;48834:10;:14;48817:31;48809:89;;;;-1:-1:-1;;;48809:89:0;;;;;;;:::i;:::-;48909:17;48929:21;48941:9;48929;:21;:::i;:::-;:29;;48953:5;48929:29;:::i;:::-;48909:49;-1:-1:-1;48969:19:0;49019:11;49027:3;49019:5;:11;:::i;:::-;48991:40;;48992:22;49005:9;48992:10;:22;:::i;:::-;48991:40;;;;:::i;:::-;48969:62;-1:-1:-1;49053:23:0;48969:62;49053:9;:23;:::i;:::-;:27;;49079:1;49053:27;:::i;:::-;49042:38;48583:505;-1:-1:-1;;;;;;;48583:505:0:o;47906:556::-;48020:14;48066:1;48055:8;:12;48047:73;;;;-1:-1:-1;;;48047:73:0;;16811:2:1;48047:73:0;;;16793:21:1;16850:2;16830:18;;;16823:30;-1:-1:-1;;;;;;;;;;;16869:18:1;;;16862:62;-1:-1:-1;;;16940:18:1;;;16933:46;16996:19;;48047:73:0;16783:238:1;48047:73:0;48151:1;48139:9;:13;:31;;;;;48169:1;48156:10;:14;48139:31;48131:89;;;;-1:-1:-1;;;48131:89:0;;;;;;;:::i;:::-;48231:23;48269:11;48277:3;48269:5;:11;:::i;:::-;48257:24;;;;:8;:24;:::i;:::-;48231:50;-1:-1:-1;48292:17:0;48312:28;48330:10;48231:50;48312:28;:::i;:::-;48292:48;-1:-1:-1;48351:19:0;48393:15;48373:17;:9;48385:5;48373:17;:::i;:::-;:35;;;;:::i;:::-;48351:57;-1:-1:-1;48431:23:0;48351:57;48431:9;:23;:::i;:::-;48419:35;47906:556;-1:-1:-1;;;;;;;;47906:556:0:o;14:134:1:-;82:20;;111:31;82:20;111:31;:::i;:::-;63:85;;;:::o;153:173::-;222:5;267:3;258:6;253:3;249:16;245:26;242:2;;;288:5;281;274:20;242:2;-1:-1:-1;314:6:1;232:94;-1:-1:-1;232:94:1:o;331:177::-;410:13;;-1:-1:-1;;;;;452:31:1;;442:42;;432:2;;498:1;495;488:12;513:257;572:6;625:2;613:9;604:7;600:23;596:32;593:2;;;646:6;638;631:22;593:2;690:9;677:23;709:31;734:5;709:31;:::i;775:261::-;845:6;898:2;886:9;877:7;873:23;869:32;866:2;;;919:6;911;904:22;866:2;956:9;950:16;975:31;1000:5;975:31;:::i;1041:398::-;1109:6;1117;1170:2;1158:9;1149:7;1145:23;1141:32;1138:2;;;1191:6;1183;1176:22;1138:2;1235:9;1222:23;1254:31;1279:5;1254:31;:::i;:::-;1304:5;-1:-1:-1;1361:2:1;1346:18;;1333:32;1374:33;1333:32;1374:33;:::i;:::-;1426:7;1416:17;;;1128:311;;;;;:::o;1444:578::-;1520:6;1528;1536;1589:2;1577:9;1568:7;1564:23;1560:32;1557:2;;;1610:6;1602;1595:22;1557:2;1654:9;1641:23;1673:31;1698:5;1673:31;:::i;:::-;1723:5;-1:-1:-1;1780:2:1;1765:18;;1752:32;1793:33;1752:32;1793:33;:::i;:::-;1845:7;-1:-1:-1;1904:2:1;1889:18;;1876:32;1952:8;1939:22;;1927:35;;1917:2;;1981:6;1973;1966:22;1917:2;2009:7;1999:17;;;1547:475;;;;;:::o;2027:466::-;2104:6;2112;2120;2173:2;2161:9;2152:7;2148:23;2144:32;2141:2;;;2194:6;2186;2179:22;2141:2;2238:9;2225:23;2257:31;2282:5;2257:31;:::i;:::-;2307:5;-1:-1:-1;2359:2:1;2344:18;;2331:32;;-1:-1:-1;2415:2:1;2400:18;;2387:32;2428:33;2387:32;2428:33;:::i;2498:676::-;2595:6;2603;2656:2;2644:9;2635:7;2631:23;2627:32;2624:2;;;2677:6;2669;2662:22;2624:2;2722:9;2709:23;-1:-1:-1;;;;;2792:2:1;2784:6;2781:14;2778:2;;;2813:6;2805;2798:22;2778:2;2856:6;2845:9;2841:22;2831:32;;2901:7;2894:4;2890:2;2886:13;2882:27;2872:2;;2928:6;2920;2913:22;2872:2;2973;2960:16;2999:2;2991:6;2988:14;2985:2;;;3020:6;3012;3005:22;2985:2;3078:7;3073:2;3063:6;3060:1;3056:14;3052:2;3048:23;3044:32;3041:45;3038:2;;;3104:6;3096;3089:22;3038:2;3140;3132:11;;;;;3162:6;;-1:-1:-1;2614:560:1;;-1:-1:-1;;;;2614:560:1:o;3179:297::-;3246:6;3299:2;3287:9;3278:7;3274:23;3270:32;3267:2;;;3320:6;3312;3305:22;3267:2;3357:9;3351:16;3410:5;3403:13;3396:21;3389:5;3386:32;3376:2;;3437:6;3429;3422:22;3481:675;3561:6;3614:2;3602:9;3593:7;3589:23;3585:32;3582:2;;;3635:6;3627;3620:22;3582:2;3673:9;3667:16;-1:-1:-1;;;;;3698:6:1;3695:30;3692:2;;;3743:6;3735;3728:22;3692:2;3771:22;;3824:4;3816:13;;3812:27;-1:-1:-1;3802:2:1;;3858:6;3850;3843:22;3802:2;3892;3886:9;3917:48;3933:31;3961:2;3933:31;:::i;:::-;3917:48;:::i;:::-;3988:2;3981:5;3974:17;4028:7;4023:2;4018;4014;4010:11;4006:20;4003:33;4000:2;;;4054:6;4046;4039:22;4000:2;4072:54;4123:2;4118;4111:5;4107:14;4102:2;4098;4094:11;4072:54;:::i;4161:394::-;4256:6;4309:2;4297:9;4288:7;4284:23;4280:32;4277:2;;;4330:6;4322;4315:22;4277:2;4375:9;4362:23;-1:-1:-1;;;;;4400:6:1;4397:30;4394:2;;;4445:6;4437;4430:22;4394:2;4473:76;4541:7;4532:6;4521:9;4517:22;4473:76;:::i;:::-;4463:86;4267:288;-1:-1:-1;;;;4267:288:1:o;4560:1118::-;4653:6;4684:2;4727;4715:9;4706:7;4702:23;4698:32;4695:2;;;4748:6;4740;4733:22;4695:2;4793:9;4780:23;-1:-1:-1;;;;;4863:2:1;4855:6;4852:14;4849:2;;;4884:6;4876;4869:22;4849:2;4912:22;;;;4968:4;4950:16;;;4946:27;4943:2;;;4991:6;4983;4976:22;4943:2;5022:22;;:::i;:::-;5082:2;5069:16;5110:2;5100:8;5097:16;5094:2;;;5131:6;5123;5116:22;5094:2;5159:17;;;-1:-1:-1;5207:4:1;5199:13;;5195:27;-1:-1:-1;5185:2:1;;5241:6;5233;5226:22;5185:2;5282;5269:16;5307:48;5323:31;5351:2;5323:31;:::i;5307:48::-;5378:2;5371:5;5364:17;5418:7;5413:2;5408;5404;5400:11;5396:20;5393:33;5390:2;;;5444:6;5436;5429:22;5390:2;5504;5499;5495;5491:11;5486:2;5479:5;5475:14;5462:45;5527:14;;;5523:23;;5516:39;;;5564:20;;5616:31;5635:11;;;5616:31;:::i;:::-;5611:2;5604:5;5600:14;5593:55;5667:5;5657:15;;;;;;4664:1014;;;;:::o;6082:605::-;6177:6;6185;6193;6201;6254:3;6242:9;6233:7;6229:23;6225:33;6222:2;;;6276:6;6268;6261:22;6222:2;6304:40;6334:9;6304:40;:::i;:::-;6294:50;;6363:49;6408:2;6397:9;6393:18;6363:49;:::i;:::-;6353:59;;6455:2;6444:9;6440:18;6434:25;6468:30;6492:5;6468:30;:::i;:::-;6567:2;6552:18;;6546:25;6517:5;;-1:-1:-1;6615:10:1;6602:24;;6590:37;;6580:2;;6646:6;6638;6631:22;6580:2;6212:475;;;;-1:-1:-1;6212:475:1;;-1:-1:-1;;6212:475:1:o;6692:306::-;6751:6;6804:2;6792:9;6783:7;6779:23;6775:32;6772:2;;;6825:6;6817;6810:22;6772:2;6856:23;;-1:-1:-1;;;;;6908:31:1;;6898:42;;6888:2;;6959:6;6951;6944:22;7003:255;7061:6;7114:2;7102:9;7093:7;7089:23;7085:32;7082:2;;;7135:6;7127;7120:22;7082:2;7179:9;7166:23;7198:30;7222:5;7198:30;:::i;7263:194::-;7333:6;7386:2;7374:9;7365:7;7361:23;7357:32;7354:2;;;7407:6;7399;7392:22;7354:2;-1:-1:-1;7435:16:1;;7344:113;-1:-1:-1;7344:113:1:o;7462:325::-;7530:6;7538;7591:2;7579:9;7570:7;7566:23;7562:32;7559:2;;;7612:6;7604;7597:22;7559:2;7653:9;7640:23;7630:33;;7713:2;7702:9;7698:18;7685:32;7726:31;7751:5;7726:31;:::i;7792:255::-;7871:6;7879;7932:2;7920:9;7911:7;7907:23;7903:32;7900:2;;;7953:6;7945;7938:22;7900:2;-1:-1:-1;;7981:16:1;;8037:2;8022:18;;;8016:25;7981:16;;8016:25;;-1:-1:-1;7890:157:1:o;8052:777::-;8140:6;8148;8156;8164;8217:2;8205:9;8196:7;8192:23;8188:32;8185:2;;;8238:6;8230;8223:22;8185:2;8279:9;8266:23;8256:33;;8336:2;8325:9;8321:18;8308:32;8298:42;;8391:2;8380:9;8376:18;8363:32;-1:-1:-1;;;;;8455:2:1;8447:6;8444:14;8441:2;;;8476:6;8468;8461:22;8441:2;8519:6;8508:9;8504:22;8494:32;;8564:7;8557:4;8553:2;8549:13;8545:27;8535:2;;8591:6;8583;8576:22;8535:2;8636;8623:16;8662:2;8654:6;8651:14;8648:2;;;8683:6;8675;8668:22;8648:2;8733:7;8728:2;8719:6;8715:2;8711:15;8707:24;8704:37;8701:2;;;8759:6;8751;8744:22;8701:2;8175:654;;;;-1:-1:-1;;8795:2:1;8787:11;;-1:-1:-1;;;8175:654:1:o;8834:257::-;8875:3;8913:5;8907:12;8940:6;8935:3;8928:19;8956:63;9012:6;9005:4;9000:3;8996:14;8989:4;8982:5;8978:16;8956:63;:::i;:::-;9073:2;9052:15;-1:-1:-1;;9048:29:1;9039:39;;;;9080:4;9035:50;;8883:208;-1:-1:-1;;8883:208:1:o;9096:431::-;9349:2;9345:15;;;-1:-1:-1;;9341:24:1;;;9329:37;;9422:3;9400:16;;;;-1:-1:-1;;;;;;9396:41:1;9391:2;9382:12;;9375:63;9472:15;;9468:24;9463:2;9454:12;;9447:46;9518:2;9509:12;;9269:258::o;9532:273::-;9715:6;9707;9702:3;9689:33;9671:3;9741:16;;9766:15;;;9741:16;9679:126;-1:-1:-1;9679:126:1:o;9810:274::-;9939:3;9977:6;9971:13;9993:53;10039:6;10034:3;10027:4;10019:6;10015:17;9993:53;:::i;:::-;10062:16;;;;;9947:137;-1:-1:-1;;9947:137:1:o;10089:203::-;-1:-1:-1;;;;;10253:32:1;;;;10235:51;;10223:2;10208:18;;10190:102::o;11379:510::-;-1:-1:-1;;;;;11621:32:1;;11603:51;;-1:-1:-1;;;;;11690:32:1;;11685:2;11670:18;;11663:60;11770:1;11759:21;;;11754:2;11739:18;;11732:49;11710:3;11812:2;11797:18;;11790:31;;;-1:-1:-1;;11838:45:1;;11863:19;;11855:6;11838:45;:::i;:::-;11830:53;11593:296;-1:-1:-1;;;;;;11593:296:1:o;12694:803::-;12854:4;12883:2;12923;12912:9;12908:18;12953:2;12942:9;12935:21;12976:6;13011;13005:13;13042:6;13034;13027:22;13080:2;13069:9;13065:18;13058:25;;13142:2;13132:6;13129:1;13125:14;13114:9;13110:30;13106:39;13092:53;;13180:2;13172:6;13168:15;13201:4;13214:254;13228:6;13225:1;13222:13;13214:254;;;13321:2;13317:7;13305:9;13297:6;13293:22;13289:36;13284:3;13277:49;13349:39;13381:6;13372;13366:13;13349:39;:::i;:::-;13339:49;-1:-1:-1;13446:12:1;;;;13411:15;;;;13250:1;13243:9;13214:254;;;-1:-1:-1;13485:6:1;;12863:634;-1:-1:-1;;;;;;;12863:634:1:o;13502:219::-;13651:2;13640:9;13633:21;13614:4;13671:44;13711:2;13700:9;13696:18;13688:6;13671:44;:::i;15585:346::-;15787:2;15769:21;;;15826:2;15806:18;;;15799:30;-1:-1:-1;;;15860:2:1;15845:18;;15838:52;15922:2;15907:18;;15759:172::o;15936:338::-;16138:2;16120:21;;;16177:2;16157:18;;;16150:30;-1:-1:-1;;;16211:2:1;16196:18;;16189:44;16265:2;16250:18;;16110:164::o;17717:340::-;17919:2;17901:21;;;17958:2;17938:18;;;17931:30;-1:-1:-1;;;17992:2:1;17977:18;;17970:46;18048:2;18033:18;;17891:166::o;19453:335::-;19655:2;19637:21;;;19694:2;19674:18;;;19667:30;-1:-1:-1;;;19728:2:1;19713:18;;19706:41;19779:2;19764:18;;19627:161::o;20500:409::-;20702:2;20684:21;;;20741:2;20721:18;;;20714:30;-1:-1:-1;;;;;;;;;;;20775:2:1;20760:18;;20753:62;-1:-1:-1;;;20846:2:1;20831:18;;20824:43;20899:3;20884:19;;20674:235::o;21261:340::-;21463:2;21445:21;;;21502:2;21482:18;;;21475:30;-1:-1:-1;;;21536:2:1;21521:18;;21514:46;21592:2;21577:18;;21435:166::o;21606:472::-;21803:2;21792:9;21785:21;21766:4;21841:6;21835:13;21884:4;21879:2;21868:9;21864:18;21857:32;21912:50;21958:2;21947:9;21943:18;21929:12;21912:50;:::i;:::-;22022:2;22010:15;;;;22004:22;-1:-1:-1;;;;;22000:48:1;21993:4;21978:20;;;;21971:78;;;;-1:-1:-1;21898:64:1;;21775:303;-1:-1:-1;21775:303:1:o;22518:458::-;22749:6;22738:9;22731:25;22792:6;22787:2;22776:9;22772:18;22765:34;22864:1;22860;22855:3;22851:11;22847:19;22839:6;22835:32;22830:2;22819:9;22815:18;22808:60;22904:3;22899:2;22888:9;22884:18;22877:31;22712:4;22925:45;22965:3;22954:9;22950:19;22942:6;22925:45;:::i;23414:533::-;23491:4;23497:6;23557:11;23544:25;23651:2;23647:7;23636:8;23620:14;23616:29;23612:43;23592:18;23588:68;23578:2;;23673:4;23667;23660:18;23578:2;23703:33;;23755:20;;;-1:-1:-1;;;;;;23787:30:1;;23784:2;;;23833:4;23827;23820:18;23784:2;23869:4;23857:17;;-1:-1:-1;23900:14:1;23896:27;;;23886:38;;23883:2;;;23937:1;23934;23927:12;23883:2;23508:439;;;;;:::o;23952:257::-;24024:4;24018:11;;;24056:17;;-1:-1:-1;;;;;24088:34:1;;24124:22;;;24085:62;24082:2;;;24150:18;;:::i;:::-;24186:4;24179:24;23998:211;:::o;24214:275::-;24285:2;24279:9;24350:2;24331:13;;-1:-1:-1;;24327:27:1;24315:40;;-1:-1:-1;;;;;24370:34:1;;24406:22;;;24367:62;24364:2;;;24432:18;;:::i;:::-;24468:2;24461:22;24259:230;;-1:-1:-1;24259:230:1:o;24494:186::-;24542:4;-1:-1:-1;;;;;24567:6:1;24564:30;24561:2;;;24597:18;;:::i;:::-;-1:-1:-1;24663:2:1;24642:15;-1:-1:-1;;24638:29:1;24669:4;24634:40;;24551:129::o;24685:128::-;24725:3;24756:1;24752:6;24749:1;24746:13;24743:2;;;24762:18;;:::i;:::-;-1:-1:-1;24798:9:1;;24733:80::o;24818:217::-;24858:1;24884;24874:2;;-1:-1:-1;;;24909:31:1;;24963:4;24960:1;24953:15;24991:4;24916:1;24981:15;24874:2;-1:-1:-1;25020:9:1;;24864:171::o;25040:168::-;25080:7;25146:1;25142;25138:6;25134:14;25131:1;25128:21;25123:1;25116:9;25109:17;25105:45;25102:2;;;25153:18;;:::i;:::-;-1:-1:-1;25193:9:1;;25092:116::o;25213:217::-;25252:4;25281:6;25337:10;;;;25307;;25359:12;;;25356:2;;;25374:18;;:::i;:::-;25411:13;;25261:169;-1:-1:-1;;;25261:169:1:o;25435:125::-;25475:4;25503:1;25500;25497:8;25494:2;;;25508:18;;:::i;:::-;-1:-1:-1;25545:9:1;;25484:76::o;25565:258::-;25637:1;25647:113;25661:6;25658:1;25655:13;25647:113;;;25737:11;;;25731:18;25718:11;;;25711:39;25683:2;25676:10;25647:113;;;25778:6;25775:1;25772:13;25769:2;;;-1:-1:-1;;25813:1:1;25795:16;;25788:27;25618:205::o;25828:135::-;25867:3;-1:-1:-1;;25888:17:1;;25885:2;;;25908:18;;:::i;:::-;-1:-1:-1;25955:1:1;25944:13;;25875:88::o;25968:127::-;26029:10;26024:3;26020:20;26017:1;26010:31;26060:4;26057:1;26050:15;26084:4;26081:1;26074:15;26100:127;26161:10;26156:3;26152:20;26149:1;26142:31;26192:4;26189:1;26182:15;26216:4;26213:1;26206:15;26232:131;-1:-1:-1;;;;;26307:31:1;;26297:42;;26287:2;;26353:1;26350;26343:12;26368:117;26453:6;26446:5;26442:18;26435:5;26432:29;26422:2;;26475:1;26472;26465:12
Swarm Source
ipfs://eb6b31921860c20ba862671739b3d360a9481afb6f969a43e8ad47f7713f05b6
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 27 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
[ Download: CSV Export ]
[ 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.