ETH Price: $2,901.38 (+2.82%)
 

Overview

Max Total Supply

10,000 MOOSE

Holders

935

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Balance
2 MOOSE
0x3cd9fca6e859e8a11b9e30e4cae5dcff9c55aaf1
Loading...
Loading
Loading...
Loading
Loading...
Loading

OVERVIEW

A fully on-chain art project. All pixels and metadata are stored on smart contracts. No IPFS, no APIs.

# Exchange Pair Price  24H Volume % Volume

Contract Source Code Verified (Exact Match)

Contract Name:
NfToken

Compiler Version
v0.8.4+commit.c7e474f2

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion, None license

Contract Source Code (Solidity)

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

// File: contracts/Library.sol


pragma solidity ^0.8.0;

library Library {
    string internal constant TABLE =
        "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

    function encode(bytes memory data) internal pure returns (string memory) {
        if (data.length == 0) return "";

        // load the table into memory
        string memory table = TABLE;

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

        // add some extra buffer at the end required for the writing
        string memory result = new string(encodedLen + 32);

        assembly {
            // set the actual output length
            mstore(result, encodedLen)

            // prepare the lookup table
            let tablePtr := add(table, 1)

            // input ptr
            let dataPtr := data
            let endPtr := add(dataPtr, mload(data))

            // result ptr, jump over length
            let resultPtr := add(result, 32)

            // run over the input, 3 bytes at a time
            for {

            } lt(dataPtr, endPtr) {

            } {
                dataPtr := add(dataPtr, 3)

                // read 3 bytes
                let input := mload(dataPtr)

                // write 4 characters
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(18, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(12, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(shr(6, input), 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
                mstore(
                    resultPtr,
                    shl(248, mload(add(tablePtr, and(input, 0x3F))))
                )
                resultPtr := add(resultPtr, 1)
            }

            // padding with '='
            switch mod(mload(data), 3)
            case 1 {
                mstore(sub(resultPtr, 2), shl(240, 0x3d3d))
            }
            case 2 {
                mstore(sub(resultPtr, 1), shl(248, 0x3d))
            }
        }

        return result;
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function toString(uint256 value) internal pure returns (string memory) {
        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);
    }

    function parseInt(string memory _a)
        internal
        pure
        returns (uint8 _parsedInt)
    {
        bytes memory bresult = bytes(_a);
        uint8 mint = 0;
        for (uint8 i = 0; i < bresult.length; i++) {
            if (
                (uint8(uint8(bresult[i])) >= 48) &&
                (uint8(uint8(bresult[i])) <= 57)
            ) {
                mint *= 10;
                mint += uint8(bresult[i]) - 48;
            }
        }
        return mint;
    }

    function substring(
        string memory str,
        uint256 startIndex,
        uint256 endIndex
    ) internal pure returns (string memory) {
        bytes memory strBytes = bytes(str);
        bytes memory result = new bytes(endIndex - startIndex);
        for (uint256 i = startIndex; i < endIndex; i++) {
            result[i - startIndex] = strBytes[i];
        }
        return string(result);
    }

    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize, which returns 0 for contracts in
        // construction, since the code is only stored at the end of the
        // constructor execution.

        uint256 size;
        assembly {
            size := extcodesize(account)
        }
        return size > 0;
    }
}
// File: @openzeppelin/contracts/token/ERC20/IERC20.sol



pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC20 standard as defined in the EIP.
 */
interface IERC20 {
    /**
     * @dev Returns the amount of tokens in existence.
     */
    function totalSupply() external view returns (uint256);

    /**
     * @dev Returns the amount of tokens owned by `account`.
     */
    function balanceOf(address account) external view returns (uint256);

    /**
     * @dev Moves `amount` tokens from the caller's account to `recipient`.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * Emits a {Transfer} event.
     */
    function transfer(address recipient, uint256 amount) external returns (bool);

    /**
     * @dev Returns the remaining number of tokens that `spender` will be
     * allowed to spend on behalf of `owner` through {transferFrom}. This is
     * zero by default.
     *
     * This value changes when {approve} or {transferFrom} are called.
     */
    function allowance(address owner, address spender) external view returns (uint256);

    /**
     * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * IMPORTANT: Beware that changing an allowance with this method brings the risk
     * that someone may use both the old and the new allowance by unfortunate
     * transaction ordering. One possible solution to mitigate this race
     * condition is to first reduce the spender's allowance to 0 and set the
     * desired value afterwards:
     * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
     *
     * Emits an {Approval} event.
     */
    function approve(address spender, uint256 amount) external returns (bool);

    /**
     * @dev Moves `amount` tokens from `sender` to `recipient` using the
     * allowance mechanism. `amount` is then deducted from the caller's
     * allowance.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(
        address sender,
        address recipient,
        uint256 amount
    ) external returns (bool);

    /**
     * @dev Emitted when `value` tokens are moved from one account (`from`) to
     * another (`to`).
     *
     * Note that `value` may be zero.
     */
    event Transfer(address indexed from, address indexed to, uint256 value);

    /**
     * @dev Emitted when the allowance of a `spender` for an `owner` is set by
     * a call to {approve}. `value` is the new allowance.
     */
    event Approval(address indexed owner, address indexed spender, uint256 value);
}

// File: contracts/IToken.sol

// contracts/IToken.sol


pragma solidity ^0.8.0;



interface IToken is IERC20 {
    function burnFrom(address account, uint256 amount) external;
}

// File: @openzeppelin/contracts/utils/math/SafeMath.sol



pragma solidity ^0.8.0;

// CAUTION
// This version of SafeMath should only be used with Solidity 0.8 or later,
// because it relies on the compiler's built in overflow checks.

/**
 * @dev Wrappers over Solidity's arithmetic operations.
 *
 * NOTE: `SafeMath` is no longer needed starting with Solidity 0.8. The compiler
 * now has built in overflow checking.
 */
library SafeMath {
    /**
     * @dev Returns the addition of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        unchecked {
            uint256 c = a + b;
            if (c < a) return (false, 0);
            return (true, c);
        }
    }

    /**
     * @dev Returns the substraction of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        unchecked {
            if (b > a) return (false, 0);
            return (true, a - b);
        }
    }

    /**
     * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        unchecked {
            // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
            // benefit is lost if 'b' is also tested.
            // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
            if (a == 0) return (true, 0);
            uint256 c = a * b;
            if (c / a != b) return (false, 0);
            return (true, c);
        }
    }

    /**
     * @dev Returns the division of two unsigned integers, with a division by zero flag.
     *
     * _Available since v3.4._
     */
    function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        unchecked {
            if (b == 0) return (false, 0);
            return (true, a / b);
        }
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
     *
     * _Available since v3.4._
     */
    function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        unchecked {
            if (b == 0) return (false, 0);
            return (true, a % b);
        }
    }

    /**
     * @dev Returns the addition of two unsigned integers, reverting on
     * overflow.
     *
     * Counterpart to Solidity's `+` operator.
     *
     * Requirements:
     *
     * - Addition cannot overflow.
     */
    function add(uint256 a, uint256 b) internal pure returns (uint256) {
        return a + b;
    }

    /**
     * @dev Returns the subtraction of two unsigned integers, reverting on
     * overflow (when the result is negative).
     *
     * Counterpart to Solidity's `-` operator.
     *
     * Requirements:
     *
     * - Subtraction cannot overflow.
     */
    function sub(uint256 a, uint256 b) internal pure returns (uint256) {
        return a - b;
    }

    /**
     * @dev Returns the multiplication of two unsigned integers, reverting on
     * overflow.
     *
     * Counterpart to Solidity's `*` operator.
     *
     * Requirements:
     *
     * - Multiplication cannot overflow.
     */
    function mul(uint256 a, uint256 b) internal pure returns (uint256) {
        return a * b;
    }

    /**
     * @dev Returns the integer division of two unsigned integers, reverting on
     * division by zero. The result is rounded towards zero.
     *
     * Counterpart to Solidity's `/` operator.
     *
     * Requirements:
     *
     * - The divisor cannot be zero.
     */
    function div(uint256 a, uint256 b) internal pure returns (uint256) {
        return a / b;
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * reverting when dividing by zero.
     *
     * Counterpart to Solidity's `%` operator. This function uses a `revert`
     * opcode (which leaves remaining gas untouched) while Solidity uses an
     * invalid opcode to revert (consuming all remaining gas).
     *
     * Requirements:
     *
     * - The divisor cannot be zero.
     */
    function mod(uint256 a, uint256 b) internal pure returns (uint256) {
        return a % b;
    }

    /**
     * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
     * overflow (when the result is negative).
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {trySub}.
     *
     * Counterpart to Solidity's `-` operator.
     *
     * Requirements:
     *
     * - Subtraction cannot overflow.
     */
    function sub(
        uint256 a,
        uint256 b,
        string memory errorMessage
    ) internal pure returns (uint256) {
        unchecked {
            require(b <= a, errorMessage);
            return a - b;
        }
    }

    /**
     * @dev Returns the integer division of two unsigned integers, reverting with custom message on
     * division by zero. The result is rounded towards zero.
     *
     * Counterpart to Solidity's `/` operator. Note: this function uses a
     * `revert` opcode (which leaves remaining gas untouched) while Solidity
     * uses an invalid opcode to revert (consuming all remaining gas).
     *
     * Requirements:
     *
     * - The divisor cannot be zero.
     */
    function div(
        uint256 a,
        uint256 b,
        string memory errorMessage
    ) internal pure returns (uint256) {
        unchecked {
            require(b > 0, errorMessage);
            return a / b;
        }
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * reverting with custom message when dividing by zero.
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {tryMod}.
     *
     * Counterpart to Solidity's `%` operator. This function uses a `revert`
     * opcode (which leaves remaining gas untouched) while Solidity uses an
     * invalid opcode to revert (consuming all remaining gas).
     *
     * Requirements:
     *
     * - The divisor cannot be zero.
     */
    function mod(
        uint256 a,
        uint256 b,
        string memory errorMessage
    ) internal pure returns (uint256) {
        unchecked {
            require(b > 0, errorMessage);
            return a % b;
        }
    }
}

// File: @openzeppelin/contracts/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);
    }
}

// File: @openzeppelin/contracts/utils/Context.sol



pragma solidity ^0.8.0;

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

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

// File: @openzeppelin/contracts/utils/Address.sol



pragma solidity ^0.8.0;

/**
 * @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
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize, which returns 0 for contracts in
        // construction, since the code is only stored at the end of the
        // constructor execution.

        uint256 size;
        assembly {
            size := extcodesize(account)
        }
        return size > 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: @openzeppelin/contracts/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: @openzeppelin/contracts/utils/introspection/IERC165.sol



pragma solidity ^0.8.0;

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

// File: @openzeppelin/contracts/utils/introspection/ERC165.sol



pragma solidity ^0.8.0;


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

// File: @openzeppelin/contracts/token/ERC721/IERC721.sol



pragma solidity ^0.8.0;


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

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

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

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

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

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`, 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: @openzeppelin/contracts/token/ERC721/extensions/IERC721Enumerable.sol



pragma solidity ^0.8.0;


/**
 * @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: @openzeppelin/contracts/token/ERC721/extensions/IERC721Metadata.sol



pragma solidity ^0.8.0;


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

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

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

// File: @openzeppelin/contracts/token/ERC721/ERC721.sol



pragma solidity ^0.8.0;








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

        _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 {
        require(operator != _msgSender(), "ERC721: approve to caller");

        _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 {
        //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);
    }

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

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

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

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

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

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

        _beforeTokenTransfer(from, to, tokenId);

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

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

        emit Transfer(from, to, tokenId);
    }

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

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
     * The call is not executed if the target address is not a contract.
     *
     * @param from address representing the previous owner of the given token ID
     * @param to target address that will receive the tokens
     * @param tokenId uint256 ID of the token to be transferred
     * @param _data bytes optional data to send along with the call
     * @return bool whether the call correctly returned the expected magic value
     */
    function _checkOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        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 {}
}

// File: @openzeppelin/contracts/token/ERC721/extensions/ERC721Enumerable.sol



pragma solidity ^0.8.0;



