ETH Price: $3,272.94 (-4.18%)
Gas: 7 Gwei

Token

PepeWojak (PEPEWOJAK)
 

Overview

Max Total Supply

420,000,000,000,000 PEPEWOJAK

Holders

47

Market

Onchain Market Cap

$0.00

Circulating Supply Market Cap

-

Other Info

Token Contract (WITH 18 Decimals)

Balance
420,000,067,200 PEPEWOJAK

Value
$0.00
0x50b0c870074a0175fff870792df512c93850d986
Loading...
Loading
Loading...
Loading
Loading...
Loading

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

Contract Source Code Verified (Exact Match)

Contract Name:
PepeWojak

Compiler Version
v0.8.17+commit.8df45f5f

Optimization Enabled:
No with 200 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2023-05-16
*/

// SPDX-License-Identifier: MIT
// File: @openzeppelin/contracts/utils/Counters.sol


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

pragma solidity ^0.8.0;

/**
 * @title Counters
 * @author Matt Condon (@shrugs)
 * @dev Provides counters that can only be incremented, decremented or reset. This can be used e.g. to track the number
 * of elements in a mapping, issuing ERC721 ids, or counting request ids.
 *
 * Include with `using Counters for Counters.Counter;`
 */
library Counters {
    struct Counter {
        // This variable should never be directly accessed by users of the library: interactions must be restricted to
        // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add
        // this feature: see https://github.com/ethereum/solidity/issues/4637
        uint256 _value; // default: 0
    }

    function current(Counter storage counter) internal view returns (uint256) {
        return counter._value;
    }

    function increment(Counter storage counter) internal {
        unchecked {
            counter._value += 1;
        }
    }

    function decrement(Counter storage counter) internal {
        uint256 value = counter._value;
        require(value > 0, "Counter: decrement overflow");
        unchecked {
            counter._value = value - 1;
        }
    }

    function reset(Counter storage counter) internal {
        counter._value = 0;
    }
}

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


// OpenZeppelin Contracts (last updated v4.8.0) (utils/math/Math.sol)

pragma solidity ^0.8.0;

/**
 * @dev Standard math utilities missing in the Solidity language.
 */
library Math {
    enum Rounding {
        Down, // Toward negative infinity
        Up, // Toward infinity
        Zero // Toward zero
    }

    /**
     * @dev Returns the largest of two numbers.
     */
    function max(uint256 a, uint256 b) internal pure returns (uint256) {
        return a > b ? a : b;
    }

    /**
     * @dev Returns the smallest of two numbers.
     */
    function min(uint256 a, uint256 b) internal pure returns (uint256) {
        return a < b ? a : b;
    }

    /**
     * @dev Returns the average of two numbers. The result is rounded towards
     * zero.
     */
    function average(uint256 a, uint256 b) internal pure returns (uint256) {
        // (a + b) / 2 can overflow.
        return (a & b) + (a ^ b) / 2;
    }

    /**
     * @dev Returns the ceiling of the division of two numbers.
     *
     * This differs from standard division with `/` in that it rounds up instead
     * of rounding down.
     */
    function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) {
        // (a + b - 1) / b can overflow on addition, so we distribute.
        return a == 0 ? 0 : (a - 1) / b + 1;
    }

    /**
     * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0
     * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv)
     * with further edits by Uniswap Labs also under MIT license.
     */
    function mulDiv(
        uint256 x,
        uint256 y,
        uint256 denominator
    ) internal pure returns (uint256 result) {
        unchecked {
            // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use
            // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256
            // variables such that product = prod1 * 2^256 + prod0.
            uint256 prod0; // Least significant 256 bits of the product
            uint256 prod1; // Most significant 256 bits of the product
            assembly {
                let mm := mulmod(x, y, not(0))
                prod0 := mul(x, y)
                prod1 := sub(sub(mm, prod0), lt(mm, prod0))
            }

            // Handle non-overflow cases, 256 by 256 division.
            if (prod1 == 0) {
                return prod0 / denominator;
            }

            // Make sure the result is less than 2^256. Also prevents denominator == 0.
            require(denominator > prod1);

            ///////////////////////////////////////////////
            // 512 by 256 division.
            ///////////////////////////////////////////////

            // Make division exact by subtracting the remainder from [prod1 prod0].
            uint256 remainder;
            assembly {
                // Compute remainder using mulmod.
                remainder := mulmod(x, y, denominator)

                // Subtract 256 bit number from 512 bit number.
                prod1 := sub(prod1, gt(remainder, prod0))
                prod0 := sub(prod0, remainder)
            }

            // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1.
            // See https://cs.stackexchange.com/q/138556/92363.

            // Does not overflow because the denominator cannot be zero at this stage in the function.
            uint256 twos = denominator & (~denominator + 1);
            assembly {
                // Divide denominator by twos.
                denominator := div(denominator, twos)

                // Divide [prod1 prod0] by twos.
                prod0 := div(prod0, twos)

                // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one.
                twos := add(div(sub(0, twos), twos), 1)
            }

            // Shift in bits from prod1 into prod0.
            prod0 |= prod1 * twos;

            // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such
            // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for
            // four bits. That is, denominator * inv = 1 mod 2^4.
            uint256 inverse = (3 * denominator) ^ 2;

            // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works
            // in modular arithmetic, doubling the correct bits in each step.
            inverse *= 2 - denominator * inverse; // inverse mod 2^8
            inverse *= 2 - denominator * inverse; // inverse mod 2^16
            inverse *= 2 - denominator * inverse; // inverse mod 2^32
            inverse *= 2 - denominator * inverse; // inverse mod 2^64
            inverse *= 2 - denominator * inverse; // inverse mod 2^128
            inverse *= 2 - denominator * inverse; // inverse mod 2^256

            // Because the division is now exact we can divide by multiplying with the modular inverse of denominator.
            // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is
            // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1
            // is no longer required.
            result = prod0 * inverse;
            return result;
        }
    }

    /**
     * @notice Calculates x * y / denominator with full precision, following the selected rounding direction.
     */
    function mulDiv(
        uint256 x,
        uint256 y,
        uint256 denominator,
        Rounding rounding
    ) internal pure returns (uint256) {
        uint256 result = mulDiv(x, y, denominator);
        if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) {
            result += 1;
        }
        return result;
    }

    /**
     * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down.
     *
     * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11).
     */
    function sqrt(uint256 a) internal pure returns (uint256) {
        if (a == 0) {
            return 0;
        }

        // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target.
        //
        // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have
        // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`.
        //
        // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)`
        // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))`
        // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)`
        //
        // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit.
        uint256 result = 1 << (log2(a) >> 1);

        // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128,
        // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at
        // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision
        // into the expected uint128 result.
        unchecked {
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            result = (result + a / result) >> 1;
            return min(result, a / result);
        }
    }

    /**
     * @notice Calculates sqrt(a), following the selected rounding direction.
     */
    function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = sqrt(a);
            return result + (rounding == Rounding.Up && result * result < a ? 1 : 0);
        }
    }

    /**
     * @dev Return the log in base 2, rounded down, of a positive value.
     * Returns 0 if given 0.
     */
    function log2(uint256 value) internal pure returns (uint256) {
        uint256 result = 0;
        unchecked {
            if (value >> 128 > 0) {
                value >>= 128;
                result += 128;
            }
            if (value >> 64 > 0) {
                value >>= 64;
                result += 64;
            }
            if (value >> 32 > 0) {
                value >>= 32;
                result += 32;
            }
            if (value >> 16 > 0) {
                value >>= 16;
                result += 16;
            }
            if (value >> 8 > 0) {
                value >>= 8;
                result += 8;
            }
            if (value >> 4 > 0) {
                value >>= 4;
                result += 4;
            }
            if (value >> 2 > 0) {
                value >>= 2;
                result += 2;
            }
            if (value >> 1 > 0) {
                result += 1;
            }
        }
        return result;
    }

    /**
     * @dev Return the log in base 2, following the selected rounding direction, of a positive value.
     * Returns 0 if given 0.
     */
    function log2(uint256 value, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = log2(value);
            return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0);
        }
    }

    /**
     * @dev Return the log in base 10, rounded down, of a positive value.
     * Returns 0 if given 0.
     */
    function log10(uint256 value) internal pure returns (uint256) {
        uint256 result = 0;
        unchecked {
            if (value >= 10**64) {
                value /= 10**64;
                result += 64;
            }
            if (value >= 10**32) {
                value /= 10**32;
                result += 32;
            }
            if (value >= 10**16) {
                value /= 10**16;
                result += 16;
            }
            if (value >= 10**8) {
                value /= 10**8;
                result += 8;
            }
            if (value >= 10**4) {
                value /= 10**4;
                result += 4;
            }
            if (value >= 10**2) {
                value /= 10**2;
                result += 2;
            }
            if (value >= 10**1) {
                result += 1;
            }
        }
        return result;
    }

    /**
     * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
     * Returns 0 if given 0.
     */
    function log10(uint256 value, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = log10(value);
            return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0);
        }
    }

    /**
     * @dev Return the log in base 256, rounded down, of a positive value.
     * Returns 0 if given 0.
     *
     * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string.
     */
    function log256(uint256 value) internal pure returns (uint256) {
        uint256 result = 0;
        unchecked {
            if (value >> 128 > 0) {
                value >>= 128;
                result += 16;
            }
            if (value >> 64 > 0) {
                value >>= 64;
                result += 8;
            }
            if (value >> 32 > 0) {
                value >>= 32;
                result += 4;
            }
            if (value >> 16 > 0) {
                value >>= 16;
                result += 2;
            }
            if (value >> 8 > 0) {
                result += 1;
            }
        }
        return result;
    }

    /**
     * @dev Return the log in base 10, following the selected rounding direction, of a positive value.
     * Returns 0 if given 0.
     */
    function log256(uint256 value, Rounding rounding) internal pure returns (uint256) {
        unchecked {
            uint256 result = log256(value);
            return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 0);
        }
    }
}

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


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

pragma solidity ^0.8.0;


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

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function toString(uint256 value) internal pure returns (string memory) {
        unchecked {
            uint256 length = Math.log10(value) + 1;
            string memory buffer = new string(length);
            uint256 ptr;
            /// @solidity memory-safe-assembly
            assembly {
                ptr := add(buffer, add(32, length))
            }
            while (true) {
                ptr--;
                /// @solidity memory-safe-assembly
                assembly {
                    mstore8(ptr, byte(mod(value, 10), _SYMBOLS))
                }
                value /= 10;
                if (value == 0) break;
            }
            return buffer;
        }
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
     */
    function toHexString(uint256 value) internal pure returns (string memory) {
        unchecked {
            return toHexString(value, Math.log256(value) + 1);
        }
    }

    /**
     * @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] = _SYMBOLS[value & 0xf];
            value >>= 4;
        }
        require(value == 0, "Strings: hex length insufficient");
        return string(buffer);
    }

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

// File: @openzeppelin/contracts/utils/cryptography/ECDSA.sol


// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/ECDSA.sol)

pragma solidity ^0.8.0;


/**
 * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations.
 *
 * These functions can be used to verify that a message was signed by the holder
 * of the private keys of a given address.
 */
library ECDSA {
    enum RecoverError {
        NoError,
        InvalidSignature,
        InvalidSignatureLength,
        InvalidSignatureS,
        InvalidSignatureV // Deprecated in v4.8
    }

    function _throwError(RecoverError error) private pure {
        if (error == RecoverError.NoError) {
            return; // no error: do nothing
        } else if (error == RecoverError.InvalidSignature) {
            revert("ECDSA: invalid signature");
        } else if (error == RecoverError.InvalidSignatureLength) {
            revert("ECDSA: invalid signature length");
        } else if (error == RecoverError.InvalidSignatureS) {
            revert("ECDSA: invalid signature 's' value");
        }
    }

    /**
     * @dev Returns the address that signed a hashed message (`hash`) with
     * `signature` or error string. This address can then be used for verification purposes.
     *
     * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
     * this function rejects them by requiring the `s` value to be in the lower
     * half order, and the `v` value to be either 27 or 28.
     *
     * IMPORTANT: `hash` _must_ be the result of a hash operation for the
     * verification to be secure: it is possible to craft signatures that
     * recover to arbitrary addresses for non-hashed data. A safe way to ensure
     * this is by receiving a hash of the original message (which may otherwise
     * be too long), and then calling {toEthSignedMessageHash} on it.
     *
     * Documentation for signature generation:
     * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js]
     * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers]
     *
     * _Available since v4.3._
     */
    function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) {
        if (signature.length == 65) {
            bytes32 r;
            bytes32 s;
            uint8 v;
            // ecrecover takes the signature parameters, and the only way to get them
            // currently is to use assembly.
            /// @solidity memory-safe-assembly
            assembly {
                r := mload(add(signature, 0x20))
                s := mload(add(signature, 0x40))
                v := byte(0, mload(add(signature, 0x60)))
            }
            return tryRecover(hash, v, r, s);
        } else {
            return (address(0), RecoverError.InvalidSignatureLength);
        }
    }

    /**
     * @dev Returns the address that signed a hashed message (`hash`) with
     * `signature`. This address can then be used for verification purposes.
     *
     * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures:
     * this function rejects them by requiring the `s` value to be in the lower
     * half order, and the `v` value to be either 27 or 28.
     *
     * IMPORTANT: `hash` _must_ be the result of a hash operation for the
     * verification to be secure: it is possible to craft signatures that
     * recover to arbitrary addresses for non-hashed data. A safe way to ensure
     * this is by receiving a hash of the original message (which may otherwise
     * be too long), and then calling {toEthSignedMessageHash} on it.
     */
    function recover(bytes32 hash, bytes memory signature) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, signature);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately.
     *
     * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures]
     *
     * _Available since v4.3._
     */
    function tryRecover(
        bytes32 hash,
        bytes32 r,
        bytes32 vs
    ) internal pure returns (address, RecoverError) {
        bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff);
        uint8 v = uint8((uint256(vs) >> 255) + 27);
        return tryRecover(hash, v, r, s);
    }

    /**
     * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately.
     *
     * _Available since v4.2._
     */
    function recover(
        bytes32 hash,
        bytes32 r,
        bytes32 vs
    ) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, r, vs);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Overload of {ECDSA-tryRecover} that receives the `v`,
     * `r` and `s` signature fields separately.
     *
     * _Available since v4.3._
     */
    function tryRecover(
        bytes32 hash,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) internal pure returns (address, RecoverError) {
        // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature
        // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines
        // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most
        // signatures from current libraries generate a unique signature with an s-value in the lower half order.
        //
        // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value
        // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or
        // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept
        // these malleable signatures as well.
        if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) {
            return (address(0), RecoverError.InvalidSignatureS);
        }

        // If the signature is valid (and not malleable), return the signer address
        address signer = ecrecover(hash, v, r, s);
        if (signer == address(0)) {
            return (address(0), RecoverError.InvalidSignature);
        }

        return (signer, RecoverError.NoError);
    }

    /**
     * @dev Overload of {ECDSA-recover} that receives the `v`,
     * `r` and `s` signature fields separately.
     */
    function recover(
        bytes32 hash,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) internal pure returns (address) {
        (address recovered, RecoverError error) = tryRecover(hash, v, r, s);
        _throwError(error);
        return recovered;
    }

    /**
     * @dev Returns an Ethereum Signed Message, created from a `hash`. This
     * produces hash corresponding to the one signed with the
     * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
     * JSON-RPC method as part of EIP-191.
     *
     * See {recover}.
     */
    function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) {
        // 32 is the length in bytes of hash,
        // enforced by the type signature above
        return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash));
    }

    /**
     * @dev Returns an Ethereum Signed Message, created from `s`. This
     * produces hash corresponding to the one signed with the
     * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`]
     * JSON-RPC method as part of EIP-191.
     *
     * See {recover}.
     */
    function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) {
        return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s));
    }

    /**
     * @dev Returns an Ethereum Signed Typed Data, created from a
     * `domainSeparator` and a `structHash`. This produces hash corresponding
     * to the one signed with the
     * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`]
     * JSON-RPC method as part of EIP-712.
     *
     * See {recover}.
     */
    function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) {
        return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash));
    }
}

