Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 1 from a total of 1 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
Value | ||||
---|---|---|---|---|---|---|---|---|---|
0x60806040 | 17774886 | 348 days ago | IN | Create: ERC721GenerativeGenesis | 0 ETH | 0.09275556 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Contract Name:
ERC721GenerativeGenesis
Compiler Version
v0.8.10+commit.fc410830
Optimization Enabled:
Yes with 1 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
//SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import { Strings } from "@openzeppelin/contracts/utils/Strings.sol"; import { Base64 } from "@openzeppelin/contracts/utils/Base64.sol"; import "./ERC721Base.sol"; import "../tokenManager/interfaces/IPostTransfer.sol"; import "../tokenManager/interfaces/IPostBurn.sol"; import "./interfaces/IERC721GeneralMint.sol"; import "./ERC721GeneralGenesisBase.sol"; /** * @title Generative ERC721 * @author highlight.xyz * @notice Generative NFT smart contract */ contract ERC721GenerativeGenesis is ERC721GeneralGenesisBase { using EnumerableSet for EnumerableSet.AddressSet; /** * @notice Generative Code URI */ string private _generativeCodeURI; /** * @notice Initialize the contract * @param data Data to initialize the contract * @ param creator Creator/owner of contract * @ param _contractURI Contract metadata * @ param defaultRoyalty Default royalty object for contract (optional) * @ param _defaultTokenManager Default token manager for contract (optional) * @ param _name Name of token edition * @ param _symbol Symbol of the token edition * @ param trustedForwarder Trusted minimal forwarder * @ param initialMinter Initial minter to register * @ param _generativeCodeURI Generative code URI * @ param newBaseURI Base URI for contract * @ param _limitSupply Initial limit supply * @ param useMarketplaceFiltererRegistry Denotes whether to use marketplace filterer registry * @param _observability Observability contract address */ function initialize(bytes calldata data, address _observability) external initializer { ( address creator, string memory _contractURI, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager, string memory _name, string memory _symbol, address trustedForwarder, address initialMinter, string memory _codeURI, string memory newBaseURI, uint256 _limitSupply, bool useMarketplaceFiltererRegistry ) = abi.decode( data, ( address, string, IRoyaltyManager.Royalty, address, string, string, address, address, string, string, uint256, bool ) ); __ERC721URIStorage_init(); __ERC721Base_initialize(creator, defaultRoyalty, _defaultTokenManager); __ERC2771ContextUpgradeable__init__(trustedForwarder); __ERC721A_init(_name, _symbol); _minters.add(initialMinter); contractURI = _contractURI; _generativeCodeURI = _codeURI; IObservability(_observability).emitGenerativeSeriesDeployed(address(this)); observability = IObservability(_observability); if (bytes(newBaseURI).length > 0) { _setBaseURI(newBaseURI); // don't emit on observability contract here } if (_limitSupply > 0) { limitSupply = _limitSupply; // don't emit on observability contract here } } /** * @notice Generative Code URI **/ function generativeCodeUri() external view returns (string memory) { return _generativeCodeURI; } function tokenImage(string memory _owner) external pure returns (string memory) { bytes memory owner = bytes(_owner); require(owner.length == 42, "Wrong wallet address"); string memory svg = '<svg xmlns="http://www.w3.org/2000/svg" id="0xeE9be9846abB49532a03C05FC27A0e4Dbc84e857" viewBox="0 0 700 700"><style>.a{stroke-width:20;}</style><g class="a" fill="none">'; uint256 r = 270; for (uint256 i = 0; i < 39; ) { string memory hexCode; if (i == 0) { hexCode = string(abi.encodePacked("#00", owner[i + 2])); } else { hexCode = string(abi.encodePacked("#", owner[i], owner[i + 1], owner[i + 2])); } svg = string( abi.encodePacked( svg, '<circle cx="350" cy="350" r="', Strings.toString(r), '" stroke="', hexCode, '"/>' ) ); r -= 20; i += 3; } return encodeSVG( abi.encodePacked( svg, '<circle cx="350" cy="350" r="10" stroke="#', owner[39], owner[40], owner[41], '"/>', "</g></svg>" ) ); } /** * @notice Generate token image based on owner wallet address. * @param chainId Chain ID contract is deployed to * @param tokenId Token ID to get animation for * @param walletAddress Wallet address that owns the token * @param contractAddress Contract Address * @param hash Transaction Hash the token was minted in. * @param blockHash Block Hash the token was minted in. * @param blockNumber Block Number the token was minted in. * @param timestamp Timestamp of the block * @param gasPrice Gas price of the transaction in gwei. **/ function tokenAnimation( uint256 chainId, uint256 tokenId, string memory walletAddress, string memory contractAddress, string memory hash, string memory blockHash, uint256 blockNumber, uint256 timestamp, uint256 gasPrice ) external view returns (string memory) { return string( abi.encodePacked( _generativeCodeURI, "/index.html?", _constructHashParams(contractAddress, walletAddress, hash, blockHash), _constructIds(tokenId, chainId, blockNumber, timestamp, gasPrice) ) ); } /** * @notice Construct Hash params. * @param walletAddress Wallet address that owns the token * @param contractAddress Contract Address * @param hash Transaction Hash the token was minted in. * @param blockHash Block Hash the token was minted in. **/ function _constructHashParams( string memory contractAddress, string memory walletAddress, string memory hash, string memory blockHash ) private pure returns (string memory) { return string(abi.encodePacked("a=", contractAddress, "&h=", hash, "&bh=", blockHash, "&wa=", walletAddress)); } /** * @notice Construct ID params. * @param chainId Chain ID contract is deployed to * @param tokenId Token ID to get animation for * @param blockNumber Block Number the token was minted in. * @param timestamp Timestamp of the block * @param gasPrice Gas price of the transaction in gwei. **/ function _constructIds( uint256 tokenId, uint256 chainId, uint256 blockNumber, uint256 timestamp, uint256 gasPrice ) private pure returns (string memory) { return string( abi.encodePacked( "&c=", Strings.toString(chainId), "&tid=", Strings.toString(tokenId), "&bn=", Strings.toString(blockNumber), "&t=", Strings.toString(timestamp), "&gp=", Strings.toString(gasPrice) ) ); } /** * @param svg Raw svg to base64 and turn into a data-uri * @dev Encodes the argument svg bytes into base64-data uri format */ function encodeSVG(bytes memory svg) private pure returns (string memory) { return string(abi.encodePacked("data:image/svg+xml;base64,", Base64.encode(svg))); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Base64.sol) pragma solidity ^0.8.0; /** * @dev Provides a set of functions to operate with Base64 strings. * * _Available since v4.5._ */ library Base64 { /** * @dev Base64 Encoding/Decoding Table */ string internal constant _TABLE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; /** * @dev Converts a `bytes` to its Bytes64 `string` representation. */ function encode(bytes memory data) internal pure returns (string memory) { /** * Inspired by Brecht Devos (Brechtpd) implementation - MIT licence * https://github.com/Brechtpd/base64/blob/e78d9fd951e7b0977ddca77d92dc85183770daf4/base64.sol */ if (data.length == 0) return ""; // Loads the table into memory string memory table = _TABLE; // Encoding takes 3 bytes chunks of binary data from `bytes` data parameter // and split into 4 numbers of 6 bits. // The final Base64 length should be `bytes` data length multiplied by 4/3 rounded up // - `data.length + 2` -> Round up // - `/ 3` -> Number of 3-bytes chunks // - `4 *` -> 4 characters for each chunk string memory result = new string(4 * ((data.length + 2) / 3)); /// @solidity memory-safe-assembly assembly { // Prepare the lookup table (skip the first "length" byte) let tablePtr := add(table, 1) // Prepare result pointer, jump over length let resultPtr := add(result, 32) // Run over the input, 3 bytes at a time for { let dataPtr := data let endPtr := add(data, mload(data)) } lt(dataPtr, endPtr) { } { // Advance 3 bytes dataPtr := add(dataPtr, 3) let input := mload(dataPtr) // To write each character, shift the 3 bytes (18 bits) chunk // 4 times in blocks of 6 bits for each character (18, 12, 6, 0) // and apply logical AND with 0x3F which is the number of // the previous character in the ASCII table prior to the Base64 Table // The result is then added to the table to get the character to write, // and finally write it in the result pointer but with a left shift // of 256 (1 byte) - 8 (1 ASCII char) = 248 bits mstore8(resultPtr, mload(add(tablePtr, and(shr(18, input), 0x3F)))) resultPtr := add(resultPtr, 1) // Advance mstore8(resultPtr, mload(add(tablePtr, and(shr(12, input), 0x3F)))) resultPtr := add(resultPtr, 1) // Advance mstore8(resultPtr, mload(add(tablePtr, and(shr(6, input), 0x3F)))) resultPtr := add(resultPtr, 1) // Advance mstore8(resultPtr, mload(add(tablePtr, and(input, 0x3F)))) resultPtr := add(resultPtr, 1) // Advance } // When data `bytes` is not exactly 3 bytes long // it is padded with `=` characters at the end switch mod(mload(data), 3) case 1 { mstore8(sub(resultPtr, 1), 0x3d) mstore8(sub(resultPtr, 2), 0x3d) } case 2 { mstore8(sub(resultPtr, 1), 0x3d) } } return result; } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "../royaltyManager/interfaces/IRoyaltyManager.sol"; import "../tokenManager/interfaces/ITokenManager.sol"; import "../utils/Ownable.sol"; import "../utils/ERC2981/IERC2981Upgradeable.sol"; import "../utils/ERC165/ERC165CheckerUpgradeable.sol"; import "../metatx/ERC2771ContextUpgradeable.sol"; import "../observability/IObservability.sol"; import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol"; import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol"; /** * @title Base ERC721 * @author highlight.xyz * @notice Core piece of Highlight NFT contracts (v2) */ abstract contract ERC721Base is OwnableUpgradeable, IERC2981Upgradeable, ERC2771ContextUpgradeable, ReentrancyGuardUpgradeable { using EnumerableSet for EnumerableSet.AddressSet; using ERC165CheckerUpgradeable for address; /** * @notice Throw when token or royalty manager is invalid */ error InvalidManager(); /** * @notice Throw when token or royalty manager does not exist */ error ManagerDoesNotExist(); /** * @notice Throw when sender is unauthorized to perform action */ error Unauthorized(); /** * @notice Throw when sender is not a minter */ error NotMinter(); /** * @notice Throw when token manager or royalty manager swap is blocked */ error ManagerSwapBlocked(); /** * @notice Throw when token manager or royalty manager remove is blocked */ error ManagerRemoveBlocked(); /** * @notice Throw when setting default or granular royalty is blocked */ error RoyaltySetBlocked(); /** * @notice Throw when royalty BPS is invalid */ error RoyaltyBPSInvalid(); /** * @notice Throw when minter registration is invalid */ error MinterRegistrationInvalid(); /** * @notice Set of minters allowed to mint on contract */ EnumerableSet.AddressSet internal _minters; /** * @notice Global token/edition manager default */ address public defaultManager; /** * @notice Token/edition managers per token grouping. * Edition ID if implemented by Editions contract, and token ID if implemented by General contract. */ mapping(uint256 => address) internal _managers; /** * @notice Default royalty for entire contract */ IRoyaltyManager.Royalty internal _defaultRoyalty; /** * @notice Royalty per token grouping. * Edition ID if implemented by Editions contract, and token ID if implemented by General contract. */ mapping(uint256 => IRoyaltyManager.Royalty) internal _royalties; /** * @notice Royalty manager - optional contract that defines the conditions around setting royalties */ address public royaltyManager; /** * @notice Freezes minting on smart contract forever */ uint8 internal _mintFrozen; /** * @notice Observability contract */ IObservability public observability; /** * @notice Emitted when minter is registered or unregistered * @param minter Minter that was changed * @param registered True if the minter was registered, false if unregistered */ event MinterRegistrationChanged(address indexed minter, bool indexed registered); /** * @notice Emitted when token managers are set for token/edition ids * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ event GranularTokenManagersSet(uint256[] _ids, address[] _tokenManagers); /** * @notice Emitted when token managers are removed for token/edition ids * @param _ids Edition / token ids to remove token managers for */ event GranularTokenManagersRemoved(uint256[] _ids); /** * @notice Emitted when default token manager changed * @param newDefaultTokenManager New default token manager. Zero address if old one was removed */ event DefaultTokenManagerChanged(address indexed newDefaultTokenManager); /** * @notice Emitted when default royalty is set * @param recipientAddress Royalty recipient * @param royaltyPercentageBPS Percentage of sale (in basis points) owed to royalty recipient */ event DefaultRoyaltySet(address indexed recipientAddress, uint16 indexed royaltyPercentageBPS); /** * @notice Emitted when royalties are set for edition / token ids * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ event GranularRoyaltiesSet(uint256[] ids, IRoyaltyManager.Royalty[] _newRoyalties); /** * @notice Emitted when royalty manager is updated * @param newRoyaltyManager New royalty manager. Zero address if old one was removed */ event RoyaltyManagerChanged(address indexed newRoyaltyManager); /** * @notice Emitted when mints are frozen permanently */ event MintsFrozen(); /** * @notice Restricts calls to minters */ modifier onlyMinter() { if (!_minters.contains(_msgSender())) { _revert(NotMinter.selector); } _; } /** * @notice Restricts calls if input royalty bps is over 10000 */ modifier royaltyValid(uint16 _royaltyBPS) { if (!_royaltyBPSValid(_royaltyBPS)) { _revert(RoyaltyBPSInvalid.selector); } _; } /** * @notice Registers a minter * @param minter New minter */ function registerMinter(address minter) external onlyOwner { if (!_minters.add(minter)) { _revert(MinterRegistrationInvalid.selector); } emit MinterRegistrationChanged(minter, true); observability.emitMinterRegistrationChanged(minter, true); } /** * @notice Unregisters a minter * @param minter Minter to unregister */ function unregisterMinter(address minter) external onlyOwner { if (!_minters.remove(minter)) { _revert(MinterRegistrationInvalid.selector); } emit MinterRegistrationChanged(minter, false); observability.emitMinterRegistrationChanged(minter, false); } /** * @notice Sets granular token managers if current token manager(s) allow it * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ function setGranularTokenManagers(uint256[] calldata _ids, address[] calldata _tokenManagers) external nonReentrant { address msgSender = _msgSender(); address tempOwner = owner(); uint256 idsLength = _ids.length; for (uint256 i = 0; i < idsLength; i++) { if (!_isValidTokenManager(_tokenManagers[i])) { _revert(InvalidManager.selector); } address currentTokenManager = tokenManager(_ids[i]); if (currentTokenManager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(currentTokenManager).canSwap(msgSender, _ids[i], _managers[i])) { _revert(ManagerSwapBlocked.selector); } } _managers[_ids[i]] = _tokenManagers[i]; } emit GranularTokenManagersSet(_ids, _tokenManagers); observability.emitGranularTokenManagersSet(_ids, _tokenManagers); } /** * @notice Remove granular token managers * @param _ids Edition / token ids to remove token managers for */ function removeGranularTokenManagers(uint256[] calldata _ids) external nonReentrant { address msgSender = _msgSender(); uint256 idsLength = _ids.length; for (uint256 i = 0; i < idsLength; i++) { address currentTokenManager = _managers[_ids[i]]; if (currentTokenManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!ITokenManager(currentTokenManager).canRemoveItself(msgSender, _ids[i])) { _revert(ManagerRemoveBlocked.selector); } _managers[_ids[i]] = address(0); } emit GranularTokenManagersRemoved(_ids); observability.emitGranularTokenManagersRemoved(_ids); } /** * @notice Set default token manager if current token manager allows it * @param _defaultTokenManager New default token manager */ function setDefaultTokenManager(address _defaultTokenManager) external nonReentrant { if (!_isValidTokenManager(_defaultTokenManager)) { _revert(InvalidManager.selector); } address msgSender = _msgSender(); address currentTokenManager = defaultManager; if (currentTokenManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(currentTokenManager).canSwap(msgSender, 0, _defaultTokenManager)) { _revert(ManagerSwapBlocked.selector); } } defaultManager = _defaultTokenManager; emit DefaultTokenManagerChanged(_defaultTokenManager); observability.emitDefaultTokenManagerChanged(_defaultTokenManager); } /** * @notice Removes default token manager if current token manager allows it */ function removeDefaultTokenManager() external nonReentrant { address msgSender = _msgSender(); address currentTokenManager = defaultManager; if (currentTokenManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!ITokenManager(currentTokenManager).canRemoveItself(msgSender, 0)) { _revert(ManagerRemoveBlocked.selector); } defaultManager = address(0); emit DefaultTokenManagerChanged(address(0)); observability.emitDefaultTokenManagerChanged(address(0)); } /** * @notice Sets default royalty if royalty manager allows it * @param _royalty New default royalty */ function setDefaultRoyalty(IRoyaltyManager.Royalty calldata _royalty) external nonReentrant royaltyValid(_royalty.royaltyPercentageBPS) { address msgSender = _msgSender(); address _royaltyManager = royaltyManager; if (_royaltyManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!IRoyaltyManager(_royaltyManager).canSetDefaultRoyalty(_royalty, msgSender)) { _revert(RoyaltySetBlocked.selector); } } _defaultRoyalty = _royalty; emit DefaultRoyaltySet(_royalty.recipientAddress, _royalty.royaltyPercentageBPS); observability.emitDefaultRoyaltySet(_royalty.recipientAddress, _royalty.royaltyPercentageBPS); } /** * @notice Sets granular royalties (per token-grouping) if royalty manager allows it * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ function setGranularRoyalties(uint256[] calldata ids, IRoyaltyManager.Royalty[] calldata _newRoyalties) external nonReentrant { address msgSender = _msgSender(); address tempOwner = owner(); address _royaltyManager = royaltyManager; uint256 idsLength = ids.length; if (_royaltyManager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } for (uint256 i = 0; i < idsLength; i++) { if (!_royaltyBPSValid(_newRoyalties[i].royaltyPercentageBPS)) { _revert(RoyaltyBPSInvalid.selector); } _royalties[ids[i]] = _newRoyalties[i]; } } else { for (uint256 i = 0; i < idsLength; i++) { if (!_royaltyBPSValid(_newRoyalties[i].royaltyPercentageBPS)) { _revert(RoyaltyBPSInvalid.selector); } if (!IRoyaltyManager(_royaltyManager).canSetGranularRoyalty(ids[i], _newRoyalties[i], msgSender)) { _revert(RoyaltySetBlocked.selector); } _royalties[ids[i]] = _newRoyalties[i]; } } emit GranularRoyaltiesSet(ids, _newRoyalties); observability.emitGranularRoyaltiesSet(ids, _newRoyalties); } /** * @notice Sets royalty manager if current one allows it * @param _royaltyManager New royalty manager */ function setRoyaltyManager(address _royaltyManager) external nonReentrant { if (!_isValidRoyaltyManager(_royaltyManager)) { _revert(InvalidManager.selector); } address msgSender = _msgSender(); address currentRoyaltyManager = royaltyManager; if (currentRoyaltyManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!IRoyaltyManager(currentRoyaltyManager).canSwap(_royaltyManager, msgSender)) { _revert(ManagerSwapBlocked.selector); } } royaltyManager = _royaltyManager; emit RoyaltyManagerChanged(_royaltyManager); observability.emitRoyaltyManagerChanged(_royaltyManager); } /** * @notice Removes royalty manager if current one allows it */ function removeRoyaltyManager() external nonReentrant { address msgSender = _msgSender(); address currentRoyaltyManager = royaltyManager; if (currentRoyaltyManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!IRoyaltyManager(currentRoyaltyManager).canRemoveItself(msgSender)) { _revert(ManagerRemoveBlocked.selector); } royaltyManager = address(0); emit RoyaltyManagerChanged(address(0)); observability.emitRoyaltyManagerChanged(address(0)); } /** * @notice Freeze mints on contract forever */ function freezeMints() external onlyOwner nonReentrant { _mintFrozen = 1; emit MintsFrozen(); observability.emitMintsFrozen(); } /** * @notice Return allowed minters on contract */ function minters() external view returns (address[] memory) { return _minters.values(); } /** * @notice Conforms to ERC-2981. Editions should overwrite to return royalty for entire edition * @param _tokenGroupingId Token id if on general, and edition id if on editions * @param _salePrice Sale price of token */ function royaltyInfo(uint256 _tokenGroupingId, uint256 _salePrice) public view virtual override returns (address receiver, uint256 royaltyAmount) { IRoyaltyManager.Royalty memory royalty = _royalties[_tokenGroupingId]; if (royalty.recipientAddress == address(0)) { royalty = _defaultRoyalty; } receiver = royalty.recipientAddress; royaltyAmount = (_salePrice * uint256(royalty.royaltyPercentageBPS)) / 10000; } /** * @notice Returns the token manager for the id passed in. * @param id Token ID or Edition ID for Editions implementing contracts */ function tokenManager(uint256 id) public view returns (address manager) { manager = defaultManager; address granularManager = _managers[id]; if (granularManager != address(0)) { manager = granularManager; } } /** * @notice Initializes the contract, setting the creator as the initial owner. * @param _creator Contract creator * @param defaultRoyalty Default royalty for the contract * @param _defaultTokenManager Default token manager for the contract */ function __ERC721Base_initialize( address _creator, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager ) internal onlyInitializing royaltyValid(defaultRoyalty.royaltyPercentageBPS) { __Ownable_init(); __ReentrancyGuard_init(); _transferOwnership(_creator); if (_defaultRoyalty.recipientAddress != address(0)) { _defaultRoyalty = defaultRoyalty; } if (_defaultTokenManager != address(0)) { defaultManager = _defaultTokenManager; } } /** * @notice Returns true if address is a valid tokenManager * @param _tokenManager Token manager being checked */ function _isValidTokenManager(address _tokenManager) internal view returns (bool) { return _tokenManager.supportsInterface(type(ITokenManager).interfaceId); } /** * @notice Returns true if address is a valid royaltyManager * @param _royaltyManager Royalty manager being checked */ function _isValidRoyaltyManager(address _royaltyManager) internal view returns (bool) { return _royaltyManager.supportsInterface(type(IRoyaltyManager).interfaceId); } /** * @notice Used for meta-transactions */ function _msgSender() internal view virtual override(ContextUpgradeable, ERC2771ContextUpgradeable) returns (address sender) { return ERC2771ContextUpgradeable._msgSender(); } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } /** * @notice Used for meta-transactions */ function _msgData() internal view virtual override(ContextUpgradeable, ERC2771ContextUpgradeable) returns (bytes calldata) { return ERC2771ContextUpgradeable._msgData(); } /** * @notice Returns true if royalty bps passed in is valid (<= 10000) * @param _royaltyBPS Royalty basis points */ function _royaltyBPSValid(uint16 _royaltyBPS) private pure returns (bool) { return _royaltyBPS <= 10000; } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; /** * @author highlight.xyz * @notice If token managers implement this, transfer actions will call * postSafeTransferFrom or postTransferFrom on the token manager. */ interface IPostTransfer { /** * @notice Hook called by community after safe transfers, if token manager of transferred token implements this * interface. * @param operator Operator transferring tokens * @param from Token(s) sender * @param to Token(s) recipient * @param id Transferred token's id * @param data Arbitrary data */ function postSafeTransferFrom( address operator, address from, address to, uint256 id, bytes memory data ) external; /** * @notice Hook called by community after transfers, if token manager of transferred token implements * this interface. * @param operator Operator transferring tokens * @param from Token(s) sender * @param to Token(s) recipient * @param id Transferred token's id */ function postTransferFrom( address operator, address from, address to, uint256 id ) external; }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; /** * @author highlight.xyz * @notice If token managers implement this, transfer actions will call * postBurn on the token manager. */ interface IPostBurn { /** * @notice Hook called by contract after burn, if token manager of burned token implements this * interface. * @param operator Operator burning tokens * @param sender Msg sender * @param id Burned token's id or id of edition of token that is burned */ function postBurn( address operator, address sender, uint256 id ) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @notice General721 mint interface * @author highlight.xyz */ interface IERC721GeneralMint { /** * @notice Mint one token to one recipient * @param recipient Recipient of minted NFT */ function mintOneToOneRecipient(address recipient) external returns (uint256); /** * @notice Mint an amount of tokens to one recipient * @param recipient Recipient of minted NFTs * @param amount Amount of NFTs minted */ function mintAmountToOneRecipient(address recipient, uint256 amount) external; /** * @notice Mint one token to multiple recipients. Useful for use-cases like airdrops * @param recipients Recipients of minted NFTs */ function mintOneToMultipleRecipients(address[] calldata recipients) external; /** * @notice Mint the same amount of tokens to multiple recipients * @param recipients Recipients of minted NFTs * @param amount Amount of NFTs minted to each recipient */ function mintSameAmountToMultipleRecipients(address[] calldata recipients, uint256 amount) external; /** * @notice Mint a chosen token id to a single recipient * @param recipient Recipient of chosen NFT * @param tokenId ID of NFT to mint */ function mintSpecificTokenToOneRecipient(address recipient, uint256 tokenId) external; /** * @notice Mint chosen token ids to a single recipient * @param recipient Recipient of chosen NFT * @param tokenIds IDs of NFTs to mint */ function mintSpecificTokensToOneRecipient(address recipient, uint256[] calldata tokenIds) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "./ERC721Base.sol"; import "../metadata/MetadataEncryption.sol"; import "../tokenManager/interfaces/IPostTransfer.sol"; import "../tokenManager/interfaces/IPostBurn.sol"; import "./interfaces/IERC721GeneralMint.sol"; import "../utils/ERC721/ERC721URIStorageUpgradeable.sol"; /** * @title Generalized Base ERC721 * @author highlight.xyz * @notice Generalized Base NFT smart contract */ abstract contract ERC721GeneralGenesisBase is ERC721Base, ERC721URIStorageUpgradeable, IERC721GeneralMint { using EnumerableSet for EnumerableSet.AddressSet; /** * @notice Throw when attempting to mint, while mint is frozen */ error MintFrozen(); /** * @notice Throw when requested token is not in range within bounds of limit supply */ error TokenNotInRange(); /** * @notice Throw when new supply is over limit supply */ error OverLimitSupply(); /** * @notice Throw when array lengths are mismatched */ error MismatchedArrayLengths(); /** * @notice Throw when string is empty */ error EmptyString(); /** * @notice Contract metadata */ string public contractURI; /** * @notice Limit the supply to take advantage of over-promising in summation with multiple mint vectors */ uint256 public limitSupply; /** * @notice Emitted when uris are set for tokens * @param ids IDs of tokens to set uris for * @param uris Uris to set on tokens */ event TokenURIsSet(uint256[] ids, string[] uris); /** * @notice Emitted when limit supply is set * @param newLimitSupply Limit supply to set */ event LimitSupplySet(uint256 indexed newLimitSupply); /** * @notice See {IERC721GeneralMint-mintOneToOneRecipient} */ function mintOneToOneRecipient(address recipient) external onlyMinter nonReentrant returns (uint256) { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = _nextTokenId(); _requireLimitSupply(tempSupply); _mint(recipient, 1, tempSupply, tempSupply); return tempSupply; } /** * @notice See {IERC721GeneralMint-mintAmountToOneRecipient} */ function mintAmountToOneRecipient(address recipient, uint256 amount) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = _nextTokenId(); // cache _requireLimitSupply(tempSupply + amount - 1); _mint(recipient, amount, tempSupply, tempSupply); } /** * @notice See {IERC721GeneralMint-mintOneToMultipleRecipients} */ function mintOneToMultipleRecipients(address[] calldata recipients) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 recipientsLength = recipients.length; uint256 tempSupply = _nextTokenId(); // cache for (uint256 i = 0; i < recipientsLength; i++) { _mint(recipients[i], 1, tempSupply, tempSupply); tempSupply++; } _requireLimitSupply(tempSupply - 1); } /** * @notice See {IERC721GeneralMint-mintSameAmountToMultipleRecipients} */ function mintSameAmountToMultipleRecipients(address[] calldata recipients, uint256 amount) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 recipientsLength = recipients.length; uint256 tempSupply = _nextTokenId(); // cache for (uint256 i = 0; i < recipientsLength; i++) { _mint(recipients[i], amount, tempSupply, tempSupply); tempSupply += amount; } _requireLimitSupply(tempSupply - 1); } /** * @notice See {IERC721GeneralMint-mintSpecificTokenToOneRecipient} */ function mintSpecificTokenToOneRecipient(address recipient, uint256 tokenId) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 _limitSupply = limitSupply; if (_limitSupply != 0) { if (tokenId > _limitSupply) { _revert(TokenNotInRange.selector); } } _mint(recipient, 1, tokenId, 0); } /** * @notice See {IERC721GeneralMint-mintSpecificTokensToOneRecipient} */ function mintSpecificTokensToOneRecipient(address recipient, uint256[] calldata tokenIds) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = _nextTokenId(); uint256 tokenIdsLength = tokenIds.length; uint256 _limitSupply = limitSupply; if (_limitSupply == 0) { // don't check that token id is within range, since _limitSupply being 0 implies unlimited range for (uint256 i = 0; i < tokenIdsLength; i++) { _mint(recipient, 1, tokenIds[i], tempSupply); tempSupply++; } } else { // check that token id is within range for (uint256 i = 0; i < tokenIdsLength; i++) { if (tokenIds[i] > _limitSupply) { _revert(TokenNotInRange.selector); } _mint(recipient, 1, tokenIds[i], tempSupply); tempSupply++; } } } /** * @notice Override base URI system for select tokens, with custom per-token metadata * @param ids IDs of tokens to override base uri system for with custom uris * @param uris Custom uris */ function setTokenURIs(uint256[] calldata ids, string[] calldata uris) external nonReentrant { uint256 idsLength = ids.length; if (idsLength != uris.length) { _revert(MismatchedArrayLengths.selector); } for (uint256 i = 0; i < idsLength; i++) { _setTokenURI(ids[i], uris[i]); } emit TokenURIsSet(ids, uris); observability.emitTokenURIsSet(ids, uris); } /** * @notice Set base uri * @param newBaseURI New base uri to set */ function setBaseURI(string calldata newBaseURI) external nonReentrant { if (bytes(newBaseURI).length == 0) { _revert(EmptyString.selector); } address _manager = defaultManager; if (_manager == address(0)) { if (_msgSender() != owner()) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(_manager).canUpdateMetadata(_msgSender(), 0, bytes(newBaseURI))) { _revert(Unauthorized.selector); } } _setBaseURI(newBaseURI); observability.emitBaseUriSet(newBaseURI); } /** * @notice Set limit supply * @param _limitSupply Limit supply to set */ function setLimitSupply(uint256 _limitSupply) external onlyOwner nonReentrant { // allow it to be 0, for post-mint limitSupply = _limitSupply; emit LimitSupplySet(_limitSupply); observability.emitLimitSupplySet(_limitSupply); } /** * @notice Set contract name * @param newName New name * @param newSymbol New symbol * @param newContractUri New contractURI */ function setContractMetadata( string calldata newName, string calldata newSymbol, string calldata newContractUri ) external onlyOwner { _setContractMetadata(newName, newSymbol); contractURI = newContractUri; observability.emitContractMetadataSet(newName, newSymbol, newContractUri); } /** * @notice Hook called after transfers * @param from Account token is being transferred from * @param to Account token is being transferred to * @param tokenId ID of token being transferred */ function _afterTokenTransfers( address from, address to, uint256 tokenId ) internal override { address msgSender = _msgSender(); address _manager = tokenManager(tokenId); if (_manager != address(0) && IERC165Upgradeable(_manager).supportsInterface(type(IPostTransfer).interfaceId)) { IPostTransfer(_manager).postSafeTransferFrom(msgSender, from, to, tokenId, ""); } observability.emitTransfer(from, to, tokenId); } /** * @notice See {IERC721-burn}. Overrides default behaviour to check associated tokenManager. */ function burn(uint256 tokenId) public nonReentrant { address _manager = tokenManager(tokenId); address msgSender = _msgSender(); if (_manager != address(0) && IERC165Upgradeable(_manager).supportsInterface(type(IPostBurn).interfaceId)) { address owner = ownerOf(tokenId); IPostBurn(_manager).postBurn(msgSender, owner, tokenId); } else { // default to restricting burn to owner or operator if a valid TM isn't present if (!_isApprovedOrOwner(msgSender, tokenId)) { _revert(Unauthorized.selector); } } _burn(tokenId); observability.emitTransfer(msgSender, address(0), tokenId); } /** * @notice Overrides tokenURI to first rotate the token id * @param tokenId ID of token to get uri for */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { return ERC721URIStorageUpgradeable.tokenURI(tokenId); } /** * @notice See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165Upgradeable, ERC721AUpgradeable) returns (bool) { return ERC721AUpgradeable.supportsInterface(interfaceId); } /** * @notice Used for meta-transactions */ function _msgSender() internal view override(ERC721Base) returns (address sender) { return ERC2771ContextUpgradeable._msgSender(); } /** * @notice Used for meta-transactions */ function _msgData() internal view override(ERC721Base) returns (bytes calldata) { return ERC2771ContextUpgradeable._msgData(); } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure override(ERC721AUpgradeable, ERC721Base) { ERC721AUpgradeable._revert(errorSelector); } /** * @notice Override base URI system for select tokens, with custom per-token metadata * @param tokenId Token to set uri for * @param _uri Uri to set on token */ function _setTokenURI(uint256 tokenId, string calldata _uri) private { address _manager = tokenManager(tokenId); address msgSender = _msgSender(); address tempOwner = owner(); if (_manager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(_manager).canUpdateMetadata(msgSender, tokenId, bytes(_uri))) { _revert(Unauthorized.selector); } } _tokenURIs[tokenId] = _uri; } /** * @notice Require the new supply of tokens after mint to be less than limit supply * @param newSupply New supply */ function _requireLimitSupply(uint256 newSupply) private view { uint256 _limitSupply = limitSupply; if (_limitSupply != 0 && newSupply > _limitSupply) { _revert(OverLimitSupply.selector); } } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @title IRoyaltyManager * @author highlight.xyz * @notice Enables interfacing with custom royalty managers that define conditions on setting royalties for * NFT contracts */ interface IRoyaltyManager { /** * @notice Struct containing values required to adhere to ERC-2981 * @param recipientAddress Royalty recipient - can be EOA, royalty splitter contract, etc. * @param royaltyPercentageBPS Royalty cut, in basis points */ struct Royalty { address recipientAddress; uint16 royaltyPercentageBPS; } /** * @notice Defines conditions around being able to swap royalty manager for another one * @param newRoyaltyManager New royalty manager being swapped in * @param sender msg sender */ function canSwap(address newRoyaltyManager, address sender) external view returns (bool); /** * @notice Defines conditions around being able to remove current royalty manager * @param sender msg sender */ function canRemoveItself(address sender) external view returns (bool); /** * @notice Defines conditions around being able to set granular royalty (per token or per edition) * @param id Edition / token ID whose royalty is being set * @param royalty Royalty being set * @param sender msg sender */ function canSetGranularRoyalty( uint256 id, Royalty calldata royalty, address sender ) external view returns (bool); /** * @notice Defines conditions around being able to set default royalty * @param royalty Royalty being set * @param sender msg sender */ function canSetDefaultRoyalty(Royalty calldata royalty, address sender) external view returns (bool); }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @title ITokenManager * @author highlight.xyz * @notice Enables interfacing with custom token managers */ interface ITokenManager { /** * @notice Returns whether metadata updater is allowed to update * @param sender Updater * @param id Token/edition who's uri is being updated * If id is 0, implementation should decide behaviour for base uri update * @param newData Token's new uri if called by general contract, and any metadata field if called by editions * @return If invocation can update metadata */ function canUpdateMetadata( address sender, uint256 id, bytes calldata newData ) external returns (bool); /** * @notice Returns whether token manager can be swapped for another one by invocator * @notice Default token manager implementations should ignore id * @param sender Swapper * @param id Token grouping id (token id or edition id) * @param newTokenManager New token manager being swapped to * @return If invocation can swap token managers */ function canSwap( address sender, uint256 id, address newTokenManager ) external returns (bool); /** * @notice Returns whether token manager can be removed * @notice Default token manager implementations should ignore id * @param sender Swapper * @param id Token grouping id (token id or edition id) * @return If invocation can remove token manager */ function canRemoveItself(address sender, uint256 id) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol) pragma solidity 0.8.10; import "@openzeppelin/contracts/utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ /* solhint-disable */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions 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 (last updated v4.6.0) (interfaces/IERC2981Upgradeable.sol) pragma solidity 0.8.10; import "../ERC165/IERC165Upgradeable.sol"; /** * @dev Interface for the NFT Royalty Standard. * * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal * support for royalty payments across all NFT marketplaces and ecosystem participants. * * _Available since v4.5._ */ interface IERC2981Upgradeable is IERC165Upgradeable { /** * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of * exchange. The royalty amount is denominated and should be paid in that same unit of exchange. */ function royaltyInfo(uint256 tokenId, uint256 salePrice) external view returns (address receiver, uint256 royaltyAmount); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165Checker.sol) pragma solidity 0.8.10; import "./IERC165Upgradeable.sol"; /** * @dev Library used to query support of an interface declared via {IERC165}. * * Note that these functions return the actual result of the query: they do not * `revert` if an interface is not supported. It is up to the caller to decide * what to do in these cases. */ /* solhint-disable */ library ERC165CheckerUpgradeable { // As per the EIP-165 spec, no interface should ever match 0xffffffff bytes4 private constant _INTERFACE_ID_INVALID = 0xffffffff; /** * @dev Returns true if `account` supports the {IERC165} interface, */ function supportsERC165(address account) internal view returns (bool) { // Any contract that implements ERC165 must explicitly indicate support of // InterfaceId_ERC165 and explicitly indicate non-support of InterfaceId_Invalid return supportsERC165InterfaceUnchecked(account, type(IERC165Upgradeable).interfaceId) && !supportsERC165InterfaceUnchecked(account, _INTERFACE_ID_INVALID); } /** * @dev Returns true if `account` supports the interface defined by * `interfaceId`. Support for {IERC165} itself is queried automatically. * * See {IERC165-supportsInterface}. */ function supportsInterface(address account, bytes4 interfaceId) internal view returns (bool) { // query support of both ERC165 as per the spec and support of _interfaceId return supportsERC165(account) && supportsERC165InterfaceUnchecked(account, interfaceId); } /** * @dev Returns a boolean array where each value corresponds to the * interfaces passed in and whether they're supported or not. This allows * you to batch check interfaces for a contract where your expectation * is that some interfaces may not be supported. * * See {IERC165-supportsInterface}. * * _Available since v3.4._ */ function getSupportedInterfaces(address account, bytes4[] memory interfaceIds) internal view returns (bool[] memory) { // an array of booleans corresponding to interfaceIds and whether they're supported or not bool[] memory interfaceIdsSupported = new bool[](interfaceIds.length); // query support of ERC165 itself if (supportsERC165(account)) { // query support of each interface in interfaceIds for (uint256 i = 0; i < interfaceIds.length; i++) { interfaceIdsSupported[i] = supportsERC165InterfaceUnchecked(account, interfaceIds[i]); } } return interfaceIdsSupported; } /** * @dev Returns true if `account` supports all the interfaces defined in * `interfaceIds`. Support for {IERC165} itself is queried automatically. * * Batch-querying can lead to gas savings by skipping repeated checks for * {IERC165} support. * * See {IERC165-supportsInterface}. */ function supportsAllInterfaces(address account, bytes4[] memory interfaceIds) internal view returns (bool) { // query support of ERC165 itself if (!supportsERC165(account)) { return false; } // query support of each interface in _interfaceIds for (uint256 i = 0; i < interfaceIds.length; i++) { if (!supportsERC165InterfaceUnchecked(account, interfaceIds[i])) { return false; } } // all interfaces supported return true; } /** * @notice Query if a contract implements an interface, does not check ERC165 support * @param account The address of the contract to query for support of an interface * @param interfaceId The interface identifier, as specified in ERC-165 * @return true if the contract at account indicates support of the interface with * identifier interfaceId, false otherwise * @dev Assumes that account contains a contract that supports ERC165, otherwise * the behavior of this method is undefined. This precondition can be checked * with {supportsERC165}. * Interface identification is specified in ERC-165. */ function supportsERC165InterfaceUnchecked(address account, bytes4 interfaceId) internal view returns (bool) { // prepare call bytes memory encodedParams = abi.encodeWithSelector(IERC165Upgradeable.supportsInterface.selector, interfaceId); // perform static call bool success; uint256 returnSize; uint256 returnValue; assembly { success := staticcall(30000, account, add(encodedParams, 0x20), mload(encodedParams), 0x00, 0x20) returnSize := returndatasize() returnValue := mload(0x00) } return success && returnSize >= 0x20 && returnValue > 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (metatx/ERC2771Context.sol) pragma solidity 0.8.10; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; /** * @dev Context variant with ERC2771 support. * Openzeppelin contract slightly modified by ishan@ highlight.xyz to be upgradeable. */ abstract contract ERC2771ContextUpgradeable is Initializable { address private _trustedForwarder; /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; function isTrustedForwarder(address forwarder) public view virtual returns (bool) { return forwarder == _trustedForwarder; } function __ERC2771ContextUpgradeable__init__(address trustedForwarder) internal onlyInitializing { _trustedForwarder = trustedForwarder; } function _msgSender() internal view virtual returns (address sender) { if (isTrustedForwarder(msg.sender)) { // The assembly code is more direct than the Solidity version using `abi.decode`. /* solhint-disable no-inline-assembly */ assembly { sender := shr(96, calldataload(sub(calldatasize(), 20))) } /* solhint-enable no-inline-assembly */ } else { return msg.sender; } } function _msgData() internal view virtual returns (bytes calldata) { if (isTrustedForwarder(msg.sender)) { return msg.data[:msg.data.length - 20]; } else { return msg.data; } } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "../royaltyManager/interfaces/IRoyaltyManager.sol"; /** * @title IObservability * @author highlight.xyz * @notice Interface to interact with the Highlight observability singleton * @dev Singleton to coalesce select Highlight protocol events */ interface IObservability { /************************** ERC721Base / ERC721MinimizedBase events **************************/ /** * @notice Emitted when minter is registered or unregistered * @param contractAddress Initial contract that emitted event * @param minter Minter that was changed * @param registered True if the minter was registered, false if unregistered */ event MinterRegistrationChanged(address indexed contractAddress, address indexed minter, bool indexed registered); /** * @notice Emitted when token managers are set for token/edition ids * @param contractAddress Initial contract that emitted event * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ event GranularTokenManagersSet(address indexed contractAddress, uint256[] _ids, address[] _tokenManagers); /** * @notice Emitted when token managers are removed for token/edition ids * @param contractAddress Initial contract that emitted event * @param _ids Edition / token ids to remove token managers for */ event GranularTokenManagersRemoved(address indexed contractAddress, uint256[] _ids); /** * @notice Emitted when default token manager changed * @param contractAddress Initial contract that emitted event * @param newDefaultTokenManager New default token manager. Zero address if old one was removed */ event DefaultTokenManagerChanged(address indexed contractAddress, address indexed newDefaultTokenManager); /** * @notice Emitted when default royalty is set * @param contractAddress Initial contract that emitted event * @param recipientAddress Royalty recipient * @param royaltyPercentageBPS Percentage of sale (in basis points) owed to royalty recipient */ event DefaultRoyaltySet( address indexed contractAddress, address indexed recipientAddress, uint16 indexed royaltyPercentageBPS ); /** * @notice Emitted when royalties are set for edition / token ids * @param contractAddress Initial contract that emitted event * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ event GranularRoyaltiesSet(address indexed contractAddress, uint256[] ids, IRoyaltyManager.Royalty[] _newRoyalties); /** * @notice Emitted when royalty manager is updated * @param contractAddress Initial contract that emitted event * @param newRoyaltyManager New royalty manager. Zero address if old one was removed */ event RoyaltyManagerChanged(address indexed contractAddress, address indexed newRoyaltyManager); /** * @notice Emitted when mints are frozen permanently * @param contractAddress Initial contract that emitted event */ event MintsFrozen(address indexed contractAddress); /** * @notice Emitted when contract metadata is set * @param contractAddress Initial contract that emitted event * @param name New name * @param symbol New symbol * @param contractURI New contract uri */ event ContractMetadataSet(address indexed contractAddress, string name, string symbol, string contractURI); /************************** ERC721General events **************************/ /** * @notice Emitted when hashed metadata config is set * @param contractAddress Initial contract that emitted event * @param hashedURIData Hashed uri data * @param hashedRotationData Hashed rotation key * @param _supply Supply of tokens to mint w/ reveal */ event HashedMetadataConfigSet( address indexed contractAddress, bytes hashedURIData, bytes hashedRotationData, uint256 indexed _supply ); /** * @notice Emitted when metadata is revealed * @param contractAddress Initial contract that emitted event * @param key Key used to decode hashed data * @param newRotationKey Actual rotation key to be used */ event Revealed(address indexed contractAddress, bytes key, uint256 newRotationKey); /************************** ERC721GeneralBase events **************************/ /** * @notice Emitted when uris are set for tokens * @param contractAddress Initial contract that emitted event * @param ids IDs of tokens to set uris for * @param uris Uris to set on tokens */ event TokenURIsSet(address indexed contractAddress, uint256[] ids, string[] uris); /** * @notice Emitted when limit supply is set * @param contractAddress Initial contract that emitted event * @param newLimitSupply Limit supply to set */ event LimitSupplySet(address indexed contractAddress, uint256 indexed newLimitSupply); /************************** ERC721StorageUri events **************************/ /** * @notice Emits when a series collection has its base uri set * @param contractAddress Contract with updated base uri * @param newBaseUri New base uri */ event BaseUriSet(address indexed contractAddress, string newBaseUri); /************************** ERC721Editions / ERC721SingleEdition events **************************/ // Not adding EditionCreated, EditionMintedToOneRecipient, EditionMintedToRecipients // EditionCreated - handled by MetadataInitialized // EditionMintedToOneRecipient / EditionMintedToRecipients - handled via mint module events /************************** Deployment events **************************/ /** * @notice Emitted when Generative Series contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event GenerativeSeriesDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when Series contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event SeriesDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when MultipleEditions contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event MultipleEditionsDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when SingleEdition contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event SingleEditionDeployed(address indexed deployer, address indexed contractAddress); /************************** ERC721 events **************************/ /** * @notice Emitted when `tokenId` token is transferred from `from` to `to` on contractAddress * @param contractAddress NFT contract token resides on * @param from Token sender * @param to Token receiver * @param tokenId Token being sent */ event Transfer(address indexed contractAddress, address indexed from, address to, uint256 indexed tokenId); /** * @notice Emit MinterRegistrationChanged */ function emitMinterRegistrationChanged(address minter, bool registered) external; /** * @notice Emit GranularTokenManagersSet */ function emitGranularTokenManagersSet(uint256[] calldata _ids, address[] calldata _tokenManagers) external; /** * @notice Emit GranularTokenManagersRemoved */ function emitGranularTokenManagersRemoved(uint256[] calldata _ids) external; /** * @notice Emit DefaultTokenManagerChanged */ function emitDefaultTokenManagerChanged(address newDefaultTokenManager) external; /** * @notice Emit DefaultRoyaltySet */ function emitDefaultRoyaltySet(address recipientAddress, uint16 royaltyPercentageBPS) external; /** * @notice Emit GranularRoyaltiesSet */ function emitGranularRoyaltiesSet(uint256[] calldata ids, IRoyaltyManager.Royalty[] calldata _newRoyalties) external; /** * @notice Emit RoyaltyManagerChanged */ function emitRoyaltyManagerChanged(address newRoyaltyManager) external; /** * @notice Emit MintsFrozen */ function emitMintsFrozen() external; /** * @notice Emit ContractMetadataSet */ function emitContractMetadataSet( string calldata name, string calldata symbol, string calldata contractURI ) external; /** * @notice Emit HashedMetadataConfigSet */ function emitHashedMetadataConfigSet( bytes calldata hashedURIData, bytes calldata hashedRotationData, uint256 _supply ) external; /** * @notice Emit Revealed */ function emitRevealed(bytes calldata key, uint256 newRotationKey) external; /** * @notice Emit TokenURIsSet */ function emitTokenURIsSet(uint256[] calldata ids, string[] calldata uris) external; /** * @notice Emit LimitSupplySet */ function emitLimitSupplySet(uint256 newLimitSupply) external; /** * @notice Emit BaseUriSet */ function emitBaseUriSet(string calldata newBaseUri) external; /** * @notice Emit GenerativeSeriesDeployed */ function emitGenerativeSeriesDeployed(address contractAddress) external; /** * @notice Emit SeriesDeployed */ function emitSeriesDeployed(address contractAddress) external; /** * @notice Emit MultipleEditionsDeployed */ function emitMultipleEditionsDeployed(address contractAddress) external; /** * @notice Emit SingleEditionDeployed */ function emitSingleEditionDeployed(address contractAddress) external; /** * @notice Emit Transfer */ function emitTransfer( address from, address to, uint256 tokenId ) external; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/structs/EnumerableSet.sol) pragma solidity ^0.8.0; /** * @dev Library for managing * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive * types. * * Sets have the following properties: * * - Elements are added, removed, and checked for existence in constant time * (O(1)). * - Elements are enumerated in O(n). No guarantees are made on the ordering. * * ``` * contract Example { * // Add the library methods * using EnumerableSet for EnumerableSet.AddressSet; * * // Declare a set state variable * EnumerableSet.AddressSet private mySet; * } * ``` * * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`) * and `uint256` (`UintSet`) are supported. * * [WARNING] * ==== * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure unusable. * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info. * * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an array of EnumerableSet. * ==== */ library EnumerableSet { // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Set type with // bytes32 values. // The Set implementation uses private functions, and user-facing // implementations (such as AddressSet) are just wrappers around the // underlying Set. // This means that we can only create new EnumerableSets for types that fit // in bytes32. struct Set { // Storage of set values bytes32[] _values; // Position of the value in the `values` array, plus 1 because index 0 // means a value is not in the set. mapping(bytes32 => uint256) _indexes; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); // The value is stored at length-1, but we add 1 to all indexes // and use 0 as a sentinel value set._indexes[value] = set._values.length; return true; } else { return false; } } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function _remove(Set storage set, bytes32 value) private returns (bool) { // We read and store the value's index to prevent multiple reads from the same storage slot uint256 valueIndex = set._indexes[value]; if (valueIndex != 0) { // Equivalent to contains(set, value) // To delete an element from the _values array in O(1), we swap the element to delete with the last one in // the array, and then remove the last element (sometimes called as 'swap and pop'). // This modifies the order of the array, as noted in {at}. uint256 toDeleteIndex = valueIndex - 1; uint256 lastIndex = set._values.length - 1; if (lastIndex != toDeleteIndex) { bytes32 lastValue = set._values[lastIndex]; // Move the last value to the index where the value to delete is set._values[toDeleteIndex] = lastValue; // Update the index for the moved value set._indexes[lastValue] = valueIndex; // Replace lastValue's index to valueIndex } // Delete the slot where the moved value was stored set._values.pop(); // Delete the index for the deleted slot delete set._indexes[value]; return true; } else { return false; } } /** * @dev Returns true if the value is in the set. O(1). */ function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._indexes[value] != 0; } /** * @dev Returns the number of values on the set. O(1). */ function _length(Set storage set) private view returns (uint256) { return set._values.length; } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // Bytes32Set struct Bytes32Set { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _add(set._inner, value); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _remove(set._inner, value); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) { return _contains(set._inner, value); } /** * @dev Returns the number of values in the set. O(1). */ function length(Bytes32Set storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) { return _at(set._inner, index); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(Bytes32Set storage set) internal view returns (bytes32[] memory) { return _values(set._inner); } // AddressSet struct AddressSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // UintSet struct UintSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(UintSet storage set, uint256 value) internal returns (bool) { return _add(set._inner, bytes32(value)); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(UintSet storage set, uint256 value) internal returns (bool) { return _remove(set._inner, bytes32(value)); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(UintSet storage set, uint256 value) internal view returns (bool) { return _contains(set._inner, bytes32(value)); } /** * @dev Returns the number of values on the set. O(1). */ function length(UintSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintSet storage set, uint256 index) internal view returns (uint256) { return uint256(_at(set._inner, index)); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(UintSet storage set) internal view returns (uint256[] memory) { bytes32[] memory store = _values(set._inner); uint256[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/ContextUpgradeable.sol"; import "../proxy/utils/Initializable.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 OwnableUpgradeable is Initializable, ContextUpgradeable { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ function __Ownable_init() internal onlyInitializing { __Ownable_init_unchained(); } function __Ownable_init_unchained() internal onlyInitializing { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions 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); } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol) pragma solidity ^0.8.0; import "../proxy/utils/Initializable.sol"; /** * @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 ReentrancyGuardUpgradeable is Initializable { // 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; function __ReentrancyGuard_init() internal onlyInitializing { __ReentrancyGuard_init_unchained(); } function __ReentrancyGuard_init_unchained() internal onlyInitializing { _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; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// 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 (utils/introspection/IERC165.sol) pragma solidity 0.8.10; /** * @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}. */ /* solhint-disable */ interface IERC165Upgradeable { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (proxy/utils/Initializable.sol) pragma solidity ^0.8.2; import "../../utils/AddressUpgradeable.sol"; /** * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect. * * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in * case an upgrade adds a module that needs to be initialized. * * For example: * * [.hljs-theme-light.nopadding] * ``` * contract MyToken is ERC20Upgradeable { * function initialize() initializer public { * __ERC20_init("MyToken", "MTK"); * } * } * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable { * function initializeV2() reinitializer(2) public { * __ERC20Permit_init("MyToken"); * } * } * ``` * * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}. * * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity. * * [CAUTION] * ==== * Avoid leaving a contract uninitialized. * * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed: * * [.hljs-theme-light.nopadding] * ``` * /// @custom:oz-upgrades-unsafe-allow constructor * constructor() { * _disableInitializers(); * } * ``` * ==== */ abstract contract Initializable { /** * @dev Indicates that the contract has been initialized. * @custom:oz-retyped-from bool */ uint8 private _initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool private _initializing; /** * @dev Triggered when the contract has been initialized or reinitialized. */ event Initialized(uint8 version); /** * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope, * `onlyInitializing` functions can be used to initialize parent contracts. Equivalent to `reinitializer(1)`. */ modifier initializer() { bool isTopLevelCall = !_initializing; require( (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1), "Initializable: contract is already initialized" ); _initialized = 1; if (isTopLevelCall) { _initializing = true; } _; if (isTopLevelCall) { _initializing = false; emit Initialized(1); } } /** * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be * used to initialize parent contracts. * * `initializer` is equivalent to `reinitializer(1)`, so a reinitializer may be used after the original * initialization step. This is essential to configure modules that are added through upgrades and that require * initialization. * * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in * a contract, executing them in the right order is up to the developer or operator. */ modifier reinitializer(uint8 version) { require(!_initializing && _initialized < version, "Initializable: contract is already initialized"); _initialized = version; _initializing = true; _; _initializing = false; emit Initialized(version); } /** * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the * {initializer} and {reinitializer} modifiers, directly or indirectly. */ modifier onlyInitializing() { require(_initializing, "Initializable: contract is not initializing"); _; } /** * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call. * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized * to any version. It is recommended to use this to lock implementation contracts that are designed to be called * through proxies. */ function _disableInitializers() internal virtual { require(!_initializing, "Initializable: contract is initializing"); if (_initialized < type(uint8).max) { _initialized = type(uint8).max; emit Initialized(type(uint8).max); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library AddressUpgradeable { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://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 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 /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; import "../proxy/utils/Initializable.sol"; /** * @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 ContextUpgradeable is Initializable { function __Context_init() internal onlyInitializing { } function __Context_init_unchained() internal onlyInitializing { } function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[50] private __gap; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @dev Utilities for metadata encryption and decryption * @author highlight.xyz */ abstract contract MetadataEncryption { /// @dev See: https://ethereum.stackexchange.com/questions/69825/decrypt-message-on-chain function encryptDecrypt(bytes memory data, bytes calldata key) public pure returns (bytes memory result) { // Store data length on stack for later use uint256 length = data.length; // solhint-disable-next-line no-inline-assembly assembly { // Set result to free memory pointer result := mload(0x40) // Increase free memory pointer by lenght + 32 mstore(0x40, add(add(result, length), 32)) // Set result length mstore(result, length) } // Iterate over the data stepping by 32 bytes for (uint256 i = 0; i < length; i += 32) { // Generate hash of the key and offset bytes32 hash = keccak256(abi.encodePacked(key, i)); bytes32 chunk; // solhint-disable-next-line no-inline-assembly assembly { // Read 32-bytes data chunk chunk := mload(add(data, add(i, 32))) } // XOR the chunk with hash chunk ^= hash; // solhint-disable-next-line no-inline-assembly assembly { // Write 32-byte encrypted chunk mstore(add(result, add(i, 32)), chunk) } } } function _sliceUint(bytes memory bs, uint256 start) internal pure returns (uint256) { require(bs.length >= start + 32, "slicing out of range"); uint256 x; assembly { x := mload(add(bs, add(0x20, start))) } return x; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/extensions/ERC721URIStorage.sol) pragma solidity 0.8.10; import "../../erc721/erc721a/ERC721AUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol"; /** * @title Appending URI storage utilities onto template ERC721 contract * @author highlight.xyz and OpenZeppelin * @dev ERC721 token with storage based token URI management. OpenZeppelin template edited by Highlight */ /* solhint-disable */ abstract contract ERC721URIStorageUpgradeable is Initializable, ERC721AUpgradeable { /** * @notice Throw when token doesn't exist */ error TokenDoesNotExist(); function __ERC721URIStorage_init() internal onlyInitializing {} function __ERC721URIStorage_init_unchained() internal onlyInitializing {} using StringsUpgradeable for uint256; // Optional mapping for token URIs mapping(uint256 => string) internal _tokenURIs; /** * @dev Hashed rotation key data */ bytes internal _hashedRotationKeyData; /** * @dev Hashed base uri data */ bytes internal _hashedBaseURIData; /** * @dev Rotation key */ uint256 internal _rotationKey; /** * @dev Contract baseURI */ string public baseURI; /** @notice Emitted when base uri is set * @param oldBaseUri Old base uri * @param newBaseURI New base uri */ event BaseURISet(string oldBaseUri, string newBaseURI); /** * @dev Set contract baseURI */ function _setBaseURI(string memory newBaseURI) internal { emit BaseURISet(baseURI, newBaseURI); baseURI = newBaseURI; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) { _revert(TokenDoesNotExist.selector); } string memory _tokenURI = _tokenURIs[tokenId]; // If there is no token URI, return the base URI. if (bytes(_tokenURI).length == 0) { return super.tokenURI(tokenId); } return _tokenURI; } /** * @dev See {ERC721-_burn}. This override additionally checks to see if a * token-specific URI was set for the token, and if so, it deletes the token URI from * the storage mapping. */ function _burn(uint256 tokenId) internal virtual override { super._burn(tokenId); if (bytes(_tokenURIs[tokenId]).length != 0) { delete _tokenURIs[tokenId]; } } function _baseURI() internal view override returns (string memory) { return baseURI; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity 0.8.10; /** solhint-disable */ import "./IERC721AUpgradeable.sol"; import { ERC721AStorage } from "./ERC721AStorage.sol"; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol"; /** * @dev Interface of ERC721 token receiver. */ interface ERC721A__IERC721ReceiverUpgradeable { function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } /** * @title ERC721AHL * @ author: Chiru Labs, modified by [email protected] * (mainly to store mint quantity data in extra bits per packed ownership) * * @dev Implementation of the [ERC721](https://eips.ethereum.org/EIPS/eip-721) * Non-Fungible Token Standard, including the Metadata extension. * Optimized for lower gas during batch mints. * * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...) * starting from `_startTokenId()`. * * Assumptions: * * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply. * - The maximum token ID cannot exceed 2**256 - 1 (max value of uint256). */ contract ERC721AUpgradeable is Initializable, IERC721AUpgradeable { using ERC721AStorage for ERC721AStorage.Layout; using StringsUpgradeable for uint256; // ============================================================= // CONSTANTS // ============================================================= // Mask of an entry in packed address data. uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1; // The bit position of `numberMinted` in packed address data. uint256 private constant _BITPOS_NUMBER_MINTED = 64; // The bit position of `numberBurned` in packed address data. uint256 private constant _BITPOS_NUMBER_BURNED = 128; // The bit position of `aux` in packed address data. uint256 private constant _BITPOS_AUX = 192; // Mask of all 256 bits in packed address data except the 64 bits for `aux`. uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1; // The bit position of `startTimestamp` in packed ownership. uint256 private constant _BITPOS_START_TIMESTAMP = 160; // The bit mask of the `burned` bit in packed ownership. uint256 private constant _BITMASK_BURNED = 1 << 224; // The bit position of the `nextInitialized` bit in packed ownership. uint256 private constant _BITPOS_NEXT_INITIALIZED = 225; // The bit mask of the `nextInitialized` bit in packed ownership. uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225; // The bit position of `extraData` in packed ownership. uint256 private constant _BITPOS_EXTRA_DATA = 232; // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`. uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1; // The mask of the lower 160 bits for addresses. uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1; uint256 private constant _UINT_24_MAX = 16777216; // The `Transfer` event signature is given by: // `keccak256(bytes("Transfer(address,address,uint256)"))`. bytes32 private constant _TRANSFER_EVENT_SIGNATURE = 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef; // ============================================================= // CONSTRUCTOR // ============================================================= function __ERC721A_init(string memory name_, string memory symbol_) internal onlyInitializing { __ERC721A_init_unchained(name_, symbol_); } function __ERC721A_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing { ERC721AStorage.layout()._name = name_; ERC721AStorage.layout()._symbol = symbol_; ERC721AStorage.layout()._currentIndex = _startTokenId(); } /** * @dev Hook that is called after a set of serially-ordered token IDs * have been transferred. * * `startTokenId` - the first token ID to be transferred. * * Calling conditions: * * - `from` and `to` are both non-zero, since this is only invoked ont transfers */ function _afterTokenTransfers( address from, address to, uint256 tokenId ) internal virtual {} /** * @dev Added by Highlight to facilitate updating of name and symbol */ function _setContractMetadata(string calldata newName, string calldata newSymbol) internal { ERC721AStorage.layout()._name = newName; ERC721AStorage.layout()._symbol = newSymbol; } // ============================================================= // TOKEN COUNTING OPERATIONS // ============================================================= /** * @dev Returns the starting token ID. * To change the starting token ID, please override this function. */ function _startTokenId() internal view virtual returns (uint256) { return 1; } /** * @dev Returns the next token ID to be minted. */ function _nextTokenId() internal view virtual returns (uint256) { return ERC721AStorage.layout()._currentIndex; } /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() external view virtual override returns (uint256) { // Counter underflow is impossible as _burnCounter cannot be incremented // more than `_currentIndex - _startTokenId()` times. unchecked { return ERC721AStorage.layout()._currentIndex - ERC721AStorage.layout()._burnCounter - _startTokenId(); } } /** * @dev Returns the total amount of tokens minted in the contract. */ function _totalMinted() internal view virtual returns (uint256) { // Counter underflow is impossible as `_currentIndex` does not decrement, // and it is initialized to `_startTokenId()`. unchecked { return ERC721AStorage.layout()._currentIndex - _startTokenId(); } } /** * @dev Returns the total number of tokens burned. */ function _totalBurned() internal view virtual returns (uint256) { return ERC721AStorage.layout()._burnCounter; } // ============================================================= // ADDRESS DATA OPERATIONS // ============================================================= /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) public view virtual override returns (uint256) { if (owner == address(0)) _revert(BalanceQueryForZeroAddress.selector); return ERC721AStorage.layout()._packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens minted by `owner`. */ function _numberMinted(address owner) internal view returns (uint256) { return (ERC721AStorage.layout()._packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens burned by or on behalf of `owner`. */ function _numberBurned(address owner) internal view returns (uint256) { return (ERC721AStorage.layout()._packedAddressData[owner] >> _BITPOS_NUMBER_BURNED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). */ function _getAux(address owner) internal view returns (uint64) { return uint64(ERC721AStorage.layout()._packedAddressData[owner] >> _BITPOS_AUX); } /** * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). * If there are multiple variables, please pack them into a uint64. */ function _setAux(address owner, uint64 aux) internal virtual { uint256 packed = ERC721AStorage.layout()._packedAddressData[owner]; uint256 auxCasted; // Cast `aux` with assembly to avoid redundant masking. assembly { auxCasted := aux } packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX); ERC721AStorage.layout()._packedAddressData[owner] = packed; } // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { // The interface IDs are constants representing the first 4 bytes // of the XOR of all function selectors in the interface. // See: [ERC165](https://eips.ethereum.org/EIPS/eip-165) // (e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`) return interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165. interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721. interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata. } // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() public view virtual override returns (string memory) { return ERC721AStorage.layout()._name; } /** * @dev Returns the token collection symbol. */ function symbol() public view virtual override returns (string memory) { return ERC721AStorage.layout()._symbol; } /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) _revert(URIQueryForNonexistentToken.selector); string memory baseURI = _baseURI(); return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, "/", tokenId.toString())) : ""; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, it can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ""; } // ============================================================= // OWNERSHIPS OPERATIONS // ============================================================= /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { return address(uint160(_packedOwnershipOf(tokenId))); } /** * @dev Gas spent here starts off proportional to the maximum mint batch size. * It gradually moves to O(1) as tokens get transferred around over time. */ function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnershipOf(tokenId)); } /** * @dev Returns the unpacked `TokenOwnership` struct at `index`. */ function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(ERC721AStorage.layout()._packedOwnerships[index]); } /** * @dev Returns whether the ownership slot at `index` is initialized. * An uninitialized slot does not necessarily mean that the slot has no owner. */ function _ownershipIsInitialized(uint256 index) internal view virtual returns (bool) { return ERC721AStorage.layout()._packedOwnerships[index] != 0; } /** * @dev Initializes the ownership slot minted at `index` for efficiency purposes. */ function _initializeOwnershipAt(uint256 index) internal virtual { if (ERC721AStorage.layout()._packedOwnerships[index] == 0) { ERC721AStorage.layout()._packedOwnerships[index] = _packedOwnershipOf(index); } } /* solhint-disable code-complexity */ /** * Returns the packed ownership data of `tokenId`. */ function _packedOwnershipOf(uint256 tokenId) private view returns (uint256 packed) { if (_startTokenId() <= tokenId) { packed = ERC721AStorage.layout()._packedOwnerships[tokenId]; // If the data at the starting slot does not exist, start the scan. if (packed == 0) { // adds run-time, potentially redundant, remove soon if (!_exists(tokenId)) { _revert(OwnerQueryForNonexistentToken.selector); } // current index tracks number of tokens, but we could mint certain tokens with different ids // if (tokenId >= ERC721AStorage.layout()._currentIndex) // _revert(OwnerQueryForNonexistentToken.selector); // Invariant: // There will always be an initialized ownership slot // (i.e. `ownership.addr != address(0) && ownership.burned == false`) // before an unintialized ownership slot // (i.e. `ownership.addr == address(0) && ownership.burned == false`) // Hence, `tokenId` will not underflow. // We can directly compare the packed value. // If the address is zero, packed will be zero. for (;;) { unchecked { if (tokenId == 0) { _revert(OwnerQueryForNonexistentToken.selector); } packed = ERC721AStorage.layout()._packedOwnerships[--tokenId]; } if (packed == 0) continue; if (packed & _BITMASK_BURNED == 0) return packed; // Otherwise, the token is burned, and we must revert. // This handles the case of batch burned tokens, where only the burned bit // of the starting slot is set, and remaining slots are left uninitialized. _revert(OwnerQueryForNonexistentToken.selector); } } // Otherwise, the data exists and we can skip the scan. // This is possible because we have already achieved the target condition. // This saves 2143 gas on transfers of initialized tokens. // If the token is not burned, return `packed`. Otherwise, revert. if (packed & _BITMASK_BURNED == 0) return packed; } _revert(OwnerQueryForNonexistentToken.selector); } /* solhint-enable code-complexity */ /** * @dev Returns the unpacked `TokenOwnership` struct from `packed`. */ function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) { ownership.addr = address(uint160(packed)); ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP); ownership.burned = packed & _BITMASK_BURNED != 0; ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA); } /** * @dev Packs ownership data into a single uint256. */ function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // `owner | (block.timestamp << _BITPOS_START_TIMESTAMP) | flags`. result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags)) } } /** * @dev Returns the `nextInitialized` flag set if `quantity` equals 1. */ function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) { // For branchless setting of the `nextInitialized` flag. assembly { // `(quantity == 1) << _BITPOS_NEXT_INITIALIZED`. result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1)) } } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. See {ERC721A-_approve}. * * Requirements: * * - The caller must own the token or be an approved operator. */ function approve(address to, uint256 tokenId) public payable virtual override { _approve(to, tokenId, true); } /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { if (!_exists(tokenId)) _revert(ApprovalQueryForNonexistentToken.selector); return ERC721AStorage.layout()._tokenApprovals[tokenId].value; } /** * @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) public virtual override { ERC721AStorage.layout()._operatorApprovals[_msgSenderERC721A()][operator] = approved; emit ApprovalForAll(_msgSenderERC721A(), operator, approved); } /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return ERC721AStorage.layout()._operatorApprovals[owner][operator]; } function _minted(uint256 tokenId) internal view virtual returns (bool result) { if (_startTokenId() <= tokenId) { uint256 initialTokenId = tokenId; uint24 mintQuantity; while ( (mintQuantity = uint24((ERC721AStorage.layout()._packedOwnerships[tokenId]) >> _BITPOS_EXTRA_DATA)) == 0 ) { // edited to avoid underflow if (tokenId == 0) { return false; } --tokenId; } // if the mint quantity at the number we landed at does extend to the initially queried tokenId, // the initially queried tokenId has been minted if (uint24(initialTokenId - tokenId) < mintQuantity) { return true; } } } /** * @dev Returns whether `tokenId` exists. * * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}. * * Tokens start existing when they are minted. See {_mint}. */ function _exists(uint256 tokenId) internal view virtual returns (bool result) { if (_startTokenId() <= tokenId) { uint256 packed = ERC721AStorage.layout()._packedOwnerships[tokenId]; // not burned, and minted (calculate burn first, to avoid minted call if token has been burned) result = ((packed & _BITMASK_BURNED) == 0) && _minted(tokenId); } } /** * @dev Returns whether `spender` is allowed to manage `tokenId`. * * Requirements: * * - `tokenId` must exist. */ function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) { address owner = ownerOf(tokenId); return (isApprovedForAll(owner, spender) || _isSenderApprovedOrOwner(getApproved(tokenId), owner, spender)); } /** * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`. */ function _isSenderApprovedOrOwner( address approvedAddress, address owner, address msgSender ) private pure returns (bool result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean. msgSender := and(msgSender, _BITMASK_ADDRESS) // `msgSender == owner || msgSender == approvedAddress`. result := or(eq(msgSender, owner), eq(msgSender, approvedAddress)) } } /** * @dev Returns the storage slot and value for the approved address of `tokenId`. */ function _getApprovedSlotAndAddress(uint256 tokenId) private view returns (uint256 approvedAddressSlot, address approvedAddress) { ERC721AStorage.TokenApprovalRef storage tokenApproval = ERC721AStorage.layout()._tokenApprovals[tokenId]; // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId].value`. assembly { approvedAddressSlot := tokenApproval.slot approvedAddress := sload(approvedAddressSlot) } } // ============================================================= // TRANSFER OPERATIONS // ============================================================= /** * @dev Transfers `tokenId` from `from` to `to`. * * 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 ) public payable virtual override { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); // Mask `from` to the lower 160 bits, in case the upper bits somehow aren't clean. from = address(uint160(uint256(uint160(from)) & _BITMASK_ADDRESS)); if (address(uint160(prevOwnershipPacked)) != from) _revert(TransferFromIncorrectOwner.selector); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // We can directly increment and decrement the balances. --ERC721AStorage.layout()._packedAddressData[from]; // Updates: `balance -= 1`. ++ERC721AStorage.layout()._packedAddressData[to]; // Updates: `balance += 1`. // Updates: // - `address` to the next owner. // - `startTimestamp` to the timestamp of transfering. // - `burned` to `false`. // - `nextInitialized` to `true`. ERC721AStorage.layout()._packedOwnerships[tokenId] = _packOwnershipData( to, _BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked, 0) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (ERC721AStorage.layout()._packedOwnerships[nextTokenId] == 0) { // If the next slot exists if (_minted(nextTokenId)) { // remove extra data from this // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. // remove mint quantity to preserve original mint spots ERC721AStorage.layout()._packedOwnerships[nextTokenId] = _removeExtraData(prevOwnershipPacked); } } } } // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. from, // `from`. toMasked, // `to`. tokenId // `tokenId`. ) } if (toMasked == 0) _revert(TransferToZeroAddress.selector); _afterTokenTransfers(from, to, tokenId); } /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) public payable virtual override { safeTransferFrom(from, to, tokenId, ""); } /** * @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 memory _data ) public payable virtual override { transferFrom(from, to, tokenId); if (to.code.length != 0) if (!_checkContractOnERC721Received(from, to, tokenId, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } } /** * @dev Private function to invoke {IERC721Receiver-onERC721Received} on a target contract. * * `from` - Previous owner of the given token ID. * `to` - Target address that will receive the token. * `tokenId` - Token ID to be transferred. * `_data` - Optional data to send along with the call. * * Returns whether the call correctly returned the expected magic value. */ function _checkContractOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { try ERC721A__IERC721ReceiverUpgradeable(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns (bytes4 retval) { return retval == ERC721A__IERC721ReceiverUpgradeable(to).onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { _revert(TransferToNonERC721ReceiverImplementer.selector); } assembly { revert(add(32, reason), mload(reason)) } } } // ============================================================= // MINT OPERATIONS // ============================================================= /** * @dev Mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {Transfer} event for each mint. */ function _mint( address to, uint256 quantity, uint256 startTokenId, uint256 currentIndex ) internal virtual { if (quantity >= _UINT_24_MAX) { _revert(QuantityTooLarge.selector); } if (currentIndex == 0) { currentIndex = ERC721AStorage.layout()._currentIndex; } if (startTokenId == 0) { startTokenId = currentIndex; } else { // only path that selects token ids erratically (collector's choice series) runs _mint per token, // so no need to check _minted for consecutive token ids here if (_minted(startTokenId)) { _revert(TokenMintedAlready.selector); } } if (quantity == 0) _revert(MintZeroQuantity.selector); // Overflows are incredibly unrealistic. // `balance` and `numberMinted` have a maximum limit of 2**64. // `tokenId` has a maximum limit of 2**256. unchecked { // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. ERC721AStorage.layout()._packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0, uint24(quantity)) ); // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. ERC721AStorage.layout()._packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; if (toMasked == 0) _revert(MintToZeroAddress.selector); uint256 end = startTokenId + quantity; uint256 tokenId = startTokenId; do { assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. 0, // `address(0)`. toMasked, // `to`. tokenId // `tokenId`. ) } // The `!=` check ensures that large values of `quantity` // that overflows uint256 will make the loop run out of gas. } while (++tokenId != end); ERC721AStorage.layout()._currentIndex = currentIndex + quantity; } } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Equivalent to `_approve(to, tokenId, false)`. */ function _approve(address to, uint256 tokenId) internal virtual { _approve(to, tokenId, false); } /** * @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: * * - `tokenId` must exist. * * Emits an {Approval} event. */ function _approve( address to, uint256 tokenId, bool approvalCheck ) internal virtual { address owner = ownerOf(tokenId); if (approvalCheck && _msgSenderERC721A() != owner) if (!isApprovedForAll(owner, _msgSenderERC721A())) { _revert(ApprovalCallerNotOwnerNorApproved.selector); } ERC721AStorage.layout()._tokenApprovals[tokenId].value = to; emit Approval(owner, to, tokenId); } // ============================================================= // BURN OPERATIONS // ============================================================= /** * @dev Equivalent to `_burn(tokenId, false)`. */ function _burn(uint256 tokenId) internal virtual { _burn(tokenId, false); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId, bool approvalCheck) internal virtual { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); address from = address(uint160(prevOwnershipPacked)); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); if (approvalCheck) { // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); } // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // Updates: // - `balance -= 1`. // - `numberBurned += 1`. // // We can directly decrement the balance, and increment the number burned. // This is equivalent to `packed -= 1; packed += 1 << _BITPOS_NUMBER_BURNED;`. ERC721AStorage.layout()._packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1; // Updates: // - `address` to the last owner. // - `startTimestamp` to the timestamp of burning. // - `burned` to `true`. // - `nextInitialized` to `true`. ERC721AStorage.layout()._packedOwnerships[tokenId] = _packOwnershipData( from, (_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked, 0) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (ERC721AStorage.layout()._packedOwnerships[nextTokenId] == 0) { // If the next slot exists if (_minted(nextTokenId)) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. // remove mint quantity to preserve original mint spots ERC721AStorage.layout()._packedOwnerships[nextTokenId] = _removeExtraData(prevOwnershipPacked); } } } } emit Transfer(from, address(0), tokenId); // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times. unchecked { ERC721AStorage.layout()._burnCounter++; } } // ============================================================= // EXTRA DATA OPERATIONS // ============================================================= /** * @dev Remove the extra data from a packed ownership, used to preserve original mint slots */ function _removeExtraData(uint256 packedOwnershipData) internal pure returns (uint256) { return packedOwnershipData & _BITMASK_EXTRA_DATA_COMPLEMENT; } /** * @dev Directly sets the extra data for the ownership data `index`. */ function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual { uint256 packed = ERC721AStorage.layout()._packedOwnerships[index]; if (packed == 0) _revert(OwnershipNotInitializedForExtraData.selector); uint256 extraDataCasted; // Cast `extraData` with assembly to avoid redundant masking. assembly { extraDataCasted := extraData } packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA); ERC721AStorage.layout()._packedOwnerships[index] = packed; } /** * @dev Called during each token transfer to set the 24bit `extraData` field. * Intended to be overridden by the cosumer contract. * * `previousExtraData` - the value of `extraData` before transfer. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _extraData( address from, address to, uint24 previousExtraData, uint24 quantityMinted ) internal view virtual returns (uint24) { if (from == address(0) && to != address(0)) { // mint return quantityMinted; } else { return previousExtraData; } } /** * @dev Returns the next extra data for the packed ownership data. * The returned result is shifted into position. */ function _nextExtraData( address from, address to, uint256 prevOwnershipPacked, uint24 quantityMinted ) private view returns (uint256) { uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA); return uint256(_extraData(from, to, extraData, quantityMinted)) << _BITPOS_EXTRA_DATA; } // ============================================================= // OTHER OPERATIONS // ============================================================= /** * @dev Returns the message sender (defaults to `msg.sender`). * * If you are writing GSN compatible contracts, you need to override this function. */ function _msgSenderERC721A() internal view virtual returns (address) { return msg.sender; } /** * @dev Converts a uint256 to its ASCII string decimal representation. */ function _toString(uint256 value) internal pure virtual returns (string memory str) { assembly { // The maximum value of a uint256 contains 78 digits (1 byte per digit), but // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned. // We will need 1 word for the trailing zeros padding, 1 word for the length, // and 3 words for a maximum of 78 digits. Total: 5 * 0x20 = 0xa0. let m := add(mload(0x40), 0xa0) // Update the free memory pointer to allocate. mstore(0x40, m) // Assign the `str` to the end. str := sub(m, 0x20) // Zeroize the slot after the string. mstore(str, 0) // Cache the end of the memory to calculate the length later. let end := str // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. // prettier-ignore for { let temp := value } 1 {} { str := sub(str, 1) // Write the character to the pointer. // The ASCII index of the '0' character is 48. mstore8(str, add(48, mod(temp, 10))) // Keep dividing `temp` until zero. temp := div(temp, 10) // prettier-ignore if iszero(temp) { break } } let length := sub(end, str) // Move the pointer 32 bytes leftwards to make room for the length. str := sub(str, 0x20) // Store the length. mstore(str, length) } } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library StringsUpgradeable { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }
// SPDX-License-Identifier: MIT // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity 0.8.10; /** solhint-disable */ /** * @dev Interface of ERC721A. * @ author Chiru Labs, slightly modified by [email protected] */ interface IERC721AUpgradeable { /** * Quantity to mint is too large */ error QuantityTooLarge(); /** * Cannot re-mint token */ error TokenMintedAlready(); /** * The caller must own the token or be an approved operator. */ error ApprovalCallerNotOwnerNorApproved(); /** * The token does not exist. */ error ApprovalQueryForNonexistentToken(); /** * Cannot query the balance for the zero address. */ error BalanceQueryForZeroAddress(); /** * Cannot mint to the zero address. */ error MintToZeroAddress(); /** * The quantity of tokens minted must be more than zero. */ error MintZeroQuantity(); /** * The token does not exist. */ error OwnerQueryForNonexistentToken(); /** * The caller must own the token or be an approved operator. */ error TransferCallerNotOwnerNorApproved(); /** * The token must be owned by `from`. */ error TransferFromIncorrectOwner(); /** * Cannot safely transfer to a contract that does not implement the * ERC721Receiver interface. */ error TransferToNonERC721ReceiverImplementer(); /** * Cannot transfer to the zero address. */ error TransferToZeroAddress(); /** * The token does not exist. */ error URIQueryForNonexistentToken(); /** * The `quantity` minted with ERC2309 exceeds the safety limit. */ error MintERC2309QuantityExceedsLimit(); /** * The `extraData` cannot be set on an unintialized ownership slot. */ error OwnershipNotInitializedForExtraData(); // ============================================================= // STRUCTS // ============================================================= struct TokenOwnership { // The address of the owner. address addr; // Stores the start time of ownership with minimal overhead for tokenomics. uint64 startTimestamp; // Whether the token has been burned. bool burned; // Arbitrary data similar to `startTimestamp` that can be set via {_extraData}. uint24 extraData; } // ============================================================= // TOKEN COUNTERS // ============================================================= /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() external view returns (uint256); // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); // ============================================================= // IERC721 // ============================================================= /** * @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, bytes calldata data ) external payable; /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external payable; /** * @dev Transfers `tokenId` 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 payable; /** * @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 payable; /** * @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 the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @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); // ============================================================= // IERC721Metadata // ============================================================= /** * @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.10; /** solhint-disable */ library ERC721AStorage { // Bypass for a `--via-ir` bug (https://github.com/chiru-labs/ERC721A/pull/364). struct TokenApprovalRef { address value; } struct Layout { // ============================================================= // STORAGE // ============================================================= // The next token ID to be minted. uint256 _currentIndex; // The number of tokens burned. uint256 _burnCounter; // Token name string _name; // Token symbol string _symbol; // Mapping from token ID to ownership details // An empty struct value does not necessarily mean the token is unowned. // See {_packedOwnershipOf} implementation for details. // // Bits Layout: // - [0..159] `addr` // - [160..223] `startTimestamp` // - [224] `burned` // - [225] `nextInitialized` // - [232..255] `extraData` mapping(uint256 => uint256) _packedOwnerships; // Mapping owner address to address data. // // Bits Layout: // - [0..63] `balance` // - [64..127] `numberMinted` // - [128..191] `numberBurned` // - [192..255] `aux` mapping(address => uint256) _packedAddressData; // Mapping from token ID to approved address. mapping(uint256 => ERC721AStorage.TokenApprovalRef) _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) _operatorApprovals; } bytes32 internal constant _STORAGE_SLOT = keccak256("ERC721A.contracts.storage.ERC721A"); function layout() internal pure returns (Layout storage l) { bytes32 slot = _STORAGE_SLOT; assembly { l.slot := slot } } }
{ "metadata": { "bytecodeHash": "none" }, "optimizer": { "enabled": true, "runs": 1 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"EmptyString","type":"error"},{"inputs":[],"name":"InvalidManager","type":"error"},{"inputs":[],"name":"ManagerDoesNotExist","type":"error"},{"inputs":[],"name":"ManagerRemoveBlocked","type":"error"},{"inputs":[],"name":"ManagerSwapBlocked","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintFrozen","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"MinterRegistrationInvalid","type":"error"},{"inputs":[],"name":"MismatchedArrayLengths","type":"error"},{"inputs":[],"name":"NotMinter","type":"error"},{"inputs":[],"name":"OverLimitSupply","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[],"name":"QuantityTooLarge","type":"error"},{"inputs":[],"name":"RoyaltyBPSInvalid","type":"error"},{"inputs":[],"name":"RoyaltySetBlocked","type":"error"},{"inputs":[],"name":"TokenDoesNotExist","type":"error"},{"inputs":[],"name":"TokenMintedAlready","type":"error"},{"inputs":[],"name":"TokenNotInRange","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"Unauthorized","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"string","name":"oldBaseUri","type":"string"},{"indexed":false,"internalType":"string","name":"newBaseURI","type":"string"}],"name":"BaseURISet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"recipientAddress","type":"address"},{"indexed":true,"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"name":"DefaultRoyaltySet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newDefaultTokenManager","type":"address"}],"name":"DefaultTokenManagerChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"indexed":false,"internalType":"struct IRoyaltyManager.Royalty[]","name":"_newRoyalties","type":"tuple[]"}],"name":"GranularRoyaltiesSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"_ids","type":"uint256[]"}],"name":"GranularTokenManagersRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"indexed":false,"internalType":"address[]","name":"_tokenManagers","type":"address[]"}],"name":"GranularTokenManagersSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"newLimitSupply","type":"uint256"}],"name":"LimitSupplySet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"minter","type":"address"},{"indexed":true,"internalType":"bool","name":"registered","type":"bool"}],"name":"MinterRegistrationChanged","type":"event"},{"anonymous":false,"inputs":[],"name":"MintsFrozen","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":"newRoyaltyManager","type":"address"}],"name":"RoyaltyManagerChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"indexed":false,"internalType":"string[]","name":"uris","type":"string[]"}],"name":"TokenURIsSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"defaultManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"freezeMints","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"generativeCodeUri","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"address","name":"_observability","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"forwarder","type":"address"}],"name":"isTrustedForwarder","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"limitSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintAmountToOneRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"}],"name":"mintOneToMultipleRecipients","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"}],"name":"mintOneToOneRecipient","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintSameAmountToMultipleRecipients","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"mintSpecificTokenToOneRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"}],"name":"mintSpecificTokensToOneRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"minters","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"observability","outputs":[{"internalType":"contract IObservability","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"minter","type":"address"}],"name":"registerMinter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"removeDefaultTokenManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_ids","type":"uint256[]"}],"name":"removeGranularTokenManagers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"removeRoyaltyManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenGroupingId","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint256","name":"royaltyAmount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"royaltyManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newBaseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newName","type":"string"},{"internalType":"string","name":"newSymbol","type":"string"},{"internalType":"string","name":"newContractUri","type":"string"}],"name":"setContractMetadata","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty","name":"_royalty","type":"tuple"}],"name":"setDefaultRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_defaultTokenManager","type":"address"}],"name":"setDefaultTokenManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty[]","name":"_newRoyalties","type":"tuple[]"}],"name":"setGranularRoyalties","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"internalType":"address[]","name":"_tokenManagers","type":"address[]"}],"name":"setGranularTokenManagers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_limitSupply","type":"uint256"}],"name":"setLimitSupply","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_royaltyManager","type":"address"}],"name":"setRoyaltyManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"string[]","name":"uris","type":"string[]"}],"name":"setTokenURIs","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":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"chainId","type":"uint256"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"string","name":"walletAddress","type":"string"},{"internalType":"string","name":"contractAddress","type":"string"},{"internalType":"string","name":"hash","type":"string"},{"internalType":"string","name":"blockHash","type":"string"},{"internalType":"uint256","name":"blockNumber","type":"uint256"},{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"uint256","name":"gasPrice","type":"uint256"}],"name":"tokenAnimation","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"string","name":"_owner","type":"string"}],"name":"tokenImage","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"id","type":"uint256"}],"name":"tokenManager","outputs":[{"internalType":"address","name":"manager","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"minter","type":"address"}],"name":"unregisterMinter","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
608060405234801561001057600080fd5b50615ff480620000216000396000f3fe60806040526004361061025a5760003560e01c806301ffc9a71461025f578063063a75021461029457806306fdde03146102ab578063081812fc146102cd578063095ea7b3146102fa5780630d640e811461030d57806318160ddd1461032d578063184a94d51461035057806318bae6c81461037057806323b872dd146103905780632a55205a146103a35780632a5ac764146103d15780632ddcb21f146103f15780632fc3a8d9146104085780633713c1a11461042857806342842e0e1461044857806342966c681461045b57806347230dcb1461047b5780634f58122a1461049b57806350cf5a31146104bb57806355f804b3146104db578063572b6c05146104fb5780635be954481461051b5780636352211e1461053b5780636371fbe71461055b5780636502abea1461057b5780636c0360eb1461059b5780636f33f145146105b057806370a08231146105c5578063715018a6146105e5578063834a20df146105fa57806389a0e03b1461061a5780638da5cb5b1461063a57806395d89b411461064f5780639b66be2b14610664578063a22cb46514610684578063a305f5b2146106a4578063b5c5801e146106c4578063b88d4fde146106d9578063bfb2995c146106ec578063c2f507011461070c578063c87b56dd14610721578063cce2df0314610741578063db3e4c8414610761578063e8a3d48514610781578063e985e9c514610796578063eb872150146107b6578063ee295d62146107d6578063eeab0589146107f6578063ef60ceaf14610816578063f2fde38b14610836578063f7441c3a14610856578063f97b57ec14610876575b600080fd5b34801561026b57600080fd5b5061027f61027a3660046149de565b610898565b60405190151581526020015b60405180910390f35b3480156102a057600080fd5b506102a96108a9565b005b3480156102b757600080fd5b506102c061098d565b60405161028b9190614a53565b3480156102d957600080fd5b506102ed6102e8366004614a66565b610a28565b60405161028b9190614a7f565b6102a9610308366004614aa8565b610a6c565b34801561031957600080fd5b506102a9610328366004614aa8565b610a7c565b34801561033957600080fd5b50610342610b2a565b60405190815260200161028b565b34801561035c57600080fd5b5061034261036b366004614ad4565b610b4a565b34801561037c57600080fd5b5060cb546102ed906001600160a01b031681565b6102a961039e366004614af1565b610be5565b3480156103af57600080fd5b506103c36103be366004614b32565b610da2565b60405161028b929190614b54565b3480156103dd57600080fd5b506102a96103ec366004614bb5565b610e3d565b3480156103fd57600080fd5b506103426101085481565b34801561041457600080fd5b506102a9610423366004614c92565b610ed3565b34801561043457600080fd5b506102a9610443366004614a66565b610fc7565b6102a9610456366004614af1565b611082565b34801561046757600080fd5b506102a9610476366004614a66565b6110a2565b34801561048757600080fd5b506102a9610496366004614ad4565b61127c565b3480156104a757600080fd5b506102a96104b6366004614cdd565b611330565b3480156104c757600080fd5b506102a96104d6366004614ad4565b61156d565b3480156104e757600080fd5b506102a96104f6366004614d1e565b6116ef565b34801561050757600080fd5b5061027f610516366004614ad4565b611879565b34801561052757600080fd5b506102a9610536366004614aa8565b61188d565b34801561054757600080fd5b506102ed610556366004614a66565b61192a565b34801561056757600080fd5b506102a9610576366004614d53565b611935565b34801561058757600080fd5b506102ed610596366004614a66565b611a91565b3480156105a757600080fd5b506102c0611ac0565b3480156105bc57600080fd5b506102c0611b4e565b3480156105d157600080fd5b506103426105e0366004614ad4565b611b5e565b3480156105f157600080fd5b506102a9611bbd565b34801561060657600080fd5b506102a9610615366004614ad4565b611bd1565b34801561062657600080fd5b506102c0610635366004614e52565b611c4f565b34801561064657600080fd5b506102ed611ebc565b34801561065b57600080fd5b506102c0611ecb565b34801561067057600080fd5b506102c061067f366004614e86565b611ee3565b34801561069057600080fd5b506102a961069f366004614f7c565b611f31565b3480156106b057600080fd5b506102a96106bf366004614ad4565b611fae565b3480156106d057600080fd5b506102a9612132565b6102a96106e7366004614fb5565b612296565b3480156106f857600080fd5b506102a9610707366004615034565b6122d7565b34801561071857600080fd5b506102a9612618565b34801561072d57600080fd5b506102c061073c366004614a66565b612749565b34801561074d57600080fd5b506102a961075c3660046150cd565b612754565b34801561076d57600080fd5b506102a961077c366004615123565b6129a5565b34801561078d57600080fd5b506102c0612af9565b3480156107a257600080fd5b5061027f6107b136600461518e565b612b07565b3480156107c257600080fd5b5060d0546102ed906001600160a01b031681565b3480156107e257600080fd5b5060cf546102ed906001600160a01b031681565b34801561080257600080fd5b506102a9610811366004615123565b612b44565b34801561082257600080fd5b506102a96108313660046151bc565b612e2f565b34801561084257600080fd5b506102a9610851366004614ad4565b613023565b34801561086257600080fd5b506102a9610871366004614cdd565b61309c565b34801561088257600080fd5b5061088b613182565b60405161028b91906151ce565b60006108a382613193565b92915050565b6108b16131e1565b600260975414156108dd5760405162461bcd60e51b81526004016108d49061521b565b60405180910390fd5b600260975560cf805460ff60a01b1916600160a01b1790556040517f9d4b38394dff663b46f53f52ced584161b5021180321b381f6104325eecebf4990600090a160d060009054906101000a90046001600160a01b03166001600160a01b031663fd120bd26040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561096e57600080fd5b505af1158015610982573d6000803e3d6000fd5b505060016097555050565b6060610997613250565b60020180546109a590615252565b80601f01602080910402602001604051908101604052809291908181526020018280546109d190615252565b8015610a1e5780601f106109f357610100808354040283529160200191610a1e565b820191906000526020600020905b815481529060010190602001808311610a0157829003601f168201915b5050505050905090565b6000610a3382613274565b610a4757610a476333d1c03960e21b6132be565b610a4f613250565b60009283526006016020525060409020546001600160a01b031690565b610a78828260016132c7565b5050565b610a8f610a87613378565b60c990613382565b610aa357610aa3633e34a41b60e21b6132be565b60026097541415610ac65760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415610af057610af06345a9661760e01b6132be565b610108548015610b125780821115610b1257610b12633740d80b60e21b6132be565b610b2083600184600061339e565b5050600160975550565b60006001610b36613250565b60010154610b42613250565b540303919050565b6000610b57610a87613378565b610b6b57610b6b633e34a41b60e21b6132be565b60026097541415610b8e5760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415610bb857610bb86345a9661760e01b6132be565b6000610bc26134cb565b9050610bcd816134db565b610bda836001838461339e565b600160975592915050565b6000610bf082613502565b6001600160a01b039485169490915081168414610c1657610c1662a1148160e81b6132be565b600080610c22846135d3565b91509150610c378187610c323390565b6135fb565b610c5957610c458633612b07565b610c5957610c59632ce44b5f60e11b6132be565b8015610c6457600082555b610c6c613250565b6001600160a01b0387166000908152600591909101602052604090208054600019019055610c98613250565b6001600160a01b03861660009081526005919091016020526040812080546001019055610cdb908690610cd090899083908890613610565b600160e11b17613637565b610ce3613250565b60008681526004919091016020526040902055600160e11b8316610d595760018401610d0d613250565b60008281526004919091016020526040902054610d5757610d2d8161364c565b15610d5757610d3b846136d0565b610d43613250565b600083815260049190910160205260409020555b505b6001600160a01b038516848188600080516020615fa8833981519152600080a480610d8e57610d8e633a954ecd60e21b6132be565b610d998787876136dc565b50505050505050565b600082815260ce602090815260408083208151808301909252546001600160a01b038116808352600160a01b90910461ffff16928201929092528291610e0d57506040805180820190915260cd546001600160a01b0381168252600160a01b900461ffff1660208201525b8051602082015190935061271090610e299061ffff168661529d565b610e3391906152d2565b9150509250929050565b610e456131e1565b610e518686868661386f565b610e5e6101078383614885565b5060d054604051635bf57bc360e01b81526001600160a01b0390911690635bf57bc390610e999089908990899089908990899060040161530f565b600060405180830381600087803b158015610eb357600080fd5b505af1158015610ec7573d6000803e3d6000fd5b50505050505050505050565b610ede610a87613378565b610ef257610ef2633e34a41b60e21b6132be565b60026097541415610f155760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415610f3f57610f3f6345a9661760e01b6132be565b816000610f4a6134cb565b905060005b82811015610fa757610f89868683818110610f6c57610f6c615358565b9050602002016020810190610f819190614ad4565b85848561339e565b610f93848361536e565b915080610f9f81615386565b915050610f4f565b50610fbb610fb66001836153a1565b6134db565b50506001609755505050565b610fcf6131e1565b60026097541415610ff25760405162461bcd60e51b81526004016108d49061521b565b600260975561010881905560405181907fad7735a1decc3db45c1eab757cc4a1cb4bff8e332877b55f6db4ea11d82320c690600090a260d054604051630941255b60e21b8152600481018390526001600160a01b0390911690632504956c90602401600060405180830381600087803b15801561106e57600080fd5b505af1158015610fbb573d6000803e3d6000fd5b61109d83838360405180602001604052806000815250612296565b505050565b600260975414156110c55760405162461bcd60e51b81526004016108d49061521b565b600260975560006110d582611a91565b905060006110e1613378565b90506001600160a01b0382161580159061116a57506040516301ffc9a760e01b81526001600160a01b038316906301ffc9a7906111299063129dae8b60e01b906004016153b8565b602060405180830381865afa158015611146573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061116a91906153cd565b156111e557600061117a8461192a565b60405163129dae8b60e01b81529091506001600160a01b0384169063129dae8b906111ad908590859089906004016153ea565b600060405180830381600087803b1580156111c757600080fd5b505af11580156111db573d6000803e3d6000fd5b5050505050611202565b6111ef81846138a4565b611202576112026282b42960e81b6132be565b61120b836138d5565b60d0546040516323de665160e01b81526001600160a01b03909116906323de66519061124090849060009088906004016153ea565b600060405180830381600087803b15801561125a57600080fd5b505af115801561126e573d6000803e3d6000fd5b505060016097555050505050565b6112846131e1565b61128f60c982613915565b6112a3576112a363ea6eb5c360e01b6132be565b6040516001906001600160a01b03831690600080516020615e9e83398151915290600090a360d05460405163a9292a6f60e01b81526001600160a01b039091169063a9292a6f906112fb90849060019060040161540e565b600060405180830381600087803b15801561131557600080fd5b505af1158015611329573d6000803e3d6000fd5b5050505050565b600260975414156113535760405162461bcd60e51b81526004016108d49061521b565b60026097556000611362613378565b90508160005b818110156114c457600060cc600087878581811061138857611388615358565b60209081029290920135835250810191909152604001600020546001600160a01b03169050806113c2576113c2635f98be1560e11b6132be565b806001600160a01b031663ce2003a5858888868181106113e4576113e4615358565b905060200201356040518363ffffffff1660e01b8152600401611408929190614b54565b6020604051808303816000875af1158015611427573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061144b91906153cd565b61145f5761145f63251c48e160e11b6132be565b600060cc600088888681811061147757611477615358565b90506020020135815260200190815260200160002060006101000a8154816001600160a01b0302191690836001600160a01b031602179055505080806114bc90615386565b915050611368565b507f4d30c2588339a383e2c3092b5383924cf202ae9c377b76c5fe5208cd41d8d2aa84846040516114f692919061545f565b60405180910390a160d0546040516320364dcd60e21b81526001600160a01b03909116906380d9373490611530908790879060040161545f565b600060405180830381600087803b15801561154a57600080fd5b505af115801561155e573d6000803e3d6000fd5b50506001609755505050505050565b600260975414156115905760405162461bcd60e51b81526004016108d49061521b565b600260975561159e8161392a565b6115b2576115b2631a7387bd60e11b6132be565b60006115bc613378565b60cf549091506001600160a01b031680611604576115d8611ebc565b6001600160a01b0316826001600160a01b0316146115ff576115ff6282b42960e81b6132be565b61168a565b60405163bcdbc94760e01b81526001600160a01b038481166004830152838116602483015282169063bcdbc94790604401602060405180830381865afa158015611652573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061167691906153cd565b61168a5761168a63e10a050560e01b6132be565b60cf80546001600160a01b0319166001600160a01b038516908117909155604051600080516020615fc883398151915290600090a260d05460405163755d5a1160e11b81526001600160a01b039091169063eabab42290611240908690600401614a7f565b600260975414156117125760405162461bcd60e51b81526004016108d49061521b565b60026097558061172c5761172c63ecd7b0d160e01b6132be565b60cb546001600160a01b03168061177857611745611ebc565b6001600160a01b0316611756613378565b6001600160a01b031614611773576117736282b42960e81b6132be565b611808565b806001600160a01b0316633d820a4d61178f613378565b600086866040518563ffffffff1660e01b81526004016117b29493929190615473565b6020604051808303816000875af11580156117d1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906117f591906153cd565b611808576118086282b42960e81b6132be565b61184783838080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061394692505050565b60d054604051632cfd654b60e11b81526001600160a01b03909116906359faca969061124090869086906004016154a5565b6065546001600160a01b0391821691161490565b611898610a87613378565b6118ac576118ac633e34a41b60e21b6132be565b600260975414156118cf5760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff16600114156118f9576118f96345a9661760e01b6132be565b60006119036134cb565b905061191e6001611914848461536e565b610fb691906153a1565b610b208383838461339e565b60006108a382613502565b611940610a87613378565b61195457611954633e34a41b60e21b6132be565b600260975414156119775760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff16600114156119a1576119a16345a9661760e01b6132be565b60006119ab6134cb565b61010854909150829080611a0f5760005b82811015611a09576119e98760018888858181106119dc576119dc615358565b905060200201358761339e565b836119f381615386565b9450508080611a0190615386565b9150506119bc565b50611a84565b60005b82811015611a825781868683818110611a2d57611a2d615358565b905060200201351115611a4a57611a4a633740d80b60e21b6132be565b611a628760018888858181106119dc576119dc615358565b83611a6c81615386565b9450508080611a7a90615386565b915050611a12565b505b5050600160975550505050565b60cb54600082815260cc60205260409020546001600160a01b0391821691168015611aba578091505b50919050565b60d58054611acd90615252565b80601f0160208091040260200160405190810160405280929190818152602001828054611af990615252565b8015611b465780601f10611b1b57610100808354040283529160200191611b46565b820191906000526020600020905b815481529060010190602001808311611b2957829003601f168201915b505050505081565b606061010980546109a590615252565b60006001600160a01b038216611b7e57611b7e6323d3ad8160e21b6132be565b6001600160401b03611b8e613250565b6005016000846001600160a01b03166001600160a01b0316815260200190815260200160002054169050919050565b611bc56131e1565b611bcf6000613993565b565b611bd96131e1565b611be460c9826139e5565b611bf857611bf863ea6eb5c360e01b6132be565b6040516000906001600160a01b03831690600080516020615e9e833981519152908390a360d05460405163a9292a6f60e01b81526001600160a01b039091169063a9292a6f906112fb90849060009060040161540e565b80516060908290602a14611c9c5760405162461bcd60e51b815260206004820152601460248201527357726f6e672077616c6c6574206164647265737360601b60448201526064016108d4565b60006040518060e0016040528060aa8152602001615efe60aa9139905061010e60005b6027811015611e2d57606081611d2b5784611cdb83600261536e565b81518110611ceb57611ceb615358565b016020908101516040516202330360ec1b928101929092526001600160f81b03191660238201526024016040516020818303038152906040529050611dde565b848281518110611d3d57611d3d615358565b01602001516001600160f81b03191685611d5884600161536e565b81518110611d6857611d68615358565b01602001516001600160f81b03191686611d8385600261536e565b81518110611d9357611d93615358565b01602090810151604051602360f81b928101929092526001600160f81b0319938416602183015291831660228201529116602382015260240160405160208183030381529060405290505b83611de8846139fa565b82604051602001611dfb939291906154b9565b60408051601f198184030181529190529350611e186014846153a1565b9250611e2560038361536e565b915050611cbf565b50611eb38284602781518110611e4557611e45615358565b602001015160f81c60f81b85602881518110611e6357611e63615358565b602001015160f81c60f81b86602981518110611e8157611e81615358565b602001015160f81c60f81b604051602001611e9f949392919061554d565b604051602081830303815290604052613af7565b95945050505050565b6033546001600160a01b031690565b6060611ed5613250565b60030180546109a590615252565b6060610109611ef4888a8989613b28565b611f018b8d888888613b5a565b604051602001611f13939291906155e1565b60405160208183030381529060405290509998505050505050505050565b80611f3a613250565b336000818152600792909201602090815260408084206001600160a01b03881680865290835293819020805460ff19169515159590951790945592518415158152919290917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b60026097541415611fd15760405162461bcd60e51b81526004016108d49061521b565b6002609755611fdf81613bb7565b611ff357611ff3631a7387bd60e11b6132be565b6000611ffd613378565b60cb549091506001600160a01b03168061204557612019611ebc565b6001600160a01b0316826001600160a01b031614612040576120406282b42960e81b6132be565b6120cd565b6040516342f19adb60e11b81526001600160a01b038216906385e335b6906120769085906000908890600401615693565b6020604051808303816000875af1158015612095573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120b991906153cd565b6120cd576120cd63e10a050560e01b6132be565b60cb80546001600160a01b0319166001600160a01b038516908117909155604051600080516020615e7e83398151915290600090a260d0546040516316365cdd60e01b81526001600160a01b03909116906316365cdd90611240908690600401614a7f565b600260975414156121555760405162461bcd60e51b81526004016108d49061521b565b60026097556000612164613378565b60cb549091506001600160a01b03168061218857612188635f98be1560e11b6132be565b60405163ce2003a560e01b81526001600160a01b0382169063ce2003a5906121b7908590600090600401614b54565b6020604051808303816000875af11580156121d6573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906121fa91906153cd565b61220e5761220e63251c48e160e11b6132be565b60cb80546001600160a01b0319169055604051600090600080516020615e7e833981519152908290a260d0546040516316365cdd60e01b81526001600160a01b03909116906316365cdd9061226890600090600401614a7f565b600060405180830381600087803b15801561228257600080fd5b505af1158015611a84573d6000803e3d6000fd5b6122a1848484610be5565b6001600160a01b0383163b156122d1576122bd84848484613bd3565b6122d1576122d16368d2bf6b60e11b6132be565b50505050565b600260975414156122fa5760405162461bcd60e51b81526004016108d49061521b565b60026097556000612309613378565b90506000612315611ebc565b60cf549091506001600160a01b0316858161241657826001600160a01b0316846001600160a01b031614612352576123526282b42960e81b6132be565b60005b818110156124105761239887878381811061237257612372615358565b905060400201602001602081019061238a91906156c6565b61271061ffff909116111590565b6123ac576123ac633daf741f60e21b6132be565b8686828181106123be576123be615358565b90506040020160ce60008b8b858181106123da576123da615358565b90506020020135815260200190815260200160002081816123fb91906156e3565b5081905061240881615386565b915050612355565b50612564565b60005b818110156125625761243687878381811061237257612372615358565b61244a5761244a633daf741f60e21b6132be565b826001600160a01b031663ad6e40c78a8a8481811061246b5761246b615358565b9050602002013589898581811061248457612484615358565b905060400201886040518463ffffffff1660e01b81526004016124a993929190615770565b602060405180830381865afa1580156124c6573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906124ea91906153cd565b6124fe576124fe63270e710760e21b6132be565b86868281811061251057612510615358565b90506040020160ce60008b8b8581811061252c5761252c615358565b905060200201358152602001908152602001600020818161254d91906156e3565b5081905061255a81615386565b915050612419565b505b7feaf422d1ab2d8f38856ab6be8378c08d2886463a33667e896ba3a6ebf2c6260c88888888604051612599949392919061579d565b60405180910390a160d054604051632be18fef60e11b81526001600160a01b03909116906357c31fde906125d7908b908b908b908b9060040161579d565b600060405180830381600087803b1580156125f157600080fd5b505af1158015612605573d6000803e3d6000fd5b5050600160975550505050505050505050565b6002609754141561263b5760405162461bcd60e51b81526004016108d49061521b565b6002609755600061264a613378565b60cf549091506001600160a01b03168061266e5761266e635f98be1560e11b6132be565b60405163680c607d60e11b81526001600160a01b0382169063d018c0fa9061269a908590600401614a7f565b602060405180830381865afa1580156126b7573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906126db91906153cd565b6126ef576126ef63251c48e160e11b6132be565b60cf80546001600160a01b0319169055604051600090600080516020615fc8833981519152908290a260d05460405163755d5a1160e11b81526001600160a01b039091169063eabab4229061226890600090600401614a7f565b60606108a382613caf565b600054610100900460ff16158080156127745750600054600160ff909116105b8061278e5750303b15801561278e575060005460ff166001145b6127f15760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b60648201526084016108d4565b6000805460ff191660011790558015612814576000805461ff0019166101001790555b6000806000806000806000806000806000808f8f8101906128359190615862565b9b509b509b509b509b509b509b509b509b509b509b509b50612855613d7a565b6128608c8b8b613da1565b61286986613e7e565b6128738888613ec7565b61287e60c986613915565b508a51612893906101079060208e0190614909565b5083516128a890610109906020870190614909565b5060405163332a79b160e01b81526001600160a01b038f169063332a79b1906128d5903090600401614a7f565b600060405180830381600087803b1580156128ef57600080fd5b505af1158015612903573d6000803e3d6000fd5b505050508d60d060006101000a8154816001600160a01b0302191690836001600160a01b031602179055506000835111156129415761294183613946565b811561294e576101088290555b50505050505050505050505080156122d1576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a150505050565b600260975414156129c85760405162461bcd60e51b81526004016108d49061521b565b6002609755828181146129e5576129e5632b477e7160e11b6132be565b60005b81811015612a4757612a35868683818110612a0557612a05615358565b90506020020135858584818110612a1e57612a1e615358565b9050602002810190612a3091906159c7565b613ef8565b80612a3f81615386565b9150506129e8565b507f2df7dd608aac5806395769062aca937bf08722cb45659d38c0cce82f5cd0cde385858585604051612a7d9493929190615a0d565b60405180910390a160d0546040516374c1ace160e01b81526001600160a01b03909116906374c1ace190612abb908890889088908890600401615a0d565b600060405180830381600087803b158015612ad557600080fd5b505af1158015612ae9573d6000803e3d6000fd5b5050600160975550505050505050565b6101078054611acd90615252565b6000612b11613250565b6001600160a01b039384166000908152600791909101602090815260408083209490951682529290925250205460ff1690565b60026097541415612b675760405162461bcd60e51b81526004016108d49061521b565b60026097556000612b76613378565b90506000612b82611ebc565b90508460005b81811015612d7b57612bbf868683818110612ba557612ba5615358565b9050602002016020810190612bba9190614ad4565b613bb7565b612bd357612bd3631a7387bd60e11b6132be565b6000612bf6898984818110612bea57612bea615358565b90506020020135611a91565b90506001600160a01b038116612c3357836001600160a01b0316856001600160a01b031614612c2e57612c2e6282b42960e81b6132be565b612cf1565b806001600160a01b03166385e335b6868b8b86818110612c5557612c55615358565b600088815260cc60209081526040918290205491516001600160e01b031960e089901b168152612c9a969591909302013592506001600160a01b031690600401615693565b6020604051808303816000875af1158015612cb9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612cdd91906153cd565b612cf157612cf163e10a050560e01b6132be565b868683818110612d0357612d03615358565b9050602002016020810190612d189190614ad4565b60cc60008b8b86818110612d2e57612d2e615358565b90506020020135815260200190815260200160002060006101000a8154816001600160a01b0302191690836001600160a01b03160217905550508080612d7390615386565b915050612b88565b507facc0f6922d8ccd133dbdaee1908d4d734f25a540223d55145e9fad2898471a4e87878787604051612db19493929190615abd565b60405180910390a160d0546040516315bd85bf60e01b81526001600160a01b03909116906315bd85bf90612def908a908a908a908a90600401615abd565b600060405180830381600087803b158015612e0957600080fd5b505af1158015612e1d573d6000803e3d6000fd5b50506001609755505050505050505050565b60026097541415612e525760405162461bcd60e51b81526004016108d49061521b565b6002609755612e6760408201602083016156c6565b61271061ffff82161115612e8557612e85633daf741f60e21b6132be565b6000612e8f613378565b60cf549091506001600160a01b031680612ed757612eab611ebc565b6001600160a01b0316826001600160a01b031614612ed257612ed26282b42960e81b6132be565b612f5a565b60405163049a5c0f60e41b81526001600160a01b038216906349a5c0f090612f059087908690600401615b15565b602060405180830381865afa158015612f22573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612f4691906153cd565b612f5a57612f5a63270e710760e21b6132be565b8360cd612f6782826156e3565b50612f7a905060408501602086016156c6565b61ffff16612f8b6020860186614ad4565b6001600160a01b03167f2c5ea6e4103e78cb101e796fb2dace540362fc542cbff5145eaa24af7dd8fe4160405160405180910390a360d0546001600160a01b031663d10072c0612fde6020870187614ad4565b612fee60408801602089016156c6565b6040516001600160e01b031960e085901b1681526001600160a01b03909216600483015261ffff166024820152604401611530565b61302b6131e1565b6001600160a01b0381166130905760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016108d4565b61309981613993565b50565b6130a7610a87613378565b6130bb576130bb633e34a41b60e21b6132be565b600260975414156130de5760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415613108576131086345a9661760e01b6132be565b8060006131136134cb565b905060005b828110156131735761315385858381811061313557613135615358565b905060200201602081019061314a9190614ad4565b6001848561339e565b8161315d81615386565b925050808061316b90615386565b915050613118565b50610982610fb66001836153a1565b606061318e60c9613ff9565b905090565b60006301ffc9a760e01b6001600160e01b0319831614806131c457506380ac58cd60e01b6001600160e01b03198316145b806108a35750506001600160e01b031916635b5e139f60e01b1490565b6131e9613378565b6001600160a01b03166131fa611ebc565b6001600160a01b031614611bcf5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016108d4565b7f2569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c4090565b6000816001116132b9576000613288613250565b600084815260049190910160205260409020549050600160e01b81161580156132b557506132b58361364c565b9150505b919050565b61309981614006565b60006132d28361192a565b90508180156132ea5750336001600160a01b03821614155b1561330d576132f98133612b07565b61330d5761330d6367d9dca160e11b6132be565b83613316613250565b6000858152600691909101602052604080822080546001600160a01b0319166001600160a01b0394851617905551859287811692908516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259190a450505050565b600061318e614010565b6000613397836001600160a01b038416614032565b9392505050565b630100000083106133b9576133b96312387aa160e01b6132be565b806133ca576133c6613250565b5490505b816133d7578091506133f5565b6133e08261364c565b156133f5576133f563638e967560e11b6132be565b8261340a5761340a63b562e8dd60e01b6132be565b6134298461341c600087600088613610565b6001861460e11b17613637565b613431613250565b600084815260049190910160205260409020556001600160401b018302613456613250565b6001600160a01b0386166000818152600592909201602052604090912080549092019091558061348f5761348f622e076360e81b6132be565b828401835b80836000600080516020615fa8833981519152600080a460010180821415613494578584016134c1613250565b5550505050505050565b60006134d5613250565b54919050565b6101085480158015906134ed57508082115b15610a7857610a7863be1a657d60e01b6132be565b6000816001116135c357613514613250565b600083815260049190910160205260409020549050806135b35761353782613274565b61354b5761354b636f96cda160e11b6132be565b8161356057613560636f96cda160e11b6132be565b613568613250565b600019909201600081815260049390930160205260409092205490508061358e5761354b565b600160e01b811661359e57919050565b6135ae636f96cda160e11b6132be565b61354b565b600160e01b81166135c357919050565b6132b9636f96cda160e11b6132be565b60008060006135e0613250565b60009485526006016020525050604090912080549092909150565b6001600160a01b039081169116811491141790565b600060e883811c906136248787848761404a565b62ffffff16901b9150505b949350505050565b4260a01b176001600160a01b03919091161790565b6000816001116132b9578160005b60e8613664613250565b60008681526004919091016020526040902054901c905062ffffff81166136a45783613694575060009392505050565b61369d84615b3b565b935061365a565b62ffffff81166136b485846153a1565b62ffffff1610156136c9575060019392505050565b5050919050565b6001600160e81b031690565b60006136e6613378565b905060006136f383611a91565b90506001600160a01b0381161580159061377c57506040516301ffc9a760e01b81526001600160a01b038216906301ffc9a79061373b90637af46b4f60e11b906004016153b8565b602060405180830381865afa158015613758573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061377c91906153cd565b1561380257604051638af6791b60e01b81526001600160a01b038381166004830152868116602483015285811660448301526064820185905260a06084830152600060a4830152821690638af6791b9060c401600060405180830381600087803b1580156137e957600080fd5b505af11580156137fd573d6000803e3d6000fd5b505050505b60d0546040516323de665160e01b81526001600160a01b03909116906323de665190613836908890889088906004016153ea565b600060405180830381600087803b15801561385057600080fd5b505af1158015613864573d6000803e3d6000fd5b505050505050505050565b8383613879613250565b6002019190613889929190614885565b508181613894613250565b6003019190611329929190614885565b6000806138b08361192a565b90506138bc8185612b07565b8061362f575061362f6138ce84610a28565b82866135fb565b6138de8161407e565b600081815260d16020526040902080546138f790615252565b15905061309957600081815260d1602052604081206130999161497d565b6000613397836001600160a01b038416614089565b60006108a36001600160a01b03831663440444c560e11b6140d3565b7f2e0a5b969d96a99aee0b35787d9a60516a02ca6f528a5f66d3f936468d8f038260d582604051613978929190615b52565b60405180910390a18051610a789060d5906020840190614909565b603380546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b6000613397836001600160a01b0384166140ef565b606081613a1e5750506040805180820190915260018152600360fc1b602082015290565b8160005b8115613a485780613a3281615386565b9150613a419050600a836152d2565b9150613a22565b6000816001600160401b03811115613a6257613a62614da7565b6040519080825280601f01601f191660200182016040528015613a8c576020820181803683370190505b5090505b841561362f57613aa16001836153a1565b9150613aae600a86615be2565b613ab990603061536e565b60f81b818381518110613ace57613ace615358565b60200101906001600160f81b031916908160001a905350613af0600a866152d2565b9450613a90565b6060613b02826141e2565b604051602001613b129190615bf6565b6040516020818303038152906040529050919050565b606084838386604051602001613b419493929190615c38565b6040516020818303038152906040529050949350505050565b6060613b65856139fa565b613b6e876139fa565b613b77866139fa565b613b80866139fa565b613b89866139fa565b604051602001613b9d959493929190615cd0565b604051602081830303815290604052905095945050505050565b60006108a36001600160a01b038316633b209e2f60e11b6140d3565b604051630a85bd0160e11b81526000906001600160a01b0385169063150b7a0290613c08903390899088908890600401615d90565b6020604051808303816000875af1925050508015613c43575060408051601f3d908101601f19168201909252613c4091810190615dc3565b60015b613c95573d808015613c71576040519150601f19603f3d011682016040523d82523d6000602084013e613c76565b606091505b508051613c8d57613c8d6368d2bf6b60e11b6132be565b805181602001fd5b6001600160e01b031916630a85bd0160e11b14905061362f565b6060613cba82613274565b613cce57613cce63677510db60e11b6132be565b600082815260d1602052604081208054613ce790615252565b80601f0160208091040260200160405190810160405280929190818152602001828054613d1390615252565b8015613d605780601f10613d3557610100808354040283529160200191613d60565b820191906000526020600020905b815481529060010190602001808311613d4357829003601f168201915b505050505090508051600014156108a3576132b583614335565b600054610100900460ff16611bcf5760405162461bcd60e51b81526004016108d490615de0565b600054610100900460ff16613dc85760405162461bcd60e51b81526004016108d490615de0565b602082015161271061ffff82161115613deb57613deb633daf741f60e21b6132be565b613df36143af565b613dfb6143de565b613e0484613993565b60cd546001600160a01b031615613e4c57825160cd8054602086015161ffff16600160a01b026001600160b01b03199091166001600160a01b03909316929092179190911790555b6001600160a01b038216156122d15760cb80546001600160a01b0384166001600160a01b031990911617905550505050565b600054610100900460ff16613ea55760405162461bcd60e51b81526004016108d490615de0565b606580546001600160a01b0319166001600160a01b0392909216919091179055565b600054610100900460ff16613eee5760405162461bcd60e51b81526004016108d490615de0565b610a78828261440d565b6000613f0384611a91565b90506000613f0f613378565b90506000613f1b611ebc565b90506001600160a01b038316613f5857806001600160a01b0316826001600160a01b031614613f5357613f536282b42960e81b6132be565b613fe0565b604051633d820a4d60e01b81526001600160a01b03841690633d820a4d90613f8a9085908a908a908a90600401615473565b6020604051808303816000875af1158015613fa9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613fcd91906153cd565b613fe057613fe06282b42960e81b6132be565b600086815260d160205260409020610d99908686614885565b6060600061339783614483565b8060005260046000fd5b600061401b33611879565b1561402d575060131936013560601c90565b503390565b60009081526001919091016020526040902054151590565b60006001600160a01b03851615801561406b57506001600160a01b03841615155b1561407757508061362f565b508161362f565b6130998160006144df565b60006140958383614032565b6140cb575081546001818101845560008481526020808220909301849055845484825282860190935260409020919091556108a3565b5060006108a3565b60006140de8361464d565b801561339757506133978383614680565b600081815260018301602052604081205480156141d85760006141136001836153a1565b8554909150600090614127906001906153a1565b905081811461418c57600086600001828154811061414757614147615358565b906000526020600020015490508087600001848154811061416a5761416a615358565b6000918252602080832090910192909255918252600188019052604090208390555b855486908061419d5761419d615e2b565b6001900381819060005260206000200160009055905585600101600086815260200190815260200160002060009055600193505050506108a3565b60009150506108a3565b606081516000141561420257505060408051602081019091526000815290565b6000604051806060016040528060408152602001615ebe6040913990506000600384516002614231919061536e565b61423b91906152d2565b61424690600461529d565b6001600160401b0381111561425d5761425d614da7565b6040519080825280601f01601f191660200182016040528015614287576020820181803683370190505b509050600182016020820185865187015b808210156142f3576003820191508151603f8160121c168501518453600184019350603f81600c1c168501518453600184019350603f8160061c168501518453600184019350603f8116850151845350600183019250614298565b505060038651066001811461430f57600281146143225761432a565b603d6001830353603d600283035361432a565b603d60018303535b509195945050505050565b606061434082613274565b61435457614354630a14c4b560e41b6132be565b600061435e614714565b9050600081511161437e57604051806020016040528060008152506132b5565b8061438884614723565b604051602001614399929190615e41565b6040516020818303038152906040529392505050565b600054610100900460ff166143d65760405162461bcd60e51b81526004016108d490615de0565b611bcf614820565b600054610100900460ff166144055760405162461bcd60e51b81526004016108d490615de0565b611bcf614857565b600054610100900460ff166144345760405162461bcd60e51b81526004016108d490615de0565b8161443d613250565b6002019080519060200190614453929190614909565b508061445d613250565b6003019080519060200190614473929190614909565b50600161447e613250565b555050565b6060816000018054806020026020016040519081016040528092919081815260200182805480156144d357602002820191906000526020600020905b8154815260200190600101908083116144bf575b50505050509050919050565b60006144ea83613502565b9050806000806144f9866135d3565b9150915084156145305761450e8184336135fb565b6145305761451c8333612b07565b61453057614530632ce44b5f60e11b6132be565b801561453b57600082555b6001600160801b0361454b613250565b6001600160a01b038516600090815260059190910160205260408120805490920190915561458d9084906145829082908881613610565b600360e01b17613637565b614595613250565b60008881526004919091016020526040902055600160e11b841661460b57600186016145bf613250565b60008281526004919091016020526040902054614609576145df8161364c565b15614609576145ed856136d0565b6145f5613250565b600083815260049190910160205260409020555b505b60405186906000906001600160a01b03861690600080516020615fa8833981519152908390a4614639613250565b600190810180549091019055505050505050565b6000614660826301ffc9a760e01b614680565b80156108a35750614679826001600160e01b0319614680565b1592915050565b6000806301ffc9a760e01b8360405160240161469c91906153b8565b604051602081830303815290604052906001600160e01b0319166020820180516001600160e01b03838183161783525050505090506000806000602060008551602087018a617530fa92503d915060005190508280156146fd575060208210155b80156147095750600081115b979650505050505050565b606060d580546109a590615252565b6060816147475750506040805180820190915260018152600360fc1b602082015290565b8160005b8115614771578061475b81615386565b915061476a9050600a836152d2565b915061474b565b6000816001600160401b0381111561478b5761478b614da7565b6040519080825280601f01601f1916602001820160405280156147b5576020820181803683370190505b5090505b841561362f576147ca6001836153a1565b91506147d7600a86615be2565b6147e290603061536e565b60f81b8183815181106147f7576147f7615358565b60200101906001600160f81b031916908160001a905350614819600a866152d2565b94506147b9565b600054610100900460ff166148475760405162461bcd60e51b81526004016108d490615de0565b611bcf614852613378565b613993565b600054610100900460ff1661487e5760405162461bcd60e51b81526004016108d490615de0565b6001609755565b82805461489190615252565b90600052602060002090601f0160209004810192826148b357600085556148f9565b82601f106148cc5782800160ff198235161785556148f9565b828001600101855582156148f9579182015b828111156148f95782358255916020019190600101906148de565b506149059291506149b3565b5090565b82805461491590615252565b90600052602060002090601f01602090048101928261493757600085556148f9565b82601f1061495057805160ff19168380011785556148f9565b828001600101855582156148f9579182015b828111156148f9578251825591602001919060010190614962565b50805461498990615252565b6000825580601f10614999575050565b601f01602090049060005260206000209081019061309991905b5b8082111561490557600081556001016149b4565b6001600160e01b03198116811461309957600080fd5b6000602082840312156149f057600080fd5b8135613397816149c8565b60005b83811015614a165781810151838201526020016149fe565b838111156122d15750506000910152565b60008151808452614a3f8160208601602086016149fb565b601f01601f19169290920160200192915050565b6020815260006133976020830184614a27565b600060208284031215614a7857600080fd5b5035919050565b6001600160a01b0391909116815260200190565b6001600160a01b038116811461309957600080fd5b60008060408385031215614abb57600080fd5b8235614ac681614a93565b946020939093013593505050565b600060208284031215614ae657600080fd5b813561339781614a93565b600080600060608486031215614b0657600080fd5b8335614b1181614a93565b92506020840135614b2181614a93565b929592945050506040919091013590565b60008060408385031215614b4557600080fd5b50508035926020909101359150565b6001600160a01b03929092168252602082015260400190565b60008083601f840112614b7f57600080fd5b5081356001600160401b03811115614b9657600080fd5b602083019150836020828501011115614bae57600080fd5b9250929050565b60008060008060008060608789031215614bce57600080fd5b86356001600160401b0380821115614be557600080fd5b614bf18a838b01614b6d565b90985096506020890135915080821115614c0a57600080fd5b614c168a838b01614b6d565b90965094506040890135915080821115614c2f57600080fd5b50614c3c89828a01614b6d565b979a9699509497509295939492505050565b60008083601f840112614c6057600080fd5b5081356001600160401b03811115614c7757600080fd5b6020830191508360208260051b8501011115614bae57600080fd5b600080600060408486031215614ca757600080fd5b83356001600160401b03811115614cbd57600080fd5b614cc986828701614c4e565b909790965060209590950135949350505050565b60008060208385031215614cf057600080fd5b82356001600160401b03811115614d0657600080fd5b614d1285828601614c4e565b90969095509350505050565b60008060208385031215614d3157600080fd5b82356001600160401b03811115614d4757600080fd5b614d1285828601614b6d565b600080600060408486031215614d6857600080fd5b8335614d7381614a93565b925060208401356001600160401b03811115614d8e57600080fd5b614d9a86828701614c4e565b9497909650939450505050565b634e487b7160e01b600052604160045260246000fd5b60006001600160401b0380841115614dd757614dd7614da7565b604051601f8501601f19908116603f01168101908282118183101715614dff57614dff614da7565b81604052809350858152868686011115614e1857600080fd5b858560208301376000602087830101525050509392505050565b600082601f830112614e4357600080fd5b61339783833560208501614dbd565b600060208284031215614e6457600080fd5b81356001600160401b03811115614e7a57600080fd5b61362f84828501614e32565b60008060008060008060008060006101208a8c031215614ea557600080fd5b8935985060208a0135975060408a01356001600160401b0380821115614eca57600080fd5b614ed68d838e01614e32565b985060608c0135915080821115614eec57600080fd5b614ef88d838e01614e32565b975060808c0135915080821115614f0e57600080fd5b614f1a8d838e01614e32565b965060a08c0135915080821115614f3057600080fd5b50614f3d8c828d01614e32565b94505060c08a0135925060e08a013591506101008a013590509295985092959850929598565b801515811461309957600080fd5b80356132b981614f63565b60008060408385031215614f8f57600080fd5b8235614f9a81614a93565b91506020830135614faa81614f63565b809150509250929050565b60008060008060808587031215614fcb57600080fd5b8435614fd681614a93565b93506020850135614fe681614a93565b92506040850135915060608501356001600160401b0381111561500857600080fd5b8501601f8101871361501957600080fd5b61502887823560208401614dbd565b91505092959194509250565b6000806000806040858703121561504a57600080fd5b84356001600160401b038082111561506157600080fd5b61506d88838901614c4e565b9096509450602087013591508082111561508657600080fd5b818701915087601f83011261509a57600080fd5b8135818111156150a957600080fd5b8860208260061b85010111156150be57600080fd5b95989497505060200194505050565b6000806000604084860312156150e257600080fd5b83356001600160401b038111156150f857600080fd5b61510486828701614b6d565b909450925050602084013561511881614a93565b809150509250925092565b6000806000806040858703121561513957600080fd5b84356001600160401b038082111561515057600080fd5b61515c88838901614c4e565b9096509450602087013591508082111561517557600080fd5b5061518287828801614c4e565b95989497509550505050565b600080604083850312156151a157600080fd5b82356151ac81614a93565b91506020830135614faa81614a93565b600060408284031215611aba57600080fd5b6020808252825182820181905260009190848201906040850190845b8181101561520f5783516001600160a01b0316835292840192918401916001016151ea565b50909695505050505050565b6020808252601f908201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604082015260600190565b600181811c9082168061526657607f821691505b60208210811415611aba57634e487b7160e01b600052602260045260246000fd5b634e487b7160e01b600052601160045260246000fd5b60008160001904831182151516156152b7576152b7615287565b500290565b634e487b7160e01b600052601260045260246000fd5b6000826152e1576152e16152bc565b500490565b81835281816020850137506000828201602090810191909152601f909101601f19169091010190565b60608152600061532360608301888a6152e6565b82810360208401526153368187896152e6565b9050828103604084015261534b8185876152e6565b9998505050505050505050565b634e487b7160e01b600052603260045260246000fd5b6000821982111561538157615381615287565b500190565b600060001982141561539a5761539a615287565b5060010190565b6000828210156153b3576153b3615287565b500390565b6001600160e01b031991909116815260200190565b6000602082840312156153df57600080fd5b815161339781614f63565b6001600160a01b039384168152919092166020820152604081019190915260600190565b6001600160a01b039290921682521515602082015260400190565b81835260006001600160fb1b0383111561544257600080fd5b8260051b8083602087013760009401602001938452509192915050565b60208152600061362f602083018486615429565b60018060a01b038516815283602082015260606040820152600061549b6060830184866152e6565b9695505050505050565b60208152600061362f6020830184866152e6565b600084516154cb8184602089016149fb565b7f3c636972636c652063783d22333530222063793d223335302220723d22000000908301908152845161550581601d8401602089016149fb565b69111039ba3937b5b29e9160b11b601d929091019182015283516155308160278401602088016149fb565b6211179f60e91b60279290910191820152602a0195945050505050565b6000855161555f818460208a016149fb565b7f3c636972636c652063783d22333530222063793d223335302220723d2231302292019182525069207374726f6b653d222360b01b60208201526001600160f81b0319938416602a820152918316602b830152909116602c8201526211179f60e91b602d820152691e17b39f1e17b9bb339f60b11b6030820152603a01919050565b60008085546155ef81615252565b60018281168015615607576001811461561857615647565b60ff19841687528287019450615647565b8960005260208060002060005b8581101561563e5781548a820152908401908201615625565b50505082870194505b505050506b2f696e6465782e68746d6c3f60a01b8152845161567081600c8401602089016149fb565b845191019061568681600c8401602088016149fb565b01600c0195945050505050565b6001600160a01b0393841681526020810192909252909116604082015260600190565b61ffff8116811461309957600080fd5b6000602082840312156156d857600080fd5b8135613397816156b6565b81356156ee81614a93565b81546001600160a01b031981166001600160a01b03929092169182178355602084013561571a816156b6565b6001600160b01b03199190911690911760a09190911b61ffff60a01b1617905550565b803561574881614a93565b6001600160a01b031682526020810135615761816156b6565b61ffff81166020840152505050565b83815260808101615784602083018561573d565b6001600160a01b03929092166060919091015292915050565b600060408083526157b18184018789615429565b83810360208086019190915285825286910160005b868110156157e8576157d8828461573d565b91830191908301906001016157c6565b5098975050505050505050565b80356132b981614a93565b60006040828403121561581257600080fd5b604080519081016001600160401b038111828210171561583457615834614da7565b604052905080823561584581614a93565b81526020830135615855816156b6565b6020919091015292915050565b6000806000806000806000806000806000806101a08d8f03121561588557600080fd5b61588e8d6157f5565b9b506001600160401b0360208e013511156158a857600080fd5b6158b88e60208f01358f01614e32565b9a506158c78e60408f01615800565b99506158d560808e016157f5565b98506001600160401b0360a08e013511156158ef57600080fd5b6158ff8e60a08f01358f01614e32565b97506001600160401b0360c08e0135111561591957600080fd5b6159298e60c08f01358f01614e32565b965061593760e08e016157f5565b95506159466101008e016157f5565b94506001600160401b036101208e0135111561596157600080fd5b6159728e6101208f01358f01614e32565b93506001600160401b036101408e0135111561598d57600080fd5b61599e8e6101408f01358f01614e32565b92506101608d013591506159b56101808e01614f71565b90509295989b509295989b509295989b565b6000808335601e198436030181126159de57600080fd5b8301803591506001600160401b038211156159f857600080fd5b602001915036819003821315614bae57600080fd5b604081526000615a21604083018688615429565b602083820381850152818583528183019050818660051b8401018760005b88811015615aad57858303601f190184528135368b9003601e19018112615a6557600080fd5b8a0180356001600160401b03811115615a7d57600080fd5b8036038c1315615a8c57600080fd5b615a9985828985016152e6565b958701959450505090840190600101615a3f565b50909a9950505050505050505050565b604081526000615ad1604083018688615429565b8281036020848101919091528482528591810160005b868110156157e8578335615afa81614a93565b6001600160a01b031682529282019290820190600101615ae7565b60608101615b23828561573d565b6001600160a01b039290921660409190910152919050565b600081615b4a57615b4a615287565b506000190190565b604081526000808454615b6481615252565b8060408601526060600180841660008114615b865760018114615b9a57615bcb565b60ff19851688840152608088019550615bcb565b8960005260208060002060005b86811015615bc25781548b8201870152908401908201615ba7565b8a018501975050505b50505050508281036020840152611eb38185614a27565b600082615bf157615bf16152bc565b500690565b7919185d184e9a5b5859d94bdcdd99cade1b5b0ed8985cd94d8d0b60321b815260008251615c2b81601a8501602087016149fb565b91909101601a0192915050565b61613d60f01b815260008551615c55816002850160208a016149fb565b6226683d60e81b6002918401918201528551615c78816005840160208a016149fb565b632662683d60e01b600592909101918201528451615c9d8160098401602089016149fb565b632677613d60e01b600992909101918201528351615cc281600d8401602088016149fb565b01600d019695505050505050565b6226633d60e81b815260008651615cee816003850160208b016149fb565b64267469643d60d81b6003918401918201528651615d13816008840160208b016149fb565b6326626e3d60e01b600892909101918201528551615d3881600c840160208a016149fb565b6226743d60e81b600c92909101918201528451615d5c81600f8401602089016149fb565b632667703d60e01b600f92909101918201528351615d818160138401602088016149fb565b01601301979650505050505050565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061549b90830184614a27565b600060208284031215615dd557600080fd5b8151613397816149c8565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b634e487b7160e01b600052603160045260246000fd5b60008351615e538184602088016149fb565b602f60f81b9083019081528351615e718160018401602088016149fb565b0160010194935050505056fec407dfe559ca4a63d7cf8439b9fce2965d7a4c8c92eb5d5047f53bb9eec295fd73e05f4c0a27cbf60841afa4a2f8577f5e99db3e3e9c293800d6e6b1f7b8a7ab4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2f3c73766720786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f737667222069643d22307865453962653938343661624234393533326130334330354643323741306534446263383465383537222076696577426f783d223020302037303020373030223e3c7374796c653e2e617b7374726f6b652d77696474683a32303b7d3c2f7374796c653e3c6720636c6173733d2261222066696c6c3d226e6f6e65223eddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efe236d251e7da4ac8a3c275f80aebc9b74cbf356425fa04864da5043511a81219a164736f6c634300080a000a
Deployed Bytecode
0x60806040526004361061025a5760003560e01c806301ffc9a71461025f578063063a75021461029457806306fdde03146102ab578063081812fc146102cd578063095ea7b3146102fa5780630d640e811461030d57806318160ddd1461032d578063184a94d51461035057806318bae6c81461037057806323b872dd146103905780632a55205a146103a35780632a5ac764146103d15780632ddcb21f146103f15780632fc3a8d9146104085780633713c1a11461042857806342842e0e1461044857806342966c681461045b57806347230dcb1461047b5780634f58122a1461049b57806350cf5a31146104bb57806355f804b3146104db578063572b6c05146104fb5780635be954481461051b5780636352211e1461053b5780636371fbe71461055b5780636502abea1461057b5780636c0360eb1461059b5780636f33f145146105b057806370a08231146105c5578063715018a6146105e5578063834a20df146105fa57806389a0e03b1461061a5780638da5cb5b1461063a57806395d89b411461064f5780639b66be2b14610664578063a22cb46514610684578063a305f5b2146106a4578063b5c5801e146106c4578063b88d4fde146106d9578063bfb2995c146106ec578063c2f507011461070c578063c87b56dd14610721578063cce2df0314610741578063db3e4c8414610761578063e8a3d48514610781578063e985e9c514610796578063eb872150146107b6578063ee295d62146107d6578063eeab0589146107f6578063ef60ceaf14610816578063f2fde38b14610836578063f7441c3a14610856578063f97b57ec14610876575b600080fd5b34801561026b57600080fd5b5061027f61027a3660046149de565b610898565b60405190151581526020015b60405180910390f35b3480156102a057600080fd5b506102a96108a9565b005b3480156102b757600080fd5b506102c061098d565b60405161028b9190614a53565b3480156102d957600080fd5b506102ed6102e8366004614a66565b610a28565b60405161028b9190614a7f565b6102a9610308366004614aa8565b610a6c565b34801561031957600080fd5b506102a9610328366004614aa8565b610a7c565b34801561033957600080fd5b50610342610b2a565b60405190815260200161028b565b34801561035c57600080fd5b5061034261036b366004614ad4565b610b4a565b34801561037c57600080fd5b5060cb546102ed906001600160a01b031681565b6102a961039e366004614af1565b610be5565b3480156103af57600080fd5b506103c36103be366004614b32565b610da2565b60405161028b929190614b54565b3480156103dd57600080fd5b506102a96103ec366004614bb5565b610e3d565b3480156103fd57600080fd5b506103426101085481565b34801561041457600080fd5b506102a9610423366004614c92565b610ed3565b34801561043457600080fd5b506102a9610443366004614a66565b610fc7565b6102a9610456366004614af1565b611082565b34801561046757600080fd5b506102a9610476366004614a66565b6110a2565b34801561048757600080fd5b506102a9610496366004614ad4565b61127c565b3480156104a757600080fd5b506102a96104b6366004614cdd565b611330565b3480156104c757600080fd5b506102a96104d6366004614ad4565b61156d565b3480156104e757600080fd5b506102a96104f6366004614d1e565b6116ef565b34801561050757600080fd5b5061027f610516366004614ad4565b611879565b34801561052757600080fd5b506102a9610536366004614aa8565b61188d565b34801561054757600080fd5b506102ed610556366004614a66565b61192a565b34801561056757600080fd5b506102a9610576366004614d53565b611935565b34801561058757600080fd5b506102ed610596366004614a66565b611a91565b3480156105a757600080fd5b506102c0611ac0565b3480156105bc57600080fd5b506102c0611b4e565b3480156105d157600080fd5b506103426105e0366004614ad4565b611b5e565b3480156105f157600080fd5b506102a9611bbd565b34801561060657600080fd5b506102a9610615366004614ad4565b611bd1565b34801561062657600080fd5b506102c0610635366004614e52565b611c4f565b34801561064657600080fd5b506102ed611ebc565b34801561065b57600080fd5b506102c0611ecb565b34801561067057600080fd5b506102c061067f366004614e86565b611ee3565b34801561069057600080fd5b506102a961069f366004614f7c565b611f31565b3480156106b057600080fd5b506102a96106bf366004614ad4565b611fae565b3480156106d057600080fd5b506102a9612132565b6102a96106e7366004614fb5565b612296565b3480156106f857600080fd5b506102a9610707366004615034565b6122d7565b34801561071857600080fd5b506102a9612618565b34801561072d57600080fd5b506102c061073c366004614a66565b612749565b34801561074d57600080fd5b506102a961075c3660046150cd565b612754565b34801561076d57600080fd5b506102a961077c366004615123565b6129a5565b34801561078d57600080fd5b506102c0612af9565b3480156107a257600080fd5b5061027f6107b136600461518e565b612b07565b3480156107c257600080fd5b5060d0546102ed906001600160a01b031681565b3480156107e257600080fd5b5060cf546102ed906001600160a01b031681565b34801561080257600080fd5b506102a9610811366004615123565b612b44565b34801561082257600080fd5b506102a96108313660046151bc565b612e2f565b34801561084257600080fd5b506102a9610851366004614ad4565b613023565b34801561086257600080fd5b506102a9610871366004614cdd565b61309c565b34801561088257600080fd5b5061088b613182565b60405161028b91906151ce565b60006108a382613193565b92915050565b6108b16131e1565b600260975414156108dd5760405162461bcd60e51b81526004016108d49061521b565b60405180910390fd5b600260975560cf805460ff60a01b1916600160a01b1790556040517f9d4b38394dff663b46f53f52ced584161b5021180321b381f6104325eecebf4990600090a160d060009054906101000a90046001600160a01b03166001600160a01b031663fd120bd26040518163ffffffff1660e01b8152600401600060405180830381600087803b15801561096e57600080fd5b505af1158015610982573d6000803e3d6000fd5b505060016097555050565b6060610997613250565b60020180546109a590615252565b80601f01602080910402602001604051908101604052809291908181526020018280546109d190615252565b8015610a1e5780601f106109f357610100808354040283529160200191610a1e565b820191906000526020600020905b815481529060010190602001808311610a0157829003601f168201915b5050505050905090565b6000610a3382613274565b610a4757610a476333d1c03960e21b6132be565b610a4f613250565b60009283526006016020525060409020546001600160a01b031690565b610a78828260016132c7565b5050565b610a8f610a87613378565b60c990613382565b610aa357610aa3633e34a41b60e21b6132be565b60026097541415610ac65760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415610af057610af06345a9661760e01b6132be565b610108548015610b125780821115610b1257610b12633740d80b60e21b6132be565b610b2083600184600061339e565b5050600160975550565b60006001610b36613250565b60010154610b42613250565b540303919050565b6000610b57610a87613378565b610b6b57610b6b633e34a41b60e21b6132be565b60026097541415610b8e5760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415610bb857610bb86345a9661760e01b6132be565b6000610bc26134cb565b9050610bcd816134db565b610bda836001838461339e565b600160975592915050565b6000610bf082613502565b6001600160a01b039485169490915081168414610c1657610c1662a1148160e81b6132be565b600080610c22846135d3565b91509150610c378187610c323390565b6135fb565b610c5957610c458633612b07565b610c5957610c59632ce44b5f60e11b6132be565b8015610c6457600082555b610c6c613250565b6001600160a01b0387166000908152600591909101602052604090208054600019019055610c98613250565b6001600160a01b03861660009081526005919091016020526040812080546001019055610cdb908690610cd090899083908890613610565b600160e11b17613637565b610ce3613250565b60008681526004919091016020526040902055600160e11b8316610d595760018401610d0d613250565b60008281526004919091016020526040902054610d5757610d2d8161364c565b15610d5757610d3b846136d0565b610d43613250565b600083815260049190910160205260409020555b505b6001600160a01b038516848188600080516020615fa8833981519152600080a480610d8e57610d8e633a954ecd60e21b6132be565b610d998787876136dc565b50505050505050565b600082815260ce602090815260408083208151808301909252546001600160a01b038116808352600160a01b90910461ffff16928201929092528291610e0d57506040805180820190915260cd546001600160a01b0381168252600160a01b900461ffff1660208201525b8051602082015190935061271090610e299061ffff168661529d565b610e3391906152d2565b9150509250929050565b610e456131e1565b610e518686868661386f565b610e5e6101078383614885565b5060d054604051635bf57bc360e01b81526001600160a01b0390911690635bf57bc390610e999089908990899089908990899060040161530f565b600060405180830381600087803b158015610eb357600080fd5b505af1158015610ec7573d6000803e3d6000fd5b50505050505050505050565b610ede610a87613378565b610ef257610ef2633e34a41b60e21b6132be565b60026097541415610f155760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415610f3f57610f3f6345a9661760e01b6132be565b816000610f4a6134cb565b905060005b82811015610fa757610f89868683818110610f6c57610f6c615358565b9050602002016020810190610f819190614ad4565b85848561339e565b610f93848361536e565b915080610f9f81615386565b915050610f4f565b50610fbb610fb66001836153a1565b6134db565b50506001609755505050565b610fcf6131e1565b60026097541415610ff25760405162461bcd60e51b81526004016108d49061521b565b600260975561010881905560405181907fad7735a1decc3db45c1eab757cc4a1cb4bff8e332877b55f6db4ea11d82320c690600090a260d054604051630941255b60e21b8152600481018390526001600160a01b0390911690632504956c90602401600060405180830381600087803b15801561106e57600080fd5b505af1158015610fbb573d6000803e3d6000fd5b61109d83838360405180602001604052806000815250612296565b505050565b600260975414156110c55760405162461bcd60e51b81526004016108d49061521b565b600260975560006110d582611a91565b905060006110e1613378565b90506001600160a01b0382161580159061116a57506040516301ffc9a760e01b81526001600160a01b038316906301ffc9a7906111299063129dae8b60e01b906004016153b8565b602060405180830381865afa158015611146573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061116a91906153cd565b156111e557600061117a8461192a565b60405163129dae8b60e01b81529091506001600160a01b0384169063129dae8b906111ad908590859089906004016153ea565b600060405180830381600087803b1580156111c757600080fd5b505af11580156111db573d6000803e3d6000fd5b5050505050611202565b6111ef81846138a4565b611202576112026282b42960e81b6132be565b61120b836138d5565b60d0546040516323de665160e01b81526001600160a01b03909116906323de66519061124090849060009088906004016153ea565b600060405180830381600087803b15801561125a57600080fd5b505af115801561126e573d6000803e3d6000fd5b505060016097555050505050565b6112846131e1565b61128f60c982613915565b6112a3576112a363ea6eb5c360e01b6132be565b6040516001906001600160a01b03831690600080516020615e9e83398151915290600090a360d05460405163a9292a6f60e01b81526001600160a01b039091169063a9292a6f906112fb90849060019060040161540e565b600060405180830381600087803b15801561131557600080fd5b505af1158015611329573d6000803e3d6000fd5b5050505050565b600260975414156113535760405162461bcd60e51b81526004016108d49061521b565b60026097556000611362613378565b90508160005b818110156114c457600060cc600087878581811061138857611388615358565b60209081029290920135835250810191909152604001600020546001600160a01b03169050806113c2576113c2635f98be1560e11b6132be565b806001600160a01b031663ce2003a5858888868181106113e4576113e4615358565b905060200201356040518363ffffffff1660e01b8152600401611408929190614b54565b6020604051808303816000875af1158015611427573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061144b91906153cd565b61145f5761145f63251c48e160e11b6132be565b600060cc600088888681811061147757611477615358565b90506020020135815260200190815260200160002060006101000a8154816001600160a01b0302191690836001600160a01b031602179055505080806114bc90615386565b915050611368565b507f4d30c2588339a383e2c3092b5383924cf202ae9c377b76c5fe5208cd41d8d2aa84846040516114f692919061545f565b60405180910390a160d0546040516320364dcd60e21b81526001600160a01b03909116906380d9373490611530908790879060040161545f565b600060405180830381600087803b15801561154a57600080fd5b505af115801561155e573d6000803e3d6000fd5b50506001609755505050505050565b600260975414156115905760405162461bcd60e51b81526004016108d49061521b565b600260975561159e8161392a565b6115b2576115b2631a7387bd60e11b6132be565b60006115bc613378565b60cf549091506001600160a01b031680611604576115d8611ebc565b6001600160a01b0316826001600160a01b0316146115ff576115ff6282b42960e81b6132be565b61168a565b60405163bcdbc94760e01b81526001600160a01b038481166004830152838116602483015282169063bcdbc94790604401602060405180830381865afa158015611652573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061167691906153cd565b61168a5761168a63e10a050560e01b6132be565b60cf80546001600160a01b0319166001600160a01b038516908117909155604051600080516020615fc883398151915290600090a260d05460405163755d5a1160e11b81526001600160a01b039091169063eabab42290611240908690600401614a7f565b600260975414156117125760405162461bcd60e51b81526004016108d49061521b565b60026097558061172c5761172c63ecd7b0d160e01b6132be565b60cb546001600160a01b03168061177857611745611ebc565b6001600160a01b0316611756613378565b6001600160a01b031614611773576117736282b42960e81b6132be565b611808565b806001600160a01b0316633d820a4d61178f613378565b600086866040518563ffffffff1660e01b81526004016117b29493929190615473565b6020604051808303816000875af11580156117d1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906117f591906153cd565b611808576118086282b42960e81b6132be565b61184783838080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061394692505050565b60d054604051632cfd654b60e11b81526001600160a01b03909116906359faca969061124090869086906004016154a5565b6065546001600160a01b0391821691161490565b611898610a87613378565b6118ac576118ac633e34a41b60e21b6132be565b600260975414156118cf5760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff16600114156118f9576118f96345a9661760e01b6132be565b60006119036134cb565b905061191e6001611914848461536e565b610fb691906153a1565b610b208383838461339e565b60006108a382613502565b611940610a87613378565b61195457611954633e34a41b60e21b6132be565b600260975414156119775760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff16600114156119a1576119a16345a9661760e01b6132be565b60006119ab6134cb565b61010854909150829080611a0f5760005b82811015611a09576119e98760018888858181106119dc576119dc615358565b905060200201358761339e565b836119f381615386565b9450508080611a0190615386565b9150506119bc565b50611a84565b60005b82811015611a825781868683818110611a2d57611a2d615358565b905060200201351115611a4a57611a4a633740d80b60e21b6132be565b611a628760018888858181106119dc576119dc615358565b83611a6c81615386565b9450508080611a7a90615386565b915050611a12565b505b5050600160975550505050565b60cb54600082815260cc60205260409020546001600160a01b0391821691168015611aba578091505b50919050565b60d58054611acd90615252565b80601f0160208091040260200160405190810160405280929190818152602001828054611af990615252565b8015611b465780601f10611b1b57610100808354040283529160200191611b46565b820191906000526020600020905b815481529060010190602001808311611b2957829003601f168201915b505050505081565b606061010980546109a590615252565b60006001600160a01b038216611b7e57611b7e6323d3ad8160e21b6132be565b6001600160401b03611b8e613250565b6005016000846001600160a01b03166001600160a01b0316815260200190815260200160002054169050919050565b611bc56131e1565b611bcf6000613993565b565b611bd96131e1565b611be460c9826139e5565b611bf857611bf863ea6eb5c360e01b6132be565b6040516000906001600160a01b03831690600080516020615e9e833981519152908390a360d05460405163a9292a6f60e01b81526001600160a01b039091169063a9292a6f906112fb90849060009060040161540e565b80516060908290602a14611c9c5760405162461bcd60e51b815260206004820152601460248201527357726f6e672077616c6c6574206164647265737360601b60448201526064016108d4565b60006040518060e0016040528060aa8152602001615efe60aa9139905061010e60005b6027811015611e2d57606081611d2b5784611cdb83600261536e565b81518110611ceb57611ceb615358565b016020908101516040516202330360ec1b928101929092526001600160f81b03191660238201526024016040516020818303038152906040529050611dde565b848281518110611d3d57611d3d615358565b01602001516001600160f81b03191685611d5884600161536e565b81518110611d6857611d68615358565b01602001516001600160f81b03191686611d8385600261536e565b81518110611d9357611d93615358565b01602090810151604051602360f81b928101929092526001600160f81b0319938416602183015291831660228201529116602382015260240160405160208183030381529060405290505b83611de8846139fa565b82604051602001611dfb939291906154b9565b60408051601f198184030181529190529350611e186014846153a1565b9250611e2560038361536e565b915050611cbf565b50611eb38284602781518110611e4557611e45615358565b602001015160f81c60f81b85602881518110611e6357611e63615358565b602001015160f81c60f81b86602981518110611e8157611e81615358565b602001015160f81c60f81b604051602001611e9f949392919061554d565b604051602081830303815290604052613af7565b95945050505050565b6033546001600160a01b031690565b6060611ed5613250565b60030180546109a590615252565b6060610109611ef4888a8989613b28565b611f018b8d888888613b5a565b604051602001611f13939291906155e1565b60405160208183030381529060405290509998505050505050505050565b80611f3a613250565b336000818152600792909201602090815260408084206001600160a01b03881680865290835293819020805460ff19169515159590951790945592518415158152919290917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b60026097541415611fd15760405162461bcd60e51b81526004016108d49061521b565b6002609755611fdf81613bb7565b611ff357611ff3631a7387bd60e11b6132be565b6000611ffd613378565b60cb549091506001600160a01b03168061204557612019611ebc565b6001600160a01b0316826001600160a01b031614612040576120406282b42960e81b6132be565b6120cd565b6040516342f19adb60e11b81526001600160a01b038216906385e335b6906120769085906000908890600401615693565b6020604051808303816000875af1158015612095573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120b991906153cd565b6120cd576120cd63e10a050560e01b6132be565b60cb80546001600160a01b0319166001600160a01b038516908117909155604051600080516020615e7e83398151915290600090a260d0546040516316365cdd60e01b81526001600160a01b03909116906316365cdd90611240908690600401614a7f565b600260975414156121555760405162461bcd60e51b81526004016108d49061521b565b60026097556000612164613378565b60cb549091506001600160a01b03168061218857612188635f98be1560e11b6132be565b60405163ce2003a560e01b81526001600160a01b0382169063ce2003a5906121b7908590600090600401614b54565b6020604051808303816000875af11580156121d6573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906121fa91906153cd565b61220e5761220e63251c48e160e11b6132be565b60cb80546001600160a01b0319169055604051600090600080516020615e7e833981519152908290a260d0546040516316365cdd60e01b81526001600160a01b03909116906316365cdd9061226890600090600401614a7f565b600060405180830381600087803b15801561228257600080fd5b505af1158015611a84573d6000803e3d6000fd5b6122a1848484610be5565b6001600160a01b0383163b156122d1576122bd84848484613bd3565b6122d1576122d16368d2bf6b60e11b6132be565b50505050565b600260975414156122fa5760405162461bcd60e51b81526004016108d49061521b565b60026097556000612309613378565b90506000612315611ebc565b60cf549091506001600160a01b0316858161241657826001600160a01b0316846001600160a01b031614612352576123526282b42960e81b6132be565b60005b818110156124105761239887878381811061237257612372615358565b905060400201602001602081019061238a91906156c6565b61271061ffff909116111590565b6123ac576123ac633daf741f60e21b6132be565b8686828181106123be576123be615358565b90506040020160ce60008b8b858181106123da576123da615358565b90506020020135815260200190815260200160002081816123fb91906156e3565b5081905061240881615386565b915050612355565b50612564565b60005b818110156125625761243687878381811061237257612372615358565b61244a5761244a633daf741f60e21b6132be565b826001600160a01b031663ad6e40c78a8a8481811061246b5761246b615358565b9050602002013589898581811061248457612484615358565b905060400201886040518463ffffffff1660e01b81526004016124a993929190615770565b602060405180830381865afa1580156124c6573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906124ea91906153cd565b6124fe576124fe63270e710760e21b6132be565b86868281811061251057612510615358565b90506040020160ce60008b8b8581811061252c5761252c615358565b905060200201358152602001908152602001600020818161254d91906156e3565b5081905061255a81615386565b915050612419565b505b7feaf422d1ab2d8f38856ab6be8378c08d2886463a33667e896ba3a6ebf2c6260c88888888604051612599949392919061579d565b60405180910390a160d054604051632be18fef60e11b81526001600160a01b03909116906357c31fde906125d7908b908b908b908b9060040161579d565b600060405180830381600087803b1580156125f157600080fd5b505af1158015612605573d6000803e3d6000fd5b5050600160975550505050505050505050565b6002609754141561263b5760405162461bcd60e51b81526004016108d49061521b565b6002609755600061264a613378565b60cf549091506001600160a01b03168061266e5761266e635f98be1560e11b6132be565b60405163680c607d60e11b81526001600160a01b0382169063d018c0fa9061269a908590600401614a7f565b602060405180830381865afa1580156126b7573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906126db91906153cd565b6126ef576126ef63251c48e160e11b6132be565b60cf80546001600160a01b0319169055604051600090600080516020615fc8833981519152908290a260d05460405163755d5a1160e11b81526001600160a01b039091169063eabab4229061226890600090600401614a7f565b60606108a382613caf565b600054610100900460ff16158080156127745750600054600160ff909116105b8061278e5750303b15801561278e575060005460ff166001145b6127f15760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b60648201526084016108d4565b6000805460ff191660011790558015612814576000805461ff0019166101001790555b6000806000806000806000806000806000808f8f8101906128359190615862565b9b509b509b509b509b509b509b509b509b509b509b509b50612855613d7a565b6128608c8b8b613da1565b61286986613e7e565b6128738888613ec7565b61287e60c986613915565b508a51612893906101079060208e0190614909565b5083516128a890610109906020870190614909565b5060405163332a79b160e01b81526001600160a01b038f169063332a79b1906128d5903090600401614a7f565b600060405180830381600087803b1580156128ef57600080fd5b505af1158015612903573d6000803e3d6000fd5b505050508d60d060006101000a8154816001600160a01b0302191690836001600160a01b031602179055506000835111156129415761294183613946565b811561294e576101088290555b50505050505050505050505080156122d1576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a150505050565b600260975414156129c85760405162461bcd60e51b81526004016108d49061521b565b6002609755828181146129e5576129e5632b477e7160e11b6132be565b60005b81811015612a4757612a35868683818110612a0557612a05615358565b90506020020135858584818110612a1e57612a1e615358565b9050602002810190612a3091906159c7565b613ef8565b80612a3f81615386565b9150506129e8565b507f2df7dd608aac5806395769062aca937bf08722cb45659d38c0cce82f5cd0cde385858585604051612a7d9493929190615a0d565b60405180910390a160d0546040516374c1ace160e01b81526001600160a01b03909116906374c1ace190612abb908890889088908890600401615a0d565b600060405180830381600087803b158015612ad557600080fd5b505af1158015612ae9573d6000803e3d6000fd5b5050600160975550505050505050565b6101078054611acd90615252565b6000612b11613250565b6001600160a01b039384166000908152600791909101602090815260408083209490951682529290925250205460ff1690565b60026097541415612b675760405162461bcd60e51b81526004016108d49061521b565b60026097556000612b76613378565b90506000612b82611ebc565b90508460005b81811015612d7b57612bbf868683818110612ba557612ba5615358565b9050602002016020810190612bba9190614ad4565b613bb7565b612bd357612bd3631a7387bd60e11b6132be565b6000612bf6898984818110612bea57612bea615358565b90506020020135611a91565b90506001600160a01b038116612c3357836001600160a01b0316856001600160a01b031614612c2e57612c2e6282b42960e81b6132be565b612cf1565b806001600160a01b03166385e335b6868b8b86818110612c5557612c55615358565b600088815260cc60209081526040918290205491516001600160e01b031960e089901b168152612c9a969591909302013592506001600160a01b031690600401615693565b6020604051808303816000875af1158015612cb9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612cdd91906153cd565b612cf157612cf163e10a050560e01b6132be565b868683818110612d0357612d03615358565b9050602002016020810190612d189190614ad4565b60cc60008b8b86818110612d2e57612d2e615358565b90506020020135815260200190815260200160002060006101000a8154816001600160a01b0302191690836001600160a01b03160217905550508080612d7390615386565b915050612b88565b507facc0f6922d8ccd133dbdaee1908d4d734f25a540223d55145e9fad2898471a4e87878787604051612db19493929190615abd565b60405180910390a160d0546040516315bd85bf60e01b81526001600160a01b03909116906315bd85bf90612def908a908a908a908a90600401615abd565b600060405180830381600087803b158015612e0957600080fd5b505af1158015612e1d573d6000803e3d6000fd5b50506001609755505050505050505050565b60026097541415612e525760405162461bcd60e51b81526004016108d49061521b565b6002609755612e6760408201602083016156c6565b61271061ffff82161115612e8557612e85633daf741f60e21b6132be565b6000612e8f613378565b60cf549091506001600160a01b031680612ed757612eab611ebc565b6001600160a01b0316826001600160a01b031614612ed257612ed26282b42960e81b6132be565b612f5a565b60405163049a5c0f60e41b81526001600160a01b038216906349a5c0f090612f059087908690600401615b15565b602060405180830381865afa158015612f22573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612f4691906153cd565b612f5a57612f5a63270e710760e21b6132be565b8360cd612f6782826156e3565b50612f7a905060408501602086016156c6565b61ffff16612f8b6020860186614ad4565b6001600160a01b03167f2c5ea6e4103e78cb101e796fb2dace540362fc542cbff5145eaa24af7dd8fe4160405160405180910390a360d0546001600160a01b031663d10072c0612fde6020870187614ad4565b612fee60408801602089016156c6565b6040516001600160e01b031960e085901b1681526001600160a01b03909216600483015261ffff166024820152604401611530565b61302b6131e1565b6001600160a01b0381166130905760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201526564647265737360d01b60648201526084016108d4565b61309981613993565b50565b6130a7610a87613378565b6130bb576130bb633e34a41b60e21b6132be565b600260975414156130de5760405162461bcd60e51b81526004016108d49061521b565b600260975560cf54600160a01b900460ff1660011415613108576131086345a9661760e01b6132be565b8060006131136134cb565b905060005b828110156131735761315385858381811061313557613135615358565b905060200201602081019061314a9190614ad4565b6001848561339e565b8161315d81615386565b925050808061316b90615386565b915050613118565b50610982610fb66001836153a1565b606061318e60c9613ff9565b905090565b60006301ffc9a760e01b6001600160e01b0319831614806131c457506380ac58cd60e01b6001600160e01b03198316145b806108a35750506001600160e01b031916635b5e139f60e01b1490565b6131e9613378565b6001600160a01b03166131fa611ebc565b6001600160a01b031614611bcf5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016108d4565b7f2569078dfb4b0305704d3008e7403993ae9601b85f7ae5e742de3de8f8011c4090565b6000816001116132b9576000613288613250565b600084815260049190910160205260409020549050600160e01b81161580156132b557506132b58361364c565b9150505b919050565b61309981614006565b60006132d28361192a565b90508180156132ea5750336001600160a01b03821614155b1561330d576132f98133612b07565b61330d5761330d6367d9dca160e11b6132be565b83613316613250565b6000858152600691909101602052604080822080546001600160a01b0319166001600160a01b0394851617905551859287811692908516917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259190a450505050565b600061318e614010565b6000613397836001600160a01b038416614032565b9392505050565b630100000083106133b9576133b96312387aa160e01b6132be565b806133ca576133c6613250565b5490505b816133d7578091506133f5565b6133e08261364c565b156133f5576133f563638e967560e11b6132be565b8261340a5761340a63b562e8dd60e01b6132be565b6134298461341c600087600088613610565b6001861460e11b17613637565b613431613250565b600084815260049190910160205260409020556001600160401b018302613456613250565b6001600160a01b0386166000818152600592909201602052604090912080549092019091558061348f5761348f622e076360e81b6132be565b828401835b80836000600080516020615fa8833981519152600080a460010180821415613494578584016134c1613250565b5550505050505050565b60006134d5613250565b54919050565b6101085480158015906134ed57508082115b15610a7857610a7863be1a657d60e01b6132be565b6000816001116135c357613514613250565b600083815260049190910160205260409020549050806135b35761353782613274565b61354b5761354b636f96cda160e11b6132be565b8161356057613560636f96cda160e11b6132be565b613568613250565b600019909201600081815260049390930160205260409092205490508061358e5761354b565b600160e01b811661359e57919050565b6135ae636f96cda160e11b6132be565b61354b565b600160e01b81166135c357919050565b6132b9636f96cda160e11b6132be565b60008060006135e0613250565b60009485526006016020525050604090912080549092909150565b6001600160a01b039081169116811491141790565b600060e883811c906136248787848761404a565b62ffffff16901b9150505b949350505050565b4260a01b176001600160a01b03919091161790565b6000816001116132b9578160005b60e8613664613250565b60008681526004919091016020526040902054901c905062ffffff81166136a45783613694575060009392505050565b61369d84615b3b565b935061365a565b62ffffff81166136b485846153a1565b62ffffff1610156136c9575060019392505050565b5050919050565b6001600160e81b031690565b60006136e6613378565b905060006136f383611a91565b90506001600160a01b0381161580159061377c57506040516301ffc9a760e01b81526001600160a01b038216906301ffc9a79061373b90637af46b4f60e11b906004016153b8565b602060405180830381865afa158015613758573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061377c91906153cd565b1561380257604051638af6791b60e01b81526001600160a01b038381166004830152868116602483015285811660448301526064820185905260a06084830152600060a4830152821690638af6791b9060c401600060405180830381600087803b1580156137e957600080fd5b505af11580156137fd573d6000803e3d6000fd5b505050505b60d0546040516323de665160e01b81526001600160a01b03909116906323de665190613836908890889088906004016153ea565b600060405180830381600087803b15801561385057600080fd5b505af1158015613864573d6000803e3d6000fd5b505050505050505050565b8383613879613250565b6002019190613889929190614885565b508181613894613250565b6003019190611329929190614885565b6000806138b08361192a565b90506138bc8185612b07565b8061362f575061362f6138ce84610a28565b82866135fb565b6138de8161407e565b600081815260d16020526040902080546138f790615252565b15905061309957600081815260d1602052604081206130999161497d565b6000613397836001600160a01b038416614089565b60006108a36001600160a01b03831663440444c560e11b6140d3565b7f2e0a5b969d96a99aee0b35787d9a60516a02ca6f528a5f66d3f936468d8f038260d582604051613978929190615b52565b60405180910390a18051610a789060d5906020840190614909565b603380546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b6000613397836001600160a01b0384166140ef565b606081613a1e5750506040805180820190915260018152600360fc1b602082015290565b8160005b8115613a485780613a3281615386565b9150613a419050600a836152d2565b9150613a22565b6000816001600160401b03811115613a6257613a62614da7565b6040519080825280601f01601f191660200182016040528015613a8c576020820181803683370190505b5090505b841561362f57613aa16001836153a1565b9150613aae600a86615be2565b613ab990603061536e565b60f81b818381518110613ace57613ace615358565b60200101906001600160f81b031916908160001a905350613af0600a866152d2565b9450613a90565b6060613b02826141e2565b604051602001613b129190615bf6565b6040516020818303038152906040529050919050565b606084838386604051602001613b419493929190615c38565b6040516020818303038152906040529050949350505050565b6060613b65856139fa565b613b6e876139fa565b613b77866139fa565b613b80866139fa565b613b89866139fa565b604051602001613b9d959493929190615cd0565b604051602081830303815290604052905095945050505050565b60006108a36001600160a01b038316633b209e2f60e11b6140d3565b604051630a85bd0160e11b81526000906001600160a01b0385169063150b7a0290613c08903390899088908890600401615d90565b6020604051808303816000875af1925050508015613c43575060408051601f3d908101601f19168201909252613c4091810190615dc3565b60015b613c95573d808015613c71576040519150601f19603f3d011682016040523d82523d6000602084013e613c76565b606091505b508051613c8d57613c8d6368d2bf6b60e11b6132be565b805181602001fd5b6001600160e01b031916630a85bd0160e11b14905061362f565b6060613cba82613274565b613cce57613cce63677510db60e11b6132be565b600082815260d1602052604081208054613ce790615252565b80601f0160208091040260200160405190810160405280929190818152602001828054613d1390615252565b8015613d605780601f10613d3557610100808354040283529160200191613d60565b820191906000526020600020905b815481529060010190602001808311613d4357829003601f168201915b505050505090508051600014156108a3576132b583614335565b600054610100900460ff16611bcf5760405162461bcd60e51b81526004016108d490615de0565b600054610100900460ff16613dc85760405162461bcd60e51b81526004016108d490615de0565b602082015161271061ffff82161115613deb57613deb633daf741f60e21b6132be565b613df36143af565b613dfb6143de565b613e0484613993565b60cd546001600160a01b031615613e4c57825160cd8054602086015161ffff16600160a01b026001600160b01b03199091166001600160a01b03909316929092179190911790555b6001600160a01b038216156122d15760cb80546001600160a01b0384166001600160a01b031990911617905550505050565b600054610100900460ff16613ea55760405162461bcd60e51b81526004016108d490615de0565b606580546001600160a01b0319166001600160a01b0392909216919091179055565b600054610100900460ff16613eee5760405162461bcd60e51b81526004016108d490615de0565b610a78828261440d565b6000613f0384611a91565b90506000613f0f613378565b90506000613f1b611ebc565b90506001600160a01b038316613f5857806001600160a01b0316826001600160a01b031614613f5357613f536282b42960e81b6132be565b613fe0565b604051633d820a4d60e01b81526001600160a01b03841690633d820a4d90613f8a9085908a908a908a90600401615473565b6020604051808303816000875af1158015613fa9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190613fcd91906153cd565b613fe057613fe06282b42960e81b6132be565b600086815260d160205260409020610d99908686614885565b6060600061339783614483565b8060005260046000fd5b600061401b33611879565b1561402d575060131936013560601c90565b503390565b60009081526001919091016020526040902054151590565b60006001600160a01b03851615801561406b57506001600160a01b03841615155b1561407757508061362f565b508161362f565b6130998160006144df565b60006140958383614032565b6140cb575081546001818101845560008481526020808220909301849055845484825282860190935260409020919091556108a3565b5060006108a3565b60006140de8361464d565b801561339757506133978383614680565b600081815260018301602052604081205480156141d85760006141136001836153a1565b8554909150600090614127906001906153a1565b905081811461418c57600086600001828154811061414757614147615358565b906000526020600020015490508087600001848154811061416a5761416a615358565b6000918252602080832090910192909255918252600188019052604090208390555b855486908061419d5761419d615e2b565b6001900381819060005260206000200160009055905585600101600086815260200190815260200160002060009055600193505050506108a3565b60009150506108a3565b606081516000141561420257505060408051602081019091526000815290565b6000604051806060016040528060408152602001615ebe6040913990506000600384516002614231919061536e565b61423b91906152d2565b61424690600461529d565b6001600160401b0381111561425d5761425d614da7565b6040519080825280601f01601f191660200182016040528015614287576020820181803683370190505b509050600182016020820185865187015b808210156142f3576003820191508151603f8160121c168501518453600184019350603f81600c1c168501518453600184019350603f8160061c168501518453600184019350603f8116850151845350600183019250614298565b505060038651066001811461430f57600281146143225761432a565b603d6001830353603d600283035361432a565b603d60018303535b509195945050505050565b606061434082613274565b61435457614354630a14c4b560e41b6132be565b600061435e614714565b9050600081511161437e57604051806020016040528060008152506132b5565b8061438884614723565b604051602001614399929190615e41565b6040516020818303038152906040529392505050565b600054610100900460ff166143d65760405162461bcd60e51b81526004016108d490615de0565b611bcf614820565b600054610100900460ff166144055760405162461bcd60e51b81526004016108d490615de0565b611bcf614857565b600054610100900460ff166144345760405162461bcd60e51b81526004016108d490615de0565b8161443d613250565b6002019080519060200190614453929190614909565b508061445d613250565b6003019080519060200190614473929190614909565b50600161447e613250565b555050565b6060816000018054806020026020016040519081016040528092919081815260200182805480156144d357602002820191906000526020600020905b8154815260200190600101908083116144bf575b50505050509050919050565b60006144ea83613502565b9050806000806144f9866135d3565b9150915084156145305761450e8184336135fb565b6145305761451c8333612b07565b61453057614530632ce44b5f60e11b6132be565b801561453b57600082555b6001600160801b0361454b613250565b6001600160a01b038516600090815260059190910160205260408120805490920190915561458d9084906145829082908881613610565b600360e01b17613637565b614595613250565b60008881526004919091016020526040902055600160e11b841661460b57600186016145bf613250565b60008281526004919091016020526040902054614609576145df8161364c565b15614609576145ed856136d0565b6145f5613250565b600083815260049190910160205260409020555b505b60405186906000906001600160a01b03861690600080516020615fa8833981519152908390a4614639613250565b600190810180549091019055505050505050565b6000614660826301ffc9a760e01b614680565b80156108a35750614679826001600160e01b0319614680565b1592915050565b6000806301ffc9a760e01b8360405160240161469c91906153b8565b604051602081830303815290604052906001600160e01b0319166020820180516001600160e01b03838183161783525050505090506000806000602060008551602087018a617530fa92503d915060005190508280156146fd575060208210155b80156147095750600081115b979650505050505050565b606060d580546109a590615252565b6060816147475750506040805180820190915260018152600360fc1b602082015290565b8160005b8115614771578061475b81615386565b915061476a9050600a836152d2565b915061474b565b6000816001600160401b0381111561478b5761478b614da7565b6040519080825280601f01601f1916602001820160405280156147b5576020820181803683370190505b5090505b841561362f576147ca6001836153a1565b91506147d7600a86615be2565b6147e290603061536e565b60f81b8183815181106147f7576147f7615358565b60200101906001600160f81b031916908160001a905350614819600a866152d2565b94506147b9565b600054610100900460ff166148475760405162461bcd60e51b81526004016108d490615de0565b611bcf614852613378565b613993565b600054610100900460ff1661487e5760405162461bcd60e51b81526004016108d490615de0565b6001609755565b82805461489190615252565b90600052602060002090601f0160209004810192826148b357600085556148f9565b82601f106148cc5782800160ff198235161785556148f9565b828001600101855582156148f9579182015b828111156148f95782358255916020019190600101906148de565b506149059291506149b3565b5090565b82805461491590615252565b90600052602060002090601f01602090048101928261493757600085556148f9565b82601f1061495057805160ff19168380011785556148f9565b828001600101855582156148f9579182015b828111156148f9578251825591602001919060010190614962565b50805461498990615252565b6000825580601f10614999575050565b601f01602090049060005260206000209081019061309991905b5b8082111561490557600081556001016149b4565b6001600160e01b03198116811461309957600080fd5b6000602082840312156149f057600080fd5b8135613397816149c8565b60005b83811015614a165781810151838201526020016149fe565b838111156122d15750506000910152565b60008151808452614a3f8160208601602086016149fb565b601f01601f19169290920160200192915050565b6020815260006133976020830184614a27565b600060208284031215614a7857600080fd5b5035919050565b6001600160a01b0391909116815260200190565b6001600160a01b038116811461309957600080fd5b60008060408385031215614abb57600080fd5b8235614ac681614a93565b946020939093013593505050565b600060208284031215614ae657600080fd5b813561339781614a93565b600080600060608486031215614b0657600080fd5b8335614b1181614a93565b92506020840135614b2181614a93565b929592945050506040919091013590565b60008060408385031215614b4557600080fd5b50508035926020909101359150565b6001600160a01b03929092168252602082015260400190565b60008083601f840112614b7f57600080fd5b5081356001600160401b03811115614b9657600080fd5b602083019150836020828501011115614bae57600080fd5b9250929050565b60008060008060008060608789031215614bce57600080fd5b86356001600160401b0380821115614be557600080fd5b614bf18a838b01614b6d565b90985096506020890135915080821115614c0a57600080fd5b614c168a838b01614b6d565b90965094506040890135915080821115614c2f57600080fd5b50614c3c89828a01614b6d565b979a9699509497509295939492505050565b60008083601f840112614c6057600080fd5b5081356001600160401b03811115614c7757600080fd5b6020830191508360208260051b8501011115614bae57600080fd5b600080600060408486031215614ca757600080fd5b83356001600160401b03811115614cbd57600080fd5b614cc986828701614c4e565b909790965060209590950135949350505050565b60008060208385031215614cf057600080fd5b82356001600160401b03811115614d0657600080fd5b614d1285828601614c4e565b90969095509350505050565b60008060208385031215614d3157600080fd5b82356001600160401b03811115614d4757600080fd5b614d1285828601614b6d565b600080600060408486031215614d6857600080fd5b8335614d7381614a93565b925060208401356001600160401b03811115614d8e57600080fd5b614d9a86828701614c4e565b9497909650939450505050565b634e487b7160e01b600052604160045260246000fd5b60006001600160401b0380841115614dd757614dd7614da7565b604051601f8501601f19908116603f01168101908282118183101715614dff57614dff614da7565b81604052809350858152868686011115614e1857600080fd5b858560208301376000602087830101525050509392505050565b600082601f830112614e4357600080fd5b61339783833560208501614dbd565b600060208284031215614e6457600080fd5b81356001600160401b03811115614e7a57600080fd5b61362f84828501614e32565b60008060008060008060008060006101208a8c031215614ea557600080fd5b8935985060208a0135975060408a01356001600160401b0380821115614eca57600080fd5b614ed68d838e01614e32565b985060608c0135915080821115614eec57600080fd5b614ef88d838e01614e32565b975060808c0135915080821115614f0e57600080fd5b614f1a8d838e01614e32565b965060a08c0135915080821115614f3057600080fd5b50614f3d8c828d01614e32565b94505060c08a0135925060e08a013591506101008a013590509295985092959850929598565b801515811461309957600080fd5b80356132b981614f63565b60008060408385031215614f8f57600080fd5b8235614f9a81614a93565b91506020830135614faa81614f63565b809150509250929050565b60008060008060808587031215614fcb57600080fd5b8435614fd681614a93565b93506020850135614fe681614a93565b92506040850135915060608501356001600160401b0381111561500857600080fd5b8501601f8101871361501957600080fd5b61502887823560208401614dbd565b91505092959194509250565b6000806000806040858703121561504a57600080fd5b84356001600160401b038082111561506157600080fd5b61506d88838901614c4e565b9096509450602087013591508082111561508657600080fd5b818701915087601f83011261509a57600080fd5b8135818111156150a957600080fd5b8860208260061b85010111156150be57600080fd5b95989497505060200194505050565b6000806000604084860312156150e257600080fd5b83356001600160401b038111156150f857600080fd5b61510486828701614b6d565b909450925050602084013561511881614a93565b809150509250925092565b6000806000806040858703121561513957600080fd5b84356001600160401b038082111561515057600080fd5b61515c88838901614c4e565b9096509450602087013591508082111561517557600080fd5b5061518287828801614c4e565b95989497509550505050565b600080604083850312156151a157600080fd5b82356151ac81614a93565b91506020830135614faa81614a93565b600060408284031215611aba57600080fd5b6020808252825182820181905260009190848201906040850190845b8181101561520f5783516001600160a01b0316835292840192918401916001016151ea565b50909695505050505050565b6020808252601f908201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604082015260600190565b600181811c9082168061526657607f821691505b60208210811415611aba57634e487b7160e01b600052602260045260246000fd5b634e487b7160e01b600052601160045260246000fd5b60008160001904831182151516156152b7576152b7615287565b500290565b634e487b7160e01b600052601260045260246000fd5b6000826152e1576152e16152bc565b500490565b81835281816020850137506000828201602090810191909152601f909101601f19169091010190565b60608152600061532360608301888a6152e6565b82810360208401526153368187896152e6565b9050828103604084015261534b8185876152e6565b9998505050505050505050565b634e487b7160e01b600052603260045260246000fd5b6000821982111561538157615381615287565b500190565b600060001982141561539a5761539a615287565b5060010190565b6000828210156153b3576153b3615287565b500390565b6001600160e01b031991909116815260200190565b6000602082840312156153df57600080fd5b815161339781614f63565b6001600160a01b039384168152919092166020820152604081019190915260600190565b6001600160a01b039290921682521515602082015260400190565b81835260006001600160fb1b0383111561544257600080fd5b8260051b8083602087013760009401602001938452509192915050565b60208152600061362f602083018486615429565b60018060a01b038516815283602082015260606040820152600061549b6060830184866152e6565b9695505050505050565b60208152600061362f6020830184866152e6565b600084516154cb8184602089016149fb565b7f3c636972636c652063783d22333530222063793d223335302220723d22000000908301908152845161550581601d8401602089016149fb565b69111039ba3937b5b29e9160b11b601d929091019182015283516155308160278401602088016149fb565b6211179f60e91b60279290910191820152602a0195945050505050565b6000855161555f818460208a016149fb565b7f3c636972636c652063783d22333530222063793d223335302220723d2231302292019182525069207374726f6b653d222360b01b60208201526001600160f81b0319938416602a820152918316602b830152909116602c8201526211179f60e91b602d820152691e17b39f1e17b9bb339f60b11b6030820152603a01919050565b60008085546155ef81615252565b60018281168015615607576001811461561857615647565b60ff19841687528287019450615647565b8960005260208060002060005b8581101561563e5781548a820152908401908201615625565b50505082870194505b505050506b2f696e6465782e68746d6c3f60a01b8152845161567081600c8401602089016149fb565b845191019061568681600c8401602088016149fb565b01600c0195945050505050565b6001600160a01b0393841681526020810192909252909116604082015260600190565b61ffff8116811461309957600080fd5b6000602082840312156156d857600080fd5b8135613397816156b6565b81356156ee81614a93565b81546001600160a01b031981166001600160a01b03929092169182178355602084013561571a816156b6565b6001600160b01b03199190911690911760a09190911b61ffff60a01b1617905550565b803561574881614a93565b6001600160a01b031682526020810135615761816156b6565b61ffff81166020840152505050565b83815260808101615784602083018561573d565b6001600160a01b03929092166060919091015292915050565b600060408083526157b18184018789615429565b83810360208086019190915285825286910160005b868110156157e8576157d8828461573d565b91830191908301906001016157c6565b5098975050505050505050565b80356132b981614a93565b60006040828403121561581257600080fd5b604080519081016001600160401b038111828210171561583457615834614da7565b604052905080823561584581614a93565b81526020830135615855816156b6565b6020919091015292915050565b6000806000806000806000806000806000806101a08d8f03121561588557600080fd5b61588e8d6157f5565b9b506001600160401b0360208e013511156158a857600080fd5b6158b88e60208f01358f01614e32565b9a506158c78e60408f01615800565b99506158d560808e016157f5565b98506001600160401b0360a08e013511156158ef57600080fd5b6158ff8e60a08f01358f01614e32565b97506001600160401b0360c08e0135111561591957600080fd5b6159298e60c08f01358f01614e32565b965061593760e08e016157f5565b95506159466101008e016157f5565b94506001600160401b036101208e0135111561596157600080fd5b6159728e6101208f01358f01614e32565b93506001600160401b036101408e0135111561598d57600080fd5b61599e8e6101408f01358f01614e32565b92506101608d013591506159b56101808e01614f71565b90509295989b509295989b509295989b565b6000808335601e198436030181126159de57600080fd5b8301803591506001600160401b038211156159f857600080fd5b602001915036819003821315614bae57600080fd5b604081526000615a21604083018688615429565b602083820381850152818583528183019050818660051b8401018760005b88811015615aad57858303601f190184528135368b9003601e19018112615a6557600080fd5b8a0180356001600160401b03811115615a7d57600080fd5b8036038c1315615a8c57600080fd5b615a9985828985016152e6565b958701959450505090840190600101615a3f565b50909a9950505050505050505050565b604081526000615ad1604083018688615429565b8281036020848101919091528482528591810160005b868110156157e8578335615afa81614a93565b6001600160a01b031682529282019290820190600101615ae7565b60608101615b23828561573d565b6001600160a01b039290921660409190910152919050565b600081615b4a57615b4a615287565b506000190190565b604081526000808454615b6481615252565b8060408601526060600180841660008114615b865760018114615b9a57615bcb565b60ff19851688840152608088019550615bcb565b8960005260208060002060005b86811015615bc25781548b8201870152908401908201615ba7565b8a018501975050505b50505050508281036020840152611eb38185614a27565b600082615bf157615bf16152bc565b500690565b7919185d184e9a5b5859d94bdcdd99cade1b5b0ed8985cd94d8d0b60321b815260008251615c2b81601a8501602087016149fb565b91909101601a0192915050565b61613d60f01b815260008551615c55816002850160208a016149fb565b6226683d60e81b6002918401918201528551615c78816005840160208a016149fb565b632662683d60e01b600592909101918201528451615c9d8160098401602089016149fb565b632677613d60e01b600992909101918201528351615cc281600d8401602088016149fb565b01600d019695505050505050565b6226633d60e81b815260008651615cee816003850160208b016149fb565b64267469643d60d81b6003918401918201528651615d13816008840160208b016149fb565b6326626e3d60e01b600892909101918201528551615d3881600c840160208a016149fb565b6226743d60e81b600c92909101918201528451615d5c81600f8401602089016149fb565b632667703d60e01b600f92909101918201528351615d818160138401602088016149fb565b01601301979650505050505050565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061549b90830184614a27565b600060208284031215615dd557600080fd5b8151613397816149c8565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b634e487b7160e01b600052603160045260246000fd5b60008351615e538184602088016149fb565b602f60f81b9083019081528351615e718160018401602088016149fb565b0160010194935050505056fec407dfe559ca4a63d7cf8439b9fce2965d7a4c8c92eb5d5047f53bb9eec295fd73e05f4c0a27cbf60841afa4a2f8577f5e99db3e3e9c293800d6e6b1f7b8a7ab4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2f3c73766720786d6c6e733d22687474703a2f2f7777772e77332e6f72672f323030302f737667222069643d22307865453962653938343661624234393533326130334330354643323741306534446263383465383537222076696577426f783d223020302037303020373030223e3c7374796c653e2e617b7374726f6b652d77696474683a32303b7d3c2f7374796c653e3c6720636c6173733d2261222066696c6c3d226e6f6e65223eddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efe236d251e7da4ac8a3c275f80aebc9b74cbf356425fa04864da5043511a81219a164736f6c634300080a000a
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 26 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.