ETH Price: $3,503.34 (+4.44%)

Token

DRGZ (DRGZ)
 

Overview

Max Total Supply

7,777 DRGZ

Holders

4,570

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A

Other Info

Filtered by Token Holder
t4eng.eth
Balance
3 DRGZ
0xc9636b935fb6b3ce48654a0009755d58f473c064
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:
DRGZ

Compiler Version
v0.8.7+commit.e28d00a7

Optimization Enabled:
No with 200 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2022-07-14
*/

// SPDX-License-Identifier: MIT

// Created by: DRGZ Dev Team aka CryptoMage

// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)

pragma solidity ^0.8.0;

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

// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol)

pragma solidity ^0.8.0;


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

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

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

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

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

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

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

// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)

pragma solidity ^0.8.0;



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



// ERC721A Contracts v3.3.0
// Creator: Chiru Labs

pragma solidity ^0.8.4;



/**
 * @dev Interface of an ERC721A compliant contract.
 */
interface IERC721A is IERC721, IERC721Metadata {
    /**
     * The caller must own the token or be an approved operator.
     */
    error ApprovalCallerNotOwnerNorApproved();

    /**
     * The token does not exist.
     */
    error ApprovalQueryForNonexistentToken();

    /**
     * The caller cannot approve to their own address.
     */
    error ApproveToCaller();

    /**
     * The caller cannot approve to the current owner.
     */
    error ApprovalToCurrentOwner();

    /**
     * Cannot query the balance for the zero address.
     */
    error BalanceQueryForZeroAddress();

    /**
     * Cannot mint to the zero address.
     */
    error MintToZeroAddress();

    /**
     * The quantity of tokens minted must be more than zero.
     */
    error MintZeroQuantity();

    /**
     * The token does not exist.
     */
    error OwnerQueryForNonexistentToken();

    /**
     * The caller must own the token or be an approved operator.
     */
    error TransferCallerNotOwnerNorApproved();

    /**
     * The token must be owned by `from`.
     */
    error TransferFromIncorrectOwner();

    /**
     * Cannot safely transfer to a contract that does not implement the ERC721Receiver interface.
     */
    error TransferToNonERC721ReceiverImplementer();

    /**
     * Cannot transfer to the zero address.
     */
    error TransferToZeroAddress();

    /**
     * The token does not exist.
     */
    error URIQueryForNonexistentToken();

    // Compiler will pack this into a single 256bit word.
    struct TokenOwnership {
        // The address of the owner.
        address addr;
        // Keeps track of the start time of ownership with minimal overhead for tokenomics.
        uint64 startTimestamp;
        // Whether the token has been burned.
        bool burned;
    }

    // Compiler will pack this into a single 256bit word.
    struct AddressData {
        // Realistically, 2**64-1 is more than enough.
        uint64 balance;
        // Keeps track of mint count with minimal overhead for tokenomics.
        uint64 numberMinted;
        // Keeps track of burn count with minimal overhead for tokenomics.
        uint64 numberBurned;
        // For miscellaneous variable(s) pertaining to the address
        // (e.g. number of whitelist mint slots used).
        // If there are multiple variables, please pack them into a uint64.
        uint64 aux;
    }

    /**
     * @dev Returns the total amount of tokens stored by the contract.
     * 
     * Burned tokens are calculated here, use `_totalMinted()` if you want to count just minted tokens.
     */
    function totalSupply() external view returns (uint256);
}


// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)

pragma solidity ^0.8.0;

/**
 * @dev Provides information about the current execution context, including the
 * sender of the transaction and its data. While these are generally available
 * via msg.sender and msg.data, they should not be accessed in such a direct
 * manner, since when dealing with meta-transactions the account sending and
 * paying for execution may not be the actual sender (as far as an application
 * is concerned).
 *
 * This contract is only required for intermediate, library-like contracts.
 */
abstract contract Context {
    function _msgSender() internal view virtual returns (address) {
        return msg.sender;
    }

    function _msgData() internal view virtual returns (bytes calldata) {
        return msg.data;
    }
}

// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)

pragma solidity ^0.8.0;



/**
 * @dev Implementation of the {IERC165} interface.
 *
 * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
 * for the additional interface id that will be supported. For example:
 *
 * ```solidity
 * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
 *     return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
 * }
 * ```
 *
 * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
 */
abstract contract ERC165 is IERC165 {
    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        return interfaceId == type(IERC165).interfaceId;
    }
}

// ERC721A Contracts v3.3.0
// Creator: Chiru Labs

pragma solidity ^0.8.4;


/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
 * the Metadata extension. Built to optimize for lower gas during batch mints.
 *
 * Assumes serials are sequentially minted starting at _startTokenId() (defaults to 0, e.g. 0, 1, 2, 3..).
 *
 * Assumes that an owner cannot have more than 2**64 - 1 (max value of uint64) of supply.
 *
 * Assumes that the maximum token id cannot exceed 2**256 - 1 (max value of uint256).
 */
contract ERC721A is Context, ERC165, IERC721A {
    using Address for address;
    using Strings for uint256;

    // The tokenId of the next token to be minted.
    uint256 internal _currentIndex;

    // The number of tokens burned.
    uint256 internal _burnCounter;

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to ownership details
    // An empty struct value does not necessarily mean the token is unowned. See _ownershipOf implementation for details.
    mapping(uint256 => TokenOwnership) internal _ownerships;

    // Mapping owner address to address data
    mapping(address => AddressData) private _addressData;

    // 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;

    constructor(string memory name_, string memory symbol_) {
        _name = name_;
        _symbol = symbol_;
        _currentIndex = _startTokenId();
    }

    /**
     * To change the starting tokenId, please override this function.
     */
    function _startTokenId() internal view virtual returns (uint256) {
        return 0;
    }

    /**
     * @dev Burned tokens are calculated here, use _totalMinted() if you want to count just minted tokens.
     */
    function totalSupply() public view override returns (uint256) {
        // Counter underflow is impossible as _burnCounter cannot be incremented
        // more than _currentIndex - _startTokenId() times
        unchecked {
            return _currentIndex - _burnCounter - _startTokenId();
        }
    }

    /**
     * Returns the total amount of tokens minted in the contract.
     */
    function _totalMinted() internal view returns (uint256) {
        // Counter underflow is impossible as _currentIndex does not decrement,
        // and it is initialized to _startTokenId()
        unchecked {
            return _currentIndex - _startTokenId();
        }
    }

    /**
     * @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 override returns (uint256) {
        if (owner == address(0)) revert BalanceQueryForZeroAddress();
        return uint256(_addressData[owner].balance);
    }

    /**
     * Returns the number of tokens minted by `owner`.
     */
    function _numberMinted(address owner) internal view returns (uint256) {
        return uint256(_addressData[owner].numberMinted);
    }

    /**
     * Returns the number of tokens burned by or on behalf of `owner`.
     */
    function _numberBurned(address owner) internal view returns (uint256) {
        return uint256(_addressData[owner].numberBurned);
    }

    /**
     * Returns the auxillary data for `owner`. (e.g. number of whitelist mint slots used).
     */
    function _getAux(address owner) internal view returns (uint64) {
        return _addressData[owner].aux;
    }

    /**
     * Sets the auxillary data for `owner`. (e.g. number of whitelist mint slots used).
     * If there are multiple variables, please pack them into a uint64.
     */
    function _setAux(address owner, uint64 aux) internal {
        _addressData[owner].aux = aux;
    }

    /**
     * Gas spent here starts off proportional to the maximum mint batch size.
     * It gradually moves to O(1) as tokens get transferred around in the collection over time.
     */
    function _ownershipOf(uint256 tokenId) internal view returns (TokenOwnership memory) {
        uint256 curr = tokenId;

        unchecked {
            if (_startTokenId() <= curr) if (curr < _currentIndex) {
                TokenOwnership memory ownership = _ownerships[curr];
                if (!ownership.burned) {
                    if (ownership.addr != address(0)) {
                        return ownership;
                    }
                    // Invariant:
                    // There will always be an ownership that has an address and is not burned
                    // before an ownership that does not have an address and is not burned.
                    // Hence, curr will not underflow.
                    while (true) {
                        curr--;
                        ownership = _ownerships[curr];
                        if (ownership.addr != address(0)) {
                            return ownership;
                        }
                    }
                }
            }
        }
        revert OwnerQueryForNonexistentToken();
    }

    /**
     * @dev See {IERC721-ownerOf}.
     */
    function ownerOf(uint256 tokenId) public view override returns (address) {
        return _ownershipOf(tokenId).addr;
    }

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

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

    /**
     * @dev See {IERC721Metadata-tokenURI}.
     */
    function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
        if (!_exists(tokenId)) revert URIQueryForNonexistentToken();

        string memory baseURI = _baseURI();
        return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : '';
    }

    /**
     * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each
     * token will be the concatenation of the `baseURI` and the `tokenId`. Empty
     * by default, can be overriden in child contracts.
     */
    function _baseURI() internal view virtual returns (string memory) {
        return '';
    }

    /**
     * @dev See {IERC721-approve}.
     */
    function approve(address to, uint256 tokenId) public override {
        address owner = ERC721A.ownerOf(tokenId);
        if (to == owner) revert ApprovalToCurrentOwner();

        if (_msgSender() != owner) if(!isApprovedForAll(owner, _msgSender())) {
            revert ApprovalCallerNotOwnerNorApproved();
        }

        _approve(to, tokenId, owner);
    }

    /**
     * @dev See {IERC721-getApproved}.
     */
    function getApproved(uint256 tokenId) public view override returns (address) {
        if (!_exists(tokenId)) revert ApprovalQueryForNonexistentToken();

        return _tokenApprovals[tokenId];
    }

    /**
     * @dev See {IERC721-setApprovalForAll}.
     */
    function setApprovalForAll(address operator, bool approved) public virtual override {
        if (operator == _msgSender()) revert ApproveToCaller();

        _operatorApprovals[_msgSender()][operator] = approved;
        emit ApprovalForAll(_msgSender(), operator, approved);
    }

    /**
     * @dev See {IERC721-isApprovedForAll}.
     */
    function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
        return _operatorApprovals[owner][operator];
    }

    /**
     * @dev See {IERC721-transferFrom}.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        _transfer(from, to, tokenId);
    }

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        safeTransferFrom(from, to, tokenId, '');
    }

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) public virtual override {
        _transfer(from, to, tokenId);
        if (to.isContract()) if(!_checkContractOnERC721Received(from, to, tokenId, _data)) {
            revert TransferToNonERC721ReceiverImplementer();
        }
    }

    /**
     * @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`),
     */
    function _exists(uint256 tokenId) internal view returns (bool) {
        return _startTokenId() <= tokenId && tokenId < _currentIndex && !_ownerships[tokenId].burned;
    }

    /**
     * @dev Equivalent to `_safeMint(to, quantity, '')`.
     */
    function _safeMint(address to, uint256 quantity) internal {
        _safeMint(to, quantity, '');
    }

    /**
     * @dev Safely mints `quantity` tokens and transfers them to `to`.
     *
     * Requirements:
     *
     * - If `to` refers to a smart contract, it must implement
     *   {IERC721Receiver-onERC721Received}, which is called for each safe transfer.
     * - `quantity` must be greater than 0.
     *
     * Emits a {Transfer} event.
     */
    function _safeMint(
        address to,
        uint256 quantity,
        bytes memory _data
    ) internal {
        uint256 startTokenId = _currentIndex;
        if (to == address(0)) revert MintToZeroAddress();
        if (quantity == 0) revert MintZeroQuantity();

        _beforeTokenTransfers(address(0), to, startTokenId, quantity);

        // Overflows are incredibly unrealistic.
        // balance or numberMinted overflow if current value of either + quantity > 1.8e19 (2**64) - 1
        // updatedIndex overflows if _currentIndex + quantity > 1.2e77 (2**256) - 1
        unchecked {
            _addressData[to].balance += uint64(quantity);
            _addressData[to].numberMinted += uint64(quantity);

            _ownerships[startTokenId].addr = to;
            _ownerships[startTokenId].startTimestamp = uint64(block.timestamp);

            uint256 updatedIndex = startTokenId;
            uint256 end = updatedIndex + quantity;

            if (to.isContract()) {
                do {
                    emit Transfer(address(0), to, updatedIndex);
                    if (!_checkContractOnERC721Received(address(0), to, updatedIndex++, _data)) {
                        revert TransferToNonERC721ReceiverImplementer();
                    }
                } while (updatedIndex < end);
                // Reentrancy protection
                if (_currentIndex != startTokenId) revert();
            } else {
                do {
                    emit Transfer(address(0), to, updatedIndex++);
                } while (updatedIndex < end);
            }
            _currentIndex = updatedIndex;
        }
        _afterTokenTransfers(address(0), to, startTokenId, quantity);
    }

    /**
     * @dev Mints `quantity` tokens and transfers them to `to`.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `quantity` must be greater than 0.
     *
     * Emits a {Transfer} event.
     */
    function _mint(address to, uint256 quantity) internal {
        uint256 startTokenId = _currentIndex;
        if (to == address(0)) revert MintToZeroAddress();
        if (quantity == 0) revert MintZeroQuantity();

        _beforeTokenTransfers(address(0), to, startTokenId, quantity);

        // Overflows are incredibly unrealistic.
        // balance or numberMinted overflow if current value of either + quantity > 1.8e19 (2**64) - 1
        // updatedIndex overflows if _currentIndex + quantity > 1.2e77 (2**256) - 1
        unchecked {
            _addressData[to].balance += uint64(quantity);
            _addressData[to].numberMinted += uint64(quantity);

            _ownerships[startTokenId].addr = to;
            _ownerships[startTokenId].startTimestamp = uint64(block.timestamp);

            uint256 updatedIndex = startTokenId;
            uint256 end = updatedIndex + quantity;

            do {
                emit Transfer(address(0), to, updatedIndex++);
            } while (updatedIndex < end);

            _currentIndex = updatedIndex;
        }
        _afterTokenTransfers(address(0), to, startTokenId, quantity);
    }

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *
     * 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
    ) private {
        TokenOwnership memory prevOwnership = _ownershipOf(tokenId);

        if (prevOwnership.addr != from) revert TransferFromIncorrectOwner();

        bool isApprovedOrOwner = (_msgSender() == from ||
            isApprovedForAll(from, _msgSender()) ||
            getApproved(tokenId) == _msgSender());

        if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved();
        if (to == address(0)) revert TransferToZeroAddress();

        _beforeTokenTransfers(from, to, tokenId, 1);

        // Clear approvals from the previous owner
        _approve(address(0), tokenId, from);

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        // Counter overflow is incredibly unrealistic as tokenId would have to be 2**256.
        unchecked {
            _addressData[from].balance -= 1;
            _addressData[to].balance += 1;

            TokenOwnership storage currSlot = _ownerships[tokenId];
            currSlot.addr = to;
            currSlot.startTimestamp = uint64(block.timestamp);

            // If the ownership slot of tokenId+1 is not explicitly set, that means the transfer initiator owns it.
            // Set the slot of tokenId+1 explicitly in storage to maintain correctness for ownerOf(tokenId+1) calls.
            uint256 nextTokenId = tokenId + 1;
            TokenOwnership storage nextSlot = _ownerships[nextTokenId];
            if (nextSlot.addr == address(0)) {
                // This will suffice for checking _exists(nextTokenId),
                // as a burned slot cannot contain the zero address.
                if (nextTokenId != _currentIndex) {
                    nextSlot.addr = from;
                    nextSlot.startTimestamp = prevOwnership.startTimestamp;
                }
            }
        }

        emit Transfer(from, to, tokenId);
        _afterTokenTransfers(from, to, tokenId, 1);
    }

    /**
     * @dev Equivalent to `_burn(tokenId, false)`.
     */
    function _burn(uint256 tokenId) internal virtual {
        _burn(tokenId, false);
    }

    /**
     * @dev Destroys `tokenId`.
     * The approval is cleared when the token is burned.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     *
     * Emits a {Transfer} event.
     */
    function _burn(uint256 tokenId, bool approvalCheck) internal virtual {
        TokenOwnership memory prevOwnership = _ownershipOf(tokenId);

        address from = prevOwnership.addr;

        if (approvalCheck) {
            bool isApprovedOrOwner = (_msgSender() == from ||
                isApprovedForAll(from, _msgSender()) ||
                getApproved(tokenId) == _msgSender());

            if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved();
        }

        _beforeTokenTransfers(from, address(0), tokenId, 1);

        // Clear approvals from the previous owner
        _approve(address(0), tokenId, from);

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        // Counter overflow is incredibly unrealistic as tokenId would have to be 2**256.
        unchecked {
            AddressData storage addressData = _addressData[from];
            addressData.balance -= 1;
            addressData.numberBurned += 1;

            // Keep track of who burned the token, and the timestamp of burning.
            TokenOwnership storage currSlot = _ownerships[tokenId];
            currSlot.addr = from;
            currSlot.startTimestamp = uint64(block.timestamp);
            currSlot.burned = true;

            // If the ownership slot of tokenId+1 is not explicitly set, that means the burn initiator owns it.
            // Set the slot of tokenId+1 explicitly in storage to maintain correctness for ownerOf(tokenId+1) calls.
            uint256 nextTokenId = tokenId + 1;
            TokenOwnership storage nextSlot = _ownerships[nextTokenId];
            if (nextSlot.addr == address(0)) {
                // This will suffice for checking _exists(nextTokenId),
                // as a burned slot cannot contain the zero address.
                if (nextTokenId != _currentIndex) {
                    nextSlot.addr = from;
                    nextSlot.startTimestamp = prevOwnership.startTimestamp;
                }
            }
        }

        emit Transfer(from, address(0), tokenId);
        _afterTokenTransfers(from, address(0), tokenId, 1);

        // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times.
        unchecked {
            _burnCounter++;
        }
    }

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

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target 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 _checkContractOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) {
            return retval == IERC721Receiver(to).onERC721Received.selector;
        } catch (bytes memory reason) {
            if (reason.length == 0) {
                revert TransferToNonERC721ReceiverImplementer();
            } else {
                assembly {
                    revert(add(32, reason), mload(reason))
                }
            }
        }
    }

    /**
     * @dev Hook that is called before a set of serially-ordered token ids are about to be transferred. This includes minting.
     * And also called before burning one token.
     *
     * startTokenId - the first token id to be transferred
     * quantity - the amount to be transferred
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, `tokenId` will be burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _beforeTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal virtual {}

    /**
     * @dev Hook that is called after a set of serially-ordered token ids have been transferred. This includes
     * minting.
     * And also called after one token has been burned.
     *
     * startTokenId - the first token id to be transferred
     * quantity - the amount to be transferred
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been
     * transferred to `to`.
     * - When `from` is zero, `tokenId` has been minted for `to`.
     * - When `to` is zero, `tokenId` has been burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _afterTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal virtual {}
}
// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol)

pragma solidity ^0.8.0;

/**
 * @title ERC721 token receiver interface
 * @dev Interface for any contract that wants to support safeTransfers
 * from ERC721 asset contracts.
 */
