ETH Price: $2,431.71 (+0.31%)
 

Overview

TokenID

115

Total Transfers

-

Market

Onchain Market Cap

$0.00

Circulating Supply Market Cap

-
Loading...
Loading
Loading...
Loading
Loading...
Loading

Click here to update the token information / general information
# Exchange Pair Price  24H Volume % Volume

Contract Source Code Verified (Exact Match)

Contract Name:
discreetNFT

Compiler Version
v0.8.7+commit.e28d00a7

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2021-09-03
*/

// SPDX-License-Identifier: MIT
pragma solidity 0.8.7;


/**
 * @dev Interface for discreet.eth in addition to the standard ERC721 interface.
 */
interface discreetNFTInterface {
    /**
     * @dev Mint token with the supplied tokenId if it is currently available.
     */
    function mint(uint256 tokenId) external;

    /**
     * @dev Mint token with the supplied tokenId if it is currently available to
     * another address.
     */
    function mint(address to, uint256 tokenId) external;

    /**
     * @dev Burn token with the supplied tokenId if it is owned, approved or
     * reclaimable. Tokens become reclaimable after ~4 million blocks without a
     * mint or transfer.
     */
    function burn(uint256 tokenId) external;

    /**
     * @dev Check the current block number at which a given token will become
     * reclaimable.
     */
    function reclaimableThreshold(uint256 tokenId) external view returns (uint256);

    /**
     * @dev Check whether a given token is currently reclaimable.
     */
    function isReclaimable(uint256 tokenId) external view returns (bool);

    /**
     * @dev Retrieve just the image URI for a given token.
     */
    function tokenImageURI(uint256 tokenId) external view returns (string memory);
}


/**
 * @dev Interface of the ERC165 standard, as defined in the
 * https://eips.ethereum.org/EIPS/eip-165[EIP].
 */
interface IERC165 {
    /**
     * @dev Returns true if this contract implements the interface defined by
     * `interfaceId`. See the corresponding
     * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
     * to learn more about how these ids are created.
     *
     * This function call must use less than 30 000 gas.
     */
    function supportsInterface(bytes4 interfaceId) external view returns (bool);
}


/**
 * @dev Required interface of an ERC721 compliant contract.
 */
interface IERC721 is IERC165 {
    /**
     * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
     */
    event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
     */
    event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.
     */
    event ApprovalForAll(address indexed owner, address indexed operator, bool approved);

    /**
     * @dev Returns the number of tokens in ``owner``'s account.
     */
    function balanceOf(address owner) external view returns (uint256 balance);

    /**
     * @dev Returns the owner of the `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function ownerOf(uint256 tokenId) external view returns (address owner);

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
     * are aware of the ERC721 protocol to prevent tokens from being forever locked.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be have been allowed to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external;

    /**
     * @dev Transfers `tokenId` token from `from` to `to`.
     *
     * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external;

    /**
     * @dev Gives permission to `to` to transfer `tokenId` token to another account.
     * The approval is cleared when the token is transferred.
     *
     * Only a single account can be approved at a time, so approving the zero address clears previous approvals.
     *
     * Requirements:
     *
     * - The caller must own the token or be an approved operator.
     * - `tokenId` must exist.
     *
     * Emits an {Approval} event.
     */
    function approve(address to, uint256 tokenId) external;

    /**
     * @dev Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) external view returns (address operator);

    /**
     * @dev Approve or remove `operator` as an operator for the caller.
     * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.
     *
     * Requirements:
     *
     * - The `operator` cannot be the caller.
     *
     * Emits an {ApprovalForAll} event.
     */
    function setApprovalForAll(address operator, bool _approved) external;

    /**
     * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
     *
     * See {setApprovalForAll}
     */
    function isApprovedForAll(address owner, address operator) external view returns (bool);

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes calldata data
    ) external;
}


/**
 * @title ERC721 token receiver interface
 * @dev Interface for any contract that wants to support safeTransfers
 * from ERC721 asset contracts.
 */
interface IERC721Receiver {
    /**
     * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
     * by `operator` from `from`, this function is called.
     *
     * It must return its Solidity selector to confirm the token transfer.
     * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.
     *
     * The selector can be obtained in Solidity with `IERC721.onERC721Received.selector`.
     */
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external returns (bytes4);
}


/**
 * @title ERC-721 Non-Fungible Token Standard, optional metadata extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Metadata is IERC721 {
    /**
     * @dev Returns the token collection name.
     */
    function name() external view returns (string memory);

    /**
     * @dev Returns the token collection symbol.
     */
    function symbol() external view returns (string memory);

    /**
     * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.
     */
    function tokenURI(uint256 tokenId) external view returns (string memory);
}


/**
 * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Enumerable is IERC721 {
    /**
     * @dev Returns the total amount of tokens stored by the contract.
     */
    function totalSupply() external view returns (uint256);

    /**
     * @dev Returns a token ID owned by `owner` at a given `index` of its token list.
     * Use along with {balanceOf} to enumerate all of ``owner``'s tokens.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256 tokenId);

    /**
     * @dev Returns a token ID at a given `index` of all the tokens stored by the contract.
     * Use along with {totalSupply} to enumerate all tokens.
     */
    function tokenByIndex(uint256 index) external view returns (uint256);
}


interface IENSReverseRegistrar {
    function claim(address owner) external returns (bytes32 node);
    function setName(string calldata name) external returns (bytes32 node);
}


/**
 * @dev Implementation of the {IERC165} interface.
 */
abstract contract ERC165 is IERC165 {
    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        return interfaceId == type(IERC165).interfaceId;
    }
}


/// [MIT License]
/// @title Base64
/// @notice Provides a function for encoding some bytes in base64
/// @author Brecht Devos <[email protected]>
library Base64 {
    bytes internal constant TABLE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

    /// @notice Encodes some bytes to the base64 representation
    function encode(bytes memory data) internal pure returns (string memory) {
        uint256 len = data.length;
        if (len == 0) return "";

        // multiply by 4/3 rounded up
        uint256 encodedLen = 4 * ((len + 2) / 3);

        // Add some extra buffer at the end
        bytes memory result = new bytes(encodedLen + 32);

        bytes memory table = TABLE;

        assembly {
            let tablePtr := add(table, 1)
            let resultPtr := add(result, 32)

            for {
                let i := 0
            } lt(i, len) {

            } {
                i := add(i, 3)
                let input := and(mload(add(data, i)), 0xffffff)

                let out := mload(add(tablePtr, and(shr(18, input), 0x3F)))
                out := shl(8, out)
                out := add(out, and(mload(add(tablePtr, and(shr(12, input), 0x3F))), 0xFF))
                out := shl(8, out)
                out := add(out, and(mload(add(tablePtr, and(shr(6, input), 0x3F))), 0xFF))
                out := shl(8, out)
                out := add(out, and(mload(add(tablePtr, and(input, 0x3F))), 0xFF))
                out := shl(224, out)

                mstore(resultPtr, out)

                resultPtr := add(resultPtr, 4)
            }

            switch mod(len, 3)
            case 1 {
                mstore(sub(resultPtr, 2), shl(240, 0x3d3d))
            }
            case 2 {
                mstore(sub(resultPtr, 1), shl(248, 0x3d))
            }

            mstore(result, encodedLen)
        }

        return string(result);
    }
}


