ETH Price: $3,156.98 (+1.80%)

Token

Doodle Monkey (MONKEY)
 

Overview

Max Total Supply

1,222 MONKEY

Holders

359

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Filtered by Token Holder
excdark.eth
Balance
3 MONKEY
0x46c0a29d668f730425a9da8a5a161c860fbeda16
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:
DoodleMonkey

Compiler Version
v0.8.7+commit.e28d00a7

Optimization Enabled:
No with 200 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity Multiple files format)

File 3 of 14: DoodleMonkey.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

import "./ERC721A.sol";

contract DoodleMonkey is ERC721A {

    constructor(string memory baseURI) ERC721A("Doodle Monkey", "MONKEY") {
        setBaseURI(baseURI);
    }

    uint256 public constant MAX_SUPPLY = 1222;

    uint256 private mintCount = 0;

    uint256 public price = 29000000000000000;

    string private baseTokenURI;
      
    bool public saleOpen = false;

    event Minted(uint256 totalMinted);
      
    function totalSupply() public view override returns (uint256) {
        return mintCount;
    }

    function setBaseURI(string memory baseURI) public onlyOwner {
        baseTokenURI = baseURI;
    }

    function changePrice(uint256 _newPrice) external onlyOwner {
        price = _newPrice;
    }

    function flipSale() external onlyOwner {
        saleOpen = !saleOpen;
    }

    function withdraw() external onlyOwner {
        (bool success, ) = payable(msg.sender).call{value: address(this).balance}("");
        require(success, "Transfer failed.");
    }

    function mint(address _to, uint256 _count) external payable {
        uint256 supply = totalSupply();

        require(supply + _count <= MAX_SUPPLY, "Exceeds maximum supply");
        require(_count > 0, "Minimum 1 NFT has to be minted per transaction");

        if (msg.sender != owner()) {
            require(saleOpen, "Sale is not open yet");
            require(
                _count <= 15,
                "Maximum 15 NFTs can be minted per transaction"
            );
            require(
                msg.value >= price * _count,
                "Ether sent with this transaction is not correct"
            );
        }
        mintCount += _count;      
        _safeMint(_to, _count);
        emit Minted(_count);       
    }

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

File 1 of 14: Address.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (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 functionCall(target, data, "Address: low-level call failed");
    }

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

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

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

        (bool success, bytes memory returndata) = target.call{value: value}(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

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

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal view returns (bytes memory) {
        require(isContract(target), "Address: static call to non-contract");

        (bool success, bytes memory returndata) = target.staticcall(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

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

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(
        address target,
        bytes memory data,
        string memory errorMessage
    ) internal returns (bytes memory) {
        require(isContract(target), "Address: delegate call to non-contract");

        (bool success, bytes memory returndata) = target.delegatecall(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
     * revert reason using the provided one.
     *
     * _Available since v4.3._
     */
    function verifyCallResult(
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) internal pure returns (bytes memory) {
        if (success) {
            return returndata;
        } else {
            // Look for revert reason and bubble it up if present
            if (returndata.length > 0) {
                // The easiest way to bubble the revert reason is using memory via assembly

                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}

File 2 of 14: Context.sol
// SPDX-License-Identifier: MIT

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) {
        this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
        return msg.data;
    }
}

File 4 of 14: ERC165.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)

pragma solidity ^0.8.0;

import "./IERC165.sol";

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

File 5 of 14: ERC721.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/ERC721.sol)

pragma solidity ^0.8.0;

import "./IERC721.sol";
import "./IERC721Receiver.sol";
import "./IERC721Metadata.sol";
import "./Address.sol";
import "./Context.sol";
import "./Strings.sol";
import "./ERC165.sol";

/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
 * the Metadata extension, but not including the Enumerable extension, which is available separately as
 * {ERC721Enumerable}.
 */
contract ERC721 is Context, ERC165, IERC721, IERC721Metadata {
    using Address for address;
    using Strings for uint256;

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to owner address
    mapping(uint256 => address) private _owners;

    // Mapping owner address to token count
    mapping(address => uint256) private _balances;

    // Mapping from token ID to approved address
    mapping(uint256 => address) private _tokenApprovals;

    // Mapping from owner to operator approvals
    mapping(address => mapping(address => bool)) private _operatorApprovals;

    /**
     * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.
     */
    constructor(string memory name_, string memory symbol_) {
        _name = name_;
        _symbol = symbol_;
    }

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

    /**
     * @dev See {IERC721-balanceOf}.
     */
    function balanceOf(address owner) public view virtual override returns (uint256) {
        require(owner != address(0), "ERC721: balance query for the zero address");
        return _balances[owner];
    }

    /**
     * @dev See {IERC721-ownerOf}.
     */
    function ownerOf(uint256 tokenId) public view virtual override returns (address) {
        address owner = _owners[tokenId];
        require(owner != address(0), "ERC721: owner query for nonexistent token");
        return owner;
    }

    /**
     * @dev See {IERC721Metadata-name}.
     */
    function name() 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) {
        require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");

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

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

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

        require(
            _msgSender() == owner || isApprovedForAll(owner, _msgSender()),
            "ERC721: approve caller is not owner nor approved for all"
        );
        if (to.isContract()) {
            revert ("Token transfer to contract address is not allowed.");
        } else {
            _approve(to, tokenId);
        }
        // _approve(to, tokenId);
    }

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

        return _tokenApprovals[tokenId];
    }

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

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

    /**
     * @dev See {IERC721-transferFrom}.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        //solhint-disable-next-line max-line-length
        require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");

        _transfer(from, to, tokenId);
    }

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

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) public virtual override {
        require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
        _safeTransfer(from, to, tokenId, _data);
    }

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
     * are aware of the ERC721 protocol to prevent tokens from being forever locked.
     *
     * `_data` is additional data, it has no specified format and it is sent in call to `to`.
     *
     * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.
     * implement alternative mechanisms to perform token transfer, such as signature-based.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function _safeTransfer(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) internal virtual {
        _transfer(from, to, tokenId);
        require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer");
    }

    /**
     * @dev Returns whether `tokenId` exists.
     *
     * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.
     *
     * Tokens start existing when they are minted (`_mint`),
     * and stop existing when they are burned (`_burn`).
     */
    function _exists(uint256 tokenId) internal view virtual returns (bool) {
        return _owners[tokenId] != address(0);
    }

    /**
     * @dev Returns whether `spender` is allowed to manage `tokenId`.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {
        require(_exists(tokenId), "ERC721: operator query for nonexistent token");
        address owner = ERC721.ownerOf(tokenId);
        return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender));
    }

    /**
     * @dev Safely mints `tokenId` and transfers it to `to`.
     *
     * Requirements:
     *
     * - `tokenId` must not exist.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function _safeMint(address to, uint256 tokenId) internal virtual {
        _safeMint(to, tokenId, "");
    }

    /**
     * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is
     * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.
     */
    function _safeMint(
        address to,
        uint256 tokenId,
        bytes memory _data
    ) internal virtual {
        _mint(to, tokenId);
        require(
            _checkOnERC721Received(address(0), to, tokenId, _data),
            "ERC721: transfer to non ERC721Receiver implementer"
        );
    }

    /**
     * @dev Mints `tokenId` and transfers it to `to`.
     *
     * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible
     *
     * Requirements:
     *
     * - `tokenId` must not exist.
     * - `to` cannot be the zero address.
     *
     * Emits a {Transfer} event.
     */
    function _mint(address to, uint256 tokenId) internal virtual {
        require(to != address(0), "ERC721: mint to the zero address");
        require(!_exists(tokenId), "ERC721: token already minted");

        _beforeTokenTransfer(address(0), to, tokenId);

        _balances[to] += 1;
        _owners[tokenId] = to;

        emit Transfer(address(0), to, tokenId);

        _afterTokenTransfer(address(0), to, tokenId);
    }

    /**
     * @dev Destroys `tokenId`.
     * The approval is cleared when the token is burned.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     *
     * Emits a {Transfer} event.
     */
    function _burn(uint256 tokenId) internal virtual {
        address owner = ERC721.ownerOf(tokenId);

        _beforeTokenTransfer(owner, address(0), tokenId);

        // Clear approvals
        _approve(address(0), tokenId);

        _balances[owner] -= 1;
        delete _owners[tokenId];

        emit Transfer(owner, address(0), tokenId);

        _afterTokenTransfer(owner, address(0), tokenId);
    }

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *  As opposed to {transferFrom}, this imposes no restrictions on msg.sender.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     *
     * Emits a {Transfer} event.
     */
    function _transfer(
        address from,
        address to,
        uint256 tokenId
    ) internal virtual {
        require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner");
        require(to != address(0), "ERC721: transfer to the zero address");

        _beforeTokenTransfer(from, to, tokenId);

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

        _balances[from] -= 1;
        _balances[to] += 1;
        _owners[tokenId] = to;

        emit Transfer(from, to, tokenId);

        _afterTokenTransfer(from, to, tokenId);
    }

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

    /**
     * @dev Approve `operator` to operate on all of `owner` tokens
     *
     * Emits a {ApprovalForAll} event.
     */
    function _setApprovalForAll(
        address owner,
        address operator,
        bool approved
    ) internal virtual {
        require(owner != operator, "ERC721: approve to caller");
        _operatorApprovals[owner][operator] = approved;
        emit ApprovalForAll(owner, operator, approved);
    }

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
     * The call is not executed if the target address is not a contract.
     *
     * @param from address representing the previous owner of the given token ID
     * @param to target address that will receive the tokens
     * @param tokenId uint256 ID of the token to be transferred
     * @param _data bytes optional data to send along with the call
     * @return bool whether the call correctly returned the expected magic value
     */
    function _checkOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        if (to.isContract()) {
            try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) {
                return retval == IERC721Receiver.onERC721Received.selector;
            } catch (bytes memory reason) {
                if (reason.length == 0) {
                    revert("ERC721: transfer to non ERC721Receiver implementer");
                } else {
                    assembly {
                        revert(add(32, reason), mload(reason))
                    }
                }
            }
        } else {
            return true;
        }
    }

    /**
     * @dev Hook that is called before any token transfer. This includes minting
     * and burning.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, ``from``'s `tokenId` will be burned.
     * - `from` and `to` are never both zero.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _beforeTokenTransfer(
        address from,
        address to,
        uint256 tokenId
    ) internal virtual {}

    /**
     * @dev Hook that is called after any transfer of tokens. This includes
     * minting and burning.
     *
     * Calling conditions:
     *
     * - when `from` and `to` are both non-zero.
     * - `from` and `to` are never both zero.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _afterTokenTransfer(
        address from,
        address to,
        uint256 tokenId
    ) internal virtual {}
}

File 6 of 14: ERC721A.sol
// SPDX-License-Identifier: MIT


pragma solidity ^0.8.0;

import './IERC721.sol';
import './IERC721Receiver.sol';
import './IERC721Metadata.sol';
import './IERC721Enumerable.sol';
import './Address.sol';
import './Context.sol';
import './Strings.sol';
import './ERC165.sol';
import './Ownable.sol';

error ApprovalCallerNotOwnerNorApproved();
error ApprovalQueryForNonexistentToken();
error ApproveToCaller();
error ApprovalToCurrentOwner();
error BalanceQueryForZeroAddress();
error MintedQueryForZeroAddress();
error BurnedQueryForZeroAddress();
error MintToZeroAddress();
error MintZeroQuantity();
error OwnerIndexOutOfBounds();
error OwnerQueryForNonexistentToken();
error TokenIndexOutOfBounds();
error TransferCallerNotOwnerNorApproved();
error TransferFromIncorrectOwner();
error TransferToNonERC721ReceiverImplementer();
error TransferToZeroAddress();
error URIQueryForNonexistentToken();

/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
 * the Metadata and Enumerable extension. Built to optimize for lower gas during batch mints.
 *
 * Assumes serials are sequentially minted starting at 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, IERC721, IERC721Metadata, IERC721Enumerable, Ownable {
    using Address for address;
    using Strings for uint256;

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

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

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

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    //Allow all tokens to transfer to contract
    bool public allowedToContract = false;

    function setAllowToContract() external onlyOwner {
        allowedToContract = !allowedToContract;
    }

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

    // Mapping token to allow to transfer to contract
    mapping(uint256 => bool) public _transferToContract;

    function setAllowTokenToContract(uint256 _tokenId, bool _allow) external onlyOwner {
        _transferToContract[_tokenId] = _allow;
    }

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

    /**
     * @dev See {IERC721Enumerable-totalSupply}.
     */
    function totalSupply() public view virtual override returns (uint256) {
        // Counter underflow is impossible as _burnCounter cannot be incremented
        // more than _currentIndex times
        unchecked {
            return _currentIndex - _burnCounter;    
        }
    }

    /**
     * @dev See {IERC721Enumerable-tokenByIndex}.
     * This read function is O(totalSupply). If calling from a separate contract, be sure to test gas first.
     * It may also degrade with extremely large collection sizes (e.g >> 10000), test for your use case.
     */
    function tokenByIndex(uint256 index) public view override returns (uint256) {
        uint256 numMintedSoFar = _currentIndex;
        uint256 tokenIdsIdx;

        // Counter overflow is impossible as the loop breaks when
        // uint256 i is equal to another uint256 numMintedSoFar.
        unchecked {
            for (uint256 i; i < numMintedSoFar; i++) {
                TokenOwnership memory ownership = _ownerships[i];
                if (!ownership.burned) {
                    if (tokenIdsIdx == index) {
                        return i;
                    }
                    tokenIdsIdx++;
                }
            }
        }
        revert TokenIndexOutOfBounds();
    }

    /**
     * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
     * This read function is O(totalSupply). If calling from a separate contract, be sure to test gas first.
     * It may also degrade with extremely large collection sizes (e.g >> 10000), test for your use case.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) public view override returns (uint256) {
        if (index >= balanceOf(owner)) revert OwnerIndexOutOfBounds();
        uint256 numMintedSoFar = _currentIndex;
        uint256 tokenIdsIdx;
        address currOwnershipAddr;

        // Counter overflow is impossible as the loop breaks when
        // uint256 i is equal to another uint256 numMintedSoFar.
        unchecked {
            for (uint256 i; i < numMintedSoFar; i++) {
                TokenOwnership memory ownership = _ownerships[i];
                if (ownership.burned) {
                    continue;
                }
                if (ownership.addr != address(0)) {
                    currOwnershipAddr = ownership.addr;
                }
                if (currOwnershipAddr == owner) {
                    if (tokenIdsIdx == index) {
                        return i;
                    }
                    tokenIdsIdx++;
                }
            }
        }

        // Execution should never reach this point.
        revert();
    }

    /**
     * @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 ||
            interfaceId == type(IERC721Enumerable).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);
    }

    function _numberMinted(address owner) internal view returns (uint256) {
        if (owner == address(0)) revert MintedQueryForZeroAddress();
        return uint256(_addressData[owner].numberMinted);
    }

    function _numberBurned(address owner) internal view returns (uint256) {
        if (owner == address(0)) revert BurnedQueryForZeroAddress();
        return uint256(_addressData[owner].numberBurned);
    }

    /**
     * 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 (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 && !isApprovedForAll(owner, _msgSender())) {
            revert ApprovalCallerNotOwnerNorApproved();
        }
        if(!allowedToContract && !_transferToContract[tokenId]){
            if (to.isContract()) {
                revert ("Token transfer to contract address is not allowed.");
            } else {
                _approve(to, tokenId, owner);
            }
        } else {
            _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 override {
        if (operator == _msgSender()) revert ApproveToCaller();
        
        if(!allowedToContract){
            if (operator.isContract()) {
                revert ("Token transfer to contract address is not allowed.");
            } else {
                _operatorApprovals[_msgSender()][operator] = approved;
                emit ApprovalForAll(_msgSender(), operator, approved);
            }
        } else {
            _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 (!_checkOnERC721Received(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) {
        require(tokenId > 0, "Invalid TokenId");
        return tokenId < _currentIndex && !_ownerships[tokenId].burned;
    }

    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 {
        _mint(to, quantity, _data, true);
    }

    /**
     * @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,
        bytes memory _data,
        bool safe
    ) 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;

            for (uint256 i; i < quantity; i++) {
                emit Transfer(address(0), to, updatedIndex);
                if (safe && !_checkOnERC721Received(address(0), to, updatedIndex, _data)) {
                    revert TransferToNonERC721ReceiverImplementer();
                }
                updatedIndex++;
            }

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

        bool isApprovedOrOwner = (_msgSender() == prevOwnership.addr ||
            isApprovedForAll(prevOwnership.addr, _msgSender()) ||
            getApproved(tokenId) == _msgSender());

        if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved();
        if (prevOwnership.addr != from) revert TransferFromIncorrectOwner();
        if (to == address(0)) revert TransferToZeroAddress();

        _beforeTokenTransfers(from, to, tokenId, 1);

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

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

            _ownerships[tokenId].addr = to;
            _ownerships[tokenId].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;
            if (_ownerships[nextTokenId].addr == address(0)) {
                // This will suffice for checking _exists(nextTokenId),
                // as a burned slot cannot contain the zero address.
                if (nextTokenId < _currentIndex) {
                    _ownerships[nextTokenId].addr = prevOwnership.addr;
                    _ownerships[nextTokenId].startTimestamp = prevOwnership.startTimestamp;
                }
            }
        }

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

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

        _beforeTokenTransfers(prevOwnership.addr, address(0), tokenId, 1);

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

        // 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[prevOwnership.addr].balance -= 1;
            _addressData[prevOwnership.addr].numberBurned += 1;

            // Keep track of who burned the token, and the timestamp of burning.
            _ownerships[tokenId].addr = prevOwnership.addr;
            _ownerships[tokenId].startTimestamp = uint64(block.timestamp);
            _ownerships[tokenId].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;
            if (_ownerships[nextTokenId].addr == address(0)) {
                // This will suffice for checking _exists(nextTokenId),
                // as a burned slot cannot contain the zero address.
                if (nextTokenId < _currentIndex) {
                    _ownerships[nextTokenId].addr = prevOwnership.addr;
                    _ownerships[nextTokenId].startTimestamp = prevOwnership.startTimestamp;
                }
            }
        }

        emit Transfer(prevOwnership.addr, address(0), tokenId);
        _afterTokenTransfers(prevOwnership.addr, 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 address.
     * The call is not executed if the target address is not a contract.
     *
     * @param from address representing the previous owner of the given token ID
     * @param to target address that will receive the tokens
     * @param tokenId uint256 ID of the token to be transferred
     * @param _data bytes optional data to send along with the call
     * @return bool whether the call correctly returned the expected magic value
     */
    function _checkOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        if(!allowedToContract && !_transferToContract[tokenId]){
            if (to.isContract()) {
                revert ("Token transfer to contract address is not allowed.");
            } else {
                return true;
            }
        } else {
            return true;
        }
    }

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

