Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 469 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Swap And Deposit | 21231139 | 33 days ago | IN | 0.098 ETH | 0.00455919 | ||||
Swap And Deposit | 21231119 | 33 days ago | IN | 0.3 ETH | 0.00437661 | ||||
Swap And Deposit | 21231101 | 33 days ago | IN | 0 ETH | 0.00740888 | ||||
Swap In Same Cha... | 21229817 | 33 days ago | IN | 0 ETH | 0.00630986 | ||||
Swap And Deposit | 21228275 | 33 days ago | IN | 0 ETH | 0.0044334 | ||||
Swap And Deposit | 21226861 | 33 days ago | IN | 0 ETH | 0.00215263 | ||||
Swap And Deposit | 21226627 | 33 days ago | IN | 0 ETH | 0.00267051 | ||||
Swap And Deposit | 21226299 | 33 days ago | IN | 1.88 ETH | 0.00494763 | ||||
Swap And Deposit | 21224489 | 34 days ago | IN | 1 ETH | 0.00604263 | ||||
Swap And Deposit | 21223351 | 34 days ago | IN | 1.55 ETH | 0.00917271 | ||||
Swap In Same Cha... | 21221523 | 34 days ago | IN | 0 ETH | 0.00248057 | ||||
Swap And Deposit | 21221362 | 34 days ago | IN | 0 ETH | 0.0039367 | ||||
Swap And Deposit | 21219058 | 35 days ago | IN | 0.65 ETH | 0.00304579 | ||||
Swap And Deposit | 21217251 | 35 days ago | IN | 0 ETH | 0.00629156 | ||||
Swap And Deposit | 21217218 | 35 days ago | IN | 0 ETH | 0.00747765 | ||||
Swap And Deposit | 21216370 | 35 days ago | IN | 1.32 ETH | 0.0061998 | ||||
Swap And Deposit | 21215662 | 35 days ago | IN | 0.014 ETH | 0.00946331 | ||||
Swap And Deposit | 21209866 | 36 days ago | IN | 0 ETH | 0.00413621 | ||||
Swap And Deposit | 21209836 | 36 days ago | IN | 0 ETH | 0.00337367 | ||||
Swap In Same Cha... | 21209450 | 36 days ago | IN | 0.008371 ETH | 0.00267653 | ||||
Swap In Same Cha... | 21208302 | 36 days ago | IN | 0 ETH | 0.00450099 | ||||
Swap And Deposit | 21207396 | 36 days ago | IN | 0.6 ETH | 0.0034689 | ||||
Swap And Deposit | 21207024 | 36 days ago | IN | 1 ETH | 0.00266239 | ||||
Swap And Deposit | 21206864 | 36 days ago | IN | 0 ETH | 0.0022835 | ||||
Swap And Deposit | 21206822 | 36 days ago | IN | 0 ETH | 0.00435604 |
Latest 25 internal transactions (View All)
Advanced mode:
Parent Transaction Hash | Block |
From
|
To
|
|||
---|---|---|---|---|---|---|
21231139 | 33 days ago | 0.098 ETH | ||||
21231119 | 33 days ago | 0.3 ETH | ||||
21226299 | 33 days ago | 1.88 ETH | ||||
21224489 | 34 days ago | 1 ETH | ||||
21223351 | 34 days ago | 1.55 ETH | ||||
21221523 | 34 days ago | 0.48689728 ETH | ||||
21221523 | 34 days ago | 0.48689728 ETH | ||||
21219058 | 35 days ago | 0.65 ETH | ||||
21216370 | 35 days ago | 1.32 ETH | ||||
21215662 | 35 days ago | 0.014 ETH | ||||
21210681 | 36 days ago | 0.33756481 ETH | ||||
21210681 | 36 days ago | 0.33756481 ETH | ||||
21209450 | 36 days ago | 0.008371 ETH | ||||
21208302 | 36 days ago | 0.18473026 ETH | ||||
21208302 | 36 days ago | 0.18473026 ETH | ||||
21207396 | 36 days ago | 0.6 ETH | ||||
21207024 | 36 days ago | 1 ETH | ||||
21202738 | 37 days ago | 0.4 ETH | ||||
21197740 | 37 days ago | 0.10241253 ETH | ||||
21197740 | 37 days ago | 0.10241253 ETH | ||||
21196465 | 38 days ago | 0.81 ETH | ||||
21195875 | 38 days ago | 0.04534651 ETH | ||||
21195875 | 38 days ago | 0.04534651 ETH | ||||
21194935 | 38 days ago | 0.5829 ETH | ||||
21194125 | 38 days ago | 0.32886419 ETH |
Loading...
Loading
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.
Contract Name:
DexSpan
Compiler Version
v0.8.20+commit.a1b79de6
Optimization Enabled:
Yes with 999999 runs
Other Settings:
paris EvmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "@openzeppelin/contracts/utils/math/SafeMath.sol"; import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/access/AccessControl.sol"; import "@openzeppelin/contracts/utils/Multicall.sol"; import "./interface/IUniswapFactory.sol"; import "./interface/IUniswapV2Factory.sol"; import "./interface/IHandlerReserve.sol"; import "./interface/IEthHandler.sol"; import "./IDexSpan.sol"; import "./UniversalERC20.sol"; import "./interface/IWETH.sol"; import "./libraries/TransferHelper.sol"; // import "./libraries/Multicall.sol"; import "./interface/IAugustusSwapper.sol"; import "../interfaces/IAssetForwarder.sol"; import "./interface/IEthHandler.sol"; import "../interfaces/IMessageHandler.sol"; abstract contract IDexSpanView is DexSpanFlags { function getExpectedReturn( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 parts, uint256 flags ) public view virtual returns (uint256 returnAmount, uint256[] memory distribution); function getExpectedReturnWithGas( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 parts, uint256 flags, uint256 destTokenEthPriceTimesGasPrice ) public view virtual returns ( uint256 returnAmount, uint256 estimateGasAmount, uint256[] memory distribution ); } library DisableFlags { function check(uint256 flags, uint256 flag) internal pure returns (bool) { return (flags & flag) != 0; } } contract DexSpanRoot { using SafeMath for uint256; using DisableFlags for uint256; using UniversalERC20 for IERC20Upgradeable; using UniversalERC20 for IWETH; using UniswapV2ExchangeLib for IUniswapV2Exchange; uint256 internal constant DEXES_COUNT = 4; uint256 internal constant DEXES_COUNT_UPDATED = 1; IERC20Upgradeable internal ZERO_ADDRESS; int256 internal constant VERY_NEGATIVE_VALUE = -1e72; IWETH public wnativeAddress; IERC20Upgradeable public nativeAddress; function _findBestDistribution( uint256 s, // parts int256[][] memory amounts // exchangesReturns ) internal pure returns (int256 returnAmount, uint256[] memory distribution) { uint256 n = amounts.length; int256[][] memory answer = new int256[][](n); // int[n][s+1] uint256[][] memory parent = new uint256[][](n); // int[n][s+1] for (uint256 i; i < n; i++) { answer[i] = new int256[](s + 1); parent[i] = new uint256[](s + 1); } for (uint256 j; j <= s; j++) { answer[0][j] = amounts[0][j]; for (uint256 i = 1; i < n; i++) { answer[i][j] = -1e72; } parent[0][j] = 0; } for (uint256 i = 1; i < n; i++) { for (uint256 j; j <= s; j++) { answer[i][j] = answer[i - 1][j]; parent[i][j] = j; for (uint256 k = 1; k <= j; k++) { if (answer[i - 1][j - k] + amounts[i][k] > answer[i][j]) { answer[i][j] = answer[i - 1][j - k] + amounts[i][k]; parent[i][j] = j - k; } } } } distribution = new uint256[](DEXES_COUNT_UPDATED); uint256 partsLeft = s; for (uint256 curExchange = n - 1; partsLeft > 0; curExchange--) { distribution[curExchange] = partsLeft - parent[curExchange][partsLeft]; partsLeft = parent[curExchange][partsLeft]; } returnAmount = (answer[n - 1][s] == VERY_NEGATIVE_VALUE) ? int256(0) : answer[n - 1][s]; } } contract DexSpan is DexSpanFlags, DexSpanRoot, AccessControl, Multicall { using UniversalERC20 for IERC20Upgradeable; using SafeMath for uint256; using DisableFlags for uint256; using UniswapV2ExchangeLib for IUniswapV2Exchange; IAssetForwarder public assetForwarder; address public assetBridge; address public univ2SkimAddress; address public newOwner; // IWETH public wnativeAddress; mapping(uint256 => address) public flagToAddress; event Swap( string indexed funcName, IERC20Upgradeable[] tokenPath, uint256 amount, address indexed sender, address indexed receiver, uint256 finalAmt, uint256[] flags, uint256 widgetID ); event SwapWithRecipient( string indexed funcName, IERC20Upgradeable[] tokenPath, uint256 amount, address indexed sender, address indexed receiver, uint256 finalAmt, uint256[] flags, uint256 widgetID ); event SwapOnSameChain( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint amount, bytes _data, uint256 flags ); event SetAssetForwarder(address assetForwarder, address admin); event SetAssetBridge(address assetBridge, address admin); event SetFlagToFactory(uint flag, address factoryAddress); event SetFactorySetter(address factorySetter); event SetWNativeAddresses(address wrappedNative); event TransferOwnership(address newOwner); event ClaimOwnership(address newOwner); error InavlidPool(); error InavlidCaller(); error ZeroAddress(); error ZeroFlag(); error InvalidCaller(); error RestrictNativeToken(); error WrongTokenSent(); error WrongDataLength(); error AmountTooLow(); error ExcecutionFailed(); error AlreadyFactorySetter(); struct DexesArgs { IERC20Upgradeable factoryAddress; uint256 _exchangeCode; } struct SwapParams { IERC20Upgradeable[] tokens; uint256 amount; uint256 minReturn; uint256[] flags; bytes[] dataTx; bool isWrapper; address recipient; bytes destToken; } bytes32 public constant FACTORY_SETTER_ROLE = keccak256("FACTORY_SETTER_ROLE"); bytes4 internal constant SWAP_MULTI_WITH_RECEPIENT_SELECTOR = 0xe738aa8d; receive() external payable {} constructor( address _assetForwarderAddress, address _native, address _wrappedNative, address _univ2SkimAddress ) { if (_assetForwarderAddress == address(0)) revert ZeroAddress(); if (_native == address(0)) revert ZeroAddress(); if (_wrappedNative == address(0)) revert ZeroAddress(); if (_univ2SkimAddress == address(0)) revert ZeroAddress(); _setupRole(DEFAULT_ADMIN_ROLE, msg.sender); assetForwarder = IAssetForwarder(_assetForwarderAddress); nativeAddress = IERC20Upgradeable(_native); wnativeAddress = IWETH(_wrappedNative); univ2SkimAddress = _univ2SkimAddress; } function transferOwnership( address _newOwner ) external onlyRole(DEFAULT_ADMIN_ROLE) { if (_newOwner == address(0)) revert ZeroAddress(); newOwner = _newOwner; emit TransferOwnership(_newOwner); } function claimOwnership() external { if (newOwner != msg.sender) { revert InavlidCaller(); } _setupRole(DEFAULT_ADMIN_ROLE, msg.sender); emit ClaimOwnership(msg.sender); } function setAssetForwarder( address _forwarder ) external onlyRole(DEFAULT_ADMIN_ROLE) { if (_forwarder == address(0)) revert ZeroAddress(); assetForwarder = IAssetForwarder(_forwarder); emit SetAssetForwarder(_forwarder, msg.sender); } function setAssetBridge( address _assetBridge ) external onlyRole(DEFAULT_ADMIN_ROLE) { if (_assetBridge == address(0)) revert ZeroAddress(); assetBridge = _assetBridge; emit SetAssetBridge(_assetBridge, msg.sender); } function setFlagToFactoryAddress( uint256 _flagCode, address _factoryAddress ) external onlyRole(FACTORY_SETTER_ROLE) { if (_flagCode == 0) revert ZeroFlag(); if (_factoryAddress == address(0)) revert ZeroAddress(); flagToAddress[_flagCode] = address(_factoryAddress); emit SetFlagToFactory(_flagCode, _factoryAddress); } function setFactorySetter( address _factorySetter ) external onlyRole(DEFAULT_ADMIN_ROLE) { if (_factorySetter == address(0)) revert ZeroAddress(); if (hasRole(FACTORY_SETTER_ROLE, _factorySetter)) revert AlreadyFactorySetter(); _setupRole(FACTORY_SETTER_ROLE, _factorySetter); emit SetFactorySetter(_factorySetter); } function setWNativeAddresses( address _native, address _wrappedNative ) external onlyRole(DEFAULT_ADMIN_ROLE) { if (_native == address(0)) revert ZeroAddress(); if (_wrappedNative == address(0)) revert ZeroAddress(); nativeAddress = IERC20Upgradeable(_native); wnativeAddress = IWETH(_wrappedNative); emit SetWNativeAddresses(_wrappedNative); } function handleMessage( address _tokenSent, uint256 _amount, bytes memory message ) external { if ( msg.sender != address(assetForwarder) && msg.sender != address(assetBridge) ) revert InvalidCaller(); messageHandler(_tokenSent, _amount, message); } function swapInSameChain( IERC20Upgradeable[] memory tokens, uint256 amount, uint256 minReturn, uint256[] memory flags, bytes[] memory dataTx, bool isWrapper, address recipient, uint256 widgetID ) public payable returns (uint256 returnAmount) { returnAmount = swapMultiWithRecipient( tokens, amount, minReturn, flags, dataTx, isWrapper, recipient ); emit Swap( "swapInSameChain", tokens, amount, msg.sender, recipient, returnAmount, flags, widgetID ); } function swapMultiWithRecipient( IERC20Upgradeable[] memory tokens, uint256 amount, uint256 minReturn, uint256[] memory flags, bytes[] memory dataTx, bool isWrapper, address recipient ) public payable returns (uint256 returnAmount) { returnAmount = _swapMultiInternal( tokens, amount, minReturn, flags, dataTx, isWrapper, recipient ); emit SwapWithRecipient( "swapMultiWithRecipient", tokens, amount, msg.sender, recipient, returnAmount, flags, 0 ); } function swapAndDeposit( uint256 partnerId, bytes32 destChainIdBytes, bytes calldata recipient, uint8 depositType, uint256 feeAmount, bytes memory message, SwapParams memory swapData, address refundRecipient ) public payable { _swapMultiInternal( swapData.tokens, swapData.amount, swapData.minReturn, swapData.flags, swapData.dataTx, swapData.isWrapper, address(this) ); IERC20Upgradeable reserveToken = swapData.tokens[ swapData.tokens.length - 1 ]; // swapAndDeposit if (depositType == 0) { uint256 amount = reserveToken.universalBalanceOf(address(this)); reserveToken.universalApprove(address(assetForwarder), amount); assetForwarder.iDeposit{value: reserveToken.isETH() ? amount : 0}( IAssetForwarder.DepositData( partnerId, amount, amount - feeAmount, address(reserveToken), refundRecipient, destChainIdBytes ), swapData.destToken, recipient ); return; } // swapAndDepositWithMessage if (depositType == 1) { uint256 amount = reserveToken.universalBalanceOf(address(this)); reserveToken.universalApprove( address(assetForwarder), swapData.minReturn ); assetForwarder.iDepositMessage{ value: reserveToken.isETH() ? amount : 0 }( IAssetForwarder.DepositData( partnerId, swapData.minReturn, swapData.minReturn - feeAmount, address(reserveToken), refundRecipient, destChainIdBytes ), swapData.destToken, recipient, message ); if (amount > swapData.minReturn) { reserveToken.universalTransfer( refundRecipient, amount - swapData.minReturn ); } return; } // swapAndDepositUSDC : Circle Flow if (depositType == 2) { //NOTE: here reserveToken should be usdc uint256 amount = reserveToken.universalBalanceOf(address(this)); reserveToken.universalApprove(address(assetForwarder), amount); // any extra fee will be left over here assetForwarder.iDepositUSDC{value: msg.value}( partnerId, destChainIdBytes, bytes32(recipient), amount ); return; } } function messageHandler( address _tokenSent, uint256 _amount, bytes memory message ) internal { ( IERC20Upgradeable[] memory tokens, uint256 minReturn, bytes[] memory dataTx, uint256[] memory flags, address recipient, bool isInstruction, bytes memory instruction ) = abi.decode( message, ( IERC20Upgradeable[], uint256, bytes[], uint256[], address, bool, bytes ) ); if (_tokenSent != address(tokens[0])) revert WrongTokenSent(); bytes memory execData; bool execFlag; (execFlag, execData) = address(this).call( abi.encodeWithSelector( SWAP_MULTI_WITH_RECEPIENT_SELECTOR, tokens, _amount, minReturn, flags, dataTx, true, recipient ) ); if (!execFlag) { tokens[0].universalTransfer(recipient, _amount); } if (isInstruction) { uint256 finalAmount = execFlag ? uint256(bytes32(execData)) : _amount; address finalToken = execFlag ? address(tokens[tokens.length - 1]) : _tokenSent; (execFlag, execData) = recipient.call( abi.encodeWithSelector( IMessageHandler.handleMessage.selector, finalToken, finalAmount, instruction ) ); } } function _swapMultiInternal( IERC20Upgradeable[] memory tokens, uint256 amount, uint256 minReturn, uint256[] memory flags, bytes[] memory dataTx, bool isWrapper, address recipient ) internal returns (uint256 returnAmount) { if (recipient == address(0)) revert ZeroAddress(); if (tokens.length - 1 != flags.length) { revert WrongDataLength(); } if (!isWrapper) { if (!tokens[0].isETH() && msg.value != 0) { revert RestrictNativeToken(); } tokens[0].universalTransferFrom(msg.sender, address(this), amount); } returnAmount = tokens[0].universalBalanceOf(address(this)); IERC20Upgradeable destinationToken = tokens[tokens.length - 1]; for (uint256 i = 1; i < tokens.length; i++) { if (tokens[i - 1] == tokens[i]) { continue; } returnAmount = _swapFloor( tokens[i - 1], tokens[i], returnAmount, 0, flags[i - 1], dataTx[i - 1] ); } if (destinationToken.isETH()) { returnAmount = wnativeAddress.balanceOf(address(this)); wnativeAddress.withdraw(returnAmount); } if (recipient != address(this)) { uint256 userBalanceOld = destinationToken.universalBalanceOf( recipient ); destinationToken.universalTransfer(recipient, returnAmount); uint256 userBalanceNew = destinationToken.universalBalanceOf( recipient ); uint receivedTokens = userBalanceNew - userBalanceOld; if (receivedTokens <= minReturn) { revert AmountTooLow(); } returnAmount = receivedTokens; } } function _swapFloor( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 minReturn, uint256 flags, bytes memory _data ) internal returns (uint returnAmount) { returnAmount = _swap( fromToken, destToken, amount, minReturn, flags, _data ); } function _swap( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 minReturn, uint256 flags, bytes memory _data ) internal returns (uint256 returnAmount) { if (fromToken == destToken) { return amount; } function( IERC20Upgradeable, IERC20Upgradeable, uint256, bytes memory, uint256 ) reserve = _getReserveExchange(flags); uint256 remainingAmount = fromToken.universalBalanceOf(address(this)); reserve(fromToken, destToken, remainingAmount, _data, flags); returnAmount = destToken.universalBalanceOf(address(this)); } function _getReserveExchange( uint256 flag ) internal pure returns ( function( IERC20Upgradeable, IERC20Upgradeable, uint256, bytes memory, uint256 ) ) { if (flag < 0x03E9 && flag >= 0x0001) { // 1 - 1000 return _swapOnUniswapV2; } else if (flag == 0x07D2) { return _swapOnParaswap; // 2002 } else { return _swapOnGenericAggregator; } revert("RA: Exchange not found"); } function _swapOnUniswapV2( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, bytes memory _data, uint256 flags ) internal { _swapOnExchangeInternal(fromToken, destToken, amount, flags); } function _swapOnGenericAggregator( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, bytes memory _data, uint256 flagCode ) internal { if (_data.length < 0) { revert WrongDataLength(); } address aggregatorFactoryAddress = flagToAddress[flagCode]; if (aggregatorFactoryAddress == address(0)) { revert ZeroAddress(); } if (fromToken.isETH()) { wnativeAddress.deposit{value: amount}(); } IERC20Upgradeable fromTokenReal = fromToken.isETH() ? wnativeAddress : fromToken; fromTokenReal.universalApprove(address(aggregatorFactoryAddress), amount); // solhint-disable-next-line avoid-low-level-calls (bool success, ) = address(aggregatorFactoryAddress).call(_data); if (!success) revert ExcecutionFailed(); } function _swapOnParaswap( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, bytes memory _data, uint256 flagCode ) internal { if (_data.length < 0) { revert WrongDataLength(); } address paraswap = flagToAddress[flagCode]; if (paraswap == address(0)) { revert ZeroAddress(); } if (fromToken.isETH()) { wnativeAddress.deposit{value: amount}(); } IERC20Upgradeable fromTokenReal = fromToken.isETH() ? wnativeAddress : fromToken; fromTokenReal.universalApprove( IAugustusSwapper(paraswap).getTokenTransferProxy(), amount ); // solhint-disable-next-line avoid-low-level-calls (bool success, ) = address(paraswap).call(_data); if (!success) { revert ExcecutionFailed(); } } function _swapOnExchangeInternal( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 flagCode ) internal returns (uint256 returnAmount) { if (fromToken.isETH()) { wnativeAddress.deposit{value: amount}(); } address dexAddress = flagToAddress[flagCode]; require(dexAddress != address(0), "RA: Exchange not found"); IUniswapV2Factory factory = IUniswapV2Factory(address(dexAddress)); IERC20Upgradeable fromTokenReal = fromToken.isETH() ? wnativeAddress : fromToken; IERC20Upgradeable toTokenReal = destToken.isETH() ? wnativeAddress : destToken; if (fromTokenReal == toTokenReal) { return amount; } IUniswapV2Exchange pool = factory.getPair(fromTokenReal, toTokenReal); if (address(pool) == address(0)) revert InavlidPool(); bool needSync; bool needSkim; (returnAmount, needSync, needSkim) = pool.getReturn( fromTokenReal, toTokenReal, amount ); if (needSync) { pool.sync(); } else if (needSkim) { pool.skim(univ2SkimAddress); } fromTokenReal.universalTransfer(address(pool), amount); if ( uint256(uint160(address(fromTokenReal))) < uint256(uint160(address(toTokenReal))) ) { pool.swap(0, returnAmount, address(this), ""); } else { pool.swap(returnAmount, 0, address(this), ""); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/SafeMath.sol) pragma solidity ^0.8.0; // CAUTION // This version of SafeMath should only be used with Solidity 0.8 or later, // because it relies on the compiler's built in overflow checks. /** * @dev Wrappers over Solidity's arithmetic operations. * * NOTE: `SafeMath` is generally not needed starting with Solidity 0.8, since the compiler * now has built in overflow checking. */ library SafeMath { /** * @dev Returns the addition of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the subtraction of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { return a + b; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return a - b; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { return a * b; } /** * @dev Returns the integer division of two unsigned integers, reverting on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return a / b; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return a % b; } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {trySub}. * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { unchecked { require(b <= a, errorMessage); return a - b; } } /** * @dev Returns the integer division of two unsigned integers, reverting with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a / b; } } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting with custom message when dividing by zero. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryMod}. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a % b; } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.0; import "../IERC20.sol"; import "../extensions/IERC20Permit.sol"; import "../../../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using Address for address; /** * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeTransfer(IERC20 token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } /** * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful. */ function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove(IERC20 token, address spender, uint256 value) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' require( (value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } /** * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 oldAllowance = token.allowance(address(this), spender); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value)); } /** * @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal { unchecked { uint256 oldAllowance = token.allowance(address(this), spender); require(oldAllowance >= value, "SafeERC20: decreased allowance below zero"); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value)); } } /** * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval * to be set to zero before setting it to a non-zero value, such as USDT. */ function forceApprove(IERC20 token, address spender, uint256 value) internal { bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value); if (!_callOptionalReturnBool(token, approvalCall)) { _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0)); _callOptionalReturn(token, approvalCall); } } /** * @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`. * Revert on invalid signature. */ function safePermit( IERC20Permit token, address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) internal { uint256 nonceBefore = token.nonces(owner); token.permit(owner, spender, value, deadline, v, r, s); uint256 nonceAfter = token.nonces(owner); require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed"); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). * * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead. */ function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false // and not revert is the subcall reverts. (bool success, bytes memory returndata) = address(token).call(data); return success && (returndata.length == 0 || abi.decode(returndata, (bool))) && Address.isContract(address(token)); } }
// SPDX-License-Identifier: MIT // 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 IERC20Upgradeable { /** * @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); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/Context.sol"; /** * @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); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (access/AccessControl.sol) pragma solidity ^0.8.0; import "./IAccessControl.sol"; import "../utils/Context.sol"; import "../utils/Strings.sol"; import "../utils/introspection/ERC165.sol"; /** * @dev Contract module that allows children to implement role-based access * control mechanisms. This is a lightweight version that doesn't allow enumerating role * members except through off-chain means by accessing the contract event logs. Some * applications may benefit from on-chain enumerability, for those cases see * {AccessControlEnumerable}. * * Roles are referred to by their `bytes32` identifier. These should be exposed * in the external API and be unique. The best way to achieve this is by * using `public constant` hash digests: * * ```solidity * bytes32 public constant MY_ROLE = keccak256("MY_ROLE"); * ``` * * Roles can be used to represent a set of permissions. To restrict access to a * function call, use {hasRole}: * * ```solidity * function foo() public { * require(hasRole(MY_ROLE, msg.sender)); * ... * } * ``` * * Roles can be granted and revoked dynamically via the {grantRole} and * {revokeRole} functions. Each role has an associated admin role, and only * accounts that have a role's admin role can call {grantRole} and {revokeRole}. * * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means * that only accounts with this role will be able to grant or revoke other * roles. More complex role relationships can be created by using * {_setRoleAdmin}. * * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to * grant and revoke this role. Extra precautions should be taken to secure * accounts that have been granted it. We recommend using {AccessControlDefaultAdminRules} * to enforce additional security measures for this role. */ abstract contract AccessControl is Context, IAccessControl, ERC165 { struct RoleData { mapping(address => bool) members; bytes32 adminRole; } mapping(bytes32 => RoleData) private _roles; bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00; /** * @dev Modifier that checks that an account has a specific role. Reverts * with a standardized message including the required role. * * The format of the revert reason is given by the following regular expression: * * /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/ * * _Available since v4.1._ */ modifier onlyRole(bytes32 role) { _checkRole(role); _; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IAccessControl).interfaceId || super.supportsInterface(interfaceId); } /** * @dev Returns `true` if `account` has been granted `role`. */ function hasRole(bytes32 role, address account) public view virtual override returns (bool) { return _roles[role].members[account]; } /** * @dev Revert with a standard message if `_msgSender()` is missing `role`. * Overriding this function changes the behavior of the {onlyRole} modifier. * * Format of the revert message is described in {_checkRole}. * * _Available since v4.6._ */ function _checkRole(bytes32 role) internal view virtual { _checkRole(role, _msgSender()); } /** * @dev Revert with a standard message if `account` is missing `role`. * * The format of the revert reason is given by the following regular expression: * * /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/ */ function _checkRole(bytes32 role, address account) internal view virtual { if (!hasRole(role, account)) { revert( string( abi.encodePacked( "AccessControl: account ", Strings.toHexString(account), " is missing role ", Strings.toHexString(uint256(role), 32) ) ) ); } } /** * @dev Returns the admin role that controls `role`. See {grantRole} and * {revokeRole}. * * To change a role's admin, use {_setRoleAdmin}. */ function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) { return _roles[role].adminRole; } /** * @dev Grants `role` to `account`. * * If `account` had not been already granted `role`, emits a {RoleGranted} * event. * * Requirements: * * - the caller must have ``role``'s admin role. * * May emit a {RoleGranted} event. */ function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) { _grantRole(role, account); } /** * @dev Revokes `role` from `account`. * * If `account` had been granted `role`, emits a {RoleRevoked} event. * * Requirements: * * - the caller must have ``role``'s admin role. * * May emit a {RoleRevoked} event. */ function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) { _revokeRole(role, account); } /** * @dev Revokes `role` from the calling account. * * Roles are often managed via {grantRole} and {revokeRole}: this function's * purpose is to provide a mechanism for accounts to lose their privileges * if they are compromised (such as when a trusted device is misplaced). * * If the calling account had been revoked `role`, emits a {RoleRevoked} * event. * * Requirements: * * - the caller must be `account`. * * May emit a {RoleRevoked} event. */ function renounceRole(bytes32 role, address account) public virtual override { require(account == _msgSender(), "AccessControl: can only renounce roles for self"); _revokeRole(role, account); } /** * @dev Grants `role` to `account`. * * If `account` had not been already granted `role`, emits a {RoleGranted} * event. Note that unlike {grantRole}, this function doesn't perform any * checks on the calling account. * * May emit a {RoleGranted} event. * * [WARNING] * ==== * This function should only be called from the constructor when setting * up the initial roles for the system. * * Using this function in any other way is effectively circumventing the admin * system imposed by {AccessControl}. * ==== * * NOTE: This function is deprecated in favor of {_grantRole}. */ function _setupRole(bytes32 role, address account) internal virtual { _grantRole(role, account); } /** * @dev Sets `adminRole` as ``role``'s admin role. * * Emits a {RoleAdminChanged} event. */ function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual { bytes32 previousAdminRole = getRoleAdmin(role); _roles[role].adminRole = adminRole; emit RoleAdminChanged(role, previousAdminRole, adminRole); } /** * @dev Grants `role` to `account`. * * Internal function without access restriction. * * May emit a {RoleGranted} event. */ function _grantRole(bytes32 role, address account) internal virtual { if (!hasRole(role, account)) { _roles[role].members[account] = true; emit RoleGranted(role, account, _msgSender()); } } /** * @dev Revokes `role` from `account`. * * Internal function without access restriction. * * May emit a {RoleRevoked} event. */ function _revokeRole(bytes32 role, address account) internal virtual { if (hasRole(role, account)) { _roles[role].members[account] = false; emit RoleRevoked(role, account, _msgSender()); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.5) (utils/Multicall.sol) pragma solidity ^0.8.0; import "./Address.sol"; import "./Context.sol"; /** * @dev Provides a function to batch together multiple calls in a single external call. * * Consider any assumption about calldata validation performed by the sender may be violated if it's not especially * careful about sending transactions invoking {multicall}. For example, a relay address that filters function * selectors won't filter calls nested within a {multicall} operation. * * NOTE: Since 5.0.1 and 4.9.4, this contract identifies non-canonical contexts (i.e. `msg.sender` is not {_msgSender}). * If a non-canonical context is identified, the following self `delegatecall` appends the last bytes of `msg.data` * to the subcall. This makes it safe to use with {ERC2771Context}. Contexts that don't affect the resolution of * {_msgSender} are not propagated to subcalls. * * _Available since v4.1._ */ abstract contract Multicall is Context { /** * @dev Receives and executes a batch of function calls on this contract. * @custom:oz-upgrades-unsafe-allow-reachable delegatecall */ function multicall(bytes[] calldata data) external virtual returns (bytes[] memory results) { bytes memory context = msg.sender == _msgSender() ? new bytes(0) : msg.data[msg.data.length - _contextSuffixLength():]; results = new bytes[](data.length); for (uint256 i = 0; i < data.length; i++) { results[i] = Address.functionDelegateCall(address(this), bytes.concat(data[i], context)); } return results; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "./IUniswapExchange.sol"; interface IUniswapFactory { function getExchange( IERC20Upgradeable token ) external view returns (IUniswapExchange exchange); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "./IUniswapV2Exchange.sol"; interface IUniswapV2Factory { function getPair( IERC20Upgradeable tokenA, IERC20Upgradeable tokenB ) external view returns (IUniswapV2Exchange pair); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; interface IHandlerReserve { function _lpToContract(address token) external returns (address); function _contractToLP(address token) external returns (address); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; interface IEthHandler { function withdraw(address WETH, uint256) external; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; contract DexSpanFlags { // flags = FLAG_DISABLE_UNISWAP + FLAG_DISABLE_BANCOR + ... uint256 internal constant FLAG_DISABLE_UNISWAP = 0x400; uint256 internal constant FLAG_DISABLE_SPLIT_RECALCULATION = 0x800000000000; uint256 internal constant FLAG_DISABLE_ALL_SPLIT_SOURCES = 0x20000000; uint256 internal constant FLAG_DISABLE_UNISWAP_V2_ALL = 0x400; uint256 internal constant FLAG_DISABLE_EMPTY = 0x100000000000; uint256 internal constant FLAG_DISABLE_DFYN = 0x800; uint256 internal constant FLAG_DISABLE_PANCAKESWAP = 0x80; uint256 internal constant FLAG_DISABLE_QUICKSWAP = 0x40000000000; uint256 internal constant FLAG_DISABLE_SUSHISWAP = 0x1000000; uint256 internal constant FLAG_DISABLE_ONEINCH = 0x100000; } abstract contract IDexSpan is DexSpanFlags { function getExpectedReturn( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 parts, uint256 flags // See constants in IOneSplit.sol ) public view virtual returns (uint256 returnAmount, uint256[] memory distribution); function getExpectedReturnWithGasMulti( IERC20Upgradeable[] memory tokens, uint256 amount, uint256[] memory parts, uint256[] memory flags, uint256[] memory destTokenEthPriceTimesGasPrices ) public view virtual returns ( uint256[] memory returnAmounts, uint256 estimateGasAmount, uint256[] memory distribution ); function getExpectedReturnWithGas( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 parts, uint256 flags, // See constants in IOneSplit.sol uint256 destTokenEthPriceTimesGasPrice ) public view virtual returns ( uint256 returnAmount, uint256 estimateGasAmount, uint256[] memory distribution ); function setHandlerAddress( address _handlerAddress ) external virtual returns (bool); function setReserveAddress( address _reserveAddress ) external virtual returns (bool); function setBridgeAddress( address _bridgeAddress ) external virtual returns (bool); function withdraw( address tokenAddress, address recipient, uint256 amount ) public payable virtual returns (bool); function swap( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 minReturn, uint256 flags, bytes memory dataTx, bool isWrapper ) public payable virtual returns (uint256 returnAmount); function swapWithRecipient( IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amount, uint256 minReturn, uint256 flags, bytes memory dataTx, bool isWrapper, address recipient ) public payable virtual returns (uint256 returnAmount); function swapMulti( IERC20Upgradeable[] memory tokens, uint256 amount, uint256 minReturn, uint256[] memory flags, bytes[] memory dataTx, bool isWrapper ) public payable virtual returns (uint256 returnAmount); function swapMultiWithRecipient( IERC20Upgradeable[] memory tokens, uint256 amount, uint256 minReturn, uint256[] memory flags, bytes[] memory dataTx, bool isWrapper, address recipient ) public payable virtual returns (uint256 returnAmount); function getExpectedReturnETH( IERC20Upgradeable srcStablefromtoken, uint256 srcStableFromTokenAmount, uint256 parts, uint256 flags ) public view virtual returns (uint256 returnAmount); function swapInSameChain( address[] memory tokens, uint256 amount, uint256 minReturn, uint256[] memory flags, bytes[] memory dataTx, bool isWrapper, address recipient, uint256 widgetID ) public payable virtual returns (uint256 returnAmount); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "@openzeppelin/contracts-upgradeable/utils/math/SafeMathUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC20/utils/SafeERC20Upgradeable.sol"; import "./libraries/TransferHelper.sol"; library UniversalERC20 { using SafeMathUpgradeable for uint256; using SafeERC20Upgradeable for IERC20Upgradeable; IERC20Upgradeable private constant ZERO_ADDRESS = IERC20Upgradeable(0x0000000000000000000000000000000000000000); IERC20Upgradeable private constant ETH_ADDRESS = IERC20Upgradeable(0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE); function universalTransfer( IERC20Upgradeable token, address to, uint256 amount ) internal returns (bool) { if (amount == 0) { return true; } if (isETH(token)) { (bool success, ) = payable(address(uint160(to))).call{ value: amount }(""); assert(success == true); } else { TransferHelper.safeTransfer(address(token), to, amount); return true; } } function universalTransferFrom( IERC20Upgradeable token, address from, address to, uint256 amount ) internal { if (amount == 0) { return; } if (isETH(token)) { require( from == msg.sender && msg.value >= amount, "Wrong usage of ETH.universalTransferFrom()" ); if (to != address(this)) { (bool success, ) = payable(address(uint160(to))).call{ value: amount }(""); assert(success == true); } if (msg.value > amount) { (bool success, ) = payable(msg.sender).call{ value: msg.value.sub(amount) }(""); assert(success == true); } } else { TransferHelper.safeTransferFrom(address(token), from, to, amount); } } function universalTransferFromSenderToThis( IERC20Upgradeable token, uint256 amount ) internal { if (amount == 0) { return; } if (isETH(token)) { if (msg.value > amount) { // Return remainder if exist (bool success, ) = payable(msg.sender).call{ value: msg.value.sub(amount) }(""); assert(success == true); } } else { TransferHelper.safeTransferFrom( address(token), msg.sender, address(this), amount ); } } function universalApprove( IERC20Upgradeable token, address to, uint256 amount ) internal { if (!isETH(token)) { // if (amount == 0) { // TransferHelper.safeApprove(address(token), to, 0); // return; // } // uint256 allowance = token.allowance(address(this), to); // if (allowance < amount) { // if (allowance > 0) { // TransferHelper.safeApprove(address(token), to, 0); // } // TransferHelper.safeApprove(address(token), to, amount); // } TransferHelper.safeApprove(address(token), to, 0); TransferHelper.safeApprove(address(token), to, amount); } } function universalBalanceOf( IERC20Upgradeable token, address who ) internal view returns (uint256) { if (isETH(token)) { return who.balance; } else { return token.balanceOf(who); } } function isETH(IERC20Upgradeable token) internal pure returns (bool) { return (address(token) == address(ZERO_ADDRESS) || address(token) == address(ETH_ADDRESS)); } function eq( IERC20Upgradeable a, IERC20Upgradeable b ) internal pure returns (bool) { return a == b || (isETH(a) && isETH(b)); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; abstract contract IWETH is IERC20Upgradeable { function deposit() external payable virtual; function withdraw(uint256 amount) external virtual; }
// SPDX-License-Identifier: GPL-3.0-or-later pragma solidity ^0.8.20; // helper methods for interacting with ERC20 tokens and sending ETH that do not consistently return true/false library TransferHelper { function safeApprove(address token, address to, uint256 value) internal { // bytes4(keccak256(bytes('approve(address,uint256)'))); (bool success, bytes memory data) = token.call( abi.encodeWithSelector(0x095ea7b3, to, value) ); require( success && (data.length == 0 || abi.decode(data, (bool))), "TransferHelper::safeApprove: approve failed" ); } function safeTransfer(address token, address to, uint256 value) internal { // bytes4(keccak256(bytes('transfer(address,uint256)'))); (bool success, bytes memory data) = token.call( abi.encodeWithSelector(0xa9059cbb, to, value) ); require( success && (data.length == 0 || abi.decode(data, (bool))), "TransferHelper::safeTransfer: transfer failed" ); } function safeTransferFrom( address token, address from, address to, uint256 value ) internal { // bytes4(keccak256(bytes('transferFrom(address,address,uint256)'))); (bool success, bytes memory data) = token.call( abi.encodeWithSelector(0x23b872dd, from, to, value) ); require( success && (data.length == 0 || abi.decode(data, (bool))), "TransferHelper::transferFrom: transferFrom failed" ); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; interface IAugustusSwapper { function getTokenTransferProxy() external view returns (address); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; /// @title Interface for handler contracts that support deposits and deposit executions. /// @author Router Protocol. interface IAssetForwarder { event FundsDeposited( uint256 partnerId, uint256 amount, bytes32 destChainIdBytes, uint256 destAmount, uint256 depositId, address srcToken, address depositor, bytes recipient, bytes destToken ); event iUSDCDeposited( uint256 partnerId, uint256 amount, bytes32 destChainIdBytes, uint256 usdcNonce, address srcToken, bytes32 recipient, address depositor ); event FundsDepositedWithMessage( uint256 partnerId, uint256 amount, bytes32 destChainIdBytes, uint256 destAmount, uint256 depositId, address srcToken, bytes recipient, address depositor, bytes destToken, bytes message ); event FundsPaid(bytes32 messageHash, address forwarder, uint256 nonce); event DepositInfoUpdate( address srcToken, uint256 feeAmount, uint256 depositId, uint256 eventNonce, bool initiatewithdrawal, address depositor ); event FundsPaidWithMessage( bytes32 messageHash, address forwarder, uint256 nonce, bool execFlag, bytes execData ); struct DestDetails { uint32 domainId; uint256 fee; bool isSet; } struct RelayData { uint256 amount; bytes32 srcChainId; uint256 depositId; address destToken; address recipient; } struct RelayDataMessage { uint256 amount; bytes32 srcChainId; uint256 depositId; address destToken; address recipient; bytes message; } struct DepositData { uint256 partnerId; uint256 amount; uint256 destAmount; address srcToken; address refundRecipient; bytes32 destChainIdBytes; } function iDepositUSDC( uint256 partnerId, bytes32 destChainIdBytes, bytes32 recipient, uint256 amount ) external payable; function iDeposit( DepositData memory depositData, bytes memory destToken, bytes memory recipient ) external payable; function iDepositInfoUpdate( address srcToken, uint256 feeAmount, uint256 depositId, bool initiatewithdrawal ) external payable; function iDepositMessage( DepositData memory depositData, bytes memory destToken, bytes memory recipient, bytes memory message ) external payable; function iRelay(RelayData memory relayData) external payable; function iRelayMessage(RelayDataMessage memory relayData) external payable; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; /// @title Handles ERC20 deposits and deposit executions. /// @author Router Protocol. /// @notice This contract is intended to be used with the Bridge contract. interface IMessageHandler { function handleMessage( address tokenSent, uint256 amount, bytes memory message ) external; }
// SPDX-License-Identifier: MIT // 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); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.4) (token/ERC20/extensions/IERC20Permit.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612]. * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. * * ==== Security Considerations * * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be * considered as an intention to spend the allowance in any specific way. The second is that because permits have * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be * generally recommended is: * * ```solidity * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public { * try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {} * doThing(..., value); * } * * function doThing(..., uint256 value) public { * token.safeTransferFrom(msg.sender, address(this), value); * ... * } * ``` * * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also * {SafeERC20-safeTransferFrom}). * * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so * contracts should have entry points that don't rely on permit. */ interface IERC20Permit { /** * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens, * given ``owner``'s signed approval. * * IMPORTANT: The same issues {IERC20-approve} has related to transaction * ordering also apply here. * * Emits an {Approval} event. * * Requirements: * * - `spender` cannot be the zero address. * - `deadline` must be a timestamp in the future. * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner` * over the EIP712-formatted function arguments. * - the signature must use ``owner``'s current nonce (see {nonces}). * * For more information on the signature format, see the * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP * section]. * * CAUTION: See Security Considerations above. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * @dev Returns the current nonce for `owner`. This value must be * included whenever a signature is generated for {permit}. * * Every successful call to {permit} increases ``owner``'s nonce by one. This * prevents a signature from being used multiple times. */ function nonces(address owner) external view returns (uint256); /** * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.4) (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; } function _contextSuffixLength() internal view virtual returns (uint256) { return 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol) pragma solidity ^0.8.0; /** * @dev External interface of AccessControl declared to support ERC165 detection. */ interface IAccessControl { /** * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole` * * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite * {RoleAdminChanged} not being emitted signaling this. * * _Available since v3.1._ */ event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole); /** * @dev Emitted when `account` is granted `role`. * * `sender` is the account that originated the contract call, an admin role * bearer except when using {AccessControl-_setupRole}. */ event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender); /** * @dev Emitted when `account` is revoked `role`. * * `sender` is the account that originated the contract call: * - if using `revokeRole`, it is the admin role bearer * - if using `renounceRole`, it is the role bearer (i.e. `account`) */ event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender); /** * @dev Returns `true` if `account` has been granted `role`. */ function hasRole(bytes32 role, address account) external view returns (bool); /** * @dev Returns the admin role that controls `role`. See {grantRole} and * {revokeRole}. * * To change a role's admin, use {AccessControl-_setRoleAdmin}. */ function getRoleAdmin(bytes32 role) external view returns (bytes32); /** * @dev Grants `role` to `account`. * * If `account` had not been already granted `role`, emits a {RoleGranted} * event. * * Requirements: * * - the caller must have ``role``'s admin role. */ function grantRole(bytes32 role, address account) external; /** * @dev Revokes `role` from `account`. * * If `account` had been granted `role`, emits a {RoleRevoked} event. * * Requirements: * * - the caller must have ``role``'s admin role. */ function revokeRole(bytes32 role, address account) external; /** * @dev Revokes `role` from the calling account. * * Roles are often managed via {grantRole} and {revokeRole}: this function's * purpose is to provide a mechanism for accounts to lose their privileges * if they are compromised (such as when a trusted device is misplaced). * * If the calling account had been granted `role`, emits a {RoleRevoked} * event. * * Requirements: * * - the caller must be `account`. */ function renounceRole(bytes32 role, address account) external; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Strings.sol) pragma solidity ^0.8.0; import "./math/Math.sol"; import "./math/SignedMath.sol"; /** * @dev String operations. */ library Strings { bytes16 private constant _SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = Math.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), _SYMBOLS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `int256` to its ASCII `string` decimal representation. */ function toString(int256 value) internal pure returns (string memory) { return string(abi.encodePacked(value < 0 ? "-" : "", toString(SignedMath.abs(value)))); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, Math.log256(value) + 1); } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } /** * @dev Returns true if the two strings are equal. */ function equal(string memory a, string memory b) internal pure returns (bool) { return keccak256(bytes(a)) == keccak256(bytes(b)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol) pragma solidity ^0.8.0; import "./IERC165.sol"; /** * @dev Implementation of the {IERC165} interface. * * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check * for the additional interface id that will be supported. For example: * * ```solidity * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId); * } * ``` * * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation. */ abstract contract ERC165 is IERC165 { /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IERC165).interfaceId; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; interface IUniswapExchange { function getEthToTokenInputPrice( uint256 ethSold ) external view returns (uint256 tokensBought); function getTokenToEthInputPrice( uint256 tokensSold ) external view returns (uint256 ethBought); function ethToTokenSwapInput( uint256 minTokens, uint256 deadline ) external payable returns (uint256 tokensBought); function tokenToEthSwapInput( uint256 tokensSold, uint256 minEth, uint256 deadline ) external returns (uint256 ethBought); function tokenToTokenSwapInput( uint256 tokensSold, uint256 minTokensBought, uint256 minEthBought, uint256 deadline, address tokenAddr ) external returns (uint256 tokensBought); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.20; import "@openzeppelin/contracts-upgradeable/utils/math/MathUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/math/SafeMathUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol"; import "../UniversalERC20.sol"; interface IUniswapV2Exchange { function getReserves() external view returns ( uint112 _reserve0, uint112 _reserve1, uint32 _blockTimestampLast ); function swap( uint256 amount0Out, uint256 amount1Out, address to, bytes calldata data ) external; function skim(address to) external; function sync() external; } library UniswapV2ExchangeLib { using MathUpgradeable for uint256; using SafeMathUpgradeable for uint256; using UniversalERC20 for IERC20Upgradeable; function getReturn( IUniswapV2Exchange exchange, IERC20Upgradeable fromToken, IERC20Upgradeable destToken, uint256 amountIn ) internal view returns (uint256 result, bool needSync, bool needSkim) { uint256 reserveIn = fromToken.universalBalanceOf(address(exchange)); uint256 reserveOut = destToken.universalBalanceOf(address(exchange)); (uint112 reserve0, uint112 reserve1, ) = exchange.getReserves(); if (fromToken > destToken) { (reserve0, reserve1) = (reserve1, reserve0); } needSync = (reserveIn < reserve0 || reserveOut < reserve1); needSkim = !needSync && (reserveIn > reserve0 || reserveOut > reserve1); uint256 amountInWithFee = amountIn.mul(997); uint256 numerator = amountInWithFee.mul( MathUpgradeable.min(reserveOut, reserve1) ); uint256 denominator = MathUpgradeable .min(reserveIn, reserve0) .mul(1000) .add(amountInWithFee); result = (denominator == 0) ? 0 : numerator.div(denominator); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/SafeMath.sol) pragma solidity ^0.8.0; // CAUTION // This version of SafeMath should only be used with Solidity 0.8 or later, // because it relies on the compiler's built in overflow checks. /** * @dev Wrappers over Solidity's arithmetic operations. * * NOTE: `SafeMath` is generally not needed starting with Solidity 0.8, since the compiler * now has built in overflow checking. */ library SafeMathUpgradeable { /** * @dev Returns the addition of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the subtraction of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { return a + b; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return a - b; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { return a * b; } /** * @dev Returns the integer division of two unsigned integers, reverting on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return a / b; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return a % b; } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {trySub}. * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { unchecked { require(b <= a, errorMessage); return a - b; } } /** * @dev Returns the integer division of two unsigned integers, reverting with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a / b; } } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting with custom message when dividing by zero. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryMod}. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a % b; } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.3) (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.0; import "../IERC20Upgradeable.sol"; import "../extensions/IERC20PermitUpgradeable.sol"; import "../../../utils/AddressUpgradeable.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20Upgradeable { using AddressUpgradeable for address; /** * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeTransfer(IERC20Upgradeable token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } /** * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful. */ function safeTransferFrom(IERC20Upgradeable token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove(IERC20Upgradeable token, address spender, uint256 value) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' require( (value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } /** * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeIncreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal { uint256 oldAllowance = token.allowance(address(this), spender); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value)); } /** * @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeDecreaseAllowance(IERC20Upgradeable token, address spender, uint256 value) internal { unchecked { uint256 oldAllowance = token.allowance(address(this), spender); require(oldAllowance >= value, "SafeERC20: decreased allowance below zero"); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value)); } } /** * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval * to be set to zero before setting it to a non-zero value, such as USDT. */ function forceApprove(IERC20Upgradeable token, address spender, uint256 value) internal { bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value); if (!_callOptionalReturnBool(token, approvalCall)) { _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0)); _callOptionalReturn(token, approvalCall); } } /** * @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`. * Revert on invalid signature. */ function safePermit( IERC20PermitUpgradeable token, address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) internal { uint256 nonceBefore = token.nonces(owner); token.permit(owner, spender, value, deadline, v, r, s); uint256 nonceAfter = token.nonces(owner); require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed"); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20Upgradeable token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). * * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead. */ function _callOptionalReturnBool(IERC20Upgradeable token, bytes memory data) private returns (bool) { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false // and not revert is the subcall reverts. (bool success, bytes memory returndata) = address(token).call(data); return success && (returndata.length == 0 || abi.decode(returndata, (bool))) && AddressUpgradeable.isContract(address(token)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol) pragma solidity ^0.8.0; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1, "Math: mulDiv overflow"); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10 ** 64) { value /= 10 ** 64; result += 64; } if (value >= 10 ** 32) { value /= 10 ** 32; result += 32; } if (value >= 10 ** 16) { value /= 10 ** 16; result += 16; } if (value >= 10 ** 8) { value /= 10 ** 8; result += 8; } if (value >= 10 ** 4) { value /= 10 ** 4; result += 4; } if (value >= 10 ** 2) { value /= 10 ** 2; result += 2; } if (value >= 10 ** 1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 256, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/SignedMath.sol) pragma solidity ^0.8.0; /** * @dev Standard signed math utilities missing in the Solidity language. */ library SignedMath { /** * @dev Returns the largest of two signed numbers. */ function max(int256 a, int256 b) internal pure returns (int256) { return a > b ? a : b; } /** * @dev Returns the smallest of two signed numbers. */ function min(int256 a, int256 b) internal pure returns (int256) { return a < b ? a : b; } /** * @dev Returns the average of two signed numbers without overflow. * The result is rounded towards zero. */ function average(int256 a, int256 b) internal pure returns (int256) { // Formula from the book "Hacker's Delight" int256 x = (a & b) + ((a ^ b) >> 1); return x + (int256(uint256(x) >> 255) & (a ^ b)); } /** * @dev Returns the absolute unsigned value of a signed value. */ function abs(int256 n) internal pure returns (uint256) { unchecked { // must be unchecked in order to support `n = type(int256).min` return uint256(n >= 0 ? n : -n); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol) pragma solidity ^0.8.0; /** * @dev Standard math utilities missing in the Solidity language. */ library MathUpgradeable { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1, "Math: mulDiv overflow"); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10 ** 64) { value /= 10 ** 64; result += 64; } if (value >= 10 ** 32) { value /= 10 ** 32; result += 32; } if (value >= 10 ** 16) { value /= 10 ** 16; result += 16; } if (value >= 10 ** 8) { value /= 10 ** 8; result += 8; } if (value >= 10 ** 4) { value /= 10 ** 4; result += 4; } if (value >= 10 ** 2) { value /= 10 ** 2; result += 2; } if (value >= 10 ** 1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 256, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.4) (token/ERC20/extensions/IERC20Permit.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612]. * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. * * ==== Security Considerations * * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be * considered as an intention to spend the allowance in any specific way. The second is that because permits have * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be * generally recommended is: * * ```solidity * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public { * try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {} * doThing(..., value); * } * * function doThing(..., uint256 value) public { * token.safeTransferFrom(msg.sender, address(this), value); * ... * } * ``` * * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also * {SafeERC20-safeTransferFrom}). * * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so * contracts should have entry points that don't rely on permit. */ interface IERC20PermitUpgradeable { /** * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens, * given ``owner``'s signed approval. * * IMPORTANT: The same issues {IERC20-approve} has related to transaction * ordering also apply here. * * Emits an {Approval} event. * * Requirements: * * - `spender` cannot be the zero address. * - `deadline` must be a timestamp in the future. * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner` * over the EIP712-formatted function arguments. * - the signature must use ``owner``'s current nonce (see {nonces}). * * For more information on the signature format, see the * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP * section]. * * CAUTION: See Security Considerations above. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * @dev Returns the current nonce for `owner`. This value must be * included whenever a signature is generated for {permit}. * * Every successful call to {permit} increases ``owner``'s nonce by one. This * prevents a signature from being used multiple times. */ function nonces(address owner) external view returns (uint256); /** * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library AddressUpgradeable { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://consensys.net/diligence/blog/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.8.0/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
{ "remappings": [ "ds-test/=lib/forge-std/lib/ds-test/src/", "forge-std/=lib/forge-std/src/", "@openzeppelin/=node_modules/@openzeppelin/", "fx-portal/=lib/contracts/contracts/", "@routerprotocol/evm-gateway-contracts/=lib/router-gateway-contracts/evm/" ], "optimizer": { "enabled": true, "runs": 999999 }, "metadata": { "useLiteralContent": false, "bytecodeHash": "ipfs", "appendCBOR": true }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "evmVersion": "paris", "viaIR": false, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_assetForwarderAddress","type":"address"},{"internalType":"address","name":"_native","type":"address"},{"internalType":"address","name":"_wrappedNative","type":"address"},{"internalType":"address","name":"_univ2SkimAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"AlreadyFactorySetter","type":"error"},{"inputs":[],"name":"AmountTooLow","type":"error"},{"inputs":[],"name":"ExcecutionFailed","type":"error"},{"inputs":[],"name":"InavlidCaller","type":"error"},{"inputs":[],"name":"InavlidPool","type":"error"},{"inputs":[],"name":"InvalidCaller","type":"error"},{"inputs":[],"name":"RestrictNativeToken","type":"error"},{"inputs":[],"name":"WrongDataLength","type":"error"},{"inputs":[],"name":"WrongTokenSent","type":"error"},{"inputs":[],"name":"ZeroAddress","type":"error"},{"inputs":[],"name":"ZeroFlag","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"newOwner","type":"address"}],"name":"ClaimOwnership","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"previousAdminRole","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"newAdminRole","type":"bytes32"}],"name":"RoleAdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleGranted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleRevoked","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"assetBridge","type":"address"},{"indexed":false,"internalType":"address","name":"admin","type":"address"}],"name":"SetAssetBridge","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"assetForwarder","type":"address"},{"indexed":false,"internalType":"address","name":"admin","type":"address"}],"name":"SetAssetForwarder","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"factorySetter","type":"address"}],"name":"SetFactorySetter","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"flag","type":"uint256"},{"indexed":false,"internalType":"address","name":"factoryAddress","type":"address"}],"name":"SetFlagToFactory","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"wrappedNative","type":"address"}],"name":"SetWNativeAddresses","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"string","name":"funcName","type":"string"},{"indexed":false,"internalType":"contract IERC20Upgradeable[]","name":"tokenPath","type":"address[]"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"address","name":"receiver","type":"address"},{"indexed":false,"internalType":"uint256","name":"finalAmt","type":"uint256"},{"indexed":false,"internalType":"uint256[]","name":"flags","type":"uint256[]"},{"indexed":false,"internalType":"uint256","name":"widgetID","type":"uint256"}],"name":"Swap","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"contract IERC20Upgradeable","name":"fromToken","type":"address"},{"indexed":false,"internalType":"contract IERC20Upgradeable","name":"destToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"bytes","name":"_data","type":"bytes"},{"indexed":false,"internalType":"uint256","name":"flags","type":"uint256"}],"name":"SwapOnSameChain","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"string","name":"funcName","type":"string"},{"indexed":false,"internalType":"contract IERC20Upgradeable[]","name":"tokenPath","type":"address[]"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"address","name":"receiver","type":"address"},{"indexed":false,"internalType":"uint256","name":"finalAmt","type":"uint256"},{"indexed":false,"internalType":"uint256[]","name":"flags","type":"uint256[]"},{"indexed":false,"internalType":"uint256","name":"widgetID","type":"uint256"}],"name":"SwapWithRecipient","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"newOwner","type":"address"}],"name":"TransferOwnership","type":"event"},{"inputs":[],"name":"DEFAULT_ADMIN_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"FACTORY_SETTER_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"assetBridge","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"assetForwarder","outputs":[{"internalType":"contract IAssetForwarder","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"flagToAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleAdmin","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_tokenSent","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bytes","name":"message","type":"bytes"}],"name":"handleMessage","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes[]","name":"data","type":"bytes[]"}],"name":"multicall","outputs":[{"internalType":"bytes[]","name":"results","type":"bytes[]"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"nativeAddress","outputs":[{"internalType":"contract IERC20Upgradeable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"newOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_assetBridge","type":"address"}],"name":"setAssetBridge","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_forwarder","type":"address"}],"name":"setAssetForwarder","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_factorySetter","type":"address"}],"name":"setFactorySetter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_flagCode","type":"uint256"},{"internalType":"address","name":"_factoryAddress","type":"address"}],"name":"setFlagToFactoryAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_native","type":"address"},{"internalType":"address","name":"_wrappedNative","type":"address"}],"name":"setWNativeAddresses","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"partnerId","type":"uint256"},{"internalType":"bytes32","name":"destChainIdBytes","type":"bytes32"},{"internalType":"bytes","name":"recipient","type":"bytes"},{"internalType":"uint8","name":"depositType","type":"uint8"},{"internalType":"uint256","name":"feeAmount","type":"uint256"},{"internalType":"bytes","name":"message","type":"bytes"},{"components":[{"internalType":"contract IERC20Upgradeable[]","name":"tokens","type":"address[]"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"minReturn","type":"uint256"},{"internalType":"uint256[]","name":"flags","type":"uint256[]"},{"internalType":"bytes[]","name":"dataTx","type":"bytes[]"},{"internalType":"bool","name":"isWrapper","type":"bool"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"bytes","name":"destToken","type":"bytes"}],"internalType":"struct DexSpan.SwapParams","name":"swapData","type":"tuple"},{"internalType":"address","name":"refundRecipient","type":"address"}],"name":"swapAndDeposit","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"contract IERC20Upgradeable[]","name":"tokens","type":"address[]"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"minReturn","type":"uint256"},{"internalType":"uint256[]","name":"flags","type":"uint256[]"},{"internalType":"bytes[]","name":"dataTx","type":"bytes[]"},{"internalType":"bool","name":"isWrapper","type":"bool"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"widgetID","type":"uint256"}],"name":"swapInSameChain","outputs":[{"internalType":"uint256","name":"returnAmount","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"contract IERC20Upgradeable[]","name":"tokens","type":"address[]"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"minReturn","type":"uint256"},{"internalType":"uint256[]","name":"flags","type":"uint256[]"},{"internalType":"bytes[]","name":"dataTx","type":"bytes[]"},{"internalType":"bool","name":"isWrapper","type":"bool"},{"internalType":"address","name":"recipient","type":"address"}],"name":"swapMultiWithRecipient","outputs":[{"internalType":"uint256","name":"returnAmount","type":"uint256"}],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"_newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"univ2SkimAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wnativeAddress","outputs":[{"internalType":"contract IWETH","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"stateMutability":"payable","type":"receive"}]
Contract Creation Code
60806040523480156200001157600080fd5b5060405162004f4338038062004f43833981016040819052620000349162000206565b6001600160a01b0384166200005c5760405163d92e233d60e01b815260040160405180910390fd5b6001600160a01b038316620000845760405163d92e233d60e01b815260040160405180910390fd5b6001600160a01b038216620000ac5760405163d92e233d60e01b815260040160405180910390fd5b6001600160a01b038116620000d45760405163d92e233d60e01b815260040160405180910390fd5b620000e160003362000135565b600480546001600160a01b039586166001600160a01b031991821617909155600280549486169482169490941790935560018054928516928416929092179091556006805491909316911617905562000263565b62000141828262000145565b5050565b60008281526003602090815260408083206001600160a01b038516845290915290205460ff16620001415760008281526003602090815260408083206001600160a01b03851684529091529020805460ff19166001179055620001a53390565b6001600160a01b0316816001600160a01b0316837f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d60405160405180910390a45050565b80516001600160a01b03811681146200020157600080fd5b919050565b600080600080608085870312156200021d57600080fd5b6200022885620001e9565b93506200023860208601620001e9565b92506200024860408601620001e9565b91506200025860608601620001e9565b905092959194509250565b614cd080620002736000396000f3fe6080604052600436106101b05760003560e01c806391d14854116100ec578063d00a2d5f1161008a578063da6689f011610064578063da6689f01461057e578063e2a4ac2d1461059e578063e738aa8d146105cb578063f2fde38b146105de57600080fd5b8063d00a2d5f14610511578063d4ee1d9014610531578063d547741f1461055e57600080fd5b8063a2625efc116100c6578063a2625efc14610454578063ac9650d814610481578063adb88982146104ae578063c75a57e4146104ce57600080fd5b806391d14854146103cc5780639e99ec391461041f578063a217fddf1461043f57600080fd5b8063248a9ca3116101595780632f2ff15d116101335780632f2ff15d1461034a57806336568abe1461036a5780634e71e0c81461038a5780637fe689171461039f57600080fd5b8063248a9ca31461029b578063285f94a0146102cb5780632e4763371461031d57600080fd5b8063169651601161018a578063169651601461024857806318137a4114610268578063183d9c951461028857600080fd5b806301ffc9a7146101bc57806306135b7c146101f1578063086efdc41461020657600080fd5b366101b757005b600080fd5b3480156101c857600080fd5b506101dc6101d73660046139d6565b6105fe565b60405190151581526020015b60405180910390f35b6102046101ff366004613e5f565b610697565b005b34801561021257600080fd5b5061023a7fafffb39e42e2a1496bc4305c8783f904fa76641e3895205b6002a03336ee097581565b6040519081526020016101e8565b34801561025457600080fd5b50610204610263366004613f2e565b610ac3565b34801561027457600080fd5b50610204610283366004613f5e565b610c01565b61023a610296366004613f7b565b610cd9565b3480156102a757600080fd5b5061023a6102b6366004614042565b60009081526003602052604090206001015490565b3480156102d757600080fd5b506005546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b60405173ffffffffffffffffffffffffffffffffffffffff90911681526020016101e8565b34801561032957600080fd5b506002546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b34801561035657600080fd5b50610204610365366004613f2e565b610d83565b34801561037657600080fd5b50610204610385366004613f2e565b610dad565b34801561039657600080fd5b50610204610e65565b3480156103ab57600080fd5b506004546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b3480156103d857600080fd5b506101dc6103e7366004613f2e565b600091825260036020908152604080842073ffffffffffffffffffffffffffffffffffffffff93909316845291905290205460ff1690565b34801561042b57600080fd5b5061020461043a366004613f5e565b610ef6565b34801561044b57600080fd5b5061023a600081565b34801561046057600080fd5b506006546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b34801561048d57600080fd5b506104a161049c36600461405b565b610fc6565b6040516101e89190614193565b3480156104ba57600080fd5b506102046104c9366004613f5e565b6110b9565b3480156104da57600080fd5b506102f86104e9366004614042565b60086020526000908152604090205473ffffffffffffffffffffffffffffffffffffffff1681565b34801561051d57600080fd5b5061020461052c3660046141a6565b611200565b34801561053d57600080fd5b506007546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b34801561056a57600080fd5b50610204610579366004613f2e565b611282565b34801561058a57600080fd5b506102046105993660046141ff565b6112a7565b3480156105aa57600080fd5b506001546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b61023a6105d936600461422d565b6113cf565b3480156105ea57600080fd5b506102046105f9366004613f5e565b611479565b60007fffffffff0000000000000000000000000000000000000000000000000000000082167f7965db0b00000000000000000000000000000000000000000000000000000000148061069157507f01ffc9a7000000000000000000000000000000000000000000000000000000007fffffffff000000000000000000000000000000000000000000000000000000008316145b92915050565b6106be826000015183602001518460400151856060015186608001518760a0015130611544565b5081518051600091906106d39060019061431f565b815181106106e3576106e3614332565b602002602001015190508560ff1660000361084157600061071a73ffffffffffffffffffffffffffffffffffffffff831630611a5a565b6004549091506107449073ffffffffffffffffffffffffffffffffffffffff848116911683611b1f565b60045473ffffffffffffffffffffffffffffffffffffffff9081169063f452ed4d90610771908516611b43565b61077c57600061077e565b825b6040518060c001604052808f81526020018581526020018a866107a1919061431f565b81526020018673ffffffffffffffffffffffffffffffffffffffff1681526020018773ffffffffffffffffffffffffffffffffffffffff1681526020018e8152508760e001518d8d6040518663ffffffff1660e01b815260040161080894939291906143aa565b6000604051808303818588803b15801561082157600080fd5b505af1158015610835573d6000803e3d6000fd5b50505050505050610ab8565b8560ff166001036109e457600061086e73ffffffffffffffffffffffffffffffffffffffff831630611a5a565b600454604086015191925061089d9173ffffffffffffffffffffffffffffffffffffffff858116921690611b1f565b60045473ffffffffffffffffffffffffffffffffffffffff90811690630421caf0906108ca908516611b43565b6108d55760006108d7565b825b6040518060c001604052808f8152602001886040015181526020018a8960400151610902919061431f565b81526020018673ffffffffffffffffffffffffffffffffffffffff1681526020018773ffffffffffffffffffffffffffffffffffffffff1681526020018e8152508760e001518d8d8b6040518763ffffffff1660e01b815260040161096b959493929190614431565b6000604051808303818588803b15801561098457600080fd5b505af1158015610998573d6000803e3d6000fd5b505050505083604001518111156109dd576109db838560400151836109bd919061431f565b73ffffffffffffffffffffffffffffffffffffffff85169190611b96565b505b5050610ab8565b8560ff16600203610ab6576000610a1173ffffffffffffffffffffffffffffffffffffffff831630611a5a565b600454909150610a3b9073ffffffffffffffffffffffffffffffffffffffff848116911683611b1f565b60045473ffffffffffffffffffffffffffffffffffffffff16633e28c7d2348d8d610a668d8f6144d9565b6040517fffffffff0000000000000000000000000000000000000000000000000000000060e087901b16815260048101939093526024830191909152604482015260648101859052608401610808565b505b505050505050505050565b7fafffb39e42e2a1496bc4305c8783f904fa76641e3895205b6002a03336ee0975610aed81611c48565b82600003610b27576040517f196de28b00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b73ffffffffffffffffffffffffffffffffffffffff8216610b74576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008381526008602090815260409182902080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff86169081179091558251868152918201527f11f5b5361c420666dc3452b8107601e1702f94b951f15bd93c26262c8147874e91015b60405180910390a1505050565b6000610c0c81611c48565b73ffffffffffffffffffffffffffffffffffffffff8216610c59576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600480547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff8416908117909155604080519182523360208301527f61b6fe2922f0180b98d305419bf3dcf432f5d8df894388c61c1f599c2194ec1d91015b60405180910390a15050565b6000610cea898989898989896113cf565b6040517f73776170496e53616d65436861696e0000000000000000000000000000000000815290915073ffffffffffffffffffffffffffffffffffffffff8416903390600f0160405180910390207fe3716b48e7d01d727d68472c873c69c2c4407365e4f68b2b6c8dd4569cd16dc78c8c868c89604051610d6f959493929190614596565b60405180910390a498975050505050505050565b600082815260036020526040902060010154610d9e81611c48565b610da88383611c55565b505050565b73ffffffffffffffffffffffffffffffffffffffff81163314610e57576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602f60248201527f416363657373436f6e74726f6c3a2063616e206f6e6c792072656e6f756e636560448201527f20726f6c657320666f722073656c66000000000000000000000000000000000060648201526084015b60405180910390fd5b610e618282611d49565b5050565b60075473ffffffffffffffffffffffffffffffffffffffff163314610eb6576040517f821de23100000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610ec1600033611e04565b6040513381527f85a8c8100c1d48466310240af79644f50cdae7d09cb4820d60dfc5b2462f0fd09060200160405180910390a1565b6000610f0181611c48565b73ffffffffffffffffffffffffffffffffffffffff8216610f4e576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600580547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff8416908117909155604080519182523360208301527f45085069ced50a4d2e82656a82ba8f5a1a28694489d8ea262252e40899a930a29101610ccd565b6040805160008152602081019091526060908267ffffffffffffffff811115610ff157610ff1613a77565b60405190808252806020026020018201604052801561102457816020015b606081526020019060019003908161100f5790505b50915060005b838110156110b1576110813086868481811061104857611048614332565b905060200281019061105a91906145da565b8560405160200161106d9392919061463f565b604051602081830303815290604052611e0e565b83828151811061109357611093614332565b602002602001018190525080806110a990614666565b91505061102a565b505092915050565b60006110c481611c48565b73ffffffffffffffffffffffffffffffffffffffff8216611111576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b73ffffffffffffffffffffffffffffffffffffffff821660009081527f3ab9add546f7e3b43565da9bc7e65e47b969d3856bb01a7dcac7bc8ff5080cbd602052604090205460ff1615611190576040517fe4e970d000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6111ba7fafffb39e42e2a1496bc4305c8783f904fa76641e3895205b6002a03336ee097583611e04565b60405173ffffffffffffffffffffffffffffffffffffffff831681527fd2b69bde5a989d23ef605a0dd4e0edf7662105058ab69f2559e482857b599ed890602001610ccd565b60045473ffffffffffffffffffffffffffffffffffffffff163314801590611240575060055473ffffffffffffffffffffffffffffffffffffffff163314155b15611277576040517f48f5c3ed00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610da8838383611e33565b60008281526003602052604090206001015461129d81611c48565b610da88383611d49565b60006112b281611c48565b73ffffffffffffffffffffffffffffffffffffffff83166112ff576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b73ffffffffffffffffffffffffffffffffffffffff821661134c576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6002805473ffffffffffffffffffffffffffffffffffffffff8581167fffffffffffffffffffffffff0000000000000000000000000000000000000000928316179092556001805492851692909116821790556040519081527fcc4166d40cf68104cdf710f65ce022efd296742868a22b42e4695f89b886363690602001610bf4565b60006113e088888888888888611544565b6040517f737761704d756c746957697468526563697069656e7400000000000000000000815290915073ffffffffffffffffffffffffffffffffffffffff831690339060160160405180910390207fc40fae9d5f584875c393ac222c6f88b6c9dced1e9cc6251483648ac2e902c8b08b8b868b6000604051611466959493929190614596565b60405180910390a4979650505050505050565b600061148481611c48565b73ffffffffffffffffffffffffffffffffffffffff82166114d1576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600780547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff84169081179091556040519081527fcfaaa26691e16e66e73290fc725eee1a6b4e0e693a1640484937aac25ffb55a490602001610ccd565b600073ffffffffffffffffffffffffffffffffffffffff8216611593576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8451600189516115a3919061431f565b146115da576040517fcc513c7200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b826116a257611618886000815181106115f5576115f5614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16611b43565b15801561162457503415155b1561165b576040517f694b150300000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6116a23330898b60008151811061167457611674614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff166121a6909392919063ffffffff16565b6116e530896000815181106116b9576116b9614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16611a5a90919063ffffffff16565b905060008860018a516116f8919061431f565b8151811061170857611708614332565b602002602001015190506000600190505b895181101561183e5789818151811061173457611734614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff168a600183611760919061431f565b8151811061177057611770614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16031561182c576118298a6117a360018461431f565b815181106117b3576117b3614332565b60200260200101518b83815181106117cd576117cd614332565b60200260200101518560008b6001876117e6919061431f565b815181106117f6576117f6614332565b60200260200101518b60018861180c919061431f565b8151811061181c5761181c614332565b6020026020010151612386565b92505b8061183681614666565b915050611719565b5061185e8173ffffffffffffffffffffffffffffffffffffffff16611b43565b1561197b576001546040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015273ffffffffffffffffffffffffffffffffffffffff909116906370a0823190602401602060405180830381865afa1580156118d1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118f5919061469e565b6001546040517f2e1a7d4d0000000000000000000000000000000000000000000000000000000081526004810183905291935073ffffffffffffffffffffffffffffffffffffffff1690632e1a7d4d90602401600060405180830381600087803b15801561196257600080fd5b505af1158015611976573d6000803e3d6000fd5b505050505b73ffffffffffffffffffffffffffffffffffffffff83163014611a4e5760006119ba73ffffffffffffffffffffffffffffffffffffffff831685611a5a565b90506119dd73ffffffffffffffffffffffffffffffffffffffff83168585611b96565b506000611a0073ffffffffffffffffffffffffffffffffffffffff841686611a5a565b90506000611a0e838361431f565b9050898111611a49576040517f1fbaba3500000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b935050505b50979650505050505050565b6000611a6583611b43565b15611a88575073ffffffffffffffffffffffffffffffffffffffff811631610691565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff83811660048301528416906370a0823190602401602060405180830381865afa158015611af4573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b18919061469e565b9050610691565b611b2883611b43565b610da857611b38838360006123a1565b610da88383836123a1565b600073ffffffffffffffffffffffffffffffffffffffff82161580610691575073ffffffffffffffffffffffffffffffffffffffff821673eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee1492915050565b600081600003611ba857506001611c41565b611bb184611b43565b15611c325760008373ffffffffffffffffffffffffffffffffffffffff168360405160006040518083038185875af1925050503d8060008114611c10576040519150601f19603f3d011682016040523d82523d6000602084013e611c15565b606091505b5090915050600181151514611c2c57611c2c6146b7565b50611c41565b611c3d848484612537565b5060015b9392505050565b611c5281336126c6565b50565b600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516845290915290205460ff16610e6157600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff85168452909152902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00166001179055611ceb3390565b73ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16837f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d60405160405180910390a45050565b600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516845290915290205460ff1615610e6157600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516808552925280832080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016905551339285917ff6391f5c32d9c69d2a47ea670b442974b53935d1edc7fd64eb21e047a839171b9190a45050565b610e618282611c55565b6060611c418383604051806060016040528060278152602001614c7460279139612780565b600080600080600080600087806020019051810190611e529190614880565b965096509650965096509650965086600081518110611e7357611e73614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff168a73ffffffffffffffffffffffffffffffffffffffff1614611edf576040517f198b098900000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b606060003073ffffffffffffffffffffffffffffffffffffffff1663e738aa8d60e01b8a8d8b8a8c60018c604051602401611f209796959493929190614958565b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529181526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff00000000000000000000000000000000000000000000000000000000909416939093179092529051611fa991906149cf565b6000604051808303816000865af19150503d8060008114611fe6576040519150601f19603f3d011682016040523d82523d6000602084013e611feb565b606091505b50925090508061203c5761203a858c8b60008151811061200d5761200d614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16611b969092919063ffffffff16565b505b83156121985760008161204f578b612058565b612058836149eb565b9050600082612067578d61208f565b8a60018c51612076919061431f565b8151811061208657612086614332565b60200260200101515b90508673ffffffffffffffffffffffffffffffffffffffff1663d00a2d5f60e01b8284886040516024016120c593929190614a30565b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529181526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff0000000000000000000000000000000000000000000000000000000090941693909317909252905161214e91906149cf565b6000604051808303816000865af19150503d806000811461218b576040519150601f19603f3d011682016040523d82523d6000602084013e612190565b606091505b509450925050505b505050505050505050505050565b8015612380576121b584611b43565b156123745773ffffffffffffffffffffffffffffffffffffffff8316331480156121df5750803410155b61226b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602a60248201527f57726f6e67207573616765206f66204554482e756e6976657273616c5472616e60448201527f7366657246726f6d2829000000000000000000000000000000000000000000006064820152608401610e4e565b73ffffffffffffffffffffffffffffffffffffffff821630146123005760008273ffffffffffffffffffffffffffffffffffffffff168260405160006040518083038185875af1925050503d80600081146122e2576040519150601f19603f3d011682016040523d82523d6000602084013e6122e7565b606091505b50909150506001811515146122fe576122fe6146b7565b505b8034111561236f576000336123153484612805565b604051600081818185875af1925050503d8060008114612351576040519150601f19603f3d011682016040523d82523d6000602084013e612356565b606091505b509091505060018115151461236d5761236d6146b7565b505b612380565b61238084848484612811565b50505050565b60006123968787878787876129b0565b979650505050505050565b6040805173ffffffffffffffffffffffffffffffffffffffff8481166024830152604480830185905283518084039091018152606490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167f095ea7b300000000000000000000000000000000000000000000000000000000179052915160009283929087169161243891906149cf565b6000604051808303816000865af19150503d8060008114612475576040519150601f19603f3d011682016040523d82523d6000602084013e61247a565b606091505b50915091508180156124a45750805115806124a45750808060200190518101906124a49190614a6e565b612530576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f5472616e7366657248656c7065723a3a73616665417070726f76653a2061707060448201527f726f7665206661696c65640000000000000000000000000000000000000000006064820152608401610e4e565b5050505050565b6040805173ffffffffffffffffffffffffffffffffffffffff8481166024830152604480830185905283518084039091018152606490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fa9059cbb0000000000000000000000000000000000000000000000000000000017905291516000928392908716916125ce91906149cf565b6000604051808303816000865af19150503d806000811461260b576040519150601f19603f3d011682016040523d82523d6000602084013e612610565b606091505b509150915081801561263a57508051158061263a57508080602001905181019061263a9190614a6e565b612530576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602d60248201527f5472616e7366657248656c7065723a3a736166655472616e736665723a20747260448201527f616e73666572206661696c6564000000000000000000000000000000000000006064820152608401610e4e565b600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516845290915290205460ff16610e615761270681612a5c565b612711836020612a7b565b604051602001612722929190614a8b565b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0818403018152908290527f08c379a0000000000000000000000000000000000000000000000000000000008252610e4e91600401614b0c565b60606000808573ffffffffffffffffffffffffffffffffffffffff16856040516127aa91906149cf565b600060405180830381855af49150503d80600081146127e5576040519150601f19603f3d011682016040523d82523d6000602084013e6127ea565b606091505b50915091506127fb86838387612cbe565b9695505050505050565b6000611c41828461431f565b6040805173ffffffffffffffffffffffffffffffffffffffff85811660248301528481166044830152606480830185905283518084039091018152608490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167f23b872dd0000000000000000000000000000000000000000000000000000000017905291516000928392908816916128b091906149cf565b6000604051808303816000865af19150503d80600081146128ed576040519150601f19603f3d011682016040523d82523d6000602084013e6128f2565b606091505b509150915081801561291c57508051158061291c57508080602001905181019061291c9190614a6e565b6129a8576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603160248201527f5472616e7366657248656c7065723a3a7472616e7366657246726f6d3a20747260448201527f616e7366657246726f6d206661696c65640000000000000000000000000000006064820152608401610e4e565b505050505050565b60008573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff16036129ec5750836127fb565b6139cc6129f884612d66565b90506000612a1c73ffffffffffffffffffffffffffffffffffffffff8a1630611a5a565b9050612a2f89898387898763ffffffff16565b612a4f73ffffffffffffffffffffffffffffffffffffffff891630611a5a565b9998505050505050505050565b606061069173ffffffffffffffffffffffffffffffffffffffff831660145b60606000612a8a836002614b1f565b612a95906002614b36565b67ffffffffffffffff811115612aad57612aad613a77565b6040519080825280601f01601f191660200182016040528015612ad7576020820181803683370190505b5090507f300000000000000000000000000000000000000000000000000000000000000081600081518110612b0e57612b0e614332565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a9053507f780000000000000000000000000000000000000000000000000000000000000081600181518110612b7157612b71614332565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a9053506000612bad846002614b1f565b612bb8906001614b36565b90505b6001811115612c55577f303132333435363738396162636465660000000000000000000000000000000085600f1660108110612bf957612bf9614332565b1a60f81b828281518110612c0f57612c0f614332565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a90535060049490941c93612c4e81614b49565b9050612bbb565b508315611c41576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e746044820152606401610e4e565b60608315612d54578251600003612d4d5773ffffffffffffffffffffffffffffffffffffffff85163b612d4d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606401610e4e565b5081612d5e565b612d5e8383612da4565b949350505050565b6139cc6103e982108015612d7b575060018210155b15612d895750612de8919050565b816107d203612d9b5750612df4919050565b5061307d919050565b815115612db45781518083602001fd5b806040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e4e9190614b0c565b6129a8858585846131e9565b60008181526008602052604090205473ffffffffffffffffffffffffffffffffffffffff1680612e50576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612e6f8673ffffffffffffffffffffffffffffffffffffffff16611b43565b15612ef857600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663d0e30db0856040518263ffffffff1660e01b81526004016000604051808303818588803b158015612ede57600080fd5b505af1158015612ef2573d6000803e3d6000fd5b50505050505b6000612f198773ffffffffffffffffffffffffffffffffffffffff16611b43565b612f235786612f3d565b60015473ffffffffffffffffffffffffffffffffffffffff165b9050612fcf8273ffffffffffffffffffffffffffffffffffffffff1663d2c4b5986040518163ffffffff1660e01b8152600401602060405180830381865afa158015612f8d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612fb19190614b7e565b73ffffffffffffffffffffffffffffffffffffffff83169087611b1f565b60008273ffffffffffffffffffffffffffffffffffffffff1685604051612ff691906149cf565b6000604051808303816000865af19150503d8060008114613033576040519150601f19603f3d011682016040523d82523d6000602084013e613038565b606091505b5050905080613073576040517f5987947500000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5050505050505050565b60008181526008602052604090205473ffffffffffffffffffffffffffffffffffffffff16806130d9576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6130f88673ffffffffffffffffffffffffffffffffffffffff16611b43565b1561318157600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663d0e30db0856040518263ffffffff1660e01b81526004016000604051808303818588803b15801561316757600080fd5b505af115801561317b573d6000803e3d6000fd5b50505050505b60006131a28773ffffffffffffffffffffffffffffffffffffffff16611b43565b6131ac57866131c6565b60015473ffffffffffffffffffffffffffffffffffffffff165b9050612fcf73ffffffffffffffffffffffffffffffffffffffff82168387611b1f565b600061320a8573ffffffffffffffffffffffffffffffffffffffff16611b43565b1561329357600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663d0e30db0846040518263ffffffff1660e01b81526004016000604051808303818588803b15801561327957600080fd5b505af115801561328d573d6000803e3d6000fd5b50505050505b60008281526008602052604090205473ffffffffffffffffffffffffffffffffffffffff168061331f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f52413a2045786368616e6765206e6f7420666f756e64000000000000000000006044820152606401610e4e565b80600061334173ffffffffffffffffffffffffffffffffffffffff8916611b43565b61334b5787613365565b60015473ffffffffffffffffffffffffffffffffffffffff165b905060006133888873ffffffffffffffffffffffffffffffffffffffff16611b43565b61339257876133ac565b60015473ffffffffffffffffffffffffffffffffffffffff165b90508073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16036133ed5786945050505050612d5e565b6040517fe6a4390500000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff838116600483015282811660248301526000919085169063e6a4390590604401602060405180830381865afa158015613465573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906134899190614b7e565b905073ffffffffffffffffffffffffffffffffffffffff81166134d8576040517f1fcdbb0900000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000806134fd73ffffffffffffffffffffffffffffffffffffffff841686868d61379c565b91995092509050811561356f578273ffffffffffffffffffffffffffffffffffffffff1663fff6cae96040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561355257600080fd5b505af1158015613566573d6000803e3d6000fd5b505050506135fb565b80156135fb576006546040517fbc25cf7700000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff91821660048201529084169063bc25cf7790602401600060405180830381600087803b1580156135e257600080fd5b505af11580156135f6573d6000803e3d6000fd5b505050505b61361c73ffffffffffffffffffffffffffffffffffffffff8616848c611b96565b508373ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1610156136f1576040517f022c0d9f000000000000000000000000000000000000000000000000000000008152600060048201819052602482018a905230604483015260806064830152608482015273ffffffffffffffffffffffffffffffffffffffff84169063022c0d9f9060a401600060405180830381600087803b1580156136d457600080fd5b505af11580156136e8573d6000803e3d6000fd5b5050505061378d565b6040517f022c0d9f0000000000000000000000000000000000000000000000000000000081526004810189905260006024820181905230604483015260806064830152608482015273ffffffffffffffffffffffffffffffffffffffff84169063022c0d9f9060a401600060405180830381600087803b15801561377457600080fd5b505af1158015613788573d6000803e3d6000fd5b505050505b50505050505050949350505050565b60008080806137c173ffffffffffffffffffffffffffffffffffffffff881689611a5a565b905060006137e573ffffffffffffffffffffffffffffffffffffffff88168a611a5a565b90506000808a73ffffffffffffffffffffffffffffffffffffffff16630902f1ac6040518163ffffffff1660e01b8152600401606060405180830381865afa158015613835573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906138599190614bb9565b50915091508873ffffffffffffffffffffffffffffffffffffffff168a73ffffffffffffffffffffffffffffffffffffffff16111561389457905b816dffffffffffffffffffffffffffff168410806138c15750806dffffffffffffffffffffffffffff1683105b9550851580156138f95750816dffffffffffffffffffffffffffff168411806138f95750806dffffffffffffffffffffffffffff1683115b94506000613909896103e5613992565b9050600061393161392a86856dffffffffffffffffffffffffffff1661399e565b8390613992565b905060006139658361395f6103e86139598b8a6dffffffffffffffffffffffffffff1661399e565b90613992565b906139b4565b9050801561397c5761397782826139c0565b61397f565b60005b9950505050505050509450945094915050565b6000611c418284614b1f565b60008183106139ad5781611c41565b5090919050565b6000611c418284614b36565b6000611c418284614c09565b6139d4614c44565b565b6000602082840312156139e857600080fd5b81357fffffffff0000000000000000000000000000000000000000000000000000000081168114611c4157600080fd5b60008083601f840112613a2a57600080fd5b50813567ffffffffffffffff811115613a4257600080fd5b602083019150836020828501011115613a5a57600080fd5b9250929050565b803560ff81168114613a7257600080fd5b919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b604051610100810167ffffffffffffffff81118282101715613aca57613aca613a77565b60405290565b604051601f82017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016810167ffffffffffffffff81118282101715613b1757613b17613a77565b604052919050565b600067ffffffffffffffff821115613b3957613b39613a77565b50601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01660200190565b600082601f830112613b7657600080fd5b8135613b89613b8482613b1f565b613ad0565b818152846020838601011115613b9e57600080fd5b816020850160208301376000918101602001919091529392505050565b600067ffffffffffffffff821115613bd557613bd5613a77565b5060051b60200190565b73ffffffffffffffffffffffffffffffffffffffff81168114611c5257600080fd5b600082601f830112613c1257600080fd5b81356020613c22613b8483613bbb565b82815260059290921b84018101918181019086841115613c4157600080fd5b8286015b84811015613c65578035613c5881613bdf565b8352918301918301613c45565b509695505050505050565b600082601f830112613c8157600080fd5b81356020613c91613b8483613bbb565b82815260059290921b84018101918181019086841115613cb057600080fd5b8286015b84811015613c655780358352918301918301613cb4565b600082601f830112613cdc57600080fd5b81356020613cec613b8483613bbb565b82815260059290921b84018101918181019086841115613d0b57600080fd5b8286015b84811015613c6557803567ffffffffffffffff811115613d2f5760008081fd5b613d3d8986838b0101613b65565b845250918301918301613d0f565b8015158114611c5257600080fd5b8035613a7281613d4b565b8035613a7281613bdf565b60006101008284031215613d8257600080fd5b613d8a613aa6565b9050813567ffffffffffffffff80821115613da457600080fd5b613db085838601613c01565b835260208401356020840152604084013560408401526060840135915080821115613dda57600080fd5b613de685838601613c70565b60608401526080840135915080821115613dff57600080fd5b613e0b85838601613ccb565b6080840152613e1c60a08501613d59565b60a0840152613e2d60c08501613d64565b60c084015260e0840135915080821115613e4657600080fd5b50613e5384828501613b65565b60e08301525092915050565b60008060008060008060008060006101008a8c031215613e7e57600080fd5b8935985060208a0135975060408a013567ffffffffffffffff80821115613ea457600080fd5b613eb08d838e01613a18565b9099509750879150613ec460608d01613a61565b965060808c0135955060a08c0135915080821115613ee157600080fd5b613eed8d838e01613b65565b945060c08c0135915080821115613f0357600080fd5b50613f108c828d01613d6f565b925050613f1f60e08b01613d64565b90509295985092959850929598565b60008060408385031215613f4157600080fd5b823591506020830135613f5381613bdf565b809150509250929050565b600060208284031215613f7057600080fd5b8135611c4181613bdf565b600080600080600080600080610100898b031215613f9857600080fd5b883567ffffffffffffffff80821115613fb057600080fd5b613fbc8c838d01613c01565b995060208b0135985060408b0135975060608b0135915080821115613fe057600080fd5b613fec8c838d01613c70565b965060808b013591508082111561400257600080fd5b5061400f8b828c01613ccb565b94505061401e60a08a01613d59565b925061402c60c08a01613d64565b915060e089013590509295985092959890939650565b60006020828403121561405457600080fd5b5035919050565b6000806020838503121561406e57600080fd5b823567ffffffffffffffff8082111561408657600080fd5b818501915085601f83011261409a57600080fd5b8135818111156140a957600080fd5b8660208260051b85010111156140be57600080fd5b60209290920196919550909350505050565b60005b838110156140eb5781810151838201526020016140d3565b50506000910152565b6000815180845261410c8160208601602086016140d0565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0169290920160200192915050565b600081518084526020808501808196508360051b8101915082860160005b858110156141865782840389526141748483516140f4565b9885019893509084019060010161415c565b5091979650505050505050565b602081526000611c41602083018461413e565b6000806000606084860312156141bb57600080fd5b83356141c681613bdf565b925060208401359150604084013567ffffffffffffffff8111156141e957600080fd5b6141f586828701613b65565b9150509250925092565b6000806040838503121561421257600080fd5b823561421d81613bdf565b91506020830135613f5381613bdf565b600080600080600080600060e0888a03121561424857600080fd5b873567ffffffffffffffff8082111561426057600080fd5b61426c8b838c01613c01565b985060208a0135975060408a0135965060608a013591508082111561429057600080fd5b61429c8b838c01613c70565b955060808a01359150808211156142b257600080fd5b506142bf8a828b01613ccb565b93505060a08801356142d081613d4b565b915060c08801356142e081613bdf565b8091505092959891949750929550565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b81810381811115610691576106916142f0565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b8183528181602085013750600060208284010152600060207fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f840116840101905092915050565b600061010061440a8388805182526020810151602083015260408101516040830152606081015173ffffffffffffffffffffffffffffffffffffffff8082166060850152806080840151166080850152505060a081015160a08301525050565b8060c084015261441c818401876140f4565b905082810360e0840152612396818587614361565b60006101206144918389805182526020810151602083015260408101516040830152606081015173ffffffffffffffffffffffffffffffffffffffff8082166060850152806080840151166080850152505060a081015160a08301525050565b8060c08401526144a3818401886140f4565b905082810360e08401526144b8818688614361565b90508281036101008401526144cd81856140f4565b98975050505050505050565b80356020831015610691577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602084900360031b1b1692915050565b600081518084526020808501945080840160005b8381101561455b57815173ffffffffffffffffffffffffffffffffffffffff1687529582019590820190600101614529565b509495945050505050565b600081518084526020808501945080840160005b8381101561455b5781518752958201959082019060010161457a565b60a0815260006145a960a0830188614515565b86602084015285604084015282810360608401526145c78186614566565b9150508260808301529695505050505050565b60008083357fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe184360301811261460f57600080fd5b83018035915067ffffffffffffffff82111561462a57600080fd5b602001915036819003821315613a5a57600080fd5b82848237600083820160008152835161465c8183602088016140d0565b0195945050505050565b60007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203614697576146976142f0565b5060010190565b6000602082840312156146b057600080fd5b5051919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052600160045260246000fd5b600082601f8301126146f757600080fd5b81516020614707613b8483613bbb565b82815260059290921b8401810191818101908684111561472657600080fd5b8286015b84811015613c6557805161473d81613bdf565b835291830191830161472a565b600082601f83011261475b57600080fd5b8151614769613b8482613b1f565b81815284602083860101111561477e57600080fd5b612d5e8260208301602087016140d0565b600082601f8301126147a057600080fd5b815160206147b0613b8483613bbb565b82815260059290921b840181019181810190868411156147cf57600080fd5b8286015b84811015613c6557805167ffffffffffffffff8111156147f35760008081fd5b6148018986838b010161474a565b8452509183019183016147d3565b600082601f83011261482057600080fd5b81516020614830613b8483613bbb565b82815260059290921b8401810191818101908684111561484f57600080fd5b8286015b84811015613c655780518352918301918301614853565b8051613a7281613bdf565b8051613a7281613d4b565b600080600080600080600060e0888a03121561489b57600080fd5b875167ffffffffffffffff808211156148b357600080fd5b6148bf8b838c016146e6565b985060208a0151975060408a01519150808211156148dc57600080fd5b6148e88b838c0161478f565b965060608a01519150808211156148fe57600080fd5b61490a8b838c0161480f565b955061491860808b0161486a565b945061492660a08b01614875565b935060c08a015191508082111561493c57600080fd5b506149498a828b0161474a565b91505092959891949750929550565b60e08152600061496b60e083018a614515565b88602084015287604084015282810360608401526149898188614566565b9050828103608084015261499d818761413e565b94151560a0840152505073ffffffffffffffffffffffffffffffffffffffff9190911660c09091015295945050505050565b600082516149e18184602087016140d0565b9190910192915050565b80516020808301519190811015614a2a577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8160200360031b1b821691505b50919050565b73ffffffffffffffffffffffffffffffffffffffff84168152826020820152606060408201526000614a6560608301846140f4565b95945050505050565b600060208284031215614a8057600080fd5b8151611c4181613d4b565b7f416363657373436f6e74726f6c3a206163636f756e7420000000000000000000815260008351614ac38160178501602088016140d0565b7f206973206d697373696e6720726f6c65200000000000000000000000000000006017918401918201528351614b008160288401602088016140d0565b01602801949350505050565b602081526000611c4160208301846140f4565b8082028115828204841417610691576106916142f0565b80820180821115610691576106916142f0565b600081614b5857614b586142f0565b507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0190565b600060208284031215614b9057600080fd5b8151611c4181613bdf565b80516dffffffffffffffffffffffffffff81168114613a7257600080fd5b600080600060608486031215614bce57600080fd5b614bd784614b9b565b9250614be560208501614b9b565b9150604084015163ffffffff81168114614bfe57600080fd5b809150509250925092565b600082614c3f577f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b500490565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052605160045260246000fdfe416464726573733a206c6f772d6c6576656c2064656c65676174652063616c6c206661696c6564a264697066735822122007c1e9de304dd51a15120baadaa17d494f8e5a23f1c132c08125823791857f3d64736f6c63430008140033000000000000000000000000c21e4ebd1d92036cb467b53fe3258f219d909eb9000000000000000000000000eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2000000000000000000000000b1b64005b11350a94c4d069eff4215592d98f2e2
Deployed Bytecode
0x6080604052600436106101b05760003560e01c806391d14854116100ec578063d00a2d5f1161008a578063da6689f011610064578063da6689f01461057e578063e2a4ac2d1461059e578063e738aa8d146105cb578063f2fde38b146105de57600080fd5b8063d00a2d5f14610511578063d4ee1d9014610531578063d547741f1461055e57600080fd5b8063a2625efc116100c6578063a2625efc14610454578063ac9650d814610481578063adb88982146104ae578063c75a57e4146104ce57600080fd5b806391d14854146103cc5780639e99ec391461041f578063a217fddf1461043f57600080fd5b8063248a9ca3116101595780632f2ff15d116101335780632f2ff15d1461034a57806336568abe1461036a5780634e71e0c81461038a5780637fe689171461039f57600080fd5b8063248a9ca31461029b578063285f94a0146102cb5780632e4763371461031d57600080fd5b8063169651601161018a578063169651601461024857806318137a4114610268578063183d9c951461028857600080fd5b806301ffc9a7146101bc57806306135b7c146101f1578063086efdc41461020657600080fd5b366101b757005b600080fd5b3480156101c857600080fd5b506101dc6101d73660046139d6565b6105fe565b60405190151581526020015b60405180910390f35b6102046101ff366004613e5f565b610697565b005b34801561021257600080fd5b5061023a7fafffb39e42e2a1496bc4305c8783f904fa76641e3895205b6002a03336ee097581565b6040519081526020016101e8565b34801561025457600080fd5b50610204610263366004613f2e565b610ac3565b34801561027457600080fd5b50610204610283366004613f5e565b610c01565b61023a610296366004613f7b565b610cd9565b3480156102a757600080fd5b5061023a6102b6366004614042565b60009081526003602052604090206001015490565b3480156102d757600080fd5b506005546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b60405173ffffffffffffffffffffffffffffffffffffffff90911681526020016101e8565b34801561032957600080fd5b506002546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b34801561035657600080fd5b50610204610365366004613f2e565b610d83565b34801561037657600080fd5b50610204610385366004613f2e565b610dad565b34801561039657600080fd5b50610204610e65565b3480156103ab57600080fd5b506004546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b3480156103d857600080fd5b506101dc6103e7366004613f2e565b600091825260036020908152604080842073ffffffffffffffffffffffffffffffffffffffff93909316845291905290205460ff1690565b34801561042b57600080fd5b5061020461043a366004613f5e565b610ef6565b34801561044b57600080fd5b5061023a600081565b34801561046057600080fd5b506006546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b34801561048d57600080fd5b506104a161049c36600461405b565b610fc6565b6040516101e89190614193565b3480156104ba57600080fd5b506102046104c9366004613f5e565b6110b9565b3480156104da57600080fd5b506102f86104e9366004614042565b60086020526000908152604090205473ffffffffffffffffffffffffffffffffffffffff1681565b34801561051d57600080fd5b5061020461052c3660046141a6565b611200565b34801561053d57600080fd5b506007546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b34801561056a57600080fd5b50610204610579366004613f2e565b611282565b34801561058a57600080fd5b506102046105993660046141ff565b6112a7565b3480156105aa57600080fd5b506001546102f89073ffffffffffffffffffffffffffffffffffffffff1681565b61023a6105d936600461422d565b6113cf565b3480156105ea57600080fd5b506102046105f9366004613f5e565b611479565b60007fffffffff0000000000000000000000000000000000000000000000000000000082167f7965db0b00000000000000000000000000000000000000000000000000000000148061069157507f01ffc9a7000000000000000000000000000000000000000000000000000000007fffffffff000000000000000000000000000000000000000000000000000000008316145b92915050565b6106be826000015183602001518460400151856060015186608001518760a0015130611544565b5081518051600091906106d39060019061431f565b815181106106e3576106e3614332565b602002602001015190508560ff1660000361084157600061071a73ffffffffffffffffffffffffffffffffffffffff831630611a5a565b6004549091506107449073ffffffffffffffffffffffffffffffffffffffff848116911683611b1f565b60045473ffffffffffffffffffffffffffffffffffffffff9081169063f452ed4d90610771908516611b43565b61077c57600061077e565b825b6040518060c001604052808f81526020018581526020018a866107a1919061431f565b81526020018673ffffffffffffffffffffffffffffffffffffffff1681526020018773ffffffffffffffffffffffffffffffffffffffff1681526020018e8152508760e001518d8d6040518663ffffffff1660e01b815260040161080894939291906143aa565b6000604051808303818588803b15801561082157600080fd5b505af1158015610835573d6000803e3d6000fd5b50505050505050610ab8565b8560ff166001036109e457600061086e73ffffffffffffffffffffffffffffffffffffffff831630611a5a565b600454604086015191925061089d9173ffffffffffffffffffffffffffffffffffffffff858116921690611b1f565b60045473ffffffffffffffffffffffffffffffffffffffff90811690630421caf0906108ca908516611b43565b6108d55760006108d7565b825b6040518060c001604052808f8152602001886040015181526020018a8960400151610902919061431f565b81526020018673ffffffffffffffffffffffffffffffffffffffff1681526020018773ffffffffffffffffffffffffffffffffffffffff1681526020018e8152508760e001518d8d8b6040518763ffffffff1660e01b815260040161096b959493929190614431565b6000604051808303818588803b15801561098457600080fd5b505af1158015610998573d6000803e3d6000fd5b505050505083604001518111156109dd576109db838560400151836109bd919061431f565b73ffffffffffffffffffffffffffffffffffffffff85169190611b96565b505b5050610ab8565b8560ff16600203610ab6576000610a1173ffffffffffffffffffffffffffffffffffffffff831630611a5a565b600454909150610a3b9073ffffffffffffffffffffffffffffffffffffffff848116911683611b1f565b60045473ffffffffffffffffffffffffffffffffffffffff16633e28c7d2348d8d610a668d8f6144d9565b6040517fffffffff0000000000000000000000000000000000000000000000000000000060e087901b16815260048101939093526024830191909152604482015260648101859052608401610808565b505b505050505050505050565b7fafffb39e42e2a1496bc4305c8783f904fa76641e3895205b6002a03336ee0975610aed81611c48565b82600003610b27576040517f196de28b00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b73ffffffffffffffffffffffffffffffffffffffff8216610b74576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008381526008602090815260409182902080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff86169081179091558251868152918201527f11f5b5361c420666dc3452b8107601e1702f94b951f15bd93c26262c8147874e91015b60405180910390a1505050565b6000610c0c81611c48565b73ffffffffffffffffffffffffffffffffffffffff8216610c59576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600480547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff8416908117909155604080519182523360208301527f61b6fe2922f0180b98d305419bf3dcf432f5d8df894388c61c1f599c2194ec1d91015b60405180910390a15050565b6000610cea898989898989896113cf565b6040517f73776170496e53616d65436861696e0000000000000000000000000000000000815290915073ffffffffffffffffffffffffffffffffffffffff8416903390600f0160405180910390207fe3716b48e7d01d727d68472c873c69c2c4407365e4f68b2b6c8dd4569cd16dc78c8c868c89604051610d6f959493929190614596565b60405180910390a498975050505050505050565b600082815260036020526040902060010154610d9e81611c48565b610da88383611c55565b505050565b73ffffffffffffffffffffffffffffffffffffffff81163314610e57576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602f60248201527f416363657373436f6e74726f6c3a2063616e206f6e6c792072656e6f756e636560448201527f20726f6c657320666f722073656c66000000000000000000000000000000000060648201526084015b60405180910390fd5b610e618282611d49565b5050565b60075473ffffffffffffffffffffffffffffffffffffffff163314610eb6576040517f821de23100000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610ec1600033611e04565b6040513381527f85a8c8100c1d48466310240af79644f50cdae7d09cb4820d60dfc5b2462f0fd09060200160405180910390a1565b6000610f0181611c48565b73ffffffffffffffffffffffffffffffffffffffff8216610f4e576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600580547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff8416908117909155604080519182523360208301527f45085069ced50a4d2e82656a82ba8f5a1a28694489d8ea262252e40899a930a29101610ccd565b6040805160008152602081019091526060908267ffffffffffffffff811115610ff157610ff1613a77565b60405190808252806020026020018201604052801561102457816020015b606081526020019060019003908161100f5790505b50915060005b838110156110b1576110813086868481811061104857611048614332565b905060200281019061105a91906145da565b8560405160200161106d9392919061463f565b604051602081830303815290604052611e0e565b83828151811061109357611093614332565b602002602001018190525080806110a990614666565b91505061102a565b505092915050565b60006110c481611c48565b73ffffffffffffffffffffffffffffffffffffffff8216611111576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b73ffffffffffffffffffffffffffffffffffffffff821660009081527f3ab9add546f7e3b43565da9bc7e65e47b969d3856bb01a7dcac7bc8ff5080cbd602052604090205460ff1615611190576040517fe4e970d000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6111ba7fafffb39e42e2a1496bc4305c8783f904fa76641e3895205b6002a03336ee097583611e04565b60405173ffffffffffffffffffffffffffffffffffffffff831681527fd2b69bde5a989d23ef605a0dd4e0edf7662105058ab69f2559e482857b599ed890602001610ccd565b60045473ffffffffffffffffffffffffffffffffffffffff163314801590611240575060055473ffffffffffffffffffffffffffffffffffffffff163314155b15611277576040517f48f5c3ed00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610da8838383611e33565b60008281526003602052604090206001015461129d81611c48565b610da88383611d49565b60006112b281611c48565b73ffffffffffffffffffffffffffffffffffffffff83166112ff576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b73ffffffffffffffffffffffffffffffffffffffff821661134c576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6002805473ffffffffffffffffffffffffffffffffffffffff8581167fffffffffffffffffffffffff0000000000000000000000000000000000000000928316179092556001805492851692909116821790556040519081527fcc4166d40cf68104cdf710f65ce022efd296742868a22b42e4695f89b886363690602001610bf4565b60006113e088888888888888611544565b6040517f737761704d756c746957697468526563697069656e7400000000000000000000815290915073ffffffffffffffffffffffffffffffffffffffff831690339060160160405180910390207fc40fae9d5f584875c393ac222c6f88b6c9dced1e9cc6251483648ac2e902c8b08b8b868b6000604051611466959493929190614596565b60405180910390a4979650505050505050565b600061148481611c48565b73ffffffffffffffffffffffffffffffffffffffff82166114d1576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600780547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff84169081179091556040519081527fcfaaa26691e16e66e73290fc725eee1a6b4e0e693a1640484937aac25ffb55a490602001610ccd565b600073ffffffffffffffffffffffffffffffffffffffff8216611593576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8451600189516115a3919061431f565b146115da576040517fcc513c7200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b826116a257611618886000815181106115f5576115f5614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16611b43565b15801561162457503415155b1561165b576040517f694b150300000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6116a23330898b60008151811061167457611674614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff166121a6909392919063ffffffff16565b6116e530896000815181106116b9576116b9614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16611a5a90919063ffffffff16565b905060008860018a516116f8919061431f565b8151811061170857611708614332565b602002602001015190506000600190505b895181101561183e5789818151811061173457611734614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff168a600183611760919061431f565b8151811061177057611770614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16031561182c576118298a6117a360018461431f565b815181106117b3576117b3614332565b60200260200101518b83815181106117cd576117cd614332565b60200260200101518560008b6001876117e6919061431f565b815181106117f6576117f6614332565b60200260200101518b60018861180c919061431f565b8151811061181c5761181c614332565b6020026020010151612386565b92505b8061183681614666565b915050611719565b5061185e8173ffffffffffffffffffffffffffffffffffffffff16611b43565b1561197b576001546040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015273ffffffffffffffffffffffffffffffffffffffff909116906370a0823190602401602060405180830381865afa1580156118d1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118f5919061469e565b6001546040517f2e1a7d4d0000000000000000000000000000000000000000000000000000000081526004810183905291935073ffffffffffffffffffffffffffffffffffffffff1690632e1a7d4d90602401600060405180830381600087803b15801561196257600080fd5b505af1158015611976573d6000803e3d6000fd5b505050505b73ffffffffffffffffffffffffffffffffffffffff83163014611a4e5760006119ba73ffffffffffffffffffffffffffffffffffffffff831685611a5a565b90506119dd73ffffffffffffffffffffffffffffffffffffffff83168585611b96565b506000611a0073ffffffffffffffffffffffffffffffffffffffff841686611a5a565b90506000611a0e838361431f565b9050898111611a49576040517f1fbaba3500000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b935050505b50979650505050505050565b6000611a6583611b43565b15611a88575073ffffffffffffffffffffffffffffffffffffffff811631610691565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff83811660048301528416906370a0823190602401602060405180830381865afa158015611af4573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b18919061469e565b9050610691565b611b2883611b43565b610da857611b38838360006123a1565b610da88383836123a1565b600073ffffffffffffffffffffffffffffffffffffffff82161580610691575073ffffffffffffffffffffffffffffffffffffffff821673eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee1492915050565b600081600003611ba857506001611c41565b611bb184611b43565b15611c325760008373ffffffffffffffffffffffffffffffffffffffff168360405160006040518083038185875af1925050503d8060008114611c10576040519150601f19603f3d011682016040523d82523d6000602084013e611c15565b606091505b5090915050600181151514611c2c57611c2c6146b7565b50611c41565b611c3d848484612537565b5060015b9392505050565b611c5281336126c6565b50565b600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516845290915290205460ff16610e6157600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff85168452909152902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00166001179055611ceb3390565b73ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16837f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d60405160405180910390a45050565b600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516845290915290205460ff1615610e6157600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516808552925280832080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016905551339285917ff6391f5c32d9c69d2a47ea670b442974b53935d1edc7fd64eb21e047a839171b9190a45050565b610e618282611c55565b6060611c418383604051806060016040528060278152602001614c7460279139612780565b600080600080600080600087806020019051810190611e529190614880565b965096509650965096509650965086600081518110611e7357611e73614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff168a73ffffffffffffffffffffffffffffffffffffffff1614611edf576040517f198b098900000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b606060003073ffffffffffffffffffffffffffffffffffffffff1663e738aa8d60e01b8a8d8b8a8c60018c604051602401611f209796959493929190614958565b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529181526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff00000000000000000000000000000000000000000000000000000000909416939093179092529051611fa991906149cf565b6000604051808303816000865af19150503d8060008114611fe6576040519150601f19603f3d011682016040523d82523d6000602084013e611feb565b606091505b50925090508061203c5761203a858c8b60008151811061200d5761200d614332565b602002602001015173ffffffffffffffffffffffffffffffffffffffff16611b969092919063ffffffff16565b505b83156121985760008161204f578b612058565b612058836149eb565b9050600082612067578d61208f565b8a60018c51612076919061431f565b8151811061208657612086614332565b60200260200101515b90508673ffffffffffffffffffffffffffffffffffffffff1663d00a2d5f60e01b8284886040516024016120c593929190614a30565b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529181526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff0000000000000000000000000000000000000000000000000000000090941693909317909252905161214e91906149cf565b6000604051808303816000865af19150503d806000811461218b576040519150601f19603f3d011682016040523d82523d6000602084013e612190565b606091505b509450925050505b505050505050505050505050565b8015612380576121b584611b43565b156123745773ffffffffffffffffffffffffffffffffffffffff8316331480156121df5750803410155b61226b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602a60248201527f57726f6e67207573616765206f66204554482e756e6976657273616c5472616e60448201527f7366657246726f6d2829000000000000000000000000000000000000000000006064820152608401610e4e565b73ffffffffffffffffffffffffffffffffffffffff821630146123005760008273ffffffffffffffffffffffffffffffffffffffff168260405160006040518083038185875af1925050503d80600081146122e2576040519150601f19603f3d011682016040523d82523d6000602084013e6122e7565b606091505b50909150506001811515146122fe576122fe6146b7565b505b8034111561236f576000336123153484612805565b604051600081818185875af1925050503d8060008114612351576040519150601f19603f3d011682016040523d82523d6000602084013e612356565b606091505b509091505060018115151461236d5761236d6146b7565b505b612380565b61238084848484612811565b50505050565b60006123968787878787876129b0565b979650505050505050565b6040805173ffffffffffffffffffffffffffffffffffffffff8481166024830152604480830185905283518084039091018152606490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167f095ea7b300000000000000000000000000000000000000000000000000000000179052915160009283929087169161243891906149cf565b6000604051808303816000865af19150503d8060008114612475576040519150601f19603f3d011682016040523d82523d6000602084013e61247a565b606091505b50915091508180156124a45750805115806124a45750808060200190518101906124a49190614a6e565b612530576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f5472616e7366657248656c7065723a3a73616665417070726f76653a2061707060448201527f726f7665206661696c65640000000000000000000000000000000000000000006064820152608401610e4e565b5050505050565b6040805173ffffffffffffffffffffffffffffffffffffffff8481166024830152604480830185905283518084039091018152606490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fa9059cbb0000000000000000000000000000000000000000000000000000000017905291516000928392908716916125ce91906149cf565b6000604051808303816000865af19150503d806000811461260b576040519150601f19603f3d011682016040523d82523d6000602084013e612610565b606091505b509150915081801561263a57508051158061263a57508080602001905181019061263a9190614a6e565b612530576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602d60248201527f5472616e7366657248656c7065723a3a736166655472616e736665723a20747260448201527f616e73666572206661696c6564000000000000000000000000000000000000006064820152608401610e4e565b600082815260036020908152604080832073ffffffffffffffffffffffffffffffffffffffff8516845290915290205460ff16610e615761270681612a5c565b612711836020612a7b565b604051602001612722929190614a8b565b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0818403018152908290527f08c379a0000000000000000000000000000000000000000000000000000000008252610e4e91600401614b0c565b60606000808573ffffffffffffffffffffffffffffffffffffffff16856040516127aa91906149cf565b600060405180830381855af49150503d80600081146127e5576040519150601f19603f3d011682016040523d82523d6000602084013e6127ea565b606091505b50915091506127fb86838387612cbe565b9695505050505050565b6000611c41828461431f565b6040805173ffffffffffffffffffffffffffffffffffffffff85811660248301528481166044830152606480830185905283518084039091018152608490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167f23b872dd0000000000000000000000000000000000000000000000000000000017905291516000928392908816916128b091906149cf565b6000604051808303816000865af19150503d80600081146128ed576040519150601f19603f3d011682016040523d82523d6000602084013e6128f2565b606091505b509150915081801561291c57508051158061291c57508080602001905181019061291c9190614a6e565b6129a8576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603160248201527f5472616e7366657248656c7065723a3a7472616e7366657246726f6d3a20747260448201527f616e7366657246726f6d206661696c65640000000000000000000000000000006064820152608401610e4e565b505050505050565b60008573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff16036129ec5750836127fb565b6139cc6129f884612d66565b90506000612a1c73ffffffffffffffffffffffffffffffffffffffff8a1630611a5a565b9050612a2f89898387898763ffffffff16565b612a4f73ffffffffffffffffffffffffffffffffffffffff891630611a5a565b9998505050505050505050565b606061069173ffffffffffffffffffffffffffffffffffffffff831660145b60606000612a8a836002614b1f565b612a95906002614b36565b67ffffffffffffffff811115612aad57612aad613a77565b6040519080825280601f01601f191660200182016040528015612ad7576020820181803683370190505b5090507f300000000000000000000000000000000000000000000000000000000000000081600081518110612b0e57612b0e614332565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a9053507f780000000000000000000000000000000000000000000000000000000000000081600181518110612b7157612b71614332565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a9053506000612bad846002614b1f565b612bb8906001614b36565b90505b6001811115612c55577f303132333435363738396162636465660000000000000000000000000000000085600f1660108110612bf957612bf9614332565b1a60f81b828281518110612c0f57612c0f614332565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a90535060049490941c93612c4e81614b49565b9050612bbb565b508315611c41576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e746044820152606401610e4e565b60608315612d54578251600003612d4d5773ffffffffffffffffffffffffffffffffffffffff85163b612d4d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606401610e4e565b5081612d5e565b612d5e8383612da4565b949350505050565b6139cc6103e982108015612d7b575060018210155b15612d895750612de8919050565b816107d203612d9b5750612df4919050565b5061307d919050565b815115612db45781518083602001fd5b806040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e4e9190614b0c565b6129a8858585846131e9565b60008181526008602052604090205473ffffffffffffffffffffffffffffffffffffffff1680612e50576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612e6f8673ffffffffffffffffffffffffffffffffffffffff16611b43565b15612ef857600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663d0e30db0856040518263ffffffff1660e01b81526004016000604051808303818588803b158015612ede57600080fd5b505af1158015612ef2573d6000803e3d6000fd5b50505050505b6000612f198773ffffffffffffffffffffffffffffffffffffffff16611b43565b612f235786612f3d565b60015473ffffffffffffffffffffffffffffffffffffffff165b9050612fcf8273ffffffffffffffffffffffffffffffffffffffff1663d2c4b5986040518163ffffffff1660e01b8152600401602060405180830381865afa158015612f8d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612fb19190614b7e565b73ffffffffffffffffffffffffffffffffffffffff83169087611b1f565b60008273ffffffffffffffffffffffffffffffffffffffff1685604051612ff691906149cf565b6000604051808303816000865af19150503d8060008114613033576040519150601f19603f3d011682016040523d82523d6000602084013e613038565b606091505b5050905080613073576040517f5987947500000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5050505050505050565b60008181526008602052604090205473ffffffffffffffffffffffffffffffffffffffff16806130d9576040517fd92e233d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6130f88673ffffffffffffffffffffffffffffffffffffffff16611b43565b1561318157600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663d0e30db0856040518263ffffffff1660e01b81526004016000604051808303818588803b15801561316757600080fd5b505af115801561317b573d6000803e3d6000fd5b50505050505b60006131a28773ffffffffffffffffffffffffffffffffffffffff16611b43565b6131ac57866131c6565b60015473ffffffffffffffffffffffffffffffffffffffff165b9050612fcf73ffffffffffffffffffffffffffffffffffffffff82168387611b1f565b600061320a8573ffffffffffffffffffffffffffffffffffffffff16611b43565b1561329357600160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1663d0e30db0846040518263ffffffff1660e01b81526004016000604051808303818588803b15801561327957600080fd5b505af115801561328d573d6000803e3d6000fd5b50505050505b60008281526008602052604090205473ffffffffffffffffffffffffffffffffffffffff168061331f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f52413a2045786368616e6765206e6f7420666f756e64000000000000000000006044820152606401610e4e565b80600061334173ffffffffffffffffffffffffffffffffffffffff8916611b43565b61334b5787613365565b60015473ffffffffffffffffffffffffffffffffffffffff165b905060006133888873ffffffffffffffffffffffffffffffffffffffff16611b43565b61339257876133ac565b60015473ffffffffffffffffffffffffffffffffffffffff165b90508073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16036133ed5786945050505050612d5e565b6040517fe6a4390500000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff838116600483015282811660248301526000919085169063e6a4390590604401602060405180830381865afa158015613465573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906134899190614b7e565b905073ffffffffffffffffffffffffffffffffffffffff81166134d8576040517f1fcdbb0900000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000806134fd73ffffffffffffffffffffffffffffffffffffffff841686868d61379c565b91995092509050811561356f578273ffffffffffffffffffffffffffffffffffffffff1663fff6cae96040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561355257600080fd5b505af1158015613566573d6000803e3d6000fd5b505050506135fb565b80156135fb576006546040517fbc25cf7700000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff91821660048201529084169063bc25cf7790602401600060405180830381600087803b1580156135e257600080fd5b505af11580156135f6573d6000803e3d6000fd5b505050505b61361c73ffffffffffffffffffffffffffffffffffffffff8616848c611b96565b508373ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1610156136f1576040517f022c0d9f000000000000000000000000000000000000000000000000000000008152600060048201819052602482018a905230604483015260806064830152608482015273ffffffffffffffffffffffffffffffffffffffff84169063022c0d9f9060a401600060405180830381600087803b1580156136d457600080fd5b505af11580156136e8573d6000803e3d6000fd5b5050505061378d565b6040517f022c0d9f0000000000000000000000000000000000000000000000000000000081526004810189905260006024820181905230604483015260806064830152608482015273ffffffffffffffffffffffffffffffffffffffff84169063022c0d9f9060a401600060405180830381600087803b15801561377457600080fd5b505af1158015613788573d6000803e3d6000fd5b505050505b50505050505050949350505050565b60008080806137c173ffffffffffffffffffffffffffffffffffffffff881689611a5a565b905060006137e573ffffffffffffffffffffffffffffffffffffffff88168a611a5a565b90506000808a73ffffffffffffffffffffffffffffffffffffffff16630902f1ac6040518163ffffffff1660e01b8152600401606060405180830381865afa158015613835573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906138599190614bb9565b50915091508873ffffffffffffffffffffffffffffffffffffffff168a73ffffffffffffffffffffffffffffffffffffffff16111561389457905b816dffffffffffffffffffffffffffff168410806138c15750806dffffffffffffffffffffffffffff1683105b9550851580156138f95750816dffffffffffffffffffffffffffff168411806138f95750806dffffffffffffffffffffffffffff1683115b94506000613909896103e5613992565b9050600061393161392a86856dffffffffffffffffffffffffffff1661399e565b8390613992565b905060006139658361395f6103e86139598b8a6dffffffffffffffffffffffffffff1661399e565b90613992565b906139b4565b9050801561397c5761397782826139c0565b61397f565b60005b9950505050505050509450945094915050565b6000611c418284614b1f565b60008183106139ad5781611c41565b5090919050565b6000611c418284614b36565b6000611c418284614c09565b6139d4614c44565b565b6000602082840312156139e857600080fd5b81357fffffffff0000000000000000000000000000000000000000000000000000000081168114611c4157600080fd5b60008083601f840112613a2a57600080fd5b50813567ffffffffffffffff811115613a4257600080fd5b602083019150836020828501011115613a5a57600080fd5b9250929050565b803560ff81168114613a7257600080fd5b919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b604051610100810167ffffffffffffffff81118282101715613aca57613aca613a77565b60405290565b604051601f82017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016810167ffffffffffffffff81118282101715613b1757613b17613a77565b604052919050565b600067ffffffffffffffff821115613b3957613b39613a77565b50601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01660200190565b600082601f830112613b7657600080fd5b8135613b89613b8482613b1f565b613ad0565b818152846020838601011115613b9e57600080fd5b816020850160208301376000918101602001919091529392505050565b600067ffffffffffffffff821115613bd557613bd5613a77565b5060051b60200190565b73ffffffffffffffffffffffffffffffffffffffff81168114611c5257600080fd5b600082601f830112613c1257600080fd5b81356020613c22613b8483613bbb565b82815260059290921b84018101918181019086841115613c4157600080fd5b8286015b84811015613c65578035613c5881613bdf565b8352918301918301613c45565b509695505050505050565b600082601f830112613c8157600080fd5b81356020613c91613b8483613bbb565b82815260059290921b84018101918181019086841115613cb057600080fd5b8286015b84811015613c655780358352918301918301613cb4565b600082601f830112613cdc57600080fd5b81356020613cec613b8483613bbb565b82815260059290921b84018101918181019086841115613d0b57600080fd5b8286015b84811015613c6557803567ffffffffffffffff811115613d2f5760008081fd5b613d3d8986838b0101613b65565b845250918301918301613d0f565b8015158114611c5257600080fd5b8035613a7281613d4b565b8035613a7281613bdf565b60006101008284031215613d8257600080fd5b613d8a613aa6565b9050813567ffffffffffffffff80821115613da457600080fd5b613db085838601613c01565b835260208401356020840152604084013560408401526060840135915080821115613dda57600080fd5b613de685838601613c70565b60608401526080840135915080821115613dff57600080fd5b613e0b85838601613ccb565b6080840152613e1c60a08501613d59565b60a0840152613e2d60c08501613d64565b60c084015260e0840135915080821115613e4657600080fd5b50613e5384828501613b65565b60e08301525092915050565b60008060008060008060008060006101008a8c031215613e7e57600080fd5b8935985060208a0135975060408a013567ffffffffffffffff80821115613ea457600080fd5b613eb08d838e01613a18565b9099509750879150613ec460608d01613a61565b965060808c0135955060a08c0135915080821115613ee157600080fd5b613eed8d838e01613b65565b945060c08c0135915080821115613f0357600080fd5b50613f108c828d01613d6f565b925050613f1f60e08b01613d64565b90509295985092959850929598565b60008060408385031215613f4157600080fd5b823591506020830135613f5381613bdf565b809150509250929050565b600060208284031215613f7057600080fd5b8135611c4181613bdf565b600080600080600080600080610100898b031215613f9857600080fd5b883567ffffffffffffffff80821115613fb057600080fd5b613fbc8c838d01613c01565b995060208b0135985060408b0135975060608b0135915080821115613fe057600080fd5b613fec8c838d01613c70565b965060808b013591508082111561400257600080fd5b5061400f8b828c01613ccb565b94505061401e60a08a01613d59565b925061402c60c08a01613d64565b915060e089013590509295985092959890939650565b60006020828403121561405457600080fd5b5035919050565b6000806020838503121561406e57600080fd5b823567ffffffffffffffff8082111561408657600080fd5b818501915085601f83011261409a57600080fd5b8135818111156140a957600080fd5b8660208260051b85010111156140be57600080fd5b60209290920196919550909350505050565b60005b838110156140eb5781810151838201526020016140d3565b50506000910152565b6000815180845261410c8160208601602086016140d0565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0169290920160200192915050565b600081518084526020808501808196508360051b8101915082860160005b858110156141865782840389526141748483516140f4565b9885019893509084019060010161415c565b5091979650505050505050565b602081526000611c41602083018461413e565b6000806000606084860312156141bb57600080fd5b83356141c681613bdf565b925060208401359150604084013567ffffffffffffffff8111156141e957600080fd5b6141f586828701613b65565b9150509250925092565b6000806040838503121561421257600080fd5b823561421d81613bdf565b91506020830135613f5381613bdf565b600080600080600080600060e0888a03121561424857600080fd5b873567ffffffffffffffff8082111561426057600080fd5b61426c8b838c01613c01565b985060208a0135975060408a0135965060608a013591508082111561429057600080fd5b61429c8b838c01613c70565b955060808a01359150808211156142b257600080fd5b506142bf8a828b01613ccb565b93505060a08801356142d081613d4b565b915060c08801356142e081613bdf565b8091505092959891949750929550565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b81810381811115610691576106916142f0565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b8183528181602085013750600060208284010152600060207fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f840116840101905092915050565b600061010061440a8388805182526020810151602083015260408101516040830152606081015173ffffffffffffffffffffffffffffffffffffffff8082166060850152806080840151166080850152505060a081015160a08301525050565b8060c084015261441c818401876140f4565b905082810360e0840152612396818587614361565b60006101206144918389805182526020810151602083015260408101516040830152606081015173ffffffffffffffffffffffffffffffffffffffff8082166060850152806080840151166080850152505060a081015160a08301525050565b8060c08401526144a3818401886140f4565b905082810360e08401526144b8818688614361565b90508281036101008401526144cd81856140f4565b98975050505050505050565b80356020831015610691577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602084900360031b1b1692915050565b600081518084526020808501945080840160005b8381101561455b57815173ffffffffffffffffffffffffffffffffffffffff1687529582019590820190600101614529565b509495945050505050565b600081518084526020808501945080840160005b8381101561455b5781518752958201959082019060010161457a565b60a0815260006145a960a0830188614515565b86602084015285604084015282810360608401526145c78186614566565b9150508260808301529695505050505050565b60008083357fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe184360301811261460f57600080fd5b83018035915067ffffffffffffffff82111561462a57600080fd5b602001915036819003821315613a5a57600080fd5b82848237600083820160008152835161465c8183602088016140d0565b0195945050505050565b60007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203614697576146976142f0565b5060010190565b6000602082840312156146b057600080fd5b5051919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052600160045260246000fd5b600082601f8301126146f757600080fd5b81516020614707613b8483613bbb565b82815260059290921b8401810191818101908684111561472657600080fd5b8286015b84811015613c6557805161473d81613bdf565b835291830191830161472a565b600082601f83011261475b57600080fd5b8151614769613b8482613b1f565b81815284602083860101111561477e57600080fd5b612d5e8260208301602087016140d0565b600082601f8301126147a057600080fd5b815160206147b0613b8483613bbb565b82815260059290921b840181019181810190868411156147cf57600080fd5b8286015b84811015613c6557805167ffffffffffffffff8111156147f35760008081fd5b6148018986838b010161474a565b8452509183019183016147d3565b600082601f83011261482057600080fd5b81516020614830613b8483613bbb565b82815260059290921b8401810191818101908684111561484f57600080fd5b8286015b84811015613c655780518352918301918301614853565b8051613a7281613bdf565b8051613a7281613d4b565b600080600080600080600060e0888a03121561489b57600080fd5b875167ffffffffffffffff808211156148b357600080fd5b6148bf8b838c016146e6565b985060208a0151975060408a01519150808211156148dc57600080fd5b6148e88b838c0161478f565b965060608a01519150808211156148fe57600080fd5b61490a8b838c0161480f565b955061491860808b0161486a565b945061492660a08b01614875565b935060c08a015191508082111561493c57600080fd5b506149498a828b0161474a565b91505092959891949750929550565b60e08152600061496b60e083018a614515565b88602084015287604084015282810360608401526149898188614566565b9050828103608084015261499d818761413e565b94151560a0840152505073ffffffffffffffffffffffffffffffffffffffff9190911660c09091015295945050505050565b600082516149e18184602087016140d0565b9190910192915050565b80516020808301519190811015614a2a577fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8160200360031b1b821691505b50919050565b73ffffffffffffffffffffffffffffffffffffffff84168152826020820152606060408201526000614a6560608301846140f4565b95945050505050565b600060208284031215614a8057600080fd5b8151611c4181613d4b565b7f416363657373436f6e74726f6c3a206163636f756e7420000000000000000000815260008351614ac38160178501602088016140d0565b7f206973206d697373696e6720726f6c65200000000000000000000000000000006017918401918201528351614b008160288401602088016140d0565b01602801949350505050565b602081526000611c4160208301846140f4565b8082028115828204841417610691576106916142f0565b80820180821115610691576106916142f0565b600081614b5857614b586142f0565b507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0190565b600060208284031215614b9057600080fd5b8151611c4181613bdf565b80516dffffffffffffffffffffffffffff81168114613a7257600080fd5b600080600060608486031215614bce57600080fd5b614bd784614b9b565b9250614be560208501614b9b565b9150604084015163ffffffff81168114614bfe57600080fd5b809150509250925092565b600082614c3f577f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b500490565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052605160045260246000fdfe416464726573733a206c6f772d6c6576656c2064656c65676174652063616c6c206661696c6564a264697066735822122007c1e9de304dd51a15120baadaa17d494f8e5a23f1c132c08125823791857f3d64736f6c63430008140033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000c21e4ebd1d92036cb467b53fe3258f219d909eb9000000000000000000000000eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2000000000000000000000000b1b64005b11350a94c4d069eff4215592d98f2e2
-----Decoded View---------------
Arg [0] : _assetForwarderAddress (address): 0xC21e4ebD1d92036Cb467b53fE3258F219d909Eb9
Arg [1] : _native (address): 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE
Arg [2] : _wrappedNative (address): 0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2
Arg [3] : _univ2SkimAddress (address): 0xB1b64005B11350a94c4D069eff4215592d98F2E2
-----Encoded View---------------
4 Constructor Arguments found :
Arg [0] : 000000000000000000000000c21e4ebd1d92036cb467b53fe3258f219d909eb9
Arg [1] : 000000000000000000000000eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee
Arg [2] : 000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc2
Arg [3] : 000000000000000000000000b1b64005b11350a94c4d069eff4215592d98f2e2
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ 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.