Feature Tip: Add private address tag to any address under My Name Tag !
More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 99 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Add Collateral | 21925878 | 36 hrs ago | IN | 0.0164 ETH | 0.00011004 | ||||
Wrap | 21911162 | 3 days ago | IN | 0.002 ETH | 0.0005426 | ||||
Un Wrap | 21876091 | 8 days ago | IN | 0 ETH | 0.00025051 | ||||
Add Collateral | 21846681 | 12 days ago | IN | 0.0004 ETH | 0.0002828 | ||||
Wrap | 21816941 | 16 days ago | IN | 0 ETH | 0.00093381 | ||||
Wrap | 21584008 | 49 days ago | IN | 0 ETH | 0.00126373 | ||||
Wrap | 21583999 | 49 days ago | IN | 0 ETH | 0.00123821 | ||||
Wrap | 21583994 | 49 days ago | IN | 0 ETH | 0.00120255 | ||||
Wrap | 21583990 | 49 days ago | IN | 0 ETH | 0.00122387 | ||||
Wrap | 21583980 | 49 days ago | IN | 0 ETH | 0.00167034 | ||||
Wrap | 21583974 | 49 days ago | IN | 0 ETH | 0.00182779 | ||||
Wrap | 21583961 | 49 days ago | IN | 0 ETH | 0.00186327 | ||||
Wrap | 21583956 | 49 days ago | IN | 0 ETH | 0.0015953 | ||||
Wrap | 21583949 | 49 days ago | IN | 0 ETH | 0.00187321 | ||||
Wrap | 21583941 | 49 days ago | IN | 0 ETH | 0.00194742 | ||||
Wrap | 21583936 | 49 days ago | IN | 0 ETH | 0.00194192 | ||||
Wrap | 21583928 | 49 days ago | IN | 0 ETH | 0.00220071 | ||||
Wrap | 21583918 | 49 days ago | IN | 0 ETH | 0.00200818 | ||||
Wrap | 21583902 | 49 days ago | IN | 0 ETH | 0.00199637 | ||||
Wrap | 21535603 | 56 days ago | IN | 0 ETH | 0.00527279 | ||||
Wrap | 21526256 | 57 days ago | IN | 0 ETH | 0.00087696 | ||||
Wrap | 21523252 | 57 days ago | IN | 0 ETH | 0.00490756 | ||||
Wrap | 21313906 | 87 days ago | IN | 0 ETH | 0.00464249 | ||||
Wrap | 21285844 | 90 days ago | IN | 0 ETH | 0.00260242 | ||||
Transfer Ownersh... | 21244039 | 96 days ago | IN | 0 ETH | 0.00051852 |
Loading...
Loading
Contract Source Code Verified (Exact Match)
Contract Name:
WrapperBaseV1
Compiler Version
v0.8.13+commit.abaa5c0e
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT // ENVELOP(NIFTSY) protocol V1 for NFT. Wrapper - main protocol contract pragma solidity 0.8.13; import "Ownable.sol"; import "ERC721Holder.sol"; import "ERC1155Holder.sol"; import "ReentrancyGuard.sol"; import "IFeeRoyaltyModel.sol"; import "IWrapper.sol"; import "IAdvancedWhiteList.sol"; import "TokenService.sol"; // #### Envelop ProtocolV1 Rules // 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0 <= Bit number(dec) // ------------------------------------------------------------------------------------ // 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 // | | | | | | | | | | | | | | | | // | | | | | | | | | | | | | | | +-No_Unwrap // | | | | | | | | | | | | | | +-No_Wrap // | | | | | | | | | | | | | +-No_Transfer // | | | | | | | | | | | | +-No_Collateral // | | | | | | | | | | | +-reserved_core // | | | | | | | | | | +-reserved_core // | | | | | | | | | +-reserved_core // | | | | | | | | +-reserved_core // | | | | | | | | // | | | | | | | | // +----+----+----+----+----+---+---+ // for use in extendings /** * @title Non-Fungible Token Wrapper * @dev Make wraping for existing ERC721 & ERC1155 and empty */ contract WrapperBaseV1 is ReentrancyGuard, ERC721Holder, ERC1155Holder, IWrapper, TokenService, Ownable { uint256 public MAX_COLLATERAL_SLOTS = 25; address public protocolTechToken; address public protocolWhiteList; // Map from wrapping asset type to wnft contract address and last minted id mapping(ETypes.AssetType => ETypes.NFTItem) public lastWNFTId; // Map from wNFT address to it's type (721, 1155) mapping(address => ETypes.AssetType) public wnftTypes; // Map from wrapped token address and id => wNFT record mapping(address => mapping(uint256 => ETypes.WNFT)) internal wrappedTokens; constructor(address _erc20) { require(_erc20 != address(0), "ProtocolTechToken cant be zero value"); protocolTechToken = _erc20; // This because default trnaferFe moddel included in techToken code IFeeRoyaltyModel(protocolTechToken).registerModel(); } function wrap( ETypes.INData calldata _inData, ETypes.AssetItem[] calldata _collateral, address _wrappFor ) public virtual payable nonReentrant returns (ETypes.AssetItem memory) { // 0. Check assetIn asset require(_checkWrap(_inData,_wrappFor), "Wrap check fail" ); // 1. Take users inAsset if ( _inData.inAsset.asset.assetType != ETypes.AssetType.NATIVE && _inData.inAsset.asset.assetType != ETypes.AssetType.EMPTY ) { require( _mustTransfered(_inData.inAsset) == _transferSafe( _inData.inAsset, msg.sender, address(this) ), "Suspicious asset for wrap" ); } // 2. Mint wNFT lastWNFTId[_inData.outType].tokenId += 1; //Save just will minted id _mintNFT( _inData.outType, // what will be minted instead of wrapping asset lastWNFTId[_inData.outType].contractAddress, // wNFT contract address _wrappFor, // wNFT receiver (1st owner) lastWNFTId[_inData.outType].tokenId, _inData.outBalance // wNFT tokenId ); // 3. Safe wNFT info _saveWNFTinfo( lastWNFTId[_inData.outType].contractAddress, lastWNFTId[_inData.outType].tokenId, _inData ); addCollateral( lastWNFTId[_inData.outType].contractAddress, lastWNFTId[_inData.outType].tokenId, _collateral ); // Charge Fee Hook // There is No Any Fees in Protocol // So this hook can be used in b2b extensions of Envelop Protocol // 0x02 - feeType for WrapFee _chargeFees( lastWNFTId[_inData.outType].contractAddress, lastWNFTId[_inData.outType].tokenId, msg.sender, address(this), 0x02 ); emit WrappedV1( _inData.inAsset.asset.contractAddress, // inAssetAddress lastWNFTId[_inData.outType].contractAddress, // outAssetAddress _inData.inAsset.tokenId, // inAssetTokenId lastWNFTId[_inData.outType].tokenId, // outTokenId _wrappFor, // wnftFirstOwner msg.value, // nativeCollateralAmount _inData.rules // rules ); return ETypes.AssetItem( ETypes.Asset(_inData.outType, lastWNFTId[_inData.outType].contractAddress), lastWNFTId[_inData.outType].tokenId, _inData.outBalance ); } function addCollateral( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetItem[] calldata _collateral ) public payable virtual { if (_collateral.length > 0 || msg.value > 0) { require( _checkAddCollateral( _wNFTAddress, _wNFTTokenId, _collateral ), "Forbidden add collateral" ); _addCollateral( _wNFTAddress, _wNFTTokenId, _collateral ); } } function unWrap(address _wNFTAddress, uint256 _wNFTTokenId) external virtual { unWrap(wnftTypes[_wNFTAddress], _wNFTAddress, _wNFTTokenId, false); } function unWrap( ETypes.AssetType _wNFTType, address _wNFTAddress, uint256 _wNFTTokenId ) external virtual { unWrap(_wNFTType, _wNFTAddress, _wNFTTokenId, false); } function unWrap( ETypes.AssetType _wNFTType, address _wNFTAddress, uint256 _wNFTTokenId, bool _isEmergency ) public virtual { // 1. Check core protocol logic: // - who and what possible to unwrap (address burnFor, uint256 burnBalance) = _checkCoreUnwrap(_wNFTType, _wNFTAddress, _wNFTTokenId); // 2. Check locks = move to _checkUnwrap require( _checkLocks(_wNFTAddress, _wNFTTokenId) ); // 3. Charge Fee Hook // There is No Any Fees in Protocol // So this hook can be used in b2b extensions of Envelop Protocol // 0x03 - feeType for UnWrapFee // _chargeFees(_wNFTAddress, _wNFTTokenId, msg.sender, address(this), 0x03); (uint256 nativeCollateralAmount, ) = getCollateralBalanceAndIndex( _wNFTAddress, _wNFTTokenId, ETypes.AssetType.NATIVE, address(0), 0 ); /////////////////////////////////////////////// /// Place for hook //// /////////////////////////////////////////////// // 4. Safe return collateral to appropriate benificiary if (!_beforeUnWrapHook(_wNFTAddress, _wNFTTokenId, _isEmergency)) { return; } // 5. BurnWNFT _burnNFT( _wNFTType, _wNFTAddress, burnFor, // msg.sender, _wNFTTokenId, burnBalance ); emit UnWrappedV1( _wNFTAddress, wrappedTokens[_wNFTAddress][_wNFTTokenId].inAsset.asset.contractAddress, _wNFTTokenId, wrappedTokens[_wNFTAddress][_wNFTTokenId].inAsset.tokenId, wrappedTokens[_wNFTAddress][_wNFTTokenId].unWrapDestination, nativeCollateralAmount, // TODO Check GAS wrappedTokens[_wNFTAddress][_wNFTTokenId].rules ); } function chargeFees( address _wNFTAddress, uint256 _wNFTTokenId, address _from, address _to, bytes1 _feeType ) public virtual returns (bool) { //TODO only wNFT contract can execute this(=charge fee) require(msg.sender == _wNFTAddress || msg.sender == address(this), "Only for wNFT or wrapper" ); require(_chargeFees(_wNFTAddress, _wNFTTokenId, _from, _to, _feeType), "Fee charge fail" ); } ///////////////////////////////////////////////////////////////////// // Admin functions // ///////////////////////////////////////////////////////////////////// function setWNFTId( ETypes.AssetType _assetOutType, address _wnftContract, uint256 _tokenId ) external onlyOwner { require(_wnftContract != address(0), "No zero address"); lastWNFTId[_assetOutType] = ETypes.NFTItem(_wnftContract, _tokenId); wnftTypes[_wnftContract] = _assetOutType; } function setWhiteList(address _wlAddress) external onlyOwner { protocolWhiteList = _wlAddress; } ///////////////////////////////////////////////////////////////////// function getWrappedToken(address _wNFTAddress, uint256 _wNFTTokenId) public view returns (ETypes.WNFT memory) { return wrappedTokens[_wNFTAddress][_wNFTTokenId]; } function getOriginalURI(address _wNFTAddress, uint256 _wNFTTokenId) public view returns(string memory uri_) { ETypes.AssetItem memory _wnftInAsset = getWrappedToken( _wNFTAddress, _wNFTTokenId ).inAsset; if (_wnftInAsset.asset.assetType == ETypes.AssetType.ERC721) { uri_ = IERC721Metadata(_wnftInAsset.asset.contractAddress).tokenURI(_wnftInAsset.tokenId); } else if (_wnftInAsset.asset.assetType == ETypes.AssetType.ERC1155) { uri_ = IERC1155MetadataURI(_wnftInAsset.asset.contractAddress).uri(_wnftInAsset.tokenId); } else { uri_ = ''; } } function getCollateralBalanceAndIndex( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetType _collateralType, address _erc, uint256 _tokenId ) public view returns (uint256, uint256) { for (uint256 i = 0; i < wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral.length; i ++) { if (wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].asset.contractAddress == _erc && wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].tokenId == _tokenId && wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].asset.assetType == _collateralType ) { return (wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].amount, i); } } } ///////////////////////////////////////////////////////////////////// // Internals // ///////////////////////////////////////////////////////////////////// function _saveWNFTinfo( address wNFTAddress, uint256 tokenId, ETypes.INData calldata _inData ) internal virtual { wrappedTokens[wNFTAddress][tokenId].inAsset = _inData.inAsset; // We will use _inData.unWrapDestination ONLY for RENT implementation // wrappedTokens[wNFTAddress][tokenId].unWrapDestination = _inData.unWrapDestination; wrappedTokens[wNFTAddress][tokenId].unWrapDestination = address(0); wrappedTokens[wNFTAddress][tokenId].rules = _inData.rules; // Copying of type struct ETypes.Fee memory[] // memory to storage not yet supported. for (uint256 i = 0; i < _inData.fees.length; i ++) { wrappedTokens[wNFTAddress][tokenId].fees.push(_inData.fees[i]); } for (uint256 i = 0; i < _inData.locks.length; i ++) { wrappedTokens[wNFTAddress][tokenId].locks.push(_inData.locks[i]); } for (uint256 i = 0; i < _inData.royalties.length; i ++) { wrappedTokens[wNFTAddress][tokenId].royalties.push(_inData.royalties[i]); } } function _addCollateral( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetItem[] calldata _collateral ) internal virtual { // Process Native Colleteral if (msg.value > 0) { _updateCollateralInfo( _wNFTAddress, _wNFTTokenId, ETypes.AssetItem( ETypes.Asset(ETypes.AssetType.NATIVE, address(0)), 0, msg.value ) ); emit CollateralAdded( _wNFTAddress, _wNFTTokenId, uint8(ETypes.AssetType.NATIVE), address(0), 0, msg.value ); } // Process Token Colleteral for (uint256 i = 0; i <_collateral.length; i ++) { if (_collateral[i].asset.assetType != ETypes.AssetType.NATIVE) { // Check WhiteList Logic if (protocolWhiteList != address(0)) { require( IAdvancedWhiteList(protocolWhiteList).enabledForCollateral( _collateral[i].asset.contractAddress), "WL:Some assets are not enabled for collateral" ); } require( _mustTransfered(_collateral[i]) == _transferSafe( _collateral[i], msg.sender, address(this) ), "Suspicious asset for wrap" ); _updateCollateralInfo( _wNFTAddress, _wNFTTokenId, _collateral[i] ); emit CollateralAdded( _wNFTAddress, _wNFTTokenId, uint8(_collateral[i].asset.assetType), _collateral[i].asset.contractAddress, _collateral[i].tokenId, _collateral[i].amount ); } } } function _updateCollateralInfo( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetItem memory collateralItem ) internal virtual { ///////////////////////////////////////// // ERC20 & NATIVE Collateral /// ///////////////////////////////////////// if (collateralItem.asset.assetType == ETypes.AssetType.ERC20 || collateralItem.asset.assetType == ETypes.AssetType.NATIVE) { require(collateralItem.tokenId == 0, "TokenId must be zero"); } ///////////////////////////////////////// // ERC1155 Collateral /// // ///////////////////////////////////////// // if (collateralItem.asset.assetType == ETypes.AssetType.ERC1155) { // No need special checks // } ///////////////////////////////////////// // ERC721 Collateral /// ///////////////////////////////////////// if (collateralItem.asset.assetType == ETypes.AssetType.ERC721 ) { require(collateralItem.amount == 0, "Amount must be zero"); } ///////////////////////////////////////// if (wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral.length == 0 || collateralItem.asset.assetType == ETypes.AssetType.ERC721 ) { // First record in collateral or 721 _newCollateralItem(_wNFTAddress,_wNFTTokenId,collateralItem); } else { // length > 0 (uint256 _amnt, uint256 _index) = getCollateralBalanceAndIndex( _wNFTAddress, _wNFTTokenId, collateralItem.asset.assetType, collateralItem.asset.contractAddress, collateralItem.tokenId ); if (_index > 0 || (_index == 0 && wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[0].asset.contractAddress == collateralItem.asset.contractAddress ) ) { // We dont need addition if for erc721 because for erc721 _amnt always be zero wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[_index].amount += collateralItem.amount; } else { // _index == 0 && and no this token record yet _newCollateralItem(_wNFTAddress,_wNFTTokenId,collateralItem); } } } function _newCollateralItem( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetItem memory collateralItem ) internal virtual { require( wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral.length < MAX_COLLATERAL_SLOTS, "Too much tokens in collateral" ); for (uint256 i = 0; i < wrappedTokens[_wNFTAddress][_wNFTTokenId].locks.length; i ++) { // Personal Collateral count Lock check if (wrappedTokens[_wNFTAddress][_wNFTTokenId].locks[i].lockType == 0x02) { require( wrappedTokens[_wNFTAddress][_wNFTTokenId].locks[i].param >= (wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral.length + 1), "Too much collateral slots for this wNFT" ); } } wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral.push(collateralItem); } function _chargeFees( address _wNFTAddress, uint256 _wNFTTokenId, address _from, address _to, bytes1 _feeType ) internal virtual returns (bool) { if (_feeType == 0x00) {// Transfer fee for (uint256 i = 0; i < wrappedTokens[_wNFTAddress][_wNFTTokenId].fees.length; i ++){ ///////////////////////////////////////// // For Transfer Fee -0x00 /// ///////////////////////////////////////// if (wrappedTokens[_wNFTAddress][_wNFTTokenId].fees[i].feeType == 0x00){ // - get modelAddress. Default feeModel adddress always live in // protocolTechToken. When white list used it is possible override that model. // default model always must be set as protocolTechToken address feeModel = protocolTechToken; if (protocolWhiteList != address(0)) { feeModel = IAdvancedWhiteList(protocolWhiteList).getWLItem( wrappedTokens[_wNFTAddress][_wNFTTokenId].fees[i].token).transferFeeModel; } // - get transfer list from external model by feetype(with royalties) (ETypes.AssetItem[] memory assetItems, address[] memory from, address[] memory to ) = IFeeRoyaltyModel(feeModel).getTransfersList( wrappedTokens[_wNFTAddress][_wNFTTokenId].fees[i], wrappedTokens[_wNFTAddress][_wNFTTokenId].royalties, _from, _to ); // - execute transfers uint256 actualTransfered; for (uint256 j = 0; j < to.length; j ++){ // if transfer receiver(to) = address(this) lets consider // wNFT as receiver. in this case received amount // will be added to collateral if (to[j]== address(this)){ _updateCollateralInfo( _wNFTAddress, _wNFTTokenId, assetItems[j] ); } actualTransfered = _transferSafe(assetItems[j], from[j], to[j]); emit EnvelopFee(to[j], _wNFTAddress, _wNFTTokenId, actualTransfered); } } ////////////////////////////////////////// } return true; } } /** * @dev This hook may be overriden in inheritor contracts for extend * base functionality. * * @param _wNFTAddress -wrapped token address * @param _wNFTTokenId -wrapped token id * * must returns true for success unwrapping enable */ function _beforeUnWrapHook( address _wNFTAddress, uint256 _wNFTTokenId, bool _emergency ) internal virtual returns (bool) { uint256 transfered; address receiver = msg.sender; if (wrappedTokens[_wNFTAddress][_wNFTTokenId].unWrapDestination != address(0)) { receiver = wrappedTokens[_wNFTAddress][_wNFTTokenId].unWrapDestination; } for (uint256 i = 0; i < wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral.length; i ++) { if (wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].asset.assetType != ETypes.AssetType.EMPTY ) { if (_emergency) { // In case of something is wrong with any collateral (attack) // user can use this mode for skip malicious asset transfered = _transferEmergency( wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i], address(this), receiver ); } else { transfered = _transferSafe( wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i], address(this), receiver ); } // we collect info about contracts with not standard behavior if (transfered != wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].amount ) { emit SuspiciousFail( _wNFTAddress, _wNFTTokenId, wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].asset.contractAddress ); } // mark collateral record as returned wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral[i].asset.assetType = ETypes.AssetType.EMPTY; } // dont pop due in some case it c can be very costly // https://docs.soliditylang.org/en/v0.8.9/types.html#array-members // For safe exit in case of low gaslimit // this strange part of code can prevent only case // when when some collateral tokens spent unexpected gas limit if ( gasleft() <= 1_000 && i < wrappedTokens[_wNFTAddress][_wNFTTokenId].collateral.length - 1 ) { emit PartialUnWrapp(_wNFTAddress, _wNFTTokenId, i); //allReturned = false; return false; } } // 5. Return Original if (wrappedTokens[_wNFTAddress][_wNFTTokenId].inAsset.asset.assetType != ETypes.AssetType.NATIVE && wrappedTokens[_wNFTAddress][_wNFTTokenId].inAsset.asset.assetType != ETypes.AssetType.EMPTY ) { if (!_emergency){ _transferSafe( wrappedTokens[_wNFTAddress][_wNFTTokenId].inAsset, address(this), receiver ); } else { _transferEmergency ( wrappedTokens[_wNFTAddress][_wNFTTokenId].inAsset, address(this), receiver ); } } return true; } //////////////////////////////////////////////////////////////////////////////////////////// function _mustTransfered(ETypes.AssetItem calldata _assetForTransfer) internal pure returns (uint256 mustTransfered) { // Available for wrap assets must be good transferable (stakable). // So for erc721 mustTransfered always be 1 if (_assetForTransfer.asset.assetType == ETypes.AssetType.ERC721) { mustTransfered = 1; } else { mustTransfered = _assetForTransfer.amount; } } function _checkRule(bytes2 _rule, bytes2 _wNFTrules) internal view returns (bool) { return _rule == (_rule & _wNFTrules); } // 0x00 - TimeLock // 0x01 - TransferFeeLock // 0x02 - Personal Collateral count Lock check function _checkLocks(address _wNFTAddress, uint256 _wNFTTokenId) internal view returns (bool) { // Lets check that inAsset for (uint256 i = 0; i < wrappedTokens[_wNFTAddress][_wNFTTokenId].locks.length; i ++) { // Time Lock check if (wrappedTokens[_wNFTAddress][_wNFTTokenId].locks[i].lockType == 0x00) { require( wrappedTokens[_wNFTAddress][_wNFTTokenId].locks[i].param <= block.timestamp, "TimeLock error" ); } // Fee Lock check if (wrappedTokens[_wNFTAddress][_wNFTTokenId].locks[i].lockType == 0x01) { // Lets check this lock rule against each fee record for (uint256 j = 0; j < wrappedTokens[_wNFTAddress][_wNFTTokenId].fees.length; j ++){ // Fee Lock depend only from Transfer Fee - 0x00 if ( wrappedTokens[_wNFTAddress][_wNFTTokenId].fees[j].feeType == 0x00) { (uint256 _bal,) = getCollateralBalanceAndIndex( _wNFTAddress, _wNFTTokenId, ETypes.AssetType.ERC20, wrappedTokens[_wNFTAddress][_wNFTTokenId].fees[j].token, 0 ); require( wrappedTokens[_wNFTAddress][_wNFTTokenId].locks[i].param <= _bal, "TransferFeeLock error" ); } } } } return true; } function _checkWrap(ETypes.INData calldata _inData, address _wrappFor) internal view returns (bool enabled) { // Lets check that inAsset // 0x0002 - this rule disable wrap already wrappednFT (NO matryoshka) enabled = !_checkRule(0x0002, getWrappedToken( _inData.inAsset.asset.contractAddress, _inData.inAsset.tokenId).rules ) && _wrappFor != address(this); // Check WhiteList Logic if (protocolWhiteList != address(0)) { require( !IAdvancedWhiteList(protocolWhiteList).getBLItem(_inData.inAsset.asset.contractAddress), "WL:Asset disabled for wrap" ); require( IAdvancedWhiteList(protocolWhiteList).rulesEnabled(_inData.inAsset.asset.contractAddress, _inData.rules), "WL:Some rules are disabled for this asset" ); for (uint256 i = 0; i < _inData.fees.length; i ++){ require( IAdvancedWhiteList(protocolWhiteList).enabledForFee( _inData.fees[i].token), "WL:Some assets are not enabled for fee" ); } } } function _checkAddCollateral( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetItem[] calldata _collateral ) internal view returns (bool enabled) { // Check that wNFT exist if (wnftTypes[_wNFTAddress] == ETypes.AssetType.ERC721) { require(IERC721Mintable(_wNFTAddress).exists(_wNFTTokenId), "wNFT not exists"); } else if(wnftTypes[_wNFTAddress] == ETypes.AssetType.ERC1155) { require(IERC1155Mintable(_wNFTAddress).exists(_wNFTTokenId), "wNFT not exists"); } else { revert UnSupportedAsset( ETypes.AssetItem(ETypes.Asset(wnftTypes[_wNFTAddress],_wNFTAddress),_wNFTTokenId, 0) ); } // Lets check wNFT rules // 0x0008 - this rule disable add collateral enabled = !_checkRule(0x0008, getWrappedToken(_wNFTAddress, _wNFTTokenId).rules); } function _checkCoreUnwrap( ETypes.AssetType _wNFTType, address _wNFTAddress, uint256 _wNFTTokenId ) internal view virtual returns (address burnFor, uint256 burnBalance) { // Lets wNFT rules // 0x0001 - this rule disable unwrap wrappednFT require(!_checkRule(0x0001, getWrappedToken(_wNFTAddress, _wNFTTokenId).rules), "UnWrapp forbidden by author" ); if (_wNFTType == ETypes.AssetType.ERC721) { // Only token owner can UnWrap burnFor = IERC721Mintable(_wNFTAddress).ownerOf(_wNFTTokenId); require(burnFor == msg.sender, 'Only owner can unwrap it' ); } else if (_wNFTType == ETypes.AssetType.ERC1155) { burnBalance = IERC1155Mintable(_wNFTAddress).totalSupply(_wNFTTokenId); burnFor = msg.sender; require( burnBalance == IERC1155Mintable(_wNFTAddress).balanceOf(burnFor, _wNFTTokenId) ,'ERC115 unwrap available only for all totalSupply' ); } else { revert UnSupportedAsset(ETypes.AssetItem(ETypes.Asset(_wNFTType,_wNFTAddress),_wNFTTokenId, 0)); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (access/Ownable.sol) pragma solidity ^0.8.0; import "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 Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(owner() == _msgSender(), "Ownable: caller is not the owner"); _; } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing 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 v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/utils/ERC721Holder.sol) pragma solidity ^0.8.0; import "IERC721Receiver.sol"; /** * @dev Implementation of the {IERC721Receiver} interface. * * Accepts all token transfers. * Make sure the contract is able to use its token with {IERC721-safeTransferFrom}, {IERC721-approve} or {IERC721-setApprovalForAll}. */ contract ERC721Holder is IERC721Receiver { /** * @dev See {IERC721Receiver-onERC721Received}. * * Always returns `IERC721Receiver.onERC721Received.selector`. */ function onERC721Received( address, address, uint256, bytes memory ) public virtual override returns (bytes4) { return this.onERC721Received.selector; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721Receiver.sol) pragma solidity ^0.8.0; /** * @title ERC721 token receiver interface * @dev Interface for any contract that wants to support safeTransfers * from ERC721 asset contracts. */ interface IERC721Receiver { /** * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom} * by `operator` from `from`, this function is called. * * It must return its Solidity selector to confirm the token transfer. * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted. * * The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`. */ function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC1155/utils/ERC1155Holder.sol) pragma solidity ^0.8.0; import "ERC1155Receiver.sol"; /** * @dev _Available since v3.1._ */ contract ERC1155Holder is ERC1155Receiver { function onERC1155Received( address, address, uint256, uint256, bytes memory ) public virtual override returns (bytes4) { return this.onERC1155Received.selector; } function onERC1155BatchReceived( address, address, uint256[] memory, uint256[] memory, bytes memory ) public virtual override returns (bytes4) { return this.onERC1155BatchReceived.selector; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC1155/utils/ERC1155Receiver.sol) pragma solidity ^0.8.0; import "IERC1155Receiver.sol"; import "ERC165.sol"; /** * @dev _Available since v3.1._ */ abstract contract ERC1155Receiver is ERC165, IERC1155Receiver { /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) { return interfaceId == type(IERC1155Receiver).interfaceId || super.supportsInterface(interfaceId); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC1155/IERC1155Receiver.sol) pragma solidity ^0.8.0; import "IERC165.sol"; /** * @dev _Available since v3.1._ */ interface IERC1155Receiver is IERC165 { /** @dev Handles the receipt of a single ERC1155 token type. This function is called at the end of a `safeTransferFrom` after the balance has been updated. To accept the transfer, this must return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))` (i.e. 0xf23a6e61, or its own function selector). @param operator The address which initiated the transfer (i.e. msg.sender) @param from The address which previously owned the token @param id The ID of the token being transferred @param value The amount of tokens being transferred @param data Additional data with no specified format @return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))` if transfer is allowed */ function onERC1155Received( address operator, address from, uint256 id, uint256 value, bytes calldata data ) external returns (bytes4); /** @dev Handles the receipt of a multiple ERC1155 token types. This function is called at the end of a `safeBatchTransferFrom` after the balances have been updated. To accept the transfer(s), this must return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))` (i.e. 0xbc197c81, or its own function selector). @param operator The address which initiated the batch transfer (i.e. msg.sender) @param from The address which previously owned the token @param ids An array containing ids of each token being transferred (order and length must match values array) @param values An array containing amounts of each token being transferred (order and length must match ids array) @param data Additional data with no specified format @return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))` if transfer is allowed */ function onERC1155BatchReceived( address operator, address from, uint256[] calldata ids, uint256[] calldata values, bytes calldata data ) external returns (bytes4); }
// 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 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 // OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol) pragma solidity ^0.8.0; /** * @dev Contract module that helps prevent reentrant calls to a function. * * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier * available, which can be applied to functions to make sure there are no nested * (reentrant) calls to them. * * Note that because there is a single `nonReentrant` guard, functions marked as * `nonReentrant` may not call one another. This can be worked around by making * those functions `private`, and then adding `external` `nonReentrant` entry * points to them. * * TIP: If you would like to learn more about reentrancy and alternative ways * to protect against it, check out our blog post * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul]. */ abstract contract ReentrancyGuard { // Booleans are more expensive than uint256 or any type that takes up a full // word because each write operation emits an extra SLOAD to first read the // slot's contents, replace the bits taken up by the boolean, and then write // back. This is the compiler's defense against contract upgrades and // pointer aliasing, and it cannot be disabled. // The values being non-zero value makes deployment a bit more expensive, // but in exchange the refund on every call to nonReentrant will be lower in // amount. Since refunds are capped to a percentage of the total // transaction's gas, it is best to keep them low in cases like this one, to // increase the likelihood of the full refund coming into effect. uint256 private constant _NOT_ENTERED = 1; uint256 private constant _ENTERED = 2; uint256 private _status; constructor() { _status = _NOT_ENTERED; } /** * @dev Prevents a contract from calling itself, directly or indirectly. * Calling a `nonReentrant` function from another `nonReentrant` * function is not supported. It is possible to prevent this from happening * by making the `nonReentrant` function external, and making it call a * `private` function that does the actual work. */ modifier nonReentrant() { // On the first call to nonReentrant, _notEntered will be true require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; _; // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) _status = _NOT_ENTERED; } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; import "LibEnvelopTypes.sol"; interface IFeeRoyaltyModel { function registerModel() external; function getTransfersList( ETypes.Fee calldata _fee, ETypes.Royalty[] calldata _royalties, address _from, address _to ) external view returns ( ETypes.AssetItem[] memory, address[] memory, address[] memory ); function wrapper() external returns (address); }
// SPDX-License-Identifier: MIT // ENVELOP(NIFTSY) protocol V1 for NFT. pragma solidity 0.8.13; library ETypes { enum AssetType {EMPTY, NATIVE, ERC20, ERC721, ERC1155, FUTURE1, FUTURE2, FUTURE3} struct Asset { AssetType assetType; address contractAddress; } struct AssetItem { Asset asset; uint256 tokenId; uint256 amount; } struct NFTItem { address contractAddress; uint256 tokenId; } struct Fee { bytes1 feeType; uint256 param; address token; } struct Lock { bytes1 lockType; uint256 param; } struct Royalty { address beneficiary; uint16 percent; } struct WNFT { AssetItem inAsset; AssetItem[] collateral; address unWrapDestination; Fee[] fees; Lock[] locks; Royalty[] royalties; bytes2 rules; } struct INData { AssetItem inAsset; address unWrapDestination; Fee[] fees; Lock[] locks; Royalty[] royalties; AssetType outType; uint256 outBalance; //0- for 721 and any amount for 1155 bytes2 rules; } struct WhiteListItem { bool enabledForFee; bool enabledForCollateral; bool enabledRemoveFromCollateral; address transferFeeModel; } struct Rules { bytes2 onlythis; bytes2 disabled; } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; //import "IERC721Enumerable.sol"; import "LibEnvelopTypes.sol"; interface IWrapper { event WrappedV1( address indexed inAssetAddress, address indexed outAssetAddress, uint256 indexed inAssetTokenId, uint256 outTokenId, address wnftFirstOwner, uint256 nativeCollateralAmount, bytes2 rules ); event UnWrappedV1( address indexed wrappedAddress, address indexed originalAddress, uint256 indexed wrappedId, uint256 originalTokenId, address beneficiary, uint256 nativeCollateralAmount, bytes2 rules ); event CollateralAdded( address indexed wrappedAddress, uint256 indexed wrappedId, uint8 assetType, address collateralAddress, uint256 collateralTokenId, uint256 collateralBalance ); event PartialUnWrapp( address indexed wrappedAddress, uint256 indexed wrappedId, uint256 lastCollateralIndex ); event SuspiciousFail( address indexed wrappedAddress, uint256 indexed wrappedId, address indexed failedContractAddress ); event EnvelopFee( address indexed receiver, address indexed wNFTConatract, uint256 indexed wNFTTokenId, uint256 amount ); function wrap( ETypes.INData calldata _inData, ETypes.AssetItem[] calldata _collateral, address _wrappFor ) external payable returns (ETypes.AssetItem memory); // function wrapUnsafe( // ETypes.INData calldata _inData, // ETypes.AssetItem[] calldata _collateral, // address _wrappFor // ) // external // payable // returns (ETypes.AssetItem memory); function addCollateral( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetItem[] calldata _collateral ) external payable; // function addCollateralUnsafe( // address _wNFTAddress, // uint256 _wNFTTokenId, // ETypes.AssetItem[] calldata _collateral // ) // external // payable; function unWrap( address _wNFTAddress, uint256 _wNFTTokenId ) external; function unWrap( ETypes.AssetType _wNFTType, address _wNFTAddress, uint256 _wNFTTokenId ) external; function unWrap( ETypes.AssetType _wNFTType, address _wNFTAddress, uint256 _wNFTTokenId, bool _isEmergency ) external; function chargeFees( address _wNFTAddress, uint256 _wNFTTokenId, address _from, address _to, bytes1 _feeType ) external returns (bool); ////////////////////////////////////////////////////////////////////// function MAX_COLLATERAL_SLOTS() external view returns (uint256); function protocolTechToken() external view returns (address); function protocolWhiteList() external view returns (address); //function trustedOperators(address _operator) external view returns (bool); //function lastWNFTId(ETypes.AssetType _assetType) external view returns (ETypes.NFTItem); function getWrappedToken(address _wNFTAddress, uint256 _wNFTTokenId) external view returns (ETypes.WNFT memory); function getOriginalURI(address _wNFTAddress, uint256 _wNFTTokenId) external view returns(string memory); function getCollateralBalanceAndIndex( address _wNFTAddress, uint256 _wNFTTokenId, ETypes.AssetType _collateralType, address _erc, uint256 _tokenId ) external view returns (uint256, uint256); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Enumerable.sol) pragma solidity ^0.8.0; import "IERC721.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Enumerable is IERC721 { /** * @dev Returns the total amount of tokens stored by the contract. */ function totalSupply() external view returns (uint256); /** * @dev Returns a token ID owned by `owner` at a given `index` of its token list. * Use along with {balanceOf} to enumerate all of ``owner``'s tokens. */ function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256 tokenId); /** * @dev Returns a token ID at a given `index` of all the tokens stored by the contract. * Use along with {totalSupply} to enumerate all tokens. */ function tokenByIndex(uint256 index) external view returns (uint256); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721.sol) pragma solidity ^0.8.0; import "IERC165.sol"; /** * @dev Required interface of an ERC721 compliant contract. */ interface IERC721 is IERC165 { /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in ``owner``'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be have been allowed to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Transfers `tokenId` token from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll} */ function isApprovedForAll(address owner, address operator) external view returns (bool); /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external; }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; //import "IERC721Enumerable.sol"; import "LibEnvelopTypes.sol"; interface IAdvancedWhiteList { event WhiteListItemChanged( address indexed asset, bool enabledForFee, bool enabledForCollateral, bool enabledRemoveFromCollateral, address transferFeeModel ); event BlackListItemChanged( address indexed asset, bool isBlackListed ); function getWLItem(address _asset) external view returns (ETypes.WhiteListItem memory); function getWLItemCount() external view returns (uint256); function getBLItem(address _asset) external view returns (bool); function getBLItemCount() external view returns (uint256); function enabledForCollateral(address _asset) external view returns (bool); function enabledForFee(address _asset) external view returns (bool); function enabledRemoveFromCollateral(address _asset) external view returns (bool); function rulesEnabled(address _asset, bytes2 _rules) external view returns (bool); function validateRules(address _asset, bytes2 _rules) external view returns (bytes2); }
// SPDX-License-Identifier: MIT // ENVELOP(NIFTSY) protocol V1 for NFT. Wrapper - main protocol contract pragma solidity 0.8.13; import "SafeERC20.sol"; import "IERC20Extended.sol"; import "LibEnvelopTypes.sol"; import "IERC721Mintable.sol"; import "IERC1155Mintable.sol"; //import "ITokenService.sol"; abstract contract TokenService { using SafeERC20 for IERC20Extended; error UnSupportedAsset(ETypes.AssetItem asset); function _mintNFT( ETypes.AssetType _mint_type, address _contract, address _mintFor, uint256 _tokenId, uint256 _outBalance ) internal virtual { if (_mint_type == ETypes.AssetType.ERC721) { IERC721Mintable(_contract).mint(_mintFor, _tokenId); } else if (_mint_type == ETypes.AssetType.ERC1155) { IERC1155Mintable(_contract).mint(_mintFor, _tokenId, _outBalance); } } function _burnNFT( ETypes.AssetType _burn_type, address _contract, address _burnFor, uint256 _tokenId, uint256 _balance ) internal virtual { if (_burn_type == ETypes.AssetType.ERC721) { IERC721Mintable(_contract).burn(_tokenId); } else if (_burn_type == ETypes.AssetType.ERC1155) { IERC1155Mintable(_contract).burn(_burnFor, _tokenId, _balance); } } function _transfer( ETypes.AssetItem memory _assetItem, address _from, address _to ) internal virtual returns (bool _transfered){ if (_assetItem.asset.assetType == ETypes.AssetType.NATIVE) { (bool success, ) = _to.call{ value: _assetItem.amount}(""); require(success, "transfer failed"); _transfered = true; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC20) { require(IERC20Extended(_assetItem.asset.contractAddress).balanceOf(_from) <= _assetItem.amount, "UPS!!!!"); IERC20Extended(_assetItem.asset.contractAddress).safeTransferFrom(_from, _to, _assetItem.amount); _transfered = true; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC721) { IERC721Mintable(_assetItem.asset.contractAddress).transferFrom(_from, _to, _assetItem.tokenId); _transfered = true; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC1155) { IERC1155Mintable(_assetItem.asset.contractAddress).safeTransferFrom(_from, _to, _assetItem.tokenId, _assetItem.amount, ""); _transfered = true; } else { revert UnSupportedAsset(_assetItem); } return _transfered; } function _transferSafe( ETypes.AssetItem memory _assetItem, address _from, address _to ) internal virtual returns (uint256 _transferedValue){ //TODO think about try catch in transfers uint256 balanceBefore; if (_assetItem.asset.assetType == ETypes.AssetType.NATIVE) { balanceBefore = _to.balance; (bool success, ) = _to.call{ value: _assetItem.amount}(""); require(success, "transfer failed"); _transferedValue = _to.balance - balanceBefore; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC20) { balanceBefore = IERC20Extended(_assetItem.asset.contractAddress).balanceOf(_to); if (_from == address(this)){ IERC20Extended(_assetItem.asset.contractAddress).safeTransfer(_to, _assetItem.amount); } else { IERC20Extended(_assetItem.asset.contractAddress).safeTransferFrom(_from, _to, _assetItem.amount); } _transferedValue = IERC20Extended(_assetItem.asset.contractAddress).balanceOf(_to) - balanceBefore; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC721 && IERC721Mintable(_assetItem.asset.contractAddress).ownerOf(_assetItem.tokenId) == _from) { balanceBefore = IERC721Mintable(_assetItem.asset.contractAddress).balanceOf(_to); IERC721Mintable(_assetItem.asset.contractAddress).transferFrom(_from, _to, _assetItem.tokenId); if (IERC721Mintable(_assetItem.asset.contractAddress).ownerOf(_assetItem.tokenId) == _to && IERC721Mintable(_assetItem.asset.contractAddress).balanceOf(_to) - balanceBefore == 1 ) { _transferedValue = 1; } } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC1155) { balanceBefore = IERC1155Mintable(_assetItem.asset.contractAddress).balanceOf(_to, _assetItem.tokenId); IERC1155Mintable(_assetItem.asset.contractAddress).safeTransferFrom(_from, _to, _assetItem.tokenId, _assetItem.amount, ""); _transferedValue = IERC1155Mintable(_assetItem.asset.contractAddress).balanceOf(_to, _assetItem.tokenId) - balanceBefore; } else { revert UnSupportedAsset(_assetItem); } return _transferedValue; } // This function must never revert. Use it for unwrap in case some // collateral transfers are revert function _transferEmergency( ETypes.AssetItem memory _assetItem, address _from, address _to ) internal virtual returns (uint256 _transferedValue){ //TODO think about try catch in transfers uint256 balanceBefore; if (_assetItem.asset.assetType == ETypes.AssetType.NATIVE) { balanceBefore = _to.balance; (bool success, ) = _to.call{ value: _assetItem.amount}(""); //require(success, "transfer failed"); _transferedValue = _to.balance - balanceBefore; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC20) { if (_from == address(this)){ (bool success, ) = _assetItem.asset.contractAddress.call( abi.encodeWithSignature("transfer(address,uint256)", _to, _assetItem.amount) ); } else { (bool success, ) = _assetItem.asset.contractAddress.call( abi.encodeWithSignature("transferFrom(address,address,uint256)", _from, _to, _assetItem.amount) ); } _transferedValue = _assetItem.amount; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC721) { (bool success, ) = _assetItem.asset.contractAddress.call( abi.encodeWithSignature("transferFrom(address,address,uint256)", _from, _to, _assetItem.tokenId) ); _transferedValue = 1; } else if (_assetItem.asset.assetType == ETypes.AssetType.ERC1155) { (bool success, ) = _assetItem.asset.contractAddress.call( abi.encodeWithSignature("safeTransferFrom(address,address,uint256,uint256,bytes)", _from, _to, _assetItem.tokenId, _assetItem.amount, "") ); _transferedValue = _assetItem.amount; } else { revert UnSupportedAsset(_assetItem); } return _transferedValue; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.0; import "IERC20.sol"; import "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; function safeTransfer( IERC20 token, address to, uint256 value ) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value)); } function safeTransferFrom( IERC20 token, address from, address to, uint256 value ) internal { _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value)); } /** * @dev Deprecated. This function has issues similar to the ones found in * {IERC20-approve}, and its usage is discouraged. * * Whenever possible, use {safeIncreaseAllowance} and * {safeDecreaseAllowance} instead. */ function safeApprove( IERC20 token, address spender, uint256 value ) internal { // safeApprove should only be called when setting an initial allowance, // or when resetting it to zero. To increase and decrease it, use // 'safeIncreaseAllowance' and 'safeDecreaseAllowance' require( (value == 0) || (token.allowance(address(this), spender) == 0), "SafeERC20: approve from non-zero to non-zero allowance" ); _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value)); } function safeIncreaseAllowance( IERC20 token, address spender, uint256 value ) internal { uint256 newAllowance = token.allowance(address(this), spender) + value; _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } 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"); uint256 newAllowance = oldAllowance - value; _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance)); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed"); if (returndata.length > 0) { // Return data is optional require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/IERC20.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `recipient`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address recipient, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `sender` to `recipient` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom( address sender, address recipient, uint256 amount ) external returns (bool); /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Address.sol) pragma solidity ^0.8.0; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize, which returns 0 for contracts in // construction, since the code is only stored at the end of the // constructor execution. uint256 size; assembly { size := extcodesize(account) } return size > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.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) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason 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 { // 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 assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; import "IERC20.sol"; interface IERC20Extended is IERC20 { function mint(address _to, uint256 _value) external; }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; import "IERC721Metadata.sol"; interface IERC721Mintable is IERC721Metadata { function mint(address _to, uint256 _tokenId) external; function burn(uint256 _tokenId) external; function exists(uint256 _tokenId) external view returns(bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol) pragma solidity ^0.8.0; import "IERC721.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional metadata extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Metadata is IERC721 { /** * @dev Returns the token collection name. */ function name() external view returns (string memory); /** * @dev Returns the token collection symbol. */ function symbol() external view returns (string memory); /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) external view returns (string memory); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.13; import "IERC1155MetadataURI.sol"; interface IERC1155Mintable is IERC1155MetadataURI { function mint(address _to, uint256 _tokenId, uint256 _amount) external; function burn(address _to, uint256 _tokenId, uint256 _amount) external; function totalSupply(uint256 _id) external view returns (uint256); function exists(uint256 _tokenId) external view returns(bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC1155/extensions/IERC1155MetadataURI.sol) pragma solidity ^0.8.0; import "IERC1155.sol"; /** * @dev Interface of the optional ERC1155MetadataExtension interface, as defined * in the https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[EIP]. * * _Available since v3.1._ */ interface IERC1155MetadataURI is IERC1155 { /** * @dev Returns the URI for token type `id`. * * If the `\{id\}` substring is present in the URI, it must be replaced by * clients with the actual token type ID. */ function uri(uint256 id) external view returns (string memory); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC1155/IERC1155.sol) pragma solidity ^0.8.0; import "IERC165.sol"; /** * @dev Required interface of an ERC1155 compliant contract, as defined in the * https://eips.ethereum.org/EIPS/eip-1155[EIP]. * * _Available since v3.1._ */ interface IERC1155 is IERC165 { /** * @dev Emitted when `value` tokens of token type `id` are transferred from `from` to `to` by `operator`. */ event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value); /** * @dev Equivalent to multiple {TransferSingle} events, where `operator`, `from` and `to` are the same for all * transfers. */ event TransferBatch( address indexed operator, address indexed from, address indexed to, uint256[] ids, uint256[] values ); /** * @dev Emitted when `account` grants or revokes permission to `operator` to transfer their tokens, according to * `approved`. */ event ApprovalForAll(address indexed account, address indexed operator, bool approved); /** * @dev Emitted when the URI for token type `id` changes to `value`, if it is a non-programmatic URI. * * If an {URI} event was emitted for `id`, the standard * https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[guarantees] that `value` will equal the value * returned by {IERC1155MetadataURI-uri}. */ event URI(string value, uint256 indexed id); /** * @dev Returns the amount of tokens of token type `id` owned by `account`. * * Requirements: * * - `account` cannot be the zero address. */ function balanceOf(address account, uint256 id) external view returns (uint256); /** * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {balanceOf}. * * Requirements: * * - `accounts` and `ids` must have the same length. */ function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids) external view returns (uint256[] memory); /** * @dev Grants or revokes permission to `operator` to transfer the caller's tokens, according to `approved`, * * Emits an {ApprovalForAll} event. * * Requirements: * * - `operator` cannot be the caller. */ function setApprovalForAll(address operator, bool approved) external; /** * @dev Returns true if `operator` is approved to transfer ``account``'s tokens. * * See {setApprovalForAll}. */ function isApprovedForAll(address account, address operator) external view returns (bool); /** * @dev Transfers `amount` tokens of token type `id` from `from` to `to`. * * Emits a {TransferSingle} event. * * Requirements: * * - `to` cannot be the zero address. * - If the caller is not `from`, it must be have been approved to spend ``from``'s tokens via {setApprovalForAll}. * - `from` must have a balance of tokens of type `id` of at least `amount`. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the * acceptance magic value. */ function safeTransferFrom( address from, address to, uint256 id, uint256 amount, bytes calldata data ) external; /** * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {safeTransferFrom}. * * Emits a {TransferBatch} event. * * Requirements: * * - `ids` and `amounts` must have the same length. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the * acceptance magic value. */ function safeBatchTransferFrom( address from, address to, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data ) external; }
// SPDX-License-Identifier: MIT // ENVELOP(NIFTSY) protocol V1 for NFT. pragma solidity 0.8.13; import "LibEnvelopTypes.sol"; interface ITokenService { error UnSupportedAsset(ETypes.AssetItem asset); function mintNFT( ETypes.AssetType _mint_type, address _contract, address _mintFor, uint256 _tokenId, uint256 _outBalance ) external; function burnNFT( ETypes.AssetType _burn_type, address _contract, address _burnFor, uint256 _tokenId, uint256 _balance ) external; function transfer( ETypes.AssetItem memory _assetItem, address _from, address _to ) external returns (bool _transfered); function transferSafe( ETypes.AssetItem memory _assetItem, address _from, address _to ) external returns (uint256 _transferedValue); // This function must never revert. Use it for unwrap in case some // collateral transfers are revert function transferEmergency( ETypes.AssetItem memory _assetItem, address _from, address _to ) external returns (uint256 _transferedValue); }
{ "evmVersion": "istanbul", "optimizer": { "enabled": true, "runs": 200 }, "libraries": { "WrapperBaseV1.sol": {} }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } } }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"address","name":"_erc20","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"components":[{"components":[{"internalType":"enum ETypes.AssetType","name":"assetType","type":"uint8"},{"internalType":"address","name":"contractAddress","type":"address"}],"internalType":"struct ETypes.Asset","name":"asset","type":"tuple"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct ETypes.AssetItem","name":"asset","type":"tuple"}],"name":"UnSupportedAsset","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"wrappedAddress","type":"address"},{"indexed":true,"internalType":"uint256","name":"wrappedId","type":"uint256"},{"indexed":false,"internalType":"uint8","name":"assetType","type":"uint8"},{"indexed":false,"internalType":"address","name":"collateralAddress","type":"address"},{"indexed":false,"internalType":"uint256","name":"collateralTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"collateralBalance","type":"uint256"}],"name":"CollateralAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"receiver","type":"address"},{"indexed":true,"internalType":"address","name":"wNFTConatract","type":"address"},{"indexed":true,"internalType":"uint256","name":"wNFTTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"EnvelopFee","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"wrappedAddress","type":"address"},{"indexed":true,"internalType":"uint256","name":"wrappedId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"lastCollateralIndex","type":"uint256"}],"name":"PartialUnWrapp","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"wrappedAddress","type":"address"},{"indexed":true,"internalType":"uint256","name":"wrappedId","type":"uint256"},{"indexed":true,"internalType":"address","name":"failedContractAddress","type":"address"}],"name":"SuspiciousFail","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"wrappedAddress","type":"address"},{"indexed":true,"internalType":"address","name":"originalAddress","type":"address"},{"indexed":true,"internalType":"uint256","name":"wrappedId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"originalTokenId","type":"uint256"},{"indexed":false,"internalType":"address","name":"beneficiary","type":"address"},{"indexed":false,"internalType":"uint256","name":"nativeCollateralAmount","type":"uint256"},{"indexed":false,"internalType":"bytes2","name":"rules","type":"bytes2"}],"name":"UnWrappedV1","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"inAssetAddress","type":"address"},{"indexed":true,"internalType":"address","name":"outAssetAddress","type":"address"},{"indexed":true,"internalType":"uint256","name":"inAssetTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"outTokenId","type":"uint256"},{"indexed":false,"internalType":"address","name":"wnftFirstOwner","type":"address"},{"indexed":false,"internalType":"uint256","name":"nativeCollateralAmount","type":"uint256"},{"indexed":false,"internalType":"bytes2","name":"rules","type":"bytes2"}],"name":"WrappedV1","type":"event"},{"inputs":[],"name":"MAX_COLLATERAL_SLOTS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"},{"components":[{"components":[{"internalType":"enum ETypes.AssetType","name":"assetType","type":"uint8"},{"internalType":"address","name":"contractAddress","type":"address"}],"internalType":"struct ETypes.Asset","name":"asset","type":"tuple"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct ETypes.AssetItem[]","name":"_collateral","type":"tuple[]"}],"name":"addCollateral","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"},{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"bytes1","name":"_feeType","type":"bytes1"}],"name":"chargeFees","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"},{"internalType":"enum ETypes.AssetType","name":"_collateralType","type":"uint8"},{"internalType":"address","name":"_erc","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"getCollateralBalanceAndIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"}],"name":"getOriginalURI","outputs":[{"internalType":"string","name":"uri_","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"}],"name":"getWrappedToken","outputs":[{"components":[{"components":[{"components":[{"internalType":"enum ETypes.AssetType","name":"assetType","type":"uint8"},{"internalType":"address","name":"contractAddress","type":"address"}],"internalType":"struct ETypes.Asset","name":"asset","type":"tuple"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct ETypes.AssetItem","name":"inAsset","type":"tuple"},{"components":[{"components":[{"internalType":"enum ETypes.AssetType","name":"assetType","type":"uint8"},{"internalType":"address","name":"contractAddress","type":"address"}],"internalType":"struct ETypes.Asset","name":"asset","type":"tuple"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct ETypes.AssetItem[]","name":"collateral","type":"tuple[]"},{"internalType":"address","name":"unWrapDestination","type":"address"},{"components":[{"internalType":"bytes1","name":"feeType","type":"bytes1"},{"internalType":"uint256","name":"param","type":"uint256"},{"internalType":"address","name":"token","type":"address"}],"internalType":"struct ETypes.Fee[]","name":"fees","type":"tuple[]"},{"components":[{"internalType":"bytes1","name":"lockType","type":"bytes1"},{"internalType":"uint256","name":"param","type":"uint256"}],"internalType":"struct ETypes.Lock[]","name":"locks","type":"tuple[]"},{"components":[{"internalType":"address","name":"beneficiary","type":"address"},{"internalType":"uint16","name":"percent","type":"uint16"}],"internalType":"struct ETypes.Royalty[]","name":"royalties","type":"tuple[]"},{"internalType":"bytes2","name":"rules","type":"bytes2"}],"internalType":"struct ETypes.WNFT","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"enum ETypes.AssetType","name":"","type":"uint8"}],"name":"lastWNFTId","outputs":[{"internalType":"address","name":"contractAddress","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"uint256[]","name":"","type":"uint256[]"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155BatchReceived","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC1155Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"bytes","name":"","type":"bytes"}],"name":"onERC721Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"protocolTechToken","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"protocolWhiteList","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"enum ETypes.AssetType","name":"_assetOutType","type":"uint8"},{"internalType":"address","name":"_wnftContract","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"setWNFTId","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_wlAddress","type":"address"}],"name":"setWhiteList","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":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"enum ETypes.AssetType","name":"_wNFTType","type":"uint8"},{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"},{"internalType":"bool","name":"_isEmergency","type":"bool"}],"name":"unWrap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"enum ETypes.AssetType","name":"_wNFTType","type":"uint8"},{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"}],"name":"unWrap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_wNFTAddress","type":"address"},{"internalType":"uint256","name":"_wNFTTokenId","type":"uint256"}],"name":"unWrap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"wnftTypes","outputs":[{"internalType":"enum ETypes.AssetType","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"components":[{"components":[{"internalType":"enum ETypes.AssetType","name":"assetType","type":"uint8"},{"internalType":"address","name":"contractAddress","type":"address"}],"internalType":"struct ETypes.Asset","name":"asset","type":"tuple"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct ETypes.AssetItem","name":"inAsset","type":"tuple"},{"internalType":"address","name":"unWrapDestination","type":"address"},{"components":[{"internalType":"bytes1","name":"feeType","type":"bytes1"},{"internalType":"uint256","name":"param","type":"uint256"},{"internalType":"address","name":"token","type":"address"}],"internalType":"struct ETypes.Fee[]","name":"fees","type":"tuple[]"},{"components":[{"internalType":"bytes1","name":"lockType","type":"bytes1"},{"internalType":"uint256","name":"param","type":"uint256"}],"internalType":"struct ETypes.Lock[]","name":"locks","type":"tuple[]"},{"components":[{"internalType":"address","name":"beneficiary","type":"address"},{"internalType":"uint16","name":"percent","type":"uint16"}],"internalType":"struct ETypes.Royalty[]","name":"royalties","type":"tuple[]"},{"internalType":"enum ETypes.AssetType","name":"outType","type":"uint8"},{"internalType":"uint256","name":"outBalance","type":"uint256"},{"internalType":"bytes2","name":"rules","type":"bytes2"}],"internalType":"struct ETypes.INData","name":"_inData","type":"tuple"},{"components":[{"components":[{"internalType":"enum ETypes.AssetType","name":"assetType","type":"uint8"},{"internalType":"address","name":"contractAddress","type":"address"}],"internalType":"struct ETypes.Asset","name":"asset","type":"tuple"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct ETypes.AssetItem[]","name":"_collateral","type":"tuple[]"},{"internalType":"address","name":"_wrappFor","type":"address"}],"name":"wrap","outputs":[{"components":[{"components":[{"internalType":"enum ETypes.AssetType","name":"assetType","type":"uint8"},{"internalType":"address","name":"contractAddress","type":"address"}],"internalType":"struct ETypes.Asset","name":"asset","type":"tuple"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct ETypes.AssetItem","name":"","type":"tuple"}],"stateMutability":"payable","type":"function"}]
Contract Creation Code
608060405260196002553480156200001657600080fd5b5060405162005a0938038062005a09833981016040819052620000399162000178565b6001600055620000493362000126565b6001600160a01b038116620000b05760405162461bcd60e51b8152602060048201526024808201527f50726f746f636f6c54656368546f6b656e2063616e74206265207a65726f2076604482015263616c756560e01b606482015260840160405180910390fd5b600380546001600160a01b0319166001600160a01b03831690811790915560408051632dceecbb60e21b8152905163b73bb2ec9160048082019260009290919082900301818387803b1580156200010657600080fd5b505af11580156200011b573d6000803e3d6000fd5b5050505050620001aa565b600180546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b6000602082840312156200018b57600080fd5b81516001600160a01b0381168114620001a357600080fd5b9392505050565b61584f80620001ba6000396000f3fe6080604052600436106101405760003560e01c80637f6d4c93116100b6578063bc197c811161006f578063bc197c811461040b578063c424d4f714610437578063f1551a9c14610464578063f23a6e6114610499578063f2fde38b146104c5578063faf7d720146104e557600080fd5b80637f6d4c93146103495780638da5cb5b1461036957806391ddb14614610387578063980550ca1461039a5780639a7b0509146103be578063a5041040146103eb57600080fd5b80633360aa3c116101085780633360aa3c1461023857806339e899ee1461029757806342fb01a8146102b75780634d36d085146102d757806366967cbb146102f7578063715018a61461033457600080fd5b806301ffc9a71461014557806310118ebb1461017a578063150b7a02146101b2578063320a18dc146101f6578063331758e614610218575b600080fd5b34801561015157600080fd5b50610165610160366004614741565b610505565b60405190151581526020015b60405180910390f35b34801561018657600080fd5b5060045461019a906001600160a01b031681565b6040516001600160a01b039091168152602001610171565b3480156101be57600080fd5b506101dd6101cd36600461488d565b630a85bd0160e11b949350505050565b6040516001600160e01b03199091168152602001610171565b34801561020257600080fd5b50610216610211366004614905565b61053c565b005b34801561022457600080fd5b5060035461019a906001600160a01b031681565b34801561024457600080fd5b50610278610253366004614946565b600560205260009081526040902080546001909101546001600160a01b039091169082565b604080516001600160a01b039093168352602083019190915201610171565b3480156102a357600080fd5b506102166102b2366004614963565b61066a565b3480156102c357600080fd5b506102166102d236600461498e565b6106b6565b6102ea6102e5366004614a2c565b6107b0565b6040516101719190614b13565b34801561030357600080fd5b50610327610312366004614963565b60066020526000908152604090205460ff1681565b6040516101719190614b21565b34801561034057600080fd5b50610216610e59565b34801561035557600080fd5b50610165610364366004614b45565b610e8f565b34801561037557600080fd5b506001546001600160a01b031661019a565b610216610395366004614bad565b610f47565b3480156103a657600080fd5b506103b060025481565b604051908152602001610171565b3480156103ca57600080fd5b506103de6103d9366004614c08565b610fbe565b6040516101719190614c60565b3480156103f757600080fd5b50610216610406366004614905565b6110e1565b34801561041757600080fd5b506101dd610426366004614d1c565b63bc197c8160e01b95945050505050565b34801561044357600080fd5b50610457610452366004614c08565b6110f3565b6040516101719190614ef4565b34801561047057600080fd5b5061048461047f366004614fc1565b611465565b60408051928352602083019190915201610171565b3480156104a557600080fd5b506101dd6104b436600461501c565b63f23a6e6160e01b95945050505050565b3480156104d157600080fd5b506102166104e0366004614963565b61163c565b3480156104f157600080fd5b50610216610500366004614c08565b6116d7565b60006001600160e01b03198216630271189760e51b148061053657506301ffc9a760e01b6001600160e01b03198316145b92915050565b6001546001600160a01b0316331461056f5760405162461bcd60e51b815260040161056690615084565b60405180910390fd5b6001600160a01b0382166105b75760405162461bcd60e51b815260206004820152600f60248201526e4e6f207a65726f206164647265737360881b6044820152606401610566565b6040518060400160405280836001600160a01b0316815260200182815250600560008560078111156105eb576105eb614aa5565b60078111156105fc576105fc614aa5565b815260208082019290925260409081016000908120845181546001600160a01b0319166001600160a01b03918216178255948401516001918201559386168152600690925290208054859260ff199091169083600781111561066057610660614aa5565b0217905550505050565b6001546001600160a01b031633146106945760405162461bcd60e51b815260040161056690615084565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b6000806106c4868686611707565b915091506106d28585611a26565b6106db57600080fd5b6106ec85853330600360f81b611d5a565b5060006106fe86866001600080611465565b50905061070c868686612145565b610718575050506107aa565b6107258787858886612789565b6001600160a01b0386811660008181526007602090815260408083208a84529091529081902080546001820154600483015460089093015493518b96610100909304831695947fabb50c1815800da62a4637d3272d1584df1ee8cbd963d90fb44eb55b63acfcb59461079e941691899160f01b906150b9565b60405180910390a45050505b50505050565b6040805160a0810182526000606082018181526080830182905282526020820181905291810182905290546001190161082b5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610566565b600260005561083a858361288d565b6108785760405162461bcd60e51b815260206004820152600f60248201526e15dc985c0818da1958dac819985a5b608a1b6044820152606401610566565b60016108876020870187614946565b600781111561089857610898614aa5565b141580156108c4575060006108b06020870187614946565b60078111156108c1576108c1614aa5565b14155b15610934576108e26108db368790038701876150e7565b3330612bf0565b6108eb8661324d565b146109345760405162461bcd60e51b81526020600482015260196024820152780537573706963696f757320617373657420666f72207772617603c1b6044820152606401610566565b60016005600061094c61012089016101008a01614946565b600781111561095d5761095d614aa5565b600781111561096e5761096e614aa5565b8152602001908152602001600020600101600082825461098e9190615191565b90915550610a5490506109a961012087016101008801614946565b600560006109bf6101208a016101008b01614946565b60078111156109d0576109d0614aa5565b60078111156109e1576109e1614aa5565b815260208101919091526040016000908120546001600160a01b0316908590600590610a156101208c016101008d01614946565b6007811115610a2657610a26614aa5565b6007811115610a3757610a37614aa5565b815260200190815260200160002060010154896101200135613284565b610afb60056000610a6d61012089016101008a01614946565b6007811115610a7e57610a7e614aa5565b6007811115610a8f57610a8f614aa5565b815260208101919091526040016000908120546001600160a01b031690600590610ac16101208a016101008b01614946565b6007811115610ad257610ad2614aa5565b6007811115610ae357610ae3614aa5565b81526020019081526020016000206001015487613327565b610ba360056000610b1461012089016101008a01614946565b6007811115610b2557610b25614aa5565b6007811115610b3657610b36614aa5565b815260208101919091526040016000908120546001600160a01b031690600590610b686101208a016101008b01614946565b6007811115610b7957610b79614aa5565b6007811115610b8a57610b8a614aa5565b8152602001908152602001600020600101548686610f47565b610c5060056000610bbc61012089016101008a01614946565b6007811115610bcd57610bcd614aa5565b6007811115610bde57610bde614aa5565b815260208101919091526040016000908120546001600160a01b031690600590610c106101208a016101008b01614946565b6007811115610c2157610c21614aa5565b6007811115610c3257610c32614aa5565b8152602001908152602001600020600101543330600260f81b611d5a565b50604085013560056000610c6c61012089016101008a01614946565b6007811115610c7d57610c7d614aa5565b6007811115610c8e57610c8e614aa5565b81526020808201929092526040908101600020546001600160a01b031691610cba918901908901614963565b6001600160a01b03167fa90a3b8dae41ae10a708d32fec7bf12da5c90879c98b9c4cca3c8fba91ddf49360056000610cfa6101208c016101008d01614946565b6007811115610d0b57610d0b614aa5565b6007811115610d1c57610d1c614aa5565b81526020810191909152604001600020600101548634610d446101608d016101408e016151a9565b604051610d5494939291906150b9565b60405180910390a46040805160a08101909152806060810180610d7f6101208a016101008b01614946565b6007811115610d9057610d90614aa5565b815260200160056000610dab6101208c016101008d01614946565b6007811115610dbc57610dbc614aa5565b6007811115610dcd57610dcd614aa5565b81526020808201929092526040016000908120546001600160a01b0316909252918352910190600590610e086101208a016101008b01614946565b6007811115610e1957610e19614aa5565b6007811115610e2a57610e2a614aa5565b815260200190815260200160002060010154815260200186610120013581525090506001600055949350505050565b6001546001600160a01b03163314610e835760405162461bcd60e51b815260040161056690615084565b610e8d60006135b2565b565b6000336001600160a01b0387161480610ea757503330145b610ef35760405162461bcd60e51b815260206004820152601860248201527f4f6e6c7920666f7220774e4654206f72207772617070657200000000000000006044820152606401610566565b610f008686868686611d5a565b610f3e5760405162461bcd60e51b815260206004820152600f60248201526e1199594818da185c99d94819985a5b608a1b6044820152606401610566565b95945050505050565b80151580610f555750600034115b156107aa57610f6684848484613604565b610fb25760405162461bcd60e51b815260206004820152601860248201527f466f7262696464656e2061646420636f6c6c61746572616c00000000000000006044820152606401610566565b6107aa848484846137e0565b60606000610fcc84846110f3565b51905060038151516007811115610fe557610fe5614aa5565b036110715780516020908101519082015160405163c87b56dd60e01b81526001600160a01b039092169163c87b56dd916110259160040190815260200190565b600060405180830381865afa158015611042573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261106a91908101906151d3565b91506110da565b6004815151600781111561108757611087614aa5565b036110c7578051602090810151908201516040516303a24d0760e21b81526001600160a01b0390921691630e89341c916110259160040190815260200190565b6040518060200160405280600081525091505b5092915050565b6110ee83838360006106b6565b505050565b604080516101808101825260006101408201818152610160830182905260e083019081526101008301829052610120830182905282526060602083018190529282018190528282018390526080820183905260a082019290925260c08101919091526001600160a01b0383166000908152600760208181526040808420868552909152918290208251610180810190935280549091839160e0830191849183916101408601918491839160ff16908111156111b0576111b0614aa5565b60078111156111c1576111c1614aa5565b8152905461010090046001600160a01b031660209182015290825260018301548282015260029092015460409182015291835260038401805483518184028101840190945280845293820193909160009084015b828210156112ac576000848152602090206040805160a08101909152600384029091018054829060608201908390829060ff16600781111561125957611259614aa5565b600781111561126a5761126a614aa5565b8152905461010090046001600160a01b031660209182015290825260018381015483830152600290930154604090920191909152918352929092019101611215565b5050509082525060048201546001600160a01b0316602080830191909152600583018054604080518285028101850182528281529401939260009084015b8282101561134c5760008481526020908190206040805160608101825260038602909201805460f81b6001600160f81b0319168352600180820154848601526002909101546001600160a01b03169183019190915290835290920191016112ea565b50505050815260200160068201805480602002602001604051908101604052809291908181526020016000905b828210156113c55760008481526020908190206040805180820190915260028502909101805460f81b6001600160f81b0319168252600190810154828401529083529092019101611379565b50505050815260200160078201805480602002602001604051908101604052809291908181526020016000905b8282101561143c57600084815260209081902060408051808201909152908401546001600160a01b0381168252600160a01b900461ffff16818301528252600190920191016113f2565b505050908252506008919091015460f01b6001600160f01b031916602090910152905092915050565b60008060005b6001600160a01b03881660009081526007602090815260408083208a8452909152902060030154811015611630576001600160a01b0388811660009081526007602090815260408083208b84529091529020600301805491871691839081106114d6576114d6615240565b600091825260209091206003909102015461010090046001600160a01b031614801561154c57506001600160a01b03881660009081526007602090815260408083208a8452909152902060030180548591908390811061153857611538615240565b906000526020600020906003020160010154145b80156115c4575085600781111561156557611565614aa5565b6001600160a01b03891660009081526007602090815260408083208b8452909152902060030180548390811061159d5761159d615240565b600091825260209091206003909102015460ff1660078111156115c2576115c2614aa5565b145b1561161e576001600160a01b03881660009081526007602090815260408083208a8452909152902060030180548290811061160157611601615240565b906000526020600020906003020160020154819250925050611632565b8061162881615256565b91505061146b565b505b9550959350505050565b6001546001600160a01b031633146116665760405162461bcd60e51b815260040161056690615084565b6001600160a01b0381166116cb5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610566565b6116d4816135b2565b50565b6001600160a01b0382166000908152600660205260408120546117039160ff90911690849084906106b6565b5050565b600080611734600160f01b61171c86866110f3565b60c0015181166001600160f01b031990811691161490565b156117815760405162461bcd60e51b815260206004820152601b60248201527f556e577261707020666f7262696464656e20627920617574686f7200000000006044820152606401610566565b600385600781111561179557611795614aa5565b03611862576040516331a9108f60e11b8152600481018490526001600160a01b03851690636352211e90602401602060405180830381865afa1580156117df573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611803919061526f565b91506001600160a01b038216331461185d5760405162461bcd60e51b815260206004820152601860248201527f4f6e6c79206f776e65722063616e20756e7772617020697400000000000000006044820152606401610566565b611a1e565b600485600781111561187657611876614aa5565b036119bf5760405163bd85b03960e01b8152600481018490526001600160a01b0385169063bd85b03990602401602060405180830381865afa1580156118c0573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118e4919061528c565b604051627eeac760e11b815233600482018190526024820186905293509091506001600160a01b0385169062fdd58e90604401602060405180830381865afa158015611934573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611958919061528c565b811461185d5760405162461bcd60e51b815260206004820152603060248201527f45524331313520756e7772617020617661696c61626c65206f6e6c7920666f7260448201526f20616c6c20746f74616c537570706c7960801b6064820152608401610566565b6040805160a081019091528060608101808860078111156119e2576119e2614aa5565b8152602001876001600160a01b03168152508152602001848152602001600081525060405163391102fb60e01b81526004016105669190614b13565b935093915050565b6000805b6001600160a01b0384166000908152600760209081526040808320868452909152902060060154811015611d50576001600160a01b03841660009081526007602090815260408083208684529091529020600601805482908110611a9057611a90615240565b600091825260208220600291909102015460f81b6001600160f81b0319169003611b40576001600160a01b03841660009081526007602090815260408083208684529091529020600601805442919083908110611aef57611aef615240565b9060005260206000209060020201600101541115611b405760405162461bcd60e51b815260206004820152600e60248201526d2a34b6b2a637b1b59032b93937b960911b6044820152606401610566565b6001600160a01b03841660009081526007602090815260408083208684529091529020600601805482908110611b7857611b78615240565b600091825260209091206002909102015460f81b6001600160f81b031916600160f81b03611d3e5760005b6001600160a01b0385166000908152600760209081526040808320878452909152902060050154811015611d3c576001600160a01b03851660009081526007602090815260408083208784529091529020600501805482908110611c0957611c09615240565b600091825260208220600391909102015460f81b6001600160f81b0319169003611d2a576001600160a01b038516600090815260076020908152604080832087845290915281206005018054611c92918891889160029187908110611c7057611c70615240565b600091825260208220600260039092020101546001600160a01b031690611465565b506001600160a01b038716600090815260076020908152604080832089845290915290206006018054919250829185908110611cd057611cd0615240565b9060005260206000209060020201600101541115611d285760405162461bcd60e51b81526020600482015260156024820152742a3930b739b332b92332b2a637b1b59032b93937b960591b6044820152606401610566565b505b80611d3481615256565b915050611ba3565b505b80611d4881615256565b915050611a2a565b5060019392505050565b60006001600160f81b031982168103610f3e5760005b6001600160a01b038716600090815260076020908152604080832089845290915290206005015481101561213b576001600160a01b03871660009081526007602090815260408083208984529091529020600501805482908110611dd657611dd6615240565b600091825260208220600391909102015460f81b6001600160f81b0319169003612129576003546004546001600160a01b03918216911615611edc576004546001600160a01b0389811660009081526007602090815260408083208c84529091529020600501805491909216916373cf00f69185908110611e5957611e59615240565b600091825260209091206003909102016002015460405160e083901b6001600160e01b03191681526001600160a01b039091166004820152602401608060405180830381865afa158015611eb1573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611ed591906152a5565b6060015190505b6000806000836001600160a01b031663ce244ce1600760008e6001600160a01b03166001600160a01b0316815260200190815260200160002060008d81526020019081526020016000206005018781548110611f3a57611f3a615240565b9060005260206000209060030201600760008f6001600160a01b03166001600160a01b0316815260200190815260200160002060008e81526020019081526020016000206007018c8c6040518563ffffffff1660e01b8152600401611fa29493929190615329565b600060405180830381865afa158015611fbf573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052611fe79190810190615441565b919450925090506000805b825181101561212257306001600160a01b031683828151811061201757612017615240565b60200260200101516001600160a01b031603612051576120518d8d87848151811061204457612044615240565b6020026020010151613bb3565b6120a785828151811061206657612066615240565b602002602001015185838151811061208057612080615240565b602002602001015185848151811061209a5761209a615240565b6020026020010151612bf0565b91508b8d6001600160a01b03168483815181106120c6576120c6615240565b60200260200101516001600160a01b03167f2e7d475f6480b44a6b26e0a71fd4bf4fbf7842b803ad204a4d23042a02bd73358560405161210891815260200190565b60405180910390a48061211a81615256565b915050611ff2565b5050505050505b8061213381615256565b915050611d70565b5060019050610f3e565b6001600160a01b03838116600090815260076020908152604080832086845290915281206004015490918291339116156121a457506001600160a01b038086166000908152600760209081526040808320888452909152902060040154165b60005b6001600160a01b03871660009081526007602090815260408083208984529091529020600301548110156125b1576001600160a01b0387166000908152600760209081526040808320898452909152812060030180548390811061220d5761220d615240565b600091825260209091206003909102015460ff16600781111561223257612232614aa5565b14612507578415612305576001600160a01b0387166000908152600760209081526040808320898452909152902060030180546122fe91908390811061227a5761227a615240565b600091825260209091206040805160a081019091526003909202018054829060608201908390829060ff1660078111156122b6576122b6614aa5565b60078111156122c7576122c7614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523084613e04565b92506123c9565b6001600160a01b0387166000908152600760209081526040808320898452909152902060030180546123c691908390811061234257612342615240565b600091825260209091206040805160a081019091526003909202018054829060608201908390829060ff16600781111561237e5761237e614aa5565b600781111561238f5761238f614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523084612bf0565b92505b6001600160a01b0387166000908152600760209081526040808320898452909152902060030180548290811061240157612401615240565b906000526020600020906003020160020154831461249f576001600160a01b0387166000908152600760209081526040808320898452909152902060030180548290811061245157612451615240565b600091825260208220600390910201546040516001600160a01b0361010090920482169289928b16917ffca203c3f6987c2a1dae80f773c277d67920e7bce0cea9c07cd0eb8142e985ca9190a45b6001600160a01b038716600090815260076020908152604080832089845290915281206003018054839081106124d7576124d7615240565b60009182526020909120600390910201805460ff1916600183600781111561250157612501614aa5565b02179055505b6103e85a1115801561254a57506001600160a01b03871660009081526007602090815260408083208984529091529020600301546125479060019061558a565b81105b1561259f5785876001600160a01b03167fd66d44264f9d44e254da71183ff08098f38da4675285592ee80cdbd3b6f5153e8360405161258b91815260200190565b60405180910390a360009350505050612782565b806125a981615256565b9150506121a7565b5060016001600160a01b03871660009081526007602081815260408084208a85529091529091205460ff16908111156125ec576125ec614aa5565b1415801561262f57506001600160a01b038616600090815260076020818152604080842089855290915282205460ff169081111561262c5761262c614aa5565b14155b1561277b57836126dc576001600160a01b038616600090815260076020818152604080842089855290915291829020825160a0810190935280546126d69392839160608301918491839160ff9091169081111561268e5761268e614aa5565b600781111561269f5761269f614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523083612bf0565b5061277b565b6001600160a01b038616600090815260076020818152604080842089855290915291829020825160a0810190935280546127799392839160608301918491839160ff9091169081111561273157612731614aa5565b600781111561274257612742614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523083613e04565b505b6001925050505b9392505050565b600385600781111561279d5761279d614aa5565b0361280257604051630852cd8d60e31b8152600481018390526001600160a01b038516906342966c68906024015b600060405180830381600087803b1580156127e557600080fd5b505af11580156127f9573d6000803e3d6000fd5b50505050612886565b600485600781111561281657612816614aa5565b0361288657604051637a94c56560e11b81526001600160a01b038481166004830152602482018490526044820183905285169063f5298aca906064015b600060405180830381600087803b15801561286d57600080fd5b505af1158015612881573d6000803e3d6000fd5b505050505b5050505050565b60006128b4600160f11b61171c6128aa6040870160208801614963565b60408701356110f3565b1580156128ca57506001600160a01b0382163014155b6004549091506001600160a01b031615610536576004546001600160a01b0316638f8b138e6128ff6040860160208701614963565b6040516001600160e01b031960e084901b1681526001600160a01b039091166004820152602401602060405180830381865afa158015612943573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061296791906155a1565b156129b45760405162461bcd60e51b815260206004820152601a60248201527f574c3a41737365742064697361626c656420666f7220777261700000000000006044820152606401610566565b6004546001600160a01b03166352cdc6a66129d56040860160208701614963565b6129e7610160870161014088016151a9565b6040516001600160e01b031960e085901b1681526001600160a01b0390921660048301526001600160f01b0319166024820152604401602060405180830381865afa158015612a3a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612a5e91906155a1565b612abc5760405162461bcd60e51b815260206004820152602960248201527f574c3a536f6d652072756c6573206172652064697361626c656420666f7220746044820152681a1a5cc8185cdcd95d60ba1b6064820152608401610566565b60005b612acc60a08501856155be565b90508110156110da576004546001600160a01b031663b6e306ac612af360a08701876155be565b84818110612b0357612b03615240565b9050606002016040016020810190612b1b9190614963565b6040516001600160e01b031960e084901b1681526001600160a01b039091166004820152602401602060405180830381865afa158015612b5f573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612b8391906155a1565b612bde5760405162461bcd60e51b815260206004820152602660248201527f574c3a536f6d652061737365747320617265206e6f7420656e61626c656420666044820152656f722066656560d01b6064820152608401610566565b80612be881615256565b915050612abf565b60008060018551516007811115612c0957612c09614aa5565b03612cc3575060408085015190516001600160a01b0384168031926000928381818185875af1925050503d8060008114612c5f576040519150601f19603f3d011682016040523d82523d6000602084013e612c64565b606091505b5050905080612ca75760405162461bcd60e51b815260206004820152600f60248201526e1d1c985b9cd9995c8819985a5b1959608a1b6044820152606401610566565b612cbb826001600160a01b0386163161558a565b925050613245565b60028551516007811115612cd957612cd9614aa5565b03612e2f578451602001516040516370a0823160e01b81526001600160a01b038581166004830152909116906370a0823190602401602060405180830381865afa158015612d2b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612d4f919061528c565b9050306001600160a01b03851603612d88576040850151855160200151612d83916001600160a01b039091169085906141d5565b612dac565b6040850151855160200151612dac916001600160a01b039091169086908690614238565b8451602001516040516370a0823160e01b81526001600160a01b038581166004830152839216906370a08231906024015b602060405180830381865afa158015612dfa573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612e1e919061528c565b612e28919061558a565b9150613245565b60038551516007811115612e4557612e45614aa5565b148015612ed157508451602090810151908601516040516331a9108f60e11b815260048101919091526001600160a01b03868116921690636352211e90602401602060405180830381865afa158015612ea2573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612ec6919061526f565b6001600160a01b0316145b156130db578451602001516040516370a0823160e01b81526001600160a01b038581166004830152909116906370a0823190602401602060405180830381865afa158015612f23573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612f47919061528c565b90508460000151602001516001600160a01b03166323b872dd858588602001516040518463ffffffff1660e01b8152600401612f8593929190615606565b600060405180830381600087803b158015612f9f57600080fd5b505af1158015612fb3573d6000803e3d6000fd5b50505050826001600160a01b03168560000151602001516001600160a01b0316636352211e87602001516040518263ffffffff1660e01b8152600401612ffb91815260200190565b602060405180830381865afa158015613018573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061303c919061526f565b6001600160a01b03161480156130cc57508451602001516040516370a0823160e01b81526001600160a01b038581166004830152839216906370a0823190602401602060405180830381865afa15801561309a573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906130be919061528c565b6130c8919061558a565b6001145b156130d657600191505b613245565b600485515160078111156130f1576130f1614aa5565b0361322a57845160209081015190860151604051627eeac760e11b81526001600160a01b038681166004830152602482019290925291169062fdd58e90604401602060405180830381865afa15801561314e573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613172919061528c565b90508460000151602001516001600160a01b031663f242432a8585886020015189604001516040518563ffffffff1660e01b81526004016131b6949392919061562a565b600060405180830381600087803b1580156131d057600080fd5b505af11580156131e4573d6000803e3d6000fd5b5050865160209081015190880151604051627eeac760e11b81526001600160a01b03888116600483015260248201929092528594509116915062fdd58e90604401612ddd565b8460405163391102fb60e01b81526004016105669190614b13565b509392505050565b6000600361325e6020840184614946565b600781111561326f5761326f614aa5565b0361327c57506001919050565b506060013590565b600385600781111561329857613298614aa5565b036132d2576040516340c10f1960e01b81526001600160a01b038481166004830152602482018490528516906340c10f19906044016127cb565b60048560078111156132e6576132e6614aa5565b0361288657604051630ab714fb60e11b81526001600160a01b038481166004830152602482018490526044820183905285169063156e29f690606401612853565b6001600160a01b0383166000908152600760209081526040808320858452909152902081906133568282615662565b50506001600160a01b0383166000908152600760209081526040808320858452909152902060040180546001600160a01b031916905561339e610160820161014083016151a9565b6001600160a01b03841660009081526007602090815260408083208684529091528120600801805461ffff191660f09390931c929092179091555b6133e660a08301836155be565b9050811015613475576001600160a01b0384166000908152600760209081526040808320868452909152902060050161342260a08401846155be565b8381811061343257613432615240565b8354600181018555600094855260209094206060909102929092019260030290910190506134608282615705565b5050808061346d90615256565b9150506133d9565b5060005b61348660c0830183615744565b9050811015613515576001600160a01b038416600090815260076020908152604080832086845290915290206006016134c260c0840184615744565b838181106134d2576134d2615240565b835460018101855560009485526020909420604090910292909201926002029091019050613500828261578d565b5050808061350d90615256565b915050613479565b5060005b61352660e0830183615744565b90508110156107aa576001600160a01b03841660009081526007602081815260408084208785529091529091200161356160e0840184615744565b8381811061357157613571615240565b8354600181018555600094855260209094206040909102929092019291909101905061359d82826157b5565b505080806135aa90615256565b915050613519565b600180546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600060036001600160a01b03861660009081526006602052604090205460ff16600781111561363557613635614aa5565b036136e757604051634f558e7960e01b8152600481018590526001600160a01b03861690634f558e79906024015b602060405180830381865afa158015613680573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906136a491906155a1565b6136e25760405162461bcd60e51b815260206004820152600f60248201526e774e4654206e6f742065786973747360881b6044820152606401610566565b6137c4565b60046001600160a01b03861660009081526006602052604090205460ff16600781111561371657613716614aa5565b0361374857604051634f558e7960e01b8152600481018590526001600160a01b03861690634f558e7990602401613663565b6040805160a0810182526001600160a01b03871660009081526006602052919091205481906060820190819060ff16600781111561378857613788614aa5565b8152602001886001600160a01b03168152508152602001858152602001600081525060405163391102fb60e01b81526004016105669190614b13565b6137d6600160f31b61171c87876110f3565b1595945050505050565b3415613874576040805160a0810190915261381f9085908590806060810180600181526000602091820181905291835282015234604090910152613bb3565b604080516001815260006020820181905281830152346060820152905184916001600160a01b038716917ff3d1350815c4f9db2be36c35f840bfb002835a83ff1c3d8f3a217b1e6227d5aa9181900360800190a35b60005b8181101561288657600183838381811061389357613893615240565b6138a99260206080909202019081019150614946565b60078111156138ba576138ba614aa5565b14613ba1576004546001600160a01b0316156139d6576004546001600160a01b031663eb9ae17c8484848181106138f3576138f3615240565b61390c9260406080909202019081019150602001614963565b6040516001600160e01b031960e084901b1681526001600160a01b039091166004820152602401602060405180830381865afa158015613950573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061397491906155a1565b6139d65760405162461bcd60e51b815260206004820152602d60248201527f574c3a536f6d652061737365747320617265206e6f7420656e61626c6564206660448201526c1bdc8818dbdb1b185d195c985b609a1b6064820152608401610566565b613a018383838181106139eb576139eb615240565b9050608002018036038101906108db91906150e7565b613a21848484818110613a1657613a16615240565b90506080020161324d565b14613a6a5760405162461bcd60e51b81526020600482015260196024820152780537573706963696f757320617373657420666f72207772617603c1b6044820152606401610566565b613a9c8585858585818110613a8157613a81615240565b905060800201803603810190613a9791906150e7565b613bb3565b83856001600160a01b03167ff3d1350815c4f9db2be36c35f840bfb002835a83ff1c3d8f3a217b1e6227d5aa858585818110613ada57613ada615240565b613af09260206080909202019081019150614946565b6007811115613b0157613b01614aa5565b868686818110613b1357613b13615240565b613b2c9260406080909202019081019150602001614963565b878787818110613b3e57613b3e615240565b90506080020160400135888888818110613b5a57613b5a615240565b90506080020160600135604051613b98949392919060ff9490941684526001600160a01b039290921660208401526040830152606082015260800190565b60405180910390a35b80613bab81615256565b915050613877565b60028151516007811115613bc957613bc9614aa5565b1480613be8575060018151516007811115613be657613be6614aa5565b145b15613c3657602081015115613c365760405162461bcd60e51b8152602060048201526014602482015273546f6b656e4964206d757374206265207a65726f60601b6044820152606401610566565b60038151516007811115613c4c57613c4c614aa5565b03613c9957604081015115613c995760405162461bcd60e51b8152602060048201526013602482015272416d6f756e74206d757374206265207a65726f60681b6044820152606401610566565b6001600160a01b03831660009081526007602090815260408083208584529091529020600301541580613cdf575060038151516007811115613cdd57613cdd614aa5565b145b15613cef576110ee838383614259565b600080613d1385858560000151600001518660000151602001518760200151611465565b915091506000811180613d8c575080158015613d8c575082516020908101516001600160a01b0387811660009081526007845260408082208983529094529283206003018054919092169290613d6b57613d6b615240565b600091825260209091206003909102015461010090046001600160a01b0316145b15613df9576040808401516001600160a01b038716600090815260076020908152838220888352905291909120600301805483908110613dce57613dce615240565b90600052602060002090600302016002016000828254613dee9190615191565b909155506128869050565b612886858585614259565b60008060018551516007811115613e1d57613e1d614aa5565b03613e92575060408085015190516001600160a01b0384168031926000928381818185875af1925050503d8060008114613e73576040519150601f19603f3d011682016040523d82523d6000602084013e613e78565b606091505b5050905081846001600160a01b031631612cbb919061558a565b60028551516007811115613ea857613ea8614aa5565b0361403257306001600160a01b03851603613f7d5760008560000151602001516001600160a01b0316848760400151604051602401613efc9291906001600160a01b03929092168252602082015260400190565b60408051601f198184030181529181526020820180516001600160e01b031663a9059cbb60e01b17905251613f3191906157fd565b6000604051808303816000865af19150503d8060008114613f6e576040519150601f19603f3d011682016040523d82523d6000602084013e613f73565b606091505b5050905050614026565b60008560000151602001516001600160a01b031685858860400151604051602401613faa93929190615606565b60408051601f198184030181529181526020820180516001600160e01b03166323b872dd60e01b17905251613fdf91906157fd565b6000604051808303816000865af19150503d806000811461401c576040519150601f19603f3d011682016040523d82523d6000602084013e614021565b606091505b505050505b84604001519150613245565b6003855151600781111561404857614048614aa5565b036140ff5760008560000151602001516001600160a01b03168585886020015160405160240161407a93929190615606565b60408051601f198184030181529181526020820180516001600160e01b03166323b872dd60e01b179052516140af91906157fd565b6000604051808303816000865af19150503d80600081146140ec576040519150601f19603f3d011682016040523d82523d6000602084013e6140f1565b606091505b505090506001925050613245565b6004855151600781111561411557614115614aa5565b0361322a5760008560000151602001516001600160a01b031685858860200151896040015160405160240161414d949392919061562a565b60408051601f198184030181529181526020820180516001600160e01b0316637921219560e11b1790525161418291906157fd565b6000604051808303816000865af19150503d80600081146141bf576040519150601f19603f3d011682016040523d82523d6000602084013e6141c4565b606091505b505090508560400151925050613245565b6040516001600160a01b0383166024820152604481018290526110ee90849063a9059cbb60e01b906064015b60408051601f198184030181529190526020810180516001600160e01b03166001600160e01b0319909316929092179091526144f7565b6107aa846323b872dd60e01b85858560405160240161420193929190615606565b6002546001600160a01b0384166000908152600760209081526040808320868452909152902060030154106142d05760405162461bcd60e51b815260206004820152601d60248201527f546f6f206d75636820746f6b656e7320696e20636f6c6c61746572616c0000006044820152606401610566565b60005b6001600160a01b038416600090815260076020908152604080832086845290915290206006015481101561444d576001600160a01b0384166000908152600760209081526040808320868452909152902060060180548290811061433957614339615240565b600091825260209091206002909102015460f81b6001600160f81b031916600160f91b0361443b576001600160a01b0384166000908152600760209081526040808320868452909152902060030154614393906001615191565b6001600160a01b038516600090815260076020908152604080832087845290915290206006018054839081106143cb576143cb615240565b906000526020600020906002020160010154101561443b5760405162461bcd60e51b815260206004820152602760248201527f546f6f206d75636820636f6c6c61746572616c20736c6f747320666f722074686044820152661a5cc81dd3919560ca1b6064820152608401610566565b8061444581615256565b9150506142d3565b506001600160a01b038316600090815260076020818152604080842086855282528320600390810180546001818101835591865292909420855180519390920201805486959194929385939092849260ff19169184908111156144b2576144b2614aa5565b021790555060209182015181546001600160a01b0390911661010002610100600160a81b03199091161790558201516001820155604090910151600290910155505050565b600061454c826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166145c99092919063ffffffff16565b8051909150156110ee578080602001905181019061456a91906155a1565b6110ee5760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b6064820152608401610566565b60606145d884846000856145e0565b949350505050565b6060824710156146415760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f6044820152651c8818d85b1b60d21b6064820152608401610566565b843b61468f5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606401610566565b600080866001600160a01b031685876040516146ab91906157fd565b60006040518083038185875af1925050503d80600081146146e8576040519150601f19603f3d011682016040523d82523d6000602084013e6146ed565b606091505b50915091506146fd828286614708565b979650505050505050565b60608315614717575081612782565b8251156147275782518084602001fd5b8160405162461bcd60e51b81526004016105669190614c60565b60006020828403121561475357600080fd5b81356001600160e01b03198116811461278257600080fd5b6001600160a01b03811681146116d457600080fd5b634e487b7160e01b600052604160045260246000fd5b604080519081016001600160401b03811182821017156147b8576147b8614780565b60405290565b604051606081016001600160401b03811182821017156147b8576147b8614780565b604051601f8201601f191681016001600160401b038111828210171561480857614808614780565b604052919050565b60006001600160401b0382111561482957614829614780565b50601f01601f191660200190565b600082601f83011261484857600080fd5b813561485b61485682614810565b6147e0565b81815284602083860101111561487057600080fd5b816020850160208301376000918101602001919091529392505050565b600080600080608085870312156148a357600080fd5b84356148ae8161476b565b935060208501356148be8161476b565b92506040850135915060608501356001600160401b038111156148e057600080fd5b6148ec87828801614837565b91505092959194509250565b600881106116d457600080fd5b60008060006060848603121561491a57600080fd5b8335614925816148f8565b925060208401356149358161476b565b929592945050506040919091013590565b60006020828403121561495857600080fd5b8135612782816148f8565b60006020828403121561497557600080fd5b81356127828161476b565b80151581146116d457600080fd5b600080600080608085870312156149a457600080fd5b84356149af816148f8565b935060208501356149bf8161476b565b92506040850135915060608501356149d681614980565b939692955090935050565b60008083601f8401126149f357600080fd5b5081356001600160401b03811115614a0a57600080fd5b6020830191508360208260071b8501011115614a2557600080fd5b9250929050565b60008060008060608587031215614a4257600080fd5b84356001600160401b0380821115614a5957600080fd5b908601906101608289031215614a6e57600080fd5b90945060208601359080821115614a8457600080fd5b50614a91878288016149e1565b90945092505060408501356149d68161476b565b634e487b7160e01b600052602160045260246000fd5b60088110614ad957634e487b7160e01b600052602160045260246000fd5b9052565b8051614aea838251614abb565b6020908101516001600160a01b0316838201528101516040808401919091520151606090910152565b608081016105368284614add565b602081016105368284614abb565b6001600160f81b0319811681146116d457600080fd5b600080600080600060a08688031215614b5d57600080fd5b8535614b688161476b565b9450602086013593506040860135614b7f8161476b565b92506060860135614b8f8161476b565b91506080860135614b9f81614b2f565b809150509295509295909350565b60008060008060608587031215614bc357600080fd5b8435614bce8161476b565b93506020850135925060408501356001600160401b03811115614bf057600080fd5b614bfc878288016149e1565b95989497509550505050565b60008060408385031215614c1b57600080fd5b8235614c268161476b565b946020939093013593505050565b60005b83811015614c4f578181015183820152602001614c37565b838111156107aa5750506000910152565b6020815260008251806020840152614c7f816040850160208701614c34565b601f01601f19169190910160400192915050565b60006001600160401b03821115614cac57614cac614780565b5060051b60200190565b600082601f830112614cc757600080fd5b81356020614cd761485683614c93565b82815260059290921b84018101918181019086841115614cf657600080fd5b8286015b84811015614d115780358352918301918301614cfa565b509695505050505050565b600080600080600060a08688031215614d3457600080fd5b8535614d3f8161476b565b94506020860135614d4f8161476b565b935060408601356001600160401b0380821115614d6b57600080fd5b614d7789838a01614cb6565b94506060880135915080821115614d8d57600080fd5b614d9989838a01614cb6565b93506080880135915080821115614daf57600080fd5b50614dbc88828901614837565b9150509295509295909350565b600081518084526020808501945080840160005b83811015614e0357614df0878351614add565b6080969096019590820190600101614ddd565b509495945050505050565b600081518084526020808501945080840160005b83811015614e0357815180516001600160f81b031916885283810151848901526040908101516001600160a01b03169088015260609096019590820190600101614e22565b600081518084526020808501945080840160005b83811015614e0357815180516001600160f81b03191688528301518388015260409096019590820190600101614e7b565b600081518084526020808501945080840160005b83811015614e0357815180516001600160a01b0316885283015161ffff168388015260409096019590820190600101614ec0565b60208152614f06602082018351614add565b600060208301516101408060a0850152614f24610160850183614dc9565b91506040850151614f4060c08601826001600160a01b03169052565b506060850151601f19808685030160e0870152614f5d8483614e0e565b9350608087015191508086850301610100870152614f7b8483614e67565b935060a08701519150808685030161012087015250614f9a8382614eac565b92505060c0850151614fb7828601826001600160f01b0319169052565b5090949350505050565b600080600080600060a08688031215614fd957600080fd5b8535614fe48161476b565b9450602086013593506040860135614ffb816148f8565b9250606086013561500b8161476b565b949793965091946080013592915050565b600080600080600060a0868803121561503457600080fd5b853561503f8161476b565b9450602086013561504f8161476b565b9350604086013592506060860135915060808601356001600160401b0381111561507857600080fd5b614dbc88828901614837565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b9384526001600160a01b0392909216602084015260408301526001600160f01b031916606082015260800190565b600081830360808112156150fa57600080fd5b604051606081018181106001600160401b038211171561511c5761511c614780565b604090815282121561512d57600080fd5b615135614796565b91508335615142816148f8565b825260208401356151528161476b565b806020840152508181526040840135602082015260608401356040820152809250505092915050565b634e487b7160e01b600052601160045260246000fd5b600082198211156151a4576151a461517b565b500190565b6000602082840312156151bb57600080fd5b81356001600160f01b03198116811461278257600080fd5b6000602082840312156151e557600080fd5b81516001600160401b038111156151fb57600080fd5b8201601f8101841361520c57600080fd5b805161521a61485682614810565b81815285602083850101111561522f57600080fd5b610f3e826020830160208601614c34565b634e487b7160e01b600052603260045260246000fd5b6000600182016152685761526861517b565b5060010190565b60006020828403121561528157600080fd5b81516127828161476b565b60006020828403121561529e57600080fd5b5051919050565b6000608082840312156152b757600080fd5b604051608081018181106001600160401b03821117156152d9576152d9614780565b60405282516152e781614980565b815260208301516152f781614980565b6020820152604083015161530a81614980565b6040820152606083015161531d8161476b565b60608201529392505050565b600060c0820160ff60f81b875460f81b1683526001808801546020818187015260018060a01b0391508160028b0154166040818189015260c060608901528591508a5480875260e0890192508b60005283600020965060005b818110156153ab578754868116855260a01c61ffff168585015296860196928201928601615382565b5050506001600160a01b038916608088015294506153c99350505050565b6001600160a01b03831660a0830152610f3e565b600082601f8301126153ee57600080fd5b815160206153fe61485683614c93565b82815260059290921b8401810191818101908684111561541d57600080fd5b8286015b84811015614d115780516154348161476b565b8352918301918301615421565b6000806000606080858703121561545757600080fd5b84516001600160401b038082111561546e57600080fd5b818701915087601f83011261548257600080fd5b8151602061549261485683614c93565b82815260079290921b8401810191818101908b8411156154b157600080fd5b948201945b8386101561553757858c0360808112156154d05760008081fd5b6154d86147be565b6040808312156154e85760008081fd5b6154f0614796565b925088516154fd816148f8565b83528886015161550c8161476b565b83870152918152878201518186015288880151918101919091528252608090950194908201906154b6565b918a015191985090945050508083111561555057600080fd5b61555c888489016153dd565b9450604087015192508083111561557257600080fd5b5050615580868287016153dd565b9150509250925092565b60008282101561559c5761559c61517b565b500390565b6000602082840312156155b357600080fd5b815161278281614980565b6000808335601e198436030181126155d557600080fd5b8301803591506001600160401b038211156155ef57600080fd5b6020019150606081023603821315614a2557600080fd5b6001600160a01b039384168152919092166020820152604081019190915260600190565b6001600160a01b0394851681529290931660208301526040820152606081019190915260a06080820181905260009082015260c00190565b813561566d816148f8565b6008811061568b57634e487b7160e01b600052602160045260246000fd5b815460ff821691508160ff19821617835560208401356156aa8161476b565b6001600160a81b03199190911690911760089190911b610100600160a81b031617815560408201356001820155606090910135600290910155565b80546001600160a01b0319166001600160a01b0392909216919091179055565b813561571081614b2f565b815460ff191660f882901c178255506020820135600182015560408201356157378161476b565b6110ee81600284016156e5565b6000808335601e1984360301811261575b57600080fd5b8301803591506001600160401b0382111561577557600080fd5b6020019150600681901b3603821315614a2557600080fd5b813561579881614b2f565b815460ff191660f882901c17825550602082013560018201555050565b81356157c08161476b565b6157ca81836156e5565b50602082013561ffff811681146157e057600080fd5b815461ffff60a01b191660a09190911b61ffff60a01b1617905550565b6000825161580f818460208701614c34565b919091019291505056fea26469706673582212207aa7f0950aefce26a018a90e4d87708cd7d647fdaf7658f6881718ef4a473ebd64736f6c634300080d003300000000000000000000000044c1bce3f284e9a672fa4ba6f5b02e8651da88a0
Deployed Bytecode
0x6080604052600436106101405760003560e01c80637f6d4c93116100b6578063bc197c811161006f578063bc197c811461040b578063c424d4f714610437578063f1551a9c14610464578063f23a6e6114610499578063f2fde38b146104c5578063faf7d720146104e557600080fd5b80637f6d4c93146103495780638da5cb5b1461036957806391ddb14614610387578063980550ca1461039a5780639a7b0509146103be578063a5041040146103eb57600080fd5b80633360aa3c116101085780633360aa3c1461023857806339e899ee1461029757806342fb01a8146102b75780634d36d085146102d757806366967cbb146102f7578063715018a61461033457600080fd5b806301ffc9a71461014557806310118ebb1461017a578063150b7a02146101b2578063320a18dc146101f6578063331758e614610218575b600080fd5b34801561015157600080fd5b50610165610160366004614741565b610505565b60405190151581526020015b60405180910390f35b34801561018657600080fd5b5060045461019a906001600160a01b031681565b6040516001600160a01b039091168152602001610171565b3480156101be57600080fd5b506101dd6101cd36600461488d565b630a85bd0160e11b949350505050565b6040516001600160e01b03199091168152602001610171565b34801561020257600080fd5b50610216610211366004614905565b61053c565b005b34801561022457600080fd5b5060035461019a906001600160a01b031681565b34801561024457600080fd5b50610278610253366004614946565b600560205260009081526040902080546001909101546001600160a01b039091169082565b604080516001600160a01b039093168352602083019190915201610171565b3480156102a357600080fd5b506102166102b2366004614963565b61066a565b3480156102c357600080fd5b506102166102d236600461498e565b6106b6565b6102ea6102e5366004614a2c565b6107b0565b6040516101719190614b13565b34801561030357600080fd5b50610327610312366004614963565b60066020526000908152604090205460ff1681565b6040516101719190614b21565b34801561034057600080fd5b50610216610e59565b34801561035557600080fd5b50610165610364366004614b45565b610e8f565b34801561037557600080fd5b506001546001600160a01b031661019a565b610216610395366004614bad565b610f47565b3480156103a657600080fd5b506103b060025481565b604051908152602001610171565b3480156103ca57600080fd5b506103de6103d9366004614c08565b610fbe565b6040516101719190614c60565b3480156103f757600080fd5b50610216610406366004614905565b6110e1565b34801561041757600080fd5b506101dd610426366004614d1c565b63bc197c8160e01b95945050505050565b34801561044357600080fd5b50610457610452366004614c08565b6110f3565b6040516101719190614ef4565b34801561047057600080fd5b5061048461047f366004614fc1565b611465565b60408051928352602083019190915201610171565b3480156104a557600080fd5b506101dd6104b436600461501c565b63f23a6e6160e01b95945050505050565b3480156104d157600080fd5b506102166104e0366004614963565b61163c565b3480156104f157600080fd5b50610216610500366004614c08565b6116d7565b60006001600160e01b03198216630271189760e51b148061053657506301ffc9a760e01b6001600160e01b03198316145b92915050565b6001546001600160a01b0316331461056f5760405162461bcd60e51b815260040161056690615084565b60405180910390fd5b6001600160a01b0382166105b75760405162461bcd60e51b815260206004820152600f60248201526e4e6f207a65726f206164647265737360881b6044820152606401610566565b6040518060400160405280836001600160a01b0316815260200182815250600560008560078111156105eb576105eb614aa5565b60078111156105fc576105fc614aa5565b815260208082019290925260409081016000908120845181546001600160a01b0319166001600160a01b03918216178255948401516001918201559386168152600690925290208054859260ff199091169083600781111561066057610660614aa5565b0217905550505050565b6001546001600160a01b031633146106945760405162461bcd60e51b815260040161056690615084565b600480546001600160a01b0319166001600160a01b0392909216919091179055565b6000806106c4868686611707565b915091506106d28585611a26565b6106db57600080fd5b6106ec85853330600360f81b611d5a565b5060006106fe86866001600080611465565b50905061070c868686612145565b610718575050506107aa565b6107258787858886612789565b6001600160a01b0386811660008181526007602090815260408083208a84529091529081902080546001820154600483015460089093015493518b96610100909304831695947fabb50c1815800da62a4637d3272d1584df1ee8cbd963d90fb44eb55b63acfcb59461079e941691899160f01b906150b9565b60405180910390a45050505b50505050565b6040805160a0810182526000606082018181526080830182905282526020820181905291810182905290546001190161082b5760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610566565b600260005561083a858361288d565b6108785760405162461bcd60e51b815260206004820152600f60248201526e15dc985c0818da1958dac819985a5b608a1b6044820152606401610566565b60016108876020870187614946565b600781111561089857610898614aa5565b141580156108c4575060006108b06020870187614946565b60078111156108c1576108c1614aa5565b14155b15610934576108e26108db368790038701876150e7565b3330612bf0565b6108eb8661324d565b146109345760405162461bcd60e51b81526020600482015260196024820152780537573706963696f757320617373657420666f72207772617603c1b6044820152606401610566565b60016005600061094c61012089016101008a01614946565b600781111561095d5761095d614aa5565b600781111561096e5761096e614aa5565b8152602001908152602001600020600101600082825461098e9190615191565b90915550610a5490506109a961012087016101008801614946565b600560006109bf6101208a016101008b01614946565b60078111156109d0576109d0614aa5565b60078111156109e1576109e1614aa5565b815260208101919091526040016000908120546001600160a01b0316908590600590610a156101208c016101008d01614946565b6007811115610a2657610a26614aa5565b6007811115610a3757610a37614aa5565b815260200190815260200160002060010154896101200135613284565b610afb60056000610a6d61012089016101008a01614946565b6007811115610a7e57610a7e614aa5565b6007811115610a8f57610a8f614aa5565b815260208101919091526040016000908120546001600160a01b031690600590610ac16101208a016101008b01614946565b6007811115610ad257610ad2614aa5565b6007811115610ae357610ae3614aa5565b81526020019081526020016000206001015487613327565b610ba360056000610b1461012089016101008a01614946565b6007811115610b2557610b25614aa5565b6007811115610b3657610b36614aa5565b815260208101919091526040016000908120546001600160a01b031690600590610b686101208a016101008b01614946565b6007811115610b7957610b79614aa5565b6007811115610b8a57610b8a614aa5565b8152602001908152602001600020600101548686610f47565b610c5060056000610bbc61012089016101008a01614946565b6007811115610bcd57610bcd614aa5565b6007811115610bde57610bde614aa5565b815260208101919091526040016000908120546001600160a01b031690600590610c106101208a016101008b01614946565b6007811115610c2157610c21614aa5565b6007811115610c3257610c32614aa5565b8152602001908152602001600020600101543330600260f81b611d5a565b50604085013560056000610c6c61012089016101008a01614946565b6007811115610c7d57610c7d614aa5565b6007811115610c8e57610c8e614aa5565b81526020808201929092526040908101600020546001600160a01b031691610cba918901908901614963565b6001600160a01b03167fa90a3b8dae41ae10a708d32fec7bf12da5c90879c98b9c4cca3c8fba91ddf49360056000610cfa6101208c016101008d01614946565b6007811115610d0b57610d0b614aa5565b6007811115610d1c57610d1c614aa5565b81526020810191909152604001600020600101548634610d446101608d016101408e016151a9565b604051610d5494939291906150b9565b60405180910390a46040805160a08101909152806060810180610d7f6101208a016101008b01614946565b6007811115610d9057610d90614aa5565b815260200160056000610dab6101208c016101008d01614946565b6007811115610dbc57610dbc614aa5565b6007811115610dcd57610dcd614aa5565b81526020808201929092526040016000908120546001600160a01b0316909252918352910190600590610e086101208a016101008b01614946565b6007811115610e1957610e19614aa5565b6007811115610e2a57610e2a614aa5565b815260200190815260200160002060010154815260200186610120013581525090506001600055949350505050565b6001546001600160a01b03163314610e835760405162461bcd60e51b815260040161056690615084565b610e8d60006135b2565b565b6000336001600160a01b0387161480610ea757503330145b610ef35760405162461bcd60e51b815260206004820152601860248201527f4f6e6c7920666f7220774e4654206f72207772617070657200000000000000006044820152606401610566565b610f008686868686611d5a565b610f3e5760405162461bcd60e51b815260206004820152600f60248201526e1199594818da185c99d94819985a5b608a1b6044820152606401610566565b95945050505050565b80151580610f555750600034115b156107aa57610f6684848484613604565b610fb25760405162461bcd60e51b815260206004820152601860248201527f466f7262696464656e2061646420636f6c6c61746572616c00000000000000006044820152606401610566565b6107aa848484846137e0565b60606000610fcc84846110f3565b51905060038151516007811115610fe557610fe5614aa5565b036110715780516020908101519082015160405163c87b56dd60e01b81526001600160a01b039092169163c87b56dd916110259160040190815260200190565b600060405180830381865afa158015611042573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261106a91908101906151d3565b91506110da565b6004815151600781111561108757611087614aa5565b036110c7578051602090810151908201516040516303a24d0760e21b81526001600160a01b0390921691630e89341c916110259160040190815260200190565b6040518060200160405280600081525091505b5092915050565b6110ee83838360006106b6565b505050565b604080516101808101825260006101408201818152610160830182905260e083019081526101008301829052610120830182905282526060602083018190529282018190528282018390526080820183905260a082019290925260c08101919091526001600160a01b0383166000908152600760208181526040808420868552909152918290208251610180810190935280549091839160e0830191849183916101408601918491839160ff16908111156111b0576111b0614aa5565b60078111156111c1576111c1614aa5565b8152905461010090046001600160a01b031660209182015290825260018301548282015260029092015460409182015291835260038401805483518184028101840190945280845293820193909160009084015b828210156112ac576000848152602090206040805160a08101909152600384029091018054829060608201908390829060ff16600781111561125957611259614aa5565b600781111561126a5761126a614aa5565b8152905461010090046001600160a01b031660209182015290825260018381015483830152600290930154604090920191909152918352929092019101611215565b5050509082525060048201546001600160a01b0316602080830191909152600583018054604080518285028101850182528281529401939260009084015b8282101561134c5760008481526020908190206040805160608101825260038602909201805460f81b6001600160f81b0319168352600180820154848601526002909101546001600160a01b03169183019190915290835290920191016112ea565b50505050815260200160068201805480602002602001604051908101604052809291908181526020016000905b828210156113c55760008481526020908190206040805180820190915260028502909101805460f81b6001600160f81b0319168252600190810154828401529083529092019101611379565b50505050815260200160078201805480602002602001604051908101604052809291908181526020016000905b8282101561143c57600084815260209081902060408051808201909152908401546001600160a01b0381168252600160a01b900461ffff16818301528252600190920191016113f2565b505050908252506008919091015460f01b6001600160f01b031916602090910152905092915050565b60008060005b6001600160a01b03881660009081526007602090815260408083208a8452909152902060030154811015611630576001600160a01b0388811660009081526007602090815260408083208b84529091529020600301805491871691839081106114d6576114d6615240565b600091825260209091206003909102015461010090046001600160a01b031614801561154c57506001600160a01b03881660009081526007602090815260408083208a8452909152902060030180548591908390811061153857611538615240565b906000526020600020906003020160010154145b80156115c4575085600781111561156557611565614aa5565b6001600160a01b03891660009081526007602090815260408083208b8452909152902060030180548390811061159d5761159d615240565b600091825260209091206003909102015460ff1660078111156115c2576115c2614aa5565b145b1561161e576001600160a01b03881660009081526007602090815260408083208a8452909152902060030180548290811061160157611601615240565b906000526020600020906003020160020154819250925050611632565b8061162881615256565b91505061146b565b505b9550959350505050565b6001546001600160a01b031633146116665760405162461bcd60e51b815260040161056690615084565b6001600160a01b0381166116cb5760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b6064820152608401610566565b6116d4816135b2565b50565b6001600160a01b0382166000908152600660205260408120546117039160ff90911690849084906106b6565b5050565b600080611734600160f01b61171c86866110f3565b60c0015181166001600160f01b031990811691161490565b156117815760405162461bcd60e51b815260206004820152601b60248201527f556e577261707020666f7262696464656e20627920617574686f7200000000006044820152606401610566565b600385600781111561179557611795614aa5565b03611862576040516331a9108f60e11b8152600481018490526001600160a01b03851690636352211e90602401602060405180830381865afa1580156117df573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611803919061526f565b91506001600160a01b038216331461185d5760405162461bcd60e51b815260206004820152601860248201527f4f6e6c79206f776e65722063616e20756e7772617020697400000000000000006044820152606401610566565b611a1e565b600485600781111561187657611876614aa5565b036119bf5760405163bd85b03960e01b8152600481018490526001600160a01b0385169063bd85b03990602401602060405180830381865afa1580156118c0573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118e4919061528c565b604051627eeac760e11b815233600482018190526024820186905293509091506001600160a01b0385169062fdd58e90604401602060405180830381865afa158015611934573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611958919061528c565b811461185d5760405162461bcd60e51b815260206004820152603060248201527f45524331313520756e7772617020617661696c61626c65206f6e6c7920666f7260448201526f20616c6c20746f74616c537570706c7960801b6064820152608401610566565b6040805160a081019091528060608101808860078111156119e2576119e2614aa5565b8152602001876001600160a01b03168152508152602001848152602001600081525060405163391102fb60e01b81526004016105669190614b13565b935093915050565b6000805b6001600160a01b0384166000908152600760209081526040808320868452909152902060060154811015611d50576001600160a01b03841660009081526007602090815260408083208684529091529020600601805482908110611a9057611a90615240565b600091825260208220600291909102015460f81b6001600160f81b0319169003611b40576001600160a01b03841660009081526007602090815260408083208684529091529020600601805442919083908110611aef57611aef615240565b9060005260206000209060020201600101541115611b405760405162461bcd60e51b815260206004820152600e60248201526d2a34b6b2a637b1b59032b93937b960911b6044820152606401610566565b6001600160a01b03841660009081526007602090815260408083208684529091529020600601805482908110611b7857611b78615240565b600091825260209091206002909102015460f81b6001600160f81b031916600160f81b03611d3e5760005b6001600160a01b0385166000908152600760209081526040808320878452909152902060050154811015611d3c576001600160a01b03851660009081526007602090815260408083208784529091529020600501805482908110611c0957611c09615240565b600091825260208220600391909102015460f81b6001600160f81b0319169003611d2a576001600160a01b038516600090815260076020908152604080832087845290915281206005018054611c92918891889160029187908110611c7057611c70615240565b600091825260208220600260039092020101546001600160a01b031690611465565b506001600160a01b038716600090815260076020908152604080832089845290915290206006018054919250829185908110611cd057611cd0615240565b9060005260206000209060020201600101541115611d285760405162461bcd60e51b81526020600482015260156024820152742a3930b739b332b92332b2a637b1b59032b93937b960591b6044820152606401610566565b505b80611d3481615256565b915050611ba3565b505b80611d4881615256565b915050611a2a565b5060019392505050565b60006001600160f81b031982168103610f3e5760005b6001600160a01b038716600090815260076020908152604080832089845290915290206005015481101561213b576001600160a01b03871660009081526007602090815260408083208984529091529020600501805482908110611dd657611dd6615240565b600091825260208220600391909102015460f81b6001600160f81b0319169003612129576003546004546001600160a01b03918216911615611edc576004546001600160a01b0389811660009081526007602090815260408083208c84529091529020600501805491909216916373cf00f69185908110611e5957611e59615240565b600091825260209091206003909102016002015460405160e083901b6001600160e01b03191681526001600160a01b039091166004820152602401608060405180830381865afa158015611eb1573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611ed591906152a5565b6060015190505b6000806000836001600160a01b031663ce244ce1600760008e6001600160a01b03166001600160a01b0316815260200190815260200160002060008d81526020019081526020016000206005018781548110611f3a57611f3a615240565b9060005260206000209060030201600760008f6001600160a01b03166001600160a01b0316815260200190815260200160002060008e81526020019081526020016000206007018c8c6040518563ffffffff1660e01b8152600401611fa29493929190615329565b600060405180830381865afa158015611fbf573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052611fe79190810190615441565b919450925090506000805b825181101561212257306001600160a01b031683828151811061201757612017615240565b60200260200101516001600160a01b031603612051576120518d8d87848151811061204457612044615240565b6020026020010151613bb3565b6120a785828151811061206657612066615240565b602002602001015185838151811061208057612080615240565b602002602001015185848151811061209a5761209a615240565b6020026020010151612bf0565b91508b8d6001600160a01b03168483815181106120c6576120c6615240565b60200260200101516001600160a01b03167f2e7d475f6480b44a6b26e0a71fd4bf4fbf7842b803ad204a4d23042a02bd73358560405161210891815260200190565b60405180910390a48061211a81615256565b915050611ff2565b5050505050505b8061213381615256565b915050611d70565b5060019050610f3e565b6001600160a01b03838116600090815260076020908152604080832086845290915281206004015490918291339116156121a457506001600160a01b038086166000908152600760209081526040808320888452909152902060040154165b60005b6001600160a01b03871660009081526007602090815260408083208984529091529020600301548110156125b1576001600160a01b0387166000908152600760209081526040808320898452909152812060030180548390811061220d5761220d615240565b600091825260209091206003909102015460ff16600781111561223257612232614aa5565b14612507578415612305576001600160a01b0387166000908152600760209081526040808320898452909152902060030180546122fe91908390811061227a5761227a615240565b600091825260209091206040805160a081019091526003909202018054829060608201908390829060ff1660078111156122b6576122b6614aa5565b60078111156122c7576122c7614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523084613e04565b92506123c9565b6001600160a01b0387166000908152600760209081526040808320898452909152902060030180546123c691908390811061234257612342615240565b600091825260209091206040805160a081019091526003909202018054829060608201908390829060ff16600781111561237e5761237e614aa5565b600781111561238f5761238f614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523084612bf0565b92505b6001600160a01b0387166000908152600760209081526040808320898452909152902060030180548290811061240157612401615240565b906000526020600020906003020160020154831461249f576001600160a01b0387166000908152600760209081526040808320898452909152902060030180548290811061245157612451615240565b600091825260208220600390910201546040516001600160a01b0361010090920482169289928b16917ffca203c3f6987c2a1dae80f773c277d67920e7bce0cea9c07cd0eb8142e985ca9190a45b6001600160a01b038716600090815260076020908152604080832089845290915281206003018054839081106124d7576124d7615240565b60009182526020909120600390910201805460ff1916600183600781111561250157612501614aa5565b02179055505b6103e85a1115801561254a57506001600160a01b03871660009081526007602090815260408083208984529091529020600301546125479060019061558a565b81105b1561259f5785876001600160a01b03167fd66d44264f9d44e254da71183ff08098f38da4675285592ee80cdbd3b6f5153e8360405161258b91815260200190565b60405180910390a360009350505050612782565b806125a981615256565b9150506121a7565b5060016001600160a01b03871660009081526007602081815260408084208a85529091529091205460ff16908111156125ec576125ec614aa5565b1415801561262f57506001600160a01b038616600090815260076020818152604080842089855290915282205460ff169081111561262c5761262c614aa5565b14155b1561277b57836126dc576001600160a01b038616600090815260076020818152604080842089855290915291829020825160a0810190935280546126d69392839160608301918491839160ff9091169081111561268e5761268e614aa5565b600781111561269f5761269f614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523083612bf0565b5061277b565b6001600160a01b038616600090815260076020818152604080842089855290915291829020825160a0810190935280546127799392839160608301918491839160ff9091169081111561273157612731614aa5565b600781111561274257612742614aa5565b8152905461010090046001600160a01b03166020918201529082526001830154908201526002909101546040909101523083613e04565b505b6001925050505b9392505050565b600385600781111561279d5761279d614aa5565b0361280257604051630852cd8d60e31b8152600481018390526001600160a01b038516906342966c68906024015b600060405180830381600087803b1580156127e557600080fd5b505af11580156127f9573d6000803e3d6000fd5b50505050612886565b600485600781111561281657612816614aa5565b0361288657604051637a94c56560e11b81526001600160a01b038481166004830152602482018490526044820183905285169063f5298aca906064015b600060405180830381600087803b15801561286d57600080fd5b505af1158015612881573d6000803e3d6000fd5b505050505b5050505050565b60006128b4600160f11b61171c6128aa6040870160208801614963565b60408701356110f3565b1580156128ca57506001600160a01b0382163014155b6004549091506001600160a01b031615610536576004546001600160a01b0316638f8b138e6128ff6040860160208701614963565b6040516001600160e01b031960e084901b1681526001600160a01b039091166004820152602401602060405180830381865afa158015612943573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061296791906155a1565b156129b45760405162461bcd60e51b815260206004820152601a60248201527f574c3a41737365742064697361626c656420666f7220777261700000000000006044820152606401610566565b6004546001600160a01b03166352cdc6a66129d56040860160208701614963565b6129e7610160870161014088016151a9565b6040516001600160e01b031960e085901b1681526001600160a01b0390921660048301526001600160f01b0319166024820152604401602060405180830381865afa158015612a3a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612a5e91906155a1565b612abc5760405162461bcd60e51b815260206004820152602960248201527f574c3a536f6d652072756c6573206172652064697361626c656420666f7220746044820152681a1a5cc8185cdcd95d60ba1b6064820152608401610566565b60005b612acc60a08501856155be565b90508110156110da576004546001600160a01b031663b6e306ac612af360a08701876155be565b84818110612b0357612b03615240565b9050606002016040016020810190612b1b9190614963565b6040516001600160e01b031960e084901b1681526001600160a01b039091166004820152602401602060405180830381865afa158015612b5f573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612b8391906155a1565b612bde5760405162461bcd60e51b815260206004820152602660248201527f574c3a536f6d652061737365747320617265206e6f7420656e61626c656420666044820152656f722066656560d01b6064820152608401610566565b80612be881615256565b915050612abf565b60008060018551516007811115612c0957612c09614aa5565b03612cc3575060408085015190516001600160a01b0384168031926000928381818185875af1925050503d8060008114612c5f576040519150601f19603f3d011682016040523d82523d6000602084013e612c64565b606091505b5050905080612ca75760405162461bcd60e51b815260206004820152600f60248201526e1d1c985b9cd9995c8819985a5b1959608a1b6044820152606401610566565b612cbb826001600160a01b0386163161558a565b925050613245565b60028551516007811115612cd957612cd9614aa5565b03612e2f578451602001516040516370a0823160e01b81526001600160a01b038581166004830152909116906370a0823190602401602060405180830381865afa158015612d2b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612d4f919061528c565b9050306001600160a01b03851603612d88576040850151855160200151612d83916001600160a01b039091169085906141d5565b612dac565b6040850151855160200151612dac916001600160a01b039091169086908690614238565b8451602001516040516370a0823160e01b81526001600160a01b038581166004830152839216906370a08231906024015b602060405180830381865afa158015612dfa573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612e1e919061528c565b612e28919061558a565b9150613245565b60038551516007811115612e4557612e45614aa5565b148015612ed157508451602090810151908601516040516331a9108f60e11b815260048101919091526001600160a01b03868116921690636352211e90602401602060405180830381865afa158015612ea2573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612ec6919061526f565b6001600160a01b0316145b156130db578451602001516040516370a0823160e01b81526001600160a01b038581166004830152909116906370a0823190602401602060405180830381865afa158015612f23573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612f47919061528c565b90508460000151602001516001600160a01b03166323b872dd858588602001516040518463ffffffff1660e01b8152600401612f8593929190615606565b600060405180830381600087803b158015612f9f57600080fd5b505af1158015612fb3573d6000803e3d6000fd5b50505050826001600160a01b03168560000151602001516001600160a01b0316636352211e87602001516040518263ffffffff1660e01b8152600401612ffb91815260200190565b602060405180830381865afa158015613018573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061303c919061526f565b6001600160a01b03161480156130cc57508451602001516040516370a0823160e01b81526001600160a01b038581166004830152839216906370a0823190602401602060405180830381865afa15801561309a573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906130be919061528c565b6130c8919061558a565b6001145b156130d657600191505b613245565b600485515160078111156130f1576130f1614aa5565b0361322a57845160209081015190860151604051627eeac760e11b81526001600160a01b038681166004830152602482019290925291169062fdd58e90604401602060405180830381865afa15801561314e573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613172919061528c565b90508460000151602001516001600160a01b031663f242432a8585886020015189604001516040518563ffffffff1660e01b81526004016131b6949392919061562a565b600060405180830381600087803b1580156131d057600080fd5b505af11580156131e4573d6000803e3d6000fd5b5050865160209081015190880151604051627eeac760e11b81526001600160a01b03888116600483015260248201929092528594509116915062fdd58e90604401612ddd565b8460405163391102fb60e01b81526004016105669190614b13565b509392505050565b6000600361325e6020840184614946565b600781111561326f5761326f614aa5565b0361327c57506001919050565b506060013590565b600385600781111561329857613298614aa5565b036132d2576040516340c10f1960e01b81526001600160a01b038481166004830152602482018490528516906340c10f19906044016127cb565b60048560078111156132e6576132e6614aa5565b0361288657604051630ab714fb60e11b81526001600160a01b038481166004830152602482018490526044820183905285169063156e29f690606401612853565b6001600160a01b0383166000908152600760209081526040808320858452909152902081906133568282615662565b50506001600160a01b0383166000908152600760209081526040808320858452909152902060040180546001600160a01b031916905561339e610160820161014083016151a9565b6001600160a01b03841660009081526007602090815260408083208684529091528120600801805461ffff191660f09390931c929092179091555b6133e660a08301836155be565b9050811015613475576001600160a01b0384166000908152600760209081526040808320868452909152902060050161342260a08401846155be565b8381811061343257613432615240565b8354600181018555600094855260209094206060909102929092019260030290910190506134608282615705565b5050808061346d90615256565b9150506133d9565b5060005b61348660c0830183615744565b9050811015613515576001600160a01b038416600090815260076020908152604080832086845290915290206006016134c260c0840184615744565b838181106134d2576134d2615240565b835460018101855560009485526020909420604090910292909201926002029091019050613500828261578d565b5050808061350d90615256565b915050613479565b5060005b61352660e0830183615744565b90508110156107aa576001600160a01b03841660009081526007602081815260408084208785529091529091200161356160e0840184615744565b8381811061357157613571615240565b8354600181018555600094855260209094206040909102929092019291909101905061359d82826157b5565b505080806135aa90615256565b915050613519565b600180546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b600060036001600160a01b03861660009081526006602052604090205460ff16600781111561363557613635614aa5565b036136e757604051634f558e7960e01b8152600481018590526001600160a01b03861690634f558e79906024015b602060405180830381865afa158015613680573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906136a491906155a1565b6136e25760405162461bcd60e51b815260206004820152600f60248201526e774e4654206e6f742065786973747360881b6044820152606401610566565b6137c4565b60046001600160a01b03861660009081526006602052604090205460ff16600781111561371657613716614aa5565b0361374857604051634f558e7960e01b8152600481018590526001600160a01b03861690634f558e7990602401613663565b6040805160a0810182526001600160a01b03871660009081526006602052919091205481906060820190819060ff16600781111561378857613788614aa5565b8152602001886001600160a01b03168152508152602001858152602001600081525060405163391102fb60e01b81526004016105669190614b13565b6137d6600160f31b61171c87876110f3565b1595945050505050565b3415613874576040805160a0810190915261381f9085908590806060810180600181526000602091820181905291835282015234604090910152613bb3565b604080516001815260006020820181905281830152346060820152905184916001600160a01b038716917ff3d1350815c4f9db2be36c35f840bfb002835a83ff1c3d8f3a217b1e6227d5aa9181900360800190a35b60005b8181101561288657600183838381811061389357613893615240565b6138a99260206080909202019081019150614946565b60078111156138ba576138ba614aa5565b14613ba1576004546001600160a01b0316156139d6576004546001600160a01b031663eb9ae17c8484848181106138f3576138f3615240565b61390c9260406080909202019081019150602001614963565b6040516001600160e01b031960e084901b1681526001600160a01b039091166004820152602401602060405180830381865afa158015613950573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061397491906155a1565b6139d65760405162461bcd60e51b815260206004820152602d60248201527f574c3a536f6d652061737365747320617265206e6f7420656e61626c6564206660448201526c1bdc8818dbdb1b185d195c985b609a1b6064820152608401610566565b613a018383838181106139eb576139eb615240565b9050608002018036038101906108db91906150e7565b613a21848484818110613a1657613a16615240565b90506080020161324d565b14613a6a5760405162461bcd60e51b81526020600482015260196024820152780537573706963696f757320617373657420666f72207772617603c1b6044820152606401610566565b613a9c8585858585818110613a8157613a81615240565b905060800201803603810190613a9791906150e7565b613bb3565b83856001600160a01b03167ff3d1350815c4f9db2be36c35f840bfb002835a83ff1c3d8f3a217b1e6227d5aa858585818110613ada57613ada615240565b613af09260206080909202019081019150614946565b6007811115613b0157613b01614aa5565b868686818110613b1357613b13615240565b613b2c9260406080909202019081019150602001614963565b878787818110613b3e57613b3e615240565b90506080020160400135888888818110613b5a57613b5a615240565b90506080020160600135604051613b98949392919060ff9490941684526001600160a01b039290921660208401526040830152606082015260800190565b60405180910390a35b80613bab81615256565b915050613877565b60028151516007811115613bc957613bc9614aa5565b1480613be8575060018151516007811115613be657613be6614aa5565b145b15613c3657602081015115613c365760405162461bcd60e51b8152602060048201526014602482015273546f6b656e4964206d757374206265207a65726f60601b6044820152606401610566565b60038151516007811115613c4c57613c4c614aa5565b03613c9957604081015115613c995760405162461bcd60e51b8152602060048201526013602482015272416d6f756e74206d757374206265207a65726f60681b6044820152606401610566565b6001600160a01b03831660009081526007602090815260408083208584529091529020600301541580613cdf575060038151516007811115613cdd57613cdd614aa5565b145b15613cef576110ee838383614259565b600080613d1385858560000151600001518660000151602001518760200151611465565b915091506000811180613d8c575080158015613d8c575082516020908101516001600160a01b0387811660009081526007845260408082208983529094529283206003018054919092169290613d6b57613d6b615240565b600091825260209091206003909102015461010090046001600160a01b0316145b15613df9576040808401516001600160a01b038716600090815260076020908152838220888352905291909120600301805483908110613dce57613dce615240565b90600052602060002090600302016002016000828254613dee9190615191565b909155506128869050565b612886858585614259565b60008060018551516007811115613e1d57613e1d614aa5565b03613e92575060408085015190516001600160a01b0384168031926000928381818185875af1925050503d8060008114613e73576040519150601f19603f3d011682016040523d82523d6000602084013e613e78565b606091505b5050905081846001600160a01b031631612cbb919061558a565b60028551516007811115613ea857613ea8614aa5565b0361403257306001600160a01b03851603613f7d5760008560000151602001516001600160a01b0316848760400151604051602401613efc9291906001600160a01b03929092168252602082015260400190565b60408051601f198184030181529181526020820180516001600160e01b031663a9059cbb60e01b17905251613f3191906157fd565b6000604051808303816000865af19150503d8060008114613f6e576040519150601f19603f3d011682016040523d82523d6000602084013e613f73565b606091505b5050905050614026565b60008560000151602001516001600160a01b031685858860400151604051602401613faa93929190615606565b60408051601f198184030181529181526020820180516001600160e01b03166323b872dd60e01b17905251613fdf91906157fd565b6000604051808303816000865af19150503d806000811461401c576040519150601f19603f3d011682016040523d82523d6000602084013e614021565b606091505b505050505b84604001519150613245565b6003855151600781111561404857614048614aa5565b036140ff5760008560000151602001516001600160a01b03168585886020015160405160240161407a93929190615606565b60408051601f198184030181529181526020820180516001600160e01b03166323b872dd60e01b179052516140af91906157fd565b6000604051808303816000865af19150503d80600081146140ec576040519150601f19603f3d011682016040523d82523d6000602084013e6140f1565b606091505b505090506001925050613245565b6004855151600781111561411557614115614aa5565b0361322a5760008560000151602001516001600160a01b031685858860200151896040015160405160240161414d949392919061562a565b60408051601f198184030181529181526020820180516001600160e01b0316637921219560e11b1790525161418291906157fd565b6000604051808303816000865af19150503d80600081146141bf576040519150601f19603f3d011682016040523d82523d6000602084013e6141c4565b606091505b505090508560400151925050613245565b6040516001600160a01b0383166024820152604481018290526110ee90849063a9059cbb60e01b906064015b60408051601f198184030181529190526020810180516001600160e01b03166001600160e01b0319909316929092179091526144f7565b6107aa846323b872dd60e01b85858560405160240161420193929190615606565b6002546001600160a01b0384166000908152600760209081526040808320868452909152902060030154106142d05760405162461bcd60e51b815260206004820152601d60248201527f546f6f206d75636820746f6b656e7320696e20636f6c6c61746572616c0000006044820152606401610566565b60005b6001600160a01b038416600090815260076020908152604080832086845290915290206006015481101561444d576001600160a01b0384166000908152600760209081526040808320868452909152902060060180548290811061433957614339615240565b600091825260209091206002909102015460f81b6001600160f81b031916600160f91b0361443b576001600160a01b0384166000908152600760209081526040808320868452909152902060030154614393906001615191565b6001600160a01b038516600090815260076020908152604080832087845290915290206006018054839081106143cb576143cb615240565b906000526020600020906002020160010154101561443b5760405162461bcd60e51b815260206004820152602760248201527f546f6f206d75636820636f6c6c61746572616c20736c6f747320666f722074686044820152661a5cc81dd3919560ca1b6064820152608401610566565b8061444581615256565b9150506142d3565b506001600160a01b038316600090815260076020818152604080842086855282528320600390810180546001818101835591865292909420855180519390920201805486959194929385939092849260ff19169184908111156144b2576144b2614aa5565b021790555060209182015181546001600160a01b0390911661010002610100600160a81b03199091161790558201516001820155604090910151600290910155505050565b600061454c826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166145c99092919063ffffffff16565b8051909150156110ee578080602001905181019061456a91906155a1565b6110ee5760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e6044820152691bdd081cdd58d8d9595960b21b6064820152608401610566565b60606145d884846000856145e0565b949350505050565b6060824710156146415760405162461bcd60e51b815260206004820152602660248201527f416464726573733a20696e73756666696369656e742062616c616e636520666f6044820152651c8818d85b1b60d21b6064820152608401610566565b843b61468f5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606401610566565b600080866001600160a01b031685876040516146ab91906157fd565b60006040518083038185875af1925050503d80600081146146e8576040519150601f19603f3d011682016040523d82523d6000602084013e6146ed565b606091505b50915091506146fd828286614708565b979650505050505050565b60608315614717575081612782565b8251156147275782518084602001fd5b8160405162461bcd60e51b81526004016105669190614c60565b60006020828403121561475357600080fd5b81356001600160e01b03198116811461278257600080fd5b6001600160a01b03811681146116d457600080fd5b634e487b7160e01b600052604160045260246000fd5b604080519081016001600160401b03811182821017156147b8576147b8614780565b60405290565b604051606081016001600160401b03811182821017156147b8576147b8614780565b604051601f8201601f191681016001600160401b038111828210171561480857614808614780565b604052919050565b60006001600160401b0382111561482957614829614780565b50601f01601f191660200190565b600082601f83011261484857600080fd5b813561485b61485682614810565b6147e0565b81815284602083860101111561487057600080fd5b816020850160208301376000918101602001919091529392505050565b600080600080608085870312156148a357600080fd5b84356148ae8161476b565b935060208501356148be8161476b565b92506040850135915060608501356001600160401b038111156148e057600080fd5b6148ec87828801614837565b91505092959194509250565b600881106116d457600080fd5b60008060006060848603121561491a57600080fd5b8335614925816148f8565b925060208401356149358161476b565b929592945050506040919091013590565b60006020828403121561495857600080fd5b8135612782816148f8565b60006020828403121561497557600080fd5b81356127828161476b565b80151581146116d457600080fd5b600080600080608085870312156149a457600080fd5b84356149af816148f8565b935060208501356149bf8161476b565b92506040850135915060608501356149d681614980565b939692955090935050565b60008083601f8401126149f357600080fd5b5081356001600160401b03811115614a0a57600080fd5b6020830191508360208260071b8501011115614a2557600080fd5b9250929050565b60008060008060608587031215614a4257600080fd5b84356001600160401b0380821115614a5957600080fd5b908601906101608289031215614a6e57600080fd5b90945060208601359080821115614a8457600080fd5b50614a91878288016149e1565b90945092505060408501356149d68161476b565b634e487b7160e01b600052602160045260246000fd5b60088110614ad957634e487b7160e01b600052602160045260246000fd5b9052565b8051614aea838251614abb565b6020908101516001600160a01b0316838201528101516040808401919091520151606090910152565b608081016105368284614add565b602081016105368284614abb565b6001600160f81b0319811681146116d457600080fd5b600080600080600060a08688031215614b5d57600080fd5b8535614b688161476b565b9450602086013593506040860135614b7f8161476b565b92506060860135614b8f8161476b565b91506080860135614b9f81614b2f565b809150509295509295909350565b60008060008060608587031215614bc357600080fd5b8435614bce8161476b565b93506020850135925060408501356001600160401b03811115614bf057600080fd5b614bfc878288016149e1565b95989497509550505050565b60008060408385031215614c1b57600080fd5b8235614c268161476b565b946020939093013593505050565b60005b83811015614c4f578181015183820152602001614c37565b838111156107aa5750506000910152565b6020815260008251806020840152614c7f816040850160208701614c34565b601f01601f19169190910160400192915050565b60006001600160401b03821115614cac57614cac614780565b5060051b60200190565b600082601f830112614cc757600080fd5b81356020614cd761485683614c93565b82815260059290921b84018101918181019086841115614cf657600080fd5b8286015b84811015614d115780358352918301918301614cfa565b509695505050505050565b600080600080600060a08688031215614d3457600080fd5b8535614d3f8161476b565b94506020860135614d4f8161476b565b935060408601356001600160401b0380821115614d6b57600080fd5b614d7789838a01614cb6565b94506060880135915080821115614d8d57600080fd5b614d9989838a01614cb6565b93506080880135915080821115614daf57600080fd5b50614dbc88828901614837565b9150509295509295909350565b600081518084526020808501945080840160005b83811015614e0357614df0878351614add565b6080969096019590820190600101614ddd565b509495945050505050565b600081518084526020808501945080840160005b83811015614e0357815180516001600160f81b031916885283810151848901526040908101516001600160a01b03169088015260609096019590820190600101614e22565b600081518084526020808501945080840160005b83811015614e0357815180516001600160f81b03191688528301518388015260409096019590820190600101614e7b565b600081518084526020808501945080840160005b83811015614e0357815180516001600160a01b0316885283015161ffff168388015260409096019590820190600101614ec0565b60208152614f06602082018351614add565b600060208301516101408060a0850152614f24610160850183614dc9565b91506040850151614f4060c08601826001600160a01b03169052565b506060850151601f19808685030160e0870152614f5d8483614e0e565b9350608087015191508086850301610100870152614f7b8483614e67565b935060a08701519150808685030161012087015250614f9a8382614eac565b92505060c0850151614fb7828601826001600160f01b0319169052565b5090949350505050565b600080600080600060a08688031215614fd957600080fd5b8535614fe48161476b565b9450602086013593506040860135614ffb816148f8565b9250606086013561500b8161476b565b949793965091946080013592915050565b600080600080600060a0868803121561503457600080fd5b853561503f8161476b565b9450602086013561504f8161476b565b9350604086013592506060860135915060808601356001600160401b0381111561507857600080fd5b614dbc88828901614837565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b9384526001600160a01b0392909216602084015260408301526001600160f01b031916606082015260800190565b600081830360808112156150fa57600080fd5b604051606081018181106001600160401b038211171561511c5761511c614780565b604090815282121561512d57600080fd5b615135614796565b91508335615142816148f8565b825260208401356151528161476b565b806020840152508181526040840135602082015260608401356040820152809250505092915050565b634e487b7160e01b600052601160045260246000fd5b600082198211156151a4576151a461517b565b500190565b6000602082840312156151bb57600080fd5b81356001600160f01b03198116811461278257600080fd5b6000602082840312156151e557600080fd5b81516001600160401b038111156151fb57600080fd5b8201601f8101841361520c57600080fd5b805161521a61485682614810565b81815285602083850101111561522f57600080fd5b610f3e826020830160208601614c34565b634e487b7160e01b600052603260045260246000fd5b6000600182016152685761526861517b565b5060010190565b60006020828403121561528157600080fd5b81516127828161476b565b60006020828403121561529e57600080fd5b5051919050565b6000608082840312156152b757600080fd5b604051608081018181106001600160401b03821117156152d9576152d9614780565b60405282516152e781614980565b815260208301516152f781614980565b6020820152604083015161530a81614980565b6040820152606083015161531d8161476b565b60608201529392505050565b600060c0820160ff60f81b875460f81b1683526001808801546020818187015260018060a01b0391508160028b0154166040818189015260c060608901528591508a5480875260e0890192508b60005283600020965060005b818110156153ab578754868116855260a01c61ffff168585015296860196928201928601615382565b5050506001600160a01b038916608088015294506153c99350505050565b6001600160a01b03831660a0830152610f3e565b600082601f8301126153ee57600080fd5b815160206153fe61485683614c93565b82815260059290921b8401810191818101908684111561541d57600080fd5b8286015b84811015614d115780516154348161476b565b8352918301918301615421565b6000806000606080858703121561545757600080fd5b84516001600160401b038082111561546e57600080fd5b818701915087601f83011261548257600080fd5b8151602061549261485683614c93565b82815260079290921b8401810191818101908b8411156154b157600080fd5b948201945b8386101561553757858c0360808112156154d05760008081fd5b6154d86147be565b6040808312156154e85760008081fd5b6154f0614796565b925088516154fd816148f8565b83528886015161550c8161476b565b83870152918152878201518186015288880151918101919091528252608090950194908201906154b6565b918a015191985090945050508083111561555057600080fd5b61555c888489016153dd565b9450604087015192508083111561557257600080fd5b5050615580868287016153dd565b9150509250925092565b60008282101561559c5761559c61517b565b500390565b6000602082840312156155b357600080fd5b815161278281614980565b6000808335601e198436030181126155d557600080fd5b8301803591506001600160401b038211156155ef57600080fd5b6020019150606081023603821315614a2557600080fd5b6001600160a01b039384168152919092166020820152604081019190915260600190565b6001600160a01b0394851681529290931660208301526040820152606081019190915260a06080820181905260009082015260c00190565b813561566d816148f8565b6008811061568b57634e487b7160e01b600052602160045260246000fd5b815460ff821691508160ff19821617835560208401356156aa8161476b565b6001600160a81b03199190911690911760089190911b610100600160a81b031617815560408201356001820155606090910135600290910155565b80546001600160a01b0319166001600160a01b0392909216919091179055565b813561571081614b2f565b815460ff191660f882901c178255506020820135600182015560408201356157378161476b565b6110ee81600284016156e5565b6000808335601e1984360301811261575b57600080fd5b8301803591506001600160401b0382111561577557600080fd5b6020019150600681901b3603821315614a2557600080fd5b813561579881614b2f565b815460ff191660f882901c17825550602082013560018201555050565b81356157c08161476b565b6157ca81836156e5565b50602082013561ffff811681146157e057600080fd5b815461ffff60a01b191660a09190911b61ffff60a01b1617905550565b6000825161580f818460208701614c34565b919091019291505056fea26469706673582212207aa7f0950aefce26a018a90e4d87708cd7d647fdaf7658f6881718ef4a473ebd64736f6c634300080d0033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
00000000000000000000000044c1bce3f284e9a672fa4ba6f5b02e8651da88a0
-----Decoded View---------------
Arg [0] : _erc20 (address): 0x44C1bCE3f284e9a672fa4ba6f5b02E8651da88a0
-----Encoded View---------------
1 Constructor Arguments found :
Arg [0] : 00000000000000000000000044c1bce3f284e9a672fa4ba6f5b02e8651da88a0
Loading...
Loading
Loading...
Loading
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.