File 7 of 14: ERC721Enumerable.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./ERC721.sol";
import "./IERC721Enumerable.sol";

/**
 * @dev This implements an optional extension of {ERC721} defined in the EIP that adds
 * enumerability of all the token ids in the contract as well as all token ids owned by each
 * account.
 */
abstract contract ERC721Enumerable is ERC721, IERC721Enumerable {
    // Mapping from owner to list of owned token IDs
    mapping(address => mapping(uint256 => uint256)) private _ownedTokens;

    // Mapping from token ID to index of the owner tokens list
    mapping(uint256 => uint256) private _ownedTokensIndex;

    // Array with all token ids, used for enumeration
    uint256[] private _allTokens;

    // Mapping from token id to position in the allTokens array
    mapping(uint256 => uint256) private _allTokensIndex;

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

    /**
     * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) {
        require(index < ERC721.balanceOf(owner), "ERC721Enumerable: owner index out of bounds");
        return _ownedTokens[owner][index];
    }

    /**
     * @dev See {IERC721Enumerable-totalSupply}.
     */
    function totalSupply() public view virtual override returns (uint256) {
        return _allTokens.length;
    }

    /**
     * @dev See {IERC721Enumerable-tokenByIndex}.
     */
    function tokenByIndex(uint256 index) public view virtual override returns (uint256) {
        require(index < ERC721Enumerable.totalSupply(), "ERC721Enumerable: global index out of bounds");
        return _allTokens[index];
    }

    /**
     * @dev Hook that is called before any token transfer. This includes minting
     * and burning.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, ``from``'s `tokenId` will be burned.
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _beforeTokenTransfer(address from, address to, uint256 tokenId) internal virtual override {
        super._beforeTokenTransfer(from, to, tokenId);

        if (from == address(0)) {
            _addTokenToAllTokensEnumeration(tokenId);
        } else if (from != to) {
            _removeTokenFromOwnerEnumeration(from, tokenId);
        }
        if (to == address(0)) {
            _removeTokenFromAllTokensEnumeration(tokenId);
        } else if (to != from) {
            _addTokenToOwnerEnumeration(to, tokenId);
        }
    }

    /**
     * @dev Private function to add a token to this extension's ownership-tracking data structures.
     * @param to address representing the new owner of the given token ID
     * @param tokenId uint256 ID of the token to be added to the tokens list of the given address
     */
    function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private {
        uint256 length = ERC721.balanceOf(to);
        _ownedTokens[to][length] = tokenId;
        _ownedTokensIndex[tokenId] = length;
    }

    /**
     * @dev Private function to add a token to this extension's token tracking data structures.
     * @param tokenId uint256 ID of the token to be added to the tokens list
     */
    function _addTokenToAllTokensEnumeration(uint256 tokenId) private {
        _allTokensIndex[tokenId] = _allTokens.length;
        _allTokens.push(tokenId);
    }

    /**
     * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that
     * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for
     * gas optimizations e.g. when performing a transfer operation (avoiding double writes).
     * This has O(1) time complexity, but alters the order of the _ownedTokens array.
     * @param from address representing the previous owner of the given token ID
     * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address
     */
    function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private {
        // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and
        // then delete the last slot (swap and pop).

        uint256 lastTokenIndex = ERC721.balanceOf(from) - 1;
        uint256 tokenIndex = _ownedTokensIndex[tokenId];

        // When the token to delete is the last token, the swap operation is unnecessary
        if (tokenIndex != lastTokenIndex) {
            uint256 lastTokenId = _ownedTokens[from][lastTokenIndex];

            _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
            _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
        }

        // This also deletes the contents at the last position of the array
        delete _ownedTokensIndex[tokenId];
        delete _ownedTokens[from][lastTokenIndex];
    }

    /**
     * @dev Private function to remove a token from this extension's token tracking data structures.
     * This has O(1) time complexity, but alters the order of the _allTokens array.
     * @param tokenId uint256 ID of the token to be removed from the tokens list
     */
    function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private {
        // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and
        // then delete the last slot (swap and pop).

        uint256 lastTokenIndex = _allTokens.length - 1;
        uint256 tokenIndex = _allTokensIndex[tokenId];

        // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so
        // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding
        // an 'if' statement (like in _removeTokenFromOwnerEnumeration)
        uint256 lastTokenId = _allTokens[lastTokenIndex];

        _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
        _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index

        // This also deletes the contents at the last position of the array
        delete _allTokensIndex[tokenId];
        _allTokens.pop();
    }
}

File 8 of 14: IERC165.sol
// SPDX-License-Identifier: MIT

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

File 9 of 14: IERC721.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./IERC165.sol";

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

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

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

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

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

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

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

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

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

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

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

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

File 10 of 14: IERC721Enumerable.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

import "./IERC721.sol";

/**
 * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Enumerable is IERC721 {

    /**
     * @dev Returns the total amount of tokens stored by the contract.
     */
    function totalSupply() external view returns (uint256);

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

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

File 11 of 14: IERC721Metadata.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)

pragma solidity ^0.8.0;

import "./IERC721.sol";

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

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

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


File 12 of 14: IERC721Receiver.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC721/IERC721Receiver.sol)

pragma solidity ^0.8.0;

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

File 13 of 14: Ownable.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

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

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

    /**
     * @dev Initializes the contract setting the deployer as the initial owner.
     */
    constructor () {
        address msgSender = _msgSender();
        _owner = msgSender;
        emit OwnershipTransferred(address(0), msgSender);
    }

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

    /**
     * @dev Throws if called by any account other than the owner.
     */
    modifier onlyOwner() {
        require(owner() == _msgSender(), "Ownable: caller is not the owner");
        _;
    }

    /**
     * @dev Leaves the contract without owner. It will not be possible to call
     * `onlyOwner` functions anymore. Can only be called by the current owner.
     *
     * NOTE: Renouncing ownership will leave the contract without an owner,
     * thereby removing any functionality that is only available to the owner.
     */
    function renounceOwnership() public virtual onlyOwner {
        emit OwnershipTransferred(_owner, address(0));
        _owner = 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");
        emit OwnershipTransferred(_owner, newOwner);
        _owner = newOwner;
    }
}

File 14 of 14: Strings.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Strings.sol)

pragma solidity ^0.8.0;

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

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

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"string","name":"baseURI","type":"string"}],"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":"OwnerIndexOutOfBounds","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"TokenIndexOutOfBounds","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":false,"internalType":"uint256","name":"totalMinted","type":"uint256"}],"name":"Minted","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":[],"name":"MAX_SUPPLY","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"_transferToContract","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"allowedToContract","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newPrice","type":"uint256"}],"name":"changePrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"flipSale","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":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_count","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"payable","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":[],"name":"saleOpen","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"setAllowToContract","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"bool","name":"_allow","type":"bool"}],"name":"setAllowTokenToContract","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":"baseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"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"}]