interface 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 `IERC721Receiver.onERC721Received.selector`.
     */
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external returns (bytes4);
}




// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)

pragma solidity ^0.8.0;

/**
 * @dev String operations.
 */
library Strings {
    bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
    uint8 private constant _ADDRESS_LENGTH = 20;

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function toString(uint256 value) internal pure returns (string memory) {
        // Inspired by OraclizeAPI's implementation - MIT licence
        // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol

        if (value == 0) {
            return "0";
        }
        uint256 temp = value;
        uint256 digits;
        while (temp != 0) {
            digits++;
            temp /= 10;
        }
        bytes memory buffer = new bytes(digits);
        while (value != 0) {
            digits -= 1;
            buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
            value /= 10;
        }
        return string(buffer);
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
     */
    function toHexString(uint256 value) internal pure returns (string memory) {
        if (value == 0) {
            return "0x00";
        }
        uint256 temp = value;
        uint256 length = 0;
        while (temp != 0) {
            length++;
            temp >>= 8;
        }
        return toHexString(value, length);
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
     */
    function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
        bytes memory buffer = new bytes(2 * length + 2);
        buffer[0] = "0";
        buffer[1] = "x";
        for (uint256 i = 2 * length + 1; i > 1; --i) {
            buffer[i] = _HEX_SYMBOLS[value & 0xf];
            value >>= 4;
        }
        require(value == 0, "Strings: hex length insufficient");
        return string(buffer);
    }

    /**
     * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.
     */
    function toHexString(address addr) internal pure returns (string memory) {
        return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
    }
}
// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)

pragma solidity ^0.8.1;

/**
 * @dev Collection of functions related to the address type
 */
library Address {
    /**
     * @dev Returns true if `account` is a contract.
     *
     * [IMPORTANT]
     * ====
     * It is unsafe to assume that an address for which this function returns
     * false is an externally-owned account (EOA) and not a contract.
     *
     * Among others, `isContract` will return false for the following
     * types of addresses:
     *
     *  - an externally-owned account
     *  - a contract in construction
     *  - an address where a contract will be created
     *  - an address where a contract lived, but was destroyed
     * ====
     *
     * [IMPORTANT]
     * ====
     * You shouldn't rely on `isContract` to protect against flash loan attacks!
     *
     * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
     * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
     * constructor.
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize/address.code.length, which returns 0
        // for contracts in construction, since the code is only stored at the end
        // of the constructor execution.

        return account.code.length > 0;
    }

    /**
     * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
     * `recipient`, forwarding all available gas and reverting on errors.
     *
     * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
     * of certain opcodes, possibly making contracts go over the 2300 gas limit
     * imposed by `transfer`, making them unable to receive funds via
     * `transfer`. {sendValue} removes this limitation.
     *
     * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
     *
     * IMPORTANT: because control is transferred to `recipient`, care must be
     * taken to not create reentrancy vulnerabilities. Consider using
     * {ReentrancyGuard} or the
     * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
     */
    function sendValue(address payable recipient, uint256 amount) internal {
        require(address(this).balance >= amount, "Address: insufficient balance");

        (bool success, ) = recipient.call{value: amount}("");
        require(success, "Address: unable to send value, recipient may have reverted");
    }

    /**
     * @dev Performs a Solidity function call using a low level `call`. A
     * plain `call` is an unsafe replacement for a function call: use this
     * function instead.
     *
     * If `target` reverts with a revert reason, it is bubbled up by this
     * function (like regular Solidity function calls).
     *
     * Returns the raw returned data. To convert to the expected return value,
     * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
     *
     * Requirements:
     *
     * - `target` must be a contract.
     * - calling `target` with `data` must not revert.
     *
     * _Available since v3.1._
     */
    function functionCall(address target, bytes memory data) internal returns (bytes memory) {
        return functionCallWithValue(target, data, 0, "Address: low-level call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
     * `errorMessage` as a fallback revert reason when `target` reverts.
     *
     * _Available since v3.1._
     */
    function functionCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal returns (bytes memory) {
        return functionCallWithValue(target, data, 0, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but also transferring `value` wei to `target`.
     *
     * Requirements:
     *
     * - the calling contract must have an ETH balance of at least `value`.
     * - the called Solidity function must be `payable`.
     *
     * _Available since v3.1._
     */
    function functionCallWithValue(
        address target,
        bytes memory data,
        uint256 value
    ) internal returns (bytes memory) {
        return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
    }

    /**
     * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
     * with `errorMessage` as a fallback revert reason when `target` reverts.
     *
     * _Available since v3.1._
     */
    function functionCallWithValue(
        address target,
        bytes memory data,
        uint256 value,
        string memory errorMessage
    ) internal returns (bytes memory) {
        require(address(this).balance >= value, "Address: insufficient balance for call");
        (bool success, bytes memory returndata) = target.call{value: value}(data);
        return verifyCallResultFromTarget(target, success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
        return functionStaticCall(target, data, "Address: low-level static call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal view returns (bytes memory) {
        (bool success, bytes memory returndata) = target.staticcall(data);
        return verifyCallResultFromTarget(target, success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
        return functionDelegateCall(target, data, "Address: low-level delegate call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal returns (bytes memory) {
        (bool success, bytes memory returndata) = target.delegatecall(data);
        return verifyCallResultFromTarget(target, success, returndata, errorMessage);
    }

    /**
     * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling
     * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract.
     *
     * _Available since v4.8._
     */
    function verifyCallResultFromTarget(
        address target,
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) internal view returns (bytes memory) {
        if (success) {
            if (returndata.length == 0) {
                // only check isContract if the call was successful and the return data is empty
                // otherwise we already know that it was a contract
                require(isContract(target), "Address: call to non-contract");
            }
            return returndata;
        } else {
            _revert(returndata, errorMessage);
        }
    }

    /**
     * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the
     * revert reason or using the provided one.
     *
     * _Available since v4.3._
     */
    function verifyCallResult(
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) internal pure returns (bytes memory) {
        if (success) {
            return returndata;
        } else {
            _revert(returndata, errorMessage);
        }
    }

    function _revert(bytes memory returndata, string memory errorMessage) private pure {
        // Look for revert reason and bubble it up if present
        if (returndata.length > 0) {
            // The easiest way to bubble the revert reason is using memory via assembly
            /// @solidity memory-safe-assembly
            assembly {
                let returndata_size := mload(returndata)
                revert(add(32, returndata), returndata_size)
            }
        } else {
            revert(errorMessage);
        }
    }
}


// OpenZeppelin Contracts (last updated v4.7.0) (utils/cryptography/MerkleProof.sol)

pragma solidity ^0.8.0;

/**
 * @dev These functions deal with verification of Merkle Tree proofs.
 *
 * The proofs can be generated using the JavaScript library
 * https://github.com/miguelmota/merkletreejs[merkletreejs].
 * Note: the hashing algorithm should be keccak256 and pair sorting should be enabled.
 *
 * See `test/utils/cryptography/MerkleProof.test.js` for some examples.
 *
 * WARNING: You should avoid using leaf values that are 64 bytes long prior to
 * hashing, or use a hash function other than keccak256 for hashing leaves.
 * This is because the concatenation of a sorted pair of internal nodes in
 * the merkle tree could be reinterpreted as a leaf value.
 */
library MerkleProof {
    /**
     * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree
     * defined by `root`. For this, a `proof` must be provided, containing
     * sibling hashes on the branch from the leaf to the root of the tree. Each
     * pair of leaves and each pair of pre-images are assumed to be sorted.
     */
    function verify(
        bytes32[] memory proof,
        bytes32 root,
        bytes32 leaf
    ) internal pure returns (bool) {
        return processProof(proof, leaf) == root;
    }

    /**
     * @dev Calldata version of {verify}
     *
     * _Available since v4.7._
     */
    function verifyCalldata(
        bytes32[] calldata proof,
        bytes32 root,
        bytes32 leaf
    ) internal pure returns (bool) {
        return processProofCalldata(proof, leaf) == root;
    }

    /**
     * @dev Returns the rebuilt hash obtained by traversing a Merkle tree up
     * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt
     * hash matches the root of the tree. When processing the proof, the pairs
     * of leafs & pre-images are assumed to be sorted.
     *
     * _Available since v4.4._
     */
    function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
        bytes32 computedHash = leaf;
        for (uint256 i = 0; i < proof.length; i++) {
            computedHash = _hashPair(computedHash, proof[i]);
        }
        return computedHash;
    }

    /**
     * @dev Calldata version of {processProof}
     *
     * _Available since v4.7._
     */
    function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) {
        bytes32 computedHash = leaf;
        for (uint256 i = 0; i < proof.length; i++) {
            computedHash = _hashPair(computedHash, proof[i]);
        }
        return computedHash;
    }

    /**
     * @dev Returns true if the `leaves` can be proved to be a part of a Merkle tree defined by
     * `root`, according to `proof` and `proofFlags` as described in {processMultiProof}.
     *
     * _Available since v4.7._
     */
    function multiProofVerify(
        bytes32[] memory proof,
        bool[] memory proofFlags,
        bytes32 root,
        bytes32[] memory leaves
    ) internal pure returns (bool) {
        return processMultiProof(proof, proofFlags, leaves) == root;
    }

    /**
     * @dev Calldata version of {multiProofVerify}
     *
     * _Available since v4.7._
     */
    function multiProofVerifyCalldata(
        bytes32[] calldata proof,
        bool[] calldata proofFlags,
        bytes32 root,
        bytes32[] memory leaves
    ) internal pure returns (bool) {
        return processMultiProofCalldata(proof, proofFlags, leaves) == root;
    }

    /**
     * @dev Returns the root of a tree reconstructed from `leaves` and the sibling nodes in `proof`,
     * consuming from one or the other at each step according to the instructions given by
     * `proofFlags`.
     *
     * _Available since v4.7._
     */
    function processMultiProof(
        bytes32[] memory proof,
        bool[] memory proofFlags,
        bytes32[] memory leaves
    ) internal pure returns (bytes32 merkleRoot) {
        // This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by
        // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
        // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
        // the merkle tree.
        uint256 leavesLen = leaves.length;
        uint256 totalHashes = proofFlags.length;

        // Check proof validity.
        require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");

        // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
        // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
        bytes32[] memory hashes = new bytes32[](totalHashes);
        uint256 leafPos = 0;
        uint256 hashPos = 0;
        uint256 proofPos = 0;
        // At each step, we compute the next hash using two values:
        // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
        //   get the next hash.
        // - depending on the flag, either another value for the "main queue" (merging branches) or an element from the
        //   `proof` array.
        for (uint256 i = 0; i < totalHashes; i++) {
            bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
            bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
            hashes[i] = _hashPair(a, b);
        }

        if (totalHashes > 0) {
            return hashes[totalHashes - 1];
        } else if (leavesLen > 0) {
            return leaves[0];
        } else {
            return proof[0];
        }
    }

    /**
     * @dev Calldata version of {processMultiProof}
     *
     * _Available since v4.7._
     */
    function processMultiProofCalldata(
        bytes32[] calldata proof,
        bool[] calldata proofFlags,
        bytes32[] memory leaves
    ) internal pure returns (bytes32 merkleRoot) {
        // This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by
        // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
        // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
        // the merkle tree.
        uint256 leavesLen = leaves.length;
        uint256 totalHashes = proofFlags.length;

        // Check proof validity.
        require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");

        // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
        // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
        bytes32[] memory hashes = new bytes32[](totalHashes);
        uint256 leafPos = 0;
        uint256 hashPos = 0;
        uint256 proofPos = 0;
        // At each step, we compute the next hash using two values:
        // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
        //   get the next hash.
        // - depending on the flag, either another value for the "main queue" (merging branches) or an element from the
        //   `proof` array.
        for (uint256 i = 0; i < totalHashes; i++) {
            bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
            bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
            hashes[i] = _hashPair(a, b);
        }

        if (totalHashes > 0) {
            return hashes[totalHashes - 1];
        } else if (leavesLen > 0) {
            return leaves[0];
        } else {
            return proof[0];
        }
    }

    function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) {
        return a < b ? _efficientHash(a, b) : _efficientHash(b, a);
    }

    function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) {
        /// @solidity memory-safe-assembly
        assembly {
            mstore(0x00, a)
            mstore(0x20, b)
            value := keccak256(0x00, 0x40)
        }
    }
}


// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)

pragma solidity ^0.8.0;



/**
 * @dev Contract module which provides a basic access control mechanism, where
 * there is an account (an owner) that can be granted exclusive access to
 * specific functions.
 *
 * By default, the owner account will be the one that deploys the contract. This
 * can later be changed with {transferOwnership}.
 *
 * This module is used through inheritance. It will make available the modifier
 * `onlyOwner`, which can be applied to your functions to restrict their use to
 * the owner.
 */
abstract contract Ownable is Context {
    address private _owner;

    event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);

    /**
     * @dev Initializes the contract setting the deployer as the initial owner.
     */
    constructor() {
        _transferOwnership(_msgSender());
    }

    /**
     * @dev Throws if called by any account other than the owner.
     */
    modifier onlyOwner() {
        _checkOwner();
        _;
    }

    /**
     * @dev Returns the address of the current owner.
     */
    function owner() public view virtual returns (address) {
        return _owner;
    }

    /**
     * @dev Throws if the sender is not the owner.
     */
    function _checkOwner() internal view virtual {
        require(owner() == _msgSender(), "Ownable: caller is not the owner");
    }

    /**
     * @dev Leaves the contract without owner. It will not be possible to call
     * `onlyOwner` functions anymore. Can only be called by the current owner.
     *
     * NOTE: Renouncing ownership will leave the contract without an owner,
     * thereby removing any functionality that is only available to the owner.
     */
    function renounceOwnership() public virtual onlyOwner {
        _transferOwnership(address(0));
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Can only be called by the current owner.
     */
    function transferOwnership(address newOwner) public virtual onlyOwner {
        require(newOwner != address(0), "Ownable: new owner is the zero address");
        _transferOwnership(newOwner);
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Internal function without access restriction.
     */
    function _transferOwnership(address newOwner) internal virtual {
        address oldOwner = _owner;
        _owner = newOwner;
        emit OwnershipTransferred(oldOwner, newOwner);
    }
}




pragma solidity >=0.8.6;


contract DRGZ is ERC721A, Ownable {

    uint public price = 0.005 ether;
    uint public maxSupply = 7777;
    uint public maxTx = 3;

    mapping(address => uint256) public _preSaleCounter;

    bytes32 public merkleRoot;

    bool private mintOpen = false;
    bool private presaleOpen = false;

    string internal baseTokenURI = '';
    

    constructor() ERC721A("DRGZ", "DRGZ") {}

    function toggleMint() external onlyOwner {
        mintOpen = !mintOpen;
    }

    function togglePresale() external onlyOwner {
        presaleOpen = !presaleOpen;
    }
    
    function setPrice(uint newPrice) external onlyOwner {
        price = newPrice;
    }
    
    function setBaseTokenURI(string calldata _uri) external onlyOwner {
        baseTokenURI = _uri;
    }

    function setMerkleRoot(bytes32 root) external onlyOwner {
        merkleRoot = root;
    }
    
    function setMaxSupply(uint newSupply) external onlyOwner {
        maxSupply = newSupply;
    }

    
    function setMaxTx(uint newMax) external onlyOwner {
        maxTx = newMax;
    }

    function _baseURI() internal override view returns (string memory) {
        return baseTokenURI;
    }

    function buyTo(address to, uint qty) external onlyOwner {
        _mintTo(to, qty);
    }

    function buyPre(uint qty, bytes32[] memory proof) external payable {
        require(presaleOpen, "store closed");
        require(
            _preSaleCounter[msg.sender] + qty <= maxTx,
            "Exceeded max available to purchase"
        );
        _preSaleCounter[msg.sender] =
            _preSaleCounter[msg.sender] +
            qty;
        require(verify(proof), "address not in whitelist");
        _buy(qty);
    }
    
    function buy(uint qty) external payable {
        require(mintOpen, "store closed");
        require(
            _preSaleCounter[msg.sender] + qty <= maxTx,
            "Exceeded max available to purchase"
        );
        _preSaleCounter[msg.sender] =
            _preSaleCounter[msg.sender] +
            qty;
        _buy(qty);
    }

    function _buy(uint qty) internal {
        require(qty <= maxTx && qty > 0, "TRANSACTION: qty of mints not alowed");
        uint free = balanceOf(_msgSender()) == 0 ? 1 : 0;
        require(msg.value >= price * (qty - free), "PAYMENT: invalid value");
        _mintTo(_msgSender(), qty);
    }

    function _mintTo(address to, uint qty) internal {
        require(qty + totalSupply() <= maxSupply, "SUPPLY: Value exceeds totalSupply");
        _mint(to, qty);
    }

    function airdrop(address[] calldata _users) external onlyOwner {
        for (uint256 i = 0; i < _users.length; i++) {
            _safeMint(_users[i], 1);
        }
    }
    
    function withdraw() external onlyOwner {
        payable(_msgSender()).transfer(address(this).balance);
    }

    function verify(bytes32[] memory proof) internal view returns (bool) {
        bytes32 leaf = keccak256(abi.encodePacked(msg.sender));
        return MerkleProof.verify(proof, merkleRoot, leaf);
    }

}

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"ApprovalToCurrentOwner","type":"error"},{"inputs":[],"name":"ApproveToCaller","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"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":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","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":"","type":"address"}],"name":"_preSaleCounter","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"_users","type":"address[]"}],"name":"airdrop","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"qty","type":"uint256"}],"name":"buy","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"qty","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"buyPre","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"qty","type":"uint256"}],"name":"buyTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxTx","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"merkleRoot","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"price","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","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":"string","name":"_uri","type":"string"}],"name":"setBaseTokenURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newSupply","type":"uint256"}],"name":"setMaxSupply","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newMax","type":"uint256"}],"name":"setMaxTx","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"root","type":"bytes32"}],"name":"setMerkleRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newPrice","type":"uint256"}],"name":"setPrice","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":[],"name":"toggleMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"togglePresale","outputs":[],"stateMutability":"nonpayable","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"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]