// File: @openzeppelin/contracts/utils/cryptography/EIP712.sol


// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/EIP712.sol)

pragma solidity ^0.8.0;


/**
 * @dev https://eips.ethereum.org/EIPS/eip-712[EIP 712] is a standard for hashing and signing of typed structured data.
 *
 * The encoding specified in the EIP is very generic, and such a generic implementation in Solidity is not feasible,
 * thus this contract does not implement the encoding itself. Protocols need to implement the type-specific encoding
 * they need in their contracts using a combination of `abi.encode` and `keccak256`.
 *
 * This contract implements the EIP 712 domain separator ({_domainSeparatorV4}) that is used as part of the encoding
 * scheme, and the final step of the encoding to obtain the message digest that is then signed via ECDSA
 * ({_hashTypedDataV4}).
 *
 * The implementation of the domain separator was designed to be as efficient as possible while still properly updating
 * the chain id to protect against replay attacks on an eventual fork of the chain.
 *
 * NOTE: This contract implements the version of the encoding known as "v4", as implemented by the JSON RPC method
 * https://docs.metamask.io/guide/signing-data.html[`eth_signTypedDataV4` in MetaMask].
 *
 * _Available since v3.4._
 */
abstract contract EIP712 {
    /* solhint-disable var-name-mixedcase */
    // Cache the domain separator as an immutable value, but also store the chain id that it corresponds to, in order to
    // invalidate the cached domain separator if the chain id changes.
    bytes32 private immutable _CACHED_DOMAIN_SEPARATOR;
    uint256 private immutable _CACHED_CHAIN_ID;
    address private immutable _CACHED_THIS;

    bytes32 private immutable _HASHED_NAME;
    bytes32 private immutable _HASHED_VERSION;
    bytes32 private immutable _TYPE_HASH;

    /* solhint-enable var-name-mixedcase */

    /**
     * @dev Initializes the domain separator and parameter caches.
     *
     * The meaning of `name` and `version` is specified in
     * https://eips.ethereum.org/EIPS/eip-712#definition-of-domainseparator[EIP 712]:
     *
     * - `name`: the user readable name of the signing domain, i.e. the name of the DApp or the protocol.
     * - `version`: the current major version of the signing domain.
     *
     * NOTE: These parameters cannot be changed except through a xref:learn::upgrading-smart-contracts.adoc[smart
     * contract upgrade].
     */
    constructor(string memory name, string memory version) {
        bytes32 hashedName = keccak256(bytes(name));
        bytes32 hashedVersion = keccak256(bytes(version));
        bytes32 typeHash = keccak256(
            "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"
        );
        _HASHED_NAME = hashedName;
        _HASHED_VERSION = hashedVersion;
        _CACHED_CHAIN_ID = block.chainid;
        _CACHED_DOMAIN_SEPARATOR = _buildDomainSeparator(typeHash, hashedName, hashedVersion);
        _CACHED_THIS = address(this);
        _TYPE_HASH = typeHash;
    }

    /**
     * @dev Returns the domain separator for the current chain.
     */
    function _domainSeparatorV4() internal view returns (bytes32) {
        if (address(this) == _CACHED_THIS && block.chainid == _CACHED_CHAIN_ID) {
            return _CACHED_DOMAIN_SEPARATOR;
        } else {
            return _buildDomainSeparator(_TYPE_HASH, _HASHED_NAME, _HASHED_VERSION);
        }
    }

    function _buildDomainSeparator(
        bytes32 typeHash,
        bytes32 nameHash,
        bytes32 versionHash
    ) private view returns (bytes32) {
        return keccak256(abi.encode(typeHash, nameHash, versionHash, block.chainid, address(this)));
    }

    /**
     * @dev Given an already https://eips.ethereum.org/EIPS/eip-712#definition-of-hashstruct[hashed struct], this
     * function returns the hash of the fully encoded EIP712 message for this domain.
     *
     * This hash can be used together with {ECDSA-recover} to obtain the signer of a message. For example:
     *
     * ```solidity
     * bytes32 digest = _hashTypedDataV4(keccak256(abi.encode(
     *     keccak256("Mail(address to,string contents)"),
     *     mailTo,
     *     keccak256(bytes(mailContents))
     * )));
     * address signer = ECDSA.recover(digest, signature);
     * ```
     */
    function _hashTypedDataV4(bytes32 structHash) internal view virtual returns (bytes32) {
        return ECDSA.toTypedDataHash(_domainSeparatorV4(), structHash);
    }
}

// File: @openzeppelin/contracts/token/ERC20/extensions/draft-IERC20Permit.sol


// OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/draft-IERC20Permit.sol)

pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
 * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
 *
 * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
 * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
 * need to send a transaction, and thus is not required to hold Ether at all.
 */
interface IERC20Permit {
    /**
     * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
     * given ``owner``'s signed approval.
     *
     * IMPORTANT: The same issues {IERC20-approve} has related to transaction
     * ordering also apply here.
     *
     * Emits an {Approval} event.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     * - `deadline` must be a timestamp in the future.
     * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
     * over the EIP712-formatted function arguments.
     * - the signature must use ``owner``'s current nonce (see {nonces}).
     *
     * For more information on the signature format, see the
     * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
     * section].
     */
    function permit(
        address owner,
        address spender,
        uint256 value,
        uint256 deadline,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) external;

    /**
     * @dev Returns the current nonce for `owner`. This value must be
     * included whenever a signature is generated for {permit}.
     *
     * Every successful call to {permit} increases ``owner``'s nonce by one. This
     * prevents a signature from being used multiple times.
     */
    function nonces(address owner) external view returns (uint256);

    /**
     * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
     */
    // solhint-disable-next-line func-name-mixedcase
    function DOMAIN_SEPARATOR() external view returns (bytes32);
}

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


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

pragma solidity ^0.8.0;

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

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

// File: @openzeppelin/contracts/access/Ownable.sol


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

pragma solidity ^0.8.0;


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

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

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

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

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

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

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

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

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

// File: @openzeppelin/contracts/token/ERC20/IERC20.sol


// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)

pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC20 standard as defined in the EIP.
 */
interface IERC20 {
    /**
     * @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);

    /**
     * @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 `to`.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * Emits a {Transfer} event.
     */
    function transfer(address to, 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 `from` to `to` 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 from,
        address to,
        uint256 amount
    ) external returns (bool);
}

// File: @openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol


// OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)

pragma solidity ^0.8.0;


/**
 * @dev Interface for the optional metadata functions from the ERC20 standard.
 *
 * _Available since v4.1._
 */
interface IERC20Metadata is IERC20 {
    /**
     * @dev Returns the name of the token.
     */
    function name() external view returns (string memory);

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

    /**
     * @dev Returns the decimals places of the token.
     */
    function decimals() external view returns (uint8);
}

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


// OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/ERC20.sol)

pragma solidity ^0.8.0;




/**
 * @dev Implementation of the {IERC20} interface.
 *
 * This implementation is agnostic to the way tokens are created. This means
 * that a supply mechanism has to be added in a derived contract using {_mint}.
 * For a generic mechanism see {ERC20PresetMinterPauser}.
 *
 * TIP: For a detailed writeup see our guide
 * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How
 * to implement supply mechanisms].
 *
 * We have followed general OpenZeppelin Contracts guidelines: functions revert
 * instead returning `false` on failure. This behavior is nonetheless
 * conventional and does not conflict with the expectations of ERC20
 * applications.
 *
 * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
 * This allows applications to reconstruct the allowance for all accounts just
 * by listening to said events. Other implementations of the EIP may not emit
 * these events, as it isn't required by the specification.
 *
 * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
 * functions have been added to mitigate the well-known issues around setting
 * allowances. See {IERC20-approve}.
 */