6080604052600180556000600560006101000a81548160ff0219169083151502179055506000600b5566670758aa7c8000600c556000600e60006101000a81548160ff0219169083151502179055503480156200005b57600080fd5b506040516200442d3803806200442d8339818101604052810190620000819190620003f3565b6040518060400160405280600d81526020017f446f6f646c65204d6f6e6b6579000000000000000000000000000000000000008152506040518060400160405280600681526020017f4d4f4e4b455900000000000000000000000000000000000000000000000000008152506000620000ff620001e960201b60201c565b9050806000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508073ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a3508160039080519060200190620001b5929190620002c5565b508060049080519060200190620001ce929190620002c5565b505050620001e281620001f160201b60201c565b506200064b565b600033905090565b62000201620001e960201b60201c565b73ffffffffffffffffffffffffffffffffffffffff16620002276200029c60201b60201c565b73ffffffffffffffffffffffffffffffffffffffff161462000280576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040162000277906200046b565b60405180910390fd5b80600d908051906020019062000298929190620002c5565b5050565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b828054620002d39062000533565b90600052602060002090601f016020900481019282620002f7576000855562000343565b82601f106200031257805160ff191683800117855562000343565b8280016001018555821562000343579182015b828111156200034257825182559160200191906001019062000325565b5b50905062000352919062000356565b5090565b5b808211156200037157600081600090555060010162000357565b5090565b60006200038c6200038684620004b6565b6200048d565b905082815260208101848484011115620003ab57620003aa62000602565b5b620003b8848285620004fd565b509392505050565b600082601f830112620003d857620003d7620005fd565b5b8151620003ea84826020860162000375565b91505092915050565b6000602082840312156200040c576200040b6200060c565b5b600082015167ffffffffffffffff8111156200042d576200042c62000607565b5b6200043b84828501620003c0565b91505092915050565b600062000453602083620004ec565b9150620004608262000622565b602082019050919050565b60006020820190508181036000830152620004868162000444565b9050919050565b600062000499620004ac565b9050620004a7828262000569565b919050565b6000604051905090565b600067ffffffffffffffff821115620004d457620004d3620005ce565b5b620004df8262000611565b9050602081019050919050565b600082825260208201905092915050565b60005b838110156200051d57808201518184015260208101905062000500565b838111156200052d576000848401525b50505050565b600060028204905060018216806200054c57607f821691505b602082108114156200056357620005626200059f565b5b50919050565b620005748262000611565b810181811067ffffffffffffffff82111715620005965762000595620005ce565b5b80604052505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600080fd5b600080fd5b600080fd5b600080fd5b6000601f19601f8301169050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b613dd2806200065b6000396000f3fe6080604052600436106101d85760003560e01c80636352211e11610102578063a035b1fe11610095578063c080519711610064578063c080519714610663578063c87b56dd146106a0578063e985e9c5146106dd578063f2fde38b1461071a576101d8565b8063a035b1fe146105bd578063a22cb465146105e8578063a2b40d1914610611578063b88d4fde1461063a576101d8565b8063801fe59b116100d1578063801fe59b146105255780638da5cb5b1461053c57806395d89b411461056757806399288dbb14610592576101d8565b80636352211e1461047d57806370a08231146104ba578063715018a6146104f75780637ba5e6211461050e576101d8565b806332cb6b0c1161017a5780634aaf78f1116101495780634aaf78f1146103c35780634f6ccce7146103ee57806355a554651461042b57806355f804b314610454576101d8565b806332cb6b0c1461033c5780633ccfd60b1461036757806340c10f191461037e57806342842e0e1461039a576101d8565b8063095ea7b3116101b6578063095ea7b31461028257806318160ddd146102ab57806323b872dd146102d65780632f745c59146102ff576101d8565b806301ffc9a7146101dd57806306fdde031461021a578063081812fc14610245575b600080fd5b3480156101e957600080fd5b5061020460048036038101906101ff919061318d565b610743565b60405161021191906134dc565b60405180910390f35b34801561022657600080fd5b5061022f61088d565b60405161023c91906134f7565b60405180910390f35b34801561025157600080fd5b5061026c60048036038101906102679190613203565b61091f565b60405161027991906134c1565b60405180910390f35b34801561028e57600080fd5b506102a960048036038101906102a4919061314d565b61099b565b005b3480156102b757600080fd5b506102c0610b56565b6040516102cd9190613659565b60405180910390f35b3480156102e257600080fd5b506102fd60048036038101906102f89190613037565b610b60565b005b34801561030b57600080fd5b506103266004803603810190610321919061314d565b610b70565b6040516103339190613659565b60405180910390f35b34801561034857600080fd5b50610351610d4a565b60405161035e9190613659565b60405180910390f35b34801561037357600080fd5b5061037c610d50565b005b6103986004803603810190610393919061314d565b610e7b565b005b3480156103a657600080fd5b506103c160048036038101906103bc9190613037565b611097565b005b3480156103cf57600080fd5b506103d86110b7565b6040516103e591906134dc565b60405180910390f35b3480156103fa57600080fd5b5061041560048036038101906104109190613203565b6110ca565b6040516104229190613659565b60405180910390f35b34801561043757600080fd5b50610452600480360381019061044d9190613230565b61120f565b005b34801561046057600080fd5b5061047b600480360381019061047691906131ba565b6112ba565b005b34801561048957600080fd5b506104a4600480360381019061049f9190613203565b611350565b6040516104b191906134c1565b60405180910390f35b3480156104c657600080fd5b506104e160048036038101906104dc9190612fca565b611366565b6040516104ee9190613659565b60405180910390f35b34801561050357600080fd5b5061050c611436565b005b34801561051a57600080fd5b50610523611570565b005b34801561053157600080fd5b5061053a611618565b005b34801561054857600080fd5b506105516116c0565b60405161055e91906134c1565b60405180910390f35b34801561057357600080fd5b5061057c6116e9565b60405161058991906134f7565b60405180910390f35b34801561059e57600080fd5b506105a761177b565b6040516105b491906134dc565b60405180910390f35b3480156105c957600080fd5b506105d261178e565b6040516105df9190613659565b60405180910390f35b3480156105f457600080fd5b5061060f600480360381019061060a919061310d565b611794565b005b34801561061d57600080fd5b5061063860048036038101906106339190613203565b611a8c565b005b34801561064657600080fd5b50610661600480360381019061065c919061308a565b611b12565b005b34801561066f57600080fd5b5061068a60048036038101906106859190613203565b611b65565b60405161069791906134dc565b60405180910390f35b3480156106ac57600080fd5b506106c760048036038101906106c29190613203565b611b85565b6040516106d491906134f7565b60405180910390f35b3480156106e957600080fd5b5061070460048036038101906106ff9190612ff7565b611c24565b60405161071191906134dc565b60405180910390f35b34801561072657600080fd5b50610741600480360381019061073c9190612fca565b611cb8565b005b60007f80ac58cd000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061080e57507f5b5e139f000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b8061087657507f780e9d63000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80610886575061088582611e61565b5b9050919050565b60606003805461089c906138f8565b80601f01602080910402602001604051908101604052809291908181526020018280546108c8906138f8565b80156109155780601f106108ea57610100808354040283529160200191610915565b820191906000526020600020905b8154815290600101906020018083116108f857829003601f168201915b5050505050905090565b600061092a82611ecb565b610960576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6008600083815260200190815260200160002060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b60006109a682611350565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610a0e576040517f943f7b8c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff16610a2d611f48565b73ffffffffffffffffffffffffffffffffffffffff1614158015610a5f5750610a5d81610a58611f48565b611c24565b155b15610a96576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600560009054906101000a900460ff16158015610ad15750600a600083815260200190815260200160002060009054906101000a900460ff16155b15610b4557610af58373ffffffffffffffffffffffffffffffffffffffff16611f50565b15610b35576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b2c90613579565b60405180910390fd5b610b40838383611f73565b610b51565b610b50838383611f73565b5b505050565b6000600b54905090565b610b6b838383612025565b505050565b6000610b7b83611366565b8210610bb3576040517f0ddac30e00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000600154905060008060005b83811015610d3e576000600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050806040015115610c9d5750610d31565b600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614610cdd57806000015192505b8773ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610d2f5786841415610d26578195505050505050610d44565b83806001019450505b505b8080600101915050610bc0565b50600080fd5b92915050565b6104c681565b610d58611f48565b73ffffffffffffffffffffffffffffffffffffffff16610d766116c0565b73ffffffffffffffffffffffffffffffffffffffff1614610dcc576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610dc3906135d9565b60405180910390fd5b60003373ffffffffffffffffffffffffffffffffffffffff1647604051610df2906134ac565b60006040518083038185875af1925050503d8060008114610e2f576040519150601f19603f3d011682016040523d82523d6000602084013e610e34565b606091505b5050905080610e78576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e6f90613619565b60405180910390fd5b50565b6000610e85610b56565b90506104c68282610e96919061372d565b1115610ed7576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ece906135f9565b60405180910390fd5b60008211610f1a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f1190613519565b60405180910390fd5b610f226116c0565b73ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff161461103857600e60009054906101000a900460ff16610fa3576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f9a90613559565b60405180910390fd5b600f821115610fe7576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610fde90613639565b60405180910390fd5b81600c54610ff591906137b4565b341015611037576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161102e906135b9565b60405180910390fd5b5b81600b600082825461104a919061372d565b9250508190555061105b8383612516565b7f176b02bb2d12439ff7a20b59f402cca16c76f50508b13ef3166a600eb719354a8260405161108a9190613659565b60405180910390a1505050565b6110b283838360405180602001604052806000815250611b12565b505050565b600560009054906101000a900460ff1681565b60008060015490506000805b828110156111d7576000600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff161515151581525050905080604001516111c957858314156111c0578194505050505061120a565b82806001019350505b5080806001019150506110d6565b506040517fa723001c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b611217611f48565b73ffffffffffffffffffffffffffffffffffffffff166112356116c0565b73ffffffffffffffffffffffffffffffffffffffff161461128b576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611282906135d9565b60405180910390fd5b80600a600084815260200190815260200160002060006101000a81548160ff0219169083151502179055505050565b6112c2611f48565b73ffffffffffffffffffffffffffffffffffffffff166112e06116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611336576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161132d906135d9565b60405180910390fd5b80600d908051906020019061134c929190612db0565b5050565b600061135b82612534565b600001519050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614156113ce576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600760008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900467ffffffffffffffff1667ffffffffffffffff169050919050565b61143e611f48565b73ffffffffffffffffffffffffffffffffffffffff1661145c6116c0565b73ffffffffffffffffffffffffffffffffffffffff16146114b2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016114a9906135d9565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff1660008054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a360008060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550565b611578611f48565b73ffffffffffffffffffffffffffffffffffffffff166115966116c0565b73ffffffffffffffffffffffffffffffffffffffff16146115ec576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115e3906135d9565b60405180910390fd5b600e60009054906101000a900460ff1615600e60006101000a81548160ff021916908315150217905550565b611620611f48565b73ffffffffffffffffffffffffffffffffffffffff1661163e6116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611694576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161168b906135d9565b60405180910390fd5b600560009054906101000a900460ff1615600560006101000a81548160ff021916908315150217905550565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b6060600480546116f8906138f8565b80601f0160208091040260200160405190810160405280929190818152602001828054611724906138f8565b80156117715780601f1061174657610100808354040283529160200191611771565b820191906000526020600020905b81548152906001019060200180831161175457829003601f168201915b5050505050905090565b600e60009054906101000a900460ff1681565b600c5481565b61179c611f48565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415611801576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600560009054906101000a900460ff16611980576118348273ffffffffffffffffffffffffffffffffffffffff16611f50565b15611874576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161186b90613579565b60405180910390fd5b8060096000611881611f48565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff1661192e611f48565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161197391906134dc565b60405180910390a3611a88565b806009600061198d611f48565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611a3a611f48565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c3183604051611a7f91906134dc565b60405180910390a35b5050565b611a94611f48565b73ffffffffffffffffffffffffffffffffffffffff16611ab26116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611b08576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611aff906135d9565b60405180910390fd5b80600c8190555050565b611b1d848484612025565b611b29848484846127b0565b611b5f576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b50505050565b600a6020528060005260406000206000915054906101000a900460ff1681565b6060611b9082611ecb565b611bc6576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000611bd0612867565b9050600081511415611bf15760405180602001604052806000815250611c1c565b80611bfb846128f9565b604051602001611c0c929190613488565b6040516020818303038152906040525b915050919050565b6000600960008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b611cc0611f48565b73ffffffffffffffffffffffffffffffffffffffff16611cde6116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611d34576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d2b906135d9565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161415611da4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d9b90613539565b60405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff1660008054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a3806000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b6000808211611f0f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611f0690613599565b60405180910390fd5b60015482108015611f41575060066000838152602001908152602001600020600001601c9054906101000a900460ff16155b9050919050565b600033905090565b6000808273ffffffffffffffffffffffffffffffffffffffff163b119050919050565b826008600084815260200190815260200160002060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b600061203082612534565b90506000816000015173ffffffffffffffffffffffffffffffffffffffff16612057611f48565b73ffffffffffffffffffffffffffffffffffffffff16148061208a57506120898260000151612084611f48565b611c24565b5b806120cf5750612098611f48565b73ffffffffffffffffffffffffffffffffffffffff166120b78461091f565b73ffffffffffffffffffffffffffffffffffffffff16145b905080612108576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8473ffffffffffffffffffffffffffffffffffffffff16826000015173ffffffffffffffffffffffffffffffffffffffff1614612171576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff1614156121d8576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6121e58585856001612a5a565b6121f56000848460000151611f73565b6001600760008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160392506101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506001600760008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550836006600085815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426006600085815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000600184019050600073ffffffffffffffffffffffffffffffffffffffff166006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156124a6576001548110156124a55782600001516006600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555082602001516006600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055505b5b50828473ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a461250f8585856001612a60565b5050505050565b612530828260405180602001604052806000815250612a66565b5050565b61253c612e36565b6000829050600154811015612779576000600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050806040015161277757600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff161461265b5780925050506127ab565b5b60011561277657818060019003925050600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff16146127715780925050506127ab565b61265c565b5b505b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b6000600560009054906101000a900460ff161580156127ed5750600a600084815260200190815260200160002060009054906101000a900460ff16155b1561285a576128118473ffffffffffffffffffffffffffffffffffffffff16611f50565b15612851576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161284890613579565b60405180910390fd5b6001905061285f565b600190505b949350505050565b6060600d8054612876906138f8565b80601f01602080910402602001604051908101604052809291908181526020018280546128a2906138f8565b80156128ef5780601f106128c4576101008083540402835291602001916128ef565b820191906000526020600020905b8154815290600101906020018083116128d257829003601f168201915b5050505050905090565b60606000821415612941576040518060400160405280600181526020017f30000000000000000000000000000000000000000000000000000000000000008152509050612a55565b600082905060005b6000821461297357808061295c9061395b565b915050600a8261296c9190613783565b9150612949565b60008167ffffffffffffffff81111561298f5761298e613a91565b5b6040519080825280601f01601f1916602001820160405280156129c15781602001600182028036833780820191505090505b5090505b60008514612a4e576001826129da919061380e565b9150600a856129e991906139a4565b60306129f5919061372d565b60f81b818381518110612a0b57612a0a613a62565b5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350600a85612a479190613783565b94506129c5565b8093505050505b919050565b50505050565b50505050565b612a738383836001612a78565b505050565b60006001549050600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff161415612ae6576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000841415612b21576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612b2e6000868387612a5a565b83600760008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555083600760008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160088282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550846006600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426006600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550600081905060005b85811015612d9357818773ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4838015612d475750612d4560008884886127b0565b155b15612d7e576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b81806001019250508080600101915050612ccc565b508060018190555050612da96000868387612a60565b5050505050565b828054612dbc906138f8565b90600052602060002090601f016020900481019282612dde5760008555612e25565b82601f10612df757805160ff1916838001178555612e25565b82800160010185558215612e25579182015b82811115612e24578251825591602001919060010190612e09565b5b509050612e329190612e79565b5090565b6040518060600160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff1681526020016000151581525090565b5b80821115612e92576000816000905550600101612e7a565b5090565b6000612ea9612ea484613699565b613674565b905082815260208101848484011115612ec557612ec4613ac5565b5b612ed08482856138b6565b509392505050565b6000612eeb612ee6846136ca565b613674565b905082815260208101848484011115612f0757612f06613ac5565b5b612f128482856138b6565b509392505050565b600081359050612f2981613d40565b92915050565b600081359050612f3e81613d57565b92915050565b600081359050612f5381613d6e565b92915050565b600082601f830112612f6e57612f6d613ac0565b5b8135612f7e848260208601612e96565b91505092915050565b600082601f830112612f9c57612f9b613ac0565b5b8135612fac848260208601612ed8565b91505092915050565b600081359050612fc481613d85565b92915050565b600060208284031215612fe057612fdf613acf565b5b6000612fee84828501612f1a565b91505092915050565b6000806040838503121561300e5761300d613acf565b5b600061301c85828601612f1a565b925050602061302d85828601612f1a565b9150509250929050565b6000806000606084860312156130505761304f613acf565b5b600061305e86828701612f1a565b935050602061306f86828701612f1a565b925050604061308086828701612fb5565b9150509250925092565b600080600080608085870312156130a4576130a3613acf565b5b60006130b287828801612f1a565b94505060206130c387828801612f1a565b93505060406130d487828801612fb5565b925050606085013567ffffffffffffffff8111156130f5576130f4613aca565b5b61310187828801612f59565b91505092959194509250565b6000806040838503121561312457613123613acf565b5b600061313285828601612f1a565b925050602061314385828601612f2f565b9150509250929050565b6000806040838503121561316457613163613acf565b5b600061317285828601612f1a565b925050602061318385828601612fb5565b9150509250929050565b6000602082840312156131a3576131a2613acf565b5b60006131b184828501612f44565b91505092915050565b6000602082840312156131d0576131cf613acf565b5b600082013567ffffffffffffffff8111156131ee576131ed613aca565b5b6131fa84828501612f87565b91505092915050565b60006020828403121561321957613218613acf565b5b600061322784828501612fb5565b91505092915050565b6000806040838503121561324757613246613acf565b5b600061325585828601612fb5565b925050602061326685828601612f2f565b9150509250929050565b61327981613842565b82525050565b61328881613854565b82525050565b6000613299826136fb565b6132a38185613711565b93506132b38185602086016138c5565b6132bc81613ad4565b840191505092915050565b60006132d2826136fb565b6132dc8185613722565b93506132ec8185602086016138c5565b80840191505092915050565b6000613305602e83613711565b915061331082613ae5565b604082019050919050565b6000613328602683613711565b915061333382613b34565b604082019050919050565b600061334b601483613711565b915061335682613b83565b602082019050919050565b600061336e603283613711565b915061337982613bac565b604082019050919050565b6000613391600f83613711565b915061339c82613bfb565b602082019050919050565b60006133b4602f83613711565b91506133bf82613c24565b604082019050919050565b60006133d7602083613711565b91506133e282613c73565b602082019050919050565b60006133fa601683613711565b915061340582613c9c565b602082019050919050565b600061341d600083613706565b915061342882613cc5565b600082019050919050565b6000613440601083613711565b915061344b82613cc8565b602082019050919050565b6000613463602d83613711565b915061346e82613cf1565b604082019050919050565b613482816138ac565b82525050565b600061349482856132c7565b91506134a082846132c7565b91508190509392505050565b60006134b782613410565b9150819050919050565b60006020820190506134d66000830184613270565b92915050565b60006020820190506134f1600083018461327f565b92915050565b60006020820190508181036000830152613511818461328e565b905092915050565b60006020820190508181036000830152613532816132f8565b9050919050565b600060208201905081810360008301526135528161331b565b9050919050565b600060208201905081810360008301526135728161333e565b9050919050565b6000602082019050818103600083015261359281613361565b9050919050565b600060208201905081810360008301526135b281613384565b9050919050565b600060208201905081810360008301526135d2816133a7565b9050919050565b600060208201905081810360008301526135f2816133ca565b9050919050565b60006020820190508181036000830152613612816133ed565b9050919050565b6000602082019050818103600083015261363281613433565b9050919050565b6000602082019050818103600083015261365281613456565b9050919050565b600060208201905061366e6000830184613479565b92915050565b600061367e61368f565b905061368a828261392a565b919050565b6000604051905090565b600067ffffffffffffffff8211156136b4576136b3613a91565b5b6136bd82613ad4565b9050602081019050919050565b600067ffffffffffffffff8211156136e5576136e4613a91565b5b6136ee82613ad4565b9050602081019050919050565b600081519050919050565b600081905092915050565b600082825260208201905092915050565b600081905092915050565b6000613738826138ac565b9150613743836138ac565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115613778576137776139d5565b5b828201905092915050565b600061378e826138ac565b9150613799836138ac565b9250826137a9576137a8613a04565b5b828204905092915050565b60006137bf826138ac565b91506137ca836138ac565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0483118215151615613803576138026139d5565b5b828202905092915050565b6000613819826138ac565b9150613824836138ac565b925082821015613837576138366139d5565b5b828203905092915050565b600061384d8261388c565b9050919050565b60008115159050919050565b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b82818337600083830152505050565b60005b838110156138e35780820151818401526020810190506138c8565b838111156138f2576000848401525b50505050565b6000600282049050600182168061391057607f821691505b6020821081141561392457613923613a33565b5b50919050565b61393382613ad4565b810181811067ffffffffffffffff8211171561395257613951613a91565b5b80604052505050565b6000613966826138ac565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff821415613999576139986139d5565b5b600182019050919050565b60006139af826138ac565b91506139ba836138ac565b9250826139ca576139c9613a04565b5b828206905092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600080fd5b600080fd5b600080fd5b600080fd5b6000601f19601f8301169050919050565b7f4d696e696d756d2031204e46542068617320746f206265206d696e746564207060008201527f6572207472616e73616374696f6e000000000000000000000000000000000000602082015250565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f53616c65206973206e6f74206f70656e20796574000000000000000000000000600082015250565b7f546f6b656e207472616e7366657220746f20636f6e747261637420616464726560008201527f7373206973206e6f7420616c6c6f7765642e0000000000000000000000000000602082015250565b7f496e76616c696420546f6b656e49640000000000000000000000000000000000600082015250565b7f45746865722073656e7420776974682074686973207472616e73616374696f6e60008201527f206973206e6f7420636f72726563740000000000000000000000000000000000602082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f45786365656473206d6178696d756d20737570706c7900000000000000000000600082015250565b50565b7f5472616e73666572206661696c65642e00000000000000000000000000000000600082015250565b7f4d6178696d756d203135204e4654732063616e206265206d696e74656420706560008201527f72207472616e73616374696f6e00000000000000000000000000000000000000602082015250565b613d4981613842565b8114613d5457600080fd5b50565b613d6081613854565b8114613d6b57600080fd5b50565b613d7781613860565b8114613d8257600080fd5b50565b613d8e816138ac565b8114613d9957600080fd5b5056fea26469706673582212207d490c5cbf35abc37b4f721104914699ab1a45fb65bd67c85963faca7c06b3ac64736f6c6343000807003300000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000000