60806040526611c37937e08000600955611e61600a556003600b556000600e60006101000a81548160ff0219169083151502179055506000600e60016101000a81548160ff02191690831515021790555060405180602001604052806000815250600f90805190602001906200007792919062000235565b503480156200008557600080fd5b506040518060400160405280600481526020017f4452475a000000000000000000000000000000000000000000000000000000008152506040518060400160405280600481526020017f4452475a0000000000000000000000000000000000000000000000000000000081525081600290805190602001906200010a92919062000235565b5080600390805190602001906200012392919062000235565b50620001346200016260201b60201c565b60008190555050506200015c620001506200016760201b60201c565b6200016f60201b60201c565b6200034a565b600090565b600033905090565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b8280546200024390620002e5565b90600052602060002090601f016020900481019282620002675760008555620002b3565b82601f106200028257805160ff1916838001178555620002b3565b82800160010185558215620002b3579182015b82811115620002b257825182559160200191906001019062000295565b5b509050620002c29190620002c6565b5090565b5b80821115620002e1576000816000905550600101620002c7565b5090565b60006002820490506001821680620002fe57607f821691505b602082108114156200031557620003146200031b565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b613dbc806200035a6000396000f3fe6080604052600436106101f95760003560e01c8063729ad39e1161010d578063a22cb465116100a0578063d3dd5fe01161006f578063d3dd5fe0146106e6578063d5abeb01146106fd578063d96a094a14610728578063e985e9c514610744578063f2fde38b14610781576101f9565b8063a22cb4651461062e578063b88d4fde14610657578063bc33718214610680578063c87b56dd146106a9576101f9565b80638da5cb5b116100dc5780638da5cb5b1461058457806391b7f5ed146105af57806395d89b41146105d8578063a035b1fe14610603576101f9565b8063729ad39e146104eb5780637437681e146105145780637cb647591461053f5780638181449214610568576101f9565b80632eb4a7ab1161019057806342842e0e1161015f57806342842e0e146104085780636352211e146104315780636f8b44b01461046e57806370a0823114610497578063715018a6146104d4576101f9565b80632eb4a7ab1461038657806330176e13146103b157806334393743146103da5780633ccfd60b146103f1576101f9565b806309bd4c31116101cc57806309bd4c31146102cc57806318160ddd146102f557806323b872dd1461032057806327b1425a14610349576101f9565b806301ffc9a7146101fe57806306fdde031461023b578063081812fc14610266578063095ea7b3146102a3575b600080fd5b34801561020a57600080fd5b506102256004803603810190610220919061311a565b6107aa565b60405161023291906134fe565b60405180910390f35b34801561024757600080fd5b5061025061088c565b60405161025d9190613534565b60405180910390f35b34801561027257600080fd5b5061028d600480360381019061028891906131c1565b61091e565b60405161029a9190613497565b60405180910390f35b3480156102af57600080fd5b506102ca60048036038101906102c59190613060565b61099a565b005b3480156102d857600080fd5b506102f360048036038101906102ee9190613060565b610a9f565b005b34801561030157600080fd5b5061030a610ab5565b6040516103179190613656565b60405180910390f35b34801561032c57600080fd5b5061034760048036038101906103429190612f4a565b610acc565b005b34801561035557600080fd5b50610370600480360381019061036b9190612edd565b610adc565b60405161037d9190613656565b60405180910390f35b34801561039257600080fd5b5061039b610af4565b6040516103a89190613519565b60405180910390f35b3480156103bd57600080fd5b506103d860048036038101906103d39190613174565b610afa565b005b3480156103e657600080fd5b506103ef610b18565b005b3480156103fd57600080fd5b50610406610b4c565b005b34801561041457600080fd5b5061042f600480360381019061042a9190612f4a565b610ba4565b005b34801561043d57600080fd5b50610458600480360381019061045391906131c1565b610bc4565b6040516104659190613497565b60405180910390f35b34801561047a57600080fd5b50610495600480360381019061049091906131c1565b610bda565b005b3480156104a357600080fd5b506104be60048036038101906104b99190612edd565b610bec565b6040516104cb9190613656565b60405180910390f35b3480156104e057600080fd5b506104e9610cbc565b005b3480156104f757600080fd5b50610512600480360381019061050d91906130a0565b610cd0565b005b34801561052057600080fd5b50610529610d30565b6040516105369190613656565b60405180910390f35b34801561054b57600080fd5b50610566600480360381019061056191906130ed565b610d36565b005b610582600480360381019061057d91906131ee565b610d48565b005b34801561059057600080fd5b50610599610f09565b6040516105a69190613497565b60405180910390f35b3480156105bb57600080fd5b506105d660048036038101906105d191906131c1565b610f33565b005b3480156105e457600080fd5b506105ed610f45565b6040516105fa9190613534565b60405180910390f35b34801561060f57600080fd5b50610618610fd7565b6040516106259190613656565b60405180910390f35b34801561063a57600080fd5b5061065560048036038101906106509190613020565b610fdd565b005b34801561066357600080fd5b5061067e60048036038101906106799190612f9d565b611155565b005b34801561068c57600080fd5b506106a760048036038101906106a291906131c1565b6111cd565b005b3480156106b557600080fd5b506106d060048036038101906106cb91906131c1565b6111df565b6040516106dd9190613534565b60405180910390f35b3480156106f257600080fd5b506106fb61127e565b005b34801561070957600080fd5b506107126112b2565b60405161071f9190613656565b60405180910390f35b610742600480360381019061073d91906131c1565b6112b8565b005b34801561075057600080fd5b5061076b60048036038101906107669190612f0a565b611430565b60405161077891906134fe565b60405180910390f35b34801561078d57600080fd5b506107a860048036038101906107a39190612edd565b6114c4565b005b60007f80ac58cd000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061087557507f5b5e139f000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80610885575061088482611548565b5b9050919050565b60606002805461089b9061390b565b80601f01602080910402602001604051908101604052809291908181526020018280546108c79061390b565b80156109145780601f106108e957610100808354040283529160200191610914565b820191906000526020600020905b8154815290600101906020018083116108f757829003601f168201915b5050505050905090565b6000610929826115b2565b61095f576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b60006109a582610bc4565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610a0d576040517f943f7b8c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff16610a2c611600565b73ffffffffffffffffffffffffffffffffffffffff1614610a8f57610a5881610a53611600565b611430565b610a8e576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b610a9a838383611608565b505050565b610aa76116ba565b610ab18282611738565b5050565b6000610abf61179d565b6001546000540303905090565b610ad78383836117a2565b505050565b600c6020528060005260406000206000915090505481565b600d5481565b610b026116ba565b8181600f9190610b13929190612bbf565b505050565b610b206116ba565b600e60019054906101000a900460ff1615600e60016101000a81548160ff021916908315150217905550565b610b546116ba565b610b5c611600565b73ffffffffffffffffffffffffffffffffffffffff166108fc479081150290604051600060405180830381858888f19350505050158015610ba1573d6000803e3d6000fd5b50565b610bbf83838360405180602001604052806000815250611155565b505050565b6000610bcf82611c58565b600001519050919050565b610be26116ba565b80600a8190555050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415610c54576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600560008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900467ffffffffffffffff1667ffffffffffffffff169050919050565b610cc46116ba565b610cce6000611ee3565b565b610cd86116ba565b60005b82829050811015610d2b57610d18838383818110610cfc57610cfb613a99565b5b9050602002016020810190610d119190612edd565b6001611fa9565b8080610d239061396e565b915050610cdb565b505050565b600b5481565b610d3e6116ba565b80600d8190555050565b600e60019054906101000a900460ff16610d97576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d8e906135b6565b60405180910390fd5b600b5482600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610de59190613736565b1115610e26576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e1d90613636565b60405180910390fd5b81600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610e719190613736565b600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550610ebd81611fc7565b610efc576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ef3906135d6565b60405180910390fd5b610f0582612008565b5050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b610f3b6116ba565b8060098190555050565b606060038054610f549061390b565b80601f0160208091040260200160405190810160405280929190818152602001828054610f809061390b565b8015610fcd5780601f10610fa257610100808354040283529160200191610fcd565b820191906000526020600020905b815481529060010190602001808311610fb057829003601f168201915b5050505050905090565b60095481565b610fe5611600565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16141561104a576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060076000611057611600565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611104611600565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161114991906134fe565b60405180910390a35050565b6111608484846117a2565b61117f8373ffffffffffffffffffffffffffffffffffffffff166120f0565b156111c75761119084848484612113565b6111c6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b6111d56116ba565b80600b8190555050565b60606111ea826115b2565b611220576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600061122a612273565b905060008151141561124b5760405180602001604052806000815250611276565b8061125584612305565b604051602001611266929190613473565b6040516020818303038152906040525b915050919050565b6112866116ba565b600e60009054906101000a900460ff1615600e60006101000a81548160ff021916908315150217905550565b600a5481565b600e60009054906101000a900460ff16611307576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016112fe906135b6565b60405180910390fd5b600b5481600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546113559190613736565b1115611396576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161138d90613636565b60405180910390fd5b80600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546113e19190613736565b600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208190555061142d81612008565b50565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b6114cc6116ba565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16141561153c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161153390613556565b60405180910390fd5b61154581611ee3565b50565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b6000816115bd61179d565b111580156115cc575060005482105b80156115f9575060046000838152602001908152602001600020600001601c9054906101000a900460ff16155b9050919050565b600033905090565b826006600084815260200190815260200160002060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6116c2611600565b73ffffffffffffffffffffffffffffffffffffffff166116e0610f09565b73ffffffffffffffffffffffffffffffffffffffff1614611736576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161172d906135f6565b60405180910390fd5b565b600a54611743610ab5565b8261174e9190613736565b111561178f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161178690613616565b60405180910390fd5b6117998282612466565b5050565b600090565b60006117ad82611c58565b90508373ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614611818576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008473ffffffffffffffffffffffffffffffffffffffff16611839611600565b73ffffffffffffffffffffffffffffffffffffffff161480611868575061186785611862611600565b611430565b5b806118ad5750611876611600565b73ffffffffffffffffffffffffffffffffffffffff166118958461091e565b73ffffffffffffffffffffffffffffffffffffffff16145b9050806118e6576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff16141561194d576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61195a8585856001612742565b61196660008487611608565b6001600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160392506101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506001600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000600460008581526020019081526020016000209050848160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550428160000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555060006001850190506000600460008381526020019081526020016000209050600073ffffffffffffffffffffffffffffffffffffffff168160000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff161415611be6576000548214611be557878160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555084602001518160000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055505b5b505050828473ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4611c518585856001612748565b5050505050565b611c60612c45565b600082905080611c6e61179d565b11611eac57600054811015611eab576000600460008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff16151515158152505090508060400151611ea957600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614611d8d578092505050611ede565b5b600115611ea857818060019003925050600460008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614611ea3578092505050611ede565b611d8e565b5b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b611fc382826040518060200160405280600081525061274e565b5050565b60008033604051602001611fdb9190613458565b60405160208183030381529060405280519060200120905061200083600d5483612b10565b915050919050565b600b54811115801561201a5750600081115b612059576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161205090613576565b60405180910390fd5b60008061206c612067611600565b610bec565b1461207857600061207b565b60015b60ff169050808261208c9190613817565b60095461209991906137bd565b3410156120db576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016120d290613596565b60405180910390fd5b6120ec6120e6611600565b83611738565b5050565b6000808273ffffffffffffffffffffffffffffffffffffffff163b119050919050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02612139611600565b8786866040518563ffffffff1660e01b815260040161215b94939291906134b2565b602060405180830381600087803b15801561217557600080fd5b505af19250505080156121a657506040513d601f19601f820116820180604052508101906121a39190613147565b60015b612220573d80600081146121d6576040519150601f19603f3d011682016040523d82523d6000602084013e6121db565b606091505b50600081511415612218576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b6060600f80546122829061390b565b80601f01602080910402602001604051908101604052809291908181526020018280546122ae9061390b565b80156122fb5780601f106122d0576101008083540402835291602001916122fb565b820191906000526020600020905b8154815290600101906020018083116122de57829003601f168201915b5050505050905090565b6060600082141561234d576040518060400160405280600181526020017f30000000000000000000000000000000000000000000000000000000000000008152509050612461565b600082905060005b6000821461237f5780806123689061396e565b915050600a82612378919061378c565b9150612355565b60008167ffffffffffffffff81111561239b5761239a613ac8565b5b6040519080825280601f01601f1916602001820160405280156123cd5781602001600182028036833780820191505090505b5090505b6000851461245a576001826123e69190613817565b9150600a856123f591906139db565b60306124019190613736565b60f81b81838151811061241757612416613a99565b5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350600a85612453919061378c565b94506123d1565b8093505050505b919050565b600080549050600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1614156124d3576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600082141561250e576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61251b6000848385612742565b81600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555081600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160088282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550826004600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426004600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000819050600083820190505b818060010192508573ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a48082106126be5781600081905550505061273d6000848385612748565b505050565b50505050565b50505050565b600080549050600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff1614156127bb576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008314156127f6576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6128036000858386612742565b82600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555082600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160088282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550836004600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426004600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000819050600084820190506129c48673ffffffffffffffffffffffffffffffffffffffff166120f0565b15612a89575b818673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612a396000878480600101955087612113565b612a6f576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8082106129ca578260005414612a8457600080fd5b612af4565b5b818060010192508673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4808210612a8a575b816000819055505050612b0a6000858386612748565b50505050565b600082612b1d8584612b27565b1490509392505050565b60008082905060005b8451811015612b7257612b5d82868381518110612b5057612b4f613a99565b5b6020026020010151612b7d565b91508080612b6a9061396e565b915050612b30565b508091505092915050565b6000818310612b9557612b908284612ba8565b612ba0565b612b9f8383612ba8565b5b905092915050565b600082600052816020526040600020905092915050565b828054612bcb9061390b565b90600052602060002090601f016020900481019282612bed5760008555612c34565b82601f10612c0657803560ff1916838001178555612c34565b82800160010185558215612c34579182015b82811115612c33578235825591602001919060010190612c18565b5b509050612c419190612c88565b5090565b6040518060600160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff1681526020016000151581525090565b5b80821115612ca1576000816000905550600101612c89565b5090565b6000612cb8612cb384613696565b613671565b90508083825260208201905082856020860282011115612cdb57612cda613b01565b5b60005b85811015612d0b5781612cf18882612e05565b845260208401935060208301925050600181019050612cde565b5050509392505050565b6000612d28612d23846136c2565b613671565b905082815260208101848484011115612d4457612d43613b06565b5b612d4f8482856138c9565b509392505050565b600081359050612d6681613d13565b92915050565b60008083601f840112612d8257612d81613afc565b5b8235905067ffffffffffffffff811115612d9f57612d9e613af7565b5b602083019150836020820283011115612dbb57612dba613b01565b5b9250929050565b600082601f830112612dd757612dd6613afc565b5b8135612de7848260208601612ca5565b91505092915050565b600081359050612dff81613d2a565b92915050565b600081359050612e1481613d41565b92915050565b600081359050612e2981613d58565b92915050565b600081519050612e3e81613d58565b92915050565b600082601f830112612e5957612e58613afc565b5b8135612e69848260208601612d15565b91505092915050565b60008083601f840112612e8857612e87613afc565b5b8235905067ffffffffffffffff811115612ea557612ea4613af7565b5b602083019150836001820283011115612ec157612ec0613b01565b5b9250929050565b600081359050612ed781613d6f565b92915050565b600060208284031215612ef357612ef2613b10565b5b6000612f0184828501612d57565b91505092915050565b60008060408385031215612f2157612f20613b10565b5b6000612f2f85828601612d57565b9250506020612f4085828601612d57565b9150509250929050565b600080600060608486031215612f6357612f62613b10565b5b6000612f7186828701612d57565b9350506020612f8286828701612d57565b9250506040612f9386828701612ec8565b9150509250925092565b60008060008060808587031215612fb757612fb6613b10565b5b6000612fc587828801612d57565b9450506020612fd687828801612d57565b9350506040612fe787828801612ec8565b925050606085013567ffffffffffffffff81111561300857613007613b0b565b5b61301487828801612e44565b91505092959194509250565b6000806040838503121561303757613036613b10565b5b600061304585828601612d57565b925050602061305685828601612df0565b9150509250929050565b6000806040838503121561307757613076613b10565b5b600061308585828601612d57565b925050602061309685828601612ec8565b9150509250929050565b600080602083850312156130b7576130b6613b10565b5b600083013567ffffffffffffffff8111156130d5576130d4613b0b565b5b6130e185828601612d6c565b92509250509250929050565b60006020828403121561310357613102613b10565b5b600061311184828501612e05565b91505092915050565b6000602082840312156131305761312f613b10565b5b600061313e84828501612e1a565b91505092915050565b60006020828403121561315d5761315c613b10565b5b600061316b84828501612e2f565b91505092915050565b6000806020838503121561318b5761318a613b10565b5b600083013567ffffffffffffffff8111156131a9576131a8613b0b565b5b6131b585828601612e72565b92509250509250929050565b6000602082840312156131d7576131d6613b10565b5b60006131e584828501612ec8565b91505092915050565b6000806040838503121561320557613204613b10565b5b600061321385828601612ec8565b925050602083013567ffffffffffffffff81111561323457613233613b0b565b5b61324085828601612dc2565b9150509250929050565b6132538161384b565b82525050565b61326a6132658261384b565b6139b7565b82525050565b6132798161385d565b82525050565b61328881613869565b82525050565b6000613299826136f3565b6132a38185613709565b93506132b38185602086016138d8565b6132bc81613b15565b840191505092915050565b60006132d2826136fe565b6132dc818561371a565b93506132ec8185602086016138d8565b6132f581613b15565b840191505092915050565b600061330b826136fe565b613315818561372b565b93506133258185602086016138d8565b80840191505092915050565b600061333e60268361371a565b915061334982613b33565b604082019050919050565b600061336160248361371a565b915061336c82613b82565b604082019050919050565b600061338460168361371a565b915061338f82613bd1565b602082019050919050565b60006133a7600c8361371a565b91506133b282613bfa565b602082019050919050565b60006133ca60188361371a565b91506133d582613c23565b602082019050919050565b60006133ed60208361371a565b91506133f882613c4c565b602082019050919050565b600061341060218361371a565b915061341b82613c75565b604082019050919050565b600061343360228361371a565b915061343e82613cc4565b604082019050919050565b613452816138bf565b82525050565b60006134648284613259565b60148201915081905092915050565b600061347f8285613300565b915061348b8284613300565b91508190509392505050565b60006020820190506134ac600083018461324a565b92915050565b60006080820190506134c7600083018761324a565b6134d4602083018661324a565b6134e16040830185613449565b81810360608301526134f3818461328e565b905095945050505050565b60006020820190506135136000830184613270565b92915050565b600060208201905061352e600083018461327f565b92915050565b6000602082019050818103600083015261354e81846132c7565b905092915050565b6000602082019050818103600083015261356f81613331565b9050919050565b6000602082019050818103600083015261358f81613354565b9050919050565b600060208201905081810360008301526135af81613377565b9050919050565b600060208201905081810360008301526135cf8161339a565b9050919050565b600060208201905081810360008301526135ef816133bd565b9050919050565b6000602082019050818103600083015261360f816133e0565b9050919050565b6000602082019050818103600083015261362f81613403565b9050919050565b6000602082019050818103600083015261364f81613426565b9050919050565b600060208201905061366b6000830184613449565b92915050565b600061367b61368c565b9050613687828261393d565b919050565b6000604051905090565b600067ffffffffffffffff8211156136b1576136b0613ac8565b5b602082029050602081019050919050565b600067ffffffffffffffff8211156136dd576136dc613ac8565b5b6136e682613b15565b9050602081019050919050565b600081519050919050565b600081519050919050565b600082825260208201905092915050565b600082825260208201905092915050565b600081905092915050565b6000613741826138bf565b915061374c836138bf565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0382111561378157613780613a0c565b5b828201905092915050565b6000613797826138bf565b91506137a2836138bf565b9250826137b2576137b1613a3b565b5b828204905092915050565b60006137c8826138bf565b91506137d3836138bf565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff048311821515161561380c5761380b613a0c565b5b828202905092915050565b6000613822826138bf565b915061382d836138bf565b9250828210156138405761383f613a0c565b5b828203905092915050565b60006138568261389f565b9050919050565b60008115159050919050565b6000819050919050565b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b82818337600083830152505050565b60005b838110156138f65780820151818401526020810190506138db565b83811115613905576000848401525b50505050565b6000600282049050600182168061392357607f821691505b6020821081141561393757613936613a6a565b5b50919050565b61394682613b15565b810181811067ffffffffffffffff8211171561396557613964613ac8565b5b80604052505050565b6000613979826138bf565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8214156139ac576139ab613a0c565b5b600182019050919050565b60006139c2826139c9565b9050919050565b60006139d482613b26565b9050919050565b60006139e6826138bf565b91506139f1836138bf565b925082613a0157613a00613a3b565b5b828206905092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b6000601f19601f8301169050919050565b60008160601b9050919050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f5452414e53414354494f4e3a20717479206f66206d696e7473206e6f7420616c60008201527f6f77656400000000000000000000000000000000000000000000000000000000602082015250565b7f5041594d454e543a20696e76616c69642076616c756500000000000000000000600082015250565b7f73746f726520636c6f7365640000000000000000000000000000000000000000600082015250565b7f61646472657373206e6f7420696e2077686974656c6973740000000000000000600082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f535550504c593a2056616c7565206578636565647320746f74616c537570706c60008201527f7900000000000000000000000000000000000000000000000000000000000000602082015250565b7f4578636565646564206d617820617661696c61626c6520746f2070757263686160008201527f7365000000000000000000000000000000000000000000000000000000000000602082015250565b613d1c8161384b565b8114613d2757600080fd5b50565b613d338161385d565b8114613d3e57600080fd5b50565b613d4a81613869565b8114613d5557600080fd5b50565b613d6181613873565b8114613d6c57600080fd5b50565b613d78816138bf565b8114613d8357600080fd5b5056fea26469706673582212205f766974bb1902178008499c747facb6cd6f5b851bd251b9c1c132d29946105d64736f6c63430008070033