/**
 * @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 ERC721 is ERC165, IERC721, IERC721Metadata {
    // Token name
    bytes8 private immutable _name;

    // Token symbol
    bytes8 private immutable _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.
     */
    constructor(bytes8 name_, bytes8 symbol_) {
        _name = name_;
        _symbol = symbol_;
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
        return
            interfaceId == type(IERC721).interfaceId ||
            interfaceId == type(IERC721Metadata).interfaceId ||
            super.supportsInterface(interfaceId);
    }

    /**
     * @dev See {IERC721-balanceOf}.
     */
    function balanceOf(address owner) public view virtual override returns (uint256) {
        require(owner != address(0), "ERC721: balance query for the zero address");
        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: owner query for nonexistent token");
        return owner;
    }

    /**
     * @dev See {IERC721Metadata-name}.
     */
    function name() external view virtual override returns (string memory) {
        return string(abi.encodePacked(_name));
    }

    /**
     * @dev See {IERC721Metadata-symbol}.
     */
    function symbol() external view virtual override returns (string memory) {
        return string(abi.encodePacked(_symbol));
    }

    /**
     * @dev NOTE: standard functionality overridden.
     */
    function tokenURI(uint256 tokenId) external view virtual override returns (string memory) {}

    /**
     * @dev See {IERC721-approve}.
     */
    function approve(address to, uint256 tokenId) external virtual override {
        address owner = ERC721.ownerOf(tokenId);
        require(to != owner, "ERC721: approval to current owner");

        require(
            msg.sender == owner || isApprovedForAll(owner, msg.sender),
            "ERC721: approve caller is not owner nor approved for all"
        );

        _approve(to, tokenId);
    }

    /**
     * @dev See {IERC721-getApproved}.
     */
    function getApproved(uint256 tokenId) public view virtual override returns (address) {
        require(_exists(tokenId), "ERC721: approved query for nonexistent token");

        return _tokenApprovals[tokenId];
    }

    /**
     * @dev See {IERC721-setApprovalForAll}.
     */
    function setApprovalForAll(address operator, bool approved) external virtual override {
        require(operator != msg.sender, "ERC721: approve to caller");

        _operatorApprovals[msg.sender][operator] = approved;
        emit ApprovalForAll(msg.sender, 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
    ) external virtual override {
        //solhint-disable-next-line max-line-length
        require(_isApprovedOrOwner(msg.sender, tokenId), "ERC721: transfer caller is not owner nor approved");

        _transfer(from, to, tokenId);
    }

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external 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(msg.sender, tokenId), "ERC721: transfer caller is not 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) {
        require(_exists(tokenId), "ERC721: operator query for nonexistent token");
        address owner = ERC721.ownerOf(tokenId);
        return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, 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 {
        _mint(to, tokenId);
        require(
            _checkOnERC721Received(address(0), to, tokenId, ""),
            "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);
    }

    /**
     * @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 = ERC721.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);
    }

    /**
     * @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(ERC721.ownerOf(tokenId) == from, "ERC721: transfer of token that is not own");
        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);
    }

    /**
     * @dev Approve `to` to operate on `tokenId`
     *
     * Emits a {Approval} event.
     */
    function _approve(address to, uint256 tokenId) internal virtual {
        _tokenApprovals[tokenId] = to;
        emit Approval(ERC721.ownerOf(tokenId), to, tokenId);
    }

    /**
     * @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) {
        uint256 size;
        assembly { size := extcodesize(to) }
        if (size > 0) {
            try IERC721Receiver(to).onERC721Received(msg.sender, from, tokenId, _data) returns (bytes4 retval) {
                return retval == IERC721Receiver(to).onERC721Received.selector;
            } catch (bytes memory reason) {
                if (reason.length == 0) {
                    revert("ERC721: transfer to non ERC721Receiver implementer");
                } else {
                    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 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 ERC721Enumerable is ERC721, IERC721Enumerable {
    // 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(IERC165, ERC721) returns (bool) {
        return interfaceId == type(IERC721Enumerable).interfaceId || super.supportsInterface(interfaceId);
    }

    /**
     * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) external view virtual override returns (uint256) {
        require(index < ERC721.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) external view virtual override returns (uint256) {
        require(index < ERC721Enumerable.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 = ERC721.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 = ERC721.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 discreet (full set — replaces the 576 orignal and 288 extra NFTs and
 * adds an additional 432 for 1296 in total)
 * @author 0age
 */
contract discreetNFT is discreetNFTInterface, ERC721, ERC721Enumerable, IERC721Receiver {
    // Map tokenIds to block numbers past which they are burnable by any caller.
    mapping(uint256 => uint256) private _reclaimableThreshold;

    // Map transaction submitters to the block number of their last token mint.
    mapping(address => uint256) private _lastTokenMinted;

    discreetNFTInterface public constant originalSet = discreetNFTInterface(
        0x3c77065B584D4Af705B3E38CC35D336b081E4948
    );

    discreetNFTInterface public constant extraSet = discreetNFTInterface(
        0x04C0567cdBB51c3a9B1C907a56A5edA0EdeeBf71
    );

    uint256 public immutable migrationEnds;

    // Fixed base64-encoded SVG fragments used across all images.
    bytes32 private constant h0 = '';
    bytes32 private constant h1 = 'wgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz';
    bytes32 private constant h2 = '0iVVRGLTgiPz48c3ZnIHZpZXdCb3g9Ij';
    bytes32 private constant h3 = 'AgMCA1MDAgNTAwIiB4bWxucz0iaHR0cD';
    bytes32 private constant h4 = 'ovL3d3dy53My5vcmcvMjAwMC9zdmciIH';
    bytes32 private constant h5 = 'N0eWxlPSJiYWNrZ3JvdW5kLWNvbG9yOi';
    bytes4 private constant m0 = 'iPjx';
    bytes10 private constant m1 = 'BmaWxsPSIj';
    bytes16 private constant f0 = 'IiAvPjwvc3ZnPg==';

    /**
     * @dev Deploy discreet as an ERC721 NFT.
     */
    constructor() ERC721("discreet", "DISCREET") {
        // Set up ENS reverse registrar.
        IENSReverseRegistrar _ensReverseRegistrar = IENSReverseRegistrar(
            0x084b1c3C81545d370f3634392De611CaaBFf8148
        );

        _ensReverseRegistrar.claim(msg.sender);
        _ensReverseRegistrar.setName("discreet.eth");

        migrationEnds = block.number + 21600; // ~3 days
    }

    /**
     * @dev Throttle minting to once a block and reset the reclamation threshold
     * whenever a new token is minted or transferred.
     */
    function _beforeTokenTransfer(
        address from,
        address to,
        uint256 tokenId
    ) internal override(ERC721, ERC721Enumerable) {
        super._beforeTokenTransfer(from, to, tokenId);

        // If minting: ensure it's the only one from this tx origin in the block.
        if (from == address(0)) {
            require(
                block.number > _lastTokenMinted[tx.origin],
                "discreet: cannot mint multiple tokens per block from a single origin"
            );

            _lastTokenMinted[tx.origin] = block.number;
        }

        // If not burning: reset tokenId's reclaimable threshold block number.
        if (to != address(0)) {
            _reclaimableThreshold[tokenId] = block.number + 0x400000;
        }
    }

    /**
     * @dev Wrap an original or extra discreet NFT when transferred to this
     * contract via `safeTransferFrom` during the migration period.
     */
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external override returns (bytes4) {
        require(
            block.number < migrationEnds,
            "discreet: token migration is complete."
        );

        require(
            msg.sender == address(originalSet) || msg.sender == address(extraSet),
            "discreet: only accepts original or extra set discreet tokens."
        );

        if (msg.sender == address(originalSet)) {
            require(
                tokenId < 0x240,
                "discreet: only accepts original set discreet tokens with metadata"
            );
            _safeMint(from, tokenId);
        } else {
            require(
                tokenId < 0x120,
                "discreet: only accepts extra set discreet tokens with metadata"
            );
            _safeMint(from, tokenId + 0x240);
        }

        return this.onERC721Received.selector;
    }

    /**
     * @dev Mint a given discreet NFT if it is currently available.
     */
    function mint(uint256 tokenId) external override {
        require(
            tokenId < 0x510,
            "discreet: cannot mint out-of-range token"
        );

        if (tokenId < 0x360) {
            require(
                block.number >= migrationEnds,
                "discreet: cannot mint tokens from original or extra set until migration is complete."
            );
        }

        _safeMint(msg.sender, tokenId);
    }

    /**
     * @dev Mint a given NFT if it is currently available to a given address.
     */
    function mint(address to, uint256 tokenId) external override {
        require(
            tokenId < 0x510,
            "discreet: cannot mint out-of-range token"
        );

        if (tokenId < 0x360) {
            require(
                block.number >= migrationEnds,
                "discreet: cannot mint tokens from original or extra set until migration is complete."
            );
        }

        _safeMint(to, tokenId);
    }

    /**
     * @dev Burn a given discreet NFT if it is owned, approved or reclaimable.
     * Tokens become reclaimable after ~4 million blocks without a transfer.
     */
    function burn(uint256 tokenId) external override {
        require(
            tokenId < 0x510,
            "discreet: cannot burn out-of-range token"
        );

        // Only enforce check if tokenId has not reached reclaimable threshold.
        if (!isReclaimable(tokenId)) {
            require(
                _isApprovedOrOwner(msg.sender, tokenId),
                "discreet: caller is not owner nor approved"
            );
        }

        _burn(tokenId);
    }

    /**
     * @dev Check the current block number at which the given token will become
     * reclaimable.
     */
    function reclaimableThreshold(uint256 tokenId) public view override returns (uint256) {
        require(tokenId < 0x510, "discreet: out-of-range token");

        return _reclaimableThreshold[tokenId];
    }

    /**
     * @dev Check whether a given token is currently reclaimable.
     */
    function isReclaimable(uint256 tokenId) public view override returns (bool) {
        return reclaimableThreshold(tokenId) < block.number;
    }

    /**
     * @dev Derive and return a discreet tokenURI image formatted as a data URI.
     */
    function tokenImageURI(uint256 tokenId) public view virtual override returns (string memory) {
        require(tokenId < 0x510, "discreet: URI image query for out-of-range token");

        // Nine base64-encoded SVG fragments for background colors.
        bytes9[9] memory c0 = [
            bytes9('MwMDAwMDA'),
            'M2OWZmMzc',
            'NmZjM3Njk',
            'MzNzY5ZmY',
            'NmZmZmOTA',
            'M5MGZmZmY',
            'NmZjkwZmY',
            'NmZmZmZmY',
            'M4MDgwODA'
        ];

        // Eighteen base64-encoded SVG fragments for primary shapes.
        string[18] memory s0 = [
            'yZWN0IHg9IjE1NSIgeT0iNTUiIHdpZHRoPSIxOTAiIGhlaWdodD0iMzkwIi',
            'yZWN0IHg9IjU1IiB5PSIxNTUiIHdpZHRoPSIzOTAiIGhlaWdodD0iMTkwIi',
            'yZWN0IHg9IjExNSIgeT0iMTE1IiB3aWR0aD0iMjcwIiBoZWlnaHQ9IjI3MCIgIC',
            'jaXJjbGUgY3g9IjI1MCIgY3k9IjI1MCIgcj0iMTY1Ii',
            'lbGxpcHNlIGN4PSIyNTAiIGN5PSIyNTAiIHJ4PSIxMjUiIHJ5PSIxOTUiIC',
            'lbGxpcHNlIGN4PSIyNTAiIGN5PSIyNTAiIHJ4PSIxOTUiIHJ5PSIxMjUiIC',
            'wb2x5Z29uIHBvaW50cz0iMTAwLDEzNSAyNTAsNDAwIDQwMCwxMzUiIC',
            'wb2x5Z29uIHBvaW50cz0iNDAwLDM2NSAyNTAsMTAwIDEwMCwzNjUiIC',
            'wb2x5Z29uIHBvaW50cz0iNDAwLDEwMCA0MDAsNDAwIDEwMCw0MDAiIC',
            'wb2x5Z29uIHBvaW50cz0iMTAwLDQwMCA0MDAsNDAwIDEwMCwxMDAiIC',
            'wb2x5Z29uIHBvaW50cz0iMTAwLDQwMCA0MDAsMTAwIDEwMCwxMDAiIC',
            'wb2x5Z29uIHBvaW50cz0iNDAwLDQwMCA0MDAsMTAwIDEwMCwxMDAiIC',
            'wb2x5Z29uIHBvaW50cz0iMjMwLDQwMCAyNzAsNDAwIDI3MCwyNzAgNDAwLDI3MCA0MDAsMjMwIDI3MCwyMzAgMjcwLDEwMCAyMzAsMTAwIDIzMCwyMzAgMTAwLDIzMCAxMDAsMjcwIDIzMCwyNzAiIC',
            'wb2x5Z29uIHBvaW50cz0iMjMwLDQwMCAyNzAsNDAwIDI3MCwyNzAgNDAwLDI3MCA0MDAsMjMwIDI3MCwyMzAgMjcwLDEwMCAyMzAsMTAwIDIzMCwyMzAgMTAwLDIzMCAxMDAsMjcwIDIzMCwyNzAiIHRyYW5zZm9ybT0icm90YXRlKDQ1LDI1MCwyNTApIi',
            'wb2x5Z29uIHBvaW50cz0iMjUwLDQwMCAzNTAsMjUwIDI1MCwxMDAgMTUwLDI1MCIgIC',
            'wb2x5Z29uIHBvaW50cz0iMjUwLDEwMCAzMzgsMzcxIDEwNywyMDQgMzkzLDIwNCAxNjIsMzcxIi',
            'wb2x5Z29uIHBvaW50cz0iMzgwLDE3NSAzODAsMzI1IDI1MCw0MDAgMTIwLDMyNSAxMjAsMTc1IDI1MCwxMDAiIC',
            'wYXRoIGQ9Ik0wIDIwMCB2LTIwMCBoMjAwIGExMywxMSAwIDAsMSAwLDIwMCBhMTEsMTMgMCAwLDEgLTIwMCwwIiB0cmFuc2Zvcm09InJvdGF0ZSgyMjUsMjA4LDE0OCkgc2NhbGUoMC45MikiIC'
        ];

        // Nine base64-encoded SVG fragments for primary colors.
        bytes8[9] memory c1 = [
            bytes8('NjlmZjM3'),
            'ZmYzNzY5',
            'Mzc2OWZm',
            'ZmZmZjkw',
            'OTBmZmZm',
            'ZmY5MGZm',
            'ZmZmZmZm',
            'ODA4MDgw',
            'MDAwMDAw'
        ];

        // Construct a discrete tokenURI from a unique combination of the above.
        uint256 c0i = (tokenId % 72) / 8;
        uint256 s0i = tokenId / 72;
        uint256 c1i = (tokenId % 8 + (tokenId / 8)) % 9;
        return string(
            abi.encodePacked(
                h0, h1, h2, h3, h4, h5, c0[c0i], m0, s0[s0i], m1, c1[c1i], f0
           )
       );
    }

    /**
     * @dev Derive and return a tokenURI json payload formatted as a
     * data URI.
     */
    function tokenURI(uint256 tokenId) external view virtual override returns (string memory) {
        string memory json = Base64.encode(
            bytes(
                string(
                    abi.encodePacked(
                        '{"name": "discreet #',
                        _toString(tokenId),
                        '", "description": "One of 1296 distinct images, stored and derived ',
                        'entirely on-chain, that comprise the discreet.eth collection. It ',
                        'will become reclaimable if 4,194,304 blocks elapse without this ',
                        'token being minted or transferred.", "image": "',
                        tokenImageURI(tokenId),
                        '"}'
                    )
                )
            )
        );

        return string(abi.encodePacked('data:application/json;base64,', json));
    }

    /**
     * @dev Derive and return a contract-level json payload formatted as a
     * data URI.
     */
    function contractURI() public view returns (string memory) {
        return string(
            abi.encodePacked(
                'data:application/json;base64,',
                Base64.encode(
                    bytes(
                        string(
                            abi.encodePacked(
                                '{"name": "discreet.eth", ',
                                '"description": "A set of 1296 distinct images, stored and derived entirely ',
                                'on-chain, that comprise the discreet.eth collection. Each token will ',
                                'become reclaimable if 4,194,304 blocks elapse without a mint or transfer of ',
                                'the token in question. Created with #1283 by 0age."}'
                            )
                        )
                    )
                )
            )
        );
    }

    /**
     * @dev Coalesce supportsInterface from inherited contracts.
     */
    function supportsInterface(bytes4 interfaceId) public view override(ERC721, ERC721Enumerable) returns (bool) {
        return super.supportsInterface(interfaceId);
    }

    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);
    }
}

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"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":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"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":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"extraSet","outputs":[{"internalType":"contract discreetNFTInterface","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"isReclaimable","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"migrationEnds","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"address","name":"from","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"onERC721Received","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"originalSet","outputs":[{"internalType":"contract discreetNFTInterface","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"reclaimableThreshold","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":"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":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","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":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenImageURI","outputs":[{"internalType":"string","name":"","type":"string"}],"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":"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"}]