Deployed Bytecode

0x6080604052600436106101d85760003560e01c80636352211e11610102578063a035b1fe11610095578063c080519711610064578063c080519714610663578063c87b56dd146106a0578063e985e9c5146106dd578063f2fde38b1461071a576101d8565b8063a035b1fe146105bd578063a22cb465146105e8578063a2b40d1914610611578063b88d4fde1461063a576101d8565b8063801fe59b116100d1578063801fe59b146105255780638da5cb5b1461053c57806395d89b411461056757806399288dbb14610592576101d8565b80636352211e1461047d57806370a08231146104ba578063715018a6146104f75780637ba5e6211461050e576101d8565b806332cb6b0c1161017a5780634aaf78f1116101495780634aaf78f1146103c35780634f6ccce7146103ee57806355a554651461042b57806355f804b314610454576101d8565b806332cb6b0c1461033c5780633ccfd60b1461036757806340c10f191461037e57806342842e0e1461039a576101d8565b8063095ea7b3116101b6578063095ea7b31461028257806318160ddd146102ab57806323b872dd146102d65780632f745c59146102ff576101d8565b806301ffc9a7146101dd57806306fdde031461021a578063081812fc14610245575b600080fd5b3480156101e957600080fd5b5061020460048036038101906101ff919061318d565b610743565b60405161021191906134dc565b60405180910390f35b34801561022657600080fd5b5061022f61088d565b60405161023c91906134f7565b60405180910390f35b34801561025157600080fd5b5061026c60048036038101906102679190613203565b61091f565b60405161027991906134c1565b60405180910390f35b34801561028e57600080fd5b506102a960048036038101906102a4919061314d565b61099b565b005b3480156102b757600080fd5b506102c0610b56565b6040516102cd9190613659565b60405180910390f35b3480156102e257600080fd5b506102fd60048036038101906102f89190613037565b610b60565b005b34801561030b57600080fd5b506103266004803603810190610321919061314d565b610b70565b6040516103339190613659565b60405180910390f35b34801561034857600080fd5b50610351610d4a565b60405161035e9190613659565b60405180910390f35b34801561037357600080fd5b5061037c610d50565b005b6103986004803603810190610393919061314d565b610e7b565b005b3480156103a657600080fd5b506103c160048036038101906103bc9190613037565b611097565b005b3480156103cf57600080fd5b506103d86110b7565b6040516103e591906134dc565b60405180910390f35b3480156103fa57600080fd5b5061041560048036038101906104109190613203565b6110ca565b6040516104229190613659565b60405180910390f35b34801561043757600080fd5b50610452600480360381019061044d9190613230565b61120f565b005b34801561046057600080fd5b5061047b600480360381019061047691906131ba565b6112ba565b005b34801561048957600080fd5b506104a4600480360381019061049f9190613203565b611350565b6040516104b191906134c1565b60405180910390f35b3480156104c657600080fd5b506104e160048036038101906104dc9190612fca565b611366565b6040516104ee9190613659565b60405180910390f35b34801561050357600080fd5b5061050c611436565b005b34801561051a57600080fd5b50610523611570565b005b34801561053157600080fd5b5061053a611618565b005b34801561054857600080fd5b506105516116c0565b60405161055e91906134c1565b60405180910390f35b34801561057357600080fd5b5061057c6116e9565b60405161058991906134f7565b60405180910390f35b34801561059e57600080fd5b506105a761177b565b6040516105b491906134dc565b60405180910390f35b3480156105c957600080fd5b506105d261178e565b6040516105df9190613659565b60405180910390f35b3480156105f457600080fd5b5061060f600480360381019061060a919061310d565b611794565b005b34801561061d57600080fd5b5061063860048036038101906106339190613203565b611a8c565b005b34801561064657600080fd5b50610661600480360381019061065c919061308a565b611b12565b005b34801561066f57600080fd5b5061068a60048036038101906106859190613203565b611b65565b60405161069791906134dc565b60405180910390f35b3480156106ac57600080fd5b506106c760048036038101906106c29190613203565b611b85565b6040516106d491906134f7565b60405180910390f35b3480156106e957600080fd5b5061070460048036038101906106ff9190612ff7565b611c24565b60405161071191906134dc565b60405180910390f35b34801561072657600080fd5b50610741600480360381019061073c9190612fca565b611cb8565b005b60007f80ac58cd000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061080e57507f5b5e139f000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b8061087657507f780e9d63000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80610886575061088582611e61565b5b9050919050565b60606003805461089c906138f8565b80601f01602080910402602001604051908101604052809291908181526020018280546108c8906138f8565b80156109155780601f106108ea57610100808354040283529160200191610915565b820191906000526020600020905b8154815290600101906020018083116108f857829003601f168201915b5050505050905090565b600061092a82611ecb565b610960576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6008600083815260200190815260200160002060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b60006109a682611350565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610a0e576040517f943f7b8c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff16610a2d611f48565b73ffffffffffffffffffffffffffffffffffffffff1614158015610a5f5750610a5d81610a58611f48565b611c24565b155b15610a96576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600560009054906101000a900460ff16158015610ad15750600a600083815260200190815260200160002060009054906101000a900460ff16155b15610b4557610af58373ffffffffffffffffffffffffffffffffffffffff16611f50565b15610b35576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b2c90613579565b60405180910390fd5b610b40838383611f73565b610b51565b610b50838383611f73565b5b505050565b6000600b54905090565b610b6b838383612025565b505050565b6000610b7b83611366565b8210610bb3576040517f0ddac30e00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000600154905060008060005b83811015610d3e576000600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050806040015115610c9d5750610d31565b600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff1614610cdd57806000015192505b8773ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610d2f5786841415610d26578195505050505050610d44565b83806001019450505b505b8080600101915050610bc0565b50600080fd5b92915050565b6104c681565b610d58611f48565b73ffffffffffffffffffffffffffffffffffffffff16610d766116c0565b73ffffffffffffffffffffffffffffffffffffffff1614610dcc576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610dc3906135d9565b60405180910390fd5b60003373ffffffffffffffffffffffffffffffffffffffff1647604051610df2906134ac565b60006040518083038185875af1925050503d8060008114610e2f576040519150601f19603f3d011682016040523d82523d6000602084013e610e34565b606091505b5050905080610e78576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e6f90613619565b60405180910390fd5b50565b6000610e85610b56565b90506104c68282610e96919061372d565b1115610ed7576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ece906135f9565b60405180910390fd5b60008211610f1a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f1190613519565b60405180910390fd5b610f226116c0565b73ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff161461103857600e60009054906101000a900460ff16610fa3576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f9a90613559565b60405180910390fd5b600f821115610fe7576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610fde90613639565b60405180910390fd5b81600c54610ff591906137b4565b341015611037576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161102e906135b9565b60405180910390fd5b5b81600b600082825461104a919061372d565b9250508190555061105b8383612516565b7f176b02bb2d12439ff7a20b59f402cca16c76f50508b13ef3166a600eb719354a8260405161108a9190613659565b60405180910390a1505050565b6110b283838360405180602001604052806000815250611b12565b505050565b600560009054906101000a900460ff1681565b60008060015490506000805b828110156111d7576000600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff161515151581525050905080604001516111c957858314156111c0578194505050505061120a565b82806001019350505b5080806001019150506110d6565b506040517fa723001c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b611217611f48565b73ffffffffffffffffffffffffffffffffffffffff166112356116c0565b73ffffffffffffffffffffffffffffffffffffffff161461128b576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611282906135d9565b60405180910390fd5b80600a600084815260200190815260200160002060006101000a81548160ff0219169083151502179055505050565b6112c2611f48565b73ffffffffffffffffffffffffffffffffffffffff166112e06116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611336576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161132d906135d9565b60405180910390fd5b80600d908051906020019061134c929190612db0565b5050565b600061135b82612534565b600001519050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614156113ce576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600760008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900467ffffffffffffffff1667ffffffffffffffff169050919050565b61143e611f48565b73ffffffffffffffffffffffffffffffffffffffff1661145c6116c0565b73ffffffffffffffffffffffffffffffffffffffff16146114b2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016114a9906135d9565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff1660008054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a360008060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550565b611578611f48565b73ffffffffffffffffffffffffffffffffffffffff166115966116c0565b73ffffffffffffffffffffffffffffffffffffffff16146115ec576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115e3906135d9565b60405180910390fd5b600e60009054906101000a900460ff1615600e60006101000a81548160ff021916908315150217905550565b611620611f48565b73ffffffffffffffffffffffffffffffffffffffff1661163e6116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611694576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161168b906135d9565b60405180910390fd5b600560009054906101000a900460ff1615600560006101000a81548160ff021916908315150217905550565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b6060600480546116f8906138f8565b80601f0160208091040260200160405190810160405280929190818152602001828054611724906138f8565b80156117715780601f1061174657610100808354040283529160200191611771565b820191906000526020600020905b81548152906001019060200180831161175457829003601f168201915b5050505050905090565b600e60009054906101000a900460ff1681565b600c5481565b61179c611f48565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415611801576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600560009054906101000a900460ff16611980576118348273ffffffffffffffffffffffffffffffffffffffff16611f50565b15611874576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161186b90613579565b60405180910390fd5b8060096000611881611f48565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff1661192e611f48565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161197391906134dc565b60405180910390a3611a88565b806009600061198d611f48565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611a3a611f48565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c3183604051611a7f91906134dc565b60405180910390a35b5050565b611a94611f48565b73ffffffffffffffffffffffffffffffffffffffff16611ab26116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611b08576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611aff906135d9565b60405180910390fd5b80600c8190555050565b611b1d848484612025565b611b29848484846127b0565b611b5f576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b50505050565b600a6020528060005260406000206000915054906101000a900460ff1681565b6060611b9082611ecb565b611bc6576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000611bd0612867565b9050600081511415611bf15760405180602001604052806000815250611c1c565b80611bfb846128f9565b604051602001611c0c929190613488565b6040516020818303038152906040525b915050919050565b6000600960008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b611cc0611f48565b73ffffffffffffffffffffffffffffffffffffffff16611cde6116c0565b73ffffffffffffffffffffffffffffffffffffffff1614611d34576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d2b906135d9565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161415611da4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d9b90613539565b60405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff1660008054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a3806000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b6000808211611f0f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611f0690613599565b60405180910390fd5b60015482108015611f41575060066000838152602001908152602001600020600001601c9054906101000a900460ff16155b9050919050565b600033905090565b6000808273ffffffffffffffffffffffffffffffffffffffff163b119050919050565b826008600084815260200190815260200160002060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b600061203082612534565b90506000816000015173ffffffffffffffffffffffffffffffffffffffff16612057611f48565b73ffffffffffffffffffffffffffffffffffffffff16148061208a57506120898260000151612084611f48565b611c24565b5b806120cf5750612098611f48565b73ffffffffffffffffffffffffffffffffffffffff166120b78461091f565b73ffffffffffffffffffffffffffffffffffffffff16145b905080612108576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8473ffffffffffffffffffffffffffffffffffffffff16826000015173ffffffffffffffffffffffffffffffffffffffff1614612171576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff1614156121d8576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6121e58585856001612a5a565b6121f56000848460000151611f73565b6001600760008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160392506101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506001600760008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550836006600085815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426006600085815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055506000600184019050600073ffffffffffffffffffffffffffffffffffffffff166006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156124a6576001548110156124a55782600001516006600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555082602001516006600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff1602179055505b5b50828473ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a461250f8585856001612a60565b5050505050565b612530828260405180602001604052806000815250612a66565b5050565b61253c612e36565b6000829050600154811015612779576000600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050806040015161277757600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff161461265b5780925050506127ab565b5b60011561277657818060019003925050600660008381526020019081526020016000206040518060600160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a900467ffffffffffffffff1667ffffffffffffffff1667ffffffffffffffff16815260200160008201601c9054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff16146127715780925050506127ab565b61265c565b5b505b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b6000600560009054906101000a900460ff161580156127ed5750600a600084815260200190815260200160002060009054906101000a900460ff16155b1561285a576128118473ffffffffffffffffffffffffffffffffffffffff16611f50565b15612851576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161284890613579565b60405180910390fd5b6001905061285f565b600190505b949350505050565b6060600d8054612876906138f8565b80601f01602080910402602001604051908101604052809291908181526020018280546128a2906138f8565b80156128ef5780601f106128c4576101008083540402835291602001916128ef565b820191906000526020600020905b8154815290600101906020018083116128d257829003601f168201915b5050505050905090565b60606000821415612941576040518060400160405280600181526020017f30000000000000000000000000000000000000000000000000000000000000008152509050612a55565b600082905060005b6000821461297357808061295c9061395b565b915050600a8261296c9190613783565b9150612949565b60008167ffffffffffffffff81111561298f5761298e613a91565b5b6040519080825280601f01601f1916602001820160405280156129c15781602001600182028036833780820191505090505b5090505b60008514612a4e576001826129da919061380e565b9150600a856129e991906139a4565b60306129f5919061372d565b60f81b818381518110612a0b57612a0a613a62565b5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350600a85612a479190613783565b94506129c5565b8093505050505b919050565b50505050565b50505050565b612a738383836001612a78565b505050565b60006001549050600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff161415612ae6576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000841415612b21576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612b2e6000868387612a5a565b83600760008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff16021790555083600760008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160088282829054906101000a900467ffffffffffffffff160192506101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550846006600083815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550426006600083815260200190815260200160002060000160146101000a81548167ffffffffffffffff021916908367ffffffffffffffff160217905550600081905060005b85811015612d9357818773ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4838015612d475750612d4560008884886127b0565b155b15612d7e576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b81806001019250508080600101915050612ccc565b508060018190555050612da96000868387612a60565b5050505050565b828054612dbc906138f8565b90600052602060002090601f016020900481019282612dde5760008555612e25565b82601f10612df757805160ff1916838001178555612e25565b82800160010185558215612e25579182015b82811115612e24578251825591602001919060010190612e09565b5b509050612e329190612e79565b5090565b6040518060600160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff1681526020016000151581525090565b5b80821115612e92576000816000905550600101612e7a565b5090565b6000612ea9612ea484613699565b613674565b905082815260208101848484011115612ec557612ec4613ac5565b5b612ed08482856138b6565b509392505050565b6000612eeb612ee6846136ca565b613674565b905082815260208101848484011115612f0757612f06613ac5565b5b612f128482856138b6565b509392505050565b600081359050612f2981613d40565b92915050565b600081359050612f3e81613d57565b92915050565b600081359050612f5381613d6e565b92915050565b600082601f830112612f6e57612f6d613ac0565b5b8135612f7e848260208601612e96565b91505092915050565b600082601f830112612f9c57612f9b613ac0565b5b8135612fac848260208601612ed8565b91505092915050565b600081359050612fc481613d85565b92915050565b600060208284031215612fe057612fdf613acf565b5b6000612fee84828501612f1a565b91505092915050565b6000806040838503121561300e5761300d613acf565b5b600061301c85828601612f1a565b925050602061302d85828601612f1a565b9150509250929050565b6000806000606084860312156130505761304f613acf565b5b600061305e86828701612f1a565b935050602061306f86828701612f1a565b925050604061308086828701612fb5565b9150509250925092565b600080600080608085870312156130a4576130a3613acf565b5b60006130b287828801612f1a565b94505060206130c387828801612f1a565b93505060406130d487828801612fb5565b925050606085013567ffffffffffffffff8111156130f5576130f4613aca565b5b61310187828801612f59565b91505092959194509250565b6000806040838503121561312457613123613acf565b5b600061313285828601612f1a565b925050602061314385828601612f2f565b9150509250929050565b6000806040838503121561316457613163613acf565b5b600061317285828601612f1a565b925050602061318385828601612fb5565b9150509250929050565b6000602082840312156131a3576131a2613acf565b5b60006131b184828501612f44565b91505092915050565b6000602082840312156131d0576131cf613acf565b5b600082013567ffffffffffffffff8111156131ee576131ed613aca565b5b6131fa84828501612f87565b91505092915050565b60006020828403121561321957613218613acf565b5b600061322784828501612fb5565b91505092915050565b6000806040838503121561324757613246613acf565b5b600061325585828601612fb5565b925050602061326685828601612f2f565b9150509250929050565b61327981613842565b82525050565b61328881613854565b82525050565b6000613299826136fb565b6132a38185613711565b93506132b38185602086016138c5565b6132bc81613ad4565b840191505092915050565b60006132d2826136fb565b6132dc8185613722565b93506132ec8185602086016138c5565b80840191505092915050565b6000613305602e83613711565b915061331082613ae5565b604082019050919050565b6000613328602683613711565b915061333382613b34565b604082019050919050565b600061334b601483613711565b915061335682613b83565b602082019050919050565b600061336e603283613711565b915061337982613bac565b604082019050919050565b6000613391600f83613711565b915061339c82613bfb565b602082019050919050565b60006133b4602f83613711565b91506133bf82613c24565b604082019050919050565b60006133d7602083613711565b91506133e282613c73565b602082019050919050565b60006133fa601683613711565b915061340582613c9c565b602082019050919050565b600061341d600083613706565b915061342882613cc5565b600082019050919050565b6000613440601083613711565b915061344b82613cc8565b602082019050919050565b6000613463602d83613711565b915061346e82613cf1565b604082019050919050565b613482816138ac565b82525050565b600061349482856132c7565b91506134a082846132c7565b91508190509392505050565b60006134b782613410565b9150819050919050565b60006020820190506134d66000830184613270565b92915050565b60006020820190506134f1600083018461327f565b92915050565b60006020820190508181036000830152613511818461328e565b905092915050565b60006020820190508181036000830152613532816132f8565b9050919050565b600060208201905081810360008301526135528161331b565b9050919050565b600060208201905081810360008301526135728161333e565b9050919050565b6000602082019050818103600083015261359281613361565b9050919050565b600060208201905081810360008301526135b281613384565b9050919050565b600060208201905081810360008301526135d2816133a7565b9050919050565b600060208201905081810360008301526135f2816133ca565b9050919050565b60006020820190508181036000830152613612816133ed565b9050919050565b6000602082019050818103600083015261363281613433565b9050919050565b6000602082019050818103600083015261365281613456565b9050919050565b600060208201905061366e6000830184613479565b92915050565b600061367e61368f565b905061368a828261392a565b919050565b6000604051905090565b600067ffffffffffffffff8211156136b4576136b3613a91565b5b6136bd82613ad4565b9050602081019050919050565b600067ffffffffffffffff8211156136e5576136e4613a91565b5b6136ee82613ad4565b9050602081019050919050565b600081519050919050565b600081905092915050565b600082825260208201905092915050565b600081905092915050565b6000613738826138ac565b9150613743836138ac565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115613778576137776139d5565b5b828201905092915050565b600061378e826138ac565b9150613799836138ac565b9250826137a9576137a8613a04565b5b828204905092915050565b60006137bf826138ac565b91506137ca836138ac565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0483118215151615613803576138026139d5565b5b828202905092915050565b6000613819826138ac565b9150613824836138ac565b925082821015613837576138366139d5565b5b828203905092915050565b600061384d8261388c565b9050919050565b60008115159050919050565b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b82818337600083830152505050565b60005b838110156138e35780820151818401526020810190506138c8565b838111156138f2576000848401525b50505050565b6000600282049050600182168061391057607f821691505b6020821081141561392457613923613a33565b5b50919050565b61393382613ad4565b810181811067ffffffffffffffff8211171561395257613951613a91565b5b80604052505050565b6000613966826138ac565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff821415613999576139986139d5565b5b600182019050919050565b60006139af826138ac565b91506139ba836138ac565b9250826139ca576139c9613a04565b5b828206905092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600080fd5b600080fd5b600080fd5b600080fd5b6000601f19601f8301169050919050565b7f4d696e696d756d2031204e46542068617320746f206265206d696e746564207060008201527f6572207472616e73616374696f6e000000000000000000000000000000000000602082015250565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f53616c65206973206e6f74206f70656e20796574000000000000000000000000600082015250565b7f546f6b656e207472616e7366657220746f20636f6e747261637420616464726560008201527f7373206973206e6f7420616c6c6f7765642e0000000000000000000000000000602082015250565b7f496e76616c696420546f6b656e49640000000000000000000000000000000000600082015250565b7f45746865722073656e7420776974682074686973207472616e73616374696f6e60008201527f206973206e6f7420636f72726563740000000000000000000000000000000000602082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f45786365656473206d6178696d756d20737570706c7900000000000000000000600082015250565b50565b7f5472616e73666572206661696c65642e00000000000000000000000000000000600082015250565b7f4d6178696d756d203135204e4654732063616e206265206d696e74656420706560008201527f72207472616e73616374696f6e00000000000000000000000000000000000000602082015250565b613d4981613842565b8114613d5457600080fd5b50565b613d6081613854565b8114613d6b57600080fd5b50565b613d7781613860565b8114613d8257600080fd5b50565b613d8e816138ac565b8114613d9957600080fd5b5056fea26469706673582212207d490c5cbf35abc37b4f721104914699ab1a45fb65bd67c85963faca7c06b3ac64736f6c63430008070033

Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)