Deployed Bytecode

0x6080604052600436106101f95760003560e01c8063729ad39e1161010d578063a22cb465116100a0578063d3dd5fe01161006f578063d3dd5fe0146106e6578063d5abeb01146106fd578063d96a094a14610728578063e985e9c514610744578063f2fde38b14610781576101f9565b8063a22cb4651461062e578063b88d4fde14610657578063bc33718214610680578063c87b56dd146106a9576101f9565b80638da5cb5b116100dc5780638da5cb5b1461058457806391b7f5ed146105af57806395d89b41146105d8578063a035b1fe14610603576101f9565b8063729ad39e146104eb5780637437681e146105145780637cb647591461053f5780638181449214610568576101f9565b80632eb4a7ab1161019057806342842e0e1161015f57806342842e0e146104085780636352211e146104315780636f8b44b01461046e57806370a0823114610497578063715018a6146104d4576101f9565b80632eb4a7ab1461038657806330176e13146103b157806334393743146103da5780633ccfd60b146103f1576101f9565b806309bd4c31116101cc57806309bd4c31146102cc57806318160ddd146102f557806323b872dd1461032057806327b1425a14610349576101f9565b806301ffc9a7146101fe57806306fdde031461023b578063081812fc14610266578063095ea7b3146102a3575b600080fd5b34801561020a57600080fd5b506102256004803603810190610220919061311a565b6107aa565b60405161023291906134fe565b60405180910390f35b34801561024757600080fd5b5061025061088c565b60405161025d9190613534565b60405180910390f35b34801561027257600080fd5b5061028d600480360381019061028891906131c1565b61091e565b60405161029a9190613497565b60405180910390f35b3480156102af57600080fd5b506102ca60048036038101906102c59190613060565b61099a565b005b3480156102d857600080fd5b506102f360048036038101906102ee9190613060565b610a9f565b005b34801561030157600080fd5b5061030a610ab5565b6040516103179190613656565b60405180910390f35b34801561032c57600080fd5b5061034760048036038101906103429190612f4a565b610acc565b005b34801561035557600080fd5b50610370600480360381019061036b9190612edd565b610adc565b60405161037d9190613656565b60405180910390f35b34801561039257600080fd5b5061039b610af4565b6040516103a89190613519565b60405180910390f35b3480156103bd57600080fd5b506103d860048036038101906103d39190613174565b610afa565b005b3480156103e657600080fd5b506103ef610b18565b005b3480156103fd57600080fd5b50610406610b4c565b005b34801561041457600080fd5b5061042f600480360381019061042a9190612f4a565b610ba4565b005b34801561043d57600080fd5b50610458600480360381019061045391906131c1565b610bc4565b6040516104659190613497565b60405180910390f35b34801561047a57600080fd5b50610495600480360381019061049091906131c1565b610bda565b005b3480156104a357600080fd5b506104be60048036038101906104b99190612edd565b610bec565b6040516104cb9190613656565b60405180910390f35b3480156104e057600080fd5b506104e9610cbc565b005b3480156104f757600080fd5b50610512600480360381019061050d91906130a0565b610cd0565b005b34801561052057600080fd5b50610529610d30565b6040516105369190613656565b60405180910390f35b34801561054b57600080fd5b50610566600480360381019061056191906130ed565b610d36565b005b610582600480360381019061057d91906131ee565b610d48565b005b34801561059057600080fd5b50610599610f09565b6040516105a69190613497565b60405180910390f35b3480156105bb57600080fd5b506105d660048036038101906105d191906131c1565b610f33565b005b3480156105e457600080fd5b506105ed610f45565b6040516105fa9190613534565b60405180910390f35b34801561060f57600080fd5b50610618610fd7565b6040516106259190613656565b60405180910390f35b34801561063a57600080fd5b5061065560048036038101906106509190613020565b610fdd565b005b34801561066357600080fd5b5061067e60048036038101906106799190612f9d565b611155565b005b34801561068c57600080fd5b506106a760048036038101906106a291906131c1565b6111cd565b005b3480156106b557600080fd5b506106d060048036038101906106cb91906131c1565b6111df565b6040516106dd9190613534565b60405180910390f35b3480156106f257600080fd5b506106fb61127e565b005b34801561070957600080fd5b506107126112b2565b60405161071f9190613656565b60405180910390f35b610742600480360381019061073d91906131c1565b6112b8565b005b34801561075057600080fd5b5061076b60048036038101906107669190612f0a565b611430565b60405161077891906134fe565b60405180910390f35b34801561078d57600080fd5b506107a860048036038101906107a39190612edd565b6114c4565b005b60007f80ac58cd000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061087557507f5b5e139f000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80610885575061088482611548565b5b9050919050565b60606002805461089b9061390b565b80601f01602080910402602001604051908101604052809291908181526020018280546108c79061390b565b80156109145780601f106108e957610100808354040283529160200191610914565b820191906000526020600020905b8154815290600101906020018083116108f757829003601f168201915b5050505050905090565b6000610929826115b2565b61095f576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b60006109a582610bc4565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610a0d576040517f943f7b8c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff16610a2c611600565b73ffffffffffffffffffffffffffffffffffffffff1614610a8f57610a5881610a53611600565b611430565b610a8e576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b610a9a838383611608565b505050565b610aa76116ba565b610ab18282611738565b5050565b6000610abf61179d565b6001546000540303905090565b610ad78383836117a2565b505050565b600c6020528060005260406000206000915090505481565b600d5481565b610b026116ba565b8181600f9190610b13929190612bbf565b505050565b610b206116ba565b600e60019054906101000a900460ff1615600e60016101000a81548160ff021916908315150217905550565b610b546116ba565b610b5c611600565b73ffffffffffffffffffffffffffffffffffffffff166108fc479081150290604051600060405180830381858888f19350505050158015610ba1573d6000803e3d6000fd5b50565b610bbf83838360405180602001604052806000815250611155565b505050565b6000610bcf82611c58565b600001519050919050565b610be26116ba565b80600a8190555050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415610c54576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600560008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900467ffffffffffffffff1667ffffffffffffffff169050919050565b610cc46116ba565b610cce6000611ee3565b565b610cd86116ba565b60005b82829050811015610d2b57610d18838383818110610cfc57610cfb613a99565b5b9050602002016020810190610d119190612edd565b6001611fa9565b8080610d239061396e565b915050610cdb565b505050565b600b5481565b610d3e6116ba565b80600d8190555050565b600e60019054906101000a900460ff16610d97576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d8e906135b6565b60405180910390fd5b600b5482600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610de59190613736565b1115610e26576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e1d90613636565b60405180910390fd5b81600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054610e719190613736565b600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550610ebd81611fc7565b610efc576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ef3906135d6565b60405180910390fd5b610f0582612008565b5050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b610f3b6116ba565b8060098190555050565b606060038054610f549061390b565b80601f0160208091040260200160405190810160405280929190818152602001828054610f809061390b565b8015610fcd5780601f10610fa257610100808354040283529160200191610fcd565b820191906000526020600020905b815481529060010190602001808311610fb057829003601f168201915b5050505050905090565b60095481565b610fe5611600565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16141561104a576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060076000611057611600565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611104611600565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161114991906134fe565b60405180910390a35050565b6111608484846117a2565b61117f8373ffffffffffffffffffffffffffffffffffffffff166120f0565b156111c75761119084848484612113565b6111c6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b6111d56116ba565b80600b8190555050565b60606111ea826115b2565b611220576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600061122a612273565b905060008151141561124b5760405180602001604052806000815250611276565b8061125584612305565b604051602001611266929190613473565b6040516020818303038152906040525b915050919050565b6112866116ba565b600e60009054906101000a900460ff1615600e60006101000a81548160ff021916908315150217905550565b600a5481565b600e60009054906101000a900460ff16611307576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016112fe906135b6565b60405180910390fd5b600b5481600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546113559190613736565b1115611396576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161138d90613636565b60405180910390fd5b80600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546113e19190613736565b600c60003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208190555061142d81612008565b50565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b6114cc6116ba565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16141561153c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161153390613556565b60405180910390fd5b61154581611ee3565b50565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b6000816115bd61179d565b111580156115cc575060005482105b80156115f9575060046000838152602001908152602001600020600001601c9054906101000a900460ff16155b9050919050565b600033905090565b826006600084815260200190815260200160002060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6116c2611600565b73ffffffffffffffffffffffffffffffffffffffff166116e0610f09565b73ffffffffffffffffffffffffffffffffffffffff1614611736576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161172d906135f6565b60405180910390fd5b565b600a54611743610ab5565b8261174e9190613736565b111561178f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161178690613616565b60405180910390fd5b6117998282612466565b5050565b600090565b60006117ad82611c58565b90508373ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614611818576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008473ffffffffffffffffffffffffffffffffffffffff16611839611600565b73ffffffffffffffffffffffffffffffffffffffff161480611868575061186785611862611600565b611430565b5b806118ad5750611876611600565b73ffffffffffffffffffffffffffffffffffffffff166118958461091e565b73ffffffffffffffffffffffffffffffffffffffff16145b9050806118e6576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff16141561194d576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61195a8585856001612742565b61196660008487611608565b6001600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160392506101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506001600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000600460008581526020019081526020016000209050848160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550428160000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555060006001850190506000600460008381526020019081526020016000209050600073ffffffffffffffffffffffffffffffffffffffff168160000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff161415611be6576000548214611be557878160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555084602001518160000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055505b5b505050828473ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4611c518585856001612748565b5050505050565b611c60612c45565b600082905080611c6e61179d565b11611eac57600054811015611eab576000600460008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff16151515158152505090508060400151611ea957600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614611d8d578092505050611ede565b5b600115611ea857818060019003925050600460008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614611ea3578092505050611ede565b611d8e565b5b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b611fc382826040518060200160405280600081525061274e565b5050565b60008033604051602001611fdb9190613458565b60405160208183030381529060405280519060200120905061200083600d5483612b10565b915050919050565b600b54811115801561201a5750600081115b612059576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161205090613576565b60405180910390fd5b60008061206c612067611600565b610bec565b1461207857600061207b565b60015b60ff169050808261208c9190613817565b60095461209991906137bd565b3410156120db576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016120d290613596565b60405180910390fd5b6120ec6120e6611600565b83611738565b5050565b6000808273ffffffffffffffffffffffffffffffffffffffff163b119050919050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02612139611600565b8786866040518563ffffffff1660e01b815260040161215b94939291906134b2565b602060405180830381600087803b15801561217557600080fd5b505af19250505080156121a657506040513d601f19601f820116820180604052508101906121a39190613147565b60015b612220573d80600081146121d6576040519150601f19603f3d011682016040523d82523d6000602084013e6121db565b606091505b50600081511415612218576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b6060600f80546122829061390b565b80601f01602080910402602001604051908101604052809291908181526020018280546122ae9061390b565b80156122fb5780601f106122d0576101008083540402835291602001916122fb565b820191906000526020600020905b8154815290600101906020018083116122de57829003601f168201915b5050505050905090565b6060600082141561234d576040518060400160405280600181526020017f30000000000000000000000000000000000000000000000000000000000000008152509050612461565b600082905060005b6000821461237f5780806123689061396e565b915050600a82612378919061378c565b9150612355565b60008167ffffffffffffffff81111561239b5761239a613ac8565b5b6040519080825280601f01601f1916602001820160405280156123cd5781602001600182028036833780820191505090505b5090505b6000851461245a576001826123e69190613817565b9150600a856123f591906139db565b60306124019190613736565b60f81b81838151811061241757612416613a99565b5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350600a85612453919061378c565b94506123d1565b8093505050505b919050565b600080549050600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1614156124d3576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600082141561250e576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61251b6000848385612742565b81600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555081600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160088282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550826004600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426004600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000819050600083820190505b818060010192508573ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a48082106126be5781600081905550505061273d6000848385612748565b505050565b50505050565b50505050565b600080549050600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff1614156127bb576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008314156127f6576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6128036000858386612742565b82600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555082600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160088282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550836004600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426004600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000819050600084820190506129c48673ffffffffffffffffffffffffffffffffffffffff166120f0565b15612a89575b818673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612a396000878480600101955087612113565b612a6f576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8082106129ca578260005414612a8457600080fd5b612af4565b5b818060010192508673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4808210612a8a575b816000819055505050612b0a6000858386612748565b50505050565b600082612b1d8584612b27565b1490509392505050565b60008082905060005b8451811015612b7257612b5d82868381518110612b5057612b4f613a99565b5b6020026020010151612b7d565b91508080612b6a9061396e565b915050612b30565b508091505092915050565b6000818310612b9557612b908284612ba8565b612ba0565b612b9f8383612ba8565b5b905092915050565b600082600052816020526040600020905092915050565b828054612bcb9061390b565b90600052602060002090601f016020900481019282612bed5760008555612c34565b82601f10612c0657803560ff1916838001178555612c34565b82800160010185558215612c34579182015b82811115612c33578235825591602001919060010190612c18565b5b509050612c419190612c88565b5090565b6040518060600160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff1681526020016000151581525090565b5b80821115612ca1576000816000905550600101612c89565b5090565b6000612cb8612cb384613696565b613671565b90508083825260208201905082856020860282011115612cdb57612cda613b01565b5b60005b85811015612d0b5781612cf18882612e05565b845260208401935060208301925050600181019050612cde565b5050509392505050565b6000612d28612d23846136c2565b613671565b905082815260208101848484011115612d4457612d43613b06565b5b612d4f8482856138c9565b509392505050565b600081359050612d6681613d13565b92915050565b60008083601f840112612d8257612d81613afc565b5b8235905067ffffffffffffffff811115612d9f57612d9e613af7565b5b602083019150836020820283011115612dbb57612dba613b01565b5b9250929050565b600082601f830112612dd757612dd6613afc565b5b8135612de7848260208601612ca5565b91505092915050565b600081359050612dff81613d2a565b92915050565b600081359050612e1481613d41565b92915050565b600081359050612e2981613d58565b92915050565b600081519050612e3e81613d58565b92915050565b600082601f830112612e5957612e58613afc565b5b8135612e69848260208601612d15565b91505092915050565b60008083601f840112612e8857612e87613afc565b5b8235905067ffffffffffffffff811115612ea557612ea4613af7565b5b602083019150836001820283011115612ec157612ec0613b01565b5b9250929050565b600081359050612ed781613d6f565b92915050565b600060208284031215612ef357612ef2613b10565b5b6000612f0184828501612d57565b91505092915050565b60008060408385031215612f2157612f20613b10565b5b6000612f2f85828601612d57565b9250506020612f4085828601612d57565b9150509250929050565b600080600060608486031215612f6357612f62613b10565b5b6000612f7186828701612d57565b9350506020612f8286828701612d57565b9250506040612f9386828701612ec8565b9150509250925092565b60008060008060808587031215612fb757612fb6613b10565b5b6000612fc587828801612d57565b9450506020612fd687828801612d57565b9350506040612fe787828801612ec8565b925050606085013567ffffffffffffffff81111561300857613007613b0b565b5b61301487828801612e44565b91505092959194509250565b6000806040838503121561303757613036613b10565b5b600061304585828601612d57565b925050602061305685828601612df0565b9150509250929050565b6000806040838503121561307757613076613b10565b5b600061308585828601612d57565b925050602061309685828601612ec8565b9150509250929050565b600080602083850312156130b7576130b6613b10565b5b600083013567ffffffffffffffff8111156130d5576130d4613b0b565b5b6130e185828601612d6c565b92509250509250929050565b60006020828403121561310357613102613b10565b5b600061311184828501612e05565b91505092915050565b6000602082840312156131305761312f613b10565b5b600061313e84828501612e1a565b91505092915050565b60006020828403121561315d5761315c613b10565b5b600061316b84828501612e2f565b91505092915050565b6000806020838503121561318b5761318a613b10565b5b600083013567ffffffffffffffff8111156131a9576131a8613b0b565b5b6131b585828601612e72565b92509250509250929050565b6000602082840312156131d7576131d6613b10565b5b60006131e584828501612ec8565b91505092915050565b6000806040838503121561320557613204613b10565b5b600061321385828601612ec8565b925050602083013567ffffffffffffffff81111561323457613233613b0b565b5b61324085828601612dc2565b9150509250929050565b6132538161384b565b82525050565b61326a6132658261384b565b6139b7565b82525050565b6132798161385d565b82525050565b61328881613869565b82525050565b6000613299826136f3565b6132a38185613709565b93506132b38185602086016138d8565b6132bc81613b15565b840191505092915050565b60006132d2826136fe565b6132dc818561371a565b93506132ec8185602086016138d8565b6132f581613b15565b840191505092915050565b600061330b826136fe565b613315818561372b565b93506133258185602086016138d8565b80840191505092915050565b600061333e60268361371a565b915061334982613b33565b604082019050919050565b600061336160248361371a565b915061336c82613b82565b604082019050919050565b600061338460168361371a565b915061338f82613bd1565b602082019050919050565b60006133a7600c8361371a565b91506133b282613bfa565b602082019050919050565b60006133ca60188361371a565b91506133d582613c23565b602082019050919050565b60006133ed60208361371a565b91506133f882613c4c565b602082019050919050565b600061341060218361371a565b915061341b82613c75565b604082019050919050565b600061343360228361371a565b915061343e82613cc4565b604082019050919050565b613452816138bf565b82525050565b60006134648284613259565b60148201915081905092915050565b600061347f8285613300565b915061348b8284613300565b91508190509392505050565b60006020820190506134ac600083018461324a565b92915050565b60006080820190506134c7600083018761324a565b6134d4602083018661324a565b6134e16040830185613449565b81810360608301526134f3818461328e565b905095945050505050565b60006020820190506135136000830184613270565b92915050565b600060208201905061352e600083018461327f565b92915050565b6000602082019050818103600083015261354e81846132c7565b905092915050565b6000602082019050818103600083015261356f81613331565b9050919050565b6000602082019050818103600083015261358f81613354565b9050919050565b600060208201905081810360008301526135af81613377565b9050919050565b600060208201905081810360008301526135cf8161339a565b9050919050565b600060208201905081810360008301526135ef816133bd565b9050919050565b6000602082019050818103600083015261360f816133e0565b9050919050565b6000602082019050818103600083015261362f81613403565b9050919050565b6000602082019050818103600083015261364f81613426565b9050919050565b600060208201905061366b6000830184613449565b92915050565b600061367b61368c565b9050613687828261393d565b919050565b6000604051905090565b600067ffffffffffffffff8211156136b1576136b0613ac8565b5b602082029050602081019050919050565b600067ffffffffffffffff8211156136dd576136dc613ac8565b5b6136e682613b15565b9050602081019050919050565b600081519050919050565b600081519050919050565b600082825260208201905092915050565b600082825260208201905092915050565b600081905092915050565b6000613741826138bf565b915061374c836138bf565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0382111561378157613780613a0c565b5b828201905092915050565b6000613797826138bf565b91506137a2836138bf565b9250826137b2576137b1613a3b565b5b828204905092915050565b60006137c8826138bf565b91506137d3836138bf565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff048311821515161561380c5761380b613a0c565b5b828202905092915050565b6000613822826138bf565b915061382d836138bf565b9250828210156138405761383f613a0c565b5b828203905092915050565b60006138568261389f565b9050919050565b60008115159050919050565b6000819050919050565b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b82818337600083830152505050565b60005b838110156138f65780820151818401526020810190506138db565b83811115613905576000848401525b50505050565b6000600282049050600182168061392357607f821691505b6020821081141561393757613936613a6a565b5b50919050565b61394682613b15565b810181811067ffffffffffffffff8211171561396557613964613ac8565b5b80604052505050565b6000613979826138bf565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8214156139ac576139ab613a0c565b5b600182019050919050565b60006139c2826139c9565b9050919050565b60006139d482613b26565b9050919050565b60006139e6826138bf565b91506139f1836138bf565b925082613a0157613a00613a3b565b5b828206905092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b6000601f19601f8301169050919050565b60008160601b9050919050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f5452414e53414354494f4e3a20717479206f66206d696e7473206e6f7420616c60008201527f6f77656400000000000000000000000000000000000000000000000000000000602082015250565b7f5041594d454e543a20696e76616c69642076616c756500000000000000000000600082015250565b7f73746f726520636c6f7365640000000000000000000000000000000000000000600082015250565b7f61646472657373206e6f7420696e2077686974656c6973740000000000000000600082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f535550504c593a2056616c7565206578636565647320746f74616c537570706c60008201527f7900000000000000000000000000000000000000000000000000000000000000602082015250565b7f4578636565646564206d617820617661696c61626c6520746f2070757263686160008201527f7365000000000000000000000000000000000000000000000000000000000000602082015250565b613d1c8161384b565b8114613d2757600080fd5b50565b613d338161385d565b8114613d3e57600080fd5b50565b613d4a81613869565b8114613d5557600080fd5b50565b613d6181613873565b8114613d6c57600080fd5b50565b613d78816138bf565b8114613d8357600080fd5b5056fea26469706673582212205f766974bb1902178008499c747facb6cd6f5b851bd251b9c1c132d29946105d64736f6c63430008070033