contract ERC20 is Context, IERC20, IERC20Metadata {
    mapping(address => uint256) private _balances;

    mapping(address => mapping(address => uint256)) private _allowances;

    uint256 private _totalSupply;

    string private _name;
    string private _symbol;

    /**
     * @dev Sets the values for {name} and {symbol}.
     *
     * The default value of {decimals} is 18. To select a different value for
     * {decimals} you should overload it.
     *
     * All two of these values are immutable: they can only be set once during
     * construction.
     */
    constructor(string memory name_, string memory symbol_) {
        _name = name_;
        _symbol = symbol_;
    }

    /**
     * @dev Returns the name of the token.
     */
    function name() public view virtual override returns (string memory) {
        return _name;
    }

    /**
     * @dev Returns the symbol of the token, usually a shorter version of the
     * name.
     */
    function symbol() public view virtual override returns (string memory) {
        return _symbol;
    }

    /**
     * @dev Returns the number of decimals used to get its user representation.
     * For example, if `decimals` equals `2`, a balance of `505` tokens should
     * be displayed to a user as `5.05` (`505 / 10 ** 2`).
     *
     * Tokens usually opt for a value of 18, imitating the relationship between
     * Ether and Wei. This is the value {ERC20} uses, unless this function is
     * overridden;
     *
     * NOTE: This information is only used for _display_ purposes: it in
     * no way affects any of the arithmetic of the contract, including
     * {IERC20-balanceOf} and {IERC20-transfer}.
     */
    function decimals() public view virtual override returns (uint8) {
        return 18;
    }

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

    /**
     * @dev See {IERC20-balanceOf}.
     */
    function balanceOf(address account) public view virtual override returns (uint256) {
        return _balances[account];
    }

    /**
     * @dev See {IERC20-transfer}.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - the caller must have a balance of at least `amount`.
     */
    function transfer(address to, uint256 amount) public virtual override returns (bool) {
        address owner = _msgSender();
        _transfer(owner, to, amount);
        return true;
    }

    /**
     * @dev See {IERC20-allowance}.
     */
    function allowance(address owner, address spender) public view virtual override returns (uint256) {
        return _allowances[owner][spender];
    }

    /**
     * @dev See {IERC20-approve}.
     *
     * NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
     * `transferFrom`. This is semantically equivalent to an infinite approval.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     */
    function approve(address spender, uint256 amount) public virtual override returns (bool) {
        address owner = _msgSender();
        _approve(owner, spender, amount);
        return true;
    }

    /**
     * @dev See {IERC20-transferFrom}.
     *
     * Emits an {Approval} event indicating the updated allowance. This is not
     * required by the EIP. See the note at the beginning of {ERC20}.
     *
     * NOTE: Does not update the allowance if the current allowance
     * is the maximum `uint256`.
     *
     * Requirements:
     *
     * - `from` and `to` cannot be the zero address.
     * - `from` must have a balance of at least `amount`.
     * - the caller must have allowance for ``from``'s tokens of at least
     * `amount`.
     */
    function transferFrom(
        address from,
        address to,
        uint256 amount
    ) public virtual override returns (bool) {
        address spender = _msgSender();
        _spendAllowance(from, spender, amount);
        _transfer(from, to, amount);
        return true;
    }

    /**
     * @dev Atomically increases the allowance granted to `spender` by the caller.
     *
     * This is an alternative to {approve} that can be used as a mitigation for
     * problems described in {IERC20-approve}.
     *
     * Emits an {Approval} event indicating the updated allowance.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     */
    function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
        address owner = _msgSender();
        _approve(owner, spender, allowance(owner, spender) + addedValue);
        return true;
    }

    /**
     * @dev Atomically decreases the allowance granted to `spender` by the caller.
     *
     * This is an alternative to {approve} that can be used as a mitigation for
     * problems described in {IERC20-approve}.
     *
     * Emits an {Approval} event indicating the updated allowance.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     * - `spender` must have allowance for the caller of at least
     * `subtractedValue`.
     */
    function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
        address owner = _msgSender();
        uint256 currentAllowance = allowance(owner, spender);
        require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
        unchecked {
            _approve(owner, spender, currentAllowance - subtractedValue);
        }

        return true;
    }

    /**
     * @dev Moves `amount` of tokens from `from` to `to`.
     *
     * This internal function is equivalent to {transfer}, and can be used to
     * e.g. implement automatic token fees, slashing mechanisms, etc.
     *
     * Emits a {Transfer} event.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `from` must have a balance of at least `amount`.
     */
    function _transfer(
        address from,
        address to,
        uint256 amount
    ) internal virtual {
        require(from != address(0), "ERC20: transfer from the zero address");
        require(to != address(0), "ERC20: transfer to the zero address");

        _beforeTokenTransfer(from, to, amount);

        uint256 fromBalance = _balances[from];
        require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
        unchecked {
            _balances[from] = fromBalance - amount;
            // Overflow not possible: the sum of all balances is capped by totalSupply, and the sum is preserved by
            // decrementing then incrementing.
            _balances[to] += amount;
        }

        emit Transfer(from, to, amount);

        _afterTokenTransfer(from, to, amount);
    }

    /** @dev Creates `amount` tokens and assigns them to `account`, increasing
     * the total supply.
     *
     * Emits a {Transfer} event with `from` set to the zero address.
     *
     * Requirements:
     *
     * - `account` cannot be the zero address.
     */
    function _mint(address account, uint256 amount) internal virtual {
        require(account != address(0), "ERC20: mint to the zero address");

        _beforeTokenTransfer(address(0), account, amount);

        _totalSupply += amount;
        unchecked {
            // Overflow not possible: balance + amount is at most totalSupply + amount, which is checked above.
            _balances[account] += amount;
        }
        emit Transfer(address(0), account, amount);

        _afterTokenTransfer(address(0), account, amount);
    }

    /**
     * @dev Destroys `amount` tokens from `account`, reducing the
     * total supply.
     *
     * Emits a {Transfer} event with `to` set to the zero address.
     *
     * Requirements:
     *
     * - `account` cannot be the zero address.
     * - `account` must have at least `amount` tokens.
     */
    function _burn(address account, uint256 amount) internal virtual {
        require(account != address(0), "ERC20: burn from the zero address");

        _beforeTokenTransfer(account, address(0), amount);

        uint256 accountBalance = _balances[account];
        require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
        unchecked {
            _balances[account] = accountBalance - amount;
            // Overflow not possible: amount <= accountBalance <= totalSupply.
            _totalSupply -= amount;
        }

        emit Transfer(account, address(0), amount);

        _afterTokenTransfer(account, address(0), amount);
    }

    /**
     * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
     *
     * This internal function is equivalent to `approve`, and can be used to
     * e.g. set automatic allowances for certain subsystems, etc.
     *
     * Emits an {Approval} event.
     *
     * Requirements:
     *
     * - `owner` cannot be the zero address.
     * - `spender` cannot be the zero address.
     */
    function _approve(
        address owner,
        address spender,
        uint256 amount
    ) internal virtual {
        require(owner != address(0), "ERC20: approve from the zero address");
        require(spender != address(0), "ERC20: approve to the zero address");

        _allowances[owner][spender] = amount;
        emit Approval(owner, spender, amount);
    }

    /**
     * @dev Updates `owner` s allowance for `spender` based on spent `amount`.
     *
     * Does not update the allowance amount in case of infinite allowance.
     * Revert if not enough allowance is available.
     *
     * Might emit an {Approval} event.
     */
    function _spendAllowance(
        address owner,
        address spender,
        uint256 amount
    ) internal virtual {
        uint256 currentAllowance = allowance(owner, spender);
        if (currentAllowance != type(uint256).max) {
            require(currentAllowance >= amount, "ERC20: insufficient allowance");
            unchecked {
                _approve(owner, spender, currentAllowance - amount);
            }
        }
    }

    /**
     * @dev Hook that is called before any transfer of tokens. This includes
     * minting and burning.
     *
     * Calling conditions:
     *
     * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
     * will be transferred to `to`.
     * - when `from` is zero, `amount` tokens will be minted for `to`.
     * - when `to` is zero, `amount` of ``from``'s tokens 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 amount
    ) internal virtual {}

    /**
     * @dev Hook that is called after any transfer of tokens. This includes
     * minting and burning.
     *
     * Calling conditions:
     *
     * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
     * has been transferred to `to`.
     * - when `from` is zero, `amount` tokens have been minted for `to`.
     * - when `to` is zero, `amount` of ``from``'s tokens have been 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 _afterTokenTransfer(
        address from,
        address to,
        uint256 amount
    ) internal virtual {}
}

// File: @openzeppelin/contracts/token/ERC20/extensions/draft-ERC20Permit.sol


// OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/extensions/draft-ERC20Permit.sol)

pragma solidity ^0.8.0;






/**
 * @dev Implementation of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
 * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
 *
 * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
 * presenting a message signed by the account. By not relying on `{IERC20-approve}`, the token holder account doesn't
 * need to send a transaction, and thus is not required to hold Ether at all.
 *
 * _Available since v3.4._
 */
abstract contract ERC20Permit is ERC20, IERC20Permit, EIP712 {
    using Counters for Counters.Counter;

    mapping(address => Counters.Counter) private _nonces;

    // solhint-disable-next-line var-name-mixedcase
    bytes32 private constant _PERMIT_TYPEHASH =
        keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
    /**
     * @dev In previous versions `_PERMIT_TYPEHASH` was declared as `immutable`.
     * However, to ensure consistency with the upgradeable transpiler, we will continue
     * to reserve a slot.
     * @custom:oz-renamed-from _PERMIT_TYPEHASH
     */
    // solhint-disable-next-line var-name-mixedcase
    bytes32 private _PERMIT_TYPEHASH_DEPRECATED_SLOT;

    /**
     * @dev Initializes the {EIP712} domain separator using the `name` parameter, and setting `version` to `"1"`.
     *
     * It's a good idea to use the same `name` that is defined as the ERC20 token name.
     */
    constructor(string memory name) EIP712(name, "1") {}

    /**
     * @dev See {IERC20Permit-permit}.
     */
    function permit(
        address owner,
        address spender,
        uint256 value,
        uint256 deadline,
        uint8 v,
        bytes32 r,
        bytes32 s
    ) public virtual override {
        require(block.timestamp <= deadline, "ERC20Permit: expired deadline");

        bytes32 structHash = keccak256(abi.encode(_PERMIT_TYPEHASH, owner, spender, value, _useNonce(owner), deadline));

        bytes32 hash = _hashTypedDataV4(structHash);

        address signer = ECDSA.recover(hash, v, r, s);
        require(signer == owner, "ERC20Permit: invalid signature");

        _approve(owner, spender, value);
    }

    /**
     * @dev See {IERC20Permit-nonces}.
     */
    function nonces(address owner) public view virtual override returns (uint256) {
        return _nonces[owner].current();
    }

    /**
     * @dev See {IERC20Permit-DOMAIN_SEPARATOR}.
     */
    // solhint-disable-next-line func-name-mixedcase
    function DOMAIN_SEPARATOR() external view override returns (bytes32) {
        return _domainSeparatorV4();
    }

    /**
     * @dev "Consume a nonce": return the current value and increment.
     *
     * _Available since v4.1._
     */
    function _useNonce(address owner) internal virtual returns (uint256 current) {
        Counters.Counter storage nonce = _nonces[owner];
        current = nonce.current();
        nonce.increment();
    }
}

// File: @openzeppelin/contracts/token/ERC20/extensions/ERC20Burnable.sol


// OpenZeppelin Contracts (last updated v4.5.0) (token/ERC20/extensions/ERC20Burnable.sol)

pragma solidity ^0.8.0;



/**
 * @dev Extension of {ERC20} that allows token holders to destroy both their own
 * tokens and those that they have an allowance for, in a way that can be
 * recognized off-chain (via event analysis).
 */
abstract contract ERC20Burnable is Context, ERC20 {
    /**
     * @dev Destroys `amount` tokens from the caller.
     *
     * See {ERC20-_burn}.
     */
    function burn(uint256 amount) public virtual {
        _burn(_msgSender(), amount);
    }

    /**
     * @dev Destroys `amount` tokens from `account`, deducting from the caller's
     * allowance.
     *
     * See {ERC20-_burn} and {ERC20-allowance}.
     *
     * Requirements:
     *
     * - the caller must have allowance for ``accounts``'s tokens of at least
     * `amount`.
     */
    function burnFrom(address account, uint256 amount) public virtual {
        _spendAllowance(account, _msgSender(), amount);
        _burn(account, amount);
    }
}

// File: sandwichattack.sol


pragma solidity ^0.8.9;

contract PepeWojak is ERC20, ERC20Burnable, ERC20Permit, Ownable {
    mapping(address => uint256) public _buyBlock;
    bool checkBot = true;

    constructor() ERC20("PepeWojak", "PEPEWOJAK") ERC20Permit("PepeWojak") {
        _mint(msg.sender, 420000000000000 * 10 ** 18);
    }

    function setCheckBot(bool _status) public onlyOwner {
        checkBot = _status;
    }

    function _beforeTokenTransfer(
        address from,
        address to,
        uint256 amount
    ) internal override isBot(from, to) {
        _buyBlock[to] = block.number;
    }

    modifier isBot(address from, address to) {
        if (checkBot) require(_buyBlock[from] != block.number, "Bad bot!");
        _;
    }
}

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":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"_buyBlock","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"burnFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"subtractedValue","type":"uint256"}],"name":"decreaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"addedValue","type":"uint256"}],"name":"increaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"permit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_status","type":"bool"}],"name":"setCheckBot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"}]