00000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000000

-----Decoded View---------------
Arg [0] : baseURI (string):

-----Encoded View---------------
2 Constructor Arguments found :
Arg [0] : 0000000000000000000000000000000000000000000000000000000000000020
Arg [1] : 0000000000000000000000000000000000000000000000000000000000000000


Deployed Bytecode Sourcemap

87:1903:2:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;6608:372:5;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;9218:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;11018:204;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;10284:668;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;509:97:2;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;12256:170:5;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;5431:1105;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;245:41:2;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;912:182;;;;;;;;;;;;;:::i;:::-;;1102:764;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;12497:185:5;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;2639:37;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;4418:713;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;3497:140;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;614:101:2;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;9027:124:5;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;7044:206;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;1746:148:12;;;;;;;;;;;;;:::i;:::-;;826:78:2;;;;;;;;;;;;;:::i;:::-;;2685:106:5;;;;;;;;;;;;;:::i;:::-;;1095:87:12;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;9387:104:5;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;424:28:2;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;333:40;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;11294:660:5;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;723:95:2;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;12753:342:5;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;3437:51;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;9562:318;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;12025:164;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;2049:244:12;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;6608:372:5;6710:4;6762:25;6747:40;;;:11;:40;;;;:105;;;;6819:33;6804:48;;;:11;:48;;;;6747:105;:172;;;;6884:35;6869:50;;;:11;:50;;;;6747:172;:225;;;;6936:36;6960:11;6936:23;:36::i;:::-;6747:225;6727:245;;6608:372;;;:::o;9218:100::-;9272:13;9305:5;9298:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;9218:100;:::o;11018:204::-;11086:7;11111:16;11119:7;11111;:16::i;:::-;11106:64;;11136:34;;;;;;;;;;;;;;11106:64;11190:15;:24;11206:7;11190:24;;;;;;;;;;;;;;;;;;;;;11183:31;;11018:204;;;:::o;10284:668::-;10357:13;10373:24;10389:7;10373:15;:24::i;:::-;10357:40;;10418:5;10412:11;;:2;:11;;;10408:48;;;10432:24;;;;;;;;;;;;;;10408:48;10489:5;10473:21;;:12;:10;:12::i;:::-;:21;;;;:63;;;;;10499:37;10516:5;10523:12;:10;:12::i;:::-;10499:16;:37::i;:::-;10498:38;10473:63;10469:138;;;10560:35;;;;;;;;;;;;;;10469:138;10621:17;;;;;;;;;;;10620:18;:51;;;;;10643:19;:28;10663:7;10643:28;;;;;;;;;;;;;;;;;;;;;10642:29;10620:51;10617:328;;;10691:15;:2;:13;;;:15::i;:::-;10687:186;;;10727:61;;;;;;;;;;:::i;:::-;;;;;;;;10687:186;10829:28;10838:2;10842:7;10851:5;10829:8;:28::i;:::-;10617:328;;;10905:28;10914:2;10918:7;10927:5;10905:8;:28::i;:::-;10617:328;10346:606;10284:668;;:::o;509:97:2:-;562:7;589:9;;582:16;;509:97;:::o;12256:170:5:-;12390:28;12400:4;12406:2;12410:7;12390:9;:28::i;:::-;12256:170;;;:::o;5431:1105::-;5520:7;5553:16;5563:5;5553:9;:16::i;:::-;5544:5;:25;5540:61;;5578:23;;;;;;;;;;;;;;5540:61;5612:22;5637:13;;5612:38;;5661:19;5691:25;5892:9;5887:557;5907:14;5903:1;:18;5887:557;;;5947:31;5981:11;:14;5993:1;5981:14;;;;;;;;;;;5947:48;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;6018:9;:16;;;6014:73;;;6059:8;;;6014:73;6135:1;6109:28;;:9;:14;;;:28;;;6105:111;;6182:9;:14;;;6162:34;;6105:111;6259:5;6238:26;;:17;:26;;;6234:195;;;6308:5;6293:11;:20;6289:85;;;6349:1;6342:8;;;;;;;;;6289:85;6396:13;;;;;;;6234:195;5928:516;5887:557;5923:3;;;;;;;5887:557;;;;6520:8;;;5431:1105;;;;;:::o;245:41:2:-;282:4;245:41;:::o;912:182::-;1326:12:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;963:12:2::1;989:10;981:24;;1013:21;981:58;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;962:77;;;1058:7;1050:36;;;;;;;;;;;;:::i;:::-;;;;;;;;;951:143;912:182::o:0;1102:764::-;1173:14;1190:13;:11;:13::i;:::-;1173:30;;282:4;1233:6;1224;:15;;;;:::i;:::-;:29;;1216:64;;;;;;;;;;;;:::i;:::-;;;;;;;;;1308:1;1299:6;:10;1291:69;;;;;;;;;;;;:::i;:::-;;;;;;;;;1391:7;:5;:7::i;:::-;1377:21;;:10;:21;;;1373:380;;1423:8;;;;;;;;;;;1415:41;;;;;;;;;;;;:::i;:::-;;;;;;;;;1507:2;1497:6;:12;;1471:119;;;;;;;;;;;;:::i;:::-;;;;;;;;;1652:6;1644:5;;:14;;;;:::i;:::-;1631:9;:27;;1605:136;;;;;;;;;;;;:::i;:::-;;;;;;;;;1373:380;1776:6;1763:9;;:19;;;;;;;:::i;:::-;;;;;;;;1799:22;1809:3;1814:6;1799:9;:22::i;:::-;1837:14;1844:6;1837:14;;;;;;:::i;:::-;;;;;;;;1162:704;1102:764;;:::o;12497:185:5:-;12635:39;12652:4;12658:2;12662:7;12635:39;;;;;;;;;;;;:16;:39::i;:::-;12497:185;;;:::o;2639:37::-;;;;;;;;;;;;;:::o;4418:713::-;4485:7;4505:22;4530:13;;4505:38;;4554:19;4749:9;4744:328;4764:14;4760:1;:18;4744:328;;;4804:31;4838:11;:14;4850:1;4838:14;;;;;;;;;;;4804:48;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;4876:9;:16;;;4871:186;;4936:5;4921:11;:20;4917:85;;;4977:1;4970:8;;;;;;;;4917:85;5024:13;;;;;;;4871:186;4785:287;4780:3;;;;;;;4744:328;;;;5100:23;;;;;;;;;;;;;;4418:713;;;;:::o;3497:140::-;1326:12:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;3623:6:5::1;3591:19;:29;3611:8;3591:29;;;;;;;;;;;;:38;;;;;;;;;;;;;;;;;;3497:140:::0;;:::o;614:101:2:-;1326:12:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;700:7:2::1;685:12;:22;;;;;;;;;;;;:::i;:::-;;614:101:::0;:::o;9027:124:5:-;9091:7;9118:20;9130:7;9118:11;:20::i;:::-;:25;;;9111:32;;9027:124;;;:::o;7044:206::-;7108:7;7149:1;7132:19;;:5;:19;;;7128:60;;;7160:28;;;;;;;;;;;;;;7128:60;7214:12;:19;7227:5;7214:19;;;;;;;;;;;;;;;:27;;;;;;;;;;;;7206:36;;7199:43;;7044:206;;;:::o;1746:148:12:-;1326:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;1853:1:::1;1816:40;;1837:6;::::0;::::1;;;;;;;;1816:40;;;;;;;;;;;;1884:1;1867:6:::0;::::1;:19;;;;;;;;;;;;;;;;;;1746:148::o:0;826:78:2:-;1326:12:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;888:8:2::1;;;;;;;;;;;887:9;876:8;;:20;;;;;;;;;;;;;;;;;;826:78::o:0;2685:106:5:-;1326:12:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;2766:17:5::1;;;;;;;;;;;2765:18;2745:17;;:38;;;;;;;;;;;;;;;;;;2685:106::o:0;1095:87:12:-;1141:7;1168:6;;;;;;;;;;;1161:13;;1095:87;:::o;9387:104:5:-;9443:13;9476:7;9469:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;9387:104;:::o;424:28:2:-;;;;;;;;;;;;;:::o;333:40::-;;;;:::o;11294:660:5:-;11397:12;:10;:12::i;:::-;11385:24;;:8;:24;;;11381:54;;;11418:17;;;;;;;;;;;;;;11381:54;11460:17;;;;;;;;;;;11456:491;;11497:21;:8;:19;;;:21::i;:::-;11493:289;;;11539:61;;;;;;;;;;:::i;:::-;;;;;;;;11493:289;11686:8;11641:18;:32;11660:12;:10;:12::i;:::-;11641:32;;;;;;;;;;;;;;;:42;11674:8;11641:42;;;;;;;;;;;;;;;;:53;;;;;;;;;;;;;;;;;;11747:8;11718:48;;11733:12;:10;:12::i;:::-;11718:48;;;11757:8;11718:48;;;;;;:::i;:::-;;;;;;;;11456:491;;;11859:8;11814:18;:32;11833:12;:10;:12::i;:::-;11814:32;;;;;;;;;;;;;;;:42;11847:8;11814:42;;;;;;;;;;;;;;;;:53;;;;;;;;;;;;;;;;;;11916:8;11887:48;;11902:12;:10;:12::i;:::-;11887:48;;;11926:8;11887:48;;;;;;:::i;:::-;;;;;;;;11456:491;11294:660;;:::o;723:95:2:-;1326:12:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;801:9:2::1;793:5;:17;;;;723:95:::0;:::o;12753:342:5:-;12920:28;12930:4;12936:2;12940:7;12920:9;:28::i;:::-;12964:48;12987:4;12993:2;12997:7;13006:5;12964:22;:48::i;:::-;12959:129;;13036:40;;;;;;;;;;;;;;12959:129;12753:342;;;;:::o;3437:51::-;;;;;;;;;;;;;;;;;;;;;;:::o;9562:318::-;9635:13;9666:16;9674:7;9666;:16::i;:::-;9661:59;;9691:29;;;;;;;;;;;;;;9661:59;9733:21;9757:10;:8;:10::i;:::-;9733:34;;9810:1;9791:7;9785:21;:26;;:87;;;;;;;;;;;;;;;;;9838:7;9847:18;:7;:16;:18::i;:::-;9821:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;9785:87;9778:94;;;9562:318;;;:::o;12025:164::-;12122:4;12146:18;:25;12165:5;12146:25;;;;;;;;;;;;;;;:35;12172:8;12146:35;;;;;;;;;;;;;;;;;;;;;;;;;12139:42;;12025:164;;;;:::o;2049:244:12:-;1326:12;:10;:12::i;:::-;1315:23;;:7;:5;:7::i;:::-;:23;;;1307:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;2158:1:::1;2138:22;;:8;:22;;;;2130:73;;;;;;;;;;;;:::i;:::-;;;;;;;;;2248:8;2219:38;;2240:6;::::0;::::1;;;;;;;;2219:38;;;;;;;;;;;;2277:8;2268:6;::::0;:17:::1;;;;;;;;;;;;;;;;;;2049:244:::0;:::o;854:157:3:-;939:4;978:25;963:40;;;:11;:40;;;;956:47;;854:157;;;:::o;13350:194:5:-;13407:4;13442:1;13432:7;:11;13424:39;;;;;;;;;;;;:::i;:::-;;;;;;;;;13491:13;;13481:7;:23;:55;;;;;13509:11;:20;13521:7;13509:20;;;;;;;;;;;:27;;;;;;;;;;;;13508:28;13481:55;13474:62;;13350:194;;;:::o;601:98:1:-;654:7;681:10;674:17;;601:98;:::o;1195:326:0:-;1255:4;1512:1;1490:7;:19;;;:23;1483:30;;1195:326;;;:::o;20606:196:5:-;20748:2;20721:15;:24;20737:7;20721:24;;;;;;;;;;;;:29;;;;;;;;;;;;;;;;;;20786:7;20782:2;20766:28;;20775:5;20766:28;;;;;;;;;;;;20606:196;;;:::o;16107:2112::-;16222:35;16260:20;16272:7;16260:11;:20::i;:::-;16222:58;;16293:22;16335:13;:18;;;16319:34;;:12;:10;:12::i;:::-;:34;;;:101;;;;16370:50;16387:13;:18;;;16407:12;:10;:12::i;:::-;16370:16;:50::i;:::-;16319:101;:154;;;;16461:12;:10;:12::i;:::-;16437:36;;:20;16449:7;16437:11;:20::i;:::-;:36;;;16319:154;16293:181;;16492:17;16487:66;;16518:35;;;;;;;;;;;;;;16487:66;16590:4;16568:26;;:13;:18;;;:26;;;16564:67;;16603:28;;;;;;;;;;;;;;16564:67;16660:1;16646:16;;:2;:16;;;16642:52;;;16671:23;;;;;;;;;;;;;;16642:52;16707:43;16729:4;16735:2;16739:7;16748:1;16707:21;:43::i;:::-;16815:49;16832:1;16836:7;16845:13;:18;;;16815:8;:49::i;:::-;17190:1;17160:12;:18;17173:4;17160:18;;;;;;;;;;;;;;;:26;;;:31;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;17234:1;17206:12;:16;17219:2;17206:16;;;;;;;;;;;;;;;:24;;;:29;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;17280:2;17252:11;:20;17264:7;17252:20;;;;;;;;;;;:25;;;:30;;;;;;;;;;;;;;;;;;17342:15;17297:11;:20;17309:7;17297:20;;;;;;;;;;;:35;;;:61;;;;;;;;;;;;;;;;;;17610:19;17642:1;17632:7;:11;17610:33;;17703:1;17662:43;;:11;:24;17674:11;17662:24;;;;;;;;;;;:29;;;;;;;;;;;;:43;;;17658:445;;;17887:13;;17873:11;:27;17869:219;;;17957:13;:18;;;17925:11;:24;17937:11;17925:24;;;;;;;;;;;:29;;;:50;;;;;;;;;;;;;;;;;;18040:13;:28;;;17998:11;:24;18010:11;17998:24;;;;;;;;;;;:39;;;:70;;;;;;;;;;;;;;;;;;17869:219;17658:445;17135:979;18150:7;18146:2;18131:27;;18140:4;18131:27;;;;;;;;;;;;18169:42;18190:4;18196:2;18200:7;18209:1;18169:20;:42::i;:::-;16211:2008;;16107:2112;;;:::o;13552:104::-;13621:27;13631:2;13635:8;13621:27;;;;;;;;;;;;:9;:27::i;:::-;13552:104;;:::o;7882:1083::-;7943:21;;:::i;:::-;7977:12;7992:7;7977:22;;8048:13;;8041:4;:20;8037:861;;;8082:31;8116:11;:17;8128:4;8116:17;;;;;;;;;;;8082:51;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;8157:9;:16;;;8152:731;;8228:1;8202:28;;:9;:14;;;:28;;;8198:101;;8266:9;8259:16;;;;;;8198:101;8603:261;8610:4;8603:261;;;8643:6;;;;;;;;8688:11;:17;8700:4;8688:17;;;;;;;;;;;8676:29;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;8762:1;8736:28;;:9;:14;;;:28;;;8732:109;;8804:9;8797:16;;;;;;8732:109;8603:261;;;8152:731;8063:835;8037:861;8926:31;;;;;;;;;;;;;;7882:1083;;;;:::o;21367:473::-;21522:4;21543:17;;;;;;;;;;;21542:18;:51;;;;;21565:19;:28;21585:7;21565:28;;;;;;;;;;;;;;;;;;;;;21564:29;21542:51;21539:294;;;21613:15;:2;:13;;;:15::i;:::-;21609:169;;;21649:61;;;;;;;;;;:::i;:::-;;;;;;;;21609:169;21758:4;21751:11;;;;21539:294;21817:4;21810:11;;21367:473;;;;;;;:::o;1874:113:2:-;1934:13;1967:12;1960:19;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;1874:113;:::o;342:723:13:-;398:13;628:1;619:5;:10;615:53;;;646:10;;;;;;;;;;;;;;;;;;;;;615:53;678:12;693:5;678:20;;709:14;734:78;749:1;741:4;:9;734:78;;767:8;;;;;:::i;:::-;;;;798:2;790:10;;;;;:::i;:::-;;;734:78;;;822:19;854:6;844:17;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;822:39;;872:154;888:1;879:5;:10;872:154;;916:1;906:11;;;;;:::i;:::-;;;983:2;975:5;:10;;;;:::i;:::-;962:2;:24;;;;:::i;:::-;949:39;;932:6;939;932:14;;;;;;;;:::i;:::-;;;;;:56;;;;;;;;;;;1012:2;1003:11;;;;;:::i;:::-;;;872:154;;;1050:6;1036:21;;;;;342:723;;;;:::o;22488:159:5:-;;;;;:::o;23306:158::-;;;;;:::o;14019:163::-;14142:32;14148:2;14152:8;14162:5;14169:4;14142:5;:32::i;:::-;14019:163;;;:::o;14441:1412::-;14580:20;14603:13;;14580:36;;14645:1;14631:16;;:2;:16;;;14627:48;;;14656:19;;;;;;;;;;;;;;14627:48;14702:1;14690:8;:13;14686:44;;;14712:18;;;;;;;;;;;;;;14686:44;14743:61;14773:1;14777:2;14781:12;14795:8;14743:21;:61::i;:::-;15116:8;15081:12;:16;15094:2;15081:16;;;;;;;;;;;;;;;:24;;;:44;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;15180:8;15140:12;:16;15153:2;15140:16;;;;;;;;;;;;;;;:29;;;:49;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;15239:2;15206:11;:25;15218:12;15206:25;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;15306:15;15256:11;:25;15268:12;15256:25;;;;;;;;;;;:40;;;:66;;;;;;;;;;;;;;;;;;15339:20;15362:12;15339:35;;15396:9;15391:328;15411:8;15407:1;:12;15391:328;;;15475:12;15471:2;15450:38;;15467:1;15450:38;;;;;;;;;;;;15511:4;:68;;;;;15520:59;15551:1;15555:2;15559:12;15573:5;15520:22;:59::i;:::-;15519:60;15511:68;15507:164;;;15611:40;;;;;;;;;;;;;;15507:164;15689:14;;;;;;;15421:3;;;;;;;15391:328;;;;15751:12;15735:13;:28;;;;15056:719;15785:60;15814:1;15818:2;15822:12;15836:8;15785:20;:60::i;:::-;14569:1284;14441:1412;;;;:::o;-1:-1:-1:-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;:::o;7:410:14:-;84:5;109:65;125:48;166:6;125:48;:::i;:::-;109:65;:::i;:::-;100:74;;197:6;190:5;183:21;235:4;228:5;224:16;273:3;264:6;259:3;255:16;252:25;249:112;;;280:79;;:::i;:::-;249:112;370:41;404:6;399:3;394;370:41;:::i;:::-;90:327;7:410;;;;;:::o;423:412::-;501:5;526:66;542:49;584:6;542:49;:::i;:::-;526:66;:::i;:::-;517:75;;615:6;608:5;601:21;653:4;646:5;642:16;691:3;682:6;677:3;673:16;670:25;667:112;;;698:79;;:::i;:::-;667:112;788:41;822:6;817:3;812;788:41;:::i;:::-;507:328;423:412;;;;;:::o;841:139::-;887:5;925:6;912:20;903:29;;941:33;968:5;941:33;:::i;:::-;841:139;;;;:::o;986:133::-;1029:5;1067:6;1054:20;1045:29;;1083:30;1107:5;1083:30;:::i;:::-;986:133;;;;:::o;1125:137::-;1170:5;1208:6;1195:20;1186:29;;1224:32;1250:5;1224:32;:::i;:::-;1125:137;;;;:::o;1281:338::-;1336:5;1385:3;1378:4;1370:6;1366:17;1362:27;1352:122;;1393:79;;:::i;:::-;1352:122;1510:6;1497:20;1535:78;1609:3;1601:6;1594:4;1586:6;1582:17;1535:78;:::i;:::-;1526:87;;1342:277;1281:338;;;;:::o;1639:340::-;1695:5;1744:3;1737:4;1729:6;1725:17;1721:27;1711:122;;1752:79;;:::i;:::-;1711:122;1869:6;1856:20;1894:79;1969:3;1961:6;1954:4;1946:6;1942:17;1894:79;:::i;:::-;1885:88;;1701:278;1639:340;;;;:::o;1985:139::-;2031:5;2069:6;2056:20;2047:29;;2085:33;2112:5;2085:33;:::i;:::-;1985:139;;;;:::o;2130:329::-;2189:6;2238:2;2226:9;2217:7;2213:23;2209:32;2206:119;;;2244:79;;:::i;:::-;2206:119;2364:1;2389:53;2434:7;2425:6;2414:9;2410:22;2389:53;:::i;:::-;2379:63;;2335:117;2130:329;;;;:::o;2465:474::-;2533:6;2541;2590:2;2578:9;2569:7;2565:23;2561:32;2558:119;;;2596:79;;:::i;:::-;2558:119;2716:1;2741:53;2786:7;2777:6;2766:9;2762:22;2741:53;:::i;:::-;2731:63;;2687:117;2843:2;2869:53;2914:7;2905:6;2894:9;2890:22;2869:53;:::i;:::-;2859:63;;2814:118;2465:474;;;;;:::o;2945:619::-;3022:6;3030;3038;3087:2;3075:9;3066:7;3062:23;3058:32;3055:119;;;3093:79;;:::i;:::-;3055:119;3213:1;3238:53;3283:7;3274:6;3263:9;3259:22;3238:53;:::i;:::-;3228:63;;3184:117;3340:2;3366:53;3411:7;3402:6;3391:9;3387:22;3366:53;:::i;:::-;3356:63;;3311:118;3468:2;3494:53;3539:7;3530:6;3519:9;3515:22;3494:53;:::i;:::-;3484:63;;3439:118;2945:619;;;;;:::o;3570:943::-;3665:6;3673;3681;3689;3738:3;3726:9;3717:7;3713:23;3709:33;3706:120;;;3745:79;;:::i;:::-;3706:120;3865:1;3890:53;3935:7;3926:6;3915:9;3911:22;3890:53;:::i;:::-;3880:63;;3836:117;3992:2;4018:53;4063:7;4054:6;4043:9;4039:22;4018:53;:::i;:::-;4008:63;;3963:118;4120:2;4146:53;4191:7;4182:6;4171:9;4167:22;4146:53;:::i;:::-;4136:63;;4091:118;4276:2;4265:9;4261:18;4248:32;4307:18;4299:6;4296:30;4293:117;;;4329:79;;:::i;:::-;4293:117;4434:62;4488:7;4479:6;4468:9;4464:22;4434:62;:::i;:::-;4424:72;;4219:287;3570:943;;;;;;;:::o;4519:468::-;4584:6;4592;4641:2;4629:9;4620:7;4616:23;4612:32;4609:119;;;4647:79;;:::i;:::-;4609:119;4767:1;4792:53;4837:7;4828:6;4817:9;4813:22;4792:53;:::i;:::-;4782:63;;4738:117;4894:2;4920:50;4962:7;4953:6;4942:9;4938:22;4920:50;:::i;:::-;4910:60;;4865:115;4519:468;;;;;:::o;4993:474::-;5061:6;5069;5118:2;5106:9;5097:7;5093:23;5089:32;5086:119;;;5124:79;;:::i;:::-;5086:119;5244:1;5269:53;5314:7;5305:6;5294:9;5290:22;5269:53;:::i;:::-;5259:63;;5215:117;5371:2;5397:53;5442:7;5433:6;5422:9;5418:22;5397:53;:::i;:::-;5387:63;;5342:118;4993:474;;;;;:::o;5473:327::-;5531:6;5580:2;5568:9;5559:7;5555:23;5551:32;5548:119;;;5586:79;;:::i;:::-;5548:119;5706:1;5731:52;5775:7;5766:6;5755:9;5751:22;5731:52;:::i;:::-;5721:62;;5677:116;5473:327;;;;:::o;5806:509::-;5875:6;5924:2;5912:9;5903:7;5899:23;5895:32;5892:119;;;5930:79;;:::i;:::-;5892:119;6078:1;6067:9;6063:17;6050:31;6108:18;6100:6;6097:30;6094:117;;;6130:79;;:::i;:::-;6094:117;6235:63;6290:7;6281:6;6270:9;6266:22;6235:63;:::i;:::-;6225:73;;6021:287;5806:509;;;;:::o;6321:329::-;6380:6;6429:2;6417:9;6408:7;6404:23;6400:32;6397:119;;;6435:79;;:::i;:::-;6397:119;6555:1;6580:53;6625:7;6616:6;6605:9;6601:22;6580:53;:::i;:::-;6570:63;;6526:117;6321:329;;;;:::o;6656:468::-;6721:6;6729;6778:2;6766:9;6757:7;6753:23;6749:32;6746:119;;;6784:79;;:::i;:::-;6746:119;6904:1;6929:53;6974:7;6965:6;6954:9;6950:22;6929:53;:::i;:::-;6919:63;;6875:117;7031:2;7057:50;7099:7;7090:6;7079:9;7075:22;7057:50;:::i;:::-;7047:60;;7002:115;6656:468;;;;;:::o;7130:118::-;7217:24;7235:5;7217:24;:::i;:::-;7212:3;7205:37;7130:118;;:::o;7254:109::-;7335:21;7350:5;7335:21;:::i;:::-;7330:3;7323:34;7254:109;;:::o;7369:364::-;7457:3;7485:39;7518:5;7485:39;:::i;:::-;7540:71;7604:6;7599:3;7540:71;:::i;:::-;7533:78;;7620:52;7665:6;7660:3;7653:4;7646:5;7642:16;7620:52;:::i;:::-;7697:29;7719:6;7697:29;:::i;:::-;7692:3;7688:39;7681:46;;7461:272;7369:364;;;;:::o;7739:377::-;7845:3;7873:39;7906:5;7873:39;:::i;:::-;7928:89;8010:6;8005:3;7928:89;:::i;:::-;7921:96;;8026:52;8071:6;8066:3;8059:4;8052:5;8048:16;8026:52;:::i;:::-;8103:6;8098:3;8094:16;8087:23;;7849:267;7739:377;;;;:::o;8122:366::-;8264:3;8285:67;8349:2;8344:3;8285:67;:::i;:::-;8278:74;;8361:93;8450:3;8361:93;:::i;:::-;8479:2;8474:3;8470:12;8463:19;;8122:366;;;:::o;8494:::-;8636:3;8657:67;8721:2;8716:3;8657:67;:::i;:::-;8650:74;;8733:93;8822:3;8733:93;:::i;:::-;8851:2;8846:3;8842:12;8835:19;;8494:366;;;:::o;8866:::-;9008:3;9029:67;9093:2;9088:3;9029:67;:::i;:::-;9022:74;;9105:93;9194:3;9105:93;:::i;:::-;9223:2;9218:3;9214:12;9207:19;;8866:366;;;:::o;9238:::-;9380:3;9401:67;9465:2;9460:3;9401:67;:::i;:::-;9394:74;;9477:93;9566:3;9477:93;:::i;:::-;9595:2;9590:3;9586:12;9579:19;;9238:366;;;:::o;9610:::-;9752:3;9773:67;9837:2;9832:3;9773:67;:::i;:::-;9766:74;;9849:93;9938:3;9849:93;:::i;:::-;9967:2;9962:3;9958:12;9951:19;;9610:366;;;:::o;9982:::-;10124:3;10145:67;10209:2;10204:3;10145:67;:::i;:::-;10138:74;;10221:93;10310:3;10221:93;:::i;:::-;10339:2;10334:3;10330:12;10323:19;;9982:366;;;:::o;10354:::-;10496:3;10517:67;10581:2;10576:3;10517:67;:::i;:::-;10510:74;;10593:93;10682:3;10593:93;:::i;:::-;10711:2;10706:3;10702:12;10695:19;;10354:366;;;:::o;10726:::-;10868:3;10889:67;10953:2;10948:3;10889:67;:::i;:::-;10882:74;;10965:93;11054:3;10965:93;:::i;:::-;11083:2;11078:3;11074:12;11067:19;;10726:366;;;:::o;11098:398::-;11257:3;11278:83;11359:1;11354:3;11278:83;:::i;:::-;11271:90;;11370:93;11459:3;11370:93;:::i;:::-;11488:1;11483:3;11479:11;11472:18;;11098:398;;;:::o;11502:366::-;11644:3;11665:67;11729:2;11724:3;11665:67;:::i;:::-;11658:74;;11741:93;11830:3;11741:93;:::i;:::-;11859:2;11854:3;11850:12;11843:19;;11502:366;;;:::o;11874:::-;12016:3;12037:67;12101:2;12096:3;12037:67;:::i;:::-;12030:74;;12113:93;12202:3;12113:93;:::i;:::-;12231:2;12226:3;12222:12;12215:19;;11874:366;;;:::o;12246:118::-;12333:24;12351:5;12333:24;:::i;:::-;12328:3;12321:37;12246:118;;:::o;12370:435::-;12550:3;12572:95;12663:3;12654:6;12572:95;:::i;:::-;12565:102;;12684:95;12775:3;12766:6;12684:95;:::i;:::-;12677:102;;12796:3;12789:10;;12370:435;;;;;:::o;12811:379::-;12995:3;13017:147;13160:3;13017:147;:::i;:::-;13010:154;;13181:3;13174:10;;12811:379;;;:::o;13196:222::-;13289:4;13327:2;13316:9;13312:18;13304:26;;13340:71;13408:1;13397:9;13393:17;13384:6;13340:71;:::i;:::-;13196:222;;;;:::o;13424:210::-;13511:4;13549:2;13538:9;13534:18;13526:26;;13562:65;13624:1;13613:9;13609:17;13600:6;13562:65;:::i;:::-;13424:210;;;;:::o;13640:313::-;13753:4;13791:2;13780:9;13776:18;13768:26;;13840:9;13834:4;13830:20;13826:1;13815:9;13811:17;13804:47;13868:78;13941:4;13932:6;13868:78;:::i;:::-;13860:86;;13640:313;;;;:::o;13959:419::-;14125:4;14163:2;14152:9;14148:18;14140:26;;14212:9;14206:4;14202:20;14198:1;14187:9;14183:17;14176:47;14240:131;14366:4;14240:131;:::i;:::-;14232:139;;13959:419;;;:::o;14384:::-;14550:4;14588:2;14577:9;14573:18;14565:26;;14637:9;14631:4;14627:20;14623:1;14612:9;14608:17;14601:47;14665:131;14791:4;14665:131;:::i;:::-;14657:139;;14384:419;;;:::o;14809:::-;14975:4;15013:2;15002:9;14998:18;14990:26;;15062:9;15056:4;15052:20;15048:1;15037:9;15033:17;15026:47;15090:131;15216:4;15090:131;:::i;:::-;15082:139;;14809:419;;;:::o;15234:::-;15400:4;15438:2;15427:9;15423:18;15415:26;;15487:9;15481:4;15477:20;15473:1;15462:9;15458:17;15451:47;15515:131;15641:4;15515:131;:::i;:::-;15507:139;;15234:419;;;:::o;15659:::-;15825:4;15863:2;15852:9;15848:18;15840:26;;15912:9;15906:4;15902:20;15898:1;15887:9;15883:17;15876:47;15940:131;16066:4;15940:131;:::i;:::-;15932:139;;15659:419;;;:::o;16084:::-;16250:4;16288:2;16277:9;16273:18;16265:26;;16337:9;16331:4;16327:20;16323:1;16312:9;16308:17;16301:47;16365:131;16491:4;16365:131;:::i;:::-;16357:139;;16084:419;;;:::o;16509:::-;16675:4;16713:2;16702:9;16698:18;16690:26;;16762:9;16756:4;16752:20;16748:1;16737:9;16733:17;16726:47;16790:131;16916:4;16790:131;:::i;:::-;16782:139;;16509:419;;;:::o;16934:::-;17100:4;17138:2;17127:9;17123:18;17115:26;;17187:9;17181:4;17177:20;17173:1;17162:9;17158:17;17151:47;17215:131;17341:4;17215:131;:::i;:::-;17207:139;;16934:419;;;:::o;17359:::-;17525:4;17563:2;17552:9;17548:18;17540:26;;17612:9;17606:4;17602:20;17598:1;17587:9;17583:17;17576:47;17640:131;17766:4;17640:131;:::i;:::-;17632:139;;17359:419;;;:::o;17784:::-;17950:4;17988:2;17977:9;17973:18;17965:26;;18037:9;18031:4;18027:20;18023:1;18012:9;18008:17;18001:47;18065:131;18191:4;18065:131;:::i;:::-;18057:139;;17784:419;;;:::o;18209:222::-;18302:4;18340:2;18329:9;18325:18;18317:26;;18353:71;18421:1;18410:9;18406:17;18397:6;18353:71;:::i;:::-;18209:222;;;;:::o;18437:129::-;18471:6;18498:20;;:::i;:::-;18488:30;;18527:33;18555:4;18547:6;18527:33;:::i;:::-;18437:129;;;:::o;18572:75::-;18605:6;18638:2;18632:9;18622:19;;18572:75;:::o;18653:307::-;18714:4;18804:18;18796:6;18793:30;18790:56;;;18826:18;;:::i;:::-;18790:56;18864:29;18886:6;18864:29;:::i;:::-;18856:37;;18948:4;18942;18938:15;18930:23;;18653:307;;;:::o;18966:308::-;19028:4;19118:18;19110:6;19107:30;19104:56;;;19140:18;;:::i;:::-;19104:56;19178:29;19200:6;19178:29;:::i;:::-;19170:37;;19262:4;19256;19252:15;19244:23;;18966:308;;;:::o;19280:99::-;19332:6;19366:5;19360:12;19350:22;;19280:99;;;:::o;19385:147::-;19486:11;19523:3;19508:18;;19385:147;;;;:::o;19538:169::-;19622:11;19656:6;19651:3;19644:19;19696:4;19691:3;19687:14;19672:29;;19538:169;;;;:::o;19713:148::-;19815:11;19852:3;19837:18;;19713:148;;;;:::o;19867:305::-;19907:3;19926:20;19944:1;19926:20;:::i;:::-;19921:25;;19960:20;19978:1;19960:20;:::i;:::-;19955:25;;20114:1;20046:66;20042:74;20039:1;20036:81;20033:107;;;20120:18;;:::i;:::-;20033:107;20164:1;20161;20157:9;20150:16;;19867:305;;;;:::o;20178:185::-;20218:1;20235:20;20253:1;20235:20;:::i;:::-;20230:25;;20269:20;20287:1;20269:20;:::i;:::-;20264:25;;20308:1;20298:35;;20313:18;;:::i;:::-;20298:35;20355:1;20352;20348:9;20343:14;;20178:185;;;;:::o;20369:348::-;20409:7;20432:20;20450:1;20432:20;:::i;:::-;20427:25;;20466:20;20484:1;20466:20;:::i;:::-;20461:25;;20654:1;20586:66;20582:74;20579:1;20576:81;20571:1;20564:9;20557:17;20553:105;20550:131;;;20661:18;;:::i;:::-;20550:131;20709:1;20706;20702:9;20691:20;;20369:348;;;;:::o;20723:191::-;20763:4;20783:20;20801:1;20783:20;:::i;:::-;20778:25;;20817:20;20835:1;20817:20;:::i;:::-;20812:25;;20856:1;20853;20850:8;20847:34;;;20861:18;;:::i;:::-;20847:34;20906:1;20903;20899:9;20891:17;;20723:191;;;;:::o;20920:96::-;20957:7;20986:24;21004:5;20986:24;:::i;:::-;20975:35;;20920:96;;;:::o;21022:90::-;21056:7;21099:5;21092:13;21085:21;21074:32;;21022:90;;;:::o;21118:149::-;21154:7;21194:66;21187:5;21183:78;21172:89;;21118:149;;;:::o;21273:126::-;21310:7;21350:42;21343:5;21339:54;21328:65;;21273:126;;;:::o;21405:77::-;21442:7;21471:5;21460:16;;21405:77;;;:::o;21488:154::-;21572:6;21567:3;21562;21549:30;21634:1;21625:6;21620:3;21616:16;21609:27;21488:154;;;:::o;21648:307::-;21716:1;21726:113;21740:6;21737:1;21734:13;21726:113;;;21825:1;21820:3;21816:11;21810:18;21806:1;21801:3;21797:11;21790:39;21762:2;21759:1;21755:10;21750:15;;21726:113;;;21857:6;21854:1;21851:13;21848:101;;;21937:1;21928:6;21923:3;21919:16;21912:27;21848:101;21697:258;21648:307;;;:::o;21961:320::-;22005:6;22042:1;22036:4;22032:12;22022:22;;22089:1;22083:4;22079:12;22110:18;22100:81;;22166:4;22158:6;22154:17;22144:27;;22100:81;22228:2;22220:6;22217:14;22197:18;22194:38;22191:84;;;22247:18;;:::i;:::-;22191:84;22012:269;21961:320;;;:::o;22287:281::-;22370:27;22392:4;22370:27;:::i;:::-;22362:6;22358:40;22500:6;22488:10;22485:22;22464:18;22452:10;22449:34;22446:62;22443:88;;;22511:18;;:::i;:::-;22443:88;22551:10;22547:2;22540:22;22330:238;22287:281;;:::o;22574:233::-;22613:3;22636:24;22654:5;22636:24;:::i;:::-;22627:33;;22682:66;22675:5;22672:77;22669:103;;;22752:18;;:::i;:::-;22669:103;22799:1;22792:5;22788:13;22781:20;;22574:233;;;:::o;22813:176::-;22845:1;22862:20;22880:1;22862:20;:::i;:::-;22857:25;;22896:20;22914:1;22896:20;:::i;:::-;22891:25;;22935:1;22925:35;;22940:18;;:::i;:::-;22925:35;22981:1;22978;22974:9;22969:14;;22813:176;;;;:::o;22995:180::-;23043:77;23040:1;23033:88;23140:4;23137:1;23130:15;23164:4;23161:1;23154:15;23181:180;23229:77;23226:1;23219:88;23326:4;23323:1;23316:15;23350:4;23347:1;23340:15;23367:180;23415:77;23412:1;23405:88;23512:4;23509:1;23502:15;23536:4;23533:1;23526:15;23553:180;23601:77;23598:1;23591:88;23698:4;23695:1;23688:15;23722:4;23719:1;23712:15;23739:180;23787:77;23784:1;23777:88;23884:4;23881:1;23874:15;23908:4;23905:1;23898:15;23925:117;24034:1;24031;24024:12;24048:117;24157:1;24154;24147:12;24171:117;24280:1;24277;24270:12;24294:117;24403:1;24400;24393:12;24417:102;24458:6;24509:2;24505:7;24500:2;24493:5;24489:14;24485:28;24475:38;;24417:102;;;:::o;24525:233::-;24665:34;24661:1;24653:6;24649:14;24642:58;24734:16;24729:2;24721:6;24717:15;24710:41;24525:233;:::o;24764:225::-;24904:34;24900:1;24892:6;24888:14;24881:58;24973:8;24968:2;24960:6;24956:15;24949:33;24764:225;:::o;24995:170::-;25135:22;25131:1;25123:6;25119:14;25112:46;24995:170;:::o;25171:237::-;25311:34;25307:1;25299:6;25295:14;25288:58;25380:20;25375:2;25367:6;25363:15;25356:45;25171:237;:::o;25414:165::-;25554:17;25550:1;25542:6;25538:14;25531:41;25414:165;:::o;25585:234::-;25725:34;25721:1;25713:6;25709:14;25702:58;25794:17;25789:2;25781:6;25777:15;25770:42;25585:234;:::o;25825:182::-;25965:34;25961:1;25953:6;25949:14;25942:58;25825:182;:::o;26013:172::-;26153:24;26149:1;26141:6;26137:14;26130:48;26013:172;:::o;26191:114::-;;:::o;26311:166::-;26451:18;26447:1;26439:6;26435:14;26428:42;26311:166;:::o;26483:232::-;26623:34;26619:1;26611:6;26607:14;26600:58;26692:15;26687:2;26679:6;26675:15;26668:40;26483:232;:::o;26721:122::-;26794:24;26812:5;26794:24;:::i;:::-;26787:5;26784:35;26774:63;;26833:1;26830;26823:12;26774:63;26721:122;:::o;26849:116::-;26919:21;26934:5;26919:21;:::i;:::-;26912:5;26909:32;26899:60;;26955:1;26952;26945:12;26899:60;26849:116;:::o;26971:120::-;27043:23;27060:5;27043:23;:::i;:::-;27036:5;27033:34;27023:62;;27081:1;27078;27071:12;27023:62;26971:120;:::o;27097:122::-;27170:24;27188:5;27170:24;:::i;:::-;27163:5;27160:35;27150:63;;27209:1;27206;27199:12;27150:63;27097:122;:::o

Swarm Source

ipfs://7d490c5cbf35abc37b4f721104914699ab1a45fb65bd67c85963faca7c06b3ac
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.