Deployed Bytecode Sourcemap

56933:3138:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;13984:305;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;17099:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;18603:204;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;18165:372;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;58164:91;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;13224:312;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;19468:170;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;57079:50;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57138:25;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57633:104;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;57433:89;;;;;;;;;;;;;:::i;:::-;;59744:111;;;;;;;;;;;;;:::i;:::-;;19709:185;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;16907:125;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57849:97;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;14353:206;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;56080:103;;;;;;;;;;;;;:::i;:::-;;59557:175;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;57049:21;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57745:92;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;58263:440;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;55432:87;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57534;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;17268:104;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;56976:31;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;18879:287;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;19965:370;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;57960:83;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;17443:318;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57345:80;;;;;;;;;;;;;:::i;:::-;;57014:28;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;58715:349;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;19237:164;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;56338:201;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;13984:305;14086:4;14138:25;14123:40;;;:11;:40;;;;:105;;;;14195:33;14180:48;;;:11;:48;;;;14123:105;:158;;;;14245:36;14269:11;14245:23;:36::i;:::-;14123:158;14103:178;;13984:305;;;:::o;17099:100::-;17153:13;17186:5;17179:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;17099:100;:::o;18603:204::-;18671:7;18696:16;18704:7;18696;:16::i;:::-;18691:64;;18721:34;;;;;;;;;;;;;;18691:64;18775:15;:24;18791:7;18775:24;;;;;;;;;;;;;;;;;;;;;18768:31;;18603:204;;;:::o;18165:372::-;18238:13;18254:24;18270:7;18254:15;:24::i;:::-;18238:40;;18299:5;18293:11;;:2;:11;;;18289:48;;;18313:24;;;;;;;;;;;;;;18289:48;18370:5;18354:21;;:12;:10;:12::i;:::-;:21;;;18350:139;;18381:37;18398:5;18405:12;:10;:12::i;:::-;18381:16;:37::i;:::-;18377:112;;18442:35;;;;;;;;;;;;;;18377:112;18350:139;18501:28;18510:2;18514:7;18523:5;18501:8;:28::i;:::-;18227:310;18165:372;;:::o;58164:91::-;55318:13;:11;:13::i;:::-;58231:16:::1;58239:2;58243:3;58231:7;:16::i;:::-;58164:91:::0;;:::o;13224:312::-;13277:7;13502:15;:13;:15::i;:::-;13487:12;;13471:13;;:28;:46;13464:53;;13224:312;:::o;19468:170::-;19602:28;19612:4;19618:2;19622:7;19602:9;:28::i;:::-;19468:170;;;:::o;57079:50::-;;;;;;;;;;;;;;;;;:::o;57138:25::-;;;;:::o;57633:104::-;55318:13;:11;:13::i;:::-;57725:4:::1;;57710:12;:19;;;;;;;:::i;:::-;;57633:104:::0;;:::o;57433:89::-;55318:13;:11;:13::i;:::-;57503:11:::1;;;;;;;;;;;57502:12;57488:11;;:26;;;;;;;;;;;;;;;;;;57433:89::o:0;59744:111::-;55318:13;:11;:13::i;:::-;59802:12:::1;:10;:12::i;:::-;59794:30;;:53;59825:21;59794:53;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;;;;;;59744:111::o:0;19709:185::-;19847:39;19864:4;19870:2;19874:7;19847:39;;;;;;;;;;;;:16;:39::i;:::-;19709:185;;;:::o;16907:125::-;16971:7;16998:21;17011:7;16998:12;:21::i;:::-;:26;;;16991:33;;16907:125;;;:::o;57849:97::-;55318:13;:11;:13::i;:::-;57929:9:::1;57917;:21;;;;57849:97:::0;:::o;14353:206::-;14417:7;14458:1;14441:19;;:5;:19;;;14437:60;;;14469:28;;;;;;;;;;;;;;14437:60;14523:12;:19;14536:5;14523:19;;;;;;;;;;;;;;;:27;;;;;;;;;;;;14515:36;;14508:43;;14353:206;;;:::o;56080:103::-;55318:13;:11;:13::i;:::-;56145:30:::1;56172:1;56145:18;:30::i;:::-;56080:103::o:0;59557:175::-;55318:13;:11;:13::i;:::-;59636:9:::1;59631:94;59655:6;;:13;;59651:1;:17;59631:94;;;59690:23;59700:6;;59707:1;59700:9;;;;;;;:::i;:::-;;;;;;;;;;;;;;;:::i;:::-;59711:1;59690:9;:23::i;:::-;59670:3;;;;;:::i;:::-;;;;59631:94;;;;59557:175:::0;;:::o;57049:21::-;;;;:::o;57745:92::-;55318:13;:11;:13::i;:::-;57825:4:::1;57812:10;:17;;;;57745:92:::0;:::o;58263:440::-;58349:11;;;;;;;;;;;58341:36;;;;;;;;;;;;:::i;:::-;;;;;;;;;58447:5;;58440:3;58410:15;:27;58426:10;58410:27;;;;;;;;;;;;;;;;:33;;;;:::i;:::-;:42;;58388:126;;;;;;;;;;;;:::i;:::-;;;;;;;;;58611:3;58568:15;:27;58584:10;58568:27;;;;;;;;;;;;;;;;:46;;;;:::i;:::-;58525:15;:27;58541:10;58525:27;;;;;;;;;;;;;;;:89;;;;58633:13;58640:5;58633:6;:13::i;:::-;58625:50;;;;;;;;;;;;:::i;:::-;;;;;;;;;58686:9;58691:3;58686:4;:9::i;:::-;58263:440;;:::o;55432:87::-;55478:7;55505:6;;;;;;;;;;;55498:13;;55432:87;:::o;57534:::-;55318:13;:11;:13::i;:::-;57605:8:::1;57597:5;:16;;;;57534:87:::0;:::o;17268:104::-;17324:13;17357:7;17350:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;17268:104;:::o;56976:31::-;;;;:::o;18879:287::-;18990:12;:10;:12::i;:::-;18978:24;;:8;:24;;;18974:54;;;19011:17;;;;;;;;;;;;;;18974:54;19086:8;19041:18;:32;19060:12;:10;:12::i;:::-;19041:32;;;;;;;;;;;;;;;:42;19074:8;19041:42;;;;;;;;;;;;;;;;:53;;;;;;;;;;;;;;;;;;19139:8;19110:48;;19125:12;:10;:12::i;:::-;19110:48;;;19149:8;19110:48;;;;;;:::i;:::-;;;;;;;;18879:287;;:::o;19965:370::-;20132:28;20142:4;20148:2;20152:7;20132:9;:28::i;:::-;20175:15;:2;:13;;;:15::i;:::-;20171:157;;;20196:56;20227:4;20233:2;20237:7;20246:5;20196:30;:56::i;:::-;20192:136;;20276:40;;;;;;;;;;;;;;20192:136;20171:157;19965:370;;;;:::o;57960:83::-;55318:13;:11;:13::i;:::-;58029:6:::1;58021:5;:14;;;;57960:83:::0;:::o;17443:318::-;17516:13;17547:16;17555:7;17547;:16::i;:::-;17542:59;;17572:29;;;;;;;;;;;;;;17542:59;17614:21;17638:10;:8;:10::i;:::-;17614:34;;17691:1;17672:7;17666:21;:26;;:87;;;;;;;;;;;;;;;;;17719:7;17728:18;:7;:16;:18::i;:::-;17702:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;17666:87;17659:94;;;17443:318;;;:::o;57345:80::-;55318:13;:11;:13::i;:::-;57409:8:::1;;;;;;;;;;;57408:9;57397:8;;:20;;;;;;;;;;;;;;;;;;57345:80::o:0;57014:28::-;;;;:::o;58715:349::-;58774:8;;;;;;;;;;;58766:33;;;;;;;;;;;;:::i;:::-;;;;;;;;;58869:5;;58862:3;58832:15;:27;58848:10;58832:27;;;;;;;;;;;;;;;;:33;;;;:::i;:::-;:42;;58810:126;;;;;;;;;;;;:::i;:::-;;;;;;;;;59033:3;58990:15;:27;59006:10;58990:27;;;;;;;;;;;;;;;;:46;;;;:::i;:::-;58947:15;:27;58963:10;58947:27;;;;;;;;;;;;;;;:89;;;;59047:9;59052:3;59047:4;:9::i;:::-;58715:349;:::o;19237:164::-;19334:4;19358:18;:25;19377:5;19358:25;;;;;;;;;;;;;;;:35;19384:8;19358:35;;;;;;;;;;;;;;;;;;;;;;;;;19351:42;;19237:164;;;;:::o;56338:201::-;55318:13;:11;:13::i;:::-;56447:1:::1;56427:22;;:8;:22;;;;56419:73;;;;;;;;;;;;:::i;:::-;;;;;;;;;56503:28;56522:8;56503:18;:28::i;:::-;56338:201:::0;:::o;11025:157::-;11110:4;11149:25;11134:40;;;:11;:40;;;;11127:47;;11025:157;;;:::o;20590:174::-;20647:4;20690:7;20671:15;:13;:15::i;:::-;:26;;:53;;;;;20711:13;;20701:7;:23;20671:53;:85;;;;;20729:11;:20;20741:7;20729:20;;;;;;;;;;;:27;;;;;;;;;;;;20728:28;20671:85;20664:92;;20590:174;;;:::o;10013:98::-;10066:7;10093:10;10086:17;;10013:98;:::o;29812:196::-;29954:2;29927:15;:24;29943:7;29927:24;;;;;;;;;;;;:29;;;;;;;;;;;;;;;;;;29992:7;29988:2;29972:28;;29981:5;29972:28;;;;;;;;;;;;29812:196;;;:::o;55597:132::-;55672:12;:10;:12::i;:::-;55661:23;;:7;:5;:7::i;:::-;:23;;;55653:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;55597:132::o;59379:170::-;59469:9;;59452:13;:11;:13::i;:::-;59446:3;:19;;;;:::i;:::-;:32;;59438:78;;;;;;;;;;;;:::i;:::-;;;;;;;;;59527:14;59533:2;59537:3;59527:5;:14::i;:::-;59379:170;;:::o;12998:92::-;13054:7;12998:92;:::o;24760:2130::-;24875:35;24913:21;24926:7;24913:12;:21::i;:::-;24875:59;;24973:4;24951:26;;:13;:18;;;:26;;;24947:67;;24986:28;;;;;;;;;;;;;;24947:67;25027:22;25069:4;25053:20;;:12;:10;:12::i;:::-;:20;;;:73;;;;25090:36;25107:4;25113:12;:10;:12::i;:::-;25090:16;:36::i;:::-;25053:73;:126;;;;25167:12;:10;:12::i;:::-;25143:36;;:20;25155:7;25143:11;:20::i;:::-;:36;;;25053:126;25027:153;;25198:17;25193:66;;25224:35;;;;;;;;;;;;;;25193:66;25288:1;25274:16;;:2;:16;;;25270:52;;;25299:23;;;;;;;;;;;;;;25270:52;25335:43;25357:4;25363:2;25367:7;25376:1;25335:21;:43::i;:::-;25443:35;25460:1;25464:7;25473:4;25443:8;:35::i;:::-;25804:1;25774:12;:18;25787:4;25774:18;;;;;;;;;;;;;;;:26;;;:31;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;25848:1;25820:12;:16;25833:2;25820:16;;;;;;;;;;;;;;;:24;;;:29;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;25866:31;25900:11;:20;25912:7;25900:20;;;;;;;;;;;25866:54;;25951:2;25935:8;:13;;;:18;;;;;;;;;;;;;;;;;;26001:15;25968:8;:23;;;:49;;;;;;;;;;;;;;;;;;26269:19;26301:1;26291:7;:11;26269:33;;26317:31;26351:11;:24;26363:11;26351:24;;;;;;;;;;;26317:58;;26419:1;26394:27;;:8;:13;;;;;;;;;;;;:27;;;26390:384;;;26604:13;;26589:11;:28;26585:174;;26658:4;26642:8;:13;;;:20;;;;;;;;;;;;;;;;;;26711:13;:28;;;26685:8;:23;;;:54;;;;;;;;;;;;;;;;;;26585:174;26390:384;25749:1036;;;26821:7;26817:2;26802:27;;26811:4;26802:27;;;;;;;;;;;;26840:42;26861:4;26867:2;26871:7;26880:1;26840:20;:42::i;:::-;24864:2026;;24760:2130;;;:::o;15734:1111::-;15796:21;;:::i;:::-;15830:12;15845:7;15830:22;;15913:4;15894:15;:13;:15::i;:::-;:23;15890:888;;15930:13;;15923:4;:20;15919:859;;;15964:31;15998:11;:17;16010:4;15998:17;;;;;;;;;;;15964:51;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;16039:9;:16;;;16034:729;;16110:1;16084:28;;:9;:14;;;:28;;;16080:101;;16148:9;16141:16;;;;;;16080:101;16483:261;16490:4;16483:261;;;16523:6;;;;;;;;16568:11;:17;16580:4;16568:17;;;;;;;;;;;16556:29;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;16642:1;16616:28;;:9;:14;;;:28;;;16612:109;;16684:9;16677:16;;;;;;16612:109;16483:261;;;16034:729;15945:833;15919:859;15890:888;16806:31;;;;;;;;;;;;;;15734:1111;;;;:::o;56699:191::-;56773:16;56792:6;;;;;;;;;;;56773:25;;56818:8;56809:6;;:17;;;;;;;;;;;;;;;;;;56873:8;56842:40;;56863:8;56842:40;;;;;;;;;;;;56762:128;56699:191;:::o;20848:104::-;20917:27;20927:2;20931:8;20917:27;;;;;;;;;;;;:9;:27::i;:::-;20848:104;;:::o;59863:203::-;59926:4;59943:12;59985:10;59968:28;;;;;;;;:::i;:::-;;;;;;;;;;;;;59958:39;;;;;;59943:54;;60015:43;60034:5;60041:10;;60053:4;60015:18;:43::i;:::-;60008:50;;;59863:203;;;:::o;59072:299::-;59131:5;;59124:3;:12;;:23;;;;;59146:1;59140:3;:7;59124:23;59116:72;;;;;;;;;;;;:::i;:::-;;;;;;;;;59199:9;59238:1;59211:23;59221:12;:10;:12::i;:::-;59211:9;:23::i;:::-;:28;:36;;59246:1;59211:36;;;59242:1;59211:36;59199:48;;;;59294:4;59288:3;:10;;;;:::i;:::-;59279:5;;:20;;;;:::i;:::-;59266:9;:33;;59258:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;59337:26;59345:12;:10;:12::i;:::-;59359:3;59337:7;:26::i;:::-;59105:266;59072:299;:::o;37401:326::-;37461:4;37718:1;37696:7;:19;;;:23;37689:30;;37401:326;;;:::o;30500:667::-;30663:4;30700:2;30684:36;;;30721:12;:10;:12::i;:::-;30735:4;30741:7;30750:5;30684:72;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;30680:480;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;30935:1;30918:6;:13;:18;30914:235;;;30964:40;;;;;;;;;;;;;;30914:235;31107:6;31101:13;31092:6;31088:2;31084:15;31077:38;30680:480;30813:45;;;30803:55;;;:6;:55;;;;30796:62;;;30500:667;;;;;;:::o;58051:105::-;58103:13;58136:12;58129:19;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;58051:105;:::o;34159:723::-;34215:13;34445:1;34436:5;:10;34432:53;;;34463:10;;;;;;;;;;;;;;;;;;;;;34432:53;34495:12;34510:5;34495:20;;34526:14;34551:78;34566:1;34558:4;:9;34551:78;;34584:8;;;;;:::i;:::-;;;;34615:2;34607:10;;;;;:::i;:::-;;;34551:78;;;34639:19;34671:6;34661:17;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;34639:39;;34689:154;34705:1;34696:5;:10;34689:154;;34733:1;34723:11;;;;;:::i;:::-;;;34800:2;34792:5;:10;;;;:::i;:::-;34779:2;:24;;;;:::i;:::-;34766:39;;34749:6;34756;34749:14;;;;;;;;:::i;:::-;;;;;:56;;;;;;;;;;;34829:2;34820:11;;;;;:::i;:::-;;;34689:154;;;34867:6;34853:21;;;;;34159:723;;;;:::o;23333:1173::-;23398:20;23421:13;;23398:36;;23463:1;23449:16;;:2;:16;;;23445:48;;;23474:19;;;;;;;;;;;;;;23445:48;23520:1;23508:8;:13;23504:44;;;23530:18;;;;;;;;;;;;;;23504:44;23561:61;23591:1;23595:2;23599:12;23613:8;23561:21;:61::i;:::-;23934:8;23899:12;:16;23912:2;23899:16;;;;;;;;;;;;;;;:24;;;:44;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;23998:8;23958:12;:16;23971:2;23958:16;;;;;;;;;;;;;;;:29;;;:49;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;24057:2;24024:11;:25;24036:12;24024:25;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;24124:15;24074:11;:25;24086:12;24074:25;;;;;;;;;;;:40;;;:66;;;;;;;;;;;;;;;;;;24157:20;24180:12;24157:35;;24207:11;24236:8;24221:12;:23;24207:37;;24261:111;24313:14;;;;;;24309:2;24288:40;;24305:1;24288:40;;;;;;;;;;;;24367:3;24352:12;:18;24261:111;;24404:12;24388:13;:28;;;;23874:554;;24438:60;24467:1;24471:2;24475:12;24489:8;24438:20;:60::i;:::-;23387:1119;23333:1173;;:::o;31815:159::-;;;;;:::o;32633:158::-;;;;;:::o;21325:1749::-;21448:20;21471:13;;21448:36;;21513:1;21499:16;;:2;:16;;;21495:48;;;21524:19;;;;;;;;;;;;;;21495:48;21570:1;21558:8;:13;21554:44;;;21580:18;;;;;;;;;;;;;;21554:44;21611:61;21641:1;21645:2;21649:12;21663:8;21611:21;:61::i;:::-;21984:8;21949:12;:16;21962:2;21949:16;;;;;;;;;;;;;;;:24;;;:44;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;22048:8;22008:12;:16;22021:2;22008:16;;;;;;;;;;;;;;;:29;;;:49;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;22107:2;22074:11;:25;22086:12;22074:25;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;22174:15;22124:11;:25;22136:12;22124:25;;;;;;;;;;;:40;;;:66;;;;;;;;;;;;;;;;;;22207:20;22230:12;22207:35;;22257:11;22286:8;22271:12;:23;22257:37;;22315:15;:2;:13;;;:15::i;:::-;22311:631;;;22351:313;22407:12;22403:2;22382:38;;22399:1;22382:38;;;;;;;;;;;;22448:69;22487:1;22491:2;22495:14;;;;;;22511:5;22448:30;:69::i;:::-;22443:174;;22553:40;;;;;;;;;;;;;;22443:174;22659:3;22644:12;:18;22351:313;;22745:12;22728:13;;:29;22724:43;;22759:8;;;22724:43;22311:631;;;22808:119;22864:14;;;;;;22860:2;22839:40;;22856:1;22839:40;;;;;;;;;;;;22922:3;22907:12;:18;22808:119;;22311:631;22972:12;22956:13;:28;;;;21924:1072;;23006:60;23035:1;23039:2;23043:12;23057:8;23006:20;:60::i;:::-;21437:1637;21325:1749;;;:::o;46744:190::-;46869:4;46922;46893:25;46906:5;46913:4;46893:12;:25::i;:::-;:33;46886:40;;46744:190;;;;;:::o;47611:296::-;47694:7;47714:20;47737:4;47714:27;;47757:9;47752:118;47776:5;:12;47772:1;:16;47752:118;;;47825:33;47835:12;47849:5;47855:1;47849:8;;;;;;;;:::i;:::-;;;;;;;;47825:9;:33::i;:::-;47810:48;;47790:3;;;;;:::i;:::-;;;;47752:118;;;;47887:12;47880:19;;;47611:296;;;;:::o;53818:149::-;53881:7;53912:1;53908;:5;:51;;53939:20;53954:1;53957;53939:14;:20::i;:::-;53908:51;;;53916:20;53931:1;53934;53916:14;:20::i;:::-;53908:51;53901:58;;53818:149;;;;:::o;53975:268::-;54043:13;54150:1;54144:4;54137:15;54179:1;54173:4;54166:15;54220:4;54214;54204:21;54195:30;;53975:268;;;;:::o;-1:-1:-1:-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;:::o;24:722:1:-;120:5;145:81;161:64;218:6;161:64;:::i;:::-;145:81;:::i;:::-;136:90;;246:5;275:6;268:5;261:21;309:4;302:5;298:16;291:23;;335:6;385:3;377:4;369:6;365:17;360:3;356:27;353:36;350:143;;;404:79;;:::i;:::-;350:143;517:1;502:238;527:6;524:1;521:13;502:238;;;595:3;624:37;657:3;645:10;624:37;:::i;:::-;619:3;612:50;691:4;686:3;682:14;675:21;;725:4;720:3;716:14;709:21;;562:178;549:1;546;542:9;537:14;;502:238;;;506:14;126:620;;24:722;;;;;:::o;752:410::-;829:5;854:65;870:48;911:6;870:48;:::i;:::-;854:65;:::i;:::-;845:74;;942:6;935:5;928:21;980:4;973:5;969:16;1018:3;1009:6;1004:3;1000:16;997:25;994:112;;;1025:79;;:::i;:::-;994:112;1115:41;1149:6;1144:3;1139;1115:41;:::i;:::-;835:327;752:410;;;;;:::o;1168:139::-;1214:5;1252:6;1239:20;1230:29;;1268:33;1295:5;1268:33;:::i;:::-;1168:139;;;;:::o;1330:568::-;1403:8;1413:6;1463:3;1456:4;1448:6;1444:17;1440:27;1430:122;;1471:79;;:::i;:::-;1430:122;1584:6;1571:20;1561:30;;1614:18;1606:6;1603:30;1600:117;;;1636:79;;:::i;:::-;1600:117;1750:4;1742:6;1738:17;1726:29;;1804:3;1796:4;1788:6;1784:17;1774:8;1770:32;1767:41;1764:128;;;1811:79;;:::i;:::-;1764:128;1330:568;;;;;:::o;1921:370::-;1992:5;2041:3;2034:4;2026:6;2022:17;2018:27;2008:122;;2049:79;;:::i;:::-;2008:122;2166:6;2153:20;2191:94;2281:3;2273:6;2266:4;2258:6;2254:17;2191:94;:::i;:::-;2182:103;;1998:293;1921:370;;;;:::o;2297:133::-;2340:5;2378:6;2365:20;2356:29;;2394:30;2418:5;2394:30;:::i;:::-;2297:133;;;;:::o;2436:139::-;2482:5;2520:6;2507:20;2498:29;;2536:33;2563:5;2536:33;:::i;:::-;2436:139;;;;:::o;2581:137::-;2626:5;2664:6;2651:20;2642:29;;2680:32;2706:5;2680:32;:::i;:::-;2581:137;;;;:::o;2724:141::-;2780:5;2811:6;2805:13;2796:22;;2827:32;2853:5;2827:32;:::i;:::-;2724:141;;;;:::o;2884:338::-;2939:5;2988:3;2981:4;2973:6;2969:17;2965:27;2955:122;;2996:79;;:::i;:::-;2955:122;3113:6;3100:20;3138:78;3212:3;3204:6;3197:4;3189:6;3185:17;3138:78;:::i;:::-;3129:87;;2945:277;2884:338;;;;:::o;3242:553::-;3300:8;3310:6;3360:3;3353:4;3345:6;3341:17;3337:27;3327:122;;3368:79;;:::i;:::-;3327:122;3481:6;3468:20;3458:30;;3511:18;3503:6;3500:30;3497:117;;;3533:79;;:::i;:::-;3497:117;3647:4;3639:6;3635:17;3623:29;;3701:3;3693:4;3685:6;3681:17;3671:8;3667:32;3664:41;3661:128;;;3708:79;;:::i;:::-;3661:128;3242:553;;;;;:::o;3801:139::-;3847:5;3885:6;3872:20;3863:29;;3901:33;3928:5;3901:33;:::i;:::-;3801:139;;;;:::o;3946:329::-;4005:6;4054:2;4042:9;4033:7;4029:23;4025:32;4022:119;;;4060:79;;:::i;:::-;4022:119;4180:1;4205:53;4250:7;4241:6;4230:9;4226:22;4205:53;:::i;:::-;4195:63;;4151:117;3946:329;;;;:::o;4281:474::-;4349:6;4357;4406:2;4394:9;4385:7;4381:23;4377:32;4374:119;;;4412:79;;:::i;:::-;4374:119;4532:1;4557:53;4602:7;4593:6;4582:9;4578:22;4557:53;:::i;:::-;4547:63;;4503:117;4659:2;4685:53;4730:7;4721:6;4710:9;4706:22;4685:53;:::i;:::-;4675:63;;4630:118;4281:474;;;;;:::o;4761:619::-;4838:6;4846;4854;4903:2;4891:9;4882:7;4878:23;4874:32;4871:119;;;4909:79;;:::i;:::-;4871:119;5029:1;5054:53;5099:7;5090:6;5079:9;5075:22;5054:53;:::i;:::-;5044:63;;5000:117;5156:2;5182:53;5227:7;5218:6;5207:9;5203:22;5182:53;:::i;:::-;5172:63;;5127:118;5284:2;5310:53;5355:7;5346:6;5335:9;5331:22;5310:53;:::i;:::-;5300:63;;5255:118;4761:619;;;;;:::o;5386:943::-;5481:6;5489;5497;5505;5554:3;5542:9;5533:7;5529:23;5525:33;5522:120;;;5561:79;;:::i;:::-;5522:120;5681:1;5706:53;5751:7;5742:6;5731:9;5727:22;5706:53;:::i;:::-;5696:63;;5652:117;5808:2;5834:53;5879:7;5870:6;5859:9;5855:22;5834:53;:::i;:::-;5824:63;;5779:118;5936:2;5962:53;6007:7;5998:6;5987:9;5983:22;5962:53;:::i;:::-;5952:63;;5907:118;6092:2;6081:9;6077:18;6064:32;6123:18;6115:6;6112:30;6109:117;;;6145:79;;:::i;:::-;6109:117;6250:62;6304:7;6295:6;6284:9;6280:22;6250:62;:::i;:::-;6240:72;;6035:287;5386:943;;;;;;;:::o;6335:468::-;6400:6;6408;6457:2;6445:9;6436:7;6432:23;6428:32;6425:119;;;6463:79;;:::i;:::-;6425:119;6583:1;6608:53;6653:7;6644:6;6633:9;6629:22;6608:53;:::i;:::-;6598:63;;6554:117;6710:2;6736:50;6778:7;6769:6;6758:9;6754:22;6736:50;:::i;:::-;6726:60;;6681:115;6335:468;;;;;:::o;6809:474::-;6877:6;6885;6934:2;6922:9;6913:7;6909:23;6905:32;6902:119;;;6940:79;;:::i;:::-;6902:119;7060:1;7085:53;7130:7;7121:6;7110:9;7106:22;7085:53;:::i;:::-;7075:63;;7031:117;7187:2;7213:53;7258:7;7249:6;7238:9;7234:22;7213:53;:::i;:::-;7203:63;;7158:118;6809:474;;;;;:::o;7289:559::-;7375:6;7383;7432:2;7420:9;7411:7;7407:23;7403:32;7400:119;;;7438:79;;:::i;:::-;7400:119;7586:1;7575:9;7571:17;7558:31;7616:18;7608:6;7605:30;7602:117;;;7638:79;;:::i;:::-;7602:117;7751:80;7823:7;7814:6;7803:9;7799:22;7751:80;:::i;:::-;7733:98;;;;7529:312;7289:559;;;;;:::o;7854:329::-;7913:6;7962:2;7950:9;7941:7;7937:23;7933:32;7930:119;;;7968:79;;:::i;:::-;7930:119;8088:1;8113:53;8158:7;8149:6;8138:9;8134:22;8113:53;:::i;:::-;8103:63;;8059:117;7854:329;;;;:::o;8189:327::-;8247:6;8296:2;8284:9;8275:7;8271:23;8267:32;8264:119;;;8302:79;;:::i;:::-;8264:119;8422:1;8447:52;8491:7;8482:6;8471:9;8467:22;8447:52;:::i;:::-;8437:62;;8393:116;8189:327;;;;:::o;8522:349::-;8591:6;8640:2;8628:9;8619:7;8615:23;8611:32;8608:119;;;8646:79;;:::i;:::-;8608:119;8766:1;8791:63;8846:7;8837:6;8826:9;8822:22;8791:63;:::i;:::-;8781:73;;8737:127;8522:349;;;;:::o;8877:529::-;8948:6;8956;9005:2;8993:9;8984:7;8980:23;8976:32;8973:119;;;9011:79;;:::i;:::-;8973:119;9159:1;9148:9;9144:17;9131:31;9189:18;9181:6;9178:30;9175:117;;;9211:79;;:::i;:::-;9175:117;9324:65;9381:7;9372:6;9361:9;9357:22;9324:65;:::i;:::-;9306:83;;;;9102:297;8877:529;;;;;:::o;9412:329::-;9471:6;9520:2;9508:9;9499:7;9495:23;9491:32;9488:119;;;9526:79;;:::i;:::-;9488:119;9646:1;9671:53;9716:7;9707:6;9696:9;9692:22;9671:53;:::i;:::-;9661:63;;9617:117;9412:329;;;;:::o;9747:684::-;9840:6;9848;9897:2;9885:9;9876:7;9872:23;9868:32;9865:119;;;9903:79;;:::i;:::-;9865:119;10023:1;10048:53;10093:7;10084:6;10073:9;10069:22;10048:53;:::i;:::-;10038:63;;9994:117;10178:2;10167:9;10163:18;10150:32;10209:18;10201:6;10198:30;10195:117;;;10231:79;;:::i;:::-;10195:117;10336:78;10406:7;10397:6;10386:9;10382:22;10336:78;:::i;:::-;10326:88;;10121:303;9747:684;;;;;:::o;10437:118::-;10524:24;10542:5;10524:24;:::i;:::-;10519:3;10512:37;10437:118;;:::o;10561:157::-;10666:45;10686:24;10704:5;10686:24;:::i;:::-;10666:45;:::i;:::-;10661:3;10654:58;10561:157;;:::o;10724:109::-;10805:21;10820:5;10805:21;:::i;:::-;10800:3;10793:34;10724:109;;:::o;10839:118::-;10926:24;10944:5;10926:24;:::i;:::-;10921:3;10914:37;10839:118;;:::o;10963:360::-;11049:3;11077:38;11109:5;11077:38;:::i;:::-;11131:70;11194:6;11189:3;11131:70;:::i;:::-;11124:77;;11210:52;11255:6;11250:3;11243:4;11236:5;11232:16;11210:52;:::i;:::-;11287:29;11309:6;11287:29;:::i;:::-;11282:3;11278:39;11271:46;;11053:270;10963:360;;;;:::o;11329:364::-;11417:3;11445:39;11478:5;11445:39;:::i;:::-;11500:71;11564:6;11559:3;11500:71;:::i;:::-;11493:78;;11580:52;11625:6;11620:3;11613:4;11606:5;11602:16;11580:52;:::i;:::-;11657:29;11679:6;11657:29;:::i;:::-;11652:3;11648:39;11641:46;;11421:272;11329:364;;;;:::o;11699:377::-;11805:3;11833:39;11866:5;11833:39;:::i;:::-;11888:89;11970:6;11965:3;11888:89;:::i;:::-;11881:96;;11986:52;12031:6;12026:3;12019:4;12012:5;12008:16;11986:52;:::i;:::-;12063:6;12058:3;12054:16;12047:23;;11809:267;11699:377;;;;:::o;12082:366::-;12224:3;12245:67;12309:2;12304:3;12245:67;:::i;:::-;12238:74;;12321:93;12410:3;12321:93;:::i;:::-;12439:2;12434:3;12430:12;12423:19;;12082:366;;;:::o;12454:::-;12596:3;12617:67;12681:2;12676:3;12617:67;:::i;:::-;12610:74;;12693:93;12782:3;12693:93;:::i;:::-;12811:2;12806:3;12802:12;12795:19;;12454:366;;;:::o;12826:::-;12968:3;12989:67;13053:2;13048:3;12989:67;:::i;:::-;12982:74;;13065:93;13154:3;13065:93;:::i;:::-;13183:2;13178:3;13174:12;13167:19;;12826:366;;;:::o;13198:::-;13340:3;13361:67;13425:2;13420:3;13361:67;:::i;:::-;13354:74;;13437:93;13526:3;13437:93;:::i;:::-;13555:2;13550:3;13546:12;13539:19;;13198:366;;;:::o;13570:::-;13712:3;13733:67;13797:2;13792:3;13733:67;:::i;:::-;13726:74;;13809:93;13898:3;13809:93;:::i;:::-;13927:2;13922:3;13918:12;13911:19;;13570:366;;;:::o;13942:::-;14084:3;14105:67;14169:2;14164:3;14105:67;:::i;:::-;14098:74;;14181:93;14270:3;14181:93;:::i;:::-;14299:2;14294:3;14290:12;14283:19;;13942:366;;;:::o;14314:::-;14456:3;14477:67;14541:2;14536:3;14477:67;:::i;:::-;14470:74;;14553:93;14642:3;14553:93;:::i;:::-;14671:2;14666:3;14662:12;14655:19;;14314:366;;;:::o;14686:::-;14828:3;14849:67;14913:2;14908:3;14849:67;:::i;:::-;14842:74;;14925:93;15014:3;14925:93;:::i;:::-;15043:2;15038:3;15034:12;15027:19;;14686:366;;;:::o;15058:118::-;15145:24;15163:5;15145:24;:::i;:::-;15140:3;15133:37;15058:118;;:::o;15182:256::-;15294:3;15309:75;15380:3;15371:6;15309:75;:::i;:::-;15409:2;15404:3;15400:12;15393:19;;15429:3;15422:10;;15182:256;;;;:::o;15444:435::-;15624:3;15646:95;15737:3;15728:6;15646:95;:::i;:::-;15639:102;;15758:95;15849:3;15840:6;15758:95;:::i;:::-;15751:102;;15870:3;15863:10;;15444:435;;;;;:::o;15885:222::-;15978:4;16016:2;16005:9;16001:18;15993:26;;16029:71;16097:1;16086:9;16082:17;16073:6;16029:71;:::i;:::-;15885:222;;;;:::o;16113:640::-;16308:4;16346:3;16335:9;16331:19;16323:27;;16360:71;16428:1;16417:9;16413:17;16404:6;16360:71;:::i;:::-;16441:72;16509:2;16498:9;16494:18;16485:6;16441:72;:::i;:::-;16523;16591:2;16580:9;16576:18;16567:6;16523:72;:::i;:::-;16642:9;16636:4;16632:20;16627:2;16616:9;16612:18;16605:48;16670:76;16741:4;16732:6;16670:76;:::i;:::-;16662:84;;16113:640;;;;;;;:::o;16759:210::-;16846:4;16884:2;16873:9;16869:18;16861:26;;16897:65;16959:1;16948:9;16944:17;16935:6;16897:65;:::i;:::-;16759:210;;;;:::o;16975:222::-;17068:4;17106:2;17095:9;17091:18;17083:26;;17119:71;17187:1;17176:9;17172:17;17163:6;17119:71;:::i;:::-;16975:222;;;;:::o;17203:313::-;17316:4;17354:2;17343:9;17339:18;17331:26;;17403:9;17397:4;17393:20;17389:1;17378:9;17374:17;17367:47;17431:78;17504:4;17495:6;17431:78;:::i;:::-;17423:86;;17203:313;;;;:::o;17522:419::-;17688:4;17726:2;17715:9;17711:18;17703:26;;17775:9;17769:4;17765:20;17761:1;17750:9;17746:17;17739:47;17803:131;17929:4;17803:131;:::i;:::-;17795:139;;17522:419;;;:::o;17947:::-;18113:4;18151:2;18140:9;18136:18;18128:26;;18200:9;18194:4;18190:20;18186:1;18175:9;18171:17;18164:47;18228:131;18354:4;18228:131;:::i;:::-;18220:139;;17947:419;;;:::o;18372:::-;18538:4;18576:2;18565:9;18561:18;18553:26;;18625:9;18619:4;18615:20;18611:1;18600:9;18596:17;18589:47;18653:131;18779:4;18653:131;:::i;:::-;18645:139;;18372:419;;;:::o;18797:::-;18963:4;19001:2;18990:9;18986:18;18978:26;;19050:9;19044:4;19040:20;19036:1;19025:9;19021:17;19014:47;19078:131;19204:4;19078:131;:::i;:::-;19070:139;;18797:419;;;:::o;19222:::-;19388:4;19426:2;19415:9;19411:18;19403:26;;19475:9;19469:4;19465:20;19461:1;19450:9;19446:17;19439:47;19503:131;19629:4;19503:131;:::i;:::-;19495:139;;19222:419;;;:::o;19647:::-;19813:4;19851:2;19840:9;19836:18;19828:26;;19900:9;19894:4;19890:20;19886:1;19875:9;19871:17;19864:47;19928:131;20054:4;19928:131;:::i;:::-;19920:139;;19647:419;;;:::o;20072:::-;20238:4;20276:2;20265:9;20261:18;20253:26;;20325:9;20319:4;20315:20;20311:1;20300:9;20296:17;20289:47;20353:131;20479:4;20353:131;:::i;:::-;20345:139;;20072:419;;;:::o;20497:::-;20663:4;20701:2;20690:9;20686:18;20678:26;;20750:9;20744:4;20740:20;20736:1;20725:9;20721:17;20714:47;20778:131;20904:4;20778:131;:::i;:::-;20770:139;;20497:419;;;:::o;20922:222::-;21015:4;21053:2;21042:9;21038:18;21030:26;;21066:71;21134:1;21123:9;21119:17;21110:6;21066:71;:::i;:::-;20922:222;;;;:::o;21150:129::-;21184:6;21211:20;;:::i;:::-;21201:30;;21240:33;21268:4;21260:6;21240:33;:::i;:::-;21150:129;;;:::o;21285:75::-;21318:6;21351:2;21345:9;21335:19;;21285:75;:::o;21366:311::-;21443:4;21533:18;21525:6;21522:30;21519:56;;;21555:18;;:::i;:::-;21519:56;21605:4;21597:6;21593:17;21585:25;;21665:4;21659;21655:15;21647:23;;21366:311;;;:::o;21683:307::-;21744:4;21834:18;21826:6;21823:30;21820:56;;;21856:18;;:::i;:::-;21820:56;21894:29;21916:6;21894:29;:::i;:::-;21886:37;;21978:4;21972;21968:15;21960:23;;21683:307;;;:::o;21996:98::-;22047:6;22081:5;22075:12;22065:22;;21996:98;;;:::o;22100:99::-;22152:6;22186:5;22180:12;22170:22;;22100:99;;;:::o;22205:168::-;22288:11;22322:6;22317:3;22310:19;22362:4;22357:3;22353:14;22338:29;;22205:168;;;;:::o;22379:169::-;22463:11;22497:6;22492:3;22485:19;22537:4;22532:3;22528:14;22513:29;;22379:169;;;;:::o;22554:148::-;22656:11;22693:3;22678:18;;22554:148;;;;:::o;22708:305::-;22748:3;22767:20;22785:1;22767:20;:::i;:::-;22762:25;;22801:20;22819:1;22801:20;:::i;:::-;22796:25;;22955:1;22887:66;22883:74;22880:1;22877:81;22874:107;;;22961:18;;:::i;:::-;22874:107;23005:1;23002;22998:9;22991:16;;22708:305;;;;:::o;23019:185::-;23059:1;23076:20;23094:1;23076:20;:::i;:::-;23071:25;;23110:20;23128:1;23110:20;:::i;:::-;23105:25;;23149:1;23139:35;;23154:18;;:::i;:::-;23139:35;23196:1;23193;23189:9;23184:14;;23019:185;;;;:::o;23210:348::-;23250:7;23273:20;23291:1;23273:20;:::i;:::-;23268:25;;23307:20;23325:1;23307:20;:::i;:::-;23302:25;;23495:1;23427:66;23423:74;23420:1;23417:81;23412:1;23405:9;23398:17;23394:105;23391:131;;;23502:18;;:::i;:::-;23391:131;23550:1;23547;23543:9;23532:20;;23210:348;;;;:::o;23564:191::-;23604:4;23624:20;23642:1;23624:20;:::i;:::-;23619:25;;23658:20;23676:1;23658:20;:::i;:::-;23653:25;;23697:1;23694;23691:8;23688:34;;;23702:18;;:::i;:::-;23688:34;23747:1;23744;23740:9;23732:17;;23564:191;;;;:::o;23761:96::-;23798:7;23827:24;23845:5;23827:24;:::i;:::-;23816:35;;23761:96;;;:::o;23863:90::-;23897:7;23940:5;23933:13;23926:21;23915:32;;23863:90;;;:::o;23959:77::-;23996:7;24025:5;24014:16;;23959:77;;;:::o;24042:149::-;24078:7;24118:66;24111:5;24107:78;24096:89;;24042:149;;;:::o;24197:126::-;24234:7;24274:42;24267:5;24263:54;24252:65;;24197:126;;;:::o;24329:77::-;24366:7;24395:5;24384:16;;24329:77;;;:::o;24412:154::-;24496:6;24491:3;24486;24473:30;24558:1;24549:6;24544:3;24540:16;24533:27;24412:154;;;:::o;24572:307::-;24640:1;24650:113;24664:6;24661:1;24658:13;24650:113;;;24749:1;24744:3;24740:11;24734:18;24730:1;24725:3;24721:11;24714:39;24686:2;24683:1;24679:10;24674:15;;24650:113;;;24781:6;24778:1;24775:13;24772:101;;;24861:1;24852:6;24847:3;24843:16;24836:27;24772:101;24621:258;24572:307;;;:::o;24885:320::-;24929:6;24966:1;24960:4;24956:12;24946:22;;25013:1;25007:4;25003:12;25034:18;25024:81;;25090:4;25082:6;25078:17;25068:27;;25024:81;25152:2;25144:6;25141:14;25121:18;25118:38;25115:84;;;25171:18;;:::i;:::-;25115:84;24936:269;24885:320;;;:::o;25211:281::-;25294:27;25316:4;25294:27;:::i;:::-;25286:6;25282:40;25424:6;25412:10;25409:22;25388:18;25376:10;25373:34;25370:62;25367:88;;;25435:18;;:::i;:::-;25367:88;25475:10;25471:2;25464:22;25254:238;25211:281;;:::o;25498:233::-;25537:3;25560:24;25578:5;25560:24;:::i;:::-;25551:33;;25606:66;25599:5;25596:77;25593:103;;;25676:18;;:::i;:::-;25593:103;25723:1;25716:5;25712:13;25705:20;;25498:233;;;:::o;25737:100::-;25776:7;25805:26;25825:5;25805:26;:::i;:::-;25794:37;;25737:100;;;:::o;25843:94::-;25882:7;25911:20;25925:5;25911:20;:::i;:::-;25900:31;;25843:94;;;:::o;25943:176::-;25975:1;25992:20;26010:1;25992:20;:::i;:::-;25987:25;;26026:20;26044:1;26026:20;:::i;:::-;26021:25;;26065:1;26055:35;;26070:18;;:::i;:::-;26055:35;26111:1;26108;26104:9;26099:14;;25943:176;;;;:::o;26125:180::-;26173:77;26170:1;26163:88;26270:4;26267:1;26260:15;26294:4;26291:1;26284:15;26311:180;26359:77;26356:1;26349:88;26456:4;26453:1;26446:15;26480:4;26477:1;26470:15;26497:180;26545:77;26542:1;26535:88;26642:4;26639:1;26632:15;26666:4;26663:1;26656:15;26683:180;26731:77;26728:1;26721:88;26828:4;26825:1;26818:15;26852:4;26849:1;26842:15;26869:180;26917:77;26914:1;26907:88;27014:4;27011:1;27004:15;27038:4;27035:1;27028:15;27055:117;27164:1;27161;27154:12;27178:117;27287:1;27284;27277:12;27301:117;27410:1;27407;27400:12;27424:117;27533:1;27530;27523:12;27547:117;27656:1;27653;27646:12;27670:117;27779:1;27776;27769:12;27793:102;27834:6;27885:2;27881:7;27876:2;27869:5;27865:14;27861:28;27851:38;;27793:102;;;:::o;27901:94::-;27934:8;27982:5;27978:2;27974:14;27953:35;;27901:94;;;:::o;28001:225::-;28141:34;28137:1;28129:6;28125:14;28118:58;28210:8;28205:2;28197:6;28193:15;28186:33;28001:225;:::o;28232:223::-;28372:34;28368:1;28360:6;28356:14;28349:58;28441:6;28436:2;28428:6;28424:15;28417:31;28232:223;:::o;28461:172::-;28601:24;28597:1;28589:6;28585:14;28578:48;28461:172;:::o;28639:162::-;28779:14;28775:1;28767:6;28763:14;28756:38;28639:162;:::o;28807:174::-;28947:26;28943:1;28935:6;28931:14;28924:50;28807:174;:::o;28987:182::-;29127:34;29123:1;29115:6;29111:14;29104:58;28987:182;:::o;29175:220::-;29315:34;29311:1;29303:6;29299:14;29292:58;29384:3;29379:2;29371:6;29367:15;29360:28;29175:220;:::o;29401:221::-;29541:34;29537:1;29529:6;29525:14;29518:58;29610:4;29605:2;29597:6;29593:15;29586:29;29401:221;:::o;29628:122::-;29701:24;29719:5;29701:24;:::i;:::-;29694:5;29691:35;29681:63;;29740:1;29737;29730:12;29681:63;29628:122;:::o;29756:116::-;29826:21;29841:5;29826:21;:::i;:::-;29819:5;29816:32;29806:60;;29862:1;29859;29852:12;29806:60;29756:116;:::o;29878:122::-;29951:24;29969:5;29951:24;:::i;:::-;29944:5;29941:35;29931:63;;29990:1;29987;29980:12;29931:63;29878:122;:::o;30006:120::-;30078:23;30095:5;30078:23;:::i;:::-;30071:5;30068:34;30058:62;;30116:1;30113;30106:12;30058:62;30006:120;:::o;30132:122::-;30205:24;30223:5;30205:24;:::i;:::-;30198:5;30195:35;30185:63;;30244:1;30241;30234:12;30185:63;30132:122;:::o

Swarm Source

ipfs://5f766974bb1902178008499c747facb6cd6f5b851bd251b9c1c132d29946105d
Loading...
Loading
Loading...
Loading
[ Download: CSV Export  ]
[ 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.