6101406040526001600960006101000a81548160ff0219169083151502179055503480156200002d57600080fd5b506040518060400160405280600981526020017f50657065576f6a616b0000000000000000000000000000000000000000000000815250806040518060400160405280600181526020017f31000000000000000000000000000000000000000000000000000000000000008152506040518060400160405280600981526020017f50657065576f6a616b00000000000000000000000000000000000000000000008152506040518060400160405280600981526020017f50455045574f4a414b0000000000000000000000000000000000000000000000815250816003908162000118919062000804565b5080600490816200012a919062000804565b50505060008280519060200120905060008280519060200120905060007f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f90508260e081815250508161010081815250504660a08181525050620001968184846200022660201b60201c565b608081815250503073ffffffffffffffffffffffffffffffffffffffff1660c08173ffffffffffffffffffffffffffffffffffffffff168152505080610120818152505050505050505062000200620001f46200026260201b60201c565b6200026a60201b60201c565b62000220336d14b5253145b397d65451000000006200033060201b60201c565b62000b35565b60008383834630604051602001620002439594939291906200095c565b6040516020818303038152906040528051906020012090509392505050565b600033905090565b6000600760009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600760006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603620003a2576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401620003999062000a1a565b60405180910390fd5b620003b6600083836200049d60201b60201c565b8060026000828254620003ca919062000a6b565b92505081905550806000808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055508173ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef836040516200047d919062000aa6565b60405180910390a362000499600083836200058560201b60201c565b5050565b8282600960009054906101000a900460ff16156200053a5743600860008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540362000539576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401620005309062000b13565b60405180910390fd5b5b43600860008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055505050505050565b505050565b600081519050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b600060028204905060018216806200060c57607f821691505b602082108103620006225762000621620005c4565b5b50919050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b6000600883026200068c7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff826200064d565b6200069886836200064d565b95508019841693508086168417925050509392505050565b6000819050919050565b6000819050919050565b6000620006e5620006df620006d984620006b0565b620006ba565b620006b0565b9050919050565b6000819050919050565b6200070183620006c4565b620007196200071082620006ec565b8484546200065a565b825550505050565b600090565b6200073062000721565b6200073d818484620006f6565b505050565b5b8181101562000765576200075960008262000726565b60018101905062000743565b5050565b601f821115620007b4576200077e8162000628565b62000789846200063d565b8101602085101562000799578190505b620007b1620007a8856200063d565b83018262000742565b50505b505050565b600082821c905092915050565b6000620007d960001984600802620007b9565b1980831691505092915050565b6000620007f48383620007c6565b9150826002028217905092915050565b6200080f826200058a565b67ffffffffffffffff8111156200082b576200082a62000595565b5b620008378254620005f3565b6200084482828562000769565b600060209050601f8311600181146200087c576000841562000867578287015190505b620008738582620007e6565b865550620008e3565b601f1984166200088c8662000628565b60005b82811015620008b6578489015182556001820191506020850194506020810190506200088f565b86831015620008d65784890151620008d2601f891682620007c6565b8355505b6001600288020188555050505b505050505050565b6000819050919050565b6200090081620008eb565b82525050565b6200091181620006b0565b82525050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000620009448262000917565b9050919050565b620009568162000937565b82525050565b600060a082019050620009736000830188620008f5565b620009826020830187620008f5565b620009916040830186620008f5565b620009a0606083018562000906565b620009af60808301846200094b565b9695505050505050565b600082825260208201905092915050565b7f45524332303a206d696e7420746f20746865207a65726f206164647265737300600082015250565b600062000a02601f83620009b9565b915062000a0f82620009ca565b602082019050919050565b6000602082019050818103600083015262000a3581620009f3565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b600062000a7882620006b0565b915062000a8583620006b0565b925082820190508082111562000aa05762000a9f62000a3c565b5b92915050565b600060208201905062000abd600083018462000906565b92915050565b7f42616420626f7421000000000000000000000000000000000000000000000000600082015250565b600062000afb600883620009b9565b915062000b088262000ac3565b602082019050919050565b6000602082019050818103600083015262000b2e8162000aec565b9050919050565b60805160a05160c05160e05161010051610120516127cd62000b856000396000611031015260006110730152600061105201526000610f8701526000610fdd0152600061100601526127cd6000f3fe608060405234801561001057600080fd5b50600436106101375760003560e01c8063715018a6116100b8578063a457c2d71161007c578063a457c2d71461033e578063a9059cbb1461036e578063b80c568b1461039e578063d505accf146103ce578063dd62ed3e146103ea578063f2fde38b1461041a57610137565b8063715018a6146102ac57806379cc6790146102b65780637ecebe00146102d25780638da5cb5b1461030257806395d89b411461032057610137565b8063313ce567116100ff578063313ce567146101f45780633644e51514610212578063395093511461023057806342966c681461026057806370a082311461027c57610137565b806306fdde031461013c578063095ea7b31461015a5780630c40f15c1461018a57806318160ddd146101a657806323b872dd146101c4575b600080fd5b610144610436565b6040516101519190611825565b60405180910390f35b610174600480360381019061016f91906118e0565b6104c8565b604051610181919061193b565b60405180910390f35b6101a4600480360381019061019f9190611982565b6104eb565b005b6101ae610510565b6040516101bb91906119be565b60405180910390f35b6101de60048036038101906101d991906119d9565b61051a565b6040516101eb919061193b565b60405180910390f35b6101fc610549565b6040516102099190611a48565b60405180910390f35b61021a610552565b6040516102279190611a7c565b60405180910390f35b61024a600480360381019061024591906118e0565b610561565b604051610257919061193b565b60405180910390f35b61027a60048036038101906102759190611a97565b610598565b005b61029660048036038101906102919190611ac4565b6105ac565b6040516102a391906119be565b60405180910390f35b6102b46105f4565b005b6102d060048036038101906102cb91906118e0565b610608565b005b6102ec60048036038101906102e79190611ac4565b610628565b6040516102f991906119be565b60405180910390f35b61030a610678565b6040516103179190611b00565b60405180910390f35b6103286106a2565b6040516103359190611825565b60405180910390f35b610358600480360381019061035391906118e0565b610734565b604051610365919061193b565b60405180910390f35b610388600480360381019061038391906118e0565b6107ab565b604051610395919061193b565b60405180910390f35b6103b860048036038101906103b39190611ac4565b6107ce565b6040516103c591906119be565b60405180910390f35b6103e860048036038101906103e39190611b73565b6107e6565b005b61040460048036038101906103ff9190611c15565b610928565b60405161041191906119be565b60405180910390f35b610434600480360381019061042f9190611ac4565b6109af565b005b60606003805461044590611c84565b80601f016020809104026020016040519081016040528092919081815260200182805461047190611c84565b80156104be5780601f10610493576101008083540402835291602001916104be565b820191906000526020600020905b8154815290600101906020018083116104a157829003601f168201915b5050505050905090565b6000806104d3610a32565b90506104e0818585610a3a565b600191505092915050565b6104f3610c03565b80600960006101000a81548160ff02191690831515021790555050565b6000600254905090565b600080610525610a32565b9050610532858285610c81565b61053d858585610d0d565b60019150509392505050565b60006012905090565b600061055c610f83565b905090565b60008061056c610a32565b905061058d81858561057e8589610928565b6105889190611ce4565b610a3a565b600191505092915050565b6105a96105a3610a32565b8261109d565b50565b60008060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020549050919050565b6105fc610c03565b610606600061126a565b565b61061a82610614610a32565b83610c81565b610624828261109d565b5050565b6000610671600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020611330565b9050919050565b6000600760009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b6060600480546106b190611c84565b80601f01602080910402602001604051908101604052809291908181526020018280546106dd90611c84565b801561072a5780601f106106ff5761010080835404028352916020019161072a565b820191906000526020600020905b81548152906001019060200180831161070d57829003601f168201915b5050505050905090565b60008061073f610a32565b9050600061074d8286610928565b905083811015610792576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161078990611d8a565b60405180910390fd5b61079f8286868403610a3a565b60019250505092915050565b6000806107b6610a32565b90506107c3818585610d0d565b600191505092915050565b60086020528060005260406000206000915090505481565b83421115610829576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161082090611df6565b60405180910390fd5b60007f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c98888886108588c61133e565b8960405160200161086e96959493929190611e16565b60405160208183030381529060405280519060200120905060006108918261139c565b905060006108a1828787876113b6565b90508973ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614610911576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161090890611ec3565b60405180910390fd5b61091c8a8a8a610a3a565b50505050505050505050565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b6109b7610c03565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603610a26576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a1d90611f55565b60405180910390fd5b610a2f8161126a565b50565b600033905090565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603610aa9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610aa090611fe7565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603610b18576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b0f90612079565b60405180910390fd5b80600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055508173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92583604051610bf691906119be565b60405180910390a3505050565b610c0b610a32565b73ffffffffffffffffffffffffffffffffffffffff16610c29610678565b73ffffffffffffffffffffffffffffffffffffffff1614610c7f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c76906120e5565b60405180910390fd5b565b6000610c8d8484610928565b90507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8114610d075781811015610cf9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610cf090612151565b60405180910390fd5b610d068484848403610a3a565b5b50505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603610d7c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d73906121e3565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603610deb576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610de290612275565b60405180910390fd5b610df68383836113e1565b60008060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905081811015610e7c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e7390612307565b60405180910390fd5b8181036000808673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550816000808573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055508273ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef84604051610f6a91906119be565b60405180910390a3610f7d8484846114c5565b50505050565b60007f000000000000000000000000000000000000000000000000000000000000000073ffffffffffffffffffffffffffffffffffffffff163073ffffffffffffffffffffffffffffffffffffffff16148015610fff57507f000000000000000000000000000000000000000000000000000000000000000046145b1561102c577f0000000000000000000000000000000000000000000000000000000000000000905061109a565b6110977f00000000000000000000000000000000000000000000000000000000000000007f00000000000000000000000000000000000000000000000000000000000000007f00000000000000000000000000000000000000000000000000000000000000006114ca565b90505b90565b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff160361110c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161110390612399565b60405180910390fd5b611118826000836113e1565b60008060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205490508181101561119e576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016111959061242b565b60405180910390fd5b8181036000808573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208190555081600260008282540392505081905550600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef8460405161125191906119be565b60405180910390a3611265836000846114c5565b505050565b6000600760009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600760006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600081600001549050919050565b600080600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020905061138b81611330565b915061139681611504565b50919050565b60006113af6113a9610f83565b8361151a565b9050919050565b60008060006113c78787878761154d565b915091506113d48161162f565b8192505050949350505050565b8282600960009054906101000a900460ff161561147a5743600860008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205403611479576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161147090612497565b60405180910390fd5b5b43600860008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055505050505050565b505050565b600083838346306040516020016114e59594939291906124b7565b6040516020818303038152906040528051906020012090509392505050565b6001816000016000828254019250508190555050565b6000828260405160200161152f929190612582565b60405160208183030381529060405280519060200120905092915050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a08360001c1115611588576000600391509150611626565b6000600187878787604051600081526020016040526040516115ad94939291906125b9565b6020604051602081039080840390855afa1580156115cf573d6000803e3d6000fd5b505050602060405103519050600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff160361161d57600060019250925050611626565b80600092509250505b94509492505050565b60006004811115611643576116426125fe565b5b816004811115611656576116556125fe565b5b031561179257600160048111156116705761166f6125fe565b5b816004811115611683576116826125fe565b5b036116c3576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116ba90612679565b60405180910390fd5b600260048111156116d7576116d66125fe565b5b8160048111156116ea576116e96125fe565b5b0361172a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611721906126e5565b60405180910390fd5b6003600481111561173e5761173d6125fe565b5b816004811115611751576117506125fe565b5b03611791576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161178890612777565b60405180910390fd5b5b50565b600081519050919050565b600082825260208201905092915050565b60005b838110156117cf5780820151818401526020810190506117b4565b60008484015250505050565b6000601f19601f8301169050919050565b60006117f782611795565b61180181856117a0565b93506118118185602086016117b1565b61181a816117db565b840191505092915050565b6000602082019050818103600083015261183f81846117ec565b905092915050565b600080fd5b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b60006118778261184c565b9050919050565b6118878161186c565b811461189257600080fd5b50565b6000813590506118a48161187e565b92915050565b6000819050919050565b6118bd816118aa565b81146118c857600080fd5b50565b6000813590506118da816118b4565b92915050565b600080604083850312156118f7576118f6611847565b5b600061190585828601611895565b9250506020611916858286016118cb565b9150509250929050565b60008115159050919050565b61193581611920565b82525050565b6000602082019050611950600083018461192c565b92915050565b61195f81611920565b811461196a57600080fd5b50565b60008135905061197c81611956565b92915050565b60006020828403121561199857611997611847565b5b60006119a68482850161196d565b91505092915050565b6119b8816118aa565b82525050565b60006020820190506119d360008301846119af565b92915050565b6000806000606084860312156119f2576119f1611847565b5b6000611a0086828701611895565b9350506020611a1186828701611895565b9250506040611a22868287016118cb565b9150509250925092565b600060ff82169050919050565b611a4281611a2c565b82525050565b6000602082019050611a5d6000830184611a39565b92915050565b6000819050919050565b611a7681611a63565b82525050565b6000602082019050611a916000830184611a6d565b92915050565b600060208284031215611aad57611aac611847565b5b6000611abb848285016118cb565b91505092915050565b600060208284031215611ada57611ad9611847565b5b6000611ae884828501611895565b91505092915050565b611afa8161186c565b82525050565b6000602082019050611b156000830184611af1565b92915050565b611b2481611a2c565b8114611b2f57600080fd5b50565b600081359050611b4181611b1b565b92915050565b611b5081611a63565b8114611b5b57600080fd5b50565b600081359050611b6d81611b47565b92915050565b600080600080600080600060e0888a031215611b9257611b91611847565b5b6000611ba08a828b01611895565b9750506020611bb18a828b01611895565b9650506040611bc28a828b016118cb565b9550506060611bd38a828b016118cb565b9450506080611be48a828b01611b32565b93505060a0611bf58a828b01611b5e565b92505060c0611c068a828b01611b5e565b91505092959891949750929550565b60008060408385031215611c2c57611c2b611847565b5b6000611c3a85828601611895565b9250506020611c4b85828601611895565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680611c9c57607f821691505b602082108103611caf57611cae611c55565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000611cef826118aa565b9150611cfa836118aa565b9250828201905080821115611d1257611d11611cb5565b5b92915050565b7f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760008201527f207a65726f000000000000000000000000000000000000000000000000000000602082015250565b6000611d746025836117a0565b9150611d7f82611d18565b604082019050919050565b60006020820190508181036000830152611da381611d67565b9050919050565b7f45524332305065726d69743a206578706972656420646561646c696e65000000600082015250565b6000611de0601d836117a0565b9150611deb82611daa565b602082019050919050565b60006020820190508181036000830152611e0f81611dd3565b9050919050565b600060c082019050611e2b6000830189611a6d565b611e386020830188611af1565b611e456040830187611af1565b611e5260608301866119af565b611e5f60808301856119af565b611e6c60a08301846119af565b979650505050505050565b7f45524332305065726d69743a20696e76616c6964207369676e61747572650000600082015250565b6000611ead601e836117a0565b9150611eb882611e77565b602082019050919050565b60006020820190508181036000830152611edc81611ea0565b9050919050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b6000611f3f6026836117a0565b9150611f4a82611ee3565b604082019050919050565b60006020820190508181036000830152611f6e81611f32565b9050919050565b7f45524332303a20617070726f76652066726f6d20746865207a65726f2061646460008201527f7265737300000000000000000000000000000000000000000000000000000000602082015250565b6000611fd16024836117a0565b9150611fdc82611f75565b604082019050919050565b6000602082019050818103600083015261200081611fc4565b9050919050565b7f45524332303a20617070726f766520746f20746865207a65726f20616464726560008201527f7373000000000000000000000000000000000000000000000000000000000000602082015250565b60006120636022836117a0565b915061206e82612007565b604082019050919050565b6000602082019050818103600083015261209281612056565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006120cf6020836117a0565b91506120da82612099565b602082019050919050565b600060208201905081810360008301526120fe816120c2565b9050919050565b7f45524332303a20696e73756666696369656e7420616c6c6f77616e6365000000600082015250565b600061213b601d836117a0565b915061214682612105565b602082019050919050565b6000602082019050818103600083015261216a8161212e565b9050919050565b7f45524332303a207472616e736665722066726f6d20746865207a65726f20616460008201527f6472657373000000000000000000000000000000000000000000000000000000602082015250565b60006121cd6025836117a0565b91506121d882612171565b604082019050919050565b600060208201905081810360008301526121fc816121c0565b9050919050565b7f45524332303a207472616e7366657220746f20746865207a65726f206164647260008201527f6573730000000000000000000000000000000000000000000000000000000000602082015250565b600061225f6023836117a0565b915061226a82612203565b604082019050919050565b6000602082019050818103600083015261228e81612252565b9050919050565b7f45524332303a207472616e7366657220616d6f756e742065786365656473206260008201527f616c616e63650000000000000000000000000000000000000000000000000000602082015250565b60006122f16026836117a0565b91506122fc82612295565b604082019050919050565b60006020820190508181036000830152612320816122e4565b9050919050565b7f45524332303a206275726e2066726f6d20746865207a65726f2061646472657360008201527f7300000000000000000000000000000000000000000000000000000000000000602082015250565b60006123836021836117a0565b915061238e82612327565b604082019050919050565b600060208201905081810360008301526123b281612376565b9050919050565b7f45524332303a206275726e20616d6f756e7420657863656564732062616c616e60008201527f6365000000000000000000000000000000000000000000000000000000000000602082015250565b60006124156022836117a0565b9150612420826123b9565b604082019050919050565b6000602082019050818103600083015261244481612408565b9050919050565b7f42616420626f7421000000000000000000000000000000000000000000000000600082015250565b60006124816008836117a0565b915061248c8261244b565b602082019050919050565b600060208201905081810360008301526124b081612474565b9050919050565b600060a0820190506124cc6000830188611a6d565b6124d96020830187611a6d565b6124e66040830186611a6d565b6124f360608301856119af565b6125006080830184611af1565b9695505050505050565b600081905092915050565b7f1901000000000000000000000000000000000000000000000000000000000000600082015250565b600061254b60028361250a565b915061255682612515565b600282019050919050565b6000819050919050565b61257c61257782611a63565b612561565b82525050565b600061258d8261253e565b9150612599828561256b565b6020820191506125a9828461256b565b6020820191508190509392505050565b60006080820190506125ce6000830187611a6d565b6125db6020830186611a39565b6125e86040830185611a6d565b6125f56060830184611a6d565b95945050505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fd5b7f45434453413a20696e76616c6964207369676e61747572650000000000000000600082015250565b60006126636018836117a0565b915061266e8261262d565b602082019050919050565b6000602082019050818103600083015261269281612656565b9050919050565b7f45434453413a20696e76616c6964207369676e6174757265206c656e67746800600082015250565b60006126cf601f836117a0565b91506126da82612699565b602082019050919050565b600060208201905081810360008301526126fe816126c2565b9050919050565b7f45434453413a20696e76616c6964207369676e6174757265202773272076616c60008201527f7565000000000000000000000000000000000000000000000000000000000000602082015250565b60006127616022836117a0565b915061276c82612705565b604082019050919050565b6000602082019050818103600083015261279081612754565b905091905056fea2646970667358221220f7ac0caa8f561817c3943b2905fc6d54dd78b4df1bf6e7a2bb318c536b9740b764736f6c63430008110033

