Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Contract Source Code Verified (Exact Match)
Contract Name:
TokenCollection
Compiler Version
v0.8.15+commit.e14f2714
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity ^0.8.9; import "@openzeppelin/contracts-upgradeable/utils/math/SafeMathUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC721/extensions/ERC721EnumerableUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC721/extensions/ERC721URIStorageUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/ERC721/extensions/ERC721BurnableUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol"; import "../BaseCollection.sol"; import "../Redeemables.sol"; contract TokenCollection is Redeemables, BaseCollection, ERC721Upgradeable, ERC721EnumerableUpgradeable, ERC721URIStorageUpgradeable, ERC721BurnableUpgradeable { using SafeMathUpgradeable for uint256; using CountersUpgradeable for CountersUpgradeable.Counter; CountersUpgradeable.Counter private _tokenIdCounter; /// @custom:oz-upgrades-unsafe-allow constructor constructor() { _disableInitializers(); } function initialize( address owner_, string memory name_, string memory symbol_, address treasury_, address royalty_, uint96 royaltyFee_ ) public override initializer { __ERC721_init(name_, symbol_); __ERC721Enumerable_init(); __ERC721URIStorage_init(); __ERC721Burnable_init(); __BaseCollection_init(owner_, treasury_, royalty_, royaltyFee_); _tokenIdCounter.increment(); } function mint( address to, uint256 quantity, string memory uri ) external onlyRolesOrOwner(MANAGER_ROLE) { _mint(to, quantity, uri); } function createRedeemable( string memory uri, uint256 price, uint256 maxQuantity, uint256 maxPerWallet, uint256 maxPerMint ) external onlyRolesOrOwner(MANAGER_ROLE) { _createRedeemable(uri, price, maxQuantity, maxPerWallet, maxPerMint); } function setMerkleRoot(uint256 redeemableId, bytes32 newRoot) external onlyRolesOrOwner(MANAGER_ROLE) { _setMerkleRoot(redeemableId, newRoot); } function setTokenURI(uint256 tokenId, string memory newUri) external onlyRolesOrOwner(MANAGER_ROLE) { _setTokenURI(tokenId, newUri); } function invalidate(uint256 redeemableId) external onlyRolesOrOwner(MANAGER_ROLE) { _invalidate(redeemableId); } function revoke(uint256 redeemableId) external onlyRolesOrOwner(MANAGER_ROLE) { _revoke(redeemableId); } function redeem( uint256 redeemableId, uint256 quantity, bytes calldata signature, bytes32[] calldata proof ) external payable { Redeemable memory redeemable = redeemableAt(redeemableId); unchecked { _totalRevenue = _totalRevenue.add(msg.value); } _niftyKit.addFees(msg.value); _mint(_msgSender(), quantity, redeemable.tokenURI); _redeem(redeemableId, quantity, signature, owner(), proof); } function _mint( address to, uint256 quantity, string memory uri ) internal { for (uint256 i = 0; i < quantity; ) { uint256 tokenId = _tokenIdCounter.current(); _tokenIdCounter.increment(); _safeMint(to, tokenId); _setTokenURI(tokenId, uri); unchecked { i++; } } } function isApprovedForAll(address owner, address operator) public view override(ERC721Upgradeable, IERC721Upgradeable) returns (bool isOperator) { if (_operatorRegistry != address(0)) { bytes32 identifier = IOperatorRegistry(_operatorRegistry) .getIdentifier(operator); if (_allowedOperators[identifier]) return true; if (_blockedOperators[identifier]) return false; } return ERC721Upgradeable.isApprovedForAll(owner, operator); } function approve(address to, uint256 tokenId) public virtual override(ERC721Upgradeable, IERC721Upgradeable) preventBlockedOperator(to) { ERC721Upgradeable.approve(to, tokenId); } function setApprovalForAll(address operator, bool approved) public virtual override(ERC721Upgradeable, IERC721Upgradeable) preventBlockedOperator(operator) { ERC721Upgradeable.setApprovalForAll(operator, approved); } // The following functions are overrides required by Solidity. function tokenURI(uint256 tokenId) public view override(ERC721Upgradeable, ERC721URIStorageUpgradeable) returns (string memory) { return ERC721URIStorageUpgradeable.tokenURI(tokenId); } function _burn(uint256 tokenId) internal override(ERC721Upgradeable, ERC721URIStorageUpgradeable) { ERC721URIStorageUpgradeable._burn(tokenId); } function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal virtual override(ERC721Upgradeable, ERC721EnumerableUpgradeable) { super._beforeTokenTransfer(from, to, tokenId); } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721Upgradeable, ERC721EnumerableUpgradeable, BaseCollection) returns (bool) { return ERC721Upgradeable.supportsInterface(interfaceId) || ERC721EnumerableUpgradeable.supportsInterface(interfaceId) || BaseCollection.supportsInterface(interfaceId) || super.supportsInterface(interfaceId); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol) pragma solidity ^0.8.0; import "../utils/introspection/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 (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) (token/common/ERC2981.sol) pragma solidity ^0.8.0; import "../../interfaces/IERC2981Upgradeable.sol"; import "../../utils/introspection/ERC165Upgradeable.sol"; import "../../proxy/utils/Initializable.sol"; /** * @dev Implementation of the NFT Royalty Standard, a standardized way to retrieve royalty payment information. * * Royalty information can be specified globally for all token ids via {_setDefaultRoyalty}, and/or individually for * specific token ids via {_setTokenRoyalty}. The latter takes precedence over the first. * * Royalty is specified as a fraction of sale price. {_feeDenominator} is overridable but defaults to 10000, meaning the * fee is specified in basis points by default. * * IMPORTANT: ERC-2981 only specifies a way to signal royalty information and does not enforce its payment. See * https://eips.ethereum.org/EIPS/eip-2981#optional-royalty-payments[Rationale] in the EIP. Marketplaces are expected to * voluntarily pay royalties together with sales, but note that this standard is not yet widely supported. * * _Available since v4.5._ */ abstract contract ERC2981Upgradeable is Initializable, IERC2981Upgradeable, ERC165Upgradeable { function __ERC2981_init() internal onlyInitializing { } function __ERC2981_init_unchained() internal onlyInitializing { } struct RoyaltyInfo { address receiver; uint96 royaltyFraction; } RoyaltyInfo private _defaultRoyaltyInfo; mapping(uint256 => RoyaltyInfo) private _tokenRoyaltyInfo; /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165Upgradeable, ERC165Upgradeable) returns (bool) { return interfaceId == type(IERC2981Upgradeable).interfaceId || super.supportsInterface(interfaceId); } /** * @inheritdoc IERC2981Upgradeable */ function royaltyInfo(uint256 _tokenId, uint256 _salePrice) public view virtual override returns (address, uint256) { RoyaltyInfo memory royalty = _tokenRoyaltyInfo[_tokenId]; if (royalty.receiver == address(0)) { royalty = _defaultRoyaltyInfo; } uint256 royaltyAmount = (_salePrice * royalty.royaltyFraction) / _feeDenominator(); return (royalty.receiver, royaltyAmount); } /** * @dev The denominator with which to interpret the fee set in {_setTokenRoyalty} and {_setDefaultRoyalty} as a * fraction of the sale price. Defaults to 10000 so fees are expressed in basis points, but may be customized by an * override. */ function _feeDenominator() internal pure virtual returns (uint96) { return 10000; } /** * @dev Sets the royalty information that all ids in this contract will default to. * * Requirements: * * - `receiver` cannot be the zero address. * - `feeNumerator` cannot be greater than the fee denominator. */ function _setDefaultRoyalty(address receiver, uint96 feeNumerator) internal virtual { require(feeNumerator <= _feeDenominator(), "ERC2981: royalty fee will exceed salePrice"); require(receiver != address(0), "ERC2981: invalid receiver"); _defaultRoyaltyInfo = RoyaltyInfo(receiver, feeNumerator); } /** * @dev Removes default royalty information. */ function _deleteDefaultRoyalty() internal virtual { delete _defaultRoyaltyInfo; } /** * @dev Sets the royalty information for a specific token id, overriding the global default. * * Requirements: * * - `receiver` cannot be the zero address. * - `feeNumerator` cannot be greater than the fee denominator. */ function _setTokenRoyalty( uint256 tokenId, address receiver, uint96 feeNumerator ) internal virtual { require(feeNumerator <= _feeDenominator(), "ERC2981: royalty fee will exceed salePrice"); require(receiver != address(0), "ERC2981: Invalid parameters"); _tokenRoyaltyInfo[tokenId] = RoyaltyInfo(receiver, feeNumerator); } /** * @dev Resets royalty information for the token id back to the global default. */ function _resetTokenRoyalty(uint256 tokenId) internal virtual { delete _tokenRoyaltyInfo[tokenId]; } /** * @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[48] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/ERC721.sol) pragma solidity ^0.8.0; import "./IERC721Upgradeable.sol"; import "./IERC721ReceiverUpgradeable.sol"; import "./extensions/IERC721MetadataUpgradeable.sol"; import "../../utils/AddressUpgradeable.sol"; import "../../utils/ContextUpgradeable.sol"; import "../../utils/StringsUpgradeable.sol"; import "../../utils/introspection/ERC165Upgradeable.sol"; import "../../proxy/utils/Initializable.sol"; /** * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including * the Metadata extension, but not including the Enumerable extension, which is available separately as * {ERC721Enumerable}. */ contract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable { using AddressUpgradeable for address; using StringsUpgradeable for uint256; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to owner address mapping(uint256 => address) private _owners; // Mapping owner address to token count mapping(address => uint256) private _balances; // Mapping from token ID to approved address mapping(uint256 => address) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; /** * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection. */ function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing { __ERC721_init_unchained(name_, symbol_); } function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing { _name = name_; _symbol = symbol_; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) { return interfaceId == type(IERC721Upgradeable).interfaceId || interfaceId == type(IERC721MetadataUpgradeable).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721-balanceOf}. */ function balanceOf(address owner) public view virtual override returns (uint256) { require(owner != address(0), "ERC721: address zero is not a valid owner"); return _balances[owner]; } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { address owner = _owners[tokenId]; require(owner != address(0), "ERC721: invalid token ID"); return owner; } /** * @dev See {IERC721Metadata-name}. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev See {IERC721Metadata-symbol}. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { _requireMinted(tokenId); 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, can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ""; } /** * @dev See {IERC721-approve}. */ function approve(address to, uint256 tokenId) public virtual override { address owner = ERC721Upgradeable.ownerOf(tokenId); require(to != owner, "ERC721: approval to current owner"); require( _msgSender() == owner || isApprovedForAll(owner, _msgSender()), "ERC721: approve caller is not token owner nor approved for all" ); _approve(to, tokenId); } /** * @dev See {IERC721-getApproved}. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { _requireMinted(tokenId); return _tokenApprovals[tokenId]; } /** * @dev See {IERC721-setApprovalForAll}. */ function setApprovalForAll(address operator, bool approved) public virtual override { _setApprovalForAll(_msgSender(), operator, approved); } /** * @dev See {IERC721-isApprovedForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev See {IERC721-transferFrom}. */ function transferFrom( address from, address to, uint256 tokenId ) public virtual override { //solhint-disable-next-line max-line-length require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved"); _transfer(from, to, tokenId); } /** * @dev See {IERC721-safeTransferFrom}. */ function safeTransferFrom( address from, address to, uint256 tokenId ) public virtual override { safeTransferFrom(from, to, tokenId, ""); } /** * @dev See {IERC721-safeTransferFrom}. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes memory data ) public virtual override { require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved"); _safeTransfer(from, to, tokenId, data); } /** * @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. * * `data` is additional data, it has no specified format and it is sent in call to `to`. * * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g. * implement alternative mechanisms to perform token transfer, such as signature-based. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeTransfer( address from, address to, uint256 tokenId, bytes memory data ) internal virtual { _transfer(from, to, tokenId); require(_checkOnERC721Received(from, to, tokenId, data), "ERC721: transfer to non ERC721Receiver implementer"); } /** * @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 (`_mint`), * and stop existing when they are burned (`_burn`). */ function _exists(uint256 tokenId) internal view virtual returns (bool) { return _owners[tokenId] != address(0); } /** * @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 = ERC721Upgradeable.ownerOf(tokenId); return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender); } /** * @dev Safely mints `tokenId` and transfers it to `to`. * * Requirements: * * - `tokenId` must not exist. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeMint(address to, uint256 tokenId) internal virtual { _safeMint(to, tokenId, ""); } /** * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is * forwarded in {IERC721Receiver-onERC721Received} to contract recipients. */ function _safeMint( address to, uint256 tokenId, bytes memory data ) internal virtual { _mint(to, tokenId); require( _checkOnERC721Received(address(0), to, tokenId, data), "ERC721: transfer to non ERC721Receiver implementer" ); } /** * @dev Mints `tokenId` and transfers it to `to`. * * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible * * Requirements: * * - `tokenId` must not exist. * - `to` cannot be the zero address. * * Emits a {Transfer} event. */ function _mint(address to, uint256 tokenId) internal virtual { require(to != address(0), "ERC721: mint to the zero address"); require(!_exists(tokenId), "ERC721: token already minted"); _beforeTokenTransfer(address(0), to, tokenId); _balances[to] += 1; _owners[tokenId] = to; emit Transfer(address(0), to, tokenId); _afterTokenTransfer(address(0), to, tokenId); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId) internal virtual { address owner = ERC721Upgradeable.ownerOf(tokenId); _beforeTokenTransfer(owner, address(0), tokenId); // Clear approvals _approve(address(0), tokenId); _balances[owner] -= 1; delete _owners[tokenId]; emit Transfer(owner, address(0), tokenId); _afterTokenTransfer(owner, address(0), tokenId); } /** * @dev Transfers `tokenId` from `from` to `to`. * As opposed to {transferFrom}, this imposes no restrictions on msg.sender. * * Requirements: * * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * * Emits a {Transfer} event. */ function _transfer( address from, address to, uint256 tokenId ) internal virtual { require(ERC721Upgradeable.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner"); require(to != address(0), "ERC721: transfer to the zero address"); _beforeTokenTransfer(from, to, tokenId); // Clear approvals from the previous owner _approve(address(0), tokenId); _balances[from] -= 1; _balances[to] += 1; _owners[tokenId] = to; emit Transfer(from, to, tokenId); _afterTokenTransfer(from, to, tokenId); } /** * @dev Approve `to` to operate on `tokenId` * * Emits an {Approval} event. */ function _approve(address to, uint256 tokenId) internal virtual { _tokenApprovals[tokenId] = to; emit Approval(ERC721Upgradeable.ownerOf(tokenId), to, tokenId); } /** * @dev Approve `operator` to operate on all of `owner` tokens * * Emits an {ApprovalForAll} event. */ function _setApprovalForAll( address owner, address operator, bool approved ) internal virtual { require(owner != operator, "ERC721: approve to caller"); _operatorApprovals[owner][operator] = approved; emit ApprovalForAll(owner, operator, approved); } /** * @dev Reverts if the `tokenId` has not been minted yet. */ function _requireMinted(uint256 tokenId) internal view virtual { require(_exists(tokenId), "ERC721: invalid token ID"); } /** * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address. * The call is not executed if the target address is not a contract. * * @param from address representing the previous owner of the given token ID * @param to target address that will receive the tokens * @param tokenId uint256 ID of the token to be transferred * @param data bytes optional data to send along with the call * @return bool whether the call correctly returned the expected magic value */ function _checkOnERC721Received( address from, address to, uint256 tokenId, bytes memory data ) private returns (bool) { if (to.isContract()) { try IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) { return retval == IERC721ReceiverUpgradeable.onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert("ERC721: transfer to non ERC721Receiver implementer"); } else { /// @solidity memory-safe-assembly assembly { revert(add(32, reason), mload(reason)) } } } } else { return true; } } /** * @dev Hook that is called before any token transfer. This includes minting * and burning. * * 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, ``from``'s `tokenId` will be burned. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal virtual {} /** * @dev Hook that is called after any transfer of tokens. This includes * minting and burning. * * Calling conditions: * * - when `from` and `to` are both non-zero. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _afterTokenTransfer( address from, address to, uint256 tokenId ) internal virtual {} /** * @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[44] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/extensions/ERC721Burnable.sol) pragma solidity ^0.8.0; import "../ERC721Upgradeable.sol"; import "../../../utils/ContextUpgradeable.sol"; import "../../../proxy/utils/Initializable.sol"; /** * @title ERC721 Burnable Token * @dev ERC721 Token that can be burned (destroyed). */ abstract contract ERC721BurnableUpgradeable is Initializable, ContextUpgradeable, ERC721Upgradeable { function __ERC721Burnable_init() internal onlyInitializing { } function __ERC721Burnable_init_unchained() internal onlyInitializing { } /** * @dev Burns `tokenId`. See {ERC721-_burn}. * * Requirements: * * - The caller must own `tokenId` or be an approved operator. */ function burn(uint256 tokenId) public virtual { //solhint-disable-next-line max-line-length require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved"); _burn(tokenId); } /** * @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: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/ERC721Enumerable.sol) pragma solidity ^0.8.0; import "../ERC721Upgradeable.sol"; import "./IERC721EnumerableUpgradeable.sol"; import "../../../proxy/utils/Initializable.sol"; /** * @dev This implements an optional extension of {ERC721} defined in the EIP that adds * enumerability of all the token ids in the contract as well as all token ids owned by each * account. */ abstract contract ERC721EnumerableUpgradeable is Initializable, ERC721Upgradeable, IERC721EnumerableUpgradeable { function __ERC721Enumerable_init() internal onlyInitializing { } function __ERC721Enumerable_init_unchained() internal onlyInitializing { } // Mapping from owner to list of owned token IDs mapping(address => mapping(uint256 => uint256)) private _ownedTokens; // Mapping from token ID to index of the owner tokens list mapping(uint256 => uint256) private _ownedTokensIndex; // Array with all token ids, used for enumeration uint256[] private _allTokens; // Mapping from token id to position in the allTokens array mapping(uint256 => uint256) private _allTokensIndex; /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165Upgradeable, ERC721Upgradeable) returns (bool) { return interfaceId == type(IERC721EnumerableUpgradeable).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}. */ function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) { require(index < ERC721Upgradeable.balanceOf(owner), "ERC721Enumerable: owner index out of bounds"); return _ownedTokens[owner][index]; } /** * @dev See {IERC721Enumerable-totalSupply}. */ function totalSupply() public view virtual override returns (uint256) { return _allTokens.length; } /** * @dev See {IERC721Enumerable-tokenByIndex}. */ function tokenByIndex(uint256 index) public view virtual override returns (uint256) { require(index < ERC721EnumerableUpgradeable.totalSupply(), "ERC721Enumerable: global index out of bounds"); return _allTokens[index]; } /** * @dev Hook that is called before any token transfer. This includes minting * and burning. * * 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, ``from``'s `tokenId` will be burned. * - `from` cannot be the zero address. * - `to` cannot be the zero address. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal virtual override { super._beforeTokenTransfer(from, to, tokenId); if (from == address(0)) { _addTokenToAllTokensEnumeration(tokenId); } else if (from != to) { _removeTokenFromOwnerEnumeration(from, tokenId); } if (to == address(0)) { _removeTokenFromAllTokensEnumeration(tokenId); } else if (to != from) { _addTokenToOwnerEnumeration(to, tokenId); } } /** * @dev Private function to add a token to this extension's ownership-tracking data structures. * @param to address representing the new owner of the given token ID * @param tokenId uint256 ID of the token to be added to the tokens list of the given address */ function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private { uint256 length = ERC721Upgradeable.balanceOf(to); _ownedTokens[to][length] = tokenId; _ownedTokensIndex[tokenId] = length; } /** * @dev Private function to add a token to this extension's token tracking data structures. * @param tokenId uint256 ID of the token to be added to the tokens list */ function _addTokenToAllTokensEnumeration(uint256 tokenId) private { _allTokensIndex[tokenId] = _allTokens.length; _allTokens.push(tokenId); } /** * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for * gas optimizations e.g. when performing a transfer operation (avoiding double writes). * This has O(1) time complexity, but alters the order of the _ownedTokens array. * @param from address representing the previous owner of the given token ID * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address */ function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private { // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and // then delete the last slot (swap and pop). uint256 lastTokenIndex = ERC721Upgradeable.balanceOf(from) - 1; uint256 tokenIndex = _ownedTokensIndex[tokenId]; // When the token to delete is the last token, the swap operation is unnecessary if (tokenIndex != lastTokenIndex) { uint256 lastTokenId = _ownedTokens[from][lastTokenIndex]; _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index } // This also deletes the contents at the last position of the array delete _ownedTokensIndex[tokenId]; delete _ownedTokens[from][lastTokenIndex]; } /** * @dev Private function to remove a token from this extension's token tracking data structures. * This has O(1) time complexity, but alters the order of the _allTokens array. * @param tokenId uint256 ID of the token to be removed from the tokens list */ function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private { // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and // then delete the last slot (swap and pop). uint256 lastTokenIndex = _allTokens.length - 1; uint256 tokenIndex = _allTokensIndex[tokenId]; // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding // an 'if' statement (like in _removeTokenFromOwnerEnumeration) uint256 lastTokenId = _allTokens[lastTokenIndex]; _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index // This also deletes the contents at the last position of the array delete _allTokensIndex[tokenId]; _allTokens.pop(); } /** * @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[46] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/extensions/ERC721URIStorage.sol) pragma solidity ^0.8.0; import "../ERC721Upgradeable.sol"; import "../../../proxy/utils/Initializable.sol"; /** * @dev ERC721 token with storage based token URI management. */ abstract contract ERC721URIStorageUpgradeable is Initializable, ERC721Upgradeable { function __ERC721URIStorage_init() internal onlyInitializing { } function __ERC721URIStorage_init_unchained() internal onlyInitializing { } using StringsUpgradeable for uint256; // Optional mapping for token URIs mapping(uint256 => string) private _tokenURIs; /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { _requireMinted(tokenId); string memory _tokenURI = _tokenURIs[tokenId]; string memory base = _baseURI(); // If there is no base URI, return the token URI. if (bytes(base).length == 0) { return _tokenURI; } // If both are set, concatenate the baseURI and tokenURI (via abi.encodePacked). if (bytes(_tokenURI).length > 0) { return string(abi.encodePacked(base, _tokenURI)); } return super.tokenURI(tokenId); } /** * @dev Sets `_tokenURI` as the tokenURI of `tokenId`. * * Requirements: * * - `tokenId` must exist. */ function _setTokenURI(uint256 tokenId, string memory _tokenURI) internal virtual { require(_exists(tokenId), "ERC721URIStorage: URI set of nonexistent token"); _tokenURIs[tokenId] = _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]; } } /** * @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 (last updated v4.5.0) (token/ERC721/extensions/IERC721Enumerable.sol) pragma solidity ^0.8.0; import "../IERC721Upgradeable.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721EnumerableUpgradeable is IERC721Upgradeable { /** * @dev Returns the total amount of tokens stored by the contract. */ function totalSupply() external view returns (uint256); /** * @dev Returns a token ID owned by `owner` at a given `index` of its token list. * Use along with {balanceOf} to enumerate all of ``owner``'s tokens. */ function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256); /** * @dev Returns a token ID at a given `index` of all the tokens stored by the contract. * Use along with {totalSupply} to enumerate all tokens. */ function tokenByIndex(uint256 index) external view returns (uint256); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol) pragma solidity ^0.8.0; import "../IERC721Upgradeable.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional metadata extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721MetadataUpgradeable is IERC721Upgradeable { /** * @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 // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol) pragma solidity ^0.8.0; /** * @title ERC721 token receiver interface * @dev Interface for any contract that wants to support safeTransfers * from ERC721 asset contracts. */ interface IERC721ReceiverUpgradeable { /** * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom} * by `operator` from `from`, this function is called. * * It must return its Solidity selector to confirm the token transfer. * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted. * * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`. */ function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol) pragma solidity ^0.8.0; import "../../utils/introspection/IERC165Upgradeable.sol"; /** * @dev Required interface of an ERC721 compliant contract. */ interface IERC721Upgradeable is IERC165Upgradeable { /** * @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`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external; /** * @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 have been allowed to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Transfers `tokenId` token from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external; /** * @dev 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); }
// 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: MIT // OpenZeppelin Contracts v4.4.1 (utils/Counters.sol) pragma solidity ^0.8.0; /** * @title Counters * @author Matt Condon (@shrugs) * @dev Provides counters that can only be incremented, decremented or reset. This can be used e.g. to track the number * of elements in a mapping, issuing ERC721 ids, or counting request ids. * * Include with `using Counters for Counters.Counter;` */ library CountersUpgradeable { struct Counter { // This variable should never be directly accessed by users of the library: interactions must be restricted to // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add // this feature: see https://github.com/ethereum/solidity/issues/4637 uint256 _value; // default: 0 } function current(Counter storage counter) internal view returns (uint256) { return counter._value; } function increment(Counter storage counter) internal { unchecked { counter._value += 1; } } function decrement(Counter storage counter) internal { uint256 value = counter._value; require(value > 0, "Counter: decrement overflow"); unchecked { counter._value = value - 1; } } function reset(Counter storage counter) internal { counter._value = 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.3) (utils/cryptography/ECDSA.sol) pragma solidity ^0.8.0; import "../StringsUpgradeable.sol"; /** * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations. * * These functions can be used to verify that a message was signed by the holder * of the private keys of a given address. */ library ECDSAUpgradeable { enum RecoverError { NoError, InvalidSignature, InvalidSignatureLength, InvalidSignatureS, InvalidSignatureV } function _throwError(RecoverError error) private pure { if (error == RecoverError.NoError) { return; // no error: do nothing } else if (error == RecoverError.InvalidSignature) { revert("ECDSA: invalid signature"); } else if (error == RecoverError.InvalidSignatureLength) { revert("ECDSA: invalid signature length"); } else if (error == RecoverError.InvalidSignatureS) { revert("ECDSA: invalid signature 's' value"); } else if (error == RecoverError.InvalidSignatureV) { revert("ECDSA: invalid signature 'v' value"); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature` or error string. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. * * Documentation for signature generation: * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js] * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers] * * _Available since v4.3._ */ function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) { if (signature.length == 65) { bytes32 r; bytes32 s; uint8 v; // ecrecover takes the signature parameters, and the only way to get them // currently is to use assembly. /// @solidity memory-safe-assembly assembly { r := mload(add(signature, 0x20)) s := mload(add(signature, 0x40)) v := byte(0, mload(add(signature, 0x60))) } return tryRecover(hash, v, r, s); } else { return (address(0), RecoverError.InvalidSignatureLength); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature`. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. */ function recover(bytes32 hash, bytes memory signature) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, signature); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately. * * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures] * * _Available since v4.3._ */ function tryRecover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address, RecoverError) { bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff); uint8 v = uint8((uint256(vs) >> 255) + 27); return tryRecover(hash, v, r, s); } /** * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately. * * _Available since v4.2._ */ function recover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, r, vs); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `v`, * `r` and `s` signature fields separately. * * _Available since v4.3._ */ function tryRecover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address, RecoverError) { // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most // signatures from current libraries generate a unique signature with an s-value in the lower half order. // // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept // these malleable signatures as well. if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) { return (address(0), RecoverError.InvalidSignatureS); } if (v != 27 && v != 28) { return (address(0), RecoverError.InvalidSignatureV); } // If the signature is valid (and not malleable), return the signer address address signer = ecrecover(hash, v, r, s); if (signer == address(0)) { return (address(0), RecoverError.InvalidSignature); } return (signer, RecoverError.NoError); } /** * @dev Overload of {ECDSA-recover} that receives the `v`, * `r` and `s` signature fields separately. */ function recover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, v, r, s); _throwError(error); return recovered; } /** * @dev Returns an Ethereum Signed Message, created from a `hash`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) { // 32 is the length in bytes of hash, // enforced by the type signature above return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash)); } /** * @dev Returns an Ethereum Signed Message, created from `s`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", StringsUpgradeable.toString(s.length), s)); } /** * @dev Returns an Ethereum Signed Typed Data, created from a * `domainSeparator` and a `structHash`. This produces hash corresponding * to the one signed with the * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`] * JSON-RPC method as part of EIP-712. * * See {recover}. */ function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/cryptography/MerkleProof.sol) pragma solidity ^0.8.0; /** * @dev These functions deal with verification of Merkle Tree proofs. * * The proofs can be generated using the JavaScript library * https://github.com/miguelmota/merkletreejs[merkletreejs]. * Note: the hashing algorithm should be keccak256 and pair sorting should be enabled. * * See `test/utils/cryptography/MerkleProof.test.js` for some examples. * * WARNING: You should avoid using leaf values that are 64 bytes long prior to * hashing, or use a hash function other than keccak256 for hashing leaves. * This is because the concatenation of a sorted pair of internal nodes in * the merkle tree could be reinterpreted as a leaf value. */ library MerkleProofUpgradeable { /** * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree * defined by `root`. For this, a `proof` must be provided, containing * sibling hashes on the branch from the leaf to the root of the tree. Each * pair of leaves and each pair of pre-images are assumed to be sorted. */ function verify( bytes32[] memory proof, bytes32 root, bytes32 leaf ) internal pure returns (bool) { return processProof(proof, leaf) == root; } /** * @dev Calldata version of {verify} * * _Available since v4.7._ */ function verifyCalldata( bytes32[] calldata proof, bytes32 root, bytes32 leaf ) internal pure returns (bool) { return processProofCalldata(proof, leaf) == root; } /** * @dev Returns the rebuilt hash obtained by traversing a Merkle tree up * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt * hash matches the root of the tree. When processing the proof, the pairs * of leafs & pre-images are assumed to be sorted. * * _Available since v4.4._ */ function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Calldata version of {processProof} * * _Available since v4.7._ */ function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Returns true if the `leaves` can be proved to be a part of a Merkle tree defined by * `root`, according to `proof` and `proofFlags` as described in {processMultiProof}. * * _Available since v4.7._ */ function multiProofVerify( bytes32[] memory proof, bool[] memory proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProof(proof, proofFlags, leaves) == root; } /** * @dev Calldata version of {multiProofVerify} * * _Available since v4.7._ */ function multiProofVerifyCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProofCalldata(proof, proofFlags, leaves) == root; } /** * @dev Returns the root of a tree reconstructed from `leaves` and the sibling nodes in `proof`, * consuming from one or the other at each step according to the instructions given by * `proofFlags`. * * _Available since v4.7._ */ function processMultiProof( bytes32[] memory proof, bool[] memory proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the merkle tree. uint256 leavesLen = leaves.length; uint256 totalHashes = proofFlags.length; // Check proof validity. require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof"); // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value for the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { return hashes[totalHashes - 1]; } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } /** * @dev Calldata version of {processMultiProof} * * _Available since v4.7._ */ function processMultiProofCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the merkle tree. uint256 leavesLen = leaves.length; uint256 totalHashes = proofFlags.length; // Check proof validity. require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof"); // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value for the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { return hashes[totalHashes - 1]; } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) { return a < b ? _efficientHash(a, b) : _efficientHash(b, a); } function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) { /// @solidity memory-safe-assembly assembly { mstore(0x00, a) mstore(0x20, b) value := keccak256(0x00, 0x40) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.2) (utils/introspection/ERC165Checker.sol) pragma solidity ^0.8.0; 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. */ 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 _supportsERC165Interface(account, type(IERC165Upgradeable).interfaceId) && !_supportsERC165Interface(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) && _supportsERC165Interface(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] = _supportsERC165Interface(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 (!_supportsERC165Interface(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 _supportsERC165Interface(address account, bytes4 interfaceId) private 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 v4.4.1 (utils/introspection/ERC165.sol) pragma solidity ^0.8.0; import "./IERC165Upgradeable.sol"; import "../../proxy/utils/Initializable.sol"; /** * @dev Implementation of the {IERC165} interface. * * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check * for the additional interface id that will be supported. For example: * * ```solidity * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId); * } * ``` * * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation. */ abstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable { function __ERC165_init() internal onlyInitializing { } function __ERC165_init_unchained() internal onlyInitializing { } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IERC165Upgradeable).interfaceId; } /** * @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: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface 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.6.0) (utils/math/SafeMath.sol) pragma solidity ^0.8.0; // CAUTION // This version of SafeMath should only be used with Solidity 0.8 or later, // because it relies on the compiler's built in overflow checks. /** * @dev Wrappers over Solidity's arithmetic operations. * * NOTE: `SafeMath` is generally not needed starting with Solidity 0.8, since the compiler * now has built in overflow checking. */ library SafeMathUpgradeable { /** * @dev Returns the addition of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the subtraction of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an overflow flag. * * _Available since v3.4._ */ function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag. * * _Available since v3.4._ */ function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @dev Returns the addition of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `+` operator. * * Requirements: * * - Addition cannot overflow. */ function add(uint256 a, uint256 b) internal pure returns (uint256) { return a + b; } /** * @dev Returns the subtraction of two unsigned integers, reverting on * overflow (when the result is negative). * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub(uint256 a, uint256 b) internal pure returns (uint256) { return a - b; } /** * @dev Returns the multiplication of two unsigned integers, reverting on * overflow. * * Counterpart to Solidity's `*` operator. * * Requirements: * * - Multiplication cannot overflow. */ function mul(uint256 a, uint256 b) internal pure returns (uint256) { return a * b; } /** * @dev Returns the integer division of two unsigned integers, reverting on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. * * Requirements: * * - The divisor cannot be zero. */ function div(uint256 a, uint256 b) internal pure returns (uint256) { return a / b; } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting when dividing by zero. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod(uint256 a, uint256 b) internal pure returns (uint256) { return a % b; } /** * @dev Returns the subtraction of two unsigned integers, reverting with custom message on * overflow (when the result is negative). * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {trySub}. * * Counterpart to Solidity's `-` operator. * * Requirements: * * - Subtraction cannot overflow. */ function sub( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b <= a, errorMessage); return a - b; } } /** * @dev Returns the integer division of two unsigned integers, reverting with custom message on * division by zero. The result is rounded towards zero. * * Counterpart to Solidity's `/` operator. Note: this function uses a * `revert` opcode (which leaves remaining gas untouched) while Solidity * uses an invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function div( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a / b; } } /** * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo), * reverting with custom message when dividing by zero. * * CAUTION: This function is deprecated because it requires allocating memory for the error * message unnecessarily. For custom revert reasons use {tryMod}. * * Counterpart to Solidity's `%` operator. This function uses a `revert` * opcode (which leaves remaining gas untouched) while Solidity uses an * invalid opcode to revert (consuming all remaining gas). * * Requirements: * * - The divisor cannot be zero. */ function mod( uint256 a, uint256 b, string memory errorMessage ) internal pure returns (uint256) { unchecked { require(b > 0, errorMessage); return a % b; } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.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 pragma solidity ^0.8.9; import "solady/src/auth/OwnableRoles.sol"; import "@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/math/SafeMathUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/token/common/ERC2981Upgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/introspection/ERC165CheckerUpgradeable.sol"; import "./interfaces/IOperatorRegistry.sol"; import "./interfaces/IBaseCollection.sol"; import "./interfaces/INiftyKit.sol"; abstract contract BaseCollection is OwnableRoles, ContextUpgradeable, ERC2981Upgradeable, IBaseCollection { using AddressUpgradeable for address; using SafeMathUpgradeable for uint256; uint256 public constant ADMIN_ROLE = 1 << 0; uint256 public constant MANAGER_ROLE = 1 << 1; uint256 public constant BURNER_ROLE = 1 << 2; INiftyKit internal _niftyKit; address internal _operatorRegistry; address internal _treasury; uint256 internal _totalRevenue; // Operators mapping(bytes32 => bool) internal _allowedOperators; mapping(bytes32 => bool) internal _blockedOperators; // modifiers modifier preventBlockedOperator(address operator) { if (_operatorRegistry != address(0)) { bytes32 identifier = IOperatorRegistry(_operatorRegistry) .getIdentifier(operator); require( !_blockedOperators[identifier], "Operator has been blocked by contract owner." ); } _; } function __BaseCollection_init( address owner_, address treasury_, address royalty_, uint96 royaltyFee_ ) internal onlyInitializing { _initializeOwner(owner_); __ERC2981_init(); _niftyKit = INiftyKit(_msgSender()); _treasury = treasury_; _setDefaultRoyalty(royalty_, royaltyFee_); } function withdraw() external onlyRolesOrOwner(ADMIN_ROLE) { require(address(this).balance > 0, "0 balance"); INiftyKit niftyKit = _niftyKit; uint256 balance = address(this).balance; uint256 fees = niftyKit.getFees(address(this)); niftyKit.addFeesClaimed(fees); AddressUpgradeable.sendValue(payable(address(niftyKit)), fees); AddressUpgradeable.sendValue(payable(_treasury), balance.sub(fees)); } function setTreasury(address newTreasury) external onlyRolesOrOwner(ADMIN_ROLE) { _treasury = newTreasury; } function setDefaultRoyalty(address receiver, uint96 feeNumerator) external onlyRolesOrOwner(ADMIN_ROLE) { _setDefaultRoyalty(receiver, feeNumerator); } function setTokenRoyalty( uint256 tokenId, address receiver, uint96 feeNumerator ) external onlyRolesOrOwner(ADMIN_ROLE) { _setTokenRoyalty(tokenId, receiver, feeNumerator); } function setOperatorRegistry(address registry) external onlyRolesOrOwner(ADMIN_ROLE) { require( ERC165CheckerUpgradeable.supportsInterface( registry, type(IOperatorRegistry).interfaceId ), "Invalid Interface" ); _operatorRegistry = registry; } function unsetOperatorRegistry() external onlyRolesOrOwner(ADMIN_ROLE) { _operatorRegistry = address(0); } function setAllowedOperator(bytes32 operator, bool allowed) external onlyRolesOrOwner(MANAGER_ROLE) { _allowedOperators[operator] = allowed; emit OperatorAllowed(operator, allowed); } function setBlockedOperator(bytes32 operator, bool blocked) external onlyRolesOrOwner(MANAGER_ROLE) { _blockedOperators[operator] = blocked; emit OperatorBlocked(operator, blocked); } function isAllowedOperator(bytes32 operator) external view returns (bool) { return _allowedOperators[operator]; } function isBlockedOperator(bytes32 operator) external view returns (bool) { return _blockedOperators[operator]; } function treasury() external view returns (address) { return _treasury; } function totalRevenue() external view returns (uint256) { return _totalRevenue; } function operatorRegistry() external view returns (address) { return _operatorRegistry; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC2981Upgradeable) returns (bool) { return interfaceId == type(IBaseCollection).interfaceId || super.supportsInterface(interfaceId); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.9; interface IBaseCollection { event OperatorAllowed(bytes32 indexed operator, bool allowed); event OperatorBlocked(bytes32 indexed operator, bool blocked); /** * @dev Contract upgradeable initializer */ function initialize( address owner, string memory name, string memory symbol, address treasury, address royalty, uint96 royaltyFee ) external; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.9; interface INiftyKit { struct Entry { uint256 value; bool isValue; } /** * @dev Emitted when collection is created */ event CollectionCreated( uint96 indexed typeId, address indexed collectionAddress ); /** * @dev Emitted when user rate is updated */ event UserRateUpdated(address indexed user, uint256 indexed rate); /** * @dev Emitted when user rate is removed */ event UserRateRemoved(address indexed user, uint256 indexed rate); /** * @dev Returns the commission amount. */ function commission(address collection, uint256 amount) external view returns (uint256); /** * @dev Add fees from Collection */ function addFees(uint256 amount) external; /** * @dev Add fees claimed by the Collection */ function addFeesClaimed(uint256 amount) external; /** * @dev Get fees accrued by the account */ function getFees(address account) external view returns (uint256); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.9; interface IOperatorRegistry { event OperatorIdentified( bytes32 indexed identifer, address indexed operator ); function getIdentifier(address operator) external view returns (bytes32); }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.9; import "@openzeppelin/contracts-upgradeable/utils/math/SafeMathUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/cryptography/MerkleProofUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/cryptography/ECDSAUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol"; contract Redeemables is ContextUpgradeable { struct Redeemable { string tokenURI; uint256 price; uint256 maxQuantity; uint256 maxPerWallet; uint256 maxPerMint; uint256 redeemedCount; bytes32 merkleRoot; bool active; uint256 nonce; } event RedeemableCreated(uint256 indexed redeemableId); event TokenRedeemed( address indexed to, uint256 indexed redeemableId, uint256 quantity ); using SafeMathUpgradeable for uint256; using ECDSAUpgradeable for bytes32; using CountersUpgradeable for CountersUpgradeable.Counter; CountersUpgradeable.Counter private _redeemablesCounter; mapping(uint256 => Redeemable) private _redeemables; mapping(uint256 => mapping(address => uint256)) private _redeemedByWallet; function totalRedeemables() external view returns (uint256) { return _redeemablesCounter.current(); } function redeemableAt(uint256 index) public view returns (Redeemable memory data) { return _redeemables[index]; } function _createRedeemable( string memory uri, uint256 price, uint256 maxQuantity, uint256 maxPerWallet, uint256 maxPerMint ) internal { uint256 redeemableId = _redeemablesCounter.current(); _redeemablesCounter.increment(); _redeemables[redeemableId] = Redeemable({ tokenURI: uri, price: price, maxQuantity: maxQuantity, maxPerWallet: maxPerWallet, maxPerMint: maxPerMint, redeemedCount: 0, merkleRoot: "", active: true, nonce: 0 }); emit RedeemableCreated(redeemableId); } function _redeem( uint256 redeemableId, uint256 quantity, bytes calldata signature, address signer, bytes32[] calldata proof ) internal { Redeemable memory redeemable = redeemableAt(redeemableId); require(redeemable.active, "Not active"); require(redeemable.price.mul(quantity) <= msg.value, "Value incorrect"); require(quantity <= redeemable.maxPerMint, "Exceeded max per mint"); require( redeemable.redeemedCount.add(quantity) <= redeemable.maxQuantity, "Exceeded max amount" ); require( _redeemedByWallet[redeemableId][_msgSender()].add(quantity) <= redeemable.maxPerWallet, "Exceeded max per wallet" ); require( keccak256(abi.encodePacked(redeemableId.add(redeemable.nonce))) .toEthSignedMessageHash() .recover(signature) == signer, "Invalid signature" ); if (redeemable.merkleRoot != "") { require( MerkleProofUpgradeable.verify( proof, redeemable.merkleRoot, keccak256(abi.encodePacked(_msgSender())) ), "Invalid proof" ); } unchecked { _redeemables[redeemableId].redeemedCount++; _redeemedByWallet[redeemableId][_msgSender()]++; } emit TokenRedeemed(_msgSender(), redeemableId, quantity); } function _setMerkleRoot(uint256 redeemableId, bytes32 newRoot) internal { require(_redeemables[redeemableId].active, "Not active"); _redeemables[redeemableId].merkleRoot = newRoot; } function _invalidate(uint256 redeemableId) internal { require(_redeemables[redeemableId].active, "Not active"); _redeemables[redeemableId].nonce = _redeemables[redeemableId].nonce.add( 1 ); } function _revoke(uint256 redeemableId) internal { require(_redeemables[redeemableId].active, "Not active"); _redeemables[redeemableId].active = false; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; /// @notice Simple single owner and multiroles authorization mixin. /// @author Solady (https://github.com/vectorized/solady/blob/main/src/auth/OwnableRoles.sol) /// @dev While the ownable portion follows [EIP-173](https://eips.ethereum.org/EIPS/eip-173) /// for compatibility, the nomenclature for the 2-step ownership handover and roles /// may be unique to this codebase. abstract contract OwnableRoles { /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* CUSTOM ERRORS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The caller is not authorized to call the function. error Unauthorized(); /// @dev The `newOwner` cannot be the zero address. error NewOwnerIsZeroAddress(); /// @dev The `pendingOwner` does not have a valid handover request. error NoHandoverRequest(); /// @dev `bytes4(keccak256(bytes("Unauthorized()")))`. uint256 private constant _UNAUTHORIZED_ERROR_SELECTOR = 0x82b42900; /// @dev `bytes4(keccak256(bytes("NewOwnerIsZeroAddress()")))`. uint256 private constant _NEW_OWNER_IS_ZERO_ADDRESS_ERROR_SELECTOR = 0x7448fbae; /// @dev `bytes4(keccak256(bytes("NoHandoverRequest()")))`. uint256 private constant _NO_HANDOVER_REQUEST_ERROR_SELECTOR = 0x6f5e8818; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* EVENTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The ownership is transferred from `oldOwner` to `newOwner`. /// This event is intentionally kept the same as OpenZeppelin's Ownable to be /// compatible with indexers and [EIP-173](https://eips.ethereum.org/EIPS/eip-173), /// despite it not being as lightweight as a single argument event. event OwnershipTransferred(address indexed oldOwner, address indexed newOwner); /// @dev An ownership handover to `pendingOwner` has been requested. event OwnershipHandoverRequested(address indexed pendingOwner); /// @dev The ownership handover to `pendingOwner` has been cancelled. event OwnershipHandoverCanceled(address indexed pendingOwner); /// @dev The `user`'s roles is updated to `roles`. /// Each bit of `roles` represents whether the role is set. event RolesUpdated(address indexed user, uint256 indexed roles); /// @dev `keccak256(bytes("OwnershipTransferred(address,address)"))`. uint256 private constant _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE = 0x8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0; /// @dev `keccak256(bytes("OwnershipHandoverRequested(address)"))`. uint256 private constant _OWNERSHIP_HANDOVER_REQUESTED_EVENT_SIGNATURE = 0xdbf36a107da19e49527a7176a1babf963b4b0ff8cde35ee35d6cd8f1f9ac7e1d; /// @dev `keccak256(bytes("OwnershipHandoverCanceled(address)"))`. uint256 private constant _OWNERSHIP_HANDOVER_CANCELED_EVENT_SIGNATURE = 0xfa7b8eab7da67f412cc9575ed43464468f9bfbae89d1675917346ca6d8fe3c92; /// @dev `keccak256(bytes("RolesUpdated(address,uint256)"))`. uint256 private constant _ROLES_UPDATED_EVENT_SIGNATURE = 0x715ad5ce61fc9595c7b415289d59cf203f23a94fa06f04af7e489a0a76e1fe26; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* STORAGE */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The owner slot is given by: `not(_OWNER_SLOT_NOT)`. /// It is intentionally choosen to be a high value /// to avoid collision with lower slots. /// The choice of manual storage layout is to enable compatibility /// with both regular and upgradeable contracts. /// /// The role slot of `user` is given by: /// ``` /// mstore(0x00, or(shl(96, user), _OWNER_SLOT_NOT)) /// let roleSlot := keccak256(0x00, 0x20) /// ``` /// This automatically ignores the upper bits of the `user` in case /// they are not clean, as well as keep the `keccak256` under 32-bytes. uint256 private constant _OWNER_SLOT_NOT = 0x8b78c6d8; /// The ownership handover slot of `newOwner` is given by: /// ``` /// mstore(0x00, or(shl(96, user), _HANDOVER_SLOT_SEED)) /// let handoverSlot := keccak256(0x00, 0x20) /// ``` /// It stores the expiry timestamp of the two-step ownership handover. uint256 private constant _HANDOVER_SLOT_SEED = 0x389a75e1; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* INTERNAL FUNCTIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Initializes the owner directly without authorization guard. /// This function must be called upon initialization, /// regardless of whether the contract is upgradeable or not. /// This is to enable generalization to both regular and upgradeable contracts, /// and to save gas in case the initial owner is not the caller. /// For performance reasons, this function will not check if there /// is an existing owner. function _initializeOwner(address newOwner) internal virtual { assembly { // Clean the upper 96 bits. newOwner := shr(96, shl(96, newOwner)) // Store the new value. sstore(not(_OWNER_SLOT_NOT), newOwner) // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, 0, newOwner) } } /// @dev Sets the owner directly without authorization guard. function _setOwner(address newOwner) internal virtual { assembly { let ownerSlot := not(_OWNER_SLOT_NOT) // Clean the upper 96 bits. newOwner := shr(96, shl(96, newOwner)) // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, sload(ownerSlot), newOwner) // Store the new value. sstore(ownerSlot, newOwner) } } /// @dev Grants the roles directly without authorization guard. /// Each bit of `roles` represents the role to turn on. function _grantRoles(address user, uint256 roles) internal virtual { assembly { // Compute the role slot. mstore(0x00, or(shl(96, user), _OWNER_SLOT_NOT)) let roleSlot := keccak256(0x00, 0x20) // Load the current value and `or` it with `roles`. let newRoles := or(sload(roleSlot), roles) // Store the new value. sstore(roleSlot, newRoles) // Emit the {RolesUpdated} event. log3(0, 0, _ROLES_UPDATED_EVENT_SIGNATURE, shr(96, shl(96, user)), newRoles) } } /// @dev Removes the roles directly without authorization guard. /// Each bit of `roles` represents the role to turn off. function _removeRoles(address user, uint256 roles) internal virtual { assembly { // Compute the role slot. mstore(0x00, or(shl(96, user), _OWNER_SLOT_NOT)) let roleSlot := keccak256(0x00, 0x20) // Load the current value. let currentRoles := sload(roleSlot) // Use `and` to compute the intersection of `currentRoles` and `roles`, // `xor` it with `currentRoles` to flip the bits in the intersection. let newRoles := xor(currentRoles, and(currentRoles, roles)) // Then, store the new value. sstore(roleSlot, newRoles) // Emit the {RolesUpdated} event. log3(0, 0, _ROLES_UPDATED_EVENT_SIGNATURE, shr(96, shl(96, user)), newRoles) } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* PUBLIC UPDATE FUNCTIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Allows the owner to transfer the ownership to `newOwner`. function transferOwnership(address newOwner) public virtual onlyOwner { assembly { // Clean the upper 96 bits. newOwner := shr(96, shl(96, newOwner)) // Reverts if the `newOwner` is the zero address. if iszero(newOwner) { mstore(0x00, _NEW_OWNER_IS_ZERO_ADDRESS_ERROR_SELECTOR) revert(0x1c, 0x04) } // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, caller(), newOwner) // Store the new value. sstore(not(_OWNER_SLOT_NOT), newOwner) } } /// @dev Allows the owner to renounce their ownership. function renounceOwnership() public virtual onlyOwner { assembly { // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, caller(), 0) // Store the new value. sstore(not(_OWNER_SLOT_NOT), 0) } } /// @dev Request a two-step ownership handover to the caller. /// The request will be automatically expire in 48 hours (172800 seconds) by default. function requestOwnershipHandover() public virtual { unchecked { uint256 expires = block.timestamp + ownershipHandoverValidFor(); assembly { // Compute and set the handover slot to 1. mstore(0x00, or(shl(96, caller()), _HANDOVER_SLOT_SEED)) sstore(keccak256(0x00, 0x20), expires) // Emit the {OwnershipHandoverRequested} event. log2(0, 0, _OWNERSHIP_HANDOVER_REQUESTED_EVENT_SIGNATURE, caller()) } } } /// @dev Cancels the two-step ownership handover to the caller, if any. function cancelOwnershipHandover() public virtual { assembly { // Compute and set the handover slot to 0. mstore(0x00, or(shl(96, caller()), _HANDOVER_SLOT_SEED)) sstore(keccak256(0x00, 0x20), 0) // Emit the {OwnershipHandoverCanceled} event. log2(0, 0, _OWNERSHIP_HANDOVER_CANCELED_EVENT_SIGNATURE, caller()) } } /// @dev Allows the owner to complete the two-step ownership handover to `pendingOwner`. /// Reverts if there is no existing ownership handover requested by `pendingOwner`. function completeOwnershipHandover(address pendingOwner) public virtual onlyOwner { assembly { // Clean the upper 96 bits. pendingOwner := shr(96, shl(96, pendingOwner)) // Compute and set the handover slot to 0. mstore(0x00, or(shl(96, pendingOwner), _HANDOVER_SLOT_SEED)) let handoverSlot := keccak256(0x00, 0x20) // If the handover does not exist, or has expired. if gt(timestamp(), sload(handoverSlot)) { mstore(0x00, _NO_HANDOVER_REQUEST_ERROR_SELECTOR) revert(0x1c, 0x04) } // Set the handover slot to 0. sstore(handoverSlot, 0) // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, caller(), pendingOwner) // Store the new value. sstore(not(_OWNER_SLOT_NOT), pendingOwner) } } /// @dev Allows the owner to grant `user` `roles`. /// If the `user` already has a role, then it will be an no-op for the role. function grantRoles(address user, uint256 roles) public virtual onlyOwner { _grantRoles(user, roles); } /// @dev Allows the owner to remove `user` `roles`. /// If the `user` does not have a role, then it will be an no-op for the role. function revokeRoles(address user, uint256 roles) public virtual onlyOwner { _removeRoles(user, roles); } /// @dev Allow the caller to remove their own roles. /// If the caller does not have a role, then it will be an no-op for the role. function renounceRoles(uint256 roles) public virtual { _removeRoles(msg.sender, roles); } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* PUBLIC READ FUNCTIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns the owner of the contract. function owner() public view virtual returns (address result) { assembly { result := sload(not(_OWNER_SLOT_NOT)) } } /// @dev Returns the expiry timestamp for the two-step ownership handover to `pendingOwner`. function ownershipHandoverExpiresAt(address pendingOwner) public view virtual returns (uint256 result) { assembly { // Compute the handover slot. mstore(0x00, or(shl(96, pendingOwner), _HANDOVER_SLOT_SEED)) // Load the handover slot. result := sload(keccak256(0x00, 0x20)) } } /// @dev Returns how long a two-step ownership handover is valid for in seconds. function ownershipHandoverValidFor() public view virtual returns (uint64) { return 48 * 3600; } /// @dev Returns whether `user` has any of `roles`. function hasAnyRole(address user, uint256 roles) public view virtual returns (bool result) { assembly { // Compute the role slot. mstore(0x00, or(shl(96, user), _OWNER_SLOT_NOT)) // Load the stored value, and set the result to whether the // `and` intersection of the value and `roles` is not zero. result := iszero(iszero(and(sload(keccak256(0x00, 0x20)), roles))) } } /// @dev Returns whether `user` has all of `roles`. function hasAllRoles(address user, uint256 roles) public view virtual returns (bool result) { assembly { // Compute the role slot. mstore(0x00, or(shl(96, user), _OWNER_SLOT_NOT)) // Whether the stored value is contains all the set bits in `roles`. result := eq(and(sload(keccak256(0x00, 0x20)), roles), roles) } } /// @dev Returns the roles of `user`. function rolesOf(address user) public view virtual returns (uint256 roles) { assembly { // Compute the role slot. mstore(0x00, or(shl(96, user), _OWNER_SLOT_NOT)) // Load the stored value. roles := sload(keccak256(0x00, 0x20)) } } /// @dev Convenience function to return a `roles` bitmap from the `ordinals`. /// This is meant for frontends like Etherscan, and is therefore not fully optimized. /// Not recommended to be called on-chain. function rolesFromOrdinals(uint8[] memory ordinals) public pure returns (uint256 roles) { assembly { // Skip the length slot. let o := add(ordinals, 0x20) // `shl` 5 is equivalent to multiplying by 0x20. let end := add(o, shl(5, mload(ordinals))) // prettier-ignore for {} iszero(eq(o, end)) { o := add(o, 0x20) } { roles := or(roles, shl(and(mload(o), 0xff), 1)) } } } /// @dev Convenience function to return a `roles` bitmap from the `ordinals`. /// This is meant for frontends like Etherscan, and is therefore not fully optimized. /// Not recommended to be called on-chain. function ordinalsFromRoles(uint256 roles) public pure returns (uint8[] memory ordinals) { assembly { // Grab the pointer to the free memory. let ptr := add(mload(0x40), 0x20) // The absence of lookup tables, De Bruijn, etc., here is intentional for // smaller bytecode, as this function is not meant to be called on-chain. // prettier-ignore for { let i := 0 } 1 { i := add(i, 1) } { mstore(ptr, i) // `shr` 5 is equivalent to multiplying by 0x20. // Push back into the ordinals array if the bit is set. ptr := add(ptr, shl(5, and(roles, 1))) roles := shr(1, roles) // prettier-ignore if iszero(roles) { break } } // Set `ordinals` to the start of the free memory. ordinals := mload(0x40) // Allocate the memory. mstore(0x40, ptr) // Store the length of `ordinals`. mstore(ordinals, shr(5, sub(ptr, add(ordinals, 0x20)))) } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* MODIFIERS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Marks a function as only callable by the owner. modifier onlyOwner() virtual { assembly { // If the caller is not the stored owner, revert. if iszero(eq(caller(), sload(not(_OWNER_SLOT_NOT)))) { mstore(0x00, _UNAUTHORIZED_ERROR_SELECTOR) revert(0x1c, 0x04) } } _; } /// @dev Marks a function as only callable by an account with `roles`. modifier onlyRoles(uint256 roles) virtual { assembly { // Compute the role slot. mstore(0x00, or(shl(96, caller()), _OWNER_SLOT_NOT)) // Load the stored value, and if the `and` intersection // of the value and `roles` is zero, revert. if iszero(and(sload(keccak256(0x00, 0x20)), roles)) { mstore(0x00, _UNAUTHORIZED_ERROR_SELECTOR) revert(0x1c, 0x04) } } _; } /// @dev Marks a function as only callable by the owner or by an account /// with `roles`. Checks for ownership first, then lazily checks for roles. modifier onlyOwnerOrRoles(uint256 roles) virtual { assembly { // If the caller is not the stored owner. if iszero(eq(caller(), sload(not(_OWNER_SLOT_NOT)))) { // Compute the role slot. mstore(0x00, or(shl(96, caller()), _OWNER_SLOT_NOT)) // Load the stored value, and if the `and` intersection // of the value and `roles` is zero, revert. if iszero(and(sload(keccak256(0x00, 0x20)), roles)) { mstore(0x00, _UNAUTHORIZED_ERROR_SELECTOR) revert(0x1c, 0x04) } } } _; } /// @dev Marks a function as only callable by an account with `roles` /// or the owner. Checks for roles first, then lazily checks for ownership. modifier onlyRolesOrOwner(uint256 roles) virtual { assembly { // Compute the role slot. mstore(0x00, or(shl(96, caller()), _OWNER_SLOT_NOT)) // Load the stored value, and if the `and` intersection // of the value and `roles` is zero, revert. if iszero(and(sload(keccak256(0x00, 0x20)), roles)) { // If the caller is not the stored owner. if iszero(eq(caller(), sload(not(_OWNER_SLOT_NOT)))) { mstore(0x00, _UNAUTHORIZED_ERROR_SELECTOR) revert(0x1c, 0x04) } } } _; } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* ROLE CONSTANTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ // IYKYK uint256 internal constant _ROLE_0 = 1 << 0; uint256 internal constant _ROLE_1 = 1 << 1; uint256 internal constant _ROLE_2 = 1 << 2; uint256 internal constant _ROLE_3 = 1 << 3; uint256 internal constant _ROLE_4 = 1 << 4; uint256 internal constant _ROLE_5 = 1 << 5; uint256 internal constant _ROLE_6 = 1 << 6; uint256 internal constant _ROLE_7 = 1 << 7; uint256 internal constant _ROLE_8 = 1 << 8; uint256 internal constant _ROLE_9 = 1 << 9; uint256 internal constant _ROLE_10 = 1 << 10; uint256 internal constant _ROLE_11 = 1 << 11; uint256 internal constant _ROLE_12 = 1 << 12; uint256 internal constant _ROLE_13 = 1 << 13; uint256 internal constant _ROLE_14 = 1 << 14; uint256 internal constant _ROLE_15 = 1 << 15; uint256 internal constant _ROLE_16 = 1 << 16; uint256 internal constant _ROLE_17 = 1 << 17; uint256 internal constant _ROLE_18 = 1 << 18; uint256 internal constant _ROLE_19 = 1 << 19; uint256 internal constant _ROLE_20 = 1 << 20; uint256 internal constant _ROLE_21 = 1 << 21; uint256 internal constant _ROLE_22 = 1 << 22; uint256 internal constant _ROLE_23 = 1 << 23; uint256 internal constant _ROLE_24 = 1 << 24; uint256 internal constant _ROLE_25 = 1 << 25; uint256 internal constant _ROLE_26 = 1 << 26; uint256 internal constant _ROLE_27 = 1 << 27; uint256 internal constant _ROLE_28 = 1 << 28; uint256 internal constant _ROLE_29 = 1 << 29; uint256 internal constant _ROLE_30 = 1 << 30; uint256 internal constant _ROLE_31 = 1 << 31; uint256 internal constant _ROLE_32 = 1 << 32; uint256 internal constant _ROLE_33 = 1 << 33; uint256 internal constant _ROLE_34 = 1 << 34; uint256 internal constant _ROLE_35 = 1 << 35; uint256 internal constant _ROLE_36 = 1 << 36; uint256 internal constant _ROLE_37 = 1 << 37; uint256 internal constant _ROLE_38 = 1 << 38; uint256 internal constant _ROLE_39 = 1 << 39; uint256 internal constant _ROLE_40 = 1 << 40; uint256 internal constant _ROLE_41 = 1 << 41; uint256 internal constant _ROLE_42 = 1 << 42; uint256 internal constant _ROLE_43 = 1 << 43; uint256 internal constant _ROLE_44 = 1 << 44; uint256 internal constant _ROLE_45 = 1 << 45; uint256 internal constant _ROLE_46 = 1 << 46; uint256 internal constant _ROLE_47 = 1 << 47; uint256 internal constant _ROLE_48 = 1 << 48; uint256 internal constant _ROLE_49 = 1 << 49; uint256 internal constant _ROLE_50 = 1 << 50; uint256 internal constant _ROLE_51 = 1 << 51; uint256 internal constant _ROLE_52 = 1 << 52; uint256 internal constant _ROLE_53 = 1 << 53; uint256 internal constant _ROLE_54 = 1 << 54; uint256 internal constant _ROLE_55 = 1 << 55; uint256 internal constant _ROLE_56 = 1 << 56; uint256 internal constant _ROLE_57 = 1 << 57; uint256 internal constant _ROLE_58 = 1 << 58; uint256 internal constant _ROLE_59 = 1 << 59; uint256 internal constant _ROLE_60 = 1 << 60; uint256 internal constant _ROLE_61 = 1 << 61; uint256 internal constant _ROLE_62 = 1 << 62; uint256 internal constant _ROLE_63 = 1 << 63; uint256 internal constant _ROLE_64 = 1 << 64; uint256 internal constant _ROLE_65 = 1 << 65; uint256 internal constant _ROLE_66 = 1 << 66; uint256 internal constant _ROLE_67 = 1 << 67; uint256 internal constant _ROLE_68 = 1 << 68; uint256 internal constant _ROLE_69 = 1 << 69; uint256 internal constant _ROLE_70 = 1 << 70; uint256 internal constant _ROLE_71 = 1 << 71; uint256 internal constant _ROLE_72 = 1 << 72; uint256 internal constant _ROLE_73 = 1 << 73; uint256 internal constant _ROLE_74 = 1 << 74; uint256 internal constant _ROLE_75 = 1 << 75; uint256 internal constant _ROLE_76 = 1 << 76; uint256 internal constant _ROLE_77 = 1 << 77; uint256 internal constant _ROLE_78 = 1 << 78; uint256 internal constant _ROLE_79 = 1 << 79; uint256 internal constant _ROLE_80 = 1 << 80; uint256 internal constant _ROLE_81 = 1 << 81; uint256 internal constant _ROLE_82 = 1 << 82; uint256 internal constant _ROLE_83 = 1 << 83; uint256 internal constant _ROLE_84 = 1 << 84; uint256 internal constant _ROLE_85 = 1 << 85; uint256 internal constant _ROLE_86 = 1 << 86; uint256 internal constant _ROLE_87 = 1 << 87; uint256 internal constant _ROLE_88 = 1 << 88; uint256 internal constant _ROLE_89 = 1 << 89; uint256 internal constant _ROLE_90 = 1 << 90; uint256 internal constant _ROLE_91 = 1 << 91; uint256 internal constant _ROLE_92 = 1 << 92; uint256 internal constant _ROLE_93 = 1 << 93; uint256 internal constant _ROLE_94 = 1 << 94; uint256 internal constant _ROLE_95 = 1 << 95; uint256 internal constant _ROLE_96 = 1 << 96; uint256 internal constant _ROLE_97 = 1 << 97; uint256 internal constant _ROLE_98 = 1 << 98; uint256 internal constant _ROLE_99 = 1 << 99; uint256 internal constant _ROLE_100 = 1 << 100; uint256 internal constant _ROLE_101 = 1 << 101; uint256 internal constant _ROLE_102 = 1 << 102; uint256 internal constant _ROLE_103 = 1 << 103; uint256 internal constant _ROLE_104 = 1 << 104; uint256 internal constant _ROLE_105 = 1 << 105; uint256 internal constant _ROLE_106 = 1 << 106; uint256 internal constant _ROLE_107 = 1 << 107; uint256 internal constant _ROLE_108 = 1 << 108; uint256 internal constant _ROLE_109 = 1 << 109; uint256 internal constant _ROLE_110 = 1 << 110; uint256 internal constant _ROLE_111 = 1 << 111; uint256 internal constant _ROLE_112 = 1 << 112; uint256 internal constant _ROLE_113 = 1 << 113; uint256 internal constant _ROLE_114 = 1 << 114; uint256 internal constant _ROLE_115 = 1 << 115; uint256 internal constant _ROLE_116 = 1 << 116; uint256 internal constant _ROLE_117 = 1 << 117; uint256 internal constant _ROLE_118 = 1 << 118; uint256 internal constant _ROLE_119 = 1 << 119; uint256 internal constant _ROLE_120 = 1 << 120; uint256 internal constant _ROLE_121 = 1 << 121; uint256 internal constant _ROLE_122 = 1 << 122; uint256 internal constant _ROLE_123 = 1 << 123; uint256 internal constant _ROLE_124 = 1 << 124; uint256 internal constant _ROLE_125 = 1 << 125; uint256 internal constant _ROLE_126 = 1 << 126; uint256 internal constant _ROLE_127 = 1 << 127; uint256 internal constant _ROLE_128 = 1 << 128; uint256 internal constant _ROLE_129 = 1 << 129; uint256 internal constant _ROLE_130 = 1 << 130; uint256 internal constant _ROLE_131 = 1 << 131; uint256 internal constant _ROLE_132 = 1 << 132; uint256 internal constant _ROLE_133 = 1 << 133; uint256 internal constant _ROLE_134 = 1 << 134; uint256 internal constant _ROLE_135 = 1 << 135; uint256 internal constant _ROLE_136 = 1 << 136; uint256 internal constant _ROLE_137 = 1 << 137; uint256 internal constant _ROLE_138 = 1 << 138; uint256 internal constant _ROLE_139 = 1 << 139; uint256 internal constant _ROLE_140 = 1 << 140; uint256 internal constant _ROLE_141 = 1 << 141; uint256 internal constant _ROLE_142 = 1 << 142; uint256 internal constant _ROLE_143 = 1 << 143; uint256 internal constant _ROLE_144 = 1 << 144; uint256 internal constant _ROLE_145 = 1 << 145; uint256 internal constant _ROLE_146 = 1 << 146; uint256 internal constant _ROLE_147 = 1 << 147; uint256 internal constant _ROLE_148 = 1 << 148; uint256 internal constant _ROLE_149 = 1 << 149; uint256 internal constant _ROLE_150 = 1 << 150; uint256 internal constant _ROLE_151 = 1 << 151; uint256 internal constant _ROLE_152 = 1 << 152; uint256 internal constant _ROLE_153 = 1 << 153; uint256 internal constant _ROLE_154 = 1 << 154; uint256 internal constant _ROLE_155 = 1 << 155; uint256 internal constant _ROLE_156 = 1 << 156; uint256 internal constant _ROLE_157 = 1 << 157; uint256 internal constant _ROLE_158 = 1 << 158; uint256 internal constant _ROLE_159 = 1 << 159; uint256 internal constant _ROLE_160 = 1 << 160; uint256 internal constant _ROLE_161 = 1 << 161; uint256 internal constant _ROLE_162 = 1 << 162; uint256 internal constant _ROLE_163 = 1 << 163; uint256 internal constant _ROLE_164 = 1 << 164; uint256 internal constant _ROLE_165 = 1 << 165; uint256 internal constant _ROLE_166 = 1 << 166; uint256 internal constant _ROLE_167 = 1 << 167; uint256 internal constant _ROLE_168 = 1 << 168; uint256 internal constant _ROLE_169 = 1 << 169; uint256 internal constant _ROLE_170 = 1 << 170; uint256 internal constant _ROLE_171 = 1 << 171; uint256 internal constant _ROLE_172 = 1 << 172; uint256 internal constant _ROLE_173 = 1 << 173; uint256 internal constant _ROLE_174 = 1 << 174; uint256 internal constant _ROLE_175 = 1 << 175; uint256 internal constant _ROLE_176 = 1 << 176; uint256 internal constant _ROLE_177 = 1 << 177; uint256 internal constant _ROLE_178 = 1 << 178; uint256 internal constant _ROLE_179 = 1 << 179; uint256 internal constant _ROLE_180 = 1 << 180; uint256 internal constant _ROLE_181 = 1 << 181; uint256 internal constant _ROLE_182 = 1 << 182; uint256 internal constant _ROLE_183 = 1 << 183; uint256 internal constant _ROLE_184 = 1 << 184; uint256 internal constant _ROLE_185 = 1 << 185; uint256 internal constant _ROLE_186 = 1 << 186; uint256 internal constant _ROLE_187 = 1 << 187; uint256 internal constant _ROLE_188 = 1 << 188; uint256 internal constant _ROLE_189 = 1 << 189; uint256 internal constant _ROLE_190 = 1 << 190; uint256 internal constant _ROLE_191 = 1 << 191; uint256 internal constant _ROLE_192 = 1 << 192; uint256 internal constant _ROLE_193 = 1 << 193; uint256 internal constant _ROLE_194 = 1 << 194; uint256 internal constant _ROLE_195 = 1 << 195; uint256 internal constant _ROLE_196 = 1 << 196; uint256 internal constant _ROLE_197 = 1 << 197; uint256 internal constant _ROLE_198 = 1 << 198; uint256 internal constant _ROLE_199 = 1 << 199; uint256 internal constant _ROLE_200 = 1 << 200; uint256 internal constant _ROLE_201 = 1 << 201; uint256 internal constant _ROLE_202 = 1 << 202; uint256 internal constant _ROLE_203 = 1 << 203; uint256 internal constant _ROLE_204 = 1 << 204; uint256 internal constant _ROLE_205 = 1 << 205; uint256 internal constant _ROLE_206 = 1 << 206; uint256 internal constant _ROLE_207 = 1 << 207; uint256 internal constant _ROLE_208 = 1 << 208; uint256 internal constant _ROLE_209 = 1 << 209; uint256 internal constant _ROLE_210 = 1 << 210; uint256 internal constant _ROLE_211 = 1 << 211; uint256 internal constant _ROLE_212 = 1 << 212; uint256 internal constant _ROLE_213 = 1 << 213; uint256 internal constant _ROLE_214 = 1 << 214; uint256 internal constant _ROLE_215 = 1 << 215; uint256 internal constant _ROLE_216 = 1 << 216; uint256 internal constant _ROLE_217 = 1 << 217; uint256 internal constant _ROLE_218 = 1 << 218; uint256 internal constant _ROLE_219 = 1 << 219; uint256 internal constant _ROLE_220 = 1 << 220; uint256 internal constant _ROLE_221 = 1 << 221; uint256 internal constant _ROLE_222 = 1 << 222; uint256 internal constant _ROLE_223 = 1 << 223; uint256 internal constant _ROLE_224 = 1 << 224; uint256 internal constant _ROLE_225 = 1 << 225; uint256 internal constant _ROLE_226 = 1 << 226; uint256 internal constant _ROLE_227 = 1 << 227; uint256 internal constant _ROLE_228 = 1 << 228; uint256 internal constant _ROLE_229 = 1 << 229; uint256 internal constant _ROLE_230 = 1 << 230; uint256 internal constant _ROLE_231 = 1 << 231; uint256 internal constant _ROLE_232 = 1 << 232; uint256 internal constant _ROLE_233 = 1 << 233; uint256 internal constant _ROLE_234 = 1 << 234; uint256 internal constant _ROLE_235 = 1 << 235; uint256 internal constant _ROLE_236 = 1 << 236; uint256 internal constant _ROLE_237 = 1 << 237; uint256 internal constant _ROLE_238 = 1 << 238; uint256 internal constant _ROLE_239 = 1 << 239; uint256 internal constant _ROLE_240 = 1 << 240; uint256 internal constant _ROLE_241 = 1 << 241; uint256 internal constant _ROLE_242 = 1 << 242; uint256 internal constant _ROLE_243 = 1 << 243; uint256 internal constant _ROLE_244 = 1 << 244; uint256 internal constant _ROLE_245 = 1 << 245; uint256 internal constant _ROLE_246 = 1 << 246; uint256 internal constant _ROLE_247 = 1 << 247; uint256 internal constant _ROLE_248 = 1 << 248; uint256 internal constant _ROLE_249 = 1 << 249; uint256 internal constant _ROLE_250 = 1 << 250; uint256 internal constant _ROLE_251 = 1 << 251; uint256 internal constant _ROLE_252 = 1 << 252; uint256 internal constant _ROLE_253 = 1 << 253; uint256 internal constant _ROLE_254 = 1 << 254; uint256 internal constant _ROLE_255 = 1 << 255; }
{ "optimizer": { "enabled": true, "runs": 200 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "metadata": { "useLiteralContent": true }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"NewOwnerIsZeroAddress","type":"error"},{"inputs":[],"name":"NoHandoverRequest","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":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"operator","type":"bytes32"},{"indexed":false,"internalType":"bool","name":"allowed","type":"bool"}],"name":"OperatorAllowed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"operator","type":"bytes32"},{"indexed":false,"internalType":"bool","name":"blocked","type":"bool"}],"name":"OperatorBlocked","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pendingOwner","type":"address"}],"name":"OwnershipHandoverCanceled","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pendingOwner","type":"address"}],"name":"OwnershipHandoverRequested","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"oldOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"redeemableId","type":"uint256"}],"name":"RedeemableCreated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":true,"internalType":"uint256","name":"roles","type":"uint256"}],"name":"RolesUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"redeemableId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"TokenRedeemed","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":[],"name":"ADMIN_ROLE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"BURNER_ROLE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MANAGER_ROLE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"cancelOwnershipHandover","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"pendingOwner","type":"address"}],"name":"completeOwnershipHandover","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"uri","type":"string"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"maxQuantity","type":"uint256"},{"internalType":"uint256","name":"maxPerWallet","type":"uint256"},{"internalType":"uint256","name":"maxPerMint","type":"uint256"}],"name":"createRedeemable","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"roles","type":"uint256"}],"name":"grantRoles","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"roles","type":"uint256"}],"name":"hasAllRoles","outputs":[{"internalType":"bool","name":"result","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"roles","type":"uint256"}],"name":"hasAnyRole","outputs":[{"internalType":"bool","name":"result","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner_","type":"address"},{"internalType":"string","name":"name_","type":"string"},{"internalType":"string","name":"symbol_","type":"string"},{"internalType":"address","name":"treasury_","type":"address"},{"internalType":"address","name":"royalty_","type":"address"},{"internalType":"uint96","name":"royaltyFee_","type":"uint96"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"redeemableId","type":"uint256"}],"name":"invalidate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"operator","type":"bytes32"}],"name":"isAllowedOperator","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"isOperator","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"operator","type":"bytes32"}],"name":"isBlockedOperator","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"quantity","type":"uint256"},{"internalType":"string","name":"uri","type":"string"}],"name":"mint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"operatorRegistry","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"roles","type":"uint256"}],"name":"ordinalsFromRoles","outputs":[{"internalType":"uint8[]","name":"ordinals","type":"uint8[]"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"result","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":"pendingOwner","type":"address"}],"name":"ownershipHandoverExpiresAt","outputs":[{"internalType":"uint256","name":"result","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ownershipHandoverValidFor","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"redeemableId","type":"uint256"},{"internalType":"uint256","name":"quantity","type":"uint256"},{"internalType":"bytes","name":"signature","type":"bytes"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"redeem","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"redeemableAt","outputs":[{"components":[{"internalType":"string","name":"tokenURI","type":"string"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"maxQuantity","type":"uint256"},{"internalType":"uint256","name":"maxPerWallet","type":"uint256"},{"internalType":"uint256","name":"maxPerMint","type":"uint256"},{"internalType":"uint256","name":"redeemedCount","type":"uint256"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"bool","name":"active","type":"bool"},{"internalType":"uint256","name":"nonce","type":"uint256"}],"internalType":"struct Redeemables.Redeemable","name":"data","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"roles","type":"uint256"}],"name":"renounceRoles","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"requestOwnershipHandover","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"redeemableId","type":"uint256"}],"name":"revoke","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"},{"internalType":"uint256","name":"roles","type":"uint256"}],"name":"revokeRoles","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint8[]","name":"ordinals","type":"uint8[]"}],"name":"rolesFromOrdinals","outputs":[{"internalType":"uint256","name":"roles","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"rolesOf","outputs":[{"internalType":"uint256","name":"roles","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"","type":"address"},{"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":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","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":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"operator","type":"bytes32"},{"internalType":"bool","name":"allowed","type":"bool"}],"name":"setAllowedOperator","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"operator","type":"bytes32"},{"internalType":"bool","name":"blocked","type":"bool"}],"name":"setBlockedOperator","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint96","name":"feeNumerator","type":"uint96"}],"name":"setDefaultRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"redeemableId","type":"uint256"},{"internalType":"bytes32","name":"newRoot","type":"bytes32"}],"name":"setMerkleRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"registry","type":"address"}],"name":"setOperatorRegistry","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint96","name":"feeNumerator","type":"uint96"}],"name":"setTokenRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"string","name":"newUri","type":"string"}],"name":"setTokenURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newTreasury","type":"address"}],"name":"setTreasury","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":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"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":"totalRedeemables","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalRevenue","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"treasury","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"unsetOperatorRegistry","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
60806040523480156200001157600080fd5b506200001c62000022565b620000e4565b600054610100900460ff16156200008f5760405162461bcd60e51b815260206004820152602760248201527f496e697469616c697a61626c653a20636f6e747261637420697320696e697469604482015266616c697a696e6760c81b606482015260840160405180910390fd5b60005460ff9081161015620000e2576000805460ff191660ff9081179091556040519081527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a15b565b6148cd80620000f46000396000f3fe6080604052600436106103975760003560e01c80635944c753116101dc578063b88d4fde11610102578063d3fc9864116100a0578063f04e283e1161006f578063f04e283e14610ad5578063f0f4426014610af5578063f2fde38b14610b15578063fee81cf414610b3557600080fd5b8063d3fc986414610a62578063d7533f0214610a82578063e985e9c514610aa0578063ec87621c14610ac057600080fd5b8063bf2d9e0b116100dc578063bf2d9e0b146109f8578063bfa9aadc14610a0d578063c13e33f314610a2d578063c87b56dd14610a4257600080fd5b8063b88d4fde14610995578063b9358c81146109b5578063b9b60747146109e557600080fd5b80637359e41f1161017a5780639b93bdd1116101495780639b93bdd1146109155780639d28fb8614610935578063a22cb46514610955578063afe735471461097557600080fd5b80637359e41f146108a557806375b238fc146108d25780638da5cb5b146108e757806395d89b411461090057600080fd5b8063691ebfb7116101b6578063691ebfb71461081357806370a0823114610840578063715018a61461086057806371dbf0a71461087557600080fd5b80635944c753146107b557806361d027b3146107d55780636352211e146107f357600080fd5b806323b872dd116102c157806342842e0e1161025f578063514e62fc1161022e578063514e62fc14610736578063521242c21461076d57806354d1f13d1461078257806358c2225b1461079757600080fd5b806342842e0e146106b657806342966c68146106d65780634a4ee7b1146106f65780634f6ccce71461071657600080fd5b80632a55205a1161029b5780632a55205a146106115780632de94807146106505780632f745c59146106815780633ccfd60b146106a157600080fd5b806323b872dd146105c757806325692962146105e7578063282c51f3146105fc57600080fd5b8063162094c4116103395780631c10893f116103085780631c10893f146105305780631cd64df4146105505780631ea1afdb1461058757806320c5429b146105a757600080fd5b8063162094c4146104bb57806318160ddd146104db578063183a4f6e146104f057806318712c211461051057600080fd5b8063081812fc11610375578063081812fc14610415578063095ea7b31461044d5780630daf00bc1461046d57806313a661ed1461048d57600080fd5b806301ffc9a71461039c57806304634d8d146103d157806306fdde03146103f3575b600080fd5b3480156103a857600080fd5b506103bc6103b7366004613c3d565b610b66565b60405190151581526020015b60405180910390f35b3480156103dd57600080fd5b506103f16103ec366004613c8d565b610ba4565b005b3480156103ff57600080fd5b50610408610be9565b6040516103c89190613d18565b34801561042157600080fd5b50610435610430366004613d2b565b610c7b565b6040516001600160a01b0390911681526020016103c8565b34801561045957600080fd5b506103f1610468366004613d44565b610ca2565b34801561047957600080fd5b506103f1610488366004613e2b565b610d6b565b34801561049957600080fd5b506104ad6104a8366004613e87565b610db6565b6040519081526020016103c8565b3480156104c757600080fd5b506103f16104d6366004613f3f565b610de9565b3480156104e757600080fd5b5060d4546104ad565b3480156104fc57600080fd5b506103f161050b366004613d2b565b610e29565b34801561051c57600080fd5b506103f161052b366004613f85565b610e36565b34801561053c57600080fd5b506103f161054b366004613d44565b610e76565b34801561055c57600080fd5b506103bc61056b366004613d44565b60609190911b638b78c6d8176000908152602090205481161490565b34801561059357600080fd5b506103f16105a2366004613d2b565b610e9f565b3480156105b357600080fd5b506103f16105c2366004613d2b565b610ede565b3480156105d357600080fd5b506103f16105e2366004613fa7565b610f1d565b3480156105f357600080fd5b506103f1610f4f565b34801561060857600080fd5b506104ad600481565b34801561061d57600080fd5b5061063161062c366004613f85565b610f9f565b604080516001600160a01b0390931683526020830191909152016103c8565b34801561065c57600080fd5b506104ad61066b366004613fe3565b60601b638b78c6d8176000908152602090205490565b34801561068d57600080fd5b506104ad61069c366004613d44565b61104d565b3480156106ad57600080fd5b506103f16110e3565b3480156106c257600080fd5b506103f16106d1366004613fa7565b611254565b3480156106e257600080fd5b506103f16106f1366004613d2b565b61126f565b34801561070257600080fd5b506103f1610711366004613d44565b61129d565b34801561072257600080fd5b506104ad610731366004613d2b565b6112c2565b34801561074257600080fd5b506103bc610751366004613d44565b60609190911b638b78c6d8176000908152602090205416151590565b34801561077957600080fd5b506103f1611355565b34801561078e57600080fd5b506103f161139e565b3480156107a357600080fd5b50609b546001600160a01b0316610435565b3480156107c157600080fd5b506103f16107d0366004613ffe565b6113db565b3480156107e157600080fd5b50609c546001600160a01b0316610435565b3480156107ff57600080fd5b5061043561080e366004613d2b565b61141c565b34801561081f57600080fd5b5061083361082e366004613d2b565b61147c565b6040516103c8919061403a565b34801561084c57600080fd5b506104ad61085b366004613fe3565b6115d8565b34801561086c57600080fd5b506103f161165e565b34801561088157600080fd5b506103bc610890366004613d2b565b6000908152609e602052604090205460ff1690565b3480156108b157600080fd5b506108c56108c0366004613d2b565b61169a565b6040516103c891906140bd565b3480156108de57600080fd5b506104ad600181565b3480156108f357600080fd5b50638b78c6d81954610435565b34801561090c57600080fd5b506104086116e2565b34801561092157600080fd5b506103f1610930366004614114565b6116f1565b34801561094157600080fd5b506103f1610950366004613fe3565b611780565b34801561096157600080fd5b506103f1610970366004614137565b61182a565b34801561098157600080fd5b506103f1610990366004614114565b6118ea565b3480156109a157600080fd5b506103f16109b0366004614161565b611970565b3480156109c157600080fd5b506103bc6109d0366004613d2b565b6000908152609f602052604090205460ff1690565b6103f16109f3366004614220565b6119a2565b348015610a0457600080fd5b50609d546104ad565b348015610a1957600080fd5b506103f1610a283660046142d0565b611a54565b348015610a3957600080fd5b506104ad611b9f565b348015610a4e57600080fd5b50610408610a5d366004613d2b565b611baf565b348015610a6e57600080fd5b506103f1610a7d366004614374565b611bba565b348015610a8e57600080fd5b506040516202a30081526020016103c8565b348015610aac57600080fd5b506103bc610abb3660046143ca565b611bfb565b348015610acc57600080fd5b506104ad600281565b348015610ae157600080fd5b506103f1610af0366004613fe3565b611cf6565b348015610b0157600080fd5b506103f1610b10366004613fe3565b611d66565b348015610b2157600080fd5b506103f1610b30366004613fe3565b611dbf565b348015610b4157600080fd5b506104ad610b50366004613fe3565b60601b63389a75e1176000908152602090205490565b6000610b7182611e14565b80610b805750610b8082611e54565b80610b8f5750610b8f82611e79565b80610b9e5750610b9e82611e54565b92915050565b6001638b78c6d83360601b176000528060206000205416610bda57638b78c6d819543314610bda576382b429006000526004601cfd5b610be48383611e9e565b505050565b606060a08054610bf8906143f4565b80601f0160208091040260200160405190810160405280929190818152602001828054610c24906143f4565b8015610c715780601f10610c4657610100808354040283529160200191610c71565b820191906000526020600020905b815481529060010190602001808311610c5457829003601f168201915b5050505050905090565b6000610c8682611f58565b50600090815260a460205260409020546001600160a01b031690565b609b5482906001600160a01b031615610d6157609b54604051632aeb048b60e01b81526001600160a01b0383811660048301526000921690632aeb048b90602401602060405180830381865afa158015610d00573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d24919061442e565b6000818152609f602052604090205490915060ff1615610d5f5760405162461bcd60e51b8152600401610d5690614447565b60405180910390fd5b505b610be48383611fb7565b6002638b78c6d83360601b176000528060206000205416610da157638b78c6d819543314610da1576382b429006000526004601cfd5b610dae86868686866120c7565b505050505050565b600060208201825160051b81015b808214610de257600160ff8351161b83179250602082019150610dc4565b5050919050565b6002638b78c6d83360601b176000528060206000205416610e1f57638b78c6d819543314610e1f576382b429006000526004601cfd5b610be483836121d9565b610e33338261226d565b50565b6002638b78c6d83360601b176000528060206000205416610e6c57638b78c6d819543314610e6c576382b429006000526004601cfd5b610be483836122be565b638b78c6d819543314610e91576382b429006000526004601cfd5b610e9b8282612304565b5050565b6002638b78c6d83360601b176000528060206000205416610ed557638b78c6d819543314610ed5576382b429006000526004601cfd5b610e9b82612350565b6002638b78c6d83360601b176000528060206000205416610f1457638b78c6d819543314610f14576382b429006000526004601cfd5b610e9b826123b3565b610f28335b826123ff565b610f445760405162461bcd60e51b8152600401610d5690614493565b610be483838361245e565b60006202a3006001600160401b03164201905063389a75e13360601b1760005280602060002055337fdbf36a107da19e49527a7176a1babf963b4b0ff8cde35ee35d6cd8f1f9ac7e1d600080a250565b60008281526069602090815260408083208151808301909252546001600160a01b038116808352600160a01b9091046001600160601b03169282019290925282916110145750604080518082019091526068546001600160a01b0381168252600160a01b90046001600160601b031660208201525b602081015160009061271090611033906001600160601b0316876144f7565b61103d919061452c565b91519350909150505b9250929050565b6000611058836115d8565b82106110ba5760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b6064820152608401610d56565b506001600160a01b0391909116600090815260d260209081526040808320938352929052205490565b6001638b78c6d83360601b17600052806020600020541661111957638b78c6d819543314611119576382b429006000526004601cfd5b600047116111555760405162461bcd60e51b8152602060048201526009602482015268302062616c616e636560b81b6044820152606401610d56565b609a54604051639af608c960e01b81523060048201526001600160a01b039091169047906000908390639af608c990602401602060405180830381865afa1580156111a4573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111c8919061442e565b60405163b9bff4bb60e01b8152600481018290529091506001600160a01b0384169063b9bff4bb90602401600060405180830381600087803b15801561120d57600080fd5b505af1158015611221573d6000803e3d6000fd5b5050505061122f8382612605565b609c5461124e906001600160a01b0316611249848461271e565b612605565b50505050565b610be483838360405180602001604052806000815250611970565b61127833610f22565b6112945760405162461bcd60e51b8152600401610d5690614493565b610e338161272a565b638b78c6d8195433146112b8576382b429006000526004601cfd5b610e9b828261226d565b60006112cd60d45490565b82106113305760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b6064820152608401610d56565b60d4828154811061134357611343614540565b90600052602060002001549050919050565b6001638b78c6d83360601b17600052806020600020541661138b57638b78c6d81954331461138b576382b429006000526004601cfd5b50609b80546001600160a01b0319169055565b63389a75e13360601b176000526000602060002055337ffa7b8eab7da67f412cc9575ed43464468f9bfbae89d1675917346ca6d8fe3c92600080a2565b6001638b78c6d83360601b17600052806020600020541661141157638b78c6d819543314611411576382b429006000526004601cfd5b61124e848484612733565b600081815260a260205260408120546001600160a01b031680610b9e5760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b6044820152606401610d56565b6114d060405180610120016040528060608152602001600081526020016000815260200160008152602001600081526020016000815260200160008019168152602001600015158152602001600081525090565b60008281526034602052604090819020815161012081019092528054829082906114f9906143f4565b80601f0160208091040260200160405190810160405280929190818152602001828054611525906143f4565b80156115725780601f1061154757610100808354040283529160200191611572565b820191906000526020600020905b81548152906001019060200180831161155557829003601f168201915b505050918352505060018201546020820152600282015460408201526003820154606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460ff16151560e08201526008909101546101009091015292915050565b60006001600160a01b0382166116425760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b6064820152608401610d56565b506001600160a01b0316600090815260a3602052604090205490565b638b78c6d819543314611679576382b429006000526004601cfd5b600033600080516020614878833981519152600080a36000638b78c6d81955565b606060206040510160005b8082526001841660051b820191508360011c935083156116c7576001016116a5565b5060405191508060405260208201810360051c825250919050565b606060a18054610bf8906143f4565b6002638b78c6d83360601b17600052806020600020541661172757638b78c6d819543314611727576382b429006000526004601cfd5b6000838152609f6020908152604091829020805460ff1916851515908117909155915191825284917ffd502da418dbb99d2724cacf064b43af780fe7b6607b623466711e8a99dedc3991015b60405180910390a2505050565b6001638b78c6d83360601b1760005280602060002054166117b657638b78c6d8195433146117b6576382b429006000526004601cfd5b6117c782632aeb048b60e01b6127fe565b6118075760405162461bcd60e51b8152602060048201526011602482015270496e76616c696420496e7465726661636560781b6044820152606401610d56565b50609b80546001600160a01b0319166001600160a01b0392909216919091179055565b609b5482906001600160a01b0316156118e057609b54604051632aeb048b60e01b81526001600160a01b0383811660048301526000921690632aeb048b90602401602060405180830381865afa158015611888573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118ac919061442e565b6000818152609f602052604090205490915060ff16156118de5760405162461bcd60e51b8152600401610d5690614447565b505b610be4838361281a565b6002638b78c6d83360601b17600052806020600020541661192057638b78c6d819543314611920576382b429006000526004601cfd5b6000838152609e6020908152604091829020805460ff1916851515908117909155915191825284917f50e00cc379ab7d213a199e50fb1aff65f74b67b5131bc6ebb6516f47b2542e659101611773565b61197a33836123ff565b6119965760405162461bcd60e51b8152600401610d5690614493565b61124e84848484612825565b60006119ad8761147c565b609d549091506119bd9034612858565b609d55609a5460405163107e9cf160e01b81523460048201526001600160a01b039091169063107e9cf190602401600060405180830381600087803b158015611a0557600080fd5b505af1158015611a19573d6000803e3d6000fd5b50505050611a30611a273390565b82518890612864565b611a4b87878787611a44638b78c6d8195490565b88886128a9565b50505050505050565b600054610100900460ff1615808015611a745750600054600160ff909116105b80611a8e5750303b158015611a8e575060005460ff166001145b611af15760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610d56565b6000805460ff191660011790558015611b14576000805461ff0019166101001790555b611b1e8686612c8a565b611b26612cbb565b611b2e612cbb565b611b36612cbb565b611b4287858585612ce4565b611b5161016880546001019055565b8015611a4b576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a150505050505050565b6000611baa60335490565b905090565b6060610b9e82612d4f565b6002638b78c6d83360601b176000528060206000205416611bf057638b78c6d819543314611bf0576382b429006000526004601cfd5b61124e848484612864565b609b546000906001600160a01b031615611cc557609b54604051632aeb048b60e01b81526001600160a01b0384811660048301526000921690632aeb048b90602401602060405180830381865afa158015611c5a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611c7e919061442e565b6000818152609e602052604090205490915060ff1615611ca2576001915050610b9e565b6000818152609f602052604090205460ff1615611cc3576000915050610b9e565b505b6001600160a01b03808416600090815260a5602090815260408083209386168352929052205460ff165b9392505050565b638b78c6d819543314611d11576382b429006000526004601cfd5b8060601b60601c905063389a75e18160601b1760005260206000208054421115611d4357636f5e88186000526004601cfd5b60008155508033600080516020614878833981519152600080a3638b78c6d81955565b6001638b78c6d83360601b176000528060206000205416611d9c57638b78c6d819543314611d9c576382b429006000526004601cfd5b50609c80546001600160a01b0319166001600160a01b0392909216919091179055565b638b78c6d819543314611dda576382b429006000526004601cfd5b6001600160a01b031680611df657637448fbae6000526004601cfd5b8033600080516020614878833981519152600080a3638b78c6d81955565b60006001600160e01b031982166380ac58cd60e01b1480611e4557506001600160e01b03198216635b5e139f60e01b145b80610b9e5750610b9e82611e79565b60006001600160e01b0319821663780e9d6360e01b1480610b9e5750610b9e82611e14565b60006001600160e01b03198216632fea6ab760e21b1480610b9e5750610b9e82612e58565b6127106001600160601b0382161115611ec95760405162461bcd60e51b8152600401610d5690614556565b6001600160a01b038216611f1f5760405162461bcd60e51b815260206004820152601960248201527f455243323938313a20696e76616c6964207265636569766572000000000000006044820152606401610d56565b604080518082019091526001600160a01b039092168083526001600160601b039091166020909201829052600160a01b90910217606855565b600081815260a260205260409020546001600160a01b0316610e335760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b6044820152606401610d56565b6000611fc28261141c565b9050806001600160a01b0316836001600160a01b03160361202f5760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b6064820152608401610d56565b336001600160a01b038216148061204b575061204b8133611bfb565b6120bd5760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c00006064820152608401610d56565b610be48383612e8d565b60006120d260335490565b90506120e2603380546001019055565b604080516101208101825287815260208082018890528183018790526060820186905260808201859052600060a0830181905260c08301819052600160e084015261010083018190528481526034909152919091208151819061214590826145e6565b50602082015160018201556040808301516002830155606083015160038301556080830151600483015560a0830151600583015560c0830151600683015560e083015160078301805460ff1916911515919091179055610100909201516008909101555181907f722868ecddb510067469fb495ac2904c250692c67941ad0bd36e01c2059fc77890600090a2505050505050565b600082815260a260205260409020546001600160a01b03166122545760405162461bcd60e51b815260206004820152602e60248201527f45524337323155524953746f726167653a2055524920736574206f66206e6f6e60448201526d32bc34b9ba32b73a103a37b5b2b760911b6064820152608401610d56565b600082815261010460205260409020610be482826145e6565b638b78c6d88260601b176000526020600020805482811681189050808255808460601b60601c7f715ad5ce61fc9595c7b415289d59cf203f23a94fa06f04af7e489a0a76e1fe26600080a350505050565b60008281526034602052604090206007015460ff166122ef5760405162461bcd60e51b8152600401610d56906146a5565b60009182526034602052604090912060060155565b638b78c6d88260601b17600052602060002081815417808255808460601b60601c7f715ad5ce61fc9595c7b415289d59cf203f23a94fa06f04af7e489a0a76e1fe26600080a350505050565b60008181526034602052604090206007015460ff166123815760405162461bcd60e51b8152600401610d56906146a5565b60008181526034602052604090206008015461239e906001612858565b60009182526034602052604090912060080155565b60008181526034602052604090206007015460ff166123e45760405162461bcd60e51b8152600401610d56906146a5565b6000908152603460205260409020600701805460ff19169055565b60008061240b8361141c565b9050806001600160a01b0316846001600160a01b0316148061243257506124328185611bfb565b806124565750836001600160a01b031661244b84610c7b565b6001600160a01b0316145b949350505050565b826001600160a01b03166124718261141c565b6001600160a01b0316146124d55760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b6064820152608401610d56565b6001600160a01b0382166125375760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610d56565b612542838383612efb565b61254d600082612e8d565b6001600160a01b038316600090815260a3602052604081208054600192906125769084906146c9565b90915550506001600160a01b038216600090815260a3602052604081208054600192906125a49084906146e0565b9091555050600081815260a2602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b804710156126555760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20696e73756666696369656e742062616c616e63650000006044820152606401610d56565b6000826001600160a01b03168260405160006040518083038185875af1925050503d80600081146126a2576040519150601f19603f3d011682016040523d82523d6000602084013e6126a7565b606091505b5050905080610be45760405162461bcd60e51b815260206004820152603a60248201527f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260448201527f6563697069656e74206d617920686176652072657665727465640000000000006064820152608401610d56565b6000611cef82846146c9565b610e3381612f06565b6127106001600160601b038216111561275e5760405162461bcd60e51b8152600401610d5690614556565b6001600160a01b0382166127b45760405162461bcd60e51b815260206004820152601b60248201527f455243323938313a20496e76616c696420706172616d657465727300000000006044820152606401610d56565b6040805180820182526001600160a01b0393841681526001600160601b0392831660208083019182526000968752606990529190942093519051909116600160a01b029116179055565b600061280983612f48565b8015611cef5750611cef8383612f7b565b610e9b338383613004565b61283084848461245e565b61283c848484846130d2565b61124e5760405162461bcd60e51b8152600401610d56906146f8565b6000611cef82846146e0565b60005b8281101561124e57600061287b6101685490565b905061288c61016880546001019055565b61289685826131d3565b6128a081846121d9565b50600101612867565b60006128b48861147c565b90508060e001516128d75760405162461bcd60e51b8152600401610d56906146a5565b602081015134906128e890896131ed565b11156129285760405162461bcd60e51b815260206004820152600f60248201526e15985b1d59481a5b98dbdc9c9958dd608a1b6044820152606401610d56565b80608001518711156129745760405162461bcd60e51b8152602060048201526015602482015274115e18d959591959081b585e081c195c881b5a5b9d605a1b6044820152606401610d56565b604081015160a08201516129889089612858565b11156129cc5760405162461bcd60e51b8152602060048201526013602482015272115e18d959591959081b585e08185b5bdd5b9d606a1b6044820152606401610d56565b606081015160008981526035602090815260408083203384529091529020546129f59089612858565b1115612a435760405162461bcd60e51b815260206004820152601760248201527f4578636565646564206d6178207065722077616c6c65740000000000000000006044820152606401610d56565b836001600160a01b0316612b1387878080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152505050610100850151612b0d9150612a9b908d90612858565b604051602001612aad91815260200190565b60408051601f1981840301815282825280516020918201207f19457468657265756d205369676e6564204d6573736167653a0a33320000000084830152603c8085019190915282518085039091018152605c909301909152815191012090565b906131f9565b6001600160a01b031614612b5d5760405162461bcd60e51b8152602060048201526011602482015270496e76616c6964207369676e617475726560781b6044820152606401610d56565b60c081015115612c1757612bdb83838080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050505060c08301516040516bffffffffffffffffffffffff193360601b1660208201526034016040516020818303038152906040528051906020012061321d565b612c175760405162461bcd60e51b815260206004820152600d60248201526c24b73b30b634b210383937b7b360991b6044820152606401610d56565b60008881526034602090815260408083206005018054600190810190915560358352818420338086529084529382902080549091019055518981528a92917fac100af3f918c30d71d2bfb00d975e36374d10738748c899376504e76c70bc73910160405180910390a35050505050505050565b600054610100900460ff16612cb15760405162461bcd60e51b8152600401610d569061474a565b610e9b8282613233565b600054610100900460ff16612ce25760405162461bcd60e51b8152600401610d569061474a565b565b600054610100900460ff16612d0b5760405162461bcd60e51b8152600401610d569061474a565b612d1484613273565b612d1c612cbb565b609a8054336001600160a01b031991821617909155609c80549091166001600160a01b03851617905561124e8282611e9e565b6060612d5a82611f58565b6000828152610104602052604081208054612d74906143f4565b80601f0160208091040260200160405190810160405280929190818152602001828054612da0906143f4565b8015612ded5780601f10612dc257610100808354040283529160200191612ded565b820191906000526020600020905b815481529060010190602001808311612dd057829003601f168201915b505050505090506000612e0b60408051602081019091526000815290565b90508051600003612e1d575092915050565b815115612e4f578082604051602001612e37929190614795565b60405160208183030381529060405292505050919050565b6124568461329d565b60006001600160e01b0319821663152a902d60e11b1480610b9e57506301ffc9a760e01b6001600160e01b0319831614610b9e565b600081815260a46020526040902080546001600160a01b0319166001600160a01b0384169081179091558190612ec28261141c565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b610be4838383613310565b612f0f816133c8565b6000818152610104602052604090208054612f29906143f4565b159050610e3357600081815261010460205260408120610e3391613bd9565b6000612f5b826301ffc9a760e01b612f7b565b8015610b9e5750612f74826001600160e01b0319612f7b565b1592915050565b604080516001600160e01b03198316602480830191909152825180830390910181526044909101909152602080820180516001600160e01b03166301ffc9a760e01b178152825160009392849283928392918391908a617530fa92503d91506000519050828015612fed575060208210155b8015612ff95750600081115b979650505050505050565b816001600160a01b0316836001600160a01b0316036130655760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c6572000000000000006044820152606401610d56565b6001600160a01b03838116600081815260a56020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b60006001600160a01b0384163b156131c857604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906131169033908990889088906004016147c4565b6020604051808303816000875af1925050508015613151575060408051601f3d908101601f1916820190925261314e91810190614801565b60015b6131ae573d80801561317f576040519150601f19603f3d011682016040523d82523d6000602084013e613184565b606091505b5080516000036131a65760405162461bcd60e51b8152600401610d56906146f8565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612456565b506001949350505050565b610e9b82826040518060200160405280600081525061346f565b6000611cef82846144f7565b600080600061320885856134a2565b91509150613215816134e4565b509392505050565b60008261322a858461369a565b14949350505050565b600054610100900460ff1661325a5760405162461bcd60e51b8152600401610d569061474a565b60a061326683826145e6565b5060a1610be482826145e6565b6001600160a01b0316638b78c6d8198190558060006000805160206148788339815191528180a350565b60606132a882611f58565b60006132bf60408051602081019091526000815290565b905060008151116132df5760405180602001604052806000815250611cef565b806132e9846136df565b6040516020016132fa929190614795565b6040516020818303038152906040529392505050565b6001600160a01b03831661336b576133668160d48054600083815260d560205260408120829055600182018355919091527f9780e26d96b1f2a9a18ef8fc72d589dbf03ef788137b64f43897e83a91e7feec0155565b61338e565b816001600160a01b0316836001600160a01b03161461338e5761338e83826137df565b6001600160a01b0382166133a557610be48161387c565b826001600160a01b0316826001600160a01b031614610be457610be4828261392b565b60006133d38261141c565b90506133e181600084612efb565b6133ec600083612e8d565b6001600160a01b038116600090815260a3602052604081208054600192906134159084906146c9565b9091555050600082815260a2602052604080822080546001600160a01b0319169055518391906001600160a01b038416907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908390a45050565b613479838361396f565b61348660008484846130d2565b610be45760405162461bcd60e51b8152600401610d56906146f8565b60008082516041036134d85760208301516040840151606085015160001a6134cc87828585613abd565b94509450505050611046565b50600090506002611046565b60008160048111156134f8576134f861481e565b036135005750565b60018160048111156135145761351461481e565b036135615760405162461bcd60e51b815260206004820152601860248201527f45434453413a20696e76616c6964207369676e617475726500000000000000006044820152606401610d56565b60028160048111156135755761357561481e565b036135c25760405162461bcd60e51b815260206004820152601f60248201527f45434453413a20696e76616c6964207369676e6174757265206c656e677468006044820152606401610d56565b60038160048111156135d6576135d661481e565b0361362e5760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202773272076616c604482015261756560f01b6064820152608401610d56565b60048160048111156136425761364261481e565b03610e335760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202776272076616c604482015261756560f01b6064820152608401610d56565b600081815b8451811015613215576136cb828683815181106136be576136be614540565b6020026020010151613baa565b9150806136d781614834565b91505061369f565b6060816000036137065750506040805180820190915260018152600360fc1b602082015290565b8160005b8115613730578061371a81614834565b91506137299050600a8361452c565b915061370a565b6000816001600160401b0381111561374a5761374a613d6e565b6040519080825280601f01601f191660200182016040528015613774576020820181803683370190505b5090505b8415612456576137896001836146c9565b9150613796600a8661484d565b6137a19060306146e0565b60f81b8183815181106137b6576137b6614540565b60200101906001600160f81b031916908160001a9053506137d8600a8661452c565b9450613778565b600060016137ec846115d8565b6137f691906146c9565b600083815260d36020526040902054909150808214613849576001600160a01b038416600090815260d260209081526040808320858452825280832054848452818420819055835260d390915290208190555b50600091825260d3602090815260408084208490556001600160a01b03909416835260d281528383209183525290812055565b60d45460009061388e906001906146c9565b600083815260d5602052604081205460d480549394509092849081106138b6576138b6614540565b906000526020600020015490508060d483815481106138d7576138d7614540565b600091825260208083209091019290925582815260d5909152604080822084905585825281205560d480548061390f5761390f614861565b6001900381819060005260206000200160009055905550505050565b6000613936836115d8565b6001600160a01b03909316600090815260d260209081526040808320868452825280832085905593825260d39052919091209190915550565b6001600160a01b0382166139c55760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610d56565b600081815260a260205260409020546001600160a01b031615613a2a5760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610d56565b613a3660008383612efb565b6001600160a01b038216600090815260a360205260408120805460019290613a5f9084906146e0565b9091555050600081815260a2602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a0831115613af45750600090506003613ba1565b8460ff16601b14158015613b0c57508460ff16601c14155b15613b1d5750600090506004613ba1565b6040805160008082526020820180845289905260ff881692820192909252606081018690526080810185905260019060a0016020604051602081039080840390855afa158015613b71573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b038116613b9a57600060019250925050613ba1565b9150600090505b94509492505050565b6000818310613bc6576000828152602084905260409020611cef565b6000838152602083905260409020611cef565b508054613be5906143f4565b6000825580601f10613bf5575050565b601f016020900490600052602060002090810190610e3391905b80821115613c235760008155600101613c0f565b5090565b6001600160e01b031981168114610e3357600080fd5b600060208284031215613c4f57600080fd5b8135611cef81613c27565b80356001600160a01b0381168114613c7157600080fd5b919050565b80356001600160601b0381168114613c7157600080fd5b60008060408385031215613ca057600080fd5b613ca983613c5a565b9150613cb760208401613c76565b90509250929050565b60005b83811015613cdb578181015183820152602001613cc3565b8381111561124e5750506000910152565b60008151808452613d04816020860160208601613cc0565b601f01601f19169290920160200192915050565b602081526000611cef6020830184613cec565b600060208284031215613d3d57600080fd5b5035919050565b60008060408385031215613d5757600080fd5b613d6083613c5a565b946020939093013593505050565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f191681016001600160401b0381118282101715613dac57613dac613d6e565b604052919050565b60006001600160401b03831115613dcd57613dcd613d6e565b613de0601f8401601f1916602001613d84565b9050828152838383011115613df457600080fd5b828260208301376000602084830101529392505050565b600082601f830112613e1c57600080fd5b611cef83833560208501613db4565b600080600080600060a08688031215613e4357600080fd5b85356001600160401b03811115613e5957600080fd5b613e6588828901613e0b565b9860208801359850604088013597606081013597506080013595509350505050565b60006020808385031215613e9a57600080fd5b82356001600160401b0380821115613eb157600080fd5b818501915085601f830112613ec557600080fd5b813581811115613ed757613ed7613d6e565b8060051b9150613ee8848301613d84565b8181529183018401918481019088841115613f0257600080fd5b938501935b83851015613f33578435925060ff83168314613f235760008081fd5b8282529385019390850190613f07565b98975050505050505050565b60008060408385031215613f5257600080fd5b8235915060208301356001600160401b03811115613f6f57600080fd5b613f7b85828601613e0b565b9150509250929050565b60008060408385031215613f9857600080fd5b50508035926020909101359150565b600080600060608486031215613fbc57600080fd5b613fc584613c5a565b9250613fd360208501613c5a565b9150604084013590509250925092565b600060208284031215613ff557600080fd5b611cef82613c5a565b60008060006060848603121561401357600080fd5b8335925061402360208501613c5a565b915061403160408501613c76565b90509250925092565b6020815260008251610120806020850152614059610140850183613cec565b9150602085015160408501526040850151606085015260608501516080850152608085015160a085015260a085015160c085015260c085015160e085015260e08501516101006140ac8187018315159052565b959095015193019290925250919050565b6020808252825182820181905260009190848201906040850190845b818110156140f857835160ff16835292840192918401916001016140d9565b50909695505050505050565b80358015158114613c7157600080fd5b6000806040838503121561412757600080fd5b82359150613cb760208401614104565b6000806040838503121561414a57600080fd5b61415383613c5a565b9150613cb760208401614104565b6000806000806080858703121561417757600080fd5b61418085613c5a565b935061418e60208601613c5a565b92506040850135915060608501356001600160401b038111156141b057600080fd5b8501601f810187136141c157600080fd5b6141d087823560208401613db4565b91505092959194509250565b60008083601f8401126141ee57600080fd5b5081356001600160401b0381111561420557600080fd5b6020830191508360208260051b850101111561104657600080fd5b6000806000806000806080878903121561423957600080fd5b863595506020870135945060408701356001600160401b038082111561425e57600080fd5b818901915089601f83011261427257600080fd5b81358181111561428157600080fd5b8a602082850101111561429357600080fd5b6020830196508095505060608901359150808211156142b157600080fd5b506142be89828a016141dc565b979a9699509497509295939492505050565b60008060008060008060c087890312156142e957600080fd5b6142f287613c5a565b955060208701356001600160401b038082111561430e57600080fd5b61431a8a838b01613e0b565b9650604089013591508082111561433057600080fd5b5061433d89828a01613e0b565b94505061434c60608801613c5a565b925061435a60808801613c5a565b915061436860a08801613c76565b90509295509295509295565b60008060006060848603121561438957600080fd5b61439284613c5a565b92506020840135915060408401356001600160401b038111156143b457600080fd5b6143c086828701613e0b565b9150509250925092565b600080604083850312156143dd57600080fd5b6143e683613c5a565b9150613cb760208401613c5a565b600181811c9082168061440857607f821691505b60208210810361442857634e487b7160e01b600052602260045260246000fd5b50919050565b60006020828403121561444057600080fd5b5051919050565b6020808252602c908201527f4f70657261746f7220686173206265656e20626c6f636b656420627920636f6e60408201526b3a3930b1ba1037bbb732b91760a11b606082015260800190565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b634e487b7160e01b600052601160045260246000fd5b6000816000190483118215151615614511576145116144e1565b500290565b634e487b7160e01b600052601260045260246000fd5b60008261453b5761453b614516565b500490565b634e487b7160e01b600052603260045260246000fd5b6020808252602a908201527f455243323938313a20726f79616c7479206665652077696c6c206578636565646040820152692073616c65507269636560b01b606082015260800190565b601f821115610be457600081815260208120601f850160051c810160208610156145c75750805b601f850160051c820191505b81811015610dae578281556001016145d3565b81516001600160401b038111156145ff576145ff613d6e565b6146138161460d84546143f4565b846145a0565b602080601f83116001811461464857600084156146305750858301515b600019600386901b1c1916600185901b178555610dae565b600085815260208120601f198616915b8281101561467757888601518255948401946001909101908401614658565b50858210156146955787850151600019600388901b60f8161c191681555b5050505050600190811b01905550565b6020808252600a90820152694e6f742061637469766560b01b604082015260600190565b6000828210156146db576146db6144e1565b500390565b600082198211156146f3576146f36144e1565b500190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b600083516147a7818460208801613cc0565b8351908301906147bb818360208801613cc0565b01949350505050565b6001600160a01b03858116825284166020820152604081018390526080606082018190526000906147f790830184613cec565b9695505050505050565b60006020828403121561481357600080fd5b8151611cef81613c27565b634e487b7160e01b600052602160045260246000fd5b600060018201614846576148466144e1565b5060010190565b60008261485c5761485c614516565b500690565b634e487b7160e01b600052603160045260246000fdfe8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0a264697066735822122099dde4802314b9e1f4c38da49ac82adb44e44ce5590540fe919aad3692f96e6964736f6c634300080f0033
Deployed Bytecode
0x6080604052600436106103975760003560e01c80635944c753116101dc578063b88d4fde11610102578063d3fc9864116100a0578063f04e283e1161006f578063f04e283e14610ad5578063f0f4426014610af5578063f2fde38b14610b15578063fee81cf414610b3557600080fd5b8063d3fc986414610a62578063d7533f0214610a82578063e985e9c514610aa0578063ec87621c14610ac057600080fd5b8063bf2d9e0b116100dc578063bf2d9e0b146109f8578063bfa9aadc14610a0d578063c13e33f314610a2d578063c87b56dd14610a4257600080fd5b8063b88d4fde14610995578063b9358c81146109b5578063b9b60747146109e557600080fd5b80637359e41f1161017a5780639b93bdd1116101495780639b93bdd1146109155780639d28fb8614610935578063a22cb46514610955578063afe735471461097557600080fd5b80637359e41f146108a557806375b238fc146108d25780638da5cb5b146108e757806395d89b411461090057600080fd5b8063691ebfb7116101b6578063691ebfb71461081357806370a0823114610840578063715018a61461086057806371dbf0a71461087557600080fd5b80635944c753146107b557806361d027b3146107d55780636352211e146107f357600080fd5b806323b872dd116102c157806342842e0e1161025f578063514e62fc1161022e578063514e62fc14610736578063521242c21461076d57806354d1f13d1461078257806358c2225b1461079757600080fd5b806342842e0e146106b657806342966c68146106d65780634a4ee7b1146106f65780634f6ccce71461071657600080fd5b80632a55205a1161029b5780632a55205a146106115780632de94807146106505780632f745c59146106815780633ccfd60b146106a157600080fd5b806323b872dd146105c757806325692962146105e7578063282c51f3146105fc57600080fd5b8063162094c4116103395780631c10893f116103085780631c10893f146105305780631cd64df4146105505780631ea1afdb1461058757806320c5429b146105a757600080fd5b8063162094c4146104bb57806318160ddd146104db578063183a4f6e146104f057806318712c211461051057600080fd5b8063081812fc11610375578063081812fc14610415578063095ea7b31461044d5780630daf00bc1461046d57806313a661ed1461048d57600080fd5b806301ffc9a71461039c57806304634d8d146103d157806306fdde03146103f3575b600080fd5b3480156103a857600080fd5b506103bc6103b7366004613c3d565b610b66565b60405190151581526020015b60405180910390f35b3480156103dd57600080fd5b506103f16103ec366004613c8d565b610ba4565b005b3480156103ff57600080fd5b50610408610be9565b6040516103c89190613d18565b34801561042157600080fd5b50610435610430366004613d2b565b610c7b565b6040516001600160a01b0390911681526020016103c8565b34801561045957600080fd5b506103f1610468366004613d44565b610ca2565b34801561047957600080fd5b506103f1610488366004613e2b565b610d6b565b34801561049957600080fd5b506104ad6104a8366004613e87565b610db6565b6040519081526020016103c8565b3480156104c757600080fd5b506103f16104d6366004613f3f565b610de9565b3480156104e757600080fd5b5060d4546104ad565b3480156104fc57600080fd5b506103f161050b366004613d2b565b610e29565b34801561051c57600080fd5b506103f161052b366004613f85565b610e36565b34801561053c57600080fd5b506103f161054b366004613d44565b610e76565b34801561055c57600080fd5b506103bc61056b366004613d44565b60609190911b638b78c6d8176000908152602090205481161490565b34801561059357600080fd5b506103f16105a2366004613d2b565b610e9f565b3480156105b357600080fd5b506103f16105c2366004613d2b565b610ede565b3480156105d357600080fd5b506103f16105e2366004613fa7565b610f1d565b3480156105f357600080fd5b506103f1610f4f565b34801561060857600080fd5b506104ad600481565b34801561061d57600080fd5b5061063161062c366004613f85565b610f9f565b604080516001600160a01b0390931683526020830191909152016103c8565b34801561065c57600080fd5b506104ad61066b366004613fe3565b60601b638b78c6d8176000908152602090205490565b34801561068d57600080fd5b506104ad61069c366004613d44565b61104d565b3480156106ad57600080fd5b506103f16110e3565b3480156106c257600080fd5b506103f16106d1366004613fa7565b611254565b3480156106e257600080fd5b506103f16106f1366004613d2b565b61126f565b34801561070257600080fd5b506103f1610711366004613d44565b61129d565b34801561072257600080fd5b506104ad610731366004613d2b565b6112c2565b34801561074257600080fd5b506103bc610751366004613d44565b60609190911b638b78c6d8176000908152602090205416151590565b34801561077957600080fd5b506103f1611355565b34801561078e57600080fd5b506103f161139e565b3480156107a357600080fd5b50609b546001600160a01b0316610435565b3480156107c157600080fd5b506103f16107d0366004613ffe565b6113db565b3480156107e157600080fd5b50609c546001600160a01b0316610435565b3480156107ff57600080fd5b5061043561080e366004613d2b565b61141c565b34801561081f57600080fd5b5061083361082e366004613d2b565b61147c565b6040516103c8919061403a565b34801561084c57600080fd5b506104ad61085b366004613fe3565b6115d8565b34801561086c57600080fd5b506103f161165e565b34801561088157600080fd5b506103bc610890366004613d2b565b6000908152609e602052604090205460ff1690565b3480156108b157600080fd5b506108c56108c0366004613d2b565b61169a565b6040516103c891906140bd565b3480156108de57600080fd5b506104ad600181565b3480156108f357600080fd5b50638b78c6d81954610435565b34801561090c57600080fd5b506104086116e2565b34801561092157600080fd5b506103f1610930366004614114565b6116f1565b34801561094157600080fd5b506103f1610950366004613fe3565b611780565b34801561096157600080fd5b506103f1610970366004614137565b61182a565b34801561098157600080fd5b506103f1610990366004614114565b6118ea565b3480156109a157600080fd5b506103f16109b0366004614161565b611970565b3480156109c157600080fd5b506103bc6109d0366004613d2b565b6000908152609f602052604090205460ff1690565b6103f16109f3366004614220565b6119a2565b348015610a0457600080fd5b50609d546104ad565b348015610a1957600080fd5b506103f1610a283660046142d0565b611a54565b348015610a3957600080fd5b506104ad611b9f565b348015610a4e57600080fd5b50610408610a5d366004613d2b565b611baf565b348015610a6e57600080fd5b506103f1610a7d366004614374565b611bba565b348015610a8e57600080fd5b506040516202a30081526020016103c8565b348015610aac57600080fd5b506103bc610abb3660046143ca565b611bfb565b348015610acc57600080fd5b506104ad600281565b348015610ae157600080fd5b506103f1610af0366004613fe3565b611cf6565b348015610b0157600080fd5b506103f1610b10366004613fe3565b611d66565b348015610b2157600080fd5b506103f1610b30366004613fe3565b611dbf565b348015610b4157600080fd5b506104ad610b50366004613fe3565b60601b63389a75e1176000908152602090205490565b6000610b7182611e14565b80610b805750610b8082611e54565b80610b8f5750610b8f82611e79565b80610b9e5750610b9e82611e54565b92915050565b6001638b78c6d83360601b176000528060206000205416610bda57638b78c6d819543314610bda576382b429006000526004601cfd5b610be48383611e9e565b505050565b606060a08054610bf8906143f4565b80601f0160208091040260200160405190810160405280929190818152602001828054610c24906143f4565b8015610c715780601f10610c4657610100808354040283529160200191610c71565b820191906000526020600020905b815481529060010190602001808311610c5457829003601f168201915b5050505050905090565b6000610c8682611f58565b50600090815260a460205260409020546001600160a01b031690565b609b5482906001600160a01b031615610d6157609b54604051632aeb048b60e01b81526001600160a01b0383811660048301526000921690632aeb048b90602401602060405180830381865afa158015610d00573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d24919061442e565b6000818152609f602052604090205490915060ff1615610d5f5760405162461bcd60e51b8152600401610d5690614447565b60405180910390fd5b505b610be48383611fb7565b6002638b78c6d83360601b176000528060206000205416610da157638b78c6d819543314610da1576382b429006000526004601cfd5b610dae86868686866120c7565b505050505050565b600060208201825160051b81015b808214610de257600160ff8351161b83179250602082019150610dc4565b5050919050565b6002638b78c6d83360601b176000528060206000205416610e1f57638b78c6d819543314610e1f576382b429006000526004601cfd5b610be483836121d9565b610e33338261226d565b50565b6002638b78c6d83360601b176000528060206000205416610e6c57638b78c6d819543314610e6c576382b429006000526004601cfd5b610be483836122be565b638b78c6d819543314610e91576382b429006000526004601cfd5b610e9b8282612304565b5050565b6002638b78c6d83360601b176000528060206000205416610ed557638b78c6d819543314610ed5576382b429006000526004601cfd5b610e9b82612350565b6002638b78c6d83360601b176000528060206000205416610f1457638b78c6d819543314610f14576382b429006000526004601cfd5b610e9b826123b3565b610f28335b826123ff565b610f445760405162461bcd60e51b8152600401610d5690614493565b610be483838361245e565b60006202a3006001600160401b03164201905063389a75e13360601b1760005280602060002055337fdbf36a107da19e49527a7176a1babf963b4b0ff8cde35ee35d6cd8f1f9ac7e1d600080a250565b60008281526069602090815260408083208151808301909252546001600160a01b038116808352600160a01b9091046001600160601b03169282019290925282916110145750604080518082019091526068546001600160a01b0381168252600160a01b90046001600160601b031660208201525b602081015160009061271090611033906001600160601b0316876144f7565b61103d919061452c565b91519350909150505b9250929050565b6000611058836115d8565b82106110ba5760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b6064820152608401610d56565b506001600160a01b0391909116600090815260d260209081526040808320938352929052205490565b6001638b78c6d83360601b17600052806020600020541661111957638b78c6d819543314611119576382b429006000526004601cfd5b600047116111555760405162461bcd60e51b8152602060048201526009602482015268302062616c616e636560b81b6044820152606401610d56565b609a54604051639af608c960e01b81523060048201526001600160a01b039091169047906000908390639af608c990602401602060405180830381865afa1580156111a4573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111c8919061442e565b60405163b9bff4bb60e01b8152600481018290529091506001600160a01b0384169063b9bff4bb90602401600060405180830381600087803b15801561120d57600080fd5b505af1158015611221573d6000803e3d6000fd5b5050505061122f8382612605565b609c5461124e906001600160a01b0316611249848461271e565b612605565b50505050565b610be483838360405180602001604052806000815250611970565b61127833610f22565b6112945760405162461bcd60e51b8152600401610d5690614493565b610e338161272a565b638b78c6d8195433146112b8576382b429006000526004601cfd5b610e9b828261226d565b60006112cd60d45490565b82106113305760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b6064820152608401610d56565b60d4828154811061134357611343614540565b90600052602060002001549050919050565b6001638b78c6d83360601b17600052806020600020541661138b57638b78c6d81954331461138b576382b429006000526004601cfd5b50609b80546001600160a01b0319169055565b63389a75e13360601b176000526000602060002055337ffa7b8eab7da67f412cc9575ed43464468f9bfbae89d1675917346ca6d8fe3c92600080a2565b6001638b78c6d83360601b17600052806020600020541661141157638b78c6d819543314611411576382b429006000526004601cfd5b61124e848484612733565b600081815260a260205260408120546001600160a01b031680610b9e5760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b6044820152606401610d56565b6114d060405180610120016040528060608152602001600081526020016000815260200160008152602001600081526020016000815260200160008019168152602001600015158152602001600081525090565b60008281526034602052604090819020815161012081019092528054829082906114f9906143f4565b80601f0160208091040260200160405190810160405280929190818152602001828054611525906143f4565b80156115725780601f1061154757610100808354040283529160200191611572565b820191906000526020600020905b81548152906001019060200180831161155557829003601f168201915b505050918352505060018201546020820152600282015460408201526003820154606082015260048201546080820152600582015460a0820152600682015460c0820152600782015460ff16151560e08201526008909101546101009091015292915050565b60006001600160a01b0382166116425760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b6064820152608401610d56565b506001600160a01b0316600090815260a3602052604090205490565b638b78c6d819543314611679576382b429006000526004601cfd5b600033600080516020614878833981519152600080a36000638b78c6d81955565b606060206040510160005b8082526001841660051b820191508360011c935083156116c7576001016116a5565b5060405191508060405260208201810360051c825250919050565b606060a18054610bf8906143f4565b6002638b78c6d83360601b17600052806020600020541661172757638b78c6d819543314611727576382b429006000526004601cfd5b6000838152609f6020908152604091829020805460ff1916851515908117909155915191825284917ffd502da418dbb99d2724cacf064b43af780fe7b6607b623466711e8a99dedc3991015b60405180910390a2505050565b6001638b78c6d83360601b1760005280602060002054166117b657638b78c6d8195433146117b6576382b429006000526004601cfd5b6117c782632aeb048b60e01b6127fe565b6118075760405162461bcd60e51b8152602060048201526011602482015270496e76616c696420496e7465726661636560781b6044820152606401610d56565b50609b80546001600160a01b0319166001600160a01b0392909216919091179055565b609b5482906001600160a01b0316156118e057609b54604051632aeb048b60e01b81526001600160a01b0383811660048301526000921690632aeb048b90602401602060405180830381865afa158015611888573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118ac919061442e565b6000818152609f602052604090205490915060ff16156118de5760405162461bcd60e51b8152600401610d5690614447565b505b610be4838361281a565b6002638b78c6d83360601b17600052806020600020541661192057638b78c6d819543314611920576382b429006000526004601cfd5b6000838152609e6020908152604091829020805460ff1916851515908117909155915191825284917f50e00cc379ab7d213a199e50fb1aff65f74b67b5131bc6ebb6516f47b2542e659101611773565b61197a33836123ff565b6119965760405162461bcd60e51b8152600401610d5690614493565b61124e84848484612825565b60006119ad8761147c565b609d549091506119bd9034612858565b609d55609a5460405163107e9cf160e01b81523460048201526001600160a01b039091169063107e9cf190602401600060405180830381600087803b158015611a0557600080fd5b505af1158015611a19573d6000803e3d6000fd5b50505050611a30611a273390565b82518890612864565b611a4b87878787611a44638b78c6d8195490565b88886128a9565b50505050505050565b600054610100900460ff1615808015611a745750600054600160ff909116105b80611a8e5750303b158015611a8e575060005460ff166001145b611af15760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610d56565b6000805460ff191660011790558015611b14576000805461ff0019166101001790555b611b1e8686612c8a565b611b26612cbb565b611b2e612cbb565b611b36612cbb565b611b4287858585612ce4565b611b5161016880546001019055565b8015611a4b576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a150505050505050565b6000611baa60335490565b905090565b6060610b9e82612d4f565b6002638b78c6d83360601b176000528060206000205416611bf057638b78c6d819543314611bf0576382b429006000526004601cfd5b61124e848484612864565b609b546000906001600160a01b031615611cc557609b54604051632aeb048b60e01b81526001600160a01b0384811660048301526000921690632aeb048b90602401602060405180830381865afa158015611c5a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611c7e919061442e565b6000818152609e602052604090205490915060ff1615611ca2576001915050610b9e565b6000818152609f602052604090205460ff1615611cc3576000915050610b9e565b505b6001600160a01b03808416600090815260a5602090815260408083209386168352929052205460ff165b9392505050565b638b78c6d819543314611d11576382b429006000526004601cfd5b8060601b60601c905063389a75e18160601b1760005260206000208054421115611d4357636f5e88186000526004601cfd5b60008155508033600080516020614878833981519152600080a3638b78c6d81955565b6001638b78c6d83360601b176000528060206000205416611d9c57638b78c6d819543314611d9c576382b429006000526004601cfd5b50609c80546001600160a01b0319166001600160a01b0392909216919091179055565b638b78c6d819543314611dda576382b429006000526004601cfd5b6001600160a01b031680611df657637448fbae6000526004601cfd5b8033600080516020614878833981519152600080a3638b78c6d81955565b60006001600160e01b031982166380ac58cd60e01b1480611e4557506001600160e01b03198216635b5e139f60e01b145b80610b9e5750610b9e82611e79565b60006001600160e01b0319821663780e9d6360e01b1480610b9e5750610b9e82611e14565b60006001600160e01b03198216632fea6ab760e21b1480610b9e5750610b9e82612e58565b6127106001600160601b0382161115611ec95760405162461bcd60e51b8152600401610d5690614556565b6001600160a01b038216611f1f5760405162461bcd60e51b815260206004820152601960248201527f455243323938313a20696e76616c6964207265636569766572000000000000006044820152606401610d56565b604080518082019091526001600160a01b039092168083526001600160601b039091166020909201829052600160a01b90910217606855565b600081815260a260205260409020546001600160a01b0316610e335760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b6044820152606401610d56565b6000611fc28261141c565b9050806001600160a01b0316836001600160a01b03160361202f5760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b6064820152608401610d56565b336001600160a01b038216148061204b575061204b8133611bfb565b6120bd5760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c00006064820152608401610d56565b610be48383612e8d565b60006120d260335490565b90506120e2603380546001019055565b604080516101208101825287815260208082018890528183018790526060820186905260808201859052600060a0830181905260c08301819052600160e084015261010083018190528481526034909152919091208151819061214590826145e6565b50602082015160018201556040808301516002830155606083015160038301556080830151600483015560a0830151600583015560c0830151600683015560e083015160078301805460ff1916911515919091179055610100909201516008909101555181907f722868ecddb510067469fb495ac2904c250692c67941ad0bd36e01c2059fc77890600090a2505050505050565b600082815260a260205260409020546001600160a01b03166122545760405162461bcd60e51b815260206004820152602e60248201527f45524337323155524953746f726167653a2055524920736574206f66206e6f6e60448201526d32bc34b9ba32b73a103a37b5b2b760911b6064820152608401610d56565b600082815261010460205260409020610be482826145e6565b638b78c6d88260601b176000526020600020805482811681189050808255808460601b60601c7f715ad5ce61fc9595c7b415289d59cf203f23a94fa06f04af7e489a0a76e1fe26600080a350505050565b60008281526034602052604090206007015460ff166122ef5760405162461bcd60e51b8152600401610d56906146a5565b60009182526034602052604090912060060155565b638b78c6d88260601b17600052602060002081815417808255808460601b60601c7f715ad5ce61fc9595c7b415289d59cf203f23a94fa06f04af7e489a0a76e1fe26600080a350505050565b60008181526034602052604090206007015460ff166123815760405162461bcd60e51b8152600401610d56906146a5565b60008181526034602052604090206008015461239e906001612858565b60009182526034602052604090912060080155565b60008181526034602052604090206007015460ff166123e45760405162461bcd60e51b8152600401610d56906146a5565b6000908152603460205260409020600701805460ff19169055565b60008061240b8361141c565b9050806001600160a01b0316846001600160a01b0316148061243257506124328185611bfb565b806124565750836001600160a01b031661244b84610c7b565b6001600160a01b0316145b949350505050565b826001600160a01b03166124718261141c565b6001600160a01b0316146124d55760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b6064820152608401610d56565b6001600160a01b0382166125375760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b6064820152608401610d56565b612542838383612efb565b61254d600082612e8d565b6001600160a01b038316600090815260a3602052604081208054600192906125769084906146c9565b90915550506001600160a01b038216600090815260a3602052604081208054600192906125a49084906146e0565b9091555050600081815260a2602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b804710156126555760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a20696e73756666696369656e742062616c616e63650000006044820152606401610d56565b6000826001600160a01b03168260405160006040518083038185875af1925050503d80600081146126a2576040519150601f19603f3d011682016040523d82523d6000602084013e6126a7565b606091505b5050905080610be45760405162461bcd60e51b815260206004820152603a60248201527f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260448201527f6563697069656e74206d617920686176652072657665727465640000000000006064820152608401610d56565b6000611cef82846146c9565b610e3381612f06565b6127106001600160601b038216111561275e5760405162461bcd60e51b8152600401610d5690614556565b6001600160a01b0382166127b45760405162461bcd60e51b815260206004820152601b60248201527f455243323938313a20496e76616c696420706172616d657465727300000000006044820152606401610d56565b6040805180820182526001600160a01b0393841681526001600160601b0392831660208083019182526000968752606990529190942093519051909116600160a01b029116179055565b600061280983612f48565b8015611cef5750611cef8383612f7b565b610e9b338383613004565b61283084848461245e565b61283c848484846130d2565b61124e5760405162461bcd60e51b8152600401610d56906146f8565b6000611cef82846146e0565b60005b8281101561124e57600061287b6101685490565b905061288c61016880546001019055565b61289685826131d3565b6128a081846121d9565b50600101612867565b60006128b48861147c565b90508060e001516128d75760405162461bcd60e51b8152600401610d56906146a5565b602081015134906128e890896131ed565b11156129285760405162461bcd60e51b815260206004820152600f60248201526e15985b1d59481a5b98dbdc9c9958dd608a1b6044820152606401610d56565b80608001518711156129745760405162461bcd60e51b8152602060048201526015602482015274115e18d959591959081b585e081c195c881b5a5b9d605a1b6044820152606401610d56565b604081015160a08201516129889089612858565b11156129cc5760405162461bcd60e51b8152602060048201526013602482015272115e18d959591959081b585e08185b5bdd5b9d606a1b6044820152606401610d56565b606081015160008981526035602090815260408083203384529091529020546129f59089612858565b1115612a435760405162461bcd60e51b815260206004820152601760248201527f4578636565646564206d6178207065722077616c6c65740000000000000000006044820152606401610d56565b836001600160a01b0316612b1387878080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152505050610100850151612b0d9150612a9b908d90612858565b604051602001612aad91815260200190565b60408051601f1981840301815282825280516020918201207f19457468657265756d205369676e6564204d6573736167653a0a33320000000084830152603c8085019190915282518085039091018152605c909301909152815191012090565b906131f9565b6001600160a01b031614612b5d5760405162461bcd60e51b8152602060048201526011602482015270496e76616c6964207369676e617475726560781b6044820152606401610d56565b60c081015115612c1757612bdb83838080602002602001604051908101604052809392919081815260200183836020028082843760009201919091525050505060c08301516040516bffffffffffffffffffffffff193360601b1660208201526034016040516020818303038152906040528051906020012061321d565b612c175760405162461bcd60e51b815260206004820152600d60248201526c24b73b30b634b210383937b7b360991b6044820152606401610d56565b60008881526034602090815260408083206005018054600190810190915560358352818420338086529084529382902080549091019055518981528a92917fac100af3f918c30d71d2bfb00d975e36374d10738748c899376504e76c70bc73910160405180910390a35050505050505050565b600054610100900460ff16612cb15760405162461bcd60e51b8152600401610d569061474a565b610e9b8282613233565b600054610100900460ff16612ce25760405162461bcd60e51b8152600401610d569061474a565b565b600054610100900460ff16612d0b5760405162461bcd60e51b8152600401610d569061474a565b612d1484613273565b612d1c612cbb565b609a8054336001600160a01b031991821617909155609c80549091166001600160a01b03851617905561124e8282611e9e565b6060612d5a82611f58565b6000828152610104602052604081208054612d74906143f4565b80601f0160208091040260200160405190810160405280929190818152602001828054612da0906143f4565b8015612ded5780601f10612dc257610100808354040283529160200191612ded565b820191906000526020600020905b815481529060010190602001808311612dd057829003601f168201915b505050505090506000612e0b60408051602081019091526000815290565b90508051600003612e1d575092915050565b815115612e4f578082604051602001612e37929190614795565b60405160208183030381529060405292505050919050565b6124568461329d565b60006001600160e01b0319821663152a902d60e11b1480610b9e57506301ffc9a760e01b6001600160e01b0319831614610b9e565b600081815260a46020526040902080546001600160a01b0319166001600160a01b0384169081179091558190612ec28261141c565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b610be4838383613310565b612f0f816133c8565b6000818152610104602052604090208054612f29906143f4565b159050610e3357600081815261010460205260408120610e3391613bd9565b6000612f5b826301ffc9a760e01b612f7b565b8015610b9e5750612f74826001600160e01b0319612f7b565b1592915050565b604080516001600160e01b03198316602480830191909152825180830390910181526044909101909152602080820180516001600160e01b03166301ffc9a760e01b178152825160009392849283928392918391908a617530fa92503d91506000519050828015612fed575060208210155b8015612ff95750600081115b979650505050505050565b816001600160a01b0316836001600160a01b0316036130655760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c6572000000000000006044820152606401610d56565b6001600160a01b03838116600081815260a56020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b60006001600160a01b0384163b156131c857604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906131169033908990889088906004016147c4565b6020604051808303816000875af1925050508015613151575060408051601f3d908101601f1916820190925261314e91810190614801565b60015b6131ae573d80801561317f576040519150601f19603f3d011682016040523d82523d6000602084013e613184565b606091505b5080516000036131a65760405162461bcd60e51b8152600401610d56906146f8565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050612456565b506001949350505050565b610e9b82826040518060200160405280600081525061346f565b6000611cef82846144f7565b600080600061320885856134a2565b91509150613215816134e4565b509392505050565b60008261322a858461369a565b14949350505050565b600054610100900460ff1661325a5760405162461bcd60e51b8152600401610d569061474a565b60a061326683826145e6565b5060a1610be482826145e6565b6001600160a01b0316638b78c6d8198190558060006000805160206148788339815191528180a350565b60606132a882611f58565b60006132bf60408051602081019091526000815290565b905060008151116132df5760405180602001604052806000815250611cef565b806132e9846136df565b6040516020016132fa929190614795565b6040516020818303038152906040529392505050565b6001600160a01b03831661336b576133668160d48054600083815260d560205260408120829055600182018355919091527f9780e26d96b1f2a9a18ef8fc72d589dbf03ef788137b64f43897e83a91e7feec0155565b61338e565b816001600160a01b0316836001600160a01b03161461338e5761338e83826137df565b6001600160a01b0382166133a557610be48161387c565b826001600160a01b0316826001600160a01b031614610be457610be4828261392b565b60006133d38261141c565b90506133e181600084612efb565b6133ec600083612e8d565b6001600160a01b038116600090815260a3602052604081208054600192906134159084906146c9565b9091555050600082815260a2602052604080822080546001600160a01b0319169055518391906001600160a01b038416907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908390a45050565b613479838361396f565b61348660008484846130d2565b610be45760405162461bcd60e51b8152600401610d56906146f8565b60008082516041036134d85760208301516040840151606085015160001a6134cc87828585613abd565b94509450505050611046565b50600090506002611046565b60008160048111156134f8576134f861481e565b036135005750565b60018160048111156135145761351461481e565b036135615760405162461bcd60e51b815260206004820152601860248201527f45434453413a20696e76616c6964207369676e617475726500000000000000006044820152606401610d56565b60028160048111156135755761357561481e565b036135c25760405162461bcd60e51b815260206004820152601f60248201527f45434453413a20696e76616c6964207369676e6174757265206c656e677468006044820152606401610d56565b60038160048111156135d6576135d661481e565b0361362e5760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202773272076616c604482015261756560f01b6064820152608401610d56565b60048160048111156136425761364261481e565b03610e335760405162461bcd60e51b815260206004820152602260248201527f45434453413a20696e76616c6964207369676e6174757265202776272076616c604482015261756560f01b6064820152608401610d56565b600081815b8451811015613215576136cb828683815181106136be576136be614540565b6020026020010151613baa565b9150806136d781614834565b91505061369f565b6060816000036137065750506040805180820190915260018152600360fc1b602082015290565b8160005b8115613730578061371a81614834565b91506137299050600a8361452c565b915061370a565b6000816001600160401b0381111561374a5761374a613d6e565b6040519080825280601f01601f191660200182016040528015613774576020820181803683370190505b5090505b8415612456576137896001836146c9565b9150613796600a8661484d565b6137a19060306146e0565b60f81b8183815181106137b6576137b6614540565b60200101906001600160f81b031916908160001a9053506137d8600a8661452c565b9450613778565b600060016137ec846115d8565b6137f691906146c9565b600083815260d36020526040902054909150808214613849576001600160a01b038416600090815260d260209081526040808320858452825280832054848452818420819055835260d390915290208190555b50600091825260d3602090815260408084208490556001600160a01b03909416835260d281528383209183525290812055565b60d45460009061388e906001906146c9565b600083815260d5602052604081205460d480549394509092849081106138b6576138b6614540565b906000526020600020015490508060d483815481106138d7576138d7614540565b600091825260208083209091019290925582815260d5909152604080822084905585825281205560d480548061390f5761390f614861565b6001900381819060005260206000200160009055905550505050565b6000613936836115d8565b6001600160a01b03909316600090815260d260209081526040808320868452825280832085905593825260d39052919091209190915550565b6001600160a01b0382166139c55760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f20616464726573736044820152606401610d56565b600081815260a260205260409020546001600160a01b031615613a2a5760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e746564000000006044820152606401610d56565b613a3660008383612efb565b6001600160a01b038216600090815260a360205260408120805460019290613a5f9084906146e0565b9091555050600081815260a2602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a0831115613af45750600090506003613ba1565b8460ff16601b14158015613b0c57508460ff16601c14155b15613b1d5750600090506004613ba1565b6040805160008082526020820180845289905260ff881692820192909252606081018690526080810185905260019060a0016020604051602081039080840390855afa158015613b71573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b038116613b9a57600060019250925050613ba1565b9150600090505b94509492505050565b6000818310613bc6576000828152602084905260409020611cef565b6000838152602083905260409020611cef565b508054613be5906143f4565b6000825580601f10613bf5575050565b601f016020900490600052602060002090810190610e3391905b80821115613c235760008155600101613c0f565b5090565b6001600160e01b031981168114610e3357600080fd5b600060208284031215613c4f57600080fd5b8135611cef81613c27565b80356001600160a01b0381168114613c7157600080fd5b919050565b80356001600160601b0381168114613c7157600080fd5b60008060408385031215613ca057600080fd5b613ca983613c5a565b9150613cb760208401613c76565b90509250929050565b60005b83811015613cdb578181015183820152602001613cc3565b8381111561124e5750506000910152565b60008151808452613d04816020860160208601613cc0565b601f01601f19169290920160200192915050565b602081526000611cef6020830184613cec565b600060208284031215613d3d57600080fd5b5035919050565b60008060408385031215613d5757600080fd5b613d6083613c5a565b946020939093013593505050565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f191681016001600160401b0381118282101715613dac57613dac613d6e565b604052919050565b60006001600160401b03831115613dcd57613dcd613d6e565b613de0601f8401601f1916602001613d84565b9050828152838383011115613df457600080fd5b828260208301376000602084830101529392505050565b600082601f830112613e1c57600080fd5b611cef83833560208501613db4565b600080600080600060a08688031215613e4357600080fd5b85356001600160401b03811115613e5957600080fd5b613e6588828901613e0b565b9860208801359850604088013597606081013597506080013595509350505050565b60006020808385031215613e9a57600080fd5b82356001600160401b0380821115613eb157600080fd5b818501915085601f830112613ec557600080fd5b813581811115613ed757613ed7613d6e565b8060051b9150613ee8848301613d84565b8181529183018401918481019088841115613f0257600080fd5b938501935b83851015613f33578435925060ff83168314613f235760008081fd5b8282529385019390850190613f07565b98975050505050505050565b60008060408385031215613f5257600080fd5b8235915060208301356001600160401b03811115613f6f57600080fd5b613f7b85828601613e0b565b9150509250929050565b60008060408385031215613f9857600080fd5b50508035926020909101359150565b600080600060608486031215613fbc57600080fd5b613fc584613c5a565b9250613fd360208501613c5a565b9150604084013590509250925092565b600060208284031215613ff557600080fd5b611cef82613c5a565b60008060006060848603121561401357600080fd5b8335925061402360208501613c5a565b915061403160408501613c76565b90509250925092565b6020815260008251610120806020850152614059610140850183613cec565b9150602085015160408501526040850151606085015260608501516080850152608085015160a085015260a085015160c085015260c085015160e085015260e08501516101006140ac8187018315159052565b959095015193019290925250919050565b6020808252825182820181905260009190848201906040850190845b818110156140f857835160ff16835292840192918401916001016140d9565b50909695505050505050565b80358015158114613c7157600080fd5b6000806040838503121561412757600080fd5b82359150613cb760208401614104565b6000806040838503121561414a57600080fd5b61415383613c5a565b9150613cb760208401614104565b6000806000806080858703121561417757600080fd5b61418085613c5a565b935061418e60208601613c5a565b92506040850135915060608501356001600160401b038111156141b057600080fd5b8501601f810187136141c157600080fd5b6141d087823560208401613db4565b91505092959194509250565b60008083601f8401126141ee57600080fd5b5081356001600160401b0381111561420557600080fd5b6020830191508360208260051b850101111561104657600080fd5b6000806000806000806080878903121561423957600080fd5b863595506020870135945060408701356001600160401b038082111561425e57600080fd5b818901915089601f83011261427257600080fd5b81358181111561428157600080fd5b8a602082850101111561429357600080fd5b6020830196508095505060608901359150808211156142b157600080fd5b506142be89828a016141dc565b979a9699509497509295939492505050565b60008060008060008060c087890312156142e957600080fd5b6142f287613c5a565b955060208701356001600160401b038082111561430e57600080fd5b61431a8a838b01613e0b565b9650604089013591508082111561433057600080fd5b5061433d89828a01613e0b565b94505061434c60608801613c5a565b925061435a60808801613c5a565b915061436860a08801613c76565b90509295509295509295565b60008060006060848603121561438957600080fd5b61439284613c5a565b92506020840135915060408401356001600160401b038111156143b457600080fd5b6143c086828701613e0b565b9150509250925092565b600080604083850312156143dd57600080fd5b6143e683613c5a565b9150613cb760208401613c5a565b600181811c9082168061440857607f821691505b60208210810361442857634e487b7160e01b600052602260045260246000fd5b50919050565b60006020828403121561444057600080fd5b5051919050565b6020808252602c908201527f4f70657261746f7220686173206265656e20626c6f636b656420627920636f6e60408201526b3a3930b1ba1037bbb732b91760a11b606082015260800190565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b634e487b7160e01b600052601160045260246000fd5b6000816000190483118215151615614511576145116144e1565b500290565b634e487b7160e01b600052601260045260246000fd5b60008261453b5761453b614516565b500490565b634e487b7160e01b600052603260045260246000fd5b6020808252602a908201527f455243323938313a20726f79616c7479206665652077696c6c206578636565646040820152692073616c65507269636560b01b606082015260800190565b601f821115610be457600081815260208120601f850160051c810160208610156145c75750805b601f850160051c820191505b81811015610dae578281556001016145d3565b81516001600160401b038111156145ff576145ff613d6e565b6146138161460d84546143f4565b846145a0565b602080601f83116001811461464857600084156146305750858301515b600019600386901b1c1916600185901b178555610dae565b600085815260208120601f198616915b8281101561467757888601518255948401946001909101908401614658565b50858210156146955787850151600019600388901b60f8161c191681555b5050505050600190811b01905550565b6020808252600a90820152694e6f742061637469766560b01b604082015260600190565b6000828210156146db576146db6144e1565b500390565b600082198211156146f3576146f36144e1565b500190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b606082015260800190565b600083516147a7818460208801613cc0565b8351908301906147bb818360208801613cc0565b01949350505050565b6001600160a01b03858116825284166020820152604081018390526080606082018190526000906147f790830184613cec565b9695505050505050565b60006020828403121561481357600080fd5b8151611cef81613c27565b634e487b7160e01b600052602160045260246000fd5b600060018201614846576148466144e1565b5060010190565b60008261485c5761485c614516565b500690565b634e487b7160e01b600052603160045260246000fdfe8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0a264697066735822122099dde4802314b9e1f4c38da49ac82adb44e44ce5590540fe919aad3692f96e6964736f6c634300080f0033
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
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.