60e06040523480156200001157600080fd5b5067191a5cd8dc99595d60c21b60805267111254d0d491515560c21b60a052604051630f41a04d60e11b815233600482015273084b1c3c81545d370f3634392de611caabff8148908190631e83409a90602401602060405180830381600087803b1580156200007f57600080fd5b505af115801562000094573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620000ba91906200016e565b5060405163c47f002760e01b815260206004820152600c60248201526b0c8d2e6c6e4cacae85ccae8d60a31b60448201526001600160a01b0382169063c47f002790606401602060405180830381600087803b1580156200011a57600080fd5b505af11580156200012f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906200015591906200016e565b50620001644361546062000188565b60c05250620001af565b6000602082840312156200018157600080fd5b5051919050565b60008219821115620001aa57634e487b7160e01b600052601160045260246000fd5b500190565b60805160c01c60c01b60a05160c01c60c01b60c0516131676200020060003960008181610390015281816106190152818161097a015261127e015260006112210152600061042501526131676000f3fe608060405234801561001057600080fd5b50600436106101a95760003560e01c806357efac42116100f9578063a22cb46511610097578063c87b56dd11610071578063c87b56dd146103c5578063e8a3d485146103d8578063e9350248146103e0578063e985e9c5146103f357600080fd5b8063a22cb46514610378578063b40de9021461038b578063b88d4fde146103b257600080fd5b8063668db319116100d3578063668db3191461033757806370a082311461034a57806395d89b411461035d578063a0712d681461036557600080fd5b806357efac42146102f6578063602e1323146103095780636352211e1461032457600080fd5b806323b872dd1161016657806340c10f191161014057806340c10f19146102aa57806342842e0e146102bd57806342966c68146102d05780634f6ccce7146102e357600080fd5b806323b872dd146102695780632f745c591461027c57806334fc0bd31461028f57600080fd5b806301ffc9a7146101ae57806306fdde03146101d6578063081812fc146101eb578063095ea7b314610216578063150b7a021461022b57806318160ddd14610257575b600080fd5b6101c16101bc366004612568565b610406565b60405190151581526020015b60405180910390f35b6101de610417565b6040516101cd91906128ca565b6101fe6101f93660046125a2565b610465565b6040516001600160a01b0390911681526020016101cd565b61022961022436600461253e565b6104ff565b005b61023e61023936600461238b565b610615565b6040516001600160e01b031990911681526020016101cd565b6006545b6040519081526020016101cd565b61022961027736600461234f565b610886565b61025b61028a36600461253e565b6108b7565b6101fe7304c0567cdbb51c3a9b1c907a56a5eda0edeebf7181565b6102296102b836600461253e565b61094d565b6102296102cb36600461234f565b6109c6565b6102296102de3660046125a2565b6109e1565b61025b6102f13660046125a2565b610ac5565b6101de6103043660046125a2565b610b58565b6101fe733c77065b584d4af705b3e38cc35d336b081e494881565b6101fe6103323660046125a2565b611102565b6101c16103453660046125a2565b611179565b61025b6103583660046122fa565b61118c565b6101de611213565b6102296103733660046125a2565b611251565b610229610386366004612502565b6112c6565b61025b7f000000000000000000000000000000000000000000000000000000000000000081565b6102296103c0366004612426565b61138b565b6101de6103d33660046125a2565b6113c3565b6101de61142b565b61025b6103ee3660046125a2565b6115d3565b6101c161040136600461231c565b611639565b600061041182611667565b92915050565b6040516001600160c01b03197f00000000000000000000000000000000000000000000000000000000000000001660208201526060906028015b604051602081830303815290604052905090565b6000818152602081905260408120546001600160a01b03166104e35760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084015b60405180910390fd5b506000908152600260205260409020546001600160a01b031690565b600061050a82611102565b9050806001600160a01b0316836001600160a01b031614156105785760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084016104da565b336001600160a01b038216148061059457506105948133611639565b6106065760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c000000000000000060648201526084016104da565b610610838361168c565b505050565b60007f000000000000000000000000000000000000000000000000000000000000000043106106955760405162461bcd60e51b815260206004820152602660248201527f64697363726565743a20746f6b656e206d6967726174696f6e20697320636f6d604482015265383632ba329760d11b60648201526084016104da565b33733c77065b584d4af705b3e38cc35d336b081e494814806106ca5750337304c0567cdbb51c3a9b1c907a56a5eda0edeebf71145b61073c5760405162461bcd60e51b815260206004820152603d60248201527f64697363726565743a206f6e6c792061636365707473206f726967696e616c2060448201527f6f722065787472612073657420646973637265657420746f6b656e732e00000060648201526084016104da565b33733c77065b584d4af705b3e38cc35d336b081e494814156107e85761024084106107d95760405162461bcd60e51b815260206004820152604160248201527f64697363726565743a206f6e6c792061636365707473206f726967696e616c2060448201527f73657420646973637265657420746f6b656e732077697468206d6574616461746064820152606160f81b608482015260a4016104da565b6107e385856116fa565b610874565b610120841061085f5760405162461bcd60e51b815260206004820152603e60248201527f64697363726565743a206f6e6c7920616363657074732065787472612073657460448201527f20646973637265657420746f6b656e732077697468206d65746164617461000060648201526084016104da565b6108748561086f86610240612a42565b6116fa565b50630a85bd0160e11b95945050505050565b610890338261173c565b6108ac5760405162461bcd60e51b81526004016104da906129a9565b610610838383611813565b60006108c28361118c565b82106109245760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b60648201526084016104da565b506001600160a01b03919091166000908152600460209081526040808320938352929052205490565b610510811061096e5760405162461bcd60e51b81526004016104da906129fa565b6103608110156109b8577f00000000000000000000000000000000000000000000000000000000000000004310156109b85760405162461bcd60e51b81526004016104da9061292f565b6109c282826116fa565b5050565b6106108383836040518060200160405280600081525061138b565b6105108110610a435760405162461bcd60e51b815260206004820152602860248201527f64697363726565743a2063616e6e6f74206275726e206f75742d6f662d72616e60448201526733b2903a37b5b2b760c11b60648201526084016104da565b610a4c81611179565b610ab957610a5a338261173c565b610ab95760405162461bcd60e51b815260206004820152602a60248201527f64697363726565743a2063616c6c6572206973206e6f74206f776e6572206e6f6044820152691c88185c1c1c9bdd995960b21b60648201526084016104da565b610ac2816119c2565b50565b6000610ad060065490565b8210610b335760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b60648201526084016104da565b60068281548110610b4657610b46612b41565b90600052602060002001549050919050565b60606105108210610bc45760405162461bcd60e51b815260206004820152603060248201527f64697363726565743a2055524920696d61676520717565727920666f72206f7560448201526f3a16b7b316b930b733b2903a37b5b2b760811b60648201526084016104da565b6040805161012081018252684d774d4441774d444160b81b8152684d324f575a6d4d7a6360b81b6020820152684e6d5a6a4d334e6a6b60b81b81830152684d7a4e7a59355a6d5960b81b6060820152684e6d5a6d5a6d4f544160b81b6080820152684d354d475a6d5a6d5960b81b60a0820152684e6d5a6a6b775a6d5960b81b60c0820152684e6d5a6d5a6d5a6d5960b81b60e0820152684d344d4467774f444160b81b61010082015281516102a08101909252603b610240830181815291926000929091829190612c2d61026084013981526020016040518060600160405280603b8152602001613095603b913981526020016040518060600160405280603f8152602001612c68603f913981526020016040518060600160405280602b81526020016130d0602b913981526020016040518060600160405280603b8152602001612e5f603b913981526020016040518060600160405280603b8152602001612bf2603b91398152602001604051806060016040528060378152602001612d956037913981526020016040518060600160405280603781526020016130fb603791398152602001604051806060016040528060378152602001612b8460379139815260200160405180606001604052806037815260200161302760379139815260200160405180606001604052806037815260200161305e603791398152602001604051806060016040528060378152602001612bbb6037913981526020016040518060c0016040528060978152602001612cfe6097913981526020016040518060e0016040528060bf8152602001612f6860bf91398152602001604051806080016040528060438152602001612e9a6043913981526020016040518060800160405280604b8152602001612edd604b91398152602001604051806080016040528060578152602001612ca76057913981526020016040518060c0016040528060938152602001612dcc6093913990526040805161012081018252674e6a6c6d5a6a4d3360c01b8152675a6d597a4e7a593560c01b6020820152674d7a63324f575a6d60c01b91810191909152675a6d5a6d5a6a6b7760c01b6060820152674f54426d5a6d5a6d60c01b6080820152675a6d59354d475a6d60c01b60a0820152675a6d5a6d5a6d5a6d60c01b60c0820152674f4441344d44677760c01b60e0820152674d4441774d44417760c01b61010082015290915060006008610f4b604888612aeb565b610f559190612a5a565b90506000610f64604888612a5a565b905060006009610f7560088a612a5a565b610f8060088b612aeb565b610f8a9190612a42565b610f949190612aeb565b90507f646174613a696d6167652f7376672b786d6c3b6261736536342c5044393462577f7767646d567963326c76626a30694d5334774969426c626d4e765a476c755a7a7f3069565652474c546769507a343863335a6e49485a705a58644362336739496a7f41674d4341314d4441674e5441774969423462577875637a30696148523063447f6f764c336433647935334d793576636d63764d6a41774d43397a646d636949487f4e306557786c50534a6959574e725a334a766457356b4c574e76624739794f698b896009811061106e5761106e612b41565b6020020151630d2a0d4f60e31b8c8a6012811061108d5761108d612b41565b6020020151692136b0abbc39a829a4b560b11b8d8b600981106110b2576110b2612b41565b60200201516f49694176506a777663335a6e50673d3d60801b6040516020016110e69c9b9a99989796959493929190612603565b6040516020818303038152906040529650505050505050919050565b6000818152602081905260408120546001600160a01b0316806104115760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201526832b73a103a37b5b2b760b91b60648201526084016104da565b600043611185836115d3565b1092915050565b60006001600160a01b0382166111f75760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604482015269726f206164647265737360b01b60648201526084016104da565b506001600160a01b031660009081526001602052604090205490565b6040516001600160c01b03197f0000000000000000000000000000000000000000000000000000000000000000166020820152606090602801610451565b61051081106112725760405162461bcd60e51b81526004016104da906129fa565b6103608110156112bc577f00000000000000000000000000000000000000000000000000000000000000004310156112bc5760405162461bcd60e51b81526004016104da9061292f565b610ac233826116fa565b6001600160a01b03821633141561131f5760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c65720000000000000060448201526064016104da565b3360008181526003602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b611395338361173c565b6113b15760405162461bcd60e51b81526004016104da906129a9565b6113bd84848484611a6b565b50505050565b606060006114016113d384611a9e565b6113dc85610b58565b6040516020016113ed9291906126b4565b604051602081830303815290604052611b9c565b9050806040516020016114149190612848565b604051602081830303815290604052915050919050565b60606115c36040516020016113ed907f7b226e616d65223a202264697363726565742e657468222c200000000000000081527f226465736372697074696f6e223a20224120736574206f66203132393620646960198201527f7374696e637420696d616765732c2073746f72656420616e642064657269766560398201526a0321032b73a34b932b63c960ad1b60598201527f6f6e2d636861696e2c207468617420636f6d707269736520746865206469736360648201527f726565742e65746820636f6c6c656374696f6e2e204561636820746f6b656e2060848201526403bb4b636160dd1b60a48201527f6265636f6d65207265636c61696d61626c6520696620342c3139342c3330342060a98201527f626c6f636b7320656c6170736520776974686f75742061206d696e74206f722060c98201526b03a3930b739b332b91037b3160a51b60e98201527f74686520746f6b656e20696e207175657374696f6e2e2043726561746564207760f58201527369746820233132383320627920306167652e227d60601b6101158201526101290190565b6040516020016104519190612848565b600061051082106116265760405162461bcd60e51b815260206004820152601c60248201527f64697363726565743a206f75742d6f662d72616e676520746f6b656e0000000060448201526064016104da565b5060009081526008602052604090205490565b6001600160a01b03918216600090815260036020908152604080832093909416825291909152205460ff1690565b60006001600160e01b0319821663780e9d6360e01b1480610411575061041182611d02565b600081815260026020526040902080546001600160a01b0319166001600160a01b03841690811790915581906116c182611102565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b6117048282611d52565b6117206000838360405180602001604052806000815250611ea2565b6109c25760405162461bcd60e51b81526004016104da906128dd565b6000818152602081905260408120546001600160a01b03166117b55760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084016104da565b60006117c083611102565b9050806001600160a01b0316846001600160a01b031614806117fb5750836001600160a01b03166117f084610465565b6001600160a01b0316145b8061180b575061180b8185611639565b949350505050565b826001600160a01b031661182682611102565b6001600160a01b03161461188e5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201526839903737ba1037bbb760b91b60648201526084016104da565b6001600160a01b0382166118f05760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b60648201526084016104da565b6118fb838383611fa8565b61190660008261168c565b6001600160a01b03831660009081526001602081905260408220805491929091611931908490612a8d565b90915550506001600160a01b03821660009081526001602081905260408220805491929091611961908490612a42565b909155505060008181526020819052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b60006119cd82611102565b90506119db81600084611fa8565b6119e660008361168c565b6001600160a01b03811660009081526001602081905260408220805491929091611a11908490612a8d565b909155505060008281526020819052604080822080546001600160a01b0319169055518391906001600160a01b038416907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908390a45050565b611a76848484611813565b611a8284848484611ea2565b6113bd5760405162461bcd60e51b81526004016104da906128dd565b606081611ac25750506040805180820190915260018152600360fc1b602082015290565b8160005b8115611aec5780611ad681612ad0565b9150611ae59050600a83612a5a565b9150611ac6565b60008167ffffffffffffffff811115611b0757611b07612b57565b6040519080825280601f01601f191660200182016040528015611b31576020820181803683370190505b5090505b841561180b57611b46600183612a8d565b9150611b53600a86612aeb565b611b5e906030612a42565b60f81b818381518110611b7357611b73612b41565b60200101906001600160f81b031916908160001a905350611b95600a86612a5a565b9450611b35565b805160609080611bbc575050604080516020810190915260008152919050565b60006003611bcb836002612a42565b611bd59190612a5a565b611be0906004612a6e565b90506000611bef826020612a42565b67ffffffffffffffff811115611c0757611c07612b57565b6040519080825280601f01601f191660200182016040528015611c31576020820181803683370190505b5090506000604051806060016040528060408152602001612f28604091399050600181016020830160005b86811015611cbd576003818a01810151603f601282901c8116860151600c83901c8216870151600684901c831688015192909316870151600891821b60ff94851601821b92841692909201901b91160160e01b835260049092019101611c5c565b506003860660018114611cd75760028114611ce857611cf4565b613d3d60f01b600119830152611cf4565b603d60f81b6000198301525b505050918152949350505050565b60006001600160e01b031982166380ac58cd60e01b1480611d3357506001600160e01b03198216635b5e139f60e01b145b8061041157506301ffc9a760e01b6001600160e01b0319831614610411565b6001600160a01b038216611da85760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f206164647265737360448201526064016104da565b6000818152602081905260409020546001600160a01b031615611e0d5760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e7465640000000060448201526064016104da565b611e1960008383611fa8565b6001600160a01b03821660009081526001602081905260408220805491929091611e44908490612a42565b909155505060008181526020819052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6000833b8015611f9e57604051630a85bd0160e11b81526001600160a01b0386169063150b7a0290611ede9033908a908990899060040161288d565b602060405180830381600087803b158015611ef857600080fd5b505af1925050508015611f28575060408051601f3d908101601f19168201909252611f2591810190612585565b60015b611f82573d808015611f56576040519150601f19603f3d011682016040523d82523d6000602084013e611f5b565b606091505b508051611f7a5760405162461bcd60e51b81526004016104da906128dd565b805181602001fd5b6001600160e01b031916630a85bd0160e11b14915061180b9050565b600191505061180b565b611fb3838383612096565b6001600160a01b038316612066573260009081526009602052604090205443116120535760405162461bcd60e51b8152602060048201526044602482018190527f64697363726565743a2063616e6e6f74206d696e74206d756c7469706c652074908201527f6f6b656e732070657220626c6f636b2066726f6d20612073696e676c65206f7260648201526334b3b4b760e11b608482015260a4016104da565b3260009081526009602052604090204390555b6001600160a01b03821615610610576120824362400000612a42565b600082815260086020526040902055505050565b6001600160a01b0383166120f1576120ec81600680546000838152600760205260408120829055600182018355919091527ff652222313e28459528d920b65115c16c04f3efc82aaedc97be59f3f377c0d3f0155565b612114565b816001600160a01b0316836001600160a01b03161461211457612114838261214e565b6001600160a01b03821661212b57610610816121eb565b826001600160a01b0316826001600160a01b03161461061057610610828261229a565b6000600161215b8461118c565b6121659190612a8d565b6000838152600560205260409020549091508082146121b8576001600160a01b03841660009081526004602090815260408083208584528252808320548484528184208190558352600590915290208190555b5060009182526005602090815260408084208490556001600160a01b039094168352600481528383209183525290812055565b6006546000906121fd90600190612a8d565b6000838152600760205260408120546006805493945090928490811061222557612225612b41565b90600052602060002001549050806006838154811061224657612246612b41565b600091825260208083209091019290925582815260079091526040808220849055858252812055600680548061227e5761227e612b2b565b6001900381819060005260206000200160009055905550505050565b60006122a58361118c565b6001600160a01b039093166000908152600460209081526040808320868452825280832085905593825260059052919091209190915550565b80356001600160a01b03811681146122f557600080fd5b919050565b60006020828403121561230c57600080fd5b612315826122de565b9392505050565b6000806040838503121561232f57600080fd5b612338836122de565b9150612346602084016122de565b90509250929050565b60008060006060848603121561236457600080fd5b61236d846122de565b925061237b602085016122de565b9150604084013590509250925092565b6000806000806000608086880312156123a357600080fd5b6123ac866122de565b94506123ba602087016122de565b935060408601359250606086013567ffffffffffffffff808211156123de57600080fd5b818801915088601f8301126123f257600080fd5b81358181111561240157600080fd5b89602082850101111561241357600080fd5b9699959850939650602001949392505050565b6000806000806080858703121561243c57600080fd5b612445856122de565b9350612453602086016122de565b925060408501359150606085013567ffffffffffffffff8082111561247757600080fd5b818701915087601f83011261248b57600080fd5b81358181111561249d5761249d612b57565b604051601f8201601f19908116603f011681019083821181831017156124c5576124c5612b57565b816040528281528a60208487010111156124de57600080fd5b82602086016020830137600060208483010152809550505050505092959194509250565b6000806040838503121561251557600080fd5b61251e836122de565b91506020830135801515811461253357600080fd5b809150509250929050565b6000806040838503121561255157600080fd5b61255a836122de565b946020939093013593505050565b60006020828403121561257a57600080fd5b813561231581612b6d565b60006020828403121561259757600080fd5b815161231581612b6d565b6000602082840312156125b457600080fd5b5035919050565b600081518084526125d3816020860160208601612aa4565b601f01601f19169290920160200192915050565b600081516125f9818560208601612aa4565b9290920192915050565b8c81528b60208201528a60408201528960608201528860808201528760a082015268ffffffffffffffffff60b81b871660c082015263ffffffff60e01b861660c98201526000855161265c8160cd850160208a01612aa4565b6001600160b01b0319861660cd918401918201526001600160c01b0319851660d782015261269f60df8201856fffffffffffffffffffffffffffffffff19169052565b60ef019e9d5050505050505050505050505050565b737b226e616d65223a20226469736372656574202360601b815282516000906126e4816014850160208801612aa4565b7f222c20226465736372697074696f6e223a20224f6e65206f66203132393620646014918401918201527f697374696e637420696d616765732c2073746f72656420616e64206465726976603482015262032b2160ed1b60548201527f656e746972656c79206f6e2d636861696e2c207468617420636f6d707269736560578201527f207468652064697363726565742e65746820636f6c6c656374696f6e2e2049746077820152600160fd1b60978201527f77696c6c206265636f6d65207265636c61696d61626c6520696620342c31393460988201527f2c33303420626c6f636b7320656c6170736520776974686f757420746869732060b88201527f746f6b656e206265696e67206d696e746564206f72207472616e73666572726560d88201526e32171116101134b6b0b3b2911d101160891b60f882015261283f6128316101078301866125e7565b61227d60f01b815260020190565b95945050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c00000081526000825161288081601d850160208701612aa4565b91909101601d0192915050565b6001600160a01b03858116825284166020820152604081018390526080606082018190526000906128c0908301846125bb565b9695505050505050565b60208152600061231560208301846125bb565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526054908201527f64697363726565743a2063616e6e6f74206d696e7420746f6b656e732066726f60408201527f6d206f726967696e616c206f722065787472612073657420756e74696c206d6960608201527333b930ba34b7b71034b99031b7b6b83632ba329760611b608082015260a00190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b60208082526028908201527f64697363726565743a2063616e6e6f74206d696e74206f75742d6f662d72616e60408201526733b2903a37b5b2b760c11b606082015260800190565b60008219821115612a5557612a55612aff565b500190565b600082612a6957612a69612b15565b500490565b6000816000190483118215151615612a8857612a88612aff565b500290565b600082821015612a9f57612a9f612aff565b500390565b60005b83811015612abf578181015183820152602001612aa7565b838111156113bd5750506000910152565b6000600019821415612ae457612ae4612aff565b5060010190565b600082612afa57612afa612b15565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052603160045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b031981168114610ac257600080fdfe77623278355a3239754948427661573530637a30694e4441774c4445774d4341304d4441734e444177494445774d4377304d444169494377623278355a3239754948427661573530637a30694e4441774c4451774d4341304d4441734d544177494445774d4377784d44416949436c6247787063484e6c49474e34505349794e54416949474e35505349794e54416949484a34505349784f54556949484a35505349784d6a55694943795a574e3049486739496a45314e534967655430694e545569494864705a48526f505349784f5441694947686c6157646f644430694d7a6b774969795a574e3049486739496a45784e534967655430694d5445314969423361575230614430694d6a63774969426f5a576c6e61485139496a49334d434967494377623278355a3239754948427661573530637a30694d7a67774c4445334e53417a4f4441734d7a4931494449314d4377304d4441674d5449774c444d794e5341784d6a41734d546331494449314d4377784d444169494377623278355a3239754948427661573530637a30694d6a4d774c4451774d4341794e7a41734e444177494449334d4377794e7a41674e4441774c4449334d4341304d4441734d6a4d77494449334d4377794d7a41674d6a63774c4445774d4341794d7a41734d5441774944497a4d4377794d7a41674d5441774c44497a4d4341784d4441734d6a63774944497a4d4377794e7a4169494377623278355a3239754948427661573530637a30694d5441774c44457a4e5341794e5441734e444177494451774d4377784d7a55694943775958526f49475139496b3077494449774d4342324c5449774d43426f4d6a4177494745784d7977784d534177494441734d5341774c4449774d4342684d5445734d544d674d4341774c4445674c5449774d43777749694230636d467563325a76636d3039496e4a76644746305a5367794d6a55734d6a41344c4445304f436b6763324e686247556f4d4334354d696b6949436c6247787063484e6c49474e34505349794e54416949474e35505349794e54416949484a34505349784d6a556949484a35505349784f545569494377623278355a3239754948427661573530637a30694d6a55774c4451774d43417a4e5441734d6a5577494449314d4377784d4441674d5455774c4449314d434967494377623278355a3239754948427661573530637a30694d6a55774c4445774d43417a4d7a67734d7a6378494445774e7977794d4451674d7a6b7a4c4449774e4341784e6a49734d7a637849694142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2f77623278355a3239754948427661573530637a30694d6a4d774c4451774d4341794e7a41734e444177494449334d4377794e7a41674e4441774c4449334d4341304d4441734d6a4d77494449334d4377794d7a41674d6a63774c4445774d4341794d7a41734d5441774944497a4d4377794d7a41674d5441774c44497a4d4341784d4441734d6a63774944497a4d4377794e7a4169494852795957357a5a6d397962543069636d39305958526c4b4451314c4449314d4377794e544170496977623278355a3239754948427661573530637a30694d5441774c4451774d4341304d4441734e444177494445774d4377784d444169494377623278355a3239754948427661573530637a30694d5441774c4451774d4341304d4441734d544177494445774d4377784d4441694943795a574e3049486739496a553149694235505349784e545569494864705a48526f5053497a4f5441694947686c6157646f644430694d546b7749696a61584a6a6247556759336739496a49314d43496759336b39496a49314d434967636a30694d545931496977623278355a3239754948427661573530637a30694e4441774c444d324e5341794e5441734d544177494445774d43777a4e6a55694943a26469706673582212202da75c66385d0ccc5f5962ea14db55e9ad846b9193345edb0c071c6bead1e95664736f6c63430008070033