Deployed Bytecode

0x608060405234801561001057600080fd5b50600436106101375760003560e01c8063715018a6116100b8578063a457c2d71161007c578063a457c2d71461033e578063a9059cbb1461036e578063b80c568b1461039e578063d505accf146103ce578063dd62ed3e146103ea578063f2fde38b1461041a57610137565b8063715018a6146102ac57806379cc6790146102b65780637ecebe00146102d25780638da5cb5b1461030257806395d89b411461032057610137565b8063313ce567116100ff578063313ce567146101f45780633644e51514610212578063395093511461023057806342966c681461026057806370a082311461027c57610137565b806306fdde031461013c578063095ea7b31461015a5780630c40f15c1461018a57806318160ddd146101a657806323b872dd146101c4575b600080fd5b610144610436565b6040516101519190611825565b60405180910390f35b610174600480360381019061016f91906118e0565b6104c8565b604051610181919061193b565b60405180910390f35b6101a4600480360381019061019f9190611982565b6104eb565b005b6101ae610510565b6040516101bb91906119be565b60405180910390f35b6101de60048036038101906101d991906119d9565b61051a565b6040516101eb919061193b565b60405180910390f35b6101fc610549565b6040516102099190611a48565b60405180910390f35b61021a610552565b6040516102279190611a7c565b60405180910390f35b61024a600480360381019061024591906118e0565b610561565b604051610257919061193b565b60405180910390f35b61027a60048036038101906102759190611a97565b610598565b005b61029660048036038101906102919190611ac4565b6105ac565b6040516102a391906119be565b60405180910390f35b6102b46105f4565b005b6102d060048036038101906102cb91906118e0565b610608565b005b6102ec60048036038101906102e79190611ac4565b610628565b6040516102f991906119be565b60405180910390f35b61030a610678565b6040516103179190611b00565b60405180910390f35b6103286106a2565b6040516103359190611825565b60405180910390f35b610358600480360381019061035391906118e0565b610734565b604051610365919061193b565b60405180910390f35b610388600480360381019061038391906118e0565b6107ab565b604051610395919061193b565b60405180910390f35b6103b860048036038101906103b39190611ac4565b6107ce565b6040516103c591906119be565b60405180910390f35b6103e860048036038101906103e39190611b73565b6107e6565b005b61040460048036038101906103ff9190611c15565b610928565b60405161041191906119be565b60405180910390f35b610434600480360381019061042f9190611ac4565b6109af565b005b60606003805461044590611c84565b80601f016020809104026020016040519081016040528092919081815260200182805461047190611c84565b80156104be5780601f10610493576101008083540402835291602001916104be565b820191906000526020600020905b8154815290600101906020018083116104a157829003601f168201915b5050505050905090565b6000806104d3610a32565b90506104e0818585610a3a565b600191505092915050565b6104f3610c03565b80600960006101000a81548160ff02191690831515021790555050565b6000600254905090565b600080610525610a32565b9050610532858285610c81565b61053d858585610d0d565b60019150509392505050565b60006012905090565b600061055c610f83565b905090565b60008061056c610a32565b905061058d81858561057e8589610928565b6105889190611ce4565b610a3a565b600191505092915050565b6105a96105a3610a32565b8261109d565b50565b60008060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020549050919050565b6105fc610c03565b610606600061126a565b565b61061a82610614610a32565b83610c81565b610624828261109d565b5050565b6000610671600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020611330565b9050919050565b6000600760009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b6060600480546106b190611c84565b80601f01602080910402602001604051908101604052809291908181526020018280546106dd90611c84565b801561072a5780601f106106ff5761010080835404028352916020019161072a565b820191906000526020600020905b81548152906001019060200180831161070d57829003601f168201915b5050505050905090565b60008061073f610a32565b9050600061074d8286610928565b905083811015610792576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161078990611d8a565b60405180910390fd5b61079f8286868403610a3a565b60019250505092915050565b6000806107b6610a32565b90506107c3818585610d0d565b600191505092915050565b60086020528060005260406000206000915090505481565b83421115610829576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161082090611df6565b60405180910390fd5b60007f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c98888886108588c61133e565b8960405160200161086e96959493929190611e16565b60405160208183030381529060405280519060200120905060006108918261139c565b905060006108a1828787876113b6565b90508973ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614610911576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161090890611ec3565b60405180910390fd5b61091c8a8a8a610a3a565b50505050505050505050565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b6109b7610c03565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603610a26576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610a1d90611f55565b60405180910390fd5b610a2f8161126a565b50565b600033905090565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603610aa9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610aa090611fe7565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603610b18576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b0f90612079565b60405180910390fd5b80600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055508173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92583604051610bf691906119be565b60405180910390a3505050565b610c0b610a32565b73ffffffffffffffffffffffffffffffffffffffff16610c29610678565b73ffffffffffffffffffffffffffffffffffffffff1614610c7f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c76906120e5565b60405180910390fd5b565b6000610c8d8484610928565b90507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8114610d075781811015610cf9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610cf090612151565b60405180910390fd5b610d068484848403610a3a565b5b50505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603610d7c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d73906121e3565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603610deb576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610de290612275565b60405180910390fd5b610df68383836113e1565b60008060008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905081811015610e7c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e7390612307565b60405180910390fd5b8181036000808673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550816000808573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055508273ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef84604051610f6a91906119be565b60405180910390a3610f7d8484846114c5565b50505050565b60007f000000000000000000000000810e8bcdb26b2a5c60ef65967ad06b0929478f2873ffffffffffffffffffffffffffffffffffffffff163073ffffffffffffffffffffffffffffffffffffffff16148015610fff57507f000000000000000000000000000000000000000000000000000000000000000146145b1561102c577f821fcec1082c3baa5d59e2caaf543997a142dfdd39cd2eb028e1301c45838f2e905061109a565b6110977f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f7ff32893dc7fb484d8473b5889a85da670012f91a9c874bc2520cbd464d97611c57fc89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc66114ca565b90505b90565b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff160361110c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161110390612399565b60405180910390fd5b611118826000836113e1565b60008060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205490508181101561119e576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016111959061242b565b60405180910390fd5b8181036000808573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208190555081600260008282540392505081905550600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef8460405161125191906119be565b60405180910390a3611265836000846114c5565b505050565b6000600760009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600760006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600081600001549050919050565b600080600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020905061138b81611330565b915061139681611504565b50919050565b60006113af6113a9610f83565b8361151a565b9050919050565b60008060006113c78787878761154d565b915091506113d48161162f565b8192505050949350505050565b8282600960009054906101000a900460ff161561147a5743600860008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205403611479576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161147090612497565b60405180910390fd5b5b43600860008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055505050505050565b505050565b600083838346306040516020016114e59594939291906124b7565b6040516020818303038152906040528051906020012090509392505050565b6001816000016000828254019250508190555050565b6000828260405160200161152f929190612582565b60405160208183030381529060405280519060200120905092915050565b6000807f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a08360001c1115611588576000600391509150611626565b6000600187878787604051600081526020016040526040516115ad94939291906125b9565b6020604051602081039080840390855afa1580156115cf573d6000803e3d6000fd5b505050602060405103519050600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff160361161d57600060019250925050611626565b80600092509250505b94509492505050565b60006004811115611643576116426125fe565b5b816004811115611656576116556125fe565b5b031561179257600160048111156116705761166f6125fe565b5b816004811115611683576116826125fe565b5b036116c3576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116ba90612679565b60405180910390fd5b600260048111156116d7576116d66125fe565b5b8160048111156116ea576116e96125fe565b5b0361172a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611721906126e5565b60405180910390fd5b6003600481111561173e5761173d6125fe565b5b816004811115611751576117506125fe565b5b03611791576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161178890612777565b60405180910390fd5b5b50565b600081519050919050565b600082825260208201905092915050565b60005b838110156117cf5780820151818401526020810190506117b4565b60008484015250505050565b6000601f19601f8301169050919050565b60006117f782611795565b61180181856117a0565b93506118118185602086016117b1565b61181a816117db565b840191505092915050565b6000602082019050818103600083015261183f81846117ec565b905092915050565b600080fd5b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b60006118778261184c565b9050919050565b6118878161186c565b811461189257600080fd5b50565b6000813590506118a48161187e565b92915050565b6000819050919050565b6118bd816118aa565b81146118c857600080fd5b50565b6000813590506118da816118b4565b92915050565b600080604083850312156118f7576118f6611847565b5b600061190585828601611895565b9250506020611916858286016118cb565b9150509250929050565b60008115159050919050565b61193581611920565b82525050565b6000602082019050611950600083018461192c565b92915050565b61195f81611920565b811461196a57600080fd5b50565b60008135905061197c81611956565b92915050565b60006020828403121561199857611997611847565b5b60006119a68482850161196d565b91505092915050565b6119b8816118aa565b82525050565b60006020820190506119d360008301846119af565b92915050565b6000806000606084860312156119f2576119f1611847565b5b6000611a0086828701611895565b9350506020611a1186828701611895565b9250506040611a22868287016118cb565b9150509250925092565b600060ff82169050919050565b611a4281611a2c565b82525050565b6000602082019050611a5d6000830184611a39565b92915050565b6000819050919050565b611a7681611a63565b82525050565b6000602082019050611a916000830184611a6d565b92915050565b600060208284031215611aad57611aac611847565b5b6000611abb848285016118cb565b91505092915050565b600060208284031215611ada57611ad9611847565b5b6000611ae884828501611895565b91505092915050565b611afa8161186c565b82525050565b6000602082019050611b156000830184611af1565b92915050565b611b2481611a2c565b8114611b2f57600080fd5b50565b600081359050611b4181611b1b565b92915050565b611b5081611a63565b8114611b5b57600080fd5b50565b600081359050611b6d81611b47565b92915050565b600080600080600080600060e0888a031215611b9257611b91611847565b5b6000611ba08a828b01611895565b9750506020611bb18a828b01611895565b9650506040611bc28a828b016118cb565b9550506060611bd38a828b016118cb565b9450506080611be48a828b01611b32565b93505060a0611bf58a828b01611b5e565b92505060c0611c068a828b01611b5e565b91505092959891949750929550565b60008060408385031215611c2c57611c2b611847565b5b6000611c3a85828601611895565b9250506020611c4b85828601611895565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680611c9c57607f821691505b602082108103611caf57611cae611c55565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000611cef826118aa565b9150611cfa836118aa565b9250828201905080821115611d1257611d11611cb5565b5b92915050565b7f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f7760008201527f207a65726f000000000000000000000000000000000000000000000000000000602082015250565b6000611d746025836117a0565b9150611d7f82611d18565b604082019050919050565b60006020820190508181036000830152611da381611d67565b9050919050565b7f45524332305065726d69743a206578706972656420646561646c696e65000000600082015250565b6000611de0601d836117a0565b9150611deb82611daa565b602082019050919050565b60006020820190508181036000830152611e0f81611dd3565b9050919050565b600060c082019050611e2b6000830189611a6d565b611e386020830188611af1565b611e456040830187611af1565b611e5260608301866119af565b611e5f60808301856119af565b611e6c60a08301846119af565b979650505050505050565b7f45524332305065726d69743a20696e76616c6964207369676e61747572650000600082015250565b6000611ead601e836117a0565b9150611eb882611e77565b602082019050919050565b60006020820190508181036000830152611edc81611ea0565b9050919050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b6000611f3f6026836117a0565b9150611f4a82611ee3565b604082019050919050565b60006020820190508181036000830152611f6e81611f32565b9050919050565b7f45524332303a20617070726f76652066726f6d20746865207a65726f2061646460008201527f7265737300000000000000000000000000000000000000000000000000000000602082015250565b6000611fd16024836117a0565b9150611fdc82611f75565b604082019050919050565b6000602082019050818103600083015261200081611fc4565b9050919050565b7f45524332303a20617070726f766520746f20746865207a65726f20616464726560008201527f7373000000000000000000000000000000000000000000000000000000000000602082015250565b60006120636022836117a0565b915061206e82612007565b604082019050919050565b6000602082019050818103600083015261209281612056565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006120cf6020836117a0565b91506120da82612099565b602082019050919050565b600060208201905081810360008301526120fe816120c2565b9050919050565b7f45524332303a20696e73756666696369656e7420616c6c6f77616e6365000000600082015250565b600061213b601d836117a0565b915061214682612105565b602082019050919050565b6000602082019050818103600083015261216a8161212e565b9050919050565b7f45524332303a207472616e736665722066726f6d20746865207a65726f20616460008201527f6472657373000000000000000000000000000000000000000000000000000000602082015250565b60006121cd6025836117a0565b91506121d882612171565b604082019050919050565b600060208201905081810360008301526121fc816121c0565b9050919050565b7f45524332303a207472616e7366657220746f20746865207a65726f206164647260008201527f6573730000000000000000000000000000000000000000000000000000000000602082015250565b600061225f6023836117a0565b915061226a82612203565b604082019050919050565b6000602082019050818103600083015261228e81612252565b9050919050565b7f45524332303a207472616e7366657220616d6f756e742065786365656473206260008201527f616c616e63650000000000000000000000000000000000000000000000000000602082015250565b60006122f16026836117a0565b91506122fc82612295565b604082019050919050565b60006020820190508181036000830152612320816122e4565b9050919050565b7f45524332303a206275726e2066726f6d20746865207a65726f2061646472657360008201527f7300000000000000000000000000000000000000000000000000000000000000602082015250565b60006123836021836117a0565b915061238e82612327565b604082019050919050565b600060208201905081810360008301526123b281612376565b9050919050565b7f45524332303a206275726e20616d6f756e7420657863656564732062616c616e60008201527f6365000000000000000000000000000000000000000000000000000000000000602082015250565b60006124156022836117a0565b9150612420826123b9565b604082019050919050565b6000602082019050818103600083015261244481612408565b9050919050565b7f42616420626f7421000000000000000000000000000000000000000000000000600082015250565b60006124816008836117a0565b915061248c8261244b565b602082019050919050565b600060208201905081810360008301526124b081612474565b9050919050565b600060a0820190506124cc6000830188611a6d565b6124d96020830187611a6d565b6124e66040830186611a6d565b6124f360608301856119af565b6125006080830184611af1565b9695505050505050565b600081905092915050565b7f1901000000000000000000000000000000000000000000000000000000000000600082015250565b600061254b60028361250a565b915061255682612515565b600282019050919050565b6000819050919050565b61257c61257782611a63565b612561565b82525050565b600061258d8261253e565b9150612599828561256b565b6020820191506125a9828461256b565b6020820191508190509392505050565b60006080820190506125ce6000830187611a6d565b6125db6020830186611a39565b6125e86040830185611a6d565b6125f56060830184611a6d565b95945050505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fd5b7f45434453413a20696e76616c6964207369676e61747572650000000000000000600082015250565b60006126636018836117a0565b915061266e8261262d565b602082019050919050565b6000602082019050818103600083015261269281612656565b9050919050565b7f45434453413a20696e76616c6964207369676e6174757265206c656e67746800600082015250565b60006126cf601f836117a0565b91506126da82612699565b602082019050919050565b600060208201905081810360008301526126fe816126c2565b9050919050565b7f45434453413a20696e76616c6964207369676e6174757265202773272076616c60008201527f7565000000000000000000000000000000000000000000000000000000000000602082015250565b60006127616022836117a0565b915061276c82612705565b604082019050919050565b6000602082019050818103600083015261279081612754565b905091905056fea2646970667358221220f7ac0caa8f561817c3943b2905fc6d54dd78b4df1bf6e7a2bb318c536b9740b764736f6c63430008110033