/**
 * @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: contracts/NfToken.sol

// contracts/NfToken.sol

pragma solidity ^0.8.0;





contract NfToken is ERC721Enumerable {
    using SafeMath for uint256;
    using Library for uint8;

    struct Trait {
        string traitName;
        string traitType;
        string pixels;
        uint256 pixelCount;
    }


    //Mappings
    mapping(uint256 => Trait[]) public traitTypes;
    mapping(string => bool) hashToMinted;
    mapping(uint256 => string) internal tokenIdToHash;

    //uint256s
    uint256 MAX_SUPPLY = 10000;
    uint256 MINTS_PER_TIER = 2000;
    uint256 SEED_NONCE = 0;
    uint256 MINT_START = 1633964400;
    uint256 MINT_DELAY = 43200;
    uint256 START_PRICE = 30000000000000000;
    uint256 PRICE_DIFF = 5000000000000000;

    //string arrays
    string[] LETTERS = [
        "a",
        "b",
        "c",
        "d",
        "e",
        "f",
        "g",
        "h",
        "i",
        "j",
        "k",
        "l",
        "m",
        "n",
        "o",
        "p",
        "q",
        "r",
        "s",
        "t",
        "u",
        "v",
        "w",
        "x",
        "y",
        "z"
    ];

    //uint arrays
    uint16[][8] TIERS;

    //address
    address tokenAddress;
    address _owner;

    constructor() ERC721("Moose", "MOOSE") {
        _owner = msg.sender;

        //Declare all the rarity tiers

        //Hat
        TIERS[0] = [4700, 3000, 1000, 1000, 200, 50, 50];
        //Antlers
        TIERS[1] = [250, 500, 2500, 2250, 2250, 2250];
        //Neck
        TIERS[2] = [4000, 2500, 1500, 1500, 500];
        //Eyes
        TIERS[3] = [50, 100, 150, 200, 500, 500, 1000, 1500, 3000, 3000];
        //Earrings
        TIERS[4] = [7000, 2000, 500, 200, 200, 50, 50];
        //Mouth
        TIERS[5] = [1428, 1428, 1428, 1429, 1429, 1429, 1429];
        //Character
        TIERS[6] = [1950, 1950, 1950, 1950, 1950, 200, 50];
        //Accessory
        TIERS[7] = [9700, 200, 50, 50];
    }

    /*
  __  __ _     _   _             ___             _   _             
 |  \/  (_)_ _| |_(_)_ _  __ _  | __|  _ _ _  __| |_(_)___ _ _  ___
 | |\/| | | ' \  _| | ' \/ _` | | _| || | ' \/ _|  _| / _ \ ' \(_-<
 |_|  |_|_|_||_\__|_|_||_\__, | |_| \_,_|_||_\__|\__|_\___/_||_/__/
                         |___/                                     
   */

    /**
     * @dev Converts a digit from 0 - 10000 into its corresponding rarity based on the given rarity tier.
     * @param _randinput The input from 0 - 10000 to use for rarity gen.
     * @param _rarityTier The tier to use.
     */
    function rarityGen(uint256 _randinput, uint8 _rarityTier)
        internal
        view
        returns (string memory)
    {
        uint16 currentLowerBound = 0;
        for (uint8 i = 0; i < TIERS[_rarityTier].length; i++) {
            uint16 thisPercentage = TIERS[_rarityTier][i];
            if (
                _randinput >= currentLowerBound &&
                _randinput < currentLowerBound + thisPercentage
            ) return i.toString();
            currentLowerBound = currentLowerBound + thisPercentage;
        }

        revert();
    }

    /**
     * @dev Generates a 9 digit hash from a tokenId, address, and random number.
     * @param _t The token id to be used within the hash.
     * @param _a The address to be used within the hash.
     * @param _c The custom nonce to be used within the hash.
     */
    function hash(
        uint256 _t,
        address _a,
        uint256 _c
    ) internal returns (string memory) {
        require(_c < 10);

        // This will generate a 9 character string.
        //The last 8 digits are random, the first is 0, due to the mouse not being burned.
        string memory currentHash = "0";

        for (uint8 i = 0; i < 8; i++) {
            SEED_NONCE++;
            uint16 _randinput = uint16(
                uint256(
                    keccak256(
                        abi.encodePacked(
                            block.timestamp,
                            block.difficulty,
                            _t,
                            _a,
                            _c,
                            SEED_NONCE
                        )
                    )
                ) % 10000
            );

            currentHash = string(
                abi.encodePacked(currentHash, rarityGen(_randinput, i))
            );
        }

        if (hashToMinted[currentHash]) return hash(_t, _a, _c + 1);

        return currentHash;
    }

    /**
     * @dev Returns the current cost of minting.
     */
    function currentTokenCost() public view returns (uint256) {
        uint256 _totalSupply = totalSupply();

        if (_totalSupply <= 2000) return 1000000000000000000;
        if (_totalSupply > 2000 && _totalSupply <= 4000)
            return 1000000000000000000;
        if (_totalSupply > 4000 && _totalSupply <= 6000)
            return 2000000000000000000;
        if (_totalSupply > 6000 && _totalSupply <= 8000)
            return 3000000000000000000;
        if (_totalSupply > 8000 && _totalSupply <= 10000)
            return 4000000000000000000;

        revert();
    }

    /**
     * @dev Returns the current ETH price to mint
     */
    function currentPrice() public view returns (uint256) {
        if(block.timestamp < MINT_START) {
            return START_PRICE;
        }

        uint256 _mintTiersComplete = (block.timestamp - MINT_START).div(MINT_DELAY);
        if(PRICE_DIFF * _mintTiersComplete >= START_PRICE) {
            return 0;
        } else {
            return START_PRICE - (PRICE_DIFF * _mintTiersComplete);
        }
    }

    /**
     * @dev Mint internal, this is to avoid code duplication.
     */
    function mintInternal() internal {
        uint256 _totalSupply = totalSupply();
        require(_totalSupply < MAX_SUPPLY);
        require(!Library.isContract(msg.sender));

        uint256 thisTokenId = _totalSupply;

        tokenIdToHash[thisTokenId] = hash(thisTokenId, msg.sender, 0);

        hashToMinted[tokenIdToHash[thisTokenId]] = true;

        _mint(msg.sender, thisTokenId);
    }

    /**
     * @dev Mints new tokens.
     */
    function mintNfToken(uint256 _times) public payable {
        uint256 _totalSupply = totalSupply();
        uint256 _price = currentPrice();
        require((_times > 0 && _times <= 5));
        require(msg.value >= _times * _price);
        require(_totalSupply < MINTS_PER_TIER);
        require(block.timestamp >= MINT_START);
        
        for(uint256 i=0; i< _times; i++){
            mintInternal();
        }
    }
    /**
     * @dev Mint for token
     */
    function mintWithToken() public {        
        //Burn this much token
        IToken(tokenAddress).burnFrom(msg.sender, currentTokenCost());
        return mintInternal();
    }

    /**
     * @dev Burns and mints new.
     * @param _tokenId The token to burn.
     */
    function burnForMint(uint256 _tokenId) public {
        require(ownerOf(_tokenId) == msg.sender);

        //Burn token
        _transfer(
            msg.sender,
            0x000000000000000000000000000000000000dEaD,
            _tokenId
        );

        mintInternal();
    }

    /*
 ____     ___   ____  ___        _____  __ __  ____     __ ______  ____  ___   ____   _____
|    \   /  _] /    ||   \      |     ||  |  ||    \   /  ]      ||    |/   \ |    \ / ___/
|  D  ) /  [_ |  o  ||    \     |   __||  |  ||  _  | /  /|      | |  ||     ||  _  (   \_ 
|    / |    _]|     ||  D  |    |  |_  |  |  ||  |  |/  / |_|  |_| |  ||  O  ||  |  |\__  |
|    \ |   [_ |  _  ||     |    |   _] |  :  ||  |  /   \_  |  |   |  ||     ||  |  |/  \ |
|  .  \|     ||  |  ||     |    |  |   |     ||  |  \     | |  |   |  ||     ||  |  |\    |
|__|\_||_____||__|__||_____|    |__|    \__,_||__|__|\____| |__|  |____|\___/ |__|__| \___|
                                                                                           
*/

    /**
     * @dev Helper function to reduce pixel size within contract
     */
    function letterToNumber(string memory _inputLetter)
        internal
        view
        returns (uint8)
    {
        for (uint8 i = 0; i < LETTERS.length; i++) {
            if (
                keccak256(abi.encodePacked((LETTERS[i]))) ==
                keccak256(abi.encodePacked((_inputLetter)))
            ) return (i);
        }
        revert();
    }

    /**
     * @dev Hash to SVG function
     */
    function hashToSVG(string memory _hash)
        public
        view
        returns (string memory)
    {
        string memory svgString;
        bool[24][24] memory placedPixels;

        for (uint8 i = 0; i < 9; i++) {
            uint8 thisTraitIndex = Library.parseInt(
                Library.substring(_hash, i, i + 1)
            );

            for (
                uint16 j = 0;
                j < traitTypes[i][thisTraitIndex].pixelCount;
                j++
            ) {
                string memory thisPixel = Library.substring(
                    traitTypes[i][thisTraitIndex].pixels,
                    j * 4,
                    j * 4 + 4
                );

                uint8 x = letterToNumber(
                    Library.substring(thisPixel, 0, 1)
                );
                uint8 y = letterToNumber(
                    Library.substring(thisPixel, 1, 2)
                );

                if (placedPixels[x][y]) continue;

                svgString = string(
                    abi.encodePacked(
                        svgString,
                        "<rect class='c",
                        Library.substring(thisPixel, 2, 4),
                        "' x='",
                        x.toString(),
                        "' y='",
                        y.toString(),
                        "'/>"
                    )
                );

                placedPixels[x][y] = true;
            }
        }

        svgString = string(
            abi.encodePacked(
                '<svg id="mouse-svg" xmlns="http://www.w3.org/2000/svg" preserveAspectRatio="xMinYMin meet" viewBox="0 0 24 24"> <rect class="bg" x="0" y="0" />',
                svgString,
                "<style>rect.bg{width:24px;height:24px;fill:#B6EAFF} rect{width:1px;height:1px;} #mouse-svg{shape-rendering: crispedges;} .c00{fill:#000000}.c01{fill:#B1ADAC}.c02{fill:#D7D7D7}.c03{fill:#FFA6A6}.c04{fill:#FFD4D5}.c05{fill:#B9AD95}.c06{fill:#E2D6BE}.c07{fill:#7F625A}.c08{fill:#A58F82}.c09{fill:#4B1E0B}.c10{fill:#6D2C10}.c11{fill:#D8D8D8}.c12{fill:#F5F5F5}.c13{fill:#433D4B}.c14{fill:#8D949C}.c15{fill:#05FF00}.c16{fill:#01C700}.c17{fill:#0B8F08}.c18{fill:#421C13}.c19{fill:#6B392A}.c20{fill:#A35E40}.c21{fill:#DCBD91}.c22{fill:#777777}.c23{fill:#848484}.c24{fill:#ABABAB}.c25{fill:#BABABA}.c26{fill:#C7C7C7}.c27{fill:#EAEAEA}.c28{fill:#0C76AA}.c29{fill:#0E97DB}.c30{fill:#10A4EC}.c31{fill:#13B0FF}.c32{fill:#2EB9FE}.c33{fill:#54CCFF}.c34{fill:#50C0F2}.c35{fill:#54CCFF}.c36{fill:#72DAFF}.c37{fill:#B6EAFF}.c38{fill:#FFFFFF}.c39{fill:#954546}.c40{fill:#0B87F7}.c41{fill:#FF2626}.c42{fill:#180F02}.c43{fill:#2B2319}.c44{fill:#FBDD4B}.c45{fill:#F5B923}.c46{fill:#CC8A18}.c47{fill:#3C2203}.c48{fill:#53320B}.c49{fill:#7B501D}.c50{fill:#FFE646}.c51{fill:#FFD627}.c52{fill:#F5B700}.c53{fill:#242424}.c54{fill:#4A4A4A}.c55{fill:#676767}.c56{fill:#F08306}.c57{fill:#FCA30E}.c58{fill:#FEBC0E}.c59{fill:#FBEC1C}.c60{fill:#14242F}.c61{fill:#B06837}.c62{fill:#8F4B0E}.c63{fill:#D88227}.c64{fill:#B06837}</style></svg>"
            )
        );

        return svgString;
    }

    /**
     * @dev Hash to metadata function
     */
    function hashToMetadata(string memory _hash)
        public
        view
        returns (string memory)
    {
        string memory metadataString;

        for (uint8 i = 0; i < 9; i++) {
            uint8 thisTraitIndex = Library.parseInt(
                Library.substring(_hash, i, i + 1)
            );

            metadataString = string(
                abi.encodePacked(
                    metadataString,
                    '{"trait_type":"',
                    traitTypes[i][thisTraitIndex].traitType,
                    '","value":"',
                    traitTypes[i][thisTraitIndex].traitName,
                    '"}'
                )
            );

            if (i != 8)
                metadataString = string(abi.encodePacked(metadataString, ","));
        }

        return string(abi.encodePacked("[", metadataString, "]"));
    }

    /**
     * @dev Returns the SVG and metadata for a token Id
     * @param _tokenId The tokenId to return the SVG and metadata for.
     */
    function tokenURI(uint256 _tokenId)
        public
        view
        override
        returns (string memory)
    {
        require(_exists(_tokenId));

        string memory tokenHash = _tokenIdToHash(_tokenId);

        return
            string(
                abi.encodePacked(
                    "data:application/json;base64,",
                    Library.encode(
                        bytes(
                            string(
                                abi.encodePacked(
                                    '{"name": "NfToken #',
                                    Library.toString(_tokenId),
                                    '", "description": "This is a collection of 10,000 unique images. All the metadata and images are generated and stored 100% on-chain. No IPFS, no API.", "image": "data:image/svg+xml;base64,',
                                    Library.encode(
                                        bytes(hashToSVG(tokenHash))
                                    ),
                                    '","attributes":',
                                    hashToMetadata(tokenHash),
                                    "}"
                                )
                            )
                        )
                    )
                )
            );
    }

    /**
     * @dev Returns a hash for a given tokenId
     * @param _tokenId The tokenId to return the hash for.
     */
    function _tokenIdToHash(uint256 _tokenId)
        public
        view
        returns (string memory)
    {
        string memory tokenHash = tokenIdToHash[_tokenId];
        //If this is a burned token, override the previous hash
        if (ownerOf(_tokenId) == 0x000000000000000000000000000000000000dEaD) {
            tokenHash = string(
                abi.encodePacked(
                    "1",
                    Library.substring(tokenHash, 1, 9)
                )
            );
        }

        return tokenHash;
    }

    /**
     * @dev Returns the wallet of a given wallet. Mainly for ease for frontend devs.
     * @param _wallet The wallet to get the tokens of.
     */
    function walletOfOwner(address _wallet)
        public
        view
        returns (uint256[] memory)
    {
        uint256 tokenCount = balanceOf(_wallet);

        uint256[] memory tokensId = new uint256[](tokenCount);
        for (uint256 i; i < tokenCount; i++) {
            tokensId[i] = tokenOfOwnerByIndex(_wallet, i);
        }
        return tokensId;
    }

    /*

  ___   __    __  ____     ___  ____       _____  __ __  ____     __ ______  ____  ___   ____   _____
 /   \ |  |__|  ||    \   /  _]|    \     |     ||  |  ||    \   /  ]      ||    |/   \ |    \ / ___/
|     ||  |  |  ||  _  | /  [_ |  D  )    |   __||  |  ||  _  | /  /|      | |  ||     ||  _  (   \_ 
|  O  ||  |  |  ||  |  ||    _]|    /     |  |_  |  |  ||  |  |/  / |_|  |_| |  ||  O  ||  |  |\__  |
|     ||  `  '  ||  |  ||   [_ |    \     |   _] |  :  ||  |  /   \_  |  |   |  ||     ||  |  |/  \ |
|     | \      / |  |  ||     ||  .  \    |  |   |     ||  |  \     | |  |   |  ||     ||  |  |\    |
 \___/   \_/\_/  |__|__||_____||__|\_|    |__|    \__,_||__|__|\____| |__|  |____|\___/ |__|__| \___|
                                                                                                     


    */

    /**
     * @dev Clears the traits.
     */
    function clearTraits() public onlyOwner {
        for (uint256 i = 0; i < 9; i++) {
            delete traitTypes[i];
        }
    }

    /**
     * @dev Add a trait type
     * @param _traitTypeIndex The trait type index
     * @param traits Array of traits to add
     */

    function addTraitType(uint256 _traitTypeIndex, Trait[] memory traits)
        public
        onlyOwner
    {
        for (uint256 i = 0; i < traits.length; i++) {
            traitTypes[_traitTypeIndex].push(
                Trait(
                    traits[i].traitName,
                    traits[i].traitType,
                    traits[i].pixels,
                    traits[i].pixelCount
                )
            );
        }

        return;
    }

    /**
     * @dev Sets the ERC20 token address
     * @param _tokenAddress The token address
     */

    function setTokenAddress(address _tokenAddress) public onlyOwner {
        tokenAddress = _tokenAddress;
    }

    /**
     * @dev Transfers ownership
     * @param _newOwner The new owner
     */
    function transferOwnership(address _newOwner) public onlyOwner {
        _owner = _newOwner;
    }

    /**
     * @dev Withdraw ETH to owner
     */
    function withdraw() public onlyOwner {
        uint256 balance = address(this).balance;

        payable(msg.sender).transfer(balance);
    }   

    /**
     * @dev Modifier to only allow owner to call functions
     */
    modifier onlyOwner() {
        require(_owner == msg.sender);
        _;
    }
}

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"_tokenIdToHash","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_traitTypeIndex","type":"uint256"},{"components":[{"internalType":"string","name":"traitName","type":"string"},{"internalType":"string","name":"traitType","type":"string"},{"internalType":"string","name":"pixels","type":"string"},{"internalType":"uint256","name":"pixelCount","type":"uint256"}],"internalType":"struct NfToken.Trait[]","name":"traits","type":"tuple[]"}],"name":"addTraitType","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"burnForMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"clearTraits","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"currentPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentTokenCost","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"string","name":"_hash","type":"string"}],"name":"hashToMetadata","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"string","name":"_hash","type":"string"}],"name":"hashToSVG","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_times","type":"uint256"}],"name":"mintNfToken","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"mintWithToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_tokenAddress","type":"address"}],"name":"setTokenAddress","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":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"traitTypes","outputs":[{"internalType":"string","name":"traitName","type":"string"},{"internalType":"string","name":"traitType","type":"string"},{"internalType":"string","name":"pixels","type":"string"},{"internalType":"uint256","name":"pixelCount","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":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"walletOfOwner","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]

612710600d556107d0600e556000600f55636164517060105561a8c0601155666a94d74f4300006012556611c37937e0800060135560016103c0818152606160f81b6103e0526080908152610400828152603160f91b6104205260a052610440828152606360f81b6104605260c052610480828152601960fa1b6104a05260e0526104c0828152606560f81b6104e05261010052610500828152603360f91b6105205261012052610540828152606760f81b6105605261014052610580828152600d60fb1b6105a052610160526105c0828152606960f81b6105e05261018052610600828152603560f91b610620526101a052610640828152606b60f81b610660526101c052610680828152601b60fa1b6106a0526101e0526106c0828152606d60f81b6106e05261020052610700828152603760f91b6107205261022052610740828152606f60f81b6107605261024052610780828152600760fc1b6107a052610260526107c0828152607160f81b6107e05261028052610800828152603960f91b610820526102a052610840828152607360f81b610860526102c052610880828152601d60fa1b6108a0526102e0526108c0828152607560f81b6108e05261030052610900828152603b60f91b6109205261032052610940828152607760f81b6109605261034052610980828152600f60fb1b6109a052610360526109c0828152607960f81b6109e05261038052610a40604052610a00918252603d60f91b610a20526103a0919091526200023b90601490601a62000528565b503480156200024957600080fd5b50604051806040016040528060058152602001644d6f6f736560d81b815250604051806040016040528060058152602001644d4f4f534560d81b81525081600090805190602001906200029e9291906200058c565b508051620002b49060019060208401906200058c565b5050601e80546001600160a01b03191633179055506040805160e08101825261125c8152610bb860208201526103e8918101829052606081019190915260c86080820152603260a0820181905260c08201526200031690601590600762000617565b506040805160c08101825260fa81526101f460208201526109c4918101919091526108ca606082018190526080820181905260a08201526200035d90601690600662000617565b506040805160a081018252610fa081526109c460208201526105dc91810182905260608101919091526101f460808201526200039e90601790600562000617565b506040805161014081018252603281526064602082015260969181019190915260c860608201526101f46080820181905260a08201526103e860c08201526105dc60e0820152610bb861010082018190526101208201526200040590601890600a62000617565b506040805160e081018252611b5881526107d060208201526101f49181019190915260c8606082018190526080820152603260a0820181905260c08201526200045390601990600762000617565b506040805160e0810182526105948082526020820181905291810191909152610595606082018190526080820181905260a0820181905260c08201526200049f90601a90600762000617565b506040805160e08101825261079e8082526020820181905291810182905260608101829052608081019190915260c860a0820152603260c0820152620004ea90601b90600762000617565b50604080516080810182526125e4815260c86020820152603291810182905260608101919091526200052190601c90600462000617565b5062000779565b8280548282559060005260206000209081019282156200057a579160200282015b828111156200057a5782518051620005699184916020909101906200058c565b509160200191906001019062000549565b5062000588929150620006c2565b5090565b8280546200059a906200073c565b90600052602060002090601f016020900481019282620005be576000855562000609565b82601f10620005d957805160ff191683800117855562000609565b8280016001018555821562000609579182015b8281111562000609578251825591602001919060010190620005ec565b5062000588929150620006e3565b82805482825590600052602060002090600f01601090048101928215620006095791602002820160005b838211156200068357835183826101000a81548161ffff021916908361ffff160217905550926020019260020160208160010104928301926001030262000641565b8015620006b35782816101000a81549061ffff021916905560020160208160010104928301926001030262000683565b505062000588929150620006e3565b8082111562000588576000620006d98282620006fa565b50600101620006c2565b5b80821115620005885760008155600101620006e4565b50805462000708906200073c565b6000825580601f1062000719575050565b601f016020900490600052602060002090810190620007399190620006e3565b50565b600181811c908216806200075157607f821691505b602082108114156200077357634e487b7160e01b600052602260045260246000fd5b50919050565b613f8880620007896000396000f3fe6080604052600436106101d75760003560e01c8063438b630011610102578063a22cb46511610095578063d93dfe8811610064578063d93dfe881461054b578063e75166de1461056b578063e985e9c51461057e578063f2fde38b146105c757600080fd5b8063a22cb465146104cb578063b88d4fde146104eb578063c4ef71c71461050b578063c87b56dd1461052b57600080fd5b806370a08231116100d157806370a082311461046157806389ce30741461048157806395d89b41146104a15780639d1b464a146104b657600080fd5b8063438b6300146103d45780634f6ccce7146104015780636352211e1461042157806366e338701461044157600080fd5b806318160ddd1161017a5780632f745c59116101495780632f745c591461034f5780632fb098d21461036f5780633ccfd60b1461039f57806342842e0e146103b457600080fd5b806318160ddd146102db5780631da8491a146102fa57806323b872dd1461030f57806326a4e8d21461032f57600080fd5b8063081812fc116101b6578063081812fc14610257578063095ea7b31461028f578063098afd4b146102b1578063171be8d0146102c657600080fd5b80625ea307146101dc57806301ffc9a71461021257806306fdde0314610242575b600080fd5b3480156101e857600080fd5b506101fc6101f7366004612e63565b6105e7565b6040516102099190613b6d565b60405180910390f35b34801561021e57600080fd5b5061023261022d366004612df8565b6106de565b6040519015158152602001610209565b34801561024e57600080fd5b506101fc610703565b34801561026357600080fd5b50610277610272366004612e63565b610795565b6040516001600160a01b039091168152602001610209565b34801561029b57600080fd5b506102af6102aa366004612dcf565b61082f565b005b3480156102bd57600080fd5b506102af610945565b3480156102d257600080fd5b506102af610994565b3480156102e757600080fd5b506008545b604051908152602001610209565b34801561030657600080fd5b506102ec610a16565b34801561031b57600080fd5b506102af61032a366004612ce1565b610ad8565b34801561033b57600080fd5b506102af61034a366004612c95565b610b09565b34801561035b57600080fd5b506102ec61036a366004612dcf565b610b42565b34801561037b57600080fd5b5061038f61038a366004612f49565b610bd8565b6040516102099493929190613b80565b3480156103ab57600080fd5b506102af610dbd565b3480156103c057600080fd5b506102af6103cf366004612ce1565b610e07565b3480156103e057600080fd5b506103f46103ef366004612c95565b610e22565b6040516102099190613b29565b34801561040d57600080fd5b506102ec61041c366004612e63565b610ee0565b34801561042d57600080fd5b5061027761043c366004612e63565b610f81565b34801561044d57600080fd5b506101fc61045c366004612e30565b610ff8565b34801561046d57600080fd5b506102ec61047c366004612c95565b611151565b34801561048d57600080fd5b506101fc61049c366004612e30565b6111d8565b3480156104ad57600080fd5b506101fc6114eb565b3480156104c257600080fd5b506102ec6114fa565b3480156104d757600080fd5b506102af6104e6366004612d95565b61156e565b3480156104f757600080fd5b506102af610506366004612d1c565b611633565b34801561051757600080fd5b506102af610526366004612e63565b61166b565b34801561053757600080fd5b506101fc610546366004612e63565b61169d565b34801561055757600080fd5b506102af610566366004612e7b565b61172a565b6102af610579366004612e63565b6118b1565b34801561058a57600080fd5b50610232610599366004612caf565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b3480156105d357600080fd5b506102af6105e2366004612c95565b61193c565b6000818152600c602052604081208054606092919061060590613e0e565b80601f016020809104026020016040519081016040528092919081815260200182805461063190613e0e565b801561067e5780601f106106535761010080835404028352916020019161067e565b820191906000526020600020905b81548152906001019060200180831161066157829003601f168201915b5050505050905061068e83610f81565b6001600160a01b031661dead6001600160a01b031614156106d8576106b68160016009611975565b6040516020016106c69190613acd565b60405160208183030381529060405290505b92915050565b60006001600160e01b0319821663780e9d6360e01b14806106d857506106d882611a6e565b60606000805461071290613e0e565b80601f016020809104026020016040519081016040528092919081815260200182805461073e90613e0e565b801561078b5780601f106107605761010080835404028352916020019161078b565b820191906000526020600020905b81548152906001019060200180831161076e57829003601f168201915b5050505050905090565b6000818152600260205260408120546001600160a01b03166108135760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084015b60405180910390fd5b506000908152600460205260409020546001600160a01b031690565b600061083a82610f81565b9050806001600160a01b0316836001600160a01b031614156108a85760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b606482015260840161080a565b336001600160a01b03821614806108c457506108c48133610599565b6109365760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c0000000000000000606482015260840161080a565b6109408383611abe565b505050565b601e546001600160a01b0316331461095c57600080fd5b60005b6009811015610991576000818152600a6020526040812061097f916129e8565b8061098981613e6b565b91505061095f565b50565b601d546001600160a01b03166379cc6790336109ae610a16565b6040516001600160e01b031960e085901b1681526001600160a01b0390921660048301526024820152604401600060405180830381600087803b1580156109f457600080fd5b505af1158015610a08573d6000803e3d6000fd5b50505050610a14611b2c565b565b600080610a2260085490565b90506107d08111610a3c57670de0b6b3a764000091505090565b6107d081118015610a4f5750610fa08111155b15610a6357670de0b6b3a764000091505090565b610fa081118015610a7657506117708111155b15610a8a57671bc16d674ec8000091505090565b61177081118015610a9d5750611f408111155b15610ab1576729a2241af62c000091505090565b611f4081118015610ac457506127108111155b156101d757673782dace9d90000091505090565b610ae23382611bd2565b610afe5760405162461bcd60e51b815260040161080a90613c1d565b610940838383611cc9565b601e546001600160a01b03163314610b2057600080fd5b601d80546001600160a01b0319166001600160a01b0392909216919091179055565b6000610b4d83611151565b8210610baf5760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b606482015260840161080a565b506001600160a01b03919091166000908152600660209081526040808320938352929052205490565b600a6020528160005260406000208181548110610bf457600080fd5b906000526020600020906004020160009150915050806000018054610c1890613e0e565b80601f0160208091040260200160405190810160405280929190818152602001828054610c4490613e0e565b8015610c915780601f10610c6657610100808354040283529160200191610c91565b820191906000526020600020905b815481529060010190602001808311610c7457829003601f168201915b505050505090806001018054610ca690613e0e565b80601f0160208091040260200160405190810160405280929190818152602001828054610cd290613e0e565b8015610d1f5780601f10610cf457610100808354040283529160200191610d1f565b820191906000526020600020905b815481529060010190602001808311610d0257829003601f168201915b505050505090806002018054610d3490613e0e565b80601f0160208091040260200160405190810160405280929190818152602001828054610d6090613e0e565b8015610dad5780601f10610d8257610100808354040283529160200191610dad565b820191906000526020600020905b815481529060010190602001808311610d9057829003601f168201915b5050505050908060030154905084565b601e546001600160a01b03163314610dd457600080fd5b6040514790339082156108fc029083906000818181858888f19350505050158015610e03573d6000803e3d6000fd5b5050565b61094083838360405180602001604052806000815250611633565b60606000610e2f83611151565b905060008167ffffffffffffffff811115610e5a57634e487b7160e01b600052604160045260246000fd5b604051908082528060200260200182016040528015610e83578160200160208202803683370190505b50905060005b82811015610ed857610e9b8582610b42565b828281518110610ebb57634e487b7160e01b600052603260045260246000fd5b602090810291909101015280610ed081613e6b565b915050610e89565b509392505050565b6000610eeb60085490565b8210610f4e5760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b606482015260840161080a565b60088281548110610f6f57634e487b7160e01b600052603260045260246000fd5b90600052602060002001549050919050565b6000818152600260205260408120546001600160a01b0316806106d85760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201526832b73a103a37b5b2b760b91b606482015260840161080a565b60608060005b60098160ff16101561112857600061102f61102a8660ff8516611022866001613cfd565b60ff16611975565b611e74565b905082600a60008460ff1681526020019081526020016000208260ff168154811061106a57634e487b7160e01b600052603260045260246000fd5b9060005260206000209060040201600101600a60008560ff1681526020019081526020016000208360ff16815481106110b357634e487b7160e01b600052603260045260246000fd5b90600052602060002090600402016000016040516020016110d693929190613161565b60405160208183030381529060405292508160ff1660081461111557826040516020016111039190613095565b60405160208183030381529060405292505b508061112081613e86565b915050610ffe565b508060405160200161113a91906138e6565b604051602081830303815290604052915050919050565b60006001600160a01b0382166111bc5760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604482015269726f206164647265737360b01b606482015260840161080a565b506001600160a01b031660009081526003602052604090205490565b6060806111e3612a09565b60005b60098160ff1610156114c157600061120a61102a8760ff8516611022866001613cfd565b905060005b600a60008460ff1681526020019081526020016000208260ff168154811061124757634e487b7160e01b600052603260045260246000fd5b9060005260206000209060040201600301548161ffff1610156114ac57600061136b600a60008660ff1681526020019081526020016000208460ff16815481106112a157634e487b7160e01b600052603260045260246000fd5b906000526020600020906004020160020180546112bd90613e0e565b80601f01602080910402602001604051908101604052809291908181526020018280546112e990613e0e565b80156113365780601f1061130b57610100808354040283529160200191611336565b820191906000526020600020905b81548152906001019060200180831161131957829003601f168201915b50505050508360046113489190613d36565b61ffff16611357856004613d36565b611362906004613cc8565b61ffff16611975565b9050600061138461137f8360006001611975565b611f54565b9050600061139861137f8460016002611975565b9050868260ff16601881106113bd57634e487b7160e01b600052603260045260246000fd5b60200201518160ff16601881106113e457634e487b7160e01b600052603260045260246000fd5b6020020151156113f65750505061149a565b876114048460026004611975565b6114108460ff16612004565b61141c8460ff16612004565b60405160200161142f94939291906130ba565b60405160208183030381529060405297506001878360ff166018811061146557634e487b7160e01b600052603260045260246000fd5b60200201518260ff166018811061148c57634e487b7160e01b600052603260045260246000fd5b911515602090920201525050505b806114a481613e49565b91505061120f565b505080806114b990613e86565b9150506111e6565b50816040516020016114d391906131da565b60408051601f19818403018152919052949350505050565b60606001805461071290613e0e565b600060105442101561150d575060125490565b6000611529601154601054426115239190613da8565b9061211e565b90506012548160135461153c9190613d60565b1061154957600091505090565b806013546115579190613d60565b6012546115649190613da8565b91505090565b5090565b6001600160a01b0382163314156115c75760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c657200000000000000604482015260640161080a565b3360008181526005602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b61163d3383611bd2565b6116595760405162461bcd60e51b815260040161080a90613c1d565b6116658484848461212a565b50505050565b3361167582610f81565b6001600160a01b03161461168857600080fd5b6116953361dead83611cc9565b610991611b2c565b6000818152600260205260409020546060906001600160a01b03166116c157600080fd5b60006116cc836105e7565b905061171a6116da84612004565b6116eb6116e6846111d8565b61215d565b6116f484610ff8565b6040516020016117069392919061395f565b60405160208183030381529060405261215d565b60405160200161113a919061391a565b601e546001600160a01b0316331461174157600080fd5b60005b815181101561094057600a6000848152602001908152602001600020604051806080016040528084848151811061178b57634e487b7160e01b600052603260045260246000fd5b60200260200101516000015181526020018484815181106117bc57634e487b7160e01b600052603260045260246000fd5b60200260200101516020015181526020018484815181106117ed57634e487b7160e01b600052603260045260246000fd5b602002602001015160400151815260200184848151811061181e57634e487b7160e01b600052603260045260246000fd5b6020908102919091018101516060015190915282546001810184556000938452928190208251805193946004029091019261185c9284920190612a37565b5060208281015180516118759260018501920190612a37565b5060408201518051611891916002840191602090910190612a37565b5060608201518160030155505080806118a990613e6b565b915050611744565b60006118bc60085490565b905060006118c86114fa565b90506000831180156118db575060058311155b6118e457600080fd5b6118ee8184613d60565b3410156118fa57600080fd5b600e54821061190857600080fd5b60105442101561191757600080fd5b60005b838110156116655761192a611b2c565b8061193481613e6b565b91505061191a565b601e546001600160a01b0316331461195357600080fd5b601e80546001600160a01b0319166001600160a01b0392909216919091179055565b60608360006119848585613da8565b67ffffffffffffffff8111156119aa57634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f1916602001820160405280156119d4576020820181803683370190505b509050845b84811015611a6257828181518110611a0157634e487b7160e01b600052603260045260246000fd5b01602001516001600160f81b03191682611a1b8884613da8565b81518110611a3957634e487b7160e01b600052603260045260246000fd5b60200101906001600160f81b031916908160001a90535080611a5a81613e6b565b9150506119d9565b509150505b9392505050565b60006001600160e01b031982166380ac58cd60e01b1480611a9f57506001600160e01b03198216635b5e139f60e01b145b806106d857506301ffc9a760e01b6001600160e01b03198316146106d8565b600081815260046020526040902080546001600160a01b0319166001600160a01b0384169081179091558190611af382610f81565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b6000611b3760085490565b9050600d548110611b4757600080fd5b333b15611b5357600080fd5b80611b60813360006122d3565b6000828152600c602090815260409091208251611b839391929190910190612a37565b506001600b600c6000848152602001908152602001600020604051611ba891906131ce565b908152604051908190036020019020805491151560ff19909216919091179055610e03338261241b565b6000818152600260205260408120546001600160a01b0316611c4b5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b606482015260840161080a565b6000611c5683610f81565b9050806001600160a01b0316846001600160a01b03161480611c915750836001600160a01b0316611c8684610795565b6001600160a01b0316145b80611cc157506001600160a01b0380821660009081526005602090815260408083209388168352929052205460ff165b949350505050565b826001600160a01b0316611cdc82610f81565b6001600160a01b031614611d445760405162461bcd60e51b815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201526839903737ba1037bbb760b91b606482015260840161080a565b6001600160a01b038216611da65760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b606482015260840161080a565b611db1838383612569565b611dbc600082611abe565b6001600160a01b0383166000908152600360205260408120805460019290611de5908490613da8565b90915550506001600160a01b0382166000908152600360205260408120805460019290611e13908490613ce5565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b60008181805b82518160ff161015610ed8576030838260ff1681518110611eab57634e487b7160e01b600052603260045260246000fd5b016020015160f81c10801590611eec57506039838260ff1681518110611ee157634e487b7160e01b600052603260045260246000fd5b016020015160f81c11155b15611f4257611efc600a83613d7f565b91506030838260ff1681518110611f2357634e487b7160e01b600052603260045260246000fd5b0160200151611f35919060f81c613dbf565b611f3f9083613cfd565b91505b80611f4c81613e86565b915050611e7a565b6000805b60145460ff82161015611ffe5782604051602001611f76919061304a565b6040516020818303038152906040528051906020012060148260ff1681548110611fb057634e487b7160e01b600052603260045260246000fd5b90600052602060002001604051602001611fca91906131ce565b604051602081830303815290604052805190602001201415611fec5792915050565b80611ff681613e86565b915050611f58565b50600080fd5b6060816120285750506040805180820190915260018152600360fc1b602082015290565b8160005b8115612052578061203c81613e6b565b915061204b9050600a83613d22565b915061202c565b60008167ffffffffffffffff81111561207b57634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f1916602001820160405280156120a5576020820181803683370190505b5090505b8415611cc1576120ba600183613da8565b91506120c7600a86613ea6565b6120d2906030613ce5565b60f81b8183815181106120f557634e487b7160e01b600052603260045260246000fd5b60200101906001600160f81b031916908160001a905350612117600a86613d22565b94506120a9565b6000611a678284613d22565b612135848484611cc9565b61214184848484612621565b6116655760405162461bcd60e51b815260040161080a90613bcb565b606081516000141561217d57505060408051602081019091526000815290565b6000604051806060016040528060408152602001613f1360409139905060006003845160026121ac9190613ce5565b6121b69190613d22565b6121c1906004613d60565b905060006121d0826020613ce5565b67ffffffffffffffff8111156121f657634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f191660200182016040528015612220576020820181803683370190505b509050818152600183018586518101602084015b8183101561228e5760039283018051603f601282901c811687015160f890811b8552600c83901c8216880151811b6001860152600683901c8216880151811b60028601529116860151901b93820193909352600401612234565b6003895106600181146122a857600281146122b9576122c5565b613d3d60f01b6001198301526122c5565b603d60f81b6000198301525b509398975050505050505050565b6060600a82106122e257600080fd5b6040805180820190915260018152600360fc1b602082015260005b60088160ff1610156123d557600f805490600061231983613e6b565b9091555050600f54604080514260208201524491810191909152606080820189905287901b6bffffffffffffffffffffffff191660808201526094810186905260b48101919091526000906127109060d4016040516020818303038152906040528051906020012060001c61238e9190613ea6565b90508261239f8261ffff168461272e565b6040516020016123b0929190613066565b60405160208183030381529060405292505080806123cd90613e86565b9150506122fd565b50600b816040516123e6919061304a565b9081526040519081900360200190205460ff1615611cc157612413858561240e866001613ce5565b6122d3565b915050611a67565b6001600160a01b0382166124715760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f2061646472657373604482015260640161080a565b6000818152600260205260409020546001600160a01b0316156124d65760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e74656400000000604482015260640161080a565b6124e260008383612569565b6001600160a01b038216600090815260036020526040812080546001929061250b908490613ce5565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6001600160a01b0383166125c4576125bf81600880546000838152600960205260408120829055600182018355919091527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee30155565b6125e7565b816001600160a01b0316836001600160a01b0316146125e7576125e7838261282e565b6001600160a01b0382166125fe57610940816128cb565b826001600160a01b0316826001600160a01b0316146109405761094082826129a4565b60006001600160a01b0384163b1561272357604051630a85bd0160e11b81526001600160a01b0385169063150b7a0290612665903390899088908890600401613af6565b602060405180830381600087803b15801561267f57600080fd5b505af19250505080156126af575060408051601f3d908101601f191682019092526126ac91810190612e14565b60015b612709573d8080156126dd576040519150601f19603f3d011682016040523d82523d6000602084013e6126e2565b606091505b5080516127015760405162461bcd60e51b815260040161080a90613bcb565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611cc1565b506001949350505050565b60606000805b60158460ff166008811061275857634e487b7160e01b600052603260045260246000fd5b015460ff82161015611ffe57600060158560ff166008811061278a57634e487b7160e01b600052603260045260246000fd5b018260ff16815481106127ad57634e487b7160e01b600052603260045260246000fd5b90600052602060002090601091828204019190066002029054906101000a900461ffff1690508261ffff1686101580156127f357506127ec8184613cc8565b61ffff1686105b1561280e576128048260ff16612004565b93505050506106d8565b6128188184613cc8565b925050808061282690613e86565b915050612734565b6000600161283b84611151565b6128459190613da8565b600083815260076020526040902054909150808214612898576001600160a01b03841660009081526006602090815260408083208584528252808320548484528184208190558352600790915290208190555b5060009182526007602090815260408084208490556001600160a01b039094168352600681528383209183525290812055565b6008546000906128dd90600190613da8565b6000838152600960205260408120546008805493945090928490811061291357634e487b7160e01b600052603260045260246000fd5b90600052602060002001549050806008838154811061294257634e487b7160e01b600052603260045260246000fd5b600091825260208083209091019290925582815260099091526040808220849055858252812055600880548061298857634e487b7160e01b600052603160045260246000fd5b6001900381819060005260206000200160009055905550505050565b60006129af83611151565b6001600160a01b039093166000908152600660209081526040808320868452825280832085905593825260079052919091209190915550565b50805460008255600402906000526020600020908101906109919190612ab7565b6040518061030001604052806018905b612a21612af7565b815260200190600190039081612a195790505090565b828054612a4390613e0e565b90600052602060002090601f016020900481019282612a655760008555612aab565b82601f10612a7e57805160ff1916838001178555612aab565b82800160010185558215612aab579182015b82811115612aab578251825591602001919060010190612a90565b5061156a929150612b16565b8082111561156a576000612acb8282612b2b565b612ad9600183016000612b2b565b612ae7600283016000612b2b565b5060006003820155600401612ab7565b6040518061030001604052806018906020820280368337509192915050565b5b8082111561156a5760008155600101612b17565b508054612b3790613e0e565b6000825580601f10612b47575050565b601f0160209004906000526020600020908101906109919190612b16565b600067ffffffffffffffff831115612b7f57612b7f613ee6565b612b92601f8401601f1916602001613c97565b9050828152838383011115612ba657600080fd5b828260208301376000602084830101529392505050565b80356001600160a01b0381168114612bd457600080fd5b919050565b600082601f830112612be9578081fd5b611a6783833560208501612b65565b600060808284031215612c09578081fd5b612c11613c6e565b9050813567ffffffffffffffff80821115612c2b57600080fd5b612c3785838601612bd9565b83526020840135915080821115612c4d57600080fd5b612c5985838601612bd9565b60208401526040840135915080821115612c7257600080fd5b50612c7f84828501612bd9565b6040830152506060820135606082015292915050565b600060208284031215612ca6578081fd5b611a6782612bbd565b60008060408385031215612cc1578081fd5b612cca83612bbd565b9150612cd860208401612bbd565b90509250929050565b600080600060608486031215612cf5578081fd5b612cfe84612bbd565b9250612d0c60208501612bbd565b9150604084013590509250925092565b60008060008060808587031215612d31578081fd5b612d3a85612bbd565b9350612d4860208601612bbd565b925060408501359150606085013567ffffffffffffffff811115612d6a578182fd5b8501601f81018713612d7a578182fd5b612d8987823560208401612b65565b91505092959194509250565b60008060408385031215612da7578182fd5b612db083612bbd565b915060208301358015158114612dc4578182fd5b809150509250929050565b60008060408385031215612de1578182fd5b612dea83612bbd565b946020939093013593505050565b600060208284031215612e09578081fd5b8135611a6781613efc565b600060208284031215612e25578081fd5b8151611a6781613efc565b600060208284031215612e41578081fd5b813567ffffffffffffffff811115612e57578182fd5b611cc184828501612bd9565b600060208284031215612e74578081fd5b5035919050565b60008060408385031215612e8d578182fd5b8235915060208084013567ffffffffffffffff80821115612eac578384fd5b818601915086601f830112612ebf578384fd5b813581811115612ed157612ed1613ee6565b8060051b612ee0858201613c97565b8281528581019085870183870188018c1015612efa578889fd5b8893505b84841015612f3757803586811115612f1457898afd5b612f228d8a838b0101612bf8565b84525060019390930192918701918701612efe565b50809750505050505050509250929050565b60008060408385031215612f5b578182fd5b50508035926020909101359150565b60008151808452612f82816020860160208601613de2565b601f01601f19169290920160200192915050565b60008151612fa8818560208601613de2565b9290920192915050565b8054600090600181811c9080831680612fcc57607f831692505b6020808410821415612fec57634e487b7160e01b86526022600452602486fd5b81801561300057600181146130115761303e565b60ff1986168952848901965061303e565b60008881526020902060005b868110156130365781548b82015290850190830161301d565b505084890196505b50505050505092915050565b6000825161305c818460208701613de2565b9190910192915050565b60008351613078818460208801613de2565b83519083019061308c818360208801613de2565b01949350505050565b600082516130a7818460208701613de2565b600b60fa1b920191825250600101919050565b600085516130cc818460208a01613de2565b6d3c7265637420636c6173733d276360901b90830190815285516130f781600e840160208a01613de2565b642720783d2760d81b600e9290910191820152845161311d816013840160208901613de2565b642720793d2760d81b601392909101918201528351613143816018840160208801613de2565b6213979f60e91b60189290910191820152601b019695505050505050565b60008451613173818460208901613de2565b6e3d913a3930b4ba2fba3cb832911d1160891b908301908152613199600f820186612fb2565b6a1116113b30b63ab2911d1160a91b815290506131b9600b820185612fb2565b61227d60f01b81526002019695505050505050565b6000611a678284612fb2565b7f3c7376672069643d226d6f7573652d7376672220786d6c6e733d22687474703a81527f2f2f7777772e77332e6f72672f323030302f737667222070726573657276654160208201527f7370656374526174696f3d22784d696e594d696e206d6565742220766965774260408201527f6f783d22302030203234203234223e203c7265637420636c6173733d2262672260608201526e103c1e911811103c9e91181110179f60891b60808201526000825161329c81608f850160208701613de2565b7f3c7374796c653e726563742e62677b77696474683a323470783b686569676874608f9390910192830152507f3a323470783b66696c6c3a234236454146467d20726563747b77696474683a3160af8201527f70783b6865696768743a3170783b7d20236d6f7573652d7376677b736861706560cf8201527f2d72656e646572696e673a20637269737065646765733b7d202e6330307b666960ef8201527f6c6c3a233030303030307d2e6330317b66696c6c3a234231414441437d2e633061010f8201527f327b66696c6c3a234437443744377d2e6330337b66696c6c3a2346464136413661012f8201527f7d2e6330347b66696c6c3a234646443444357d2e6330357b66696c6c3a23423961014f8201527f414439357d2e6330367b66696c6c3a234532443642457d2e6330377b66696c6c61016f8201527f3a233746363235417d2e6330387b66696c6c3a234135384638327d2e6330397b61018f8201527f66696c6c3a233442314530427d2e6331307b66696c6c3a233644324331307d2e6101af8201527f6331317b66696c6c3a234438443844387d2e6331327b66696c6c3a23463546356101cf8201527f46357d2e6331337b66696c6c3a233433334434427d2e6331347b66696c6c3a236101ef8201527f3844393439437d2e6331357b66696c6c3a233035464630307d2e6331367b666961020f8201527f6c6c3a233031433730307d2e6331377b66696c6c3a233042384630387d2e633161022f8201527f387b66696c6c3a233432314331337d2e6331397b66696c6c3a2336423339324161024f8201527f7d2e6332307b66696c6c3a234133354534307d2e6332317b66696c6c3a23444361026f8201527f424439317d2e6332327b66696c6c3a233737373737377d2e6332337b66696c6c61028f8201527f3a233834383438347d2e6332347b66696c6c3a234142414241427d2e6332357b6102af8201527f66696c6c3a234241424142417d2e6332367b66696c6c3a234337433743377d2e6102cf8201527f6332377b66696c6c3a234541454145417d2e6332387b66696c6c3a23304337366102ef8201527f41417d2e6332397b66696c6c3a233045393744427d2e6333307b66696c6c3a2361030f8201527f3130413445437d2e6333317b66696c6c3a233133423046467d2e6333327b666961032f8201527f6c6c3a233245423946457d2e6333337b66696c6c3a233534434346467d2e633361034f8201527f347b66696c6c3a233530433046327d2e6333357b66696c6c3a2335344343464661036f8201527f7d2e6333367b66696c6c3a233732444146467d2e6333377b66696c6c3a23423661038f8201527f454146467d2e6333387b66696c6c3a234646464646467d2e6333397b66696c6c6103af8201527f3a233935343534367d2e6334307b66696c6c3a233042383746377d2e6334317b6103cf8201527f66696c6c3a234646323632367d2e6334327b66696c6c3a233138304630327d2e6103ef8201527f6334337b66696c6c3a233242323331397d2e6334347b66696c6c3a234642444461040f8201527f34427d2e6334357b66696c6c3a234635423932337d2e6334367b66696c6c3a2361042f8201527f4343384131387d2e6334377b66696c6c3a233343323230337d2e6334387b666961044f8201527f6c6c3a233533333230427d2e6334397b66696c6c3a233742353031447d2e633561046f8201527f307b66696c6c3a234646453634367d2e6335317b66696c6c3a2346464436323761048f8201527f7d2e6335327b66696c6c3a234635423730307d2e6335337b66696c6c3a2332346104af8201527f323432347d2e6335347b66696c6c3a233441344134417d2e6335357b66696c6c6104cf8201527f3a233637363736377d2e6335367b66696c6c3a234630383330367d2e6335377b6104ef8201527f66696c6c3a234643413330457d2e6335387b66696c6c3a234645424330457d2e61050f8201527f6335397b66696c6c3a234642454331437d2e6336307b66696c6c3a233134323461052f8201527f32467d2e6336317b66696c6c3a234230363833377d2e6336327b66696c6c3a2361054f8201527f3846344230457d2e6336337b66696c6c3a234438383232377d2e6336347b666961056f8201527f6c6c3a234230363833377d3c2f7374796c653e3c2f7376673e0000000000000061058f8201526105a801919050565b605b60f81b815260008251613902816001850160208701613de2565b605d60f81b6001939091019283015250600201919050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c00000081526000825161395281601d850160208701613de2565b91909101601d0192915050565b727b226e616d65223a20224e66546f6b656e202360681b8152835160009061398e816013850160208901613de2565b7f222c20226465736372697074696f6e223a202254686973206973206120636f6c6013918401918201527f6c656374696f6e206f662031302c30303020756e6971756520696d616765732e60338201527f20416c6c20746865206d6574616461746120616e6420696d616765732061726560538201527f2067656e65726174656420616e642073746f7265642031303025206f6e2d636860738201527f61696e2e204e6f20495046532c206e6f204150492e222c2022696d616765223a60938201527f2022646174613a696d6167652f7376672b786d6c3b6261736536342c0000000060b38201528451613a898160cf840160208901613de2565b6e11161130ba3a3934b13aba32b9911d60891b60cf9290910191820152613ac3613ab660de830186612f96565b607d60f81b815260010190565b9695505050505050565b603160f81b815260008251613ae9816001850160208701613de2565b9190910160010192915050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090613ac390830184612f6a565b6020808252825182820181905260009190848201906040850190845b81811015613b6157835183529284019291840191600101613b45565b50909695505050505050565b602081526000611a676020830184612f6a565b608081526000613b936080830187612f6a565b8281036020840152613ba58187612f6a565b90508281036040840152613bb98186612f6a565b91505082606083015295945050505050565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b6040516080810167ffffffffffffffff81118282101715613c9157613c91613ee6565b60405290565b604051601f8201601f1916810167ffffffffffffffff81118282101715613cc057613cc0613ee6565b604052919050565b600061ffff80831681851680830382111561308c5761308c613eba565b60008219821115613cf857613cf8613eba565b500190565b600060ff821660ff84168060ff03821115613d1a57613d1a613eba565b019392505050565b600082613d3157613d31613ed0565b500490565b600061ffff80831681851681830481118215151615613d5757613d57613eba565b02949350505050565b6000816000190483118215151615613d7a57613d7a613eba565b500290565b600060ff821660ff84168160ff0481118215151615613da057613da0613eba565b029392505050565b600082821015613dba57613dba613eba565b500390565b600060ff821660ff841680821015613dd957613dd9613eba565b90039392505050565b60005b83811015613dfd578181015183820152602001613de5565b838111156116655750506000910152565b600181811c90821680613e2257607f821691505b60208210811415613e4357634e487b7160e01b600052602260045260246000fd5b50919050565b600061ffff80831681811415613e6157613e61613eba565b6001019392505050565b6000600019821415613e7f57613e7f613eba565b5060010190565b600060ff821660ff811415613e9d57613e9d613eba565b60010192915050565b600082613eb557613eb5613ed0565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b03198116811461099157600080fdfe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa264697066735822122095df3398ec2a2913850d0e25d77bea187e5bc19916b0df9ae930719f2d8b641f64736f6c63430008040033

Deployed Bytecode

0x6080604052600436106101d75760003560e01c8063438b630011610102578063a22cb46511610095578063d93dfe8811610064578063d93dfe881461054b578063e75166de1461056b578063e985e9c51461057e578063f2fde38b146105c757600080fd5b8063a22cb465146104cb578063b88d4fde146104eb578063c4ef71c71461050b578063c87b56dd1461052b57600080fd5b806370a08231116100d157806370a082311461046157806389ce30741461048157806395d89b41146104a15780639d1b464a146104b657600080fd5b8063438b6300146103d45780634f6ccce7146104015780636352211e1461042157806366e338701461044157600080fd5b806318160ddd1161017a5780632f745c59116101495780632f745c591461034f5780632fb098d21461036f5780633ccfd60b1461039f57806342842e0e146103b457600080fd5b806318160ddd146102db5780631da8491a146102fa57806323b872dd1461030f57806326a4e8d21461032f57600080fd5b8063081812fc116101b6578063081812fc14610257578063095ea7b31461028f578063098afd4b146102b1578063171be8d0146102c657600080fd5b80625ea307146101dc57806301ffc9a71461021257806306fdde0314610242575b600080fd5b3480156101e857600080fd5b506101fc6101f7366004612e63565b6105e7565b6040516102099190613b6d565b60405180910390f35b34801561021e57600080fd5b5061023261022d366004612df8565b6106de565b6040519015158152602001610209565b34801561024e57600080fd5b506101fc610703565b34801561026357600080fd5b50610277610272366004612e63565b610795565b6040516001600160a01b039091168152602001610209565b34801561029b57600080fd5b506102af6102aa366004612dcf565b61082f565b005b3480156102bd57600080fd5b506102af610945565b3480156102d257600080fd5b506102af610994565b3480156102e757600080fd5b506008545b604051908152602001610209565b34801561030657600080fd5b506102ec610a16565b34801561031b57600080fd5b506102af61032a366004612ce1565b610ad8565b34801561033b57600080fd5b506102af61034a366004612c95565b610b09565b34801561035b57600080fd5b506102ec61036a366004612dcf565b610b42565b34801561037b57600080fd5b5061038f61038a366004612f49565b610bd8565b6040516102099493929190613b80565b3480156103ab57600080fd5b506102af610dbd565b3480156103c057600080fd5b506102af6103cf366004612ce1565b610e07565b3480156103e057600080fd5b506103f46103ef366004612c95565b610e22565b6040516102099190613b29565b34801561040d57600080fd5b506102ec61041c366004612e63565b610ee0565b34801561042d57600080fd5b5061027761043c366004612e63565b610f81565b34801561044d57600080fd5b506101fc61045c366004612e30565b610ff8565b34801561046d57600080fd5b506102ec61047c366004612c95565b611151565b34801561048d57600080fd5b506101fc61049c366004612e30565b6111d8565b3480156104ad57600080fd5b506101fc6114eb565b3480156104c257600080fd5b506102ec6114fa565b3480156104d757600080fd5b506102af6104e6366004612d95565b61156e565b3480156104f757600080fd5b506102af610506366004612d1c565b611633565b34801561051757600080fd5b506102af610526366004612e63565b61166b565b34801561053757600080fd5b506101fc610546366004612e63565b61169d565b34801561055757600080fd5b506102af610566366004612e7b565b61172a565b6102af610579366004612e63565b6118b1565b34801561058a57600080fd5b50610232610599366004612caf565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b3480156105d357600080fd5b506102af6105e2366004612c95565b61193c565b6000818152600c602052604081208054606092919061060590613e0e565b80601f016020809104026020016040519081016040528092919081815260200182805461063190613e0e565b801561067e5780601f106106535761010080835404028352916020019161067e565b820191906000526020600020905b81548152906001019060200180831161066157829003601f168201915b5050505050905061068e83610f81565b6001600160a01b031661dead6001600160a01b031614156106d8576106b68160016009611975565b6040516020016106c69190613acd565b60405160208183030381529060405290505b92915050565b60006001600160e01b0319821663780e9d6360e01b14806106d857506106d882611a6e565b60606000805461071290613e0e565b80601f016020809104026020016040519081016040528092919081815260200182805461073e90613e0e565b801561078b5780601f106107605761010080835404028352916020019161078b565b820191906000526020600020905b81548152906001019060200180831161076e57829003601f168201915b5050505050905090565b6000818152600260205260408120546001600160a01b03166108135760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084015b60405180910390fd5b506000908152600460205260409020546001600160a01b031690565b600061083a82610f81565b9050806001600160a01b0316836001600160a01b031614156108a85760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b606482015260840161080a565b336001600160a01b03821614806108c457506108c48133610599565b6109365760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c0000000000000000606482015260840161080a565b6109408383611abe565b505050565b601e546001600160a01b0316331461095c57600080fd5b60005b6009811015610991576000818152600a6020526040812061097f916129e8565b8061098981613e6b565b91505061095f565b50565b601d546001600160a01b03166379cc6790336109ae610a16565b6040516001600160e01b031960e085901b1681526001600160a01b0390921660048301526024820152604401600060405180830381600087803b1580156109f457600080fd5b505af1158015610a08573d6000803e3d6000fd5b50505050610a14611b2c565b565b600080610a2260085490565b90506107d08111610a3c57670de0b6b3a764000091505090565b6107d081118015610a4f5750610fa08111155b15610a6357670de0b6b3a764000091505090565b610fa081118015610a7657506117708111155b15610a8a57671bc16d674ec8000091505090565b61177081118015610a9d5750611f408111155b15610ab1576729a2241af62c000091505090565b611f4081118015610ac457506127108111155b156101d757673782dace9d90000091505090565b610ae23382611bd2565b610afe5760405162461bcd60e51b815260040161080a90613c1d565b610940838383611cc9565b601e546001600160a01b03163314610b2057600080fd5b601d80546001600160a01b0319166001600160a01b0392909216919091179055565b6000610b4d83611151565b8210610baf5760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b606482015260840161080a565b506001600160a01b03919091166000908152600660209081526040808320938352929052205490565b600a6020528160005260406000208181548110610bf457600080fd5b906000526020600020906004020160009150915050806000018054610c1890613e0e565b80601f0160208091040260200160405190810160405280929190818152602001828054610c4490613e0e565b8015610c915780601f10610c6657610100808354040283529160200191610c91565b820191906000526020600020905b815481529060010190602001808311610c7457829003601f168201915b505050505090806001018054610ca690613e0e565b80601f0160208091040260200160405190810160405280929190818152602001828054610cd290613e0e565b8015610d1f5780601f10610cf457610100808354040283529160200191610d1f565b820191906000526020600020905b815481529060010190602001808311610d0257829003601f168201915b505050505090806002018054610d3490613e0e565b80601f0160208091040260200160405190810160405280929190818152602001828054610d6090613e0e565b8015610dad5780601f10610d8257610100808354040283529160200191610dad565b820191906000526020600020905b815481529060010190602001808311610d9057829003601f168201915b5050505050908060030154905084565b601e546001600160a01b03163314610dd457600080fd5b6040514790339082156108fc029083906000818181858888f19350505050158015610e03573d6000803e3d6000fd5b5050565b61094083838360405180602001604052806000815250611633565b60606000610e2f83611151565b905060008167ffffffffffffffff811115610e5a57634e487b7160e01b600052604160045260246000fd5b604051908082528060200260200182016040528015610e83578160200160208202803683370190505b50905060005b82811015610ed857610e9b8582610b42565b828281518110610ebb57634e487b7160e01b600052603260045260246000fd5b602090810291909101015280610ed081613e6b565b915050610e89565b509392505050565b6000610eeb60085490565b8210610f4e5760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b606482015260840161080a565b60088281548110610f6f57634e487b7160e01b600052603260045260246000fd5b90600052602060002001549050919050565b6000818152600260205260408120546001600160a01b0316806106d85760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201526832b73a103a37b5b2b760b91b606482015260840161080a565b60608060005b60098160ff16101561112857600061102f61102a8660ff8516611022866001613cfd565b60ff16611975565b611e74565b905082600a60008460ff1681526020019081526020016000208260ff168154811061106a57634e487b7160e01b600052603260045260246000fd5b9060005260206000209060040201600101600a60008560ff1681526020019081526020016000208360ff16815481106110b357634e487b7160e01b600052603260045260246000fd5b90600052602060002090600402016000016040516020016110d693929190613161565b60405160208183030381529060405292508160ff1660081461111557826040516020016111039190613095565b60405160208183030381529060405292505b508061112081613e86565b915050610ffe565b508060405160200161113a91906138e6565b604051602081830303815290604052915050919050565b60006001600160a01b0382166111bc5760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604482015269726f206164647265737360b01b606482015260840161080a565b506001600160a01b031660009081526003602052604090205490565b6060806111e3612a09565b60005b60098160ff1610156114c157600061120a61102a8760ff8516611022866001613cfd565b905060005b600a60008460ff1681526020019081526020016000208260ff168154811061124757634e487b7160e01b600052603260045260246000fd5b9060005260206000209060040201600301548161ffff1610156114ac57600061136b600a60008660ff1681526020019081526020016000208460ff16815481106112a157634e487b7160e01b600052603260045260246000fd5b906000526020600020906004020160020180546112bd90613e0e565b80601f01602080910402602001604051908101604052809291908181526020018280546112e990613e0e565b80156113365780601f1061130b57610100808354040283529160200191611336565b820191906000526020600020905b81548152906001019060200180831161131957829003601f168201915b50505050508360046113489190613d36565b61ffff16611357856004613d36565b611362906004613cc8565b61ffff16611975565b9050600061138461137f8360006001611975565b611f54565b9050600061139861137f8460016002611975565b9050868260ff16601881106113bd57634e487b7160e01b600052603260045260246000fd5b60200201518160ff16601881106113e457634e487b7160e01b600052603260045260246000fd5b6020020151156113f65750505061149a565b876114048460026004611975565b6114108460ff16612004565b61141c8460ff16612004565b60405160200161142f94939291906130ba565b60405160208183030381529060405297506001878360ff166018811061146557634e487b7160e01b600052603260045260246000fd5b60200201518260ff166018811061148c57634e487b7160e01b600052603260045260246000fd5b911515602090920201525050505b806114a481613e49565b91505061120f565b505080806114b990613e86565b9150506111e6565b50816040516020016114d391906131da565b60408051601f19818403018152919052949350505050565b60606001805461071290613e0e565b600060105442101561150d575060125490565b6000611529601154601054426115239190613da8565b9061211e565b90506012548160135461153c9190613d60565b1061154957600091505090565b806013546115579190613d60565b6012546115649190613da8565b91505090565b5090565b6001600160a01b0382163314156115c75760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c657200000000000000604482015260640161080a565b3360008181526005602090815260408083206001600160a01b03871680855290835292819020805460ff191686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b61163d3383611bd2565b6116595760405162461bcd60e51b815260040161080a90613c1d565b6116658484848461212a565b50505050565b3361167582610f81565b6001600160a01b03161461168857600080fd5b6116953361dead83611cc9565b610991611b2c565b6000818152600260205260409020546060906001600160a01b03166116c157600080fd5b60006116cc836105e7565b905061171a6116da84612004565b6116eb6116e6846111d8565b61215d565b6116f484610ff8565b6040516020016117069392919061395f565b60405160208183030381529060405261215d565b60405160200161113a919061391a565b601e546001600160a01b0316331461174157600080fd5b60005b815181101561094057600a6000848152602001908152602001600020604051806080016040528084848151811061178b57634e487b7160e01b600052603260045260246000fd5b60200260200101516000015181526020018484815181106117bc57634e487b7160e01b600052603260045260246000fd5b60200260200101516020015181526020018484815181106117ed57634e487b7160e01b600052603260045260246000fd5b602002602001015160400151815260200184848151811061181e57634e487b7160e01b600052603260045260246000fd5b6020908102919091018101516060015190915282546001810184556000938452928190208251805193946004029091019261185c9284920190612a37565b5060208281015180516118759260018501920190612a37565b5060408201518051611891916002840191602090910190612a37565b5060608201518160030155505080806118a990613e6b565b915050611744565b60006118bc60085490565b905060006118c86114fa565b90506000831180156118db575060058311155b6118e457600080fd5b6118ee8184613d60565b3410156118fa57600080fd5b600e54821061190857600080fd5b60105442101561191757600080fd5b60005b838110156116655761192a611b2c565b8061193481613e6b565b91505061191a565b601e546001600160a01b0316331461195357600080fd5b601e80546001600160a01b0319166001600160a01b0392909216919091179055565b60608360006119848585613da8565b67ffffffffffffffff8111156119aa57634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f1916602001820160405280156119d4576020820181803683370190505b509050845b84811015611a6257828181518110611a0157634e487b7160e01b600052603260045260246000fd5b01602001516001600160f81b03191682611a1b8884613da8565b81518110611a3957634e487b7160e01b600052603260045260246000fd5b60200101906001600160f81b031916908160001a90535080611a5a81613e6b565b9150506119d9565b509150505b9392505050565b60006001600160e01b031982166380ac58cd60e01b1480611a9f57506001600160e01b03198216635b5e139f60e01b145b806106d857506301ffc9a760e01b6001600160e01b03198316146106d8565b600081815260046020526040902080546001600160a01b0319166001600160a01b0384169081179091558190611af382610f81565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b6000611b3760085490565b9050600d548110611b4757600080fd5b333b15611b5357600080fd5b80611b60813360006122d3565b6000828152600c602090815260409091208251611b839391929190910190612a37565b506001600b600c6000848152602001908152602001600020604051611ba891906131ce565b908152604051908190036020019020805491151560ff19909216919091179055610e03338261241b565b6000818152600260205260408120546001600160a01b0316611c4b5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b606482015260840161080a565b6000611c5683610f81565b9050806001600160a01b0316846001600160a01b03161480611c915750836001600160a01b0316611c8684610795565b6001600160a01b0316145b80611cc157506001600160a01b0380821660009081526005602090815260408083209388168352929052205460ff165b949350505050565b826001600160a01b0316611cdc82610f81565b6001600160a01b031614611d445760405162461bcd60e51b815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201526839903737ba1037bbb760b91b606482015260840161080a565b6001600160a01b038216611da65760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b606482015260840161080a565b611db1838383612569565b611dbc600082611abe565b6001600160a01b0383166000908152600360205260408120805460019290611de5908490613da8565b90915550506001600160a01b0382166000908152600360205260408120805460019290611e13908490613ce5565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b60008181805b82518160ff161015610ed8576030838260ff1681518110611eab57634e487b7160e01b600052603260045260246000fd5b016020015160f81c10801590611eec57506039838260ff1681518110611ee157634e487b7160e01b600052603260045260246000fd5b016020015160f81c11155b15611f4257611efc600a83613d7f565b91506030838260ff1681518110611f2357634e487b7160e01b600052603260045260246000fd5b0160200151611f35919060f81c613dbf565b611f3f9083613cfd565b91505b80611f4c81613e86565b915050611e7a565b6000805b60145460ff82161015611ffe5782604051602001611f76919061304a565b6040516020818303038152906040528051906020012060148260ff1681548110611fb057634e487b7160e01b600052603260045260246000fd5b90600052602060002001604051602001611fca91906131ce565b604051602081830303815290604052805190602001201415611fec5792915050565b80611ff681613e86565b915050611f58565b50600080fd5b6060816120285750506040805180820190915260018152600360fc1b602082015290565b8160005b8115612052578061203c81613e6b565b915061204b9050600a83613d22565b915061202c565b60008167ffffffffffffffff81111561207b57634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f1916602001820160405280156120a5576020820181803683370190505b5090505b8415611cc1576120ba600183613da8565b91506120c7600a86613ea6565b6120d2906030613ce5565b60f81b8183815181106120f557634e487b7160e01b600052603260045260246000fd5b60200101906001600160f81b031916908160001a905350612117600a86613d22565b94506120a9565b6000611a678284613d22565b612135848484611cc9565b61214184848484612621565b6116655760405162461bcd60e51b815260040161080a90613bcb565b606081516000141561217d57505060408051602081019091526000815290565b6000604051806060016040528060408152602001613f1360409139905060006003845160026121ac9190613ce5565b6121b69190613d22565b6121c1906004613d60565b905060006121d0826020613ce5565b67ffffffffffffffff8111156121f657634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f191660200182016040528015612220576020820181803683370190505b509050818152600183018586518101602084015b8183101561228e5760039283018051603f601282901c811687015160f890811b8552600c83901c8216880151811b6001860152600683901c8216880151811b60028601529116860151901b93820193909352600401612234565b6003895106600181146122a857600281146122b9576122c5565b613d3d60f01b6001198301526122c5565b603d60f81b6000198301525b509398975050505050505050565b6060600a82106122e257600080fd5b6040805180820190915260018152600360fc1b602082015260005b60088160ff1610156123d557600f805490600061231983613e6b565b9091555050600f54604080514260208201524491810191909152606080820189905287901b6bffffffffffffffffffffffff191660808201526094810186905260b48101919091526000906127109060d4016040516020818303038152906040528051906020012060001c61238e9190613ea6565b90508261239f8261ffff168461272e565b6040516020016123b0929190613066565b60405160208183030381529060405292505080806123cd90613e86565b9150506122fd565b50600b816040516123e6919061304a565b9081526040519081900360200190205460ff1615611cc157612413858561240e866001613ce5565b6122d3565b915050611a67565b6001600160a01b0382166124715760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f2061646472657373604482015260640161080a565b6000818152600260205260409020546001600160a01b0316156124d65760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e74656400000000604482015260640161080a565b6124e260008383612569565b6001600160a01b038216600090815260036020526040812080546001929061250b908490613ce5565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6001600160a01b0383166125c4576125bf81600880546000838152600960205260408120829055600182018355919091527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee30155565b6125e7565b816001600160a01b0316836001600160a01b0316146125e7576125e7838261282e565b6001600160a01b0382166125fe57610940816128cb565b826001600160a01b0316826001600160a01b0316146109405761094082826129a4565b60006001600160a01b0384163b1561272357604051630a85bd0160e11b81526001600160a01b0385169063150b7a0290612665903390899088908890600401613af6565b602060405180830381600087803b15801561267f57600080fd5b505af19250505080156126af575060408051601f3d908101601f191682019092526126ac91810190612e14565b60015b612709573d8080156126dd576040519150601f19603f3d011682016040523d82523d6000602084013e6126e2565b606091505b5080516127015760405162461bcd60e51b815260040161080a90613bcb565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611cc1565b506001949350505050565b60606000805b60158460ff166008811061275857634e487b7160e01b600052603260045260246000fd5b015460ff82161015611ffe57600060158560ff166008811061278a57634e487b7160e01b600052603260045260246000fd5b018260ff16815481106127ad57634e487b7160e01b600052603260045260246000fd5b90600052602060002090601091828204019190066002029054906101000a900461ffff1690508261ffff1686101580156127f357506127ec8184613cc8565b61ffff1686105b1561280e576128048260ff16612004565b93505050506106d8565b6128188184613cc8565b925050808061282690613e86565b915050612734565b6000600161283b84611151565b6128459190613da8565b600083815260076020526040902054909150808214612898576001600160a01b03841660009081526006602090815260408083208584528252808320548484528184208190558352600790915290208190555b5060009182526007602090815260408084208490556001600160a01b039094168352600681528383209183525290812055565b6008546000906128dd90600190613da8565b6000838152600960205260408120546008805493945090928490811061291357634e487b7160e01b600052603260045260246000fd5b90600052602060002001549050806008838154811061294257634e487b7160e01b600052603260045260246000fd5b600091825260208083209091019290925582815260099091526040808220849055858252812055600880548061298857634e487b7160e01b600052603160045260246000fd5b6001900381819060005260206000200160009055905550505050565b60006129af83611151565b6001600160a01b039093166000908152600660209081526040808320868452825280832085905593825260079052919091209190915550565b50805460008255600402906000526020600020908101906109919190612ab7565b6040518061030001604052806018905b612a21612af7565b815260200190600190039081612a195790505090565b828054612a4390613e0e565b90600052602060002090601f016020900481019282612a655760008555612aab565b82601f10612a7e57805160ff1916838001178555612aab565b82800160010185558215612aab579182015b82811115612aab578251825591602001919060010190612a90565b5061156a929150612b16565b8082111561156a576000612acb8282612b2b565b612ad9600183016000612b2b565b612ae7600283016000612b2b565b5060006003820155600401612ab7565b6040518061030001604052806018906020820280368337509192915050565b5b8082111561156a5760008155600101612b17565b508054612b3790613e0e565b6000825580601f10612b47575050565b601f0160209004906000526020600020908101906109919190612b16565b600067ffffffffffffffff831115612b7f57612b7f613ee6565b612b92601f8401601f1916602001613c97565b9050828152838383011115612ba657600080fd5b828260208301376000602084830101529392505050565b80356001600160a01b0381168114612bd457600080fd5b919050565b600082601f830112612be9578081fd5b611a6783833560208501612b65565b600060808284031215612c09578081fd5b612c11613c6e565b9050813567ffffffffffffffff80821115612c2b57600080fd5b612c3785838601612bd9565b83526020840135915080821115612c4d57600080fd5b612c5985838601612bd9565b60208401526040840135915080821115612c7257600080fd5b50612c7f84828501612bd9565b6040830152506060820135606082015292915050565b600060208284031215612ca6578081fd5b611a6782612bbd565b60008060408385031215612cc1578081fd5b612cca83612bbd565b9150612cd860208401612bbd565b90509250929050565b600080600060608486031215612cf5578081fd5b612cfe84612bbd565b9250612d0c60208501612bbd565b9150604084013590509250925092565b60008060008060808587031215612d31578081fd5b612d3a85612bbd565b9350612d4860208601612bbd565b925060408501359150606085013567ffffffffffffffff811115612d6a578182fd5b8501601f81018713612d7a578182fd5b612d8987823560208401612b65565b91505092959194509250565b60008060408385031215612da7578182fd5b612db083612bbd565b915060208301358015158114612dc4578182fd5b809150509250929050565b60008060408385031215612de1578182fd5b612dea83612bbd565b946020939093013593505050565b600060208284031215612e09578081fd5b8135611a6781613efc565b600060208284031215612e25578081fd5b8151611a6781613efc565b600060208284031215612e41578081fd5b813567ffffffffffffffff811115612e57578182fd5b611cc184828501612bd9565b600060208284031215612e74578081fd5b5035919050565b60008060408385031215612e8d578182fd5b8235915060208084013567ffffffffffffffff80821115612eac578384fd5b818601915086601f830112612ebf578384fd5b813581811115612ed157612ed1613ee6565b8060051b612ee0858201613c97565b8281528581019085870183870188018c1015612efa578889fd5b8893505b84841015612f3757803586811115612f1457898afd5b612f228d8a838b0101612bf8565b84525060019390930192918701918701612efe565b50809750505050505050509250929050565b60008060408385031215612f5b578182fd5b50508035926020909101359150565b60008151808452612f82816020860160208601613de2565b601f01601f19169290920160200192915050565b60008151612fa8818560208601613de2565b9290920192915050565b8054600090600181811c9080831680612fcc57607f831692505b6020808410821415612fec57634e487b7160e01b86526022600452602486fd5b81801561300057600181146130115761303e565b60ff1986168952848901965061303e565b60008881526020902060005b868110156130365781548b82015290850190830161301d565b505084890196505b50505050505092915050565b6000825161305c818460208701613de2565b9190910192915050565b60008351613078818460208801613de2565b83519083019061308c818360208801613de2565b01949350505050565b600082516130a7818460208701613de2565b600b60fa1b920191825250600101919050565b600085516130cc818460208a01613de2565b6d3c7265637420636c6173733d276360901b90830190815285516130f781600e840160208a01613de2565b642720783d2760d81b600e9290910191820152845161311d816013840160208901613de2565b642720793d2760d81b601392909101918201528351613143816018840160208801613de2565b6213979f60e91b60189290910191820152601b019695505050505050565b60008451613173818460208901613de2565b6e3d913a3930b4ba2fba3cb832911d1160891b908301908152613199600f820186612fb2565b6a1116113b30b63ab2911d1160a91b815290506131b9600b820185612fb2565b61227d60f01b81526002019695505050505050565b6000611a678284612fb2565b7f3c7376672069643d226d6f7573652d7376672220786d6c6e733d22687474703a81527f2f2f7777772e77332e6f72672f323030302f737667222070726573657276654160208201527f7370656374526174696f3d22784d696e594d696e206d6565742220766965774260408201527f6f783d22302030203234203234223e203c7265637420636c6173733d2262672260608201526e103c1e911811103c9e91181110179f60891b60808201526000825161329c81608f850160208701613de2565b7f3c7374796c653e726563742e62677b77696474683a323470783b686569676874608f9390910192830152507f3a323470783b66696c6c3a234236454146467d20726563747b77696474683a3160af8201527f70783b6865696768743a3170783b7d20236d6f7573652d7376677b736861706560cf8201527f2d72656e646572696e673a20637269737065646765733b7d202e6330307b666960ef8201527f6c6c3a233030303030307d2e6330317b66696c6c3a234231414441437d2e633061010f8201527f327b66696c6c3a234437443744377d2e6330337b66696c6c3a2346464136413661012f8201527f7d2e6330347b66696c6c3a234646443444357d2e6330357b66696c6c3a23423961014f8201527f414439357d2e6330367b66696c6c3a234532443642457d2e6330377b66696c6c61016f8201527f3a233746363235417d2e6330387b66696c6c3a234135384638327d2e6330397b61018f8201527f66696c6c3a233442314530427d2e6331307b66696c6c3a233644324331307d2e6101af8201527f6331317b66696c6c3a234438443844387d2e6331327b66696c6c3a23463546356101cf8201527f46357d2e6331337b66696c6c3a233433334434427d2e6331347b66696c6c3a236101ef8201527f3844393439437d2e6331357b66696c6c3a233035464630307d2e6331367b666961020f8201527f6c6c3a233031433730307d2e6331377b66696c6c3a233042384630387d2e633161022f8201527f387b66696c6c3a233432314331337d2e6331397b66696c6c3a2336423339324161024f8201527f7d2e6332307b66696c6c3a234133354534307d2e6332317b66696c6c3a23444361026f8201527f424439317d2e6332327b66696c6c3a233737373737377d2e6332337b66696c6c61028f8201527f3a233834383438347d2e6332347b66696c6c3a234142414241427d2e6332357b6102af8201527f66696c6c3a234241424142417d2e6332367b66696c6c3a234337433743377d2e6102cf8201527f6332377b66696c6c3a234541454145417d2e6332387b66696c6c3a23304337366102ef8201527f41417d2e6332397b66696c6c3a233045393744427d2e6333307b66696c6c3a2361030f8201527f3130413445437d2e6333317b66696c6c3a233133423046467d2e6333327b666961032f8201527f6c6c3a233245423946457d2e6333337b66696c6c3a233534434346467d2e633361034f8201527f347b66696c6c3a233530433046327d2e6333357b66696c6c3a2335344343464661036f8201527f7d2e6333367b66696c6c3a233732444146467d2e6333377b66696c6c3a23423661038f8201527f454146467d2e6333387b66696c6c3a234646464646467d2e6333397b66696c6c6103af8201527f3a233935343534367d2e6334307b66696c6c3a233042383746377d2e6334317b6103cf8201527f66696c6c3a234646323632367d2e6334327b66696c6c3a233138304630327d2e6103ef8201527f6334337b66696c6c3a233242323331397d2e6334347b66696c6c3a234642444461040f8201527f34427d2e6334357b66696c6c3a234635423932337d2e6334367b66696c6c3a2361042f8201527f4343384131387d2e6334377b66696c6c3a233343323230337d2e6334387b666961044f8201527f6c6c3a233533333230427d2e6334397b66696c6c3a233742353031447d2e633561046f8201527f307b66696c6c3a234646453634367d2e6335317b66696c6c3a2346464436323761048f8201527f7d2e6335327b66696c6c3a234635423730307d2e6335337b66696c6c3a2332346104af8201527f323432347d2e6335347b66696c6c3a233441344134417d2e6335357b66696c6c6104cf8201527f3a233637363736377d2e6335367b66696c6c3a234630383330367d2e6335377b6104ef8201527f66696c6c3a234643413330457d2e6335387b66696c6c3a234645424330457d2e61050f8201527f6335397b66696c6c3a234642454331437d2e6336307b66696c6c3a233134323461052f8201527f32467d2e6336317b66696c6c3a234230363833377d2e6336327b66696c6c3a2361054f8201527f3846344230457d2e6336337b66696c6c3a234438383232377d2e6336347b666961056f8201527f6c6c3a234230363833377d3c2f7374796c653e3c2f7376673e0000000000000061058f8201526105a801919050565b605b60f81b815260008251613902816001850160208701613de2565b605d60f81b6001939091019283015250600201919050565b7f646174613a6170706c69636174696f6e2f6a736f6e3b6261736536342c00000081526000825161395281601d850160208701613de2565b91909101601d0192915050565b727b226e616d65223a20224e66546f6b656e202360681b8152835160009061398e816013850160208901613de2565b7f222c20226465736372697074696f6e223a202254686973206973206120636f6c6013918401918201527f6c656374696f6e206f662031302c30303020756e6971756520696d616765732e60338201527f20416c6c20746865206d6574616461746120616e6420696d616765732061726560538201527f2067656e65726174656420616e642073746f7265642031303025206f6e2d636860738201527f61696e2e204e6f20495046532c206e6f204150492e222c2022696d616765223a60938201527f2022646174613a696d6167652f7376672b786d6c3b6261736536342c0000000060b38201528451613a898160cf840160208901613de2565b6e11161130ba3a3934b13aba32b9911d60891b60cf9290910191820152613ac3613ab660de830186612f96565b607d60f81b815260010190565b9695505050505050565b603160f81b815260008251613ae9816001850160208701613de2565b9190910160010192915050565b6001600160a01b0385811682528416602082015260408101839052608060608201819052600090613ac390830184612f6a565b6020808252825182820181905260009190848201906040850190845b81811015613b6157835183529284019291840191600101613b45565b50909695505050505050565b602081526000611a676020830184612f6a565b608081526000613b936080830187612f6a565b8281036020840152613ba58187612f6a565b90508281036040840152613bb98186612f6a565b91505082606083015295945050505050565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b6040516080810167ffffffffffffffff81118282101715613c9157613c91613ee6565b60405290565b604051601f8201601f1916810167ffffffffffffffff81118282101715613cc057613cc0613ee6565b604052919050565b600061ffff80831681851680830382111561308c5761308c613eba565b60008219821115613cf857613cf8613eba565b500190565b600060ff821660ff84168060ff03821115613d1a57613d1a613eba565b019392505050565b600082613d3157613d31613ed0565b500490565b600061ffff80831681851681830481118215151615613d5757613d57613eba565b02949350505050565b6000816000190483118215151615613d7a57613d7a613eba565b500290565b600060ff821660ff84168160ff0481118215151615613da057613da0613eba565b029392505050565b600082821015613dba57613dba613eba565b500390565b600060ff821660ff841680821015613dd957613dd9613eba565b90039392505050565b60005b83811015613dfd578181015183820152602001613de5565b838111156116655750506000910152565b600181811c90821680613e2257607f821691505b60208210811415613e4357634e487b7160e01b600052602260045260246000fd5b50919050565b600061ffff80831681811415613e6157613e61613eba565b6001019392505050565b6000600019821415613e7f57613e7f613eba565b5060010190565b600060ff821660ff811415613e9d57613e9d613eba565b60010192915050565b600082613eb557613eb5613ed0565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160e01b03198116811461099157600080fdfe4142434445464748494a4b4c4d4e4f505152535455565758595a6162636465666768696a6b6c6d6e6f707172737475767778797a303132333435363738392b2fa264697066735822122095df3398ec2a2913850d0e25d77bea187e5bc19916b0df9ae930719f2d8b641f64736f6c63430008040033

Deployed Bytecode Sourcemap

55357:17924:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;69707:548;;;;;;;;;;-1:-1:-1;69707:548:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;49112:224;;;;;;;;;;-1:-1:-1;49112:224:0;;;;;:::i;:::-;;:::i;:::-;;;20908:14:1;;20901:22;20883:41;;20871:2;20856:18;49112:224:0;20838:92:1;37004:100:0;;;;;;;;;;;;;:::i;38563:221::-;;;;;;;;;;-1:-1:-1;38563:221:0;;;;;:::i;:::-;;:::i;:::-;;;-1:-1:-1;;;;;19287:32:1;;;19269:51;;19257:2;19242:18;38563:221:0;19224:102:1;38086:411:0;;;;;;;;;;-1:-1:-1;38086:411:0;;;;;:::i;:::-;;:::i;:::-;;71709:137;;;;;;;;;;;;;:::i;62111:184::-;;;;;;;;;;;;;:::i;49752:113::-;;;;;;;;;;-1:-1:-1;49840:10:0;:17;49752:113;;;27949:25:1;;;27937:2;27922:18;49752:113:0;27904:76:1;59976:596:0;;;;;;;;;;;;;:::i;39453:339::-;;;;;;;;;;-1:-1:-1;39453:339:0;;;;;:::i;:::-;;:::i;72592:112::-;;;;;;;;;;-1:-1:-1;72592:112:0;;;;;:::i;:::-;;:::i;49420:256::-;;;;;;;;;;-1:-1:-1;49420:256:0;;;;;:::i;:::-;;:::i;55620:45::-;;;;;;;;;;-1:-1:-1;55620:45:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;;;;:::i;72963:145::-;;;;;;;;;;;;;:::i;39863:185::-;;;;;;;;;;-1:-1:-1;39863:185:0;;;;;:::i;:::-;;:::i;70423:380::-;;;;;;;;;;-1:-1:-1;70423:380:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;49942:233::-;;;;;;;;;;-1:-1:-1;49942:233:0;;;;;:::i;:::-;;:::i;36698:239::-;;;;;;;;;;-1:-1:-1;36698:239:0;;;;;:::i;:::-;;:::i;67188:887::-;;;;;;;;;;-1:-1:-1;67188:887:0;;;;;:::i;:::-;;:::i;36428:208::-;;;;;;;;;;-1:-1:-1;36428:208:0;;;;;:::i;:::-;;:::i;63974:3149::-;;;;;;;;;;-1:-1:-1;63974:3149:0;;;;;:::i;:::-;;:::i;37173:104::-;;;;;;;;;;;;;:::i;60649:421::-;;;;;;;;;;;;;:::i;38856:295::-;;;;;;;;;;-1:-1:-1;38856:295:0;;;;;:::i;:::-;;:::i;40119:328::-;;;;;;;;;;-1:-1:-1;40119:328:0;;;;;:::i;:::-;;:::i;62398:292::-;;;;;;;;;;-1:-1:-1;62398:292:0;;;;;:::i;:::-;;:::i;68230:1343::-;;;;;;;;;;-1:-1:-1;68230:1343:0;;;;;:::i;:::-;;:::i;72001:474::-;;;;;;;;;;-1:-1:-1;72001:474:0;;;;;:::i;:::-;;:::i;61624:435::-;;;;;;:::i;:::-;;:::i;39222:164::-;;;;;;;;;;-1:-1:-1;39222:164:0;;;;;:::i;:::-;-1:-1:-1;;;;;39343:25:0;;;39319:4;39343:25;;;:18;:25;;;;;;;;:35;;;;;;;;;;;;;;;39222:164;72802:100;;;;;;;;;;-1:-1:-1;72802:100:0;;;;;:::i;:::-;;:::i;69707:548::-;69828:23;69854;;;:13;:23;;;;;69828:49;;69797:13;;69828:23;69854;69828:49;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;69957:17;69965:8;69957:7;:17::i;:::-;-1:-1:-1;;;;;69957:63:0;69978:42;-1:-1:-1;;;;;69957:63:0;;69953:266;;;70139:34;70157:9;70168:1;70171;70139:17;:34::i;:::-;70074:118;;;;;;;;:::i;:::-;;;;;;;;;;;;;70037:170;;69953:266;70238:9;69707:548;-1:-1:-1;;69707:548:0:o;49112:224::-;49214:4;-1:-1:-1;;;;;;49238:50:0;;-1:-1:-1;;;49238:50:0;;:90;;;49292:36;49316:11;49292:23;:36::i;37004:100::-;37058:13;37091:5;37084:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;37004:100;:::o;38563:221::-;38639:7;42046:16;;;:7;:16;;;;;;-1:-1:-1;;;;;42046:16:0;38659:73;;;;-1:-1:-1;;;38659:73:0;;25949:2:1;38659:73:0;;;25931:21:1;25988:2;25968:18;;;25961:30;26027:34;26007:18;;;26000:62;-1:-1:-1;;;26078:18:1;;;26071:42;26130:19;;38659:73:0;;;;;;;;;-1:-1:-1;38752:24:0;;;;:15;:24;;;;;;-1:-1:-1;;;;;38752:24:0;;38563:221::o;38086:411::-;38167:13;38183:23;38198:7;38183:14;:23::i;:::-;38167:39;;38231:5;-1:-1:-1;;;;;38225:11:0;:2;-1:-1:-1;;;;;38225:11:0;;;38217:57;;;;-1:-1:-1;;;38217:57:0;;26772:2:1;38217:57:0;;;26754:21:1;26811:2;26791:18;;;26784:30;26850:34;26830:18;;;26823:62;-1:-1:-1;;;26901:18:1;;;26894:31;26942:19;;38217:57:0;26744:223:1;38217:57:0;17198:10;-1:-1:-1;;;;;38309:21:0;;;;:62;;-1:-1:-1;38334:37:0;38351:5;17198:10;39222:164;:::i;38334:37::-;38287:168;;;;-1:-1:-1;;;38287:168:0;;24342:2:1;38287:168:0;;;24324:21:1;24381:2;24361:18;;;24354:30;24420:34;24400:18;;;24393:62;24491:26;24471:18;;;24464:54;24535:19;;38287:168:0;24314:246:1;38287:168:0;38468:21;38477:2;38481:7;38468:8;:21::i;:::-;38086:411;;;:::o;71709:137::-;73237:6;;-1:-1:-1;;;;;73237:6:0;73247:10;73237:20;73229:29;;;;;;71765:9:::1;71760:79;71784:1;71780;:5;71760:79;;;71814:13;::::0;;;:10:::1;:13;::::0;;;;71807:20:::1;::::0;::::1;:::i;:::-;71787:3:::0;::::1;::::0;::::1;:::i;:::-;;;;71760:79;;;;71709:137::o:0;62111:184::-;62201:12;;-1:-1:-1;;;;;62201:12:0;62194:29;62224:10;62236:18;:16;:18::i;:::-;62194:61;;-1:-1:-1;;;;;;62194:61:0;;;;;;;-1:-1:-1;;;;;20016:32:1;;;62194:61:0;;;19998:51:1;20065:18;;;20058:34;19971:18;;62194:61:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;62273:14;:12;:14::i;:::-;62111:184::o;59976:596::-;60025:7;60045:20;60068:13;49840:10;:17;;49752:113;60068:13;60045:36;;60114:4;60098:12;:20;60094:52;;60127:19;60120:26;;;59976:596;:::o;60094:52::-;60176:4;60161:12;:19;:43;;;;;60200:4;60184:12;:20;;60161:43;60157:88;;;60226:19;60219:26;;;59976:596;:::o;60157:88::-;60275:4;60260:12;:19;:43;;;;;60299:4;60283:12;:20;;60260:43;60256:88;;;60325:19;60318:26;;;59976:596;:::o;60256:88::-;60374:4;60359:12;:19;:43;;;;;60398:4;60382:12;:20;;60359:43;60355:88;;;60424:19;60417:26;;;59976:596;:::o;60355:88::-;60473:4;60458:12;:19;:44;;;;;60497:5;60481:12;:21;;60458:44;60454:89;;;60524:19;60517:26;;;59976:596;:::o;39453:339::-;39648:41;17198:10;39681:7;39648:18;:41::i;:::-;39640:103;;;;-1:-1:-1;;;39640:103:0;;;;;;;:::i;:::-;39756:28;39766:4;39772:2;39776:7;39756:9;:28::i;72592:112::-;73237:6;;-1:-1:-1;;;;;73237:6:0;73247:10;73237:20;73229:29;;;;;;72668:12:::1;:28:::0;;-1:-1:-1;;;;;;72668:28:0::1;-1:-1:-1::0;;;;;72668:28:0;;;::::1;::::0;;;::::1;::::0;;72592:112::o;49420:256::-;49517:7;49553:23;49570:5;49553:16;:23::i;:::-;49545:5;:31;49537:87;;;;-1:-1:-1;;;49537:87:0;;21982:2:1;49537:87:0;;;21964:21:1;22021:2;22001:18;;;21994:30;22060:34;22040:18;;;22033:62;-1:-1:-1;;;22111:18:1;;;22104:41;22162:19;;49537:87:0;21954:233:1;49537:87:0;-1:-1:-1;;;;;;49642:19:0;;;;;;;;:12;:19;;;;;;;;:26;;;;;;;;;49420:256::o;55620:45::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;72963:145::-;73237:6;;-1:-1:-1;;;;;73237:6:0;73247:10;73237:20;73229:29;;;;;;73063:37:::1;::::0;73029:21:::1;::::0;73071:10:::1;::::0;73063:37;::::1;;;::::0;73029:21;;73011:15:::1;73063:37:::0;73011:15;73063:37;73029:21;73071:10;73063:37;::::1;;;;;;;;;;;;;::::0;::::1;;;;;;73269:1;72963:145::o:0;39863:185::-;40001:39;40018:4;40024:2;40028:7;40001:39;;;;;;;;;;;;:16;:39::i;70423:380::-;70511:16;70545:18;70566;70576:7;70566:9;:18::i;:::-;70545:39;;70597:25;70639:10;70625:25;;;;;;-1:-1:-1;;;70625:25:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;70625:25:0;;70597:53;;70666:9;70661:109;70681:10;70677:1;:14;70661:109;;;70727:31;70747:7;70756:1;70727:19;:31::i;:::-;70713:8;70722:1;70713:11;;;;;;-1:-1:-1;;;70713:11:0;;;;;;;;;;;;;;;;;;:45;70693:3;;;;:::i;:::-;;;;70661:109;;;-1:-1:-1;70787:8:0;70423:380;-1:-1:-1;;;70423:380:0:o;49942:233::-;50017:7;50053:30;49840:10;:17;;49752:113;50053:30;50045:5;:38;50037:95;;;;-1:-1:-1;;;50037:95:0;;27592:2:1;50037:95:0;;;27574:21:1;27631:2;27611:18;;;27604:30;27670:34;27650:18;;;27643:62;-1:-1:-1;;;27721:18:1;;;27714:42;27773:19;;50037:95:0;27564:234:1;50037:95:0;50150:10;50161:5;50150:17;;;;;;-1:-1:-1;;;50150:17:0;;;;;;;;;;;;;;;;;50143:24;;49942:233;;;:::o;36698:239::-;36770:7;36806:16;;;:7;:16;;;;;;-1:-1:-1;;;;;36806:16:0;36841:19;36833:73;;;;-1:-1:-1;;;36833:73:0;;25178:2:1;36833:73:0;;;25160:21:1;25217:2;25197:18;;;25190:30;25256:34;25236:18;;;25229:62;-1:-1:-1;;;25307:18:1;;;25300:39;25356:19;;36833:73:0;25150:231:1;67188:887:0;67281:13;67312:28;67358:7;67353:645;67375:1;67371;:5;;;67353:645;;;67398:20;67421:84;67456:34;67474:5;67456:34;;;67484:5;67481:1;67488;67484:5;:::i;:::-;67456:34;;:17;:34::i;:::-;67421:16;:84::i;:::-;67398:107;;67603:14;67680:10;:13;67691:1;67680:13;;;;;;;;;;;;;67694:14;67680:29;;;;;;;;-1:-1:-1;;;67680:29:0;;;;;;;;;;;;;;;;;;;:39;;67778:10;:13;67789:1;67778:13;;;;;;;;;;;;;67792:14;67778:29;;;;;;;;-1:-1:-1;;;67778:29:0;;;;;;;;;;;;;;;;;;;:39;;67564:299;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;67522:356;;67899:1;:6;;67904:1;67899:6;67895:91;;67965:14;67948:37;;;;;;;;:::i;:::-;;;;;;;;;;;;;67924:62;;67895:91;-1:-1:-1;67378:3:0;;;;:::i;:::-;;;;67353:645;;;;68046:14;68024:42;;;;;;;;:::i;:::-;;;;;;;;;;;;;68010:57;;;67188:887;;;:::o;36428:208::-;36500:7;-1:-1:-1;;;;;36528:19:0;;36520:74;;;;-1:-1:-1;;;36520:74:0;;24767:2:1;36520:74:0;;;24749:21:1;24806:2;24786:18;;;24779:30;24845:34;24825:18;;;24818:62;-1:-1:-1;;;24896:18:1;;;24889:40;24946:19;;36520:74:0;24739:232:1;36520:74:0;-1:-1:-1;;;;;;36612:16:0;;;;;:9;:16;;;;;;;36428:208::o;63974:3149::-;64062:13;64093:23;64127:32;;:::i;:::-;64177:7;64172:1309;64194:1;64190;:5;;;64172:1309;;;64217:20;64240:84;64275:34;64293:5;64275:34;;;64303:5;64300:1;64307;64303:5;:::i;64240:84::-;64217:107;;64364:8;64341:1129;64399:10;:13;64410:1;64399:13;;;;;;;;;;;;;64413:14;64399:29;;;;;;;;-1:-1:-1;;;64399:29:0;;;;;;;;;;;;;;;;;;;:40;;;64395:1;:44;;;64341:1129;;;64496:23;64522:155;64562:10;:13;64573:1;64562:13;;;;;;;;;;;;;64576:14;64562:29;;;;;;;;-1:-1:-1;;;64562:29:0;;;;;;;;;;;;;;;;;;;:36;;64522:155;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;64621:1;64625;64621:5;;;;:::i;:::-;64522:155;;64649:5;:1;64653;64649:5;:::i;:::-;:9;;64657:1;64649:9;:::i;:::-;64522:155;;:17;:155::i;:::-;64496:181;;64698:7;64708:90;64745:34;64763:9;64774:1;64777;64745:17;:34::i;:::-;64708:14;:90::i;:::-;64698:100;;64817:7;64827:90;64864:34;64882:9;64893:1;64896;64864:17;:34::i;64827:90::-;64817:100;;64942:12;64955:1;64942:15;;;;;;;-1:-1:-1;;;64942:15:0;;;;;;;;;;;;;64958:1;64942:18;;;;;;;-1:-1:-1;;;64942:18:0;;;;;;;;;;;;;64938:32;;;64962:8;;;;;64938:32;65075:9;65154:34;65172:9;65183:1;65186;65154:17;:34::i;:::-;65249:12;:1;:10;;;:12::i;:::-;65322;:1;:10;;;:12::i;:::-;65032:357;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;64991:417;;65450:4;65429:12;65442:1;65429:15;;;;;;;-1:-1:-1;;;65429:15:0;;;;;;;;;;;;;65445:1;65429:18;;;;;;;-1:-1:-1;;;65429:18:0;;;;;;;;;:25;;;:18;;;;;:25;-1:-1:-1;;;64341:1129:0;64458:3;;;;:::i;:::-;;;;64341:1129;;;;64172:1309;64197:3;;;;;:::i;:::-;;;;64172:1309;;;;65725:9;65526:1549;;;;;;;;:::i;:::-;;;;-1:-1:-1;;65526:1549:0;;;;;;;;;;63974:3149;-1:-1:-1;;;;63974:3149:0:o;37173:104::-;37229:13;37262:7;37255:14;;;;;:::i;60649:421::-;60694:7;60735:10;;60717:15;:28;60714:78;;;-1:-1:-1;60769:11:0;;;60649:421::o;60714:78::-;60804:26;60833:46;60868:10;;60852;;60834:15;:28;;;;:::i;:::-;60833:34;;:46::i;:::-;60804:75;;60928:11;;60906:18;60893:10;;:31;;;;:::i;:::-;:46;60890:173;;60963:1;60956:8;;;60649:421;:::o;60890:173::-;61032:18;61019:10;;:31;;;;:::i;:::-;61004:11;;:47;;;;:::i;:::-;60997:54;;;60649:421;:::o;60890:173::-;60649:421;;:::o;38856:295::-;-1:-1:-1;;;;;38959:24:0;;17198:10;38959:24;;38951:62;;;;-1:-1:-1;;;38951:62:0;;23575:2:1;38951:62:0;;;23557:21:1;23614:2;23594:18;;;23587:30;23653:27;23633:18;;;23626:55;23698:18;;38951:62:0;23547:175:1;38951:62:0;17198:10;39026:32;;;;:18;:32;;;;;;;;-1:-1:-1;;;;;39026:42:0;;;;;;;;;;;;:53;;-1:-1:-1;;39026:53:0;;;;;;;;;;39095:48;;20883:41:1;;;39026:42:0;;17198:10;39095:48;;20856:18:1;39095:48:0;;;;;;;38856:295;;:::o;40119:328::-;40294:41;17198:10;40327:7;40294:18;:41::i;:::-;40286:103;;;;-1:-1:-1;;;40286:103:0;;;;;;;:::i;:::-;40400:39;40414:4;40420:2;40424:7;40433:5;40400:13;:39::i;:::-;40119:328;;;;:::o;62398:292::-;62484:10;62463:17;62471:8;62463:7;:17::i;:::-;-1:-1:-1;;;;;62463:31:0;;62455:40;;;;;;62530:125;62554:10;62579:42;62636:8;62530:9;:125::i;:::-;62668:14;:12;:14::i;68230:1343::-;42022:4;42046:16;;;:7;:16;;;;;;68332:13;;-1:-1:-1;;;;;42046:16:0;68363:26;;;;;;68402:23;68428:24;68443:8;68428:14;:24::i;:::-;68402:50;;68603:928;68836:26;68853:8;68836:16;:26::i;:::-;69130:123;69193:20;69203:9;69193;:20::i;:::-;69130:14;:123::i;:::-;69348:25;69363:9;69348:14;:25::i;:::-;68721:729;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;68603:14;:928::i;:::-;68510:1040;;;;;;;;:::i;72001:474::-;73237:6;;-1:-1:-1;;;;;73237:6:0;73247:10;73237:20;73229:29;;;;;;72127:9:::1;72122:327;72146:6;:13;72142:1;:17;72122:327;;;72181:10;:27;72192:15;72181:27;;;;;;;;;;;72232:190;;;;;;;;72260:6;72267:1;72260:9;;;;;;-1:-1:-1::0;;;72260:9:0::1;;;;;;;;;;;;;;;:19;;;72232:190;;;;72302:6;72309:1;72302:9;;;;;;-1:-1:-1::0;;;72302:9:0::1;;;;;;;;;;;;;;;:19;;;72232:190;;;;72344:6;72351:1;72344:9;;;;;;-1:-1:-1::0;;;72344:9:0::1;;;;;;;;;;;;;;;:16;;;72232:190;;;;72383:6;72390:1;72383:9;;;;;;-1:-1:-1::0;;;72383:9:0::1;;;;;;;;;;::::0;;::::1;::::0;;;;;;;:20:::1;;::::0;72232:190;;;72181:256;;::::1;::::0;::::1;::::0;;-1:-1:-1;72181:256:0;;;;;;;;;;;;;::::1;;::::0;;::::1;::::0;::::1;::::0;;;::::1;::::0;::::1;:::i;:::-;-1:-1:-1::0;72181:256:0::1;::::0;;::::1;::::0;;;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;::::1;:::i;:::-;-1:-1:-1::0;72181:256:0::1;::::0;::::1;::::0;;;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;;::::1;::::0;::::1;:::i;:::-;;;;;;;;;;;;72161:3;;;;;:::i;:::-;;;;72122:327;;61624:435:::0;61687:20;61710:13;49840:10;:17;;49752:113;61710:13;61687:36;;61734:14;61751;:12;:14::i;:::-;61734:31;;61794:1;61785:6;:10;:25;;;;;61809:1;61799:6;:11;;61785:25;61776:36;;;;;;61844:15;61853:6;61844;:15;:::i;:::-;61831:9;:28;;61823:37;;;;;;61894:14;;61879:12;:29;61871:38;;;;;;61947:10;;61928:15;:29;;61920:38;;;;;;61983:9;61979:73;61999:6;61996:1;:9;61979:73;;;62026:14;:12;:14::i;:::-;62007:3;;;;:::i;:::-;;;;61979:73;;72802:100;73237:6;;-1:-1:-1;;;;;73237:6:0;73247:10;73237:20;73229:29;;;;;;72876:6:::1;:18:::0;;-1:-1:-1;;;;;;72876:18:0::1;-1:-1:-1::0;;;;;72876:18:0;;;::::1;::::0;;;::::1;::::0;;72802:100::o;3616:419::-;3749:13;3805:3;3775:21;3852;3863:10;3852:8;:21;:::i;:::-;3842:32;;;;;;-1:-1:-1;;;3842:32:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;3842:32:0;-1:-1:-1;3820:54:0;-1:-1:-1;3902:10:0;3885:111;3918:8;3914:1;:12;3885:111;;;3973:8;3982:1;3973:11;;;;;;-1:-1:-1;;;3973:11:0;;;;;;;;;;;;;-1:-1:-1;;;;;;3973:11:0;3948:6;3955:14;3959:10;3955:1;:14;:::i;:::-;3948:22;;;;;;-1:-1:-1;;;3948:22:0;;;;;;;;;;;;:36;-1:-1:-1;;;;;3948:36:0;;;;;;;;-1:-1:-1;3928:3:0;;;;:::i;:::-;;;;3885:111;;;-1:-1:-1;4020:6:0;-1:-1:-1;;3616:419:0;;;;;;:::o;36059:305::-;36161:4;-1:-1:-1;;;;;;36198:40:0;;-1:-1:-1;;;36198:40:0;;:105;;-1:-1:-1;;;;;;;36255:48:0;;-1:-1:-1;;;36255:48:0;36198:105;:158;;;-1:-1:-1;;;;;;;;;;28147:40:0;;;36320:36;28038:157;45939:174;46014:24;;;;:15;:24;;;;;:29;;-1:-1:-1;;;;;;46014:29:0;-1:-1:-1;;;;;46014:29:0;;;;;;;;:24;;46068:23;46014:24;46068:14;:23::i;:::-;-1:-1:-1;;;;;46059:46:0;;;;;;;;;;;45939:174;;:::o;61159:408::-;61203:20;61226:13;49840:10;:17;;49752:113;61226:13;61203:36;;61273:10;;61258:12;:25;61250:34;;;;;;61323:10;4366:20;4414:8;61295:40;;;;;;61370:12;61424:32;61370:12;61442:10;61348:19;61424:4;:32::i;:::-;61395:26;;;;:13;:26;;;;;;;;:61;;;;:26;;:61;;;;;;:::i;:::-;;61512:4;61469:12;61482:13;:26;61496:11;61482:26;;;;;;;;;;;61469:40;;;;;;:::i;:::-;;;;;;;;;;;;;;:47;;;;;-1:-1:-1;;61469:47:0;;;;;;;;;61529:30;61535:10;61547:11;61529:5;:30::i;42251:348::-;42344:4;42046:16;;;:7;:16;;;;;;-1:-1:-1;;;;;42046:16:0;42361:73;;;;-1:-1:-1;;;42361:73:0;;23929:2:1;42361:73:0;;;23911:21:1;23968:2;23948:18;;;23941:30;24007:34;23987:18;;;23980:62;-1:-1:-1;;;24058:18:1;;;24051:42;24110:19;;42361:73:0;23901:234:1;42361:73:0;42445:13;42461:23;42476:7;42461:14;:23::i;:::-;42445:39;;42514:5;-1:-1:-1;;;;;42503:16:0;:7;-1:-1:-1;;;;;42503:16:0;;:51;;;;42547:7;-1:-1:-1;;;;;42523:31:0;:20;42535:7;42523:11;:20::i;:::-;-1:-1:-1;;;;;42523:31:0;;42503:51;:87;;;-1:-1:-1;;;;;;39343:25:0;;;39319:4;39343:25;;;:18;:25;;;;;;;;:35;;;;;;;;;;;;42558:32;42495:96;42251:348;-1:-1:-1;;;;42251:348:0:o;45243:578::-;45402:4;-1:-1:-1;;;;;45375:31:0;:23;45390:7;45375:14;:23::i;:::-;-1:-1:-1;;;;;45375:31:0;;45367:85;;;;-1:-1:-1;;;45367:85:0;;26362:2:1;45367:85:0;;;26344:21:1;26401:2;26381:18;;;26374:30;26440:34;26420:18;;;26413:62;-1:-1:-1;;;26491:18:1;;;26484:39;26540:19;;45367:85:0;26334:231:1;45367:85:0;-1:-1:-1;;;;;45471:16:0;;45463:65;;;;-1:-1:-1;;;45463:65:0;;23170:2:1;45463:65:0;;;23152:21:1;23209:2;23189:18;;;23182:30;23248:34;23228:18;;;23221:62;-1:-1:-1;;;23299:18:1;;;23292:34;23343:19;;45463:65:0;23142:226:1;45463:65:0;45541:39;45562:4;45568:2;45572:7;45541:20;:39::i;:::-;45645:29;45662:1;45666:7;45645:8;:29::i;:::-;-1:-1:-1;;;;;45687:15:0;;;;;;:9;:15;;;;;:20;;45706:1;;45687:15;:20;;45706:1;;45687:20;:::i;:::-;;;;-1:-1:-1;;;;;;;45718:13:0;;;;;;:9;:13;;;;;:18;;45735:1;;45718:13;:18;;45735:1;;45718:18;:::i;:::-;;;;-1:-1:-1;;45747:16:0;;;;:7;:16;;;;;;:21;;-1:-1:-1;;;;;;45747:21:0;-1:-1:-1;;;;;45747:21:0;;;;;;;;;45786:27;;45747:16;;45786:27;;;;;;;45243:578;;;:::o;3105:503::-;3191:16;3254:2;3191:16;;3293:286;3315:7;:14;3311:1;:18;;;3293:286;;;3402:2;3386:7;3394:1;3386:10;;;;;;;;-1:-1:-1;;;3386:10:0;;;;;;;;;;;;;;;3374:30;;;;3373:85;;;3455:2;3439:7;3447:1;3439:10;;;;;;;;-1:-1:-1;;;3439:10:0;;;;;;;;;;;;;;;3427:30;;3373:85;3351:217;;;3493:10;3501:2;3493:10;;:::i;:::-;;;3550:2;3536:7;3544:1;3536:10;;;;;;;;-1:-1:-1;;;3536:10:0;;;;;;;;;;;;;3530:22;;;3536:10;;3530:22;:::i;:::-;3522:30;;;;:::i;:::-;;;3351:217;3331:3;;;;:::i;:::-;;;;3293:286;;63540:374;63642:5;;63665:223;63687:7;:14;63683:18;;;;63665:223;;;63835:12;63817:32;;;;;;;;:::i;:::-;;;;;;;;;;;;;63807:43;;;;;;63773:7;63781:1;63773:10;;;;;;;;-1:-1:-1;;;63773:10:0;;;;;;;;;;;;;;;;63755:30;;;;;;;;:::i;:::-;;;;;;;;;;;;;63745:41;;;;;;:105;63723:153;;;63874:1;63540:374;-1:-1:-1;;63540:374:0:o;63723:153::-;63703:3;;;;:::i;:::-;;;;63665:223;;;;63898:8;;;2565:532;2621:13;2651:10;2647:53;;-1:-1:-1;;2678:10:0;;;;;;;;;;;;-1:-1:-1;;;2678:10:0;;;;;2565:532::o;2647:53::-;2725:5;2710:12;2766:78;2773:9;;2766:78;;2799:8;;;;:::i;:::-;;-1:-1:-1;2822:10:0;;-1:-1:-1;2830:2:0;2822:10;;:::i;:::-;;;2766:78;;;2854:19;2886:6;2876:17;;;;;;-1:-1:-1;;;2876:17:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;2876:17:0;;2854:39;;2904:154;2911:10;;2904:154;;2938:11;2948:1;2938:11;;:::i;:::-;;-1:-1:-1;3007:10:0;3015:2;3007:5;:10;:::i;:::-;2994:24;;:2;:24;:::i;:::-;2981:39;;2964:6;2971;2964:14;;;;;;-1:-1:-1;;;2964:14:0;;;;;;;;;;;;:56;-1:-1:-1;;;;;2964:56:0;;;;;;;;-1:-1:-1;3035:11:0;3044:2;3035:11;;:::i;:::-;;;2904:154;;11395:98;11453:7;11480:5;11484:1;11480;:5;:::i;41329:315::-;41486:28;41496:4;41502:2;41506:7;41486:9;:28::i;:::-;41533:48;41556:4;41562:2;41566:7;41575:5;41533:22;:48::i;:::-;41525:111;;;;-1:-1:-1;;;41525:111:0;;;;;;;:::i;203:2256::-;261:13;291:4;:11;306:1;291:16;287:31;;;-1:-1:-1;;309:9:0;;;;;;;;;-1:-1:-1;309:9:0;;;203:2256::o;287:31::-;370:19;392:5;;;;;;;;;;;;;;;;;370:27;;449:18;495:1;476:4;:11;490:1;476:15;;;;:::i;:::-;475:21;;;;:::i;:::-;470:27;;:1;:27;:::i;:::-;449:48;-1:-1:-1;580:20:0;614:15;449:48;627:2;614:15;:::i;:::-;603:27;;;;;;-1:-1:-1;;;603:27:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;603:27:0;;580:50;;727:10;719:6;712:26;822:1;815:5;811:13;881:4;932;926:11;917:7;913:25;1028:2;1020:6;1016:15;1101:1045;1136:6;1127:7;1124:19;1101:1045;;;1206:1;1193:15;;;1274:14;;1457:4;1445:2;1441:14;;;1437:25;;1423:40;;1417:47;1412:3;1408:57;;;1347:137;;1648:2;1644:14;;;1640:25;;1626:40;;1620:47;1611:57;;1530:1;1515:17;;1550:137;1851:1;1847:13;;;1843:24;;1829:39;;1823:46;1814:56;;1718:17;;;1753:136;2045:16;;2031:31;;2025:38;2016:48;;1920:17;;;1955:128;;;;2114:17;;1101:1045;;;2219:1;2212:4;2206:11;2202:19;2240:1;2235:84;;;;2338:1;2333:82;;;;2195:220;;2235:84;-1:-1:-1;;;;;2268:17:0;;2261:43;2235:84;;2333:82;-1:-1:-1;;;;;2366:17:0;;2359:41;2195:220;-1:-1:-1;2445:6:0;;203:2256;-1:-1:-1;;;;;;;;203:2256:0:o;58783:1117::-;58885:13;58924:2;58919;:7;58911:16;;;;;;59085:31;;;;;;;;;;;;-1:-1:-1;;;59085:31:0;;;;:25;59129:662;59151:1;59147;:5;;;59129:662;;;59174:10;:12;;;:10;:12;;;:::i;:::-;;;;-1:-1:-1;;59551:10:0;;59312:276;;;59359:15;59312:276;;;18830:19:1;59405:16:0;18865:12:1;;;18858:28;;;;18902:12;;;;18895:28;;;18957:15;;;-1:-1:-1;;18953:53:1;18939:12;;;18932:75;19023:13;;;19016:29;;;19061:13;;;19054:29;;;;-1:-1:-1;;59633:5:0;;19099:13:1;;59312:276:0;;;;;;;;;;;;59276:335;;;;;;59246:384;;:392;;;;:::i;:::-;59201:452;;59726:11;59739:24;59749:10;59739:24;;59761:1;59739:9;:24::i;:::-;59709:55;;;;;;;;;:::i;:::-;;;;;;;;;;;;;59670:109;;59129:662;59154:3;;;;;:::i;:::-;;;;59129:662;;;;59807:12;59820:11;59807:25;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;59803:58;;;59841:20;59846:2;59850;59854:6;:2;59859:1;59854:6;:::i;:::-;59841:4;:20::i;:::-;59834:27;;;;;43935:382;-1:-1:-1;;;;;44015:16:0;;44007:61;;;;-1:-1:-1;;;44007:61:0;;25588:2:1;44007:61:0;;;25570:21:1;;;25607:18;;;25600:30;25666:34;25646:18;;;25639:62;25718:18;;44007:61:0;25560:182:1;44007:61:0;42022:4;42046:16;;;:7;:16;;;;;;-1:-1:-1;;;;;42046:16:0;:30;44079:58;;;;-1:-1:-1;;;44079:58:0;;22813:2:1;44079:58:0;;;22795:21:1;22852:2;22832:18;;;22825:30;22891;22871:18;;;22864:58;22939:18;;44079:58:0;22785:178:1;44079:58:0;44150:45;44179:1;44183:2;44187:7;44150:20;:45::i;:::-;-1:-1:-1;;;;;44208:13:0;;;;;;:9;:13;;;;;:18;;44225:1;;44208:13;:18;;44225:1;;44208:18;:::i;:::-;;;;-1:-1:-1;;44237:16:0;;;;:7;:16;;;;;;:21;;-1:-1:-1;;;;;;44237:21:0;-1:-1:-1;;;;;44237:21:0;;;;;;;;44276:33;;44237:16;;;44276:33;;44237:16;;44276:33;43935:382;;:::o;50788:589::-;-1:-1:-1;;;;;50994:18:0;;50990:187;;51029:40;51061:7;52204:10;:17;;52177:24;;;;:15;:24;;;;;:44;;;52232:24;;;;;;;;;;;;52100:164;51029:40;50990:187;;;51099:2;-1:-1:-1;;;;;51091:10:0;:4;-1:-1:-1;;;;;51091:10:0;;51087:90;;51118:47;51151:4;51157:7;51118:32;:47::i;:::-;-1:-1:-1;;;;;51191:16:0;;51187:183;;51224:45;51261:7;51224:36;:45::i;51187:183::-;51297:4;-1:-1:-1;;;;;51291:10:0;:2;-1:-1:-1;;;;;51291:10:0;;51287:83;;51318:40;51346:2;51350:7;51318:27;:40::i;46678:799::-;46833:4;-1:-1:-1;;;;;46854:13:0;;4366:20;4414:8;46850:620;;46890:72;;-1:-1:-1;;;46890:72:0;;-1:-1:-1;;;;;46890:36:0;;;;;:72;;17198:10;;46941:4;;46947:7;;46956:5;;46890:72;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;46890:72:0;;;;;;;;-1:-1:-1;;46890:72:0;;;;;;;;;;;;:::i;:::-;;;46886:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;47132:13:0;;47128:272;;47175:60;;-1:-1:-1;;;47175:60:0;;;;;;;:::i;47128:272::-;47350:6;47344:13;47335:6;47331:2;47327:15;47320:38;46886:529;-1:-1:-1;;;;;;47013:51:0;-1:-1:-1;;;47013:51:0;;-1:-1:-1;47006:58:0;;46850:620;-1:-1:-1;47454:4:0;46678:799;;;;;;:::o;57923:572::-;58031:13;58062:24;58106:7;58101:366;58123:5;58129:11;58123:18;;;;;;;-1:-1:-1;;;58123:18:0;;;;;;;;;;:25;58119:29;;;;58101:366;;;58170:21;58194:5;58200:11;58194:18;;;;;;;-1:-1:-1;;;58194:18:0;;;;;;;;;;58213:1;58194:21;;;;;;;;-1:-1:-1;;;58194:21:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;58170:45;;58266:17;58252:31;;:10;:31;;:99;;;;-1:-1:-1;58317:34:0;58337:14;58317:17;:34;:::i;:::-;58304:47;;:10;:47;58252:99;58230:156;;;58374:12;:1;:10;;;:12::i;:::-;58367:19;;;;;;;58230:156;58421:34;58441:14;58421:17;:34;:::i;:::-;58401:54;;58101:366;58150:3;;;;;:::i;:::-;;;;58101:366;;52891:988;53157:22;53207:1;53182:22;53199:4;53182:16;:22::i;:::-;:26;;;;:::i;:::-;53219:18;53240:26;;;:17;:26;;;;;;53157:51;;-1:-1:-1;53373:28:0;;;53369:328;;-1:-1:-1;;;;;53440:18:0;;53418:19;53440:18;;;:12;:18;;;;;;;;:34;;;;;;;;;53491:30;;;;;;:44;;;53608:30;;:17;:30;;;;;:43;;;53369:328;-1:-1:-1;53793:26:0;;;;:17;:26;;;;;;;;53786:33;;;-1:-1:-1;;;;;53837:18:0;;;;;:12;:18;;;;;:34;;;;;;;53830:41;52891:988::o;54174:1079::-;54452:10;:17;54427:22;;54452:21;;54472:1;;54452:21;:::i;:::-;54484:18;54505:24;;;:15;:24;;;;;;54878:10;:26;;54427:46;;-1:-1:-1;54505:24:0;;54427:46;;54878:26;;;;-1:-1:-1;;;54878:26:0;;;;;;;;;;;;;;;;;54856:48;;54942:11;54917:10;54928;54917:22;;;;;;-1:-1:-1;;;54917:22:0;;;;;;;;;;;;;;;;;;;;:36;;;;55022:28;;;:15;:28;;;;;;;:41;;;55194:24;;;;;55187:31;55229:10;:16;;;;;-1:-1:-1;;;55229:16:0;;;;;;;;;;;;;;;;;;;;;;;;;;54174:1079;;;;:::o;51678:221::-;51763:14;51780:20;51797:2;51780:16;:20::i;:::-;-1:-1:-1;;;;;51811:16:0;;;;;;;:12;:16;;;;;;;;:24;;;;;;;;:34;;;51856:26;;;:17;:26;;;;;;:35;;;;-1:-1:-1;51678:221:0:o;-1:-1:-1:-;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;:::i;14:406:1:-;78:5;112:18;104:6;101:30;98:2;;;134:18;;:::i;:::-;172:57;217:2;196:15;;-1:-1:-1;;192:29:1;223:4;188:40;172:57;:::i;:::-;163:66;;252:6;245:5;238:21;292:3;283:6;278:3;274:16;271:25;268:2;;;309:1;306;299:12;268:2;358:6;353:3;346:4;339:5;335:16;322:43;412:1;405:4;396:6;389:5;385:18;381:29;374:40;88:332;;;;;:::o;425:173::-;493:20;;-1:-1:-1;;;;;542:31:1;;532:42;;522:2;;588:1;585;578:12;522:2;474:124;;;:::o;603:229::-;646:5;699:3;692:4;684:6;680:17;676:27;666:2;;721:5;714;707:20;666:2;747:79;822:3;813:6;800:20;793:4;785:6;781:17;747:79;:::i;837:803::-;889:5;937:4;925:9;920:3;916:19;912:30;909:2;;;959:5;952;945:20;909:2;985:22;;:::i;:::-;976:31;;1043:9;1030:23;1072:18;1113:2;1105:6;1102:14;1099:2;;;1129:1;1126;1119:12;1099:2;1156:46;1198:3;1189:6;1178:9;1174:22;1156:46;:::i;:::-;1149:5;1142:61;1256:2;1245:9;1241:18;1228:32;1212:48;;1285:2;1275:8;1272:16;1269:2;;;1301:1;1298;1291:12;1269:2;1337:48;1381:3;1370:8;1359:9;1355:24;1337:48;:::i;:::-;1332:2;1325:5;1321:14;1314:72;1439:2;1428:9;1424:18;1411:32;1395:48;;1468:2;1458:8;1455:16;1452:2;;;1484:1;1481;1474:12;1452:2;;1520:48;1564:3;1553:8;1542:9;1538:24;1520:48;:::i;:::-;1515:2;1508:5;1504:14;1497:72;;1629:2;1618:9;1614:18;1601:32;1596:2;1589:5;1585:14;1578:56;899:741;;;;:::o;1645:196::-;1704:6;1757:2;1745:9;1736:7;1732:23;1728:32;1725:2;;;1778:6;1770;1763:22;1725:2;1806:29;1825:9;1806:29;:::i;1846:270::-;1914:6;1922;1975:2;1963:9;1954:7;1950:23;1946:32;1943:2;;;1996:6;1988;1981:22;1943:2;2024:29;2043:9;2024:29;:::i;:::-;2014:39;;2072:38;2106:2;2095:9;2091:18;2072:38;:::i;:::-;2062:48;;1933:183;;;;;:::o;2121:338::-;2198:6;2206;2214;2267:2;2255:9;2246:7;2242:23;2238:32;2235:2;;;2288:6;2280;2273:22;2235:2;2316:29;2335:9;2316:29;:::i;:::-;2306:39;;2364:38;2398:2;2387:9;2383:18;2364:38;:::i;:::-;2354:48;;2449:2;2438:9;2434:18;2421:32;2411:42;;2225:234;;;;;:::o;2464:696::-;2559:6;2567;2575;2583;2636:3;2624:9;2615:7;2611:23;2607:33;2604:2;;;2658:6;2650;2643:22;2604:2;2686:29;2705:9;2686:29;:::i;:::-;2676:39;;2734:38;2768:2;2757:9;2753:18;2734:38;:::i;:::-;2724:48;;2819:2;2808:9;2804:18;2791:32;2781:42;;2874:2;2863:9;2859:18;2846:32;2901:18;2893:6;2890:30;2887:2;;;2938:6;2930;2923:22;2887:2;2966:22;;3019:4;3011:13;;3007:27;-1:-1:-1;2997:2:1;;3053:6;3045;3038:22;2997:2;3081:73;3146:7;3141:2;3128:16;3123:2;3119;3115:11;3081:73;:::i;:::-;3071:83;;;2594:566;;;;;;;:::o;3165:367::-;3230:6;3238;3291:2;3279:9;3270:7;3266:23;3262:32;3259:2;;;3312:6;3304;3297:22;3259:2;3340:29;3359:9;3340:29;:::i;:::-;3330:39;;3419:2;3408:9;3404:18;3391:32;3466:5;3459:13;3452:21;3445:5;3442:32;3432:2;;3493:6;3485;3478:22;3432:2;3521:5;3511:15;;;3249:283;;;;;:::o;3537:264::-;3605:6;3613;3666:2;3654:9;3645:7;3641:23;3637:32;3634:2;;;3687:6;3679;3672:22;3634:2;3715:29;3734:9;3715:29;:::i;:::-;3705:39;3791:2;3776:18;;;;3763:32;;-1:-1:-1;;;3624:177:1:o;3806:255::-;3864:6;3917:2;3905:9;3896:7;3892:23;3888:32;3885:2;;;3938:6;3930;3923:22;3885:2;3982:9;3969:23;4001:30;4025:5;4001:30;:::i;4066:259::-;4135:6;4188:2;4176:9;4167:7;4163:23;4159:32;4156:2;;;4209:6;4201;4194:22;4156:2;4246:9;4240:16;4265:30;4289:5;4265:30;:::i;4330:342::-;4399:6;4452:2;4440:9;4431:7;4427:23;4423:32;4420:2;;;4473:6;4465;4458:22;4420:2;4518:9;4505:23;4551:18;4543:6;4540:30;4537:2;;;4588:6;4580;4573:22;4537:2;4616:50;4658:7;4649:6;4638:9;4634:22;4616:50;:::i;4677:190::-;4736:6;4789:2;4777:9;4768:7;4764:23;4760:32;4757:2;;;4810:6;4802;4795:22;4757:2;-1:-1:-1;4838:23:1;;4747:120;-1:-1:-1;4747:120:1:o;4872:1250::-;4988:6;4996;5049:2;5037:9;5028:7;5024:23;5020:32;5017:2;;;5070:6;5062;5055:22;5017:2;5111:9;5098:23;5088:33;;5140:2;5193;5182:9;5178:18;5165:32;5216:18;5257:2;5249:6;5246:14;5243:2;;;5278:6;5270;5263:22;5243:2;5321:6;5310:9;5306:22;5296:32;;5366:7;5359:4;5355:2;5351:13;5347:27;5337:2;;5393:6;5385;5378:22;5337:2;5434;5421:16;5456:2;5452;5449:10;5446:2;;;5462:18;;:::i;:::-;5508:2;5505:1;5501:10;5531:28;5555:2;5551;5547:11;5531:28;:::i;:::-;5593:15;;;5624:12;;;;5656:11;;;5686;;;5682:20;;5679:33;-1:-1:-1;5676:2:1;;;5730:6;5722;5715:22;5676:2;5757:6;5748:15;;5772:320;5786:2;5783:1;5780:9;5772:320;;;5863:3;5850:17;5899:2;5886:11;5883:19;5880:2;;;5920:6;5912;5905:22;5880:2;5954:63;6009:7;6004:2;5990:11;5986:2;5982:20;5978:29;5954:63;:::i;:::-;5942:76;;-1:-1:-1;5804:1:1;5797:9;;;;;6038:12;;;;6070;;5772:320;;;5776:3;6111:5;6101:15;;;;;;;;;5007:1115;;;;;:::o;6127:258::-;6195:6;6203;6256:2;6244:9;6235:7;6231:23;6227:32;6224:2;;;6277:6;6269;6262:22;6224:2;-1:-1:-1;;6305:23:1;;;6375:2;6360:18;;;6347:32;;-1:-1:-1;6214:171:1:o;6390:257::-;6431:3;6469:5;6463:12;6496:6;6491:3;6484:19;6512:63;6568:6;6561:4;6556:3;6552:14;6545:4;6538:5;6534:16;6512:63;:::i;:::-;6629:2;6608:15;-1:-1:-1;;6604:29:1;6595:39;;;;6636:4;6591:50;;6439:208;-1:-1:-1;;6439:208:1:o;6652:185::-;6694:3;6732:5;6726:12;6747:52;6792:6;6787:3;6780:4;6773:5;6769:16;6747:52;:::i;:::-;6815:16;;;;;6702:135;-1:-1:-1;;6702:135:1:o;6842:979::-;6927:12;;6892:3;;6984:1;7004:18;;;;7057;;;;7084:2;;7138:4;7130:6;7126:17;7116:27;;7084:2;7164;7212;7204:6;7201:14;7181:18;7178:38;7175:2;;;-1:-1:-1;;;7239:33:1;;7295:4;7292:1;7285:15;7325:4;7246:3;7313:17;7175:2;7356:18;7383:104;;;;7501:1;7496:319;;;;7349:466;;7383:104;-1:-1:-1;;7416:24:1;;7404:37;;7461:16;;;;-1:-1:-1;7383:104:1;;7496:319;28570:4;28589:17;;;28639:4;28623:21;;7590:1;7604:165;7618:6;7615:1;7612:13;7604:165;;;7696:14;;7683:11;;;7676:35;7739:16;;;;7633:10;;7604:165;;;7608:3;;7798:6;7793:3;7789:16;7782:23;;7349:466;;;;;;;6900:921;;;;:::o;7945:276::-;8076:3;8114:6;8108:13;8130:53;8176:6;8171:3;8164:4;8156:6;8152:17;8130:53;:::i;:::-;8199:16;;;;;8084:137;-1:-1:-1;;8084:137:1:o;8226:470::-;8405:3;8443:6;8437:13;8459:53;8505:6;8500:3;8493:4;8485:6;8481:17;8459:53;:::i;:::-;8575:13;;8534:16;;;;8597:57;8575:13;8534:16;8631:4;8619:17;;8597:57;:::i;:::-;8670:20;;8413:283;-1:-1:-1;;;;8413:283:1:o;8701:439::-;8933:3;8971:6;8965:13;8987:53;9033:6;9028:3;9021:4;9013:6;9009:17;8987:53;:::i;:::-;-1:-1:-1;;;9062:16:1;;9087:18;;;-1:-1:-1;9132:1:1;9121:13;;8941:199;-1:-1:-1;8941:199:1:o;9145:1459::-;9824:3;9862:6;9856:13;9878:53;9924:6;9919:3;9912:4;9904:6;9900:17;9878:53;:::i;:::-;-1:-1:-1;;;9953:16:1;;;9978:31;;;10034:13;;10056:66;10034:13;10108:2;10097:14;;10090:4;10078:17;;10056:66;:::i;:::-;-1:-1:-1;;;10185:2:1;10141:20;;;;10177:11;;;10170:28;10223:13;;10245:63;10223:13;10294:2;10286:11;;10279:4;10267:17;;10245:63;:::i;:::-;-1:-1:-1;;;10368:2:1;10327:17;;;;10360:11;;;10353:28;10406:13;;10428:63;10406:13;10477:2;10469:11;;10462:4;10450:17;;10428:63;:::i;:::-;-1:-1:-1;;;10552:11:1;10514:17;;;;10552:11;;;10545:26;10587:11;;;9832:772;-1:-1:-1;;;;;;9832:772:1:o;10609:990::-;11133:3;11171:6;11165:13;11187:53;11233:6;11228:3;11221:4;11213:6;11209:17;11187:53;:::i;:::-;-1:-1:-1;;;11262:16:1;;;11287:57;;;11363:49;11408:2;11397:14;;11389:6;11363:49;:::i;:::-;-1:-1:-1;;;11421:46:1;;11353:59;-1:-1:-1;11486:46:1;11528:2;11520:11;;11512:6;11486:46;:::i;:::-;-1:-1:-1;;;11541:26:1;;11591:1;11583:10;;11141:458;-1:-1:-1;;;;;;11141:458:1:o;11604:197::-;11732:3;11757:38;11791:3;11783:6;11757:38;:::i;11806:3643::-;12169:66;12164:3;12157:79;12266:66;12261:2;12256:3;12252:12;12245:88;12363:66;12358:2;12353:3;12349:12;12342:88;12460:66;12455:2;12450:3;12446:12;12439:88;12567:32;12562:3;12558:42;12552:3;12547;12543:13;12536:65;12139:3;12630:6;12624:13;12646:61;12700:6;12694:3;12689;12685:13;12680:2;12672:6;12668:15;12646:61;:::i;:::-;12772:34;12766:3;12726:16;;;;12758:12;;;12751:56;-1:-1:-1;12837:34:1;12831:3;12823:12;;12816:56;12902:34;12896:3;12888:12;;12881:56;12967:34;12961:3;12953:12;;12946:56;13032:34;13026:3;13018:12;;13011:56;13097:34;13091:3;13083:12;;13076:56;13162:34;13156:3;13148:12;;13141:56;13227:34;13221:3;13213:12;;13206:56;13292:34;13286:3;13278:12;;13271:56;13357:34;13351:3;13343:12;;13336:56;13422:34;13416:3;13408:12;;13401:56;13487:34;13481:3;13473:12;;13466:56;13552:34;13546:3;13538:12;;13531:56;13617:34;13611:3;13603:12;;13596:56;13682:34;13676:3;13668:12;;13661:56;13747:34;13741:3;13733:12;;13726:56;13812:34;13806:3;13798:12;;13791:56;13877:34;13871:3;13863:12;;13856:56;13942:34;13936:3;13928:12;;13921:56;14007:34;14001:3;13993:12;;13986:56;14072:34;14066:3;14058:12;;14051:56;14137:34;14131:3;14123:12;;14116:56;14202:34;14196:3;14188:12;;14181:56;14267:34;14261:3;14253:12;;14246:56;14332:34;14326:3;14318:12;;14311:56;14397:34;14391:3;14383:12;;14376:56;14462:34;14456:3;14448:12;;14441:56;14528:34;14521:4;14513:13;;14506:57;14594:34;14587:4;14579:13;;14572:57;14660:34;14653:4;14645:13;;14638:57;14726:34;14719:4;14711:13;;14704:57;14792:34;14785:4;14777:13;;14770:57;14858:34;14851:4;14843:13;;14836:57;14924:34;14917:4;14909:13;;14902:57;14990:34;14983:4;14975:13;;14968:57;15056:34;15049:4;15041:13;;15034:57;15122:34;15115:4;15107:13;;15100:57;15188:34;15181:4;15173:13;;15166:57;15254:34;15247:4;15239:13;;15232:57;15320:34;15313:4;15305:13;;15298:57;15386:27;15379:4;15371:13;;15364:50;15438:4;15430:13;;12147:3302;-1:-1:-1;12147:3302:1:o;15454:572::-;-1:-1:-1;;;15812:3:1;15805:16;15787:3;15850:6;15844:13;15866:61;15920:6;15916:1;15911:3;15907:11;15900:4;15892:6;15888:17;15866:61;:::i;:::-;-1:-1:-1;;;15986:1:1;15946:16;;;;15978:10;;;15971:23;-1:-1:-1;16018:1:1;16010:10;;15795:231;-1:-1:-1;15795:231:1:o;16031:448::-;16293:31;16288:3;16281:44;16263:3;16354:6;16348:13;16370:62;16425:6;16420:2;16415:3;16411:12;16404:4;16396:6;16392:17;16370:62;:::i;:::-;16452:16;;;;16470:2;16448:25;;16271:208;-1:-1:-1;;16271:208:1:o;16484:1649::-;-1:-1:-1;;;17133:63:1;;17219:13;;17115:3;;17241:62;17219:13;17291:2;17282:12;;17275:4;17263:17;;17241:62;:::i;:::-;17367:66;17362:2;17322:16;;;17354:11;;;17347:87;17463:34;17458:2;17450:11;;17443:55;17527:34;17522:2;17514:11;;17507:55;17592:34;17586:3;17578:12;;17571:56;17657:66;17651:3;17643:12;;17636:88;17754:66;17748:3;17740:12;;17733:88;17846:13;;17868:64;17846:13;17917:3;17909:12;;17902:4;17890:17;;17868:64;:::i;:::-;-1:-1:-1;;;17992:3:1;17951:17;;;;17984:12;;;17977:64;18057:70;18087:39;18121:3;18113:12;;18105:6;18087:39;:::i;:::-;-1:-1:-1;;;7891:16:1;;7932:1;7923:11;;7881:59;18057:70;18050:77;17123:1010;-1:-1:-1;;;;;;17123:1010:1:o;18138:418::-;-1:-1:-1;;;18395:3:1;18388:16;18370:3;18433:6;18427:13;18449:61;18503:6;18499:1;18494:3;18490:11;18483:4;18475:6;18471:17;18449:61;:::i;:::-;18530:16;;;;18548:1;18526:24;;18378:178;-1:-1:-1;;18378:178:1:o;19331:488::-;-1:-1:-1;;;;;19600:15:1;;;19582:34;;19652:15;;19647:2;19632:18;;19625:43;19699:2;19684:18;;19677:34;;;19747:3;19742:2;19727:18;;19720:31;;;19525:4;;19768:45;;19793:19;;19785:6;19768:45;:::i;20103:635::-;20274:2;20326:21;;;20396:13;;20299:18;;;20418:22;;;20245:4;;20274:2;20497:15;;;;20471:2;20456:18;;;20245:4;20543:169;20557:6;20554:1;20551:13;20543:169;;;20618:13;;20606:26;;20687:15;;;;20652:12;;;;20579:1;20572:9;20543:169;;;-1:-1:-1;20729:3:1;;20254:484;-1:-1:-1;;;;;;20254:484:1:o;20935:219::-;21084:2;21073:9;21066:21;21047:4;21104:44;21144:2;21133:9;21129:18;21121:6;21104:44;:::i;21159:616::-;21432:3;21421:9;21414:22;21395:4;21459:45;21499:3;21488:9;21484:19;21476:6;21459:45;:::i;:::-;21552:9;21544:6;21540:22;21535:2;21524:9;21520:18;21513:50;21586:32;21611:6;21603;21586:32;:::i;:::-;21572:46;;21666:9;21658:6;21654:22;21649:2;21638:9;21634:18;21627:50;21694:32;21719:6;21711;21694:32;:::i;:::-;21686:40;;;21762:6;21757:2;21746:9;21742:18;21735:34;21404:371;;;;;;;:::o;22192:414::-;22394:2;22376:21;;;22433:2;22413:18;;;22406:30;22472:34;22467:2;22452:18;;22445:62;-1:-1:-1;;;22538:2:1;22523:18;;22516:48;22596:3;22581:19;;22366:240::o;26972:413::-;27174:2;27156:21;;;27213:2;27193:18;;;27186:30;27252:34;27247:2;27232:18;;27225:62;-1:-1:-1;;;27318:2:1;27303:18;;27296:47;27375:3;27360:19;;27146:239::o;27985:253::-;28057:2;28051:9;28099:4;28087:17;;28134:18;28119:34;;28155:22;;;28116:62;28113:2;;;28181:18;;:::i;:::-;28217:2;28210:22;28031:207;:::o;28243:275::-;28314:2;28308:9;28379:2;28360:13;;-1:-1:-1;;28356:27:1;28344:40;;28414:18;28399:34;;28435:22;;;28396:62;28393:2;;;28461:18;;:::i;:::-;28497:2;28490:22;28288:230;;-1:-1:-1;28288:230:1:o;28655:224::-;28694:3;28722:6;28755:2;28752:1;28748:10;28785:2;28782:1;28778:10;28816:3;28812:2;28808:12;28803:3;28800:21;28797:2;;;28824:18;;:::i;28884:128::-;28924:3;28955:1;28951:6;28948:1;28945:13;28942:2;;;28961:18;;:::i;:::-;-1:-1:-1;28997:9:1;;28932:80::o;29017:204::-;29055:3;29091:4;29088:1;29084:12;29123:4;29120:1;29116:12;29158:3;29152:4;29148:14;29143:3;29140:23;29137:2;;;29166:18;;:::i;:::-;29202:13;;29063:158;-1:-1:-1;;;29063:158:1:o;29226:120::-;29266:1;29292;29282:2;;29297:18;;:::i;:::-;-1:-1:-1;29331:9:1;;29272:74::o;29351:258::-;29390:7;29422:6;29455:2;29452:1;29448:10;29485:2;29482:1;29478:10;29541:3;29537:2;29533:12;29528:3;29525:21;29518:3;29511:11;29504:19;29500:47;29497:2;;;29550:18;;:::i;:::-;29590:13;;29402:207;-1:-1:-1;;;;29402:207:1:o;29614:168::-;29654:7;29720:1;29716;29712:6;29708:14;29705:1;29702:21;29697:1;29690:9;29683:17;29679:45;29676:2;;;29727:18;;:::i;:::-;-1:-1:-1;29767:9:1;;29666:116::o;29787:238::-;29825:7;29865:4;29862:1;29858:12;29897:4;29894:1;29890:12;29957:3;29951:4;29947:14;29942:3;29939:23;29932:3;29925:11;29918:19;29914:49;29911:2;;;29966:18;;:::i;:::-;30006:13;;29837:188;-1:-1:-1;;;29837:188:1:o;30030:125::-;30070:4;30098:1;30095;30092:8;30089:2;;;30103:18;;:::i;:::-;-1:-1:-1;30140:9:1;;30079:76::o;30160:195::-;30198:4;30235;30232:1;30228:12;30267:4;30264:1;30260:12;30292:3;30287;30284:12;30281:2;;;30299:18;;:::i;:::-;30336:13;;;30207:148;-1:-1:-1;;;30207:148:1:o;30360:258::-;30432:1;30442:113;30456:6;30453:1;30450:13;30442:113;;;30532:11;;;30526:18;30513:11;;;30506:39;30478:2;30471:10;30442:113;;;30573:6;30570:1;30567:13;30564:2;;;-1:-1:-1;;30608:1:1;30590:16;;30583:27;30413:205::o;30623:380::-;30702:1;30698:12;;;;30745;;;30766:2;;30820:4;30812:6;30808:17;30798:27;;30766:2;30873;30865:6;30862:14;30842:18;30839:38;30836:2;;;30919:10;30914:3;30910:20;30907:1;30900:31;30954:4;30951:1;30944:15;30982:4;30979:1;30972:15;30836:2;;30678:325;;;:::o;31008:197::-;31046:3;31074:6;31115:2;31108:5;31104:14;31142:2;31133:7;31130:15;31127:2;;;31148:18;;:::i;:::-;31197:1;31184:15;;31054:151;-1:-1:-1;;;31054:151:1:o;31210:135::-;31249:3;-1:-1:-1;;31270:17:1;;31267:2;;;31290:18;;:::i;:::-;-1:-1:-1;31337:1:1;31326:13;;31257:88::o;31350:175::-;31387:3;31431:4;31424:5;31420:16;31460:4;31451:7;31448:17;31445:2;;;31468:18;;:::i;:::-;31517:1;31504:15;;31395:130;-1:-1:-1;;31395:130:1:o;31530:112::-;31562:1;31588;31578:2;;31593:18;;:::i;:::-;-1:-1:-1;31627:9:1;;31568:74::o;31647:127::-;31708:10;31703:3;31699:20;31696:1;31689:31;31739:4;31736:1;31729:15;31763:4;31760:1;31753:15;31779:127;31840:10;31835:3;31831:20;31828:1;31821:31;31871:4;31868:1;31861:15;31895:4;31892:1;31885:15;31911:127;31972:10;31967:3;31963:20;31960:1;31953:31;32003:4;32000:1;31993:15;32027:4;32024:1;32017:15;32043:131;-1:-1:-1;;;;;;32117:32:1;;32107:43;;32097:2;;32164:1;32161;32154:12

Swarm Source

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