Deployed Bytecode

0x608060405234801561001057600080fd5b50600436106101a95760003560e01c806357efac42116100f9578063a22cb46511610097578063c87b56dd11610071578063c87b56dd146103c5578063e8a3d485146103d8578063e9350248146103e0578063e985e9c5146103f357600080fd5b8063a22cb46514610378578063b40de9021461038b578063b88d4fde146103b257600080fd5b8063668db319116100d3578063668db3191461033757806370a082311461034a57806395d89b411461035d578063a0712d681461036557600080fd5b806357efac42146102f6578063602e1323146103095780636352211e1461032457600080fd5b806323b872dd1161016657806340c10f191161014057806340c10f19146102aa57806342842e0e146102bd57806342966c68146102d05780634f6ccce7146102e357600080fd5b806323b872dd146102695780632f745c591461027c57806334fc0bd31461028f57600080fd5b806301ffc9a7146101ae57806306fdde03146101d6578063081812fc146101eb578063095ea7b314610216578063150b7a021461022b57806318160ddd14610257575b600080fd5b6101c16101bc366004612568565b610406565b60405190151581526020015b60405180910390f35b6101de610417565b6040516101cd91906128ca565b6101fe6101f93660046125a2565b610465565b6040516001600160a01b0390911681526020016101cd565b61022961022436600461253e565b6104ff565b005b61023e61023936600461238b565b610615565b6040516001600160e01b031990911681526020016101cd565b6006545b6040519081526020016101cd565b61022961027736600461234f565b610886565b61025b61028a36600461253e565b6108b7565b6101fe7304c0567cdbb51c3a9b1c907a56a5eda0edeebf7181565b6102296102b836600461253e565b61094d565b6102296102cb36600461234f565b6109c6565b6102296102de3660046125a2565b6109e1565b61025b6102f13660046125a2565b610ac5565b6101de6103043660046125a2565b610b58565b6101fe733c77065b584d4af705b3e38cc35d336b081e494881565b6101fe6103323660046125a2565b611102565b6101c16103453660046125a2565b611179565b61025b6103583660046122fa565b61118c565b6101de611213565b6102296103733660046125a2565b611251565b610229610386366004612502565b6112c6565b61025b7f0000000000000000000000000000000000000000000000000000000000c8fa0b81565b6102296103c0366004612426565b61138b565b6101de6103d33660046125a2565b6113c3565b6101de61142b565b61025b6103ee3660046125a2565b6115d3565b6101c161040136600461231c565b611639565b600061041182611667565b92915050565b6040516001600160c01b03197f64697363726565740000000000000000000000000000000000000000000000001660208201526060906028015b604051602081830303815290604052905090565b6000818152602081905260408120546001600160a01b03166104e35760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084015b60405180910390fd5b506000908152600260205260409020546001600160a01b031690565b600061050a82611102565b9050806001600160a01b0316836001600160a01b031614156105785760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084016104da565b336001600160a01b038216148061059457506105948133611639565b6106065760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c000000000000000060648201526084016104da565b610610838361168c565b505050565b60007f0000000000000000000000000000000000000000000000000000000000c8fa0b43106106955760405162461bcd60e51b815260206004820152602660248201527f64697363726565743a20746f6b656e206d6967726174696f6e20697320636f6d604482015265383632ba329760d11b60648201526084016104da565b33733c77065b584d4af705b3e38cc35d336b081e494814806106ca5750337304c0567cdbb51c3a9b1c907a56a5eda0edeebf71145b61073c5760405162461bcd60e51b815260206004820152603d60248201527f64697363726565743a206f6e6c792061636365707473206f726967696e616c2060448201527f6f722065787472612073657420646973637265657420746f6b656e732e00000060648201526084016104da565b33733c77065b584d4af705b3e38cc35d336b081e494814156107e85761024084106107d95760405162461bcd60e51b815260206004820152604160248201527f64697363726565743a206f6e6c792061636365707473206f726967696e616c2060448201527f73657420646973637265657420746f6b656e732077697468206d6574616461746064820152606160f81b608482015260a4016104da565b6107e385856116fa565b610874565b610120841061085f5760405162461bcd60e51b815260206004820152603e60248201527f64697363726565743a206f6e6c7920616363657074732065787472612073657460448201527f20646973637265657420746f6b656e732077697468206d65746164617461000060648201526084016104da565b6108748561086f86610240612a42565b6116fa565b50630a85bd0160e11b95945050505050565b610890338261173c565b6108ac5760405162461bcd60e51b81526004016104da906129a9565b610610838383611813565b60006108c28361118c565b82106109245760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b60648201526084016104da565b506001600160a01b03919091166000908152600460209081526040808320938352929052205490565b610510811061096e5760405162461bcd60e51b81526004016104da906129fa565b6103608110156109b8577f0000000000000000000000000000000000000000000000000000000000c8fa0b4310156109b85760405162461bcd60e51b81526004016104da9061292f565b6109c282826116fa565b5050565b6106108383836040518060200160405280600081525061138b565b6105108110610a435760405162461bcd60e51b815260206004820152602860248201527f64697363726565743a2063616e6e6f74206275726e206f75742d6f662d72616e60448201526733b2903a37b5b2b760c11b60648201526084016104da565b610a4c81611179565b610ab957610a5a338261173c565b610ab95760405162461bcd60e51b815260206004820152602a60248201527f64697363726565743a2063616c6c6572206973206e6f74206f776e6572206e6f6044820152691c88185c1c1c9bdd995960b21b60648201526084016104da565b610ac2816119c2565b50565b6000610ad060065490565b8210610b335760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b60648201526084016104da565b60068281548110610b4657610b46612b41565b90600052602060002001549050919050565b60606105108210610bc45760405162461bcd60e51b815260206004820152603060248201527f64697363726565743a2055524920696d61676520717565727920666f72206f7560448201526f3a16b7b316b930b733b2903a37b5b2b760811b60648201526084016104da565b6040805161012081018252684d774d4441774d444160b81b8152684d324f575a6d4d7a6360b81b6020820152684e6d5a6a4d334e6a6b60b81b81830152684d7a4e7a59355a6d5960b81b6060820152684e6d5a6d5a6d4f544160b81b6080820152684d354d475a6d5a6d5960b81b60a0820152684e6d5a6a6b775a6d5960b81b60c0820152684e6d5a6d5a6d5a6d5960b81b60e0820152684d344d4467774f444160b81b61010082015281516102a08101909252603b610240830181815291926000929091829190612c2d61026084013981526020016040518060600160405280603b8152602001613095603b913981526020016040518060600160405280603f8152602001612c68603f913981526020016040518060600160405280602b81526020016130d0602b913981526020016040518060600160405280603b8152602001612e5f603b913981526020016040518060600160405280603b8152602001612bf2603b91398152602001604051806060016040528060378152602001612d956037913981526020016040518060600160405280603781526020016130fb603791398152602001604051806060016040528060378152602001612b8460379139815260200160405180606001604052806037815260200161302760379139815260200160405180606001604052806037815260200161305e603791398152602001604051806060016040528060378152602001612bbb6037913981526020016040518060c0016040528060978152602001612cfe6097913981526020016040518060e0016040528060bf8152602001612f6860bf91398152602001604051806080016040528060438152602001612e9a6043913981526020016040518060800160405280604b8152602001612edd604b91398152602001604051806080016040528060578152602001612ca76057913981526020016040518060c0016040528060938152602001612dcc6093913990526040805161012081018252674e6a6c6d5a6a4d3360c01b8152675a6d597a4e7a593560c01b6020820152674d7a63324f575a6d60c01b91810191909152675a6d5a6d5a6a6b7760c01b6060820152674f54426d5a6d5a6d60c01b6080820152675a6d59354d475a6d60c01b60a0820152675a6d5a6d5a6d5a6d60c01b60c0820152674f4441344d44677760c01b60e0820152674d4441774d44417760c01b61010082015290915060006008610f4b604888612aeb565b610f559190612a5a565b90506000610f64604888612a5a565b905060006009610f7560088a612a5a565b610f8060088b612aeb565b610f8a9190612a42565b610f949190612aeb565b90507f646174613a696d6167652f7376672b786d6c3b6261736536342c5044393462577f7767646d567963326c76626a30694d5334774969426c626d4e765a476c755a7a7f3069565652474c546769507a343863335a6e49485a705a58644362336739496a7f41674d4341314d4441674e5441774969423462577875637a30696148523063447f6f764c336433647935334d793576636d63764d6a41774d43397a646d636949487f4e306557786c50534a6959574e725a334a766457356b4c574e76624739794f698b896009811061106e5761106e612b41565b6020020151630d2a0d4f60e31b8c8a6012811061108d5761108d612b41565b6020020151692136b0abbc39a829a4b560b11b8d8b600981106110b2576110b2612b41565b60200201516f49694176506a777663335a6e50673d3d60801b6040516020016110e69c9b9a99989796959493929190612603565b6040516020818303038152906040529650505050505050919050565b6000818152602081905260408120546001600160a01b0316806104115760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201526832b73a103a37b5b2b760b91b60648201526084016104da565b600043611185836115d3565b1092915050565b60006001600160a01b0382166111f75760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604482015269726f206164647265737360b01b60648201526084016104da565b506001600160a01b031660009081526001602052604090205490565b6040516001600160c01b03197f4449534352454554000000000000000000000000000000000000000000000000166020820152606090602801610451565b61051081106112725760405162461bcd60e51b81526004016104da906129fa565b6103608110156112bc577f0000000000000000000000000000000000000000000000000000000000c8fa0b4310156112bc5760405162461bcd60e51b81526004016104da9061292f565b610ac233826116fa565b6001600160a01b03821633141561131f5760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c65720000000000000060448201526064016104da565b3360008181526003602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b611395338361173c565b6113b15760405162461bcd60e51b81526004016104da906129a9565b6113bd84848484611a6b565b50505050565b606060006114016113d384611a9e565b6113dc85610b58565b6040516020016113ed9291906126b4565b604051602081830303815290604052611b9c565b9050806040516020016114149190612848565b604051602081830303815290604052915050919050565b60606115c36040516020016113ed907f7b226e616d65223a202264697363726565742e657468222c200000000000000081527f226465736372697074696f6e223a20224120736574206f66203132393620646960198201527f7374696e637420696d616765732c2073746f72656420616e642064657269766560398201526a0321032b73a34b932b63c960ad1b60598201527f6f6e2d636861696e2c207468617420636f6d707269736520746865206469736360648201527f726565742e65746820636f6c6c656374696f6e2e204561636820746f6b656e2060848201526403bb4b636160dd1b60a48201527f6265636f6d65207265636c61696d61626c6520696620342c3139342c3330342060a98201527f626c6f636b7320656c6170736520776974686f75742061206d696e74206f722060c98201526b03a3930b739b332b91037b3160a51b60e98201527f74686520746f6b656e20696e207175657374696f6e2e2043726561746564207760f58201527369746820233132383320627920306167652e227d60601b6101158201526101290190565b6040516020016104519190612848565b600061051082106116265760405162461bcd60e51b815260206004820152601c60248201527f64697363726565743a206f75742d6f662d72616e676520746f6b656e0000000060448201526064016104da565b5060009081526008602052604090205490565b6001600160a01b03918216600090815260036020908152604080832093909416825291909152205460ff1690565b60006001600160e01b0319821663780e9d6360e01b1480610411575061041182611d02565b600081815260026020526040902080546001600160a01b0319166001600160a01b03841690811790915581906116c182611102565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b6117048282611d52565b6117206000838360405180602001604052806000815250611ea2565b6109c25760405162461bcd60e51b81526004016104da906128dd565b6000818152602081905260408120546001600160a01b03166117b55760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084016104da565b60006117c083611102565b9050806001600160a01b0316846001600160a01b031614806117fb5750836001600160a01b03166117f084610465565b6001600160a01b0316145b8061180b575061180b8185611639565b949350505050565b826001600160a01b031661182682611102565b6001600160a01b03161461188e5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201526839903737ba1037bbb760b91b60648201526084016104da565b6001600160a01b0382166118f05760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b60648201526084016104da565b6118fb838383611fa8565b61190660008261168c565b6001600160a01b03831660009081526001602081905260408220805491929091611931908490612a8d565b90915550506001600160a01b03821660009081526001602081905260408220805491929091611961908490612a42565b909155505060008181526020819052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b60006119cd82611102565b90506119db81600084611fa8565b6119e660008361168c565b6001600160a01b03811660009081526001602081905260408220805491929091611a11908490612a8d565b909155505060008281526020819052604080822080546001600160a01b0319169055518391906001600160a01b038416907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908390a45050565b611a76848484611813565b611a8284848484611ea2565b6113bd5760405162461bcd60e51b81526004016104da906128dd565b606081611ac25750506040805180820190915260018152600360fc1b602082015290565b8160005b8115611aec5780611ad681612ad0565b9150611ae59050600a83612a5a565b9150611ac6565b60008167ffffffffffffffff811115611b0757611b07612b57565b6040519080825280601f01601f191660200182016040528015611b31576020820181803683370190505b5090505b841561180b57611b46600183612a8d565b9150611b53600a86612aeb565b611b5e906030612a42565b60f81b818381518110611b7357611b73612b41565b60200101906001600160f81b031916908160001a905350611b95600a86612a5a565b9450611b35565b805160609080611bbc575050604080516020810190915260008152919050565b60006003611bcb836002612a42565b611bd59190612a5a565b611be0906004612a6e565b90506000611bef826020612a42565b67ffffffffffffffff811115611c0757611c07612b57565b6040519080825280601f01601f191660200182016040528015611c31576020820181803683370190505b5090506000604051806060016040528060408152602001612f28604091399050600181016020830160005b86811015611cbd576003818a01810151603f601282901c8116860151600c83901c8216870151600684901c831688015192909316870151600891821b60ff94851601821b92841692909201901b91160160e01b835260049092019101611c5c565b506003860660018114611cd75760028114611ce857611cf4565b613d3d60f01b600119830152611cf4565b603d60f81b6000198301525b505050918152949350505050565b60006001600160e01b031982166380ac58cd60e01b1480611d3357506001600160e01b03198216635b5e139f60e01b145b8061041157506301ffc9a760e01b6001600160e01b0319831614610411565b6001600160a01b038216611da85760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f206164647265737360448201526064016104da565b6000818152602081905260409020546001600160a01b031615611e0d5760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e7465640000000060448201526064016104da565b611e1960008383611fa8565b6001600160a01b03821660009081526001602081905260408220805491929091611e44908490612a42565b909155505060008181526020819052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6000833b8015611f9e57604051630a85bd0160e11b81526001600160a01b0386169063150b7a0290611ede9033908a908990899060040161288d565b602060405180830381600087803b158015611ef857600080fd5b505af1925050508015611f28575060408051601f3d908101601f19168201909252611f2591810190612585565b60015b611f82573d808015611f56576040519150601f19603f3d011682016040523d82523d6000602084013e611f5b565b606091505b508051611f7a5760405162461bcd60e51b81526004016104da906128dd565b805181602001fd5b6001600160e01b031916630a85bd0160e11b14915061180b9050565b600191505061180b565b611fb3838383612096565b6001600160a01b038316612066573260009081526009602052604090205443116120535760405162461bcd60e51b8152602060048201526044602482018190527f64697363726565743a2063616e6e6f74206d696e74206d756c7469706c652074908201527f6f6b656e732070657220626c6f636b2066726f6d20612073696e676c65206f7260648201526334b3b4b760e11b608482015260a4016104da565b3260009081526009602052604090204390555b6001600160a01b03821615610610576120824362400000612a42565b600082815260086020526040902055505050565b6001600160a01b0383166120f1576120ec81600680546000838152600760205260408120829055600182018355919091527ff652222313e28459528d920b65115c16c04f3efc82aaedc97be59f3f377c0d3f0155565b612114565b816001600160a01b0316836001600160a01b03161461211457612114838261214e565b6001600160a01b03821661212b57610610816121eb565b826001600160a01b0316826001600160a01b03161461061057610610828261229a565b6000600161215b8461118c565b6121659190612a8d565b6000838152600560205260409020549091508082146121b8576001600160a01b03841660009081526004602090815260408083208584528252808320548484528184208190558352600590915290208190555b5060009182526005602090815260408084208490556001600160a01b039094168352600481528383209183525290812055565b6006546000906121fd90600190612a8d565b6000838152600760205260408120546006805493945090928490811061222557612225612b41565b90600052602060002001549050806006838154811061224657612246612b41565b600091825260208083209091019290925582815260079091526040808220849055858252812055600680548061227e5761227e612b2b565b6001900381819060005260206000200160009055905550505050565b60006122a58361118c565b6001600160a01b039093166000908152600460209081526040808320868452825280832085905593825260059052919091209190915550565b80356001600160a01b03811681146122f557600080fd5b919050565b60006020828403121561230c57600080fd5b612315826122de565b9392505050565b6000806040838503121561232f57600080fd5b612338836122de565b9150612346602084016122de565b90509250929050565b60008060006060848603121561236457600080fd5b61236d846122de565b925061237b602085016122de565b9150604084013590509250925092565b6000806000806000608086880312156123a357600080fd5b6123ac866122de565b94506123ba602087016122de565b935060408601359250606086013567ffffffffffffffff808211156123de57600080fd5b818801915088601f8301126123f257600080fd5b81358181111561240157600080fd5b89602082850101111561241357600080fd5b9699959850939650602001949392505050565b6000806000806080858703121561243c57600080fd5b612445856122de565b9350612453602086016122de565b925060408501359150606085013567ffffffffffffffff8082111561247757600080fd5b818701915087601f83011261248b57600080fd5b81358181111561249d5761249d612b57565b604051601f8201601f19908116603f011681019083821181831017156124c5576124c5612b57565b816040528281528a60208487010111156124de57600080fd5b82602086016020830137600060208483010152809550505050505092959194509250565b6000806040838503121561251557600080fd5b61251e836122de565b91506020830135801515811461253357600080fd5b809150509250929050565b6000806040838503121561255157600080fd5b61255a836122de565b946020939093013593505050565b60006020828403121561257a57600080fd5b813561231581612b6d565b60006020828403121561259757600080fd5b815161231581612b6d565b6000602082840312156125b457600080fd5b5035919050565b600081518084526125d3816020860160208601612aa4565b601f01601f19169290920160200192915050565b600081516125f9818560208601612aa4565b9290920192915050565b8c81528b60208201528a60408201528960608201528860808201528760a082015268ffffffffffffffffff60b81b871660c082015263ffffffff60e01b861660c98201526000855161265c8160cd850160208a01612aa4565b6001600160b01b0319861660cd918401918201526001600160c01b0319851660d782015261269f60df8201856fffffffffffffffffffffffffffffffff19169052565b60ef019e9d5050505050505050505050505050565b737b226e616d65223a20226469736372656574202360601b815282516000906126e4816014850160208801612aa4565b7f222c20226465736372697074696f6e223a20224f6e65206f66203132393620646014918401918201527f697374696e637420696d616765732c2073746f72656420616e64206465726976603482015262032b2160ed1b60548201527f656e746972656c79206f6e2d636861696e2c207468617420636f6d707269736560578201527f207468652064697363726565742e65746820636f6c6c656374696f6e2e2049746077820152600160fd1b60978201527f77696c6c206265636f6d65207265636c61696d61626c6520696620342c31393460988201527f2c33303420626c6f636b7320656c6170736520776974686f757420746869732060b88201527f746f6b656e206265696e67206d696e746564206f72207472616e73666572726560d88201526e32171116101134b6b0b3b2911d101160891b60f882015261283f6128316101078301866125e7565b61227d60f01b815260020190565b95945050505050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c00000081526000825161288081601d850160208701612aa4565b91909101601d0192915050565b6001600160a01b03858116825284166020820152604081018390526080606082018190526000906128c0908301846125bb565b9695505050505050565b60208152600061231560208301846125bb565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526054908201527f64697363726565743a2063616e6e6f74206d696e7420746f6b656e732066726f60408201527f6d206f726967696e616c206f722065787472612073657420756e74696c206d6960608201527333b930ba34b7b71034b99031b7b6b83632ba329760611b608082015260a00190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b60208082526028908201527f64697363726565743a2063616e6e6f74206d696e74206f75742d6f662d72616e60408201526733b2903a37b5b2b760c11b606082015260800190565b60008219821115612a5557612a55612aff565b500190565b600082612a6957612a69612b15565b500490565b6000816000190483118215151615612a8857612a88612aff565b500290565b600082821015612a9f57612a9f612aff565b500390565b60005b83811015612abf578181015183820152602001612aa7565b838111156113bd5750506000910152565b6000600019821415612ae457612ae4612aff565b5060010190565b600082612afa57612afa612b15565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052603160045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b031981168114610ac257600080fdfe77623278355a3239754948427661573530637a30694e4441774c4445774d4341304d4441734e444177494445774d4377304d444169494377623278355a3239754948427661573530637a30694e4441774c4451774d4341304d4441734d544177494445774d4377784d44416949436c6247787063484e6c49474e34505349794e54416949474e35505349794e54416949484a34505349784f54556949484a35505349784d6a55694943795a574e3049486739496a45314e534967655430694e545569494864705a48526f505349784f5441694947686c6157646f644430694d7a6b774969795a574e3049486739496a45784e534967655430694d5445314969423361575230614430694d6a63774969426f5a576c6e61485139496a49334d434967494377623278355a3239754948427661573530637a30694d7a67774c4445334e53417a4f4441734d7a4931494449314d4377304d4441674d5449774c444d794e5341784d6a41734d546331494449314d4377784d444169494377623278355a3239754948427661573530637a30694d6a4d774c4451774d4341794e7a41734e444177494449334d4377794e7a41674e4441774c4449334d4341304d4441734d6a4d77494449334d4377794d7a41674d6a63774c4445774d4341794d7a41734d5441774944497a4d4377794d7a41674d5441774c44497a4d4341784d4441734d6a63774944497a4d4377794e7a4169494377623278355a3239754948427661573530637a30694d5441774c44457a4e5341794e5441734e444177494451774d4377784d7a55694943775958526f49475139496b3077494449774d4342324c5449774d43426f4d6a4177494745784d7977784d534177494441734d5341774c4449774d4342684d5445734d544d674d4341774c4445674c5449774d43777749694230636d467563325a76636d3039496e4a76644746305a5367794d6a55734d6a41344c4445304f436b6763324e686247556f4d4334354d696b6949436c6247787063484e6c49474e34505349794e54416949474e35505349794e54416949484a34505349784d6a556949484a35505349784f545569494377623278355a3239754948427661573530637a30694d6a55774c4451774d43417a4e5441734d6a5577494449314d4377784d4441674d5455774c4449314d434967494377623278355a3239754948427661573530637a30694d6a55774c4445774d43417a4d7a67734d7a6378494445774e7977794d4451674d7a6b7a4c4449774e4341784e6a49734d7a637849694142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2f77623278355a3239754948427661573530637a30694d6a4d774c4451774d4341794e7a41734e444177494449334d4377794e7a41674e4441774c4449334d4341304d4441734d6a4d77494449334d4377794d7a41674d6a63774c4445774d4341794d7a41734d5441774944497a4d4377794d7a41674d5441774c44497a4d4341784d4441734d6a63774944497a4d4377794e7a4169494852795957357a5a6d397962543069636d39305958526c4b4451314c4449314d4377794e544170496977623278355a3239754948427661573530637a30694d5441774c4451774d4341304d4441734e444177494445774d4377784d444169494377623278355a3239754948427661573530637a30694d5441774c4451774d4341304d4441734d544177494445774d4377784d4441694943795a574e3049486739496a553149694235505349784e545569494864705a48526f5053497a4f5441694947686c6157646f644430694d546b7749696a61584a6a6247556759336739496a49314d43496759336b39496a49314d434967636a30694d545931496977623278355a3239754948427661573530637a30694e4441774c444d324e5341794e5441734d544177494445774d43777a4e6a55694943a26469706673582212202da75c66385d0ccc5f5962ea14db55e9ad846b9193345edb0c071c6bead1e95664736f6c63430008070033