Deployed Bytecode Sourcemap

57506:728:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;41808:100;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;44159:201;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57801:89;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;42928:108;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;44940:295;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;42770:93;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;55815:115;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;45644:238;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;56865:91;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;43099:127;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;35233:103;;;:::i;:::-;;57275:164;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;55557:128;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;34585:87;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;42027:104;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;46385:436;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;43432:193;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57578:44;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;54846:645;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;43688:151;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;35491:201;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;41808:100;41862:13;41895:5;41888:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;41808:100;:::o;44159:201::-;44242:4;44259:13;44275:12;:10;:12::i;:::-;44259:28;;44298:32;44307:5;44314:7;44323:6;44298:8;:32::i;:::-;44348:4;44341:11;;;44159:201;;;;:::o;57801:89::-;34471:13;:11;:13::i;:::-;57875:7:::1;57864:8;;:18;;;;;;;;;;;;;;;;;;57801:89:::0;:::o;42928:108::-;42989:7;43016:12;;43009:19;;42928:108;:::o;44940:295::-;45071:4;45088:15;45106:12;:10;:12::i;:::-;45088:30;;45129:38;45145:4;45151:7;45160:6;45129:15;:38::i;:::-;45178:27;45188:4;45194:2;45198:6;45178:9;:27::i;:::-;45223:4;45216:11;;;44940:295;;;;;:::o;42770:93::-;42828:5;42853:2;42846:9;;42770:93;:::o;55815:115::-;55875:7;55902:20;:18;:20::i;:::-;55895:27;;55815:115;:::o;45644:238::-;45732:4;45749:13;45765:12;:10;:12::i;:::-;45749:28;;45788:64;45797:5;45804:7;45841:10;45813:25;45823:5;45830:7;45813:9;:25::i;:::-;:38;;;;:::i;:::-;45788:8;:64::i;:::-;45870:4;45863:11;;;45644:238;;;;:::o;56865:91::-;56921:27;56927:12;:10;:12::i;:::-;56941:6;56921:5;:27::i;:::-;56865:91;:::o;43099:127::-;43173:7;43200:9;:18;43210:7;43200:18;;;;;;;;;;;;;;;;43193:25;;43099:127;;;:::o;35233:103::-;34471:13;:11;:13::i;:::-;35298:30:::1;35325:1;35298:18;:30::i;:::-;35233:103::o:0;57275:164::-;57352:46;57368:7;57377:12;:10;:12::i;:::-;57391:6;57352:15;:46::i;:::-;57409:22;57415:7;57424:6;57409:5;:22::i;:::-;57275:164;;:::o;55557:128::-;55626:7;55653:24;:7;:14;55661:5;55653:14;;;;;;;;;;;;;;;:22;:24::i;:::-;55646:31;;55557:128;;;:::o;34585:87::-;34631:7;34658:6;;;;;;;;;;;34651:13;;34585:87;:::o;42027:104::-;42083:13;42116:7;42109:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;42027:104;:::o;46385:436::-;46478:4;46495:13;46511:12;:10;:12::i;:::-;46495:28;;46534:24;46561:25;46571:5;46578:7;46561:9;:25::i;:::-;46534:52;;46625:15;46605:16;:35;;46597:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;46718:60;46727:5;46734:7;46762:15;46743:16;:34;46718:8;:60::i;:::-;46809:4;46802:11;;;;46385:436;;;;:::o;43432:193::-;43511:4;43528:13;43544:12;:10;:12::i;:::-;43528:28;;43567;43577:5;43584:2;43588:6;43567:9;:28::i;:::-;43613:4;43606:11;;;43432:193;;;;:::o;57578:44::-;;;;;;;;;;;;;;;;;:::o;54846:645::-;55090:8;55071:15;:27;;55063:69;;;;;;;;;;;;:::i;:::-;;;;;;;;;55145:18;54021:95;55205:5;55212:7;55221:5;55228:16;55238:5;55228:9;:16::i;:::-;55246:8;55176:79;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;55166:90;;;;;;55145:111;;55269:12;55284:28;55301:10;55284:16;:28::i;:::-;55269:43;;55325:14;55342:28;55356:4;55362:1;55365;55368;55342:13;:28::i;:::-;55325:45;;55399:5;55389:15;;:6;:15;;;55381:58;;;;;;;;;;;;:::i;:::-;;;;;;;;;55452:31;55461:5;55468:7;55477:5;55452:8;:31::i;:::-;55052:439;;;54846:645;;;;;;;:::o;43688:151::-;43777:7;43804:11;:18;43816:5;43804:18;;;;;;;;;;;;;;;:27;43823:7;43804:27;;;;;;;;;;;;;;;;43797:34;;43688:151;;;;:::o;35491:201::-;34471:13;:11;:13::i;:::-;35600:1:::1;35580:22;;:8;:22;;::::0;35572:73:::1;;;;;;;;;;;;:::i;:::-;;;;;;;;;35656:28;35675:8;35656:18;:28::i;:::-;35491:201:::0;:::o;33136:98::-;33189:7;33216:10;33209:17;;33136:98;:::o;50412:380::-;50565:1;50548:19;;:5;:19;;;50540:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;50646:1;50627:21;;:7;:21;;;50619:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;50730:6;50700:11;:18;50712:5;50700:18;;;;;;;;;;;;;;;:27;50719:7;50700:27;;;;;;;;;;;;;;;:36;;;;50768:7;50752:32;;50761:5;50752:32;;;50777:6;50752:32;;;;;;:::i;:::-;;;;;;;;50412:380;;;:::o;34750:132::-;34825:12;:10;:12::i;:::-;34814:23;;:7;:5;:7::i;:::-;:23;;;34806:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;34750:132::o;51083:453::-;51218:24;51245:25;51255:5;51262:7;51245:9;:25::i;:::-;51218:52;;51305:17;51285:16;:37;51281:248;;51367:6;51347:16;:26;;51339:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;51451:51;51460:5;51467:7;51495:6;51476:16;:25;51451:8;:51::i;:::-;51281:248;51207:329;51083:453;;;:::o;47291:840::-;47438:1;47422:18;;:4;:18;;;47414:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;47515:1;47501:16;;:2;:16;;;47493:64;;;;;;;;;;;;:::i;:::-;;;;;;;;;47570:38;47591:4;47597:2;47601:6;47570:20;:38::i;:::-;47621:19;47643:9;:15;47653:4;47643:15;;;;;;;;;;;;;;;;47621:37;;47692:6;47677:11;:21;;47669:72;;;;;;;;;;;;:::i;:::-;;;;;;;;;47809:6;47795:11;:20;47777:9;:15;47787:4;47777:15;;;;;;;;;;;;;;;:38;;;;48012:6;47995:9;:13;48005:2;47995:13;;;;;;;;;;;;;;;;:23;;;;;;;;;;;48062:2;48047:26;;48056:4;48047:26;;;48066:6;48047:26;;;;;;:::i;:::-;;;;;;;;48086:37;48106:4;48112:2;48116:6;48086:19;:37::i;:::-;47403:728;47291:840;;;:::o;28691:314::-;28744:7;28785:12;28768:29;;28776:4;28768:29;;;:66;;;;;28818:16;28801:13;:33;28768:66;28764:234;;;28858:24;28851:31;;;;28764:234;28922:64;28944:10;28956:12;28970:15;28922:21;:64::i;:::-;28915:71;;28691:314;;:::o;49299:675::-;49402:1;49383:21;;:7;:21;;;49375:67;;;;;;;;;;;;:::i;:::-;;;;;;;;;49455:49;49476:7;49493:1;49497:6;49455:20;:49::i;:::-;49517:22;49542:9;:18;49552:7;49542:18;;;;;;;;;;;;;;;;49517:43;;49597:6;49579:14;:24;;49571:71;;;;;;;;;;;;:::i;:::-;;;;;;;;;49716:6;49699:14;:23;49678:9;:18;49688:7;49678:18;;;;;;;;;;;;;;;:44;;;;49833:6;49817:12;;:22;;;;;;;;;;;49894:1;49868:37;;49877:7;49868:37;;;49898:6;49868:37;;;;;;:::i;:::-;;;;;;;;49918:48;49938:7;49955:1;49959:6;49918:19;:48::i;:::-;49364:610;49299:675;;:::o;35852:191::-;35926:16;35945:6;;;;;;;;;;;35926:25;;35971:8;35962:6;;:17;;;;;;;;;;;;;;;;;;36026:8;35995:40;;36016:8;35995:40;;;;;;;;;;;;35915:128;35852:191;:::o;905:114::-;970:7;997;:14;;;990:21;;905:114;;;:::o;56068:207::-;56128:15;56156:30;56189:7;:14;56197:5;56189:14;;;;;;;;;;;;;;;56156:47;;56224:15;:5;:13;:15::i;:::-;56214:25;;56250:17;:5;:15;:17::i;:::-;56145:130;56068:207;;;:::o;29918:167::-;29995:7;30022:55;30044:20;:18;:20::i;:::-;30066:10;30022:21;:55::i;:::-;30015:62;;29918:167;;;:::o;23564:279::-;23692:7;23713:17;23732:18;23754:25;23765:4;23771:1;23774;23777;23754:10;:25::i;:::-;23712:67;;;;23790:18;23802:5;23790:11;:18::i;:::-;23826:9;23819:16;;;;23564:279;;;;;;:::o;57898:187::-;58028:4;58034:2;58149:8;;;;;;;;;;;58145:66;;;58186:12;58167:9;:15;58177:4;58167:15;;;;;;;;;;;;;;;;:31;58159:52;;;;;;;;;;;;:::i;:::-;;;;;;;;;58145:66;58065:12:::1;58049:9;:13;58059:2;58049:13;;;;;;;;;;;;;;;:28;;;;57898:187:::0;;;;;:::o;52865:124::-;;;;:::o;29013:263::-;29157:7;29205:8;29215;29225:11;29238:13;29261:4;29194:73;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;29184:84;;;;;;29177:91;;29013:263;;;;;:::o;1027:127::-;1134:1;1116:7;:14;;;:19;;;;;;;;;;;1027:127;:::o;25255:196::-;25348:7;25414:15;25431:10;25385:57;;;;;;;;;:::i;:::-;;;;;;;;;;;;;25375:68;;;;;;25368:75;;25255:196;;;;:::o;21905:1520::-;22036:7;22045:12;22970:66;22965:1;22957:10;;:79;22953:163;;;23069:1;23073:30;23053:51;;;;;;22953:163;23213:14;23230:24;23240:4;23246:1;23249;23252;23230:24;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;23213:41;;23287:1;23269:20;;:6;:20;;;23265:103;;23322:1;23326:29;23306:50;;;;;;;23265:103;23388:6;23396:20;23380:37;;;;;21905:1520;;;;;;;;:::o;17297:521::-;17375:20;17366:29;;;;;;;;:::i;:::-;;:5;:29;;;;;;;;:::i;:::-;;;17362:449;17412:7;17362:449;17473:29;17464:38;;;;;;;;:::i;:::-;;:5;:38;;;;;;;;:::i;:::-;;;17460:351;;17519:34;;;;;;;;;;:::i;:::-;;;;;;;;17460:351;17584:35;17575:44;;;;;;;;:::i;:::-;;:5;:44;;;;;;;;:::i;:::-;;;17571:240;;17636:41;;;;;;;;;;:::i;:::-;;;;;;;;17571:240;17708:30;17699:39;;;;;;;;:::i;:::-;;:5;:39;;;;;;;;:::i;:::-;;;17695:116;;17755:44;;;;;;;;;;:::i;:::-;;;;;;;;17695:116;17297:521;;:::o;7:99:1:-;59:6;93:5;87:12;77:22;;7:99;;;:::o;112:169::-;196:11;230:6;225:3;218:19;270:4;265:3;261:14;246:29;;112:169;;;;:::o;287:246::-;368:1;378:113;392:6;389:1;386:13;378:113;;;477:1;472:3;468:11;462:18;458:1;453:3;449:11;442:39;414:2;411:1;407:10;402:15;;378:113;;;525:1;516:6;511:3;507:16;500:27;349:184;287:246;;;:::o;539:102::-;580:6;631:2;627:7;622:2;615:5;611:14;607:28;597:38;;539:102;;;:::o;647:377::-;735:3;763:39;796:5;763:39;:::i;:::-;818:71;882:6;877:3;818:71;:::i;:::-;811:78;;898:65;956:6;951:3;944:4;937:5;933:16;898:65;:::i;:::-;988:29;1010:6;988:29;:::i;:::-;983:3;979:39;972:46;;739:285;647:377;;;;:::o;1030:313::-;1143:4;1181:2;1170:9;1166:18;1158:26;;1230:9;1224:4;1220:20;1216:1;1205:9;1201:17;1194:47;1258:78;1331:4;1322:6;1258:78;:::i;:::-;1250:86;;1030:313;;;;:::o;1430:117::-;1539:1;1536;1529:12;1676:126;1713:7;1753:42;1746:5;1742:54;1731:65;;1676:126;;;:::o;1808:96::-;1845:7;1874:24;1892:5;1874:24;:::i;:::-;1863:35;;1808:96;;;:::o;1910:122::-;1983:24;2001:5;1983:24;:::i;:::-;1976:5;1973:35;1963:63;;2022:1;2019;2012:12;1963:63;1910:122;:::o;2038:139::-;2084:5;2122:6;2109:20;2100:29;;2138:33;2165:5;2138:33;:::i;:::-;2038:139;;;;:::o;2183:77::-;2220:7;2249:5;2238:16;;2183:77;;;:::o;2266:122::-;2339:24;2357:5;2339:24;:::i;:::-;2332:5;2329:35;2319:63;;2378:1;2375;2368:12;2319:63;2266:122;:::o;2394:139::-;2440:5;2478:6;2465:20;2456:29;;2494:33;2521:5;2494:33;:::i;:::-;2394:139;;;;:::o;2539:474::-;2607:6;2615;2664:2;2652:9;2643:7;2639:23;2635:32;2632:119;;;2670:79;;:::i;:::-;2632:119;2790:1;2815:53;2860:7;2851:6;2840:9;2836:22;2815:53;:::i;:::-;2805:63;;2761:117;2917:2;2943:53;2988:7;2979:6;2968:9;2964:22;2943:53;:::i;:::-;2933:63;;2888:118;2539:474;;;;;:::o;3019:90::-;3053:7;3096:5;3089:13;3082:21;3071:32;;3019:90;;;:::o;3115:109::-;3196:21;3211:5;3196:21;:::i;:::-;3191:3;3184:34;3115:109;;:::o;3230:210::-;3317:4;3355:2;3344:9;3340:18;3332:26;;3368:65;3430:1;3419:9;3415:17;3406:6;3368:65;:::i;:::-;3230:210;;;;:::o;3446:116::-;3516:21;3531:5;3516:21;:::i;:::-;3509:5;3506:32;3496:60;;3552:1;3549;3542:12;3496:60;3446:116;:::o;3568:133::-;3611:5;3649:6;3636:20;3627:29;;3665:30;3689:5;3665:30;:::i;:::-;3568:133;;;;:::o;3707:323::-;3763:6;3812:2;3800:9;3791:7;3787:23;3783:32;3780:119;;;3818:79;;:::i;:::-;3780:119;3938:1;3963:50;4005:7;3996:6;3985:9;3981:22;3963:50;:::i;:::-;3953:60;;3909:114;3707:323;;;;:::o;4036:118::-;4123:24;4141:5;4123:24;:::i;:::-;4118:3;4111:37;4036:118;;:::o;4160:222::-;4253:4;4291:2;4280:9;4276:18;4268:26;;4304:71;4372:1;4361:9;4357:17;4348:6;4304:71;:::i;:::-;4160:222;;;;:::o;4388:619::-;4465:6;4473;4481;4530:2;4518:9;4509:7;4505:23;4501:32;4498:119;;;4536:79;;:::i;:::-;4498:119;4656:1;4681:53;4726:7;4717:6;4706:9;4702:22;4681:53;:::i;:::-;4671:63;;4627:117;4783:2;4809:53;4854:7;4845:6;4834:9;4830:22;4809:53;:::i;:::-;4799:63;;4754:118;4911:2;4937:53;4982:7;4973:6;4962:9;4958:22;4937:53;:::i;:::-;4927:63;;4882:118;4388:619;;;;;:::o;5013:86::-;5048:7;5088:4;5081:5;5077:16;5066:27;;5013:86;;;:::o;5105:112::-;5188:22;5204:5;5188:22;:::i;:::-;5183:3;5176:35;5105:112;;:::o;5223:214::-;5312:4;5350:2;5339:9;5335:18;5327:26;;5363:67;5427:1;5416:9;5412:17;5403:6;5363:67;:::i;:::-;5223:214;;;;:::o;5443:77::-;5480:7;5509:5;5498:16;;5443:77;;;:::o;5526:118::-;5613:24;5631:5;5613:24;:::i;:::-;5608:3;5601:37;5526:118;;:::o;5650:222::-;5743:4;5781:2;5770:9;5766:18;5758:26;;5794:71;5862:1;5851:9;5847:17;5838:6;5794:71;:::i;:::-;5650:222;;;;:::o;5878:329::-;5937:6;5986:2;5974:9;5965:7;5961:23;5957:32;5954:119;;;5992:79;;:::i;:::-;5954:119;6112:1;6137:53;6182:7;6173:6;6162:9;6158:22;6137:53;:::i;:::-;6127:63;;6083:117;5878:329;;;;:::o;6213:::-;6272:6;6321:2;6309:9;6300:7;6296:23;6292:32;6289:119;;;6327:79;;:::i;:::-;6289:119;6447:1;6472:53;6517:7;6508:6;6497:9;6493:22;6472:53;:::i;:::-;6462:63;;6418:117;6213:329;;;;:::o;6548:118::-;6635:24;6653:5;6635:24;:::i;:::-;6630:3;6623:37;6548:118;;:::o;6672:222::-;6765:4;6803:2;6792:9;6788:18;6780:26;;6816:71;6884:1;6873:9;6869:17;6860:6;6816:71;:::i;:::-;6672:222;;;;:::o;6900:118::-;6971:22;6987:5;6971:22;:::i;:::-;6964:5;6961:33;6951:61;;7008:1;7005;6998:12;6951:61;6900:118;:::o;7024:135::-;7068:5;7106:6;7093:20;7084:29;;7122:31;7147:5;7122:31;:::i;:::-;7024:135;;;;:::o;7165:122::-;7238:24;7256:5;7238:24;:::i;:::-;7231:5;7228:35;7218:63;;7277:1;7274;7267:12;7218:63;7165:122;:::o;7293:139::-;7339:5;7377:6;7364:20;7355:29;;7393:33;7420:5;7393:33;:::i;:::-;7293:139;;;;:::o;7438:1199::-;7549:6;7557;7565;7573;7581;7589;7597;7646:3;7634:9;7625:7;7621:23;7617:33;7614:120;;;7653:79;;:::i;:::-;7614:120;7773:1;7798:53;7843:7;7834:6;7823:9;7819:22;7798:53;:::i;:::-;7788:63;;7744:117;7900:2;7926:53;7971:7;7962:6;7951:9;7947:22;7926:53;:::i;:::-;7916:63;;7871:118;8028:2;8054:53;8099:7;8090:6;8079:9;8075:22;8054:53;:::i;:::-;8044:63;;7999:118;8156:2;8182:53;8227:7;8218:6;8207:9;8203:22;8182:53;:::i;:::-;8172:63;;8127:118;8284:3;8311:51;8354:7;8345:6;8334:9;8330:22;8311:51;:::i;:::-;8301:61;;8255:117;8411:3;8438:53;8483:7;8474:6;8463:9;8459:22;8438:53;:::i;:::-;8428:63;;8382:119;8540:3;8567:53;8612:7;8603:6;8592:9;8588:22;8567:53;:::i;:::-;8557:63;;8511:119;7438:1199;;;;;;;;;;:::o;8643:474::-;8711:6;8719;8768:2;8756:9;8747:7;8743:23;8739:32;8736:119;;;8774:79;;:::i;:::-;8736:119;8894:1;8919:53;8964:7;8955:6;8944:9;8940:22;8919:53;:::i;:::-;8909:63;;8865:117;9021:2;9047:53;9092:7;9083:6;9072:9;9068:22;9047:53;:::i;:::-;9037:63;;8992:118;8643:474;;;;;:::o;9123:180::-;9171:77;9168:1;9161:88;9268:4;9265:1;9258:15;9292:4;9289:1;9282:15;9309:320;9353:6;9390:1;9384:4;9380:12;9370:22;;9437:1;9431:4;9427:12;9458:18;9448:81;;9514:4;9506:6;9502:17;9492:27;;9448:81;9576:2;9568:6;9565:14;9545:18;9542:38;9539:84;;9595:18;;:::i;:::-;9539:84;9360:269;9309:320;;;:::o;9635:180::-;9683:77;9680:1;9673:88;9780:4;9777:1;9770:15;9804:4;9801:1;9794:15;9821:191;9861:3;9880:20;9898:1;9880:20;:::i;:::-;9875:25;;9914:20;9932:1;9914:20;:::i;:::-;9909:25;;9957:1;9954;9950:9;9943:16;;9978:3;9975:1;9972:10;9969:36;;;9985:18;;:::i;:::-;9969:36;9821:191;;;;:::o;10018:224::-;10158:34;10154:1;10146:6;10142:14;10135:58;10227:7;10222:2;10214:6;10210:15;10203:32;10018:224;:::o;10248:366::-;10390:3;10411:67;10475:2;10470:3;10411:67;:::i;:::-;10404:74;;10487:93;10576:3;10487:93;:::i;:::-;10605:2;10600:3;10596:12;10589:19;;10248:366;;;:::o;10620:419::-;10786:4;10824:2;10813:9;10809:18;10801:26;;10873:9;10867:4;10863:20;10859:1;10848:9;10844:17;10837:47;10901:131;11027:4;10901:131;:::i;:::-;10893:139;;10620:419;;;:::o;11045:179::-;11185:31;11181:1;11173:6;11169:14;11162:55;11045:179;:::o;11230:366::-;11372:3;11393:67;11457:2;11452:3;11393:67;:::i;:::-;11386:74;;11469:93;11558:3;11469:93;:::i;:::-;11587:2;11582:3;11578:12;11571:19;;11230:366;;;:::o;11602:419::-;11768:4;11806:2;11795:9;11791:18;11783:26;;11855:9;11849:4;11845:20;11841:1;11830:9;11826:17;11819:47;11883:131;12009:4;11883:131;:::i;:::-;11875:139;;11602:419;;;:::o;12027:775::-;12260:4;12298:3;12287:9;12283:19;12275:27;;12312:71;12380:1;12369:9;12365:17;12356:6;12312:71;:::i;:::-;12393:72;12461:2;12450:9;12446:18;12437:6;12393:72;:::i;:::-;12475;12543:2;12532:9;12528:18;12519:6;12475:72;:::i;:::-;12557;12625:2;12614:9;12610:18;12601:6;12557:72;:::i;:::-;12639:73;12707:3;12696:9;12692:19;12683:6;12639:73;:::i;:::-;12722;12790:3;12779:9;12775:19;12766:6;12722:73;:::i;:::-;12027:775;;;;;;;;;:::o;12808:180::-;12948:32;12944:1;12936:6;12932:14;12925:56;12808:180;:::o;12994:366::-;13136:3;13157:67;13221:2;13216:3;13157:67;:::i;:::-;13150:74;;13233:93;13322:3;13233:93;:::i;:::-;13351:2;13346:3;13342:12;13335:19;;12994:366;;;:::o;13366:419::-;13532:4;13570:2;13559:9;13555:18;13547:26;;13619:9;13613:4;13609:20;13605:1;13594:9;13590:17;13583:47;13647:131;13773:4;13647:131;:::i;:::-;13639:139;;13366:419;;;:::o;13791:225::-;13931:34;13927:1;13919:6;13915:14;13908:58;14000:8;13995:2;13987:6;13983:15;13976:33;13791:225;:::o;14022:366::-;14164:3;14185:67;14249:2;14244:3;14185:67;:::i;:::-;14178:74;;14261:93;14350:3;14261:93;:::i;:::-;14379:2;14374:3;14370:12;14363:19;;14022:366;;;:::o;14394:419::-;14560:4;14598:2;14587:9;14583:18;14575:26;;14647:9;14641:4;14637:20;14633:1;14622:9;14618:17;14611:47;14675:131;14801:4;14675:131;:::i;:::-;14667:139;;14394:419;;;:::o;14819:223::-;14959:34;14955:1;14947:6;14943:14;14936:58;15028:6;15023:2;15015:6;15011:15;15004:31;14819:223;:::o;15048:366::-;15190:3;15211:67;15275:2;15270:3;15211:67;:::i;:::-;15204:74;;15287:93;15376:3;15287:93;:::i;:::-;15405:2;15400:3;15396:12;15389:19;;15048:366;;;:::o;15420:419::-;15586:4;15624:2;15613:9;15609:18;15601:26;;15673:9;15667:4;15663:20;15659:1;15648:9;15644:17;15637:47;15701:131;15827:4;15701:131;:::i;:::-;15693:139;;15420:419;;;:::o;15845:221::-;15985:34;15981:1;15973:6;15969:14;15962:58;16054:4;16049:2;16041:6;16037:15;16030:29;15845:221;:::o;16072:366::-;16214:3;16235:67;16299:2;16294:3;16235:67;:::i;:::-;16228:74;;16311:93;16400:3;16311:93;:::i;:::-;16429:2;16424:3;16420:12;16413:19;;16072:366;;;:::o;16444:419::-;16610:4;16648:2;16637:9;16633:18;16625:26;;16697:9;16691:4;16687:20;16683:1;16672:9;16668:17;16661:47;16725:131;16851:4;16725:131;:::i;:::-;16717:139;;16444:419;;;:::o;16869:182::-;17009:34;17005:1;16997:6;16993:14;16986:58;16869:182;:::o;17057:366::-;17199:3;17220:67;17284:2;17279:3;17220:67;:::i;:::-;17213:74;;17296:93;17385:3;17296:93;:::i;:::-;17414:2;17409:3;17405:12;17398:19;;17057:366;;;:::o;17429:419::-;17595:4;17633:2;17622:9;17618:18;17610:26;;17682:9;17676:4;17672:20;17668:1;17657:9;17653:17;17646:47;17710:131;17836:4;17710:131;:::i;:::-;17702:139;;17429:419;;;:::o;17854:179::-;17994:31;17990:1;17982:6;17978:14;17971:55;17854:179;:::o;18039:366::-;18181:3;18202:67;18266:2;18261:3;18202:67;:::i;:::-;18195:74;;18278:93;18367:3;18278:93;:::i;:::-;18396:2;18391:3;18387:12;18380:19;;18039:366;;;:::o;18411:419::-;18577:4;18615:2;18604:9;18600:18;18592:26;;18664:9;18658:4;18654:20;18650:1;18639:9;18635:17;18628:47;18692:131;18818:4;18692:131;:::i;:::-;18684:139;;18411:419;;;:::o;18836:224::-;18976:34;18972:1;18964:6;18960:14;18953:58;19045:7;19040:2;19032:6;19028:15;19021:32;18836:224;:::o;19066:366::-;19208:3;19229:67;19293:2;19288:3;19229:67;:::i;:::-;19222:74;;19305:93;19394:3;19305:93;:::i;:::-;19423:2;19418:3;19414:12;19407:19;;19066:366;;;:::o;19438:419::-;19604:4;19642:2;19631:9;19627:18;19619:26;;19691:9;19685:4;19681:20;19677:1;19666:9;19662:17;19655:47;19719:131;19845:4;19719:131;:::i;:::-;19711:139;;19438:419;;;:::o;19863:222::-;20003:34;19999:1;19991:6;19987:14;19980:58;20072:5;20067:2;20059:6;20055:15;20048:30;19863:222;:::o;20091:366::-;20233:3;20254:67;20318:2;20313:3;20254:67;:::i;:::-;20247:74;;20330:93;20419:3;20330:93;:::i;:::-;20448:2;20443:3;20439:12;20432:19;;20091:366;;;:::o;20463:419::-;20629:4;20667:2;20656:9;20652:18;20644:26;;20716:9;20710:4;20706:20;20702:1;20691:9;20687:17;20680:47;20744:131;20870:4;20744:131;:::i;:::-;20736:139;;20463:419;;;:::o;20888:225::-;21028:34;21024:1;21016:6;21012:14;21005:58;21097:8;21092:2;21084:6;21080:15;21073:33;20888:225;:::o;21119:366::-;21261:3;21282:67;21346:2;21341:3;21282:67;:::i;:::-;21275:74;;21358:93;21447:3;21358:93;:::i;:::-;21476:2;21471:3;21467:12;21460:19;;21119:366;;;:::o;21491:419::-;21657:4;21695:2;21684:9;21680:18;21672:26;;21744:9;21738:4;21734:20;21730:1;21719:9;21715:17;21708:47;21772:131;21898:4;21772:131;:::i;:::-;21764:139;;21491:419;;;:::o;21916:220::-;22056:34;22052:1;22044:6;22040:14;22033:58;22125:3;22120:2;22112:6;22108:15;22101:28;21916:220;:::o;22142:366::-;22284:3;22305:67;22369:2;22364:3;22305:67;:::i;:::-;22298:74;;22381:93;22470:3;22381:93;:::i;:::-;22499:2;22494:3;22490:12;22483:19;;22142:366;;;:::o;22514:419::-;22680:4;22718:2;22707:9;22703:18;22695:26;;22767:9;22761:4;22757:20;22753:1;22742:9;22738:17;22731:47;22795:131;22921:4;22795:131;:::i;:::-;22787:139;;22514:419;;;:::o;22939:221::-;23079:34;23075:1;23067:6;23063:14;23056:58;23148:4;23143:2;23135:6;23131:15;23124:29;22939:221;:::o;23166:366::-;23308:3;23329:67;23393:2;23388:3;23329:67;:::i;:::-;23322:74;;23405:93;23494:3;23405:93;:::i;:::-;23523:2;23518:3;23514:12;23507:19;;23166:366;;;:::o;23538:419::-;23704:4;23742:2;23731:9;23727:18;23719:26;;23791:9;23785:4;23781:20;23777:1;23766:9;23762:17;23755:47;23819:131;23945:4;23819:131;:::i;:::-;23811:139;;23538:419;;;:::o;23963:158::-;24103:10;24099:1;24091:6;24087:14;24080:34;23963:158;:::o;24127:365::-;24269:3;24290:66;24354:1;24349:3;24290:66;:::i;:::-;24283:73;;24365:93;24454:3;24365:93;:::i;:::-;24483:2;24478:3;24474:12;24467:19;;24127:365;;;:::o;24498:419::-;24664:4;24702:2;24691:9;24687:18;24679:26;;24751:9;24745:4;24741:20;24737:1;24726:9;24722:17;24715:47;24779:131;24905:4;24779:131;:::i;:::-;24771:139;;24498:419;;;:::o;24923:664::-;25128:4;25166:3;25155:9;25151:19;25143:27;;25180:71;25248:1;25237:9;25233:17;25224:6;25180:71;:::i;:::-;25261:72;25329:2;25318:9;25314:18;25305:6;25261:72;:::i;:::-;25343;25411:2;25400:9;25396:18;25387:6;25343:72;:::i;:::-;25425;25493:2;25482:9;25478:18;25469:6;25425:72;:::i;:::-;25507:73;25575:3;25564:9;25560:19;25551:6;25507:73;:::i;:::-;24923:664;;;;;;;;:::o;25593:148::-;25695:11;25732:3;25717:18;;25593:148;;;;:::o;25747:214::-;25887:66;25883:1;25875:6;25871:14;25864:90;25747:214;:::o;25967:400::-;26127:3;26148:84;26230:1;26225:3;26148:84;:::i;:::-;26141:91;;26241:93;26330:3;26241:93;:::i;:::-;26359:1;26354:3;26350:11;26343:18;;25967:400;;;:::o;26373:79::-;26412:7;26441:5;26430:16;;26373:79;;;:::o;26458:157::-;26563:45;26583:24;26601:5;26583:24;:::i;:::-;26563:45;:::i;:::-;26558:3;26551:58;26458:157;;:::o;26621:663::-;26862:3;26884:148;27028:3;26884:148;:::i;:::-;26877:155;;27042:75;27113:3;27104:6;27042:75;:::i;:::-;27142:2;27137:3;27133:12;27126:19;;27155:75;27226:3;27217:6;27155:75;:::i;:::-;27255:2;27250:3;27246:12;27239:19;;27275:3;27268:10;;26621:663;;;;;:::o;27290:545::-;27463:4;27501:3;27490:9;27486:19;27478:27;;27515:71;27583:1;27572:9;27568:17;27559:6;27515:71;:::i;:::-;27596:68;27660:2;27649:9;27645:18;27636:6;27596:68;:::i;:::-;27674:72;27742:2;27731:9;27727:18;27718:6;27674:72;:::i;:::-;27756;27824:2;27813:9;27809:18;27800:6;27756:72;:::i;:::-;27290:545;;;;;;;:::o;27841:180::-;27889:77;27886:1;27879:88;27986:4;27983:1;27976:15;28010:4;28007:1;28000:15;28027:174;28167:26;28163:1;28155:6;28151:14;28144:50;28027:174;:::o;28207:366::-;28349:3;28370:67;28434:2;28429:3;28370:67;:::i;:::-;28363:74;;28446:93;28535:3;28446:93;:::i;:::-;28564:2;28559:3;28555:12;28548:19;;28207:366;;;:::o;28579:419::-;28745:4;28783:2;28772:9;28768:18;28760:26;;28832:9;28826:4;28822:20;28818:1;28807:9;28803:17;28796:47;28860:131;28986:4;28860:131;:::i;:::-;28852:139;;28579:419;;;:::o;29004:181::-;29144:33;29140:1;29132:6;29128:14;29121:57;29004:181;:::o;29191:366::-;29333:3;29354:67;29418:2;29413:3;29354:67;:::i;:::-;29347:74;;29430:93;29519:3;29430:93;:::i;:::-;29548:2;29543:3;29539:12;29532:19;;29191:366;;;:::o;29563:419::-;29729:4;29767:2;29756:9;29752:18;29744:26;;29816:9;29810:4;29806:20;29802:1;29791:9;29787:17;29780:47;29844:131;29970:4;29844:131;:::i;:::-;29836:139;;29563:419;;;:::o;29988:221::-;30128:34;30124:1;30116:6;30112:14;30105:58;30197:4;30192:2;30184:6;30180:15;30173:29;29988:221;:::o;30215:366::-;30357:3;30378:67;30442:2;30437:3;30378:67;:::i;:::-;30371:74;;30454:93;30543:3;30454:93;:::i;:::-;30572:2;30567:3;30563:12;30556:19;;30215:366;;;:::o;30587:419::-;30753:4;30791:2;30780:9;30776:18;30768:26;;30840:9;30834:4;30830:20;30826:1;30815:9;30811:17;30804:47;30868:131;30994:4;30868:131;:::i;:::-;30860:139;;30587:419;;;:::o

Swarm Source

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