Deployed Bytecode Sourcemap

30994:12587:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;42677:171;;;;;;:::i;:::-;;:::i;:::-;;;10825:14:1;;10818:22;10800:41;;10788:2;10773:18;42677:171:0;;;;;;;;13548:128;;;:::i;:::-;;;;;;;:::i;14586:221::-;;;;;;:::i;:::-;;:::i;:::-;;;-1:-1:-1;;;;;10123:32:1;;;10105:51;;10093:2;10078:18;14586:221:0;9959:203:1;14111:409:0;;;;;;:::i;:::-;;:::i;:::-;;33937:1031;;;;;;:::i;:::-;;:::i;:::-;;;-1:-1:-1;;;;;;11014:33:1;;;10996:52;;10984:2;10969:18;33937:1031:0;10852:202:1;25329:113:0;25417:10;:17;25329:113;;;22402:25:1;;;22390:2;22375:18;25329:113:0;22256:177:1;15472:339:0;;;;;;:::i;:::-;;:::i;24995:258::-;;;;;;:::i;:::-;;:::i;31519:128::-;;31598:42;31519:128;;35619:455;;;;;;:::i;:::-;;:::i;15882:187::-;;;;;;:::i;:::-;;:::i;36258:492::-;;;;;;:::i;:::-;;:::i;25519:235::-;;;;;;:::i;:::-;;:::i;37436:3084::-;;;;;;:::i;:::-;;:::i;31379:131::-;;31461:42;31379:131;;13242:239;;;;;;:::i;:::-;;:::i;37182:146::-;;;;;;:::i;:::-;;:::i;12972:208::-;;;;;;:::i;:::-;;:::i;13745:132::-;;;:::i;35063:451::-;;;;;;:::i;:::-;;:::i;14879:291::-;;;;;;:::i;:::-;;:::i;31656:38::-;;;;;16140:326;;;;;;:::i;:::-;;:::i;40634:911::-;;;;;;:::i;:::-;;:::i;41665:920::-;;;:::i;36878:211::-;;;;;;:::i;:::-;;:::i;15241:164::-;;;;;;:::i;:::-;;:::i;42677:171::-;42780:4;42804:36;42828:11;42804:23;:36::i;:::-;42797:43;42677:171;-1:-1:-1;;42677:171:0:o;13548:128::-;13644:23;;-1:-1:-1;;;;;;13661:5:0;6265:41:1;13644:23:0;;;6253:54:1;13604:13:0;;6323:11:1;;13644:23:0;;;;;;;;;;;;;13630:38;;13548:128;:::o;14586:221::-;14662:7;18065:16;;;;;;;;;;;-1:-1:-1;;;;;18065:16:0;14682:73;;;;-1:-1:-1;;;14682:73:0;;18609:2:1;14682:73:0;;;18591:21:1;18648:2;18628:18;;;18621:30;18687:34;18667:18;;;18660:62;-1:-1:-1;;;18738:18:1;;;18731:42;18790:19;;14682:73:0;;;;;;;;;-1:-1:-1;14775:24:0;;;;:15;:24;;;;;;-1:-1:-1;;;;;14775:24:0;;14586:221::o;14111:409::-;14194:13;14210:23;14225:7;14210:14;:23::i;:::-;14194:39;;14258:5;-1:-1:-1;;;;;14252:11:0;:2;-1:-1:-1;;;;;14252:11:0;;;14244:57;;;;-1:-1:-1;;;14244:57:0;;19925:2:1;14244:57:0;;;19907:21:1;19964:2;19944:18;;;19937:30;20003:34;19983:18;;;19976:62;-1:-1:-1;;;20054:18:1;;;20047:31;20095:19;;14244:57:0;19723:397:1;14244:57:0;14336:10;-1:-1:-1;;;;;14336:19:0;;;;:58;;;14359:35;14376:5;14383:10;14359:16;:35::i;:::-;14314:164;;;;-1:-1:-1;;;14314:164:0;;16214:2:1;14314:164:0;;;16196:21:1;16253:2;16233:18;;;16226:30;16292:34;16272:18;;;16265:62;16363:26;16343:18;;;16336:54;16407:19;;14314:164:0;16012:420:1;14314:164:0;14491:21;14500:2;14504:7;14491:8;:21::i;:::-;14183:337;14111:409;;:::o;33937:1031::-;34103:6;34159:13;34144:12;:28;34122:116;;;;-1:-1:-1;;;34122:116:0;;14983:2:1;34122:116:0;;;14965:21:1;15022:2;15002:18;;;14995:30;15061:34;15041:18;;;15034:62;-1:-1:-1;;;15112:18:1;;;15105:36;15158:19;;34122:116:0;14781:402:1;34122:116:0;34273:10;31461:42;34273:34;;:69;;-1:-1:-1;34311:10:0;31598:42;34311:31;34273:69;34251:180;;;;-1:-1:-1;;;34251:180:0;;13317:2:1;34251:180:0;;;13299:21:1;13356:2;13336:18;;;13329:30;13395:34;13375:18;;;13368:62;13466:31;13446:18;;;13439:59;13515:19;;34251:180:0;13115:425:1;34251:180:0;34448:10;31461:42;34448:34;34444:467;;;34535:5;34525:7;:15;34499:142;;;;-1:-1:-1;;;34499:142:0;;21575:2:1;34499:142:0;;;21557:21:1;21614:2;21594:18;;;21587:30;21653:34;21633:18;;;21626:62;21724:34;21704:18;;;21697:62;-1:-1:-1;;;21775:19:1;;;21768:32;21817:19;;34499:142:0;21373:469:1;34499:142:0;34656:24;34666:4;34672:7;34656:9;:24::i;:::-;34444:467;;;34749:5;34739:7;:15;34713:139;;;;-1:-1:-1;;;34713:139:0;;16639:2:1;34713:139:0;;;16621:21:1;16678:2;16658:18;;;16651:30;16717:34;16697:18;;;16690:62;16788:32;16768:18;;;16761:60;16838:19;;34713:139:0;16437:426:1;34713:139:0;34867:32;34877:4;34883:15;:7;34893:5;34883:15;:::i;:::-;34867:9;:32::i;:::-;-1:-1:-1;;;;33937:1031:0;;;;;;;:::o;15472:339::-;15669:39;15688:10;15700:7;15669:18;:39::i;:::-;15661:101;;;;-1:-1:-1;;;15661:101:0;;;;;;;:::i;:::-;15775:28;15785:4;15791:2;15795:7;15775:9;:28::i;24995:258::-;25094:7;25130:23;25147:5;25130:16;:23::i;:::-;25122:5;:31;25114:87;;;;-1:-1:-1;;;25114:87:0;;12129:2:1;25114:87:0;;;12111:21:1;12168:2;12148:18;;;12141:30;12207:34;12187:18;;;12180:62;-1:-1:-1;;;12258:18:1;;;12251:41;12309:19;;25114:87:0;11927:407:1;25114:87:0;-1:-1:-1;;;;;;25219:19:0;;;;;;;;:12;:19;;;;;;;;:26;;;;;;;;;24995:258::o;35619:455::-;35723:5;35713:7;:15;35691:105;;;;-1:-1:-1;;;35691:105:0;;;;;;;:::i;:::-;35823:5;35813:7;:15;35809:223;;;35887:13;35871:12;:29;;35845:175;;;;-1:-1:-1;;;35845:175:0;;;;;;;:::i;:::-;36044:22;36054:2;36058:7;36044:9;:22::i;:::-;35619:455;;:::o;15882:187::-;16022:39;16039:4;16045:2;16049:7;16022:39;;;;;;;;;;;;:16;:39::i;36258:492::-;36350:5;36340:7;:15;36318:105;;;;-1:-1:-1;;;36318:105:0;;11720:2:1;36318:105:0;;;11702:21:1;11759:2;11739:18;;;11732:30;11798:34;11778:18;;;11771:62;-1:-1:-1;;;11849:18:1;;;11842:38;11897:19;;36318:105:0;11518:404:1;36318:105:0;36522:22;36536:7;36522:13;:22::i;:::-;36517:199;;36587:39;36606:10;36618:7;36587:18;:39::i;:::-;36561:143;;;;-1:-1:-1;;;36561:143:0;;15803:2:1;36561:143:0;;;15785:21:1;15842:2;15822:18;;;15815:30;15881:34;15861:18;;;15854:62;-1:-1:-1;;;15932:18:1;;;15925:40;15982:19;;36561:143:0;15601:406:1;36561:143:0;36728:14;36734:7;36728:5;:14::i;:::-;36258:492;:::o;25519:235::-;25596:7;25632:30;25417:10;:17;;25329:113;25632:30;25624:5;:38;25616:95;;;;-1:-1:-1;;;25616:95:0;;21162:2:1;25616:95:0;;;21144:21:1;21201:2;21181:18;;;21174:30;21240:34;21220:18;;;21213:62;-1:-1:-1;;;21291:18:1;;;21284:42;21343:19;;25616:95:0;20960:408:1;25616:95:0;25729:10;25740:5;25729:17;;;;;;;;:::i;:::-;;;;;;;;;25722:24;;25519:235;;;:::o;37436:3084::-;37514:13;37558:5;37548:7;:15;37540:76;;;;-1:-1:-1;;;37540:76:0;;20327:2:1;37540:76:0;;;20309:21:1;20366:2;20346:18;;;20339:30;20405:34;20385:18;;;20378:62;-1:-1:-1;;;20456:18:1;;;20449:46;20512:19;;37540:76:0;20125:412:1;37540:76:0;37698:275;;;;;;;;-1:-1:-1;;;37698:275:0;;-1:-1:-1;;;37698:275:0;;;;-1:-1:-1;;;37698:275:0;;;;-1:-1:-1;;;37698:275:0;;;;-1:-1:-1;;;37698:275:0;;;;-1:-1:-1;;;37698:275:0;;;;-1:-1:-1;;;37698:275:0;;;;-1:-1:-1;;;37698:275:0;;;;-1:-1:-1;;;37698:275:0;;;;38056:1730;;;;;;;;;;;;;;;37698:275;;:19;;38056:1730;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;39865:266;;;;;;;;-1:-1:-1;;;39865:266:0;;-1:-1:-1;;;39865:266:0;;;;-1:-1:-1;;;39865:266:0;;;;;;;-1:-1:-1;;;39865:266:0;;;;-1:-1:-1;;;39865:266:0;;;;-1:-1:-1;;;39865:266:0;;;;-1:-1:-1;;;39865:266:0;;;;-1:-1:-1;;;39865:266:0;;;;-1:-1:-1;;;39865:266:0;;;;38056:1730;;-1:-1:-1;39865:19:0;40257:1;40241:12;40251:2;40241:7;:12;:::i;:::-;40240:18;;;;:::i;:::-;40226:32;-1:-1:-1;40269:11:0;40283:12;40293:2;40283:7;:12;:::i;:::-;40269:26;-1:-1:-1;40306:11:0;40352:1;40336:11;40346:1;40336:7;:11;:::i;:::-;40321;40331:1;40321:7;:11;:::i;:::-;:27;;;;:::i;:::-;40320:33;;;;:::i;:::-;40306:47;;40427:2;40431;40435;40439;40443;40447;40451;40454:3;40451:7;;;;;;;:::i;:::-;;;;;-1:-1:-1;;;40464:2:0;40467:3;40464:7;;;;;;;:::i;:::-;;;;;-1:-1:-1;;;40477:2:0;40480:3;40477:7;;;;;;;:::i;:::-;;;;;-1:-1:-1;;;40392:110:0;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;40364:148;;;;;;;;37436:3084;;;:::o;13242:239::-;13314:7;13350:16;;;;;;;;;;;-1:-1:-1;;;;;13350:16:0;13385:19;13377:73;;;;-1:-1:-1;;;13377:73:0;;17481:2:1;13377:73:0;;;17463:21:1;17520:2;17500:18;;;17493:30;17559:34;17539:18;;;17532:62;-1:-1:-1;;;17610:18:1;;;17603:39;17659:19;;13377:73:0;17279:405:1;37182:146:0;37252:4;37308:12;37276:29;37297:7;37276:20;:29::i;:::-;:44;;37182:146;-1:-1:-1;;37182:146:0:o;12972:208::-;13044:7;-1:-1:-1;;;;;13072:19:0;;13064:74;;;;-1:-1:-1;;;13064:74:0;;17070:2:1;13064:74:0;;;17052:21:1;17109:2;17089:18;;;17082:30;17148:34;17128:18;;;17121:62;-1:-1:-1;;;17199:18:1;;;17192:40;17249:19;;13064:74:0;16868:406:1;13064:74:0;-1:-1:-1;;;;;;13156:16:0;;;;;:9;:16;;;;;;;12972:208::o;13745:132::-;13843:25;;-1:-1:-1;;;;;;13860:7:0;6265:41:1;13843:25:0;;;6253:54:1;13803:13:0;;6323:11:1;;13843:25:0;6126:214:1;35063:451:0;35155:5;35145:7;:15;35123:105;;;;-1:-1:-1;;;35123:105:0;;;;;;;:::i;:::-;35255:5;35245:7;:15;35241:223;;;35319:13;35303:12;:29;;35277:175;;;;-1:-1:-1;;;35277:175:0;;;;;;;:::i;:::-;35476:30;35486:10;35498:7;35476:9;:30::i;14879:291::-;-1:-1:-1;;;;;14984:22:0;;14996:10;14984:22;;14976:60;;;;-1:-1:-1;;;14976:60:0;;14629:2:1;14976:60:0;;;14611:21:1;14668:2;14648:18;;;14641:30;14707:27;14687:18;;;14680:55;14752:18;;14976:60:0;14427:349:1;14976:60:0;15068:10;15049:30;;;;:18;:30;;;;;;;;-1:-1:-1;;;;;15049:40:0;;;;;;;;;;;;:51;;-1:-1:-1;;15049:51:0;;;;;;;;;;15116:46;;10800:41:1;;;15049:40:0;;15068:10;15116:46;;10773:18:1;15116:46:0;;;;;;;14879:291;;:::o;16140:326::-;16315:39;16334:10;16346:7;16315:18;:39::i;:::-;16307:101;;;;-1:-1:-1;;;16307:101:0;;;;;;;:::i;:::-;16419:39;16433:4;16439:2;16443:7;16452:5;16419:13;:39::i;:::-;16140:326;;;;:::o;40634:911::-;40709:13;40735:18;40756:698;40929:18;40939:7;40929:9;:18::i;:::-;41333:22;41347:7;41333:13;:22::i;:::-;40837:572;;;;;;;;;:::i;:::-;;;;;;;;;;;;;40756:13;:698::i;:::-;40735:719;;41531:4;41481:55;;;;;;;;:::i;:::-;;;;;;;;;;;;;41467:70;;;40634:911;;;:::o;41665:920::-;41709:13;41848:703;41953:529;;;;;;9148:66:1;9136:79;;9245:66;9240:2;9231:12;;9224:88;9342:34;9337:2;9328:12;;9321:56;-1:-1:-1;;;9402:2:1;9393:12;;9386:35;9452:34;9446:3;9437:13;;9430:57;9518:34;9512:3;9503:13;;9496:57;-1:-1:-1;;;9578:3:1;9569:13;;9562:30;9623:34;9617:3;9608:13;;9601:57;9689:34;9683:3;9674:13;;9667:57;-1:-1:-1;;;9749:3:1;9740:13;;9733:37;9801:34;9795:3;9786:13;;9779:57;-1:-1:-1;;;9861:3:1;9852:13;;9845:74;9944:3;9935:13;;8530:1424;41848:703:0;41763:803;;;;;;;;:::i;36878:211::-;36955:7;36993:5;36983:7;:15;36975:56;;;;-1:-1:-1;;;36975:56:0;;17891:2:1;36975:56:0;;;17873:21:1;17930:2;17910:18;;;17903:30;17969;17949:18;;;17942:58;18017:18;;36975:56:0;17689:352:1;36975:56:0;-1:-1:-1;37051:30:0;;;;:21;:30;;;;;;;36878:211::o;15241:164::-;-1:-1:-1;;;;;15362:25:0;;;15338:4;15362:25;;;:18;:25;;;;;;;;:35;;;;;;;;;;;;;;;15241:164::o;24687:224::-;24789:4;-1:-1:-1;;;;;;24813:50:0;;-1:-1:-1;;;24813:50:0;;:90;;;24867:36;24891:11;24867:23;:36::i;21564:174::-;21639:24;;;;:15;:24;;;;;:29;;-1:-1:-1;;;;;;21639:29:0;-1:-1:-1;;;;;21639:29:0;;;;;;;;:24;;21693:23;21639:24;21693:14;:23::i;:::-;-1:-1:-1;;;;;21684:46:0;;;;;;;;;;;21564:174;;:::o;18960:264::-;19036:18;19042:2;19046:7;19036:5;:18::i;:::-;19087:51;19118:1;19122:2;19126:7;19087:51;;;;;;;;;;;;:22;:51::i;:::-;19065:151;;;;-1:-1:-1;;;19065:151:0;;;;;;;:::i;18270:348::-;18363:4;18065:16;;;;;;;;;;;-1:-1:-1;;;;;18065:16:0;18380:73;;;;-1:-1:-1;;;18380:73:0;;15390:2:1;18380:73:0;;;15372:21:1;15429:2;15409:18;;;15402:30;15468:34;15448:18;;;15441:62;-1:-1:-1;;;15519:18:1;;;15512:42;15571:19;;18380:73:0;15188:408:1;18380:73:0;18464:13;18480:23;18495:7;18480:14;:23::i;:::-;18464:39;;18533:5;-1:-1:-1;;;;;18522:16:0;:7;-1:-1:-1;;;;;18522:16:0;;:51;;;;18566:7;-1:-1:-1;;;;;18542:31:0;:20;18554:7;18542:11;:20::i;:::-;-1:-1:-1;;;;;18542:31:0;;18522:51;:87;;;;18577:32;18594:5;18601:7;18577:16;:32::i;:::-;18514:96;18270:348;-1:-1:-1;;;;18270:348:0:o;20868:578::-;21027:4;-1:-1:-1;;;;;21000:31:0;:23;21015:7;21000:14;:23::i;:::-;-1:-1:-1;;;;;21000:31:0;;20992:85;;;;-1:-1:-1;;;20992:85:0;;19022:2:1;20992:85:0;;;19004:21:1;19061:2;19041:18;;;19034:30;19100:34;19080:18;;;19073:62;-1:-1:-1;;;19151:18:1;;;19144:39;19200:19;;20992:85:0;18820:405:1;20992:85:0;-1:-1:-1;;;;;21096:16:0;;21088:65;;;;-1:-1:-1;;;21088:65:0;;14224:2:1;21088:65:0;;;14206:21:1;14263:2;14243:18;;;14236:30;14302:34;14282:18;;;14275:62;-1:-1:-1;;;14353:18:1;;;14346:34;14397:19;;21088:65:0;14022:400:1;21088:65:0;21166:39;21187:4;21193:2;21197:7;21166:20;:39::i;:::-;21270:29;21287:1;21291:7;21270:8;:29::i;:::-;-1:-1:-1;;;;;21312:15:0;;;;;;21331:1;21312:15;;;;;;;:20;;21331:1;;21312:15;;:20;;21331:1;;21312:20;:::i;:::-;;;;-1:-1:-1;;;;;;;21343:13:0;;;;;;21360:1;21343:13;;;;;;;:18;;21360:1;;21343:13;;:18;;21360:1;;21343:18;:::i;:::-;;;;-1:-1:-1;;21372:7:0;:16;;;;;;;;;;;:21;;-1:-1:-1;;;;;;21372:21:0;-1:-1:-1;;;;;21372:21:0;;;;;;;;;21411:27;;21372:16;;21411:27;;;;;;;20868:578;;;:::o;20171:360::-;20231:13;20247:23;20262:7;20247:14;:23::i;:::-;20231:39;;20283:48;20304:5;20319:1;20323:7;20283:20;:48::i;:::-;20372:29;20389:1;20393:7;20372:8;:29::i;:::-;-1:-1:-1;;;;;20414:16:0;;;;;;20434:1;20414:16;;;;;;;:21;;20434:1;;20414:16;;:21;;20434:1;;20414:21;:::i;:::-;;;;-1:-1:-1;;20453:7:0;:16;;;;;;;;;;;20446:23;;-1:-1:-1;;;;;;20446:23:0;;;20487:36;20461:7;;20453;-1:-1:-1;;;;;20487:36:0;;;;;20453:7;;20487:36;20220:311;20171:360;:::o;17348:315::-;17505:28;17515:4;17521:2;17525:7;17505:9;:28::i;:::-;17552:48;17575:4;17581:2;17585:7;17594:5;17552:22;:48::i;:::-;17544:111;;;;-1:-1:-1;;;17544:111:0;;;;;;;:::i;42856:722::-;42913:13;43132:10;43128:53;;-1:-1:-1;;43159:10:0;;;;;;;;;;;;-1:-1:-1;;;43159:10:0;;;;;42856:722::o;43128:53::-;43206:5;43191:12;43247:78;43254:9;;43247:78;;43280:8;;;;:::i;:::-;;-1:-1:-1;43303:10:0;;-1:-1:-1;43311:2:0;43303:10;;:::i;:::-;;;43247:78;;;43335:19;43367:6;43357:17;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;43357:17:0;;43335:39;;43385:154;43392:10;;43385:154;;43419:11;43429:1;43419:11;;:::i;:::-;;-1:-1:-1;43488:10:0;43496:2;43488:5;:10;:::i;:::-;43475:24;;:2;:24;:::i;:::-;43462:39;;43445:6;43452;43445:14;;;;;;;;:::i;:::-;;;;:56;-1:-1:-1;;;;;43445:56:0;;;;;;;;-1:-1:-1;43516:11:0;43525:2;43516:11;;:::i;:::-;;;43385:154;;9829:1607;9927:11;;9887:13;;9953:8;9949:23;;-1:-1:-1;;9963:9:0;;;;;;;;;-1:-1:-1;9963:9:0;;;9829:1607;-1:-1:-1;9829:1607:0:o;9949:23::-;10024:18;10062:1;10051:7;:3;10057:1;10051:7;:::i;:::-;10050:13;;;;:::i;:::-;10045:19;;:1;:19;:::i;:::-;10024:40;-1:-1:-1;10122:19:0;10154:15;10024:40;10167:2;10154:15;:::i;:::-;10144:26;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;10144:26:0;;10122:48;;10183:18;10204:5;;;;;;;;;;;;;;;;;10183:26;;10273:1;10266:5;10262:13;10318:2;10310:6;10306:15;10369:1;10337:777;10392:3;10389:1;10386:10;10337:777;;;10447:1;10490:12;;;;;10484:19;10585:4;10573:2;10569:14;;;;;10551:40;;10545:47;10694:2;10690:14;;;10686:25;;10672:40;;10666:47;10823:1;10819:13;;;10815:24;;10801:39;;10795:46;10943:16;;;;10929:31;;10923:38;10621:1;10617:11;;;10715:4;10662:58;;;10653:68;10746:11;;10791:57;;;10782:67;;;;10874:11;;10919:49;;10910:59;10998:3;10994:13;11027:22;;11097:1;11082:17;;;;10440:9;10337:777;;;10341:44;11146:1;11141:3;11137:11;11167:1;11162:84;;;;11265:1;11260:82;;;;11130:212;;11162:84;-1:-1:-1;;;;;11195:17:0;;11188:43;11162:84;;11260:82;-1:-1:-1;;;;;11293:17:0;;11286:41;11130:212;-1:-1:-1;;;11358:26:0;;;11365:6;9829:1607;-1:-1:-1;;;;9829:1607:0:o;12603:305::-;12705:4;-1:-1:-1;;;;;;12742:40:0;;-1:-1:-1;;;12742:40:0;;:105;;-1:-1:-1;;;;;;;12799:48:0;;-1:-1:-1;;;12799:48:0;12742:105;:158;;;-1:-1:-1;;;;;;;;;;9425:40:0;;;12864:36;9316:157;19560:382;-1:-1:-1;;;;;19640:16:0;;19632:61;;;;-1:-1:-1;;;19632:61:0;;18248:2:1;19632:61:0;;;18230:21:1;;;18267:18;;;18260:30;18326:34;18306:18;;;18299:62;18378:18;;19632:61:0;18046:356:1;19632:61:0;18041:4;18065:16;;;;;;;;;;;-1:-1:-1;;;;;18065:16:0;:30;19704:58;;;;-1:-1:-1;;;19704:58:0;;12960:2:1;19704:58:0;;;12942:21:1;12999:2;12979:18;;;12972:30;13038;13018:18;;;13011:58;13086:18;;19704:58:0;12758:352:1;19704:58:0;19775:45;19804:1;19808:2;19812:7;19775:20;:45::i;:::-;-1:-1:-1;;;;;19833:13:0;;;;;;19850:1;19833:13;;;;;;;:18;;19850:1;;19833:13;;:18;;19850:1;;19833:18;:::i;:::-;;;;-1:-1:-1;;19862:7:0;:16;;;;;;;;;;;:21;;-1:-1:-1;;;;;;19862:21:0;-1:-1:-1;;;;;19862:21:0;;;;;;;;19901:33;;19862:16;;:7;19901:33;;19862:7;;19901:33;19560:382;;:::o;22303:863::-;22458:4;22517:15;;22548:8;;22544:615;;22577:70;;-1:-1:-1;;;22577:70:0;;-1:-1:-1;;;;;22577:36:0;;;;;:70;;22614:10;;22626:4;;22632:7;;22641:5;;22577:70;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;22577:70:0;;;;;;;;-1:-1:-1;;22577:70:0;;;;;;;;;;;;:::i;:::-;;;22573:531;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;22821:13:0;;22817:272;;22864:60;;-1:-1:-1;;;22864:60:0;;;;;;;:::i;22817:272::-;23039:6;23033:13;23024:6;23020:2;23016:15;23009:38;22573:531;-1:-1:-1;;;;;;22698:55:0;-1:-1:-1;;;22698:55:0;;-1:-1:-1;22691:62:0;;-1:-1:-1;22691:62:0;22544:615;23143:4;23136:11;;;;;32976:789;33138:45;33165:4;33171:2;33175:7;33138:26;:45::i;:::-;-1:-1:-1;;;;;33283:18:0;;33279:282;;33376:9;33359:27;;;;:16;:27;;;;;;33344:12;:42;33318:172;;;;-1:-1:-1;;;33318:172:0;;13747:2:1;33318:172:0;;;13729:21:1;13786:2;13766:18;;;13759:30;;;13825:34;13805:18;;;13798:62;13896:34;13876:18;;;13869:62;-1:-1:-1;;;13947:19:1;;;13940:35;13992:19;;33318:172:0;13545:472:1;33318:172:0;33524:9;33507:27;;;;:16;:27;;;;;33537:12;33507:42;;33279:282;-1:-1:-1;;;;;33657:16:0;;;33653:105;;33723:23;:12;33738:8;33723:23;:::i;:::-;33690:30;;;;:21;:30;;;;;:56;32976:789;;;:::o;26367:589::-;-1:-1:-1;;;;;26573:18:0;;26569:187;;26608:40;26640:7;27783:10;:17;;27756:24;;;;:15;:24;;;;;:44;;;27811:24;;;;;;;;;;;;27679:164;26608:40;26569:187;;;26678:2;-1:-1:-1;;;;;26670:10:0;:4;-1:-1:-1;;;;;26670:10:0;;26666:90;;26697:47;26730:4;26736:7;26697:32;:47::i;:::-;-1:-1:-1;;;;;26770:16:0;;26766:183;;26803:45;26840:7;26803:36;:45::i;26766:183::-;26876:4;-1:-1:-1;;;;;26870:10:0;:2;-1:-1:-1;;;;;26870:10:0;;26866:83;;26897:40;26925:2;26929:7;26897:27;:40::i;28470:988::-;28736:22;28786:1;28761:22;28778:4;28761:16;:22::i;:::-;:26;;;;:::i;:::-;28798:18;28819:26;;;:17;:26;;;;;;28736:51;;-1:-1:-1;28952:28:0;;;28948:328;;-1:-1:-1;;;;;29019:18:0;;28997:19;29019:18;;;:12;:18;;;;;;;;:34;;;;;;;;;29070:30;;;;;;:44;;;29187:30;;:17;:30;;;;;:43;;;28948:328;-1:-1:-1;29372:26:0;;;;:17;:26;;;;;;;;29365:33;;;-1:-1:-1;;;;;29416:18:0;;;;;:12;:18;;;;;:34;;;;;;;29409:41;28470:988::o;29753:1079::-;30031:10;:17;30006:22;;30031:21;;30051:1;;30031:21;:::i;:::-;30063:18;30084:24;;;:15;:24;;;;;;30457:10;:26;;30006:46;;-1:-1:-1;30084:24:0;;30006:46;;30457:26;;;;;;:::i;:::-;;;;;;;;;30435:48;;30521:11;30496:10;30507;30496:22;;;;;;;;:::i;:::-;;;;;;;;;;;;:36;;;;30601:28;;;:15;:28;;;;;;;:41;;;30773:24;;;;;30766:31;30808:10;:16;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;29824:1008;;;29753:1079;:::o;27257:221::-;27342:14;27359:20;27376:2;27359:16;:20::i;:::-;-1:-1:-1;;;;;27390:16:0;;;;;;;:12;:16;;;;;;;;:24;;;;;;;;:34;;;27435:26;;;:17;:26;;;;;;:35;;;;-1:-1:-1;27257:221:0:o;14:173:1:-;82:20;;-1:-1:-1;;;;;131:31:1;;121:42;;111:70;;177:1;174;167:12;111:70;14:173;;;:::o;192:186::-;251:6;304:2;292:9;283:7;279:23;275:32;272:52;;;320:1;317;310:12;272:52;343:29;362:9;343:29;:::i;:::-;333:39;192:186;-1:-1:-1;;;192:186:1:o;383:260::-;451:6;459;512:2;500:9;491:7;487:23;483:32;480:52;;;528:1;525;518:12;480:52;551:29;570:9;551:29;:::i;:::-;541:39;;599:38;633:2;622:9;618:18;599:38;:::i;:::-;589:48;;383:260;;;;;:::o;648:328::-;725:6;733;741;794:2;782:9;773:7;769:23;765:32;762:52;;;810:1;807;800:12;762:52;833:29;852:9;833:29;:::i;:::-;823:39;;881:38;915:2;904:9;900:18;881:38;:::i;:::-;871:48;;966:2;955:9;951:18;938:32;928:42;;648:328;;;;;:::o;981:808::-;1078:6;1086;1094;1102;1110;1163:3;1151:9;1142:7;1138:23;1134:33;1131:53;;;1180:1;1177;1170:12;1131:53;1203:29;1222:9;1203:29;:::i;:::-;1193:39;;1251:38;1285:2;1274:9;1270:18;1251:38;:::i;:::-;1241:48;;1336:2;1325:9;1321:18;1308:32;1298:42;;1391:2;1380:9;1376:18;1363:32;1414:18;1455:2;1447:6;1444:14;1441:34;;;1471:1;1468;1461:12;1441:34;1509:6;1498:9;1494:22;1484:32;;1554:7;1547:4;1543:2;1539:13;1535:27;1525:55;;1576:1;1573;1566:12;1525:55;1616:2;1603:16;1642:2;1634:6;1631:14;1628:34;;;1658:1;1655;1648:12;1628:34;1703:7;1698:2;1689:6;1685:2;1681:15;1677:24;1674:37;1671:57;;;1724:1;1721;1714:12;1671:57;981:808;;;;-1:-1:-1;981:808:1;;-1:-1:-1;1755:2:1;1747:11;;1777:6;981:808;-1:-1:-1;;;981:808:1:o;1794:1138::-;1889:6;1897;1905;1913;1966:3;1954:9;1945:7;1941:23;1937:33;1934:53;;;1983:1;1980;1973:12;1934:53;2006:29;2025:9;2006:29;:::i;:::-;1996:39;;2054:38;2088:2;2077:9;2073:18;2054:38;:::i;:::-;2044:48;;2139:2;2128:9;2124:18;2111:32;2101:42;;2194:2;2183:9;2179:18;2166:32;2217:18;2258:2;2250:6;2247:14;2244:34;;;2274:1;2271;2264:12;2244:34;2312:6;2301:9;2297:22;2287:32;;2357:7;2350:4;2346:2;2342:13;2338:27;2328:55;;2379:1;2376;2369:12;2328:55;2415:2;2402:16;2437:2;2433;2430:10;2427:36;;;2443:18;;:::i;:::-;2518:2;2512:9;2486:2;2572:13;;-1:-1:-1;;2568:22:1;;;2592:2;2564:31;2560:40;2548:53;;;2616:18;;;2636:22;;;2613:46;2610:72;;;2662:18;;:::i;:::-;2702:10;2698:2;2691:22;2737:2;2729:6;2722:18;2777:7;2772:2;2767;2763;2759:11;2755:20;2752:33;2749:53;;;2798:1;2795;2788:12;2749:53;2854:2;2849;2845;2841:11;2836:2;2828:6;2824:15;2811:46;2899:1;2894:2;2889;2881:6;2877:15;2873:24;2866:35;2920:6;2910:16;;;;;;;1794:1138;;;;;;;:::o;2937:347::-;3002:6;3010;3063:2;3051:9;3042:7;3038:23;3034:32;3031:52;;;3079:1;3076;3069:12;3031:52;3102:29;3121:9;3102:29;:::i;:::-;3092:39;;3181:2;3170:9;3166:18;3153:32;3228:5;3221:13;3214:21;3207:5;3204:32;3194:60;;3250:1;3247;3240:12;3194:60;3273:5;3263:15;;;2937:347;;;;;:::o;3289:254::-;3357:6;3365;3418:2;3406:9;3397:7;3393:23;3389:32;3386:52;;;3434:1;3431;3424:12;3386:52;3457:29;3476:9;3457:29;:::i;:::-;3447:39;3533:2;3518:18;;;;3505:32;;-1:-1:-1;;;3289:254:1:o;3548:245::-;3606:6;3659:2;3647:9;3638:7;3634:23;3630:32;3627:52;;;3675:1;3672;3665:12;3627:52;3714:9;3701:23;3733:30;3757:5;3733:30;:::i;3798:249::-;3867:6;3920:2;3908:9;3899:7;3895:23;3891:32;3888:52;;;3936:1;3933;3926:12;3888:52;3968:9;3962:16;3987:30;4011:5;3987:30;:::i;4052:180::-;4111:6;4164:2;4152:9;4143:7;4139:23;4135:32;4132:52;;;4180:1;4177;4170:12;4132:52;-1:-1:-1;4203:23:1;;4052:180;-1:-1:-1;4052:180:1:o;4366:257::-;4407:3;4445:5;4439:12;4472:6;4467:3;4460:19;4488:63;4544:6;4537:4;4532:3;4528:14;4521:4;4514:5;4510:16;4488:63;:::i;:::-;4605:2;4584:15;-1:-1:-1;;4580:29:1;4571:39;;;;4612:4;4567:50;;4366:257;-1:-1:-1;;4366:257:1:o;4628:185::-;4670:3;4708:5;4702:12;4723:52;4768:6;4763:3;4756:4;4749:5;4745:16;4723:52;:::i;:::-;4791:16;;;;;4628:185;-1:-1:-1;;4628:185:1:o;4948:1173::-;5413:6;5408:3;5401:19;5450:6;5445:2;5440:3;5436:12;5429:28;5487:6;5482:2;5477:3;5473:12;5466:28;5524:6;5519:2;5514:3;5510:12;5503:28;5562:6;5556:3;5551;5547:13;5540:29;5600:6;5594:3;5589;5585:13;5578:29;5659:20;5654:3;5650:30;5642:6;5638:43;5632:3;5627;5623:13;5616:66;5734:10;5729:3;5725:20;5717:6;5713:33;5707:3;5702;5698:13;5691:56;5383:3;5776:6;5770:13;5792:61;5846:6;5840:3;5835;5831:13;5826:2;5818:6;5814:15;5792:61;:::i;:::-;-1:-1:-1;;;;;;5918:45:1;;5912:3;5872:16;;;5904:12;;;5897:67;-1:-1:-1;;;;;;5994:42:1;;5988:3;5980:12;;5973:64;6046:41;6082:3;6074:12;;6065:7;-1:-1:-1;;4303:51:1;4291:64;;4237:124;6046:41;6111:3;6103:12;;4948:1173;-1:-1:-1;;;;;;;;;;;;;;4948:1173:1:o;6345:1727::-;-1:-1:-1;;;7148:64:1;;7235:13;;7130:3;;7257:62;7235:13;7307:2;7298:12;;7291:4;7279:17;;7257:62;:::i;:::-;7383:66;7378:2;7338:16;;;7370:11;;;7363:87;7479:34;7474:2;7466:11;;7459:55;-1:-1:-1;;;7538:2:1;7530:11;;7523:26;7578:34;7573:2;7565:11;;7558:55;7643:34;7637:3;7629:12;;7622:56;-1:-1:-1;;;7702:3:1;7694:12;;7687:25;7742:34;7736:3;7728:12;;7721:56;7807:34;7801:3;7793:12;;7786:56;7872:34;7866:3;7858:12;;7851:56;-1:-1:-1;;;7931:3:1;7923:12;;7916:64;7996:70;8026:39;8060:3;8052:12;;8044:6;8026:39;:::i;:::-;-1:-1:-1;;;4883:27:1;;4935:1;4926:11;;4818:125;7996:70;7989:77;6345:1727;-1:-1:-1;;;;;6345:1727:1:o;8077:448::-;8339:31;8334:3;8327:44;8309:3;8400:6;8394:13;8416:62;8471:6;8466:2;8461:3;8457:12;8450:4;8442:6;8438:17;8416:62;:::i;:::-;8498:16;;;;8516:2;8494:25;;8077:448;-1:-1:-1;;8077:448:1:o;10167:488::-;-1:-1:-1;;;;;10436:15:1;;;10418:34;;10488:15;;10483:2;10468:18;;10461:43;10535:2;10520:18;;10513:34;;;10583:3;10578:2;10563:18;;10556:31;;;10361:4;;10604:45;;10629:19;;10621:6;10604:45;:::i;:::-;10596:53;10167:488;-1:-1:-1;;;;;;10167:488:1:o;11294:219::-;11443:2;11432:9;11425:21;11406:4;11463:44;11503:2;11492:9;11488:18;11480:6;11463:44;:::i;12339:414::-;12541:2;12523:21;;;12580:2;12560:18;;;12553:30;12619:34;12614:2;12599:18;;12592:62;-1:-1:-1;;;12685:2:1;12670:18;;12663:48;12743:3;12728:19;;12339:414::o;19230:488::-;19432:2;19414:21;;;19471:2;19451:18;;;19444:30;19510:34;19505:2;19490:18;;19483:62;19581:34;19576:2;19561:18;;19554:62;-1:-1:-1;;;19647:3:1;19632:19;;19625:51;19708:3;19693:19;;19230:488::o;20542:413::-;20744:2;20726:21;;;20783:2;20763:18;;;20756:30;20822:34;20817:2;20802:18;;20795:62;-1:-1:-1;;;20888:2:1;20873:18;;20866:47;20945:3;20930:19;;20542:413::o;21847:404::-;22049:2;22031:21;;;22088:2;22068:18;;;22061:30;22127:34;22122:2;22107:18;;22100:62;-1:-1:-1;;;22193:2:1;22178:18;;22171:38;22241:3;22226:19;;21847:404::o;22438:128::-;22478:3;22509:1;22505:6;22502:1;22499:13;22496:39;;;22515:18;;:::i;:::-;-1:-1:-1;22551:9:1;;22438:128::o;22571:120::-;22611:1;22637;22627:35;;22642:18;;:::i;:::-;-1:-1:-1;22676:9:1;;22571:120::o;22696:168::-;22736:7;22802:1;22798;22794:6;22790:14;22787:1;22784:21;22779:1;22772:9;22765:17;22761:45;22758:71;;;22809:18;;:::i;:::-;-1:-1:-1;22849:9:1;;22696:168::o;22869:125::-;22909:4;22937:1;22934;22931:8;22928:34;;;22942:18;;:::i;:::-;-1:-1:-1;22979:9:1;;22869:125::o;22999:258::-;23071:1;23081:113;23095:6;23092:1;23089:13;23081:113;;;23171:11;;;23165:18;23152:11;;;23145:39;23117:2;23110:10;23081:113;;;23212:6;23209:1;23206:13;23203:48;;;-1:-1:-1;;23247:1:1;23229:16;;23222:27;22999:258::o;23262:135::-;23301:3;-1:-1:-1;;23322:17:1;;23319:43;;;23342:18;;:::i;:::-;-1:-1:-1;23389:1:1;23378:13;;23262:135::o;23402:112::-;23434:1;23460;23450:35;;23465:18;;:::i;:::-;-1:-1:-1;23499:9:1;;23402:112::o;23519:127::-;23580:10;23575:3;23571:20;23568:1;23561:31;23611:4;23608:1;23601:15;23635:4;23632:1;23625:15;23651:127;23712:10;23707:3;23703:20;23700:1;23693:31;23743:4;23740:1;23733:15;23767:4;23764:1;23757:15;23783:127;23844:10;23839:3;23835:20;23832:1;23825:31;23875:4;23872:1;23865:15;23899:4;23896:1;23889:15;23915:127;23976:10;23971:3;23967:20;23964:1;23957:31;24007:4;24004:1;23997:15;24031:4;24028:1;24021:15;24047:127;24108:10;24103:3;24099:20;24096:1;24089:31;24139:4;24136:1;24129:15;24163:4;24160:1;24153:15;24179:131;-1:-1:-1;;;;;;24253:32:1;;24243:43;;24233:71;;24300:1;24297;24290:12

Swarm Source

ipfs://2da75c66385d0ccc5f5962ea14db55e9ad846b9193345edb0c071c6bead1e956
Loading...
Loading
Loading...
Loading
[ Download: CSV Export  ]

A token is a representation of an on-chain or off-chain asset. The token page shows information such as price, total supply, holders, transfers and social links. Learn more about this page in our Knowledge Base.