ETH Price: $3,386.02 (-1.50%)
Gas: 2 Gwei

Token

Hacker Frens (HACKERFRENS)
 

Overview

Max Total Supply

617 HACKERFRENS

Holders

173

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Balance
1 HACKERFRENS
0x89914e35a8f5fd735f4245037ff67f4775605bf1
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:
HackerFrensContractNew

Compiler Version
v0.7.6+commit.7338295f

Optimization Enabled:
Yes with 200 runs

Other Settings:
default evmVersion, None license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2022-02-11
*/

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

pragma solidity >=0.6.0 <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 GSN 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 payable) {
        return msg.sender;
    }

    function _msgData() internal view virtual returns (bytes memory) {
        this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
        return msg.data;
    }
}

// File: @openzeppelin/contracts/introspection/IERC165.sol



pragma solidity >=0.6.0 <0.8.0;

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

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



pragma solidity >=0.6.2 <0.8.0;


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

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

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

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

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

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

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

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

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

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

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

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

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



pragma solidity >=0.6.2 <0.8.0;


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

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

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

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

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



pragma solidity >=0.6.2 <0.8.0;


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

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

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

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

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



pragma solidity >=0.6.0 <0.8.0;

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

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



pragma solidity >=0.6.0 <0.8.0;


/**
 * @dev Implementation of the {IERC165} interface.
 *
 * Contracts may inherit from this and call {_registerInterface} to declare
 * their support of an interface.
 */
abstract contract ERC165 is IERC165 {
    /*
     * bytes4(keccak256('supportsInterface(bytes4)')) == 0x01ffc9a7
     */
    bytes4 private constant _INTERFACE_ID_ERC165 = 0x01ffc9a7;

    /**
     * @dev Mapping of interface ids to whether or not it's supported.
     */
    mapping(bytes4 => bool) private _supportedInterfaces;

    constructor () internal {
        // Derived contracts need only register support for their own interfaces,
        // we register support for ERC165 itself here
        _registerInterface(_INTERFACE_ID_ERC165);
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     *
     * Time complexity O(1), guaranteed to always use less than 30 000 gas.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        return _supportedInterfaces[interfaceId];
    }

    /**
     * @dev Registers the contract as an implementer of the interface defined by
     * `interfaceId`. Support of the actual ERC165 interface is automatic and
     * registering its interface id is not required.
     *
     * See {IERC165-supportsInterface}.
     *
     * Requirements:
     *
     * - `interfaceId` cannot be the ERC165 invalid interface (`0xffffffff`).
     */
    function _registerInterface(bytes4 interfaceId) internal virtual {
        require(interfaceId != 0xffffffff, "ERC165: invalid interface id");
        _supportedInterfaces[interfaceId] = true;
    }
}

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



pragma solidity >=0.6.0 <0.8.0;

/**
 * @dev Wrappers over Solidity's arithmetic operations with added overflow
 * checks.
 *
 * Arithmetic operations in Solidity wrap on overflow. This can easily result
 * in bugs, because programmers usually assume that an overflow raises an
 * error, which is the standard behavior in high level programming languages.
 * `SafeMath` restores this intuition by reverting the transaction when an
 * operation overflows.
 *
 * Using this library instead of the unchecked operations eliminates an entire
 * class of bugs, so it's recommended to use it always.
 */
library SafeMath {
    /**
     * @dev Returns the addition of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        uint256 c = a + b;
        if (c < a) return (false, 0);
        return (true, c);
    }

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

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

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

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

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

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

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

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

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

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

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

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

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



pragma solidity >=0.6.2 <0.8.0;

/**
 * @dev Collection of functions related to the address type
 */
library Address {
    /**
     * @dev Returns true if `account` is a contract.
     *
     * [IMPORTANT]
     * ====
     * It is unsafe to assume that an address for which this function returns
     * false is an externally-owned account (EOA) and not a contract.
     *
     * Among others, `isContract` will return false for the following
     * types of addresses:
     *
     *  - an externally-owned account
     *  - a contract in construction
     *  - an address where a contract will be created
     *  - an address where a contract lived, but was destroyed
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize, which returns 0 for contracts in
        // construction, since the code is only stored at the end of the
        // constructor execution.

        uint256 size;
        // solhint-disable-next-line no-inline-assembly
        assembly { size := extcodesize(account) }
        return size > 0;
    }

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

        // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
        (bool success, ) = recipient.call{ value: amount }("");
        require(success, "Address: unable to send value, recipient may have reverted");
    }

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

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

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

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

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.call{ value: value }(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

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

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

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.staticcall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

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

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

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.delegatecall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) {
        if (success) {
            return returndata;
        } else {
            // Look for revert reason and bubble it up if present
            if (returndata.length > 0) {
                // The easiest way to bubble the revert reason is using memory via assembly

                // solhint-disable-next-line no-inline-assembly
                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}

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



pragma solidity >=0.6.0 <0.8.0;

/**
 * @dev Library for managing
 * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive
 * types.
 *
 * Sets have the following properties:
 *
 * - Elements are added, removed, and checked for existence in constant time
 * (O(1)).
 * - Elements are enumerated in O(n). No guarantees are made on the ordering.
 *
 * ```
 * contract Example {
 *     // Add the library methods
 *     using EnumerableSet for EnumerableSet.AddressSet;
 *
 *     // Declare a set state variable
 *     EnumerableSet.AddressSet private mySet;
 * }
 * ```
 *
 * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`)
 * and `uint256` (`UintSet`) are supported.
 */
library EnumerableSet {
    // To implement this library for multiple types with as little code
    // repetition as possible, we write it in terms of a generic Set type with
    // bytes32 values.
    // The Set implementation uses private functions, and user-facing
    // implementations (such as AddressSet) are just wrappers around the
    // underlying Set.
    // This means that we can only create new EnumerableSets for types that fit
    // in bytes32.

    struct Set {
        // Storage of set values
        bytes32[] _values;

        // Position of the value in the `values` array, plus 1 because index 0
        // means a value is not in the set.
        mapping (bytes32 => uint256) _indexes;
    }

    /**
     * @dev Add a value to a set. O(1).
     *
     * Returns true if the value was added to the set, that is if it was not
     * already present.
     */
    function _add(Set storage set, bytes32 value) private returns (bool) {
        if (!_contains(set, value)) {
            set._values.push(value);
            // The value is stored at length-1, but we add 1 to all indexes
            // and use 0 as a sentinel value
            set._indexes[value] = set._values.length;
            return true;
        } else {
            return false;
        }
    }

    /**
     * @dev Removes a value from a set. O(1).
     *
     * Returns true if the value was removed from the set, that is if it was
     * present.
     */
    function _remove(Set storage set, bytes32 value) private returns (bool) {
        // We read and store the value's index to prevent multiple reads from the same storage slot
        uint256 valueIndex = set._indexes[value];

        if (valueIndex != 0) { // Equivalent to contains(set, value)
            // To delete an element from the _values array in O(1), we swap the element to delete with the last one in
            // the array, and then remove the last element (sometimes called as 'swap and pop').
            // This modifies the order of the array, as noted in {at}.

            uint256 toDeleteIndex = valueIndex - 1;
            uint256 lastIndex = set._values.length - 1;

            // When the value to delete is the last one, the swap operation is unnecessary. However, since this occurs
            // so rarely, we still do the swap anyway to avoid the gas cost of adding an 'if' statement.

            bytes32 lastvalue = set._values[lastIndex];

            // Move the last value to the index where the value to delete is
            set._values[toDeleteIndex] = lastvalue;
            // Update the index for the moved value
            set._indexes[lastvalue] = toDeleteIndex + 1; // All indexes are 1-based

            // Delete the slot where the moved value was stored
            set._values.pop();

            // Delete the index for the deleted slot
            delete set._indexes[value];

            return true;
        } else {
            return false;
        }
    }

    /**
     * @dev Returns true if the value is in the set. O(1).
     */
    function _contains(Set storage set, bytes32 value) private view returns (bool) {
        return set._indexes[value] != 0;
    }

    /**
     * @dev Returns the number of values on the set. O(1).
     */
    function _length(Set storage set) private view returns (uint256) {
        return set._values.length;
    }

   /**
    * @dev Returns the value stored at position `index` in the set. O(1).
    *
    * Note that there are no guarantees on the ordering of values inside the
    * array, and it may change when more values are added or removed.
    *
    * Requirements:
    *
    * - `index` must be strictly less than {length}.
    */
    function _at(Set storage set, uint256 index) private view returns (bytes32) {
        require(set._values.length > index, "EnumerableSet: index out of bounds");
        return set._values[index];
    }

    // Bytes32Set

    struct Bytes32Set {
        Set _inner;
    }

    /**
     * @dev Add a value to a set. O(1).
     *
     * Returns true if the value was added to the set, that is if it was not
     * already present.
     */
    function add(Bytes32Set storage set, bytes32 value) internal returns (bool) {
        return _add(set._inner, value);
    }

    /**
     * @dev Removes a value from a set. O(1).
     *
     * Returns true if the value was removed from the set, that is if it was
     * present.
     */
    function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) {
        return _remove(set._inner, value);
    }

    /**
     * @dev Returns true if the value is in the set. O(1).
     */
    function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) {
        return _contains(set._inner, value);
    }

    /**
     * @dev Returns the number of values in the set. O(1).
     */
    function length(Bytes32Set storage set) internal view returns (uint256) {
        return _length(set._inner);
    }

   /**
    * @dev Returns the value stored at position `index` in the set. O(1).
    *
    * Note that there are no guarantees on the ordering of values inside the
    * array, and it may change when more values are added or removed.
    *
    * Requirements:
    *
    * - `index` must be strictly less than {length}.
    */
    function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) {
        return _at(set._inner, index);
    }

    // AddressSet

    struct AddressSet {
        Set _inner;
    }

    /**
     * @dev Add a value to a set. O(1).
     *
     * Returns true if the value was added to the set, that is if it was not
     * already present.
     */
    function add(AddressSet storage set, address value) internal returns (bool) {
        return _add(set._inner, bytes32(uint256(uint160(value))));
    }

    /**
     * @dev Removes a value from a set. O(1).
     *
     * Returns true if the value was removed from the set, that is if it was
     * present.
     */
    function remove(AddressSet storage set, address value) internal returns (bool) {
        return _remove(set._inner, bytes32(uint256(uint160(value))));
    }

    /**
     * @dev Returns true if the value is in the set. O(1).
     */
    function contains(AddressSet storage set, address value) internal view returns (bool) {
        return _contains(set._inner, bytes32(uint256(uint160(value))));
    }

    /**
     * @dev Returns the number of values in the set. O(1).
     */
    function length(AddressSet storage set) internal view returns (uint256) {
        return _length(set._inner);
    }

   /**
    * @dev Returns the value stored at position `index` in the set. O(1).
    *
    * Note that there are no guarantees on the ordering of values inside the
    * array, and it may change when more values are added or removed.
    *
    * Requirements:
    *
    * - `index` must be strictly less than {length}.
    */
    function at(AddressSet storage set, uint256 index) internal view returns (address) {
        return address(uint160(uint256(_at(set._inner, index))));
    }


    // UintSet

    struct UintSet {
        Set _inner;
    }

    /**
     * @dev Add a value to a set. O(1).
     *
     * Returns true if the value was added to the set, that is if it was not
     * already present.
     */
    function add(UintSet storage set, uint256 value) internal returns (bool) {
        return _add(set._inner, bytes32(value));
    }

    /**
     * @dev Removes a value from a set. O(1).
     *
     * Returns true if the value was removed from the set, that is if it was
     * present.
     */
    function remove(UintSet storage set, uint256 value) internal returns (bool) {
        return _remove(set._inner, bytes32(value));
    }

    /**
     * @dev Returns true if the value is in the set. O(1).
     */
    function contains(UintSet storage set, uint256 value) internal view returns (bool) {
        return _contains(set._inner, bytes32(value));
    }

    /**
     * @dev Returns the number of values on the set. O(1).
     */
    function length(UintSet storage set) internal view returns (uint256) {
        return _length(set._inner);
    }

   /**
    * @dev Returns the value stored at position `index` in the set. O(1).
    *
    * Note that there are no guarantees on the ordering of values inside the
    * array, and it may change when more values are added or removed.
    *
    * Requirements:
    *
    * - `index` must be strictly less than {length}.
    */
    function at(UintSet storage set, uint256 index) internal view returns (uint256) {
        return uint256(_at(set._inner, index));
    }
}

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



pragma solidity >=0.6.0 <0.8.0;

/**
 * @dev Library for managing an enumerable variant of Solidity's
 * https://solidity.readthedocs.io/en/latest/types.html#mapping-types[`mapping`]
 * type.
 *
 * Maps have the following properties:
 *
 * - Entries are added, removed, and checked for existence in constant time
 * (O(1)).
 * - Entries are enumerated in O(n). No guarantees are made on the ordering.
 *
 * ```
 * contract Example {
 *     // Add the library methods
 *     using EnumerableMap for EnumerableMap.UintToAddressMap;
 *
 *     // Declare a set state variable
 *     EnumerableMap.UintToAddressMap private myMap;
 * }
 * ```
 *
 * As of v3.0.0, only maps of type `uint256 -> address` (`UintToAddressMap`) are
 * supported.
 */
library EnumerableMap {
    // To implement this library for multiple types with as little code
    // repetition as possible, we write it in terms of a generic Map type with
    // bytes32 keys and values.
    // The Map implementation uses private functions, and user-facing
    // implementations (such as Uint256ToAddressMap) are just wrappers around
    // the underlying Map.
    // This means that we can only create new EnumerableMaps for types that fit
    // in bytes32.

    struct MapEntry {
        bytes32 _key;
        bytes32 _value;
    }

    struct Map {
        // Storage of map keys and values
        MapEntry[] _entries;

        // Position of the entry defined by a key in the `entries` array, plus 1
        // because index 0 means a key is not in the map.
        mapping (bytes32 => uint256) _indexes;
    }

    /**
     * @dev Adds a key-value pair to a map, or updates the value for an existing
     * key. O(1).
     *
     * Returns true if the key was added to the map, that is if it was not
     * already present.
     */
    function _set(Map storage map, bytes32 key, bytes32 value) private returns (bool) {
        // We read and store the key's index to prevent multiple reads from the same storage slot
        uint256 keyIndex = map._indexes[key];

        if (keyIndex == 0) { // Equivalent to !contains(map, key)
            map._entries.push(MapEntry({ _key: key, _value: value }));
            // The entry is stored at length-1, but we add 1 to all indexes
            // and use 0 as a sentinel value
            map._indexes[key] = map._entries.length;
            return true;
        } else {
            map._entries[keyIndex - 1]._value = value;
            return false;
        }
    }

    /**
     * @dev Removes a key-value pair from a map. O(1).
     *
     * Returns true if the key was removed from the map, that is if it was present.
     */
    function _remove(Map storage map, bytes32 key) private returns (bool) {
        // We read and store the key's index to prevent multiple reads from the same storage slot
        uint256 keyIndex = map._indexes[key];

        if (keyIndex != 0) { // Equivalent to contains(map, key)
            // To delete a key-value pair from the _entries array in O(1), we swap the entry to delete with the last one
            // in the array, and then remove the last entry (sometimes called as 'swap and pop').
            // This modifies the order of the array, as noted in {at}.

            uint256 toDeleteIndex = keyIndex - 1;
            uint256 lastIndex = map._entries.length - 1;

            // When the entry to delete is the last one, the swap operation is unnecessary. However, since this occurs
            // so rarely, we still do the swap anyway to avoid the gas cost of adding an 'if' statement.

            MapEntry storage lastEntry = map._entries[lastIndex];

            // Move the last entry to the index where the entry to delete is
            map._entries[toDeleteIndex] = lastEntry;
            // Update the index for the moved entry
            map._indexes[lastEntry._key] = toDeleteIndex + 1; // All indexes are 1-based

            // Delete the slot where the moved entry was stored
            map._entries.pop();

            // Delete the index for the deleted slot
            delete map._indexes[key];

            return true;
        } else {
            return false;
        }
    }

    /**
     * @dev Returns true if the key is in the map. O(1).
     */
    function _contains(Map storage map, bytes32 key) private view returns (bool) {
        return map._indexes[key] != 0;
    }

    /**
     * @dev Returns the number of key-value pairs in the map. O(1).
     */
    function _length(Map storage map) private view returns (uint256) {
        return map._entries.length;
    }

   /**
    * @dev Returns the key-value pair stored at position `index` in the map. O(1).
    *
    * Note that there are no guarantees on the ordering of entries inside the
    * array, and it may change when more entries are added or removed.
    *
    * Requirements:
    *
    * - `index` must be strictly less than {length}.
    */
    function _at(Map storage map, uint256 index) private view returns (bytes32, bytes32) {
        require(map._entries.length > index, "EnumerableMap: index out of bounds");

        MapEntry storage entry = map._entries[index];
        return (entry._key, entry._value);
    }

    /**
     * @dev Tries to returns the value associated with `key`.  O(1).
     * Does not revert if `key` is not in the map.
     */
    function _tryGet(Map storage map, bytes32 key) private view returns (bool, bytes32) {
        uint256 keyIndex = map._indexes[key];
        if (keyIndex == 0) return (false, 0); // Equivalent to contains(map, key)
        return (true, map._entries[keyIndex - 1]._value); // All indexes are 1-based
    }

    /**
     * @dev Returns the value associated with `key`.  O(1).
     *
     * Requirements:
     *
     * - `key` must be in the map.
     */
    function _get(Map storage map, bytes32 key) private view returns (bytes32) {
        uint256 keyIndex = map._indexes[key];
        require(keyIndex != 0, "EnumerableMap: nonexistent key"); // Equivalent to contains(map, key)
        return map._entries[keyIndex - 1]._value; // All indexes are 1-based
    }

    /**
     * @dev Same as {_get}, with a custom error message when `key` is not in the map.
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {_tryGet}.
     */
    function _get(Map storage map, bytes32 key, string memory errorMessage) private view returns (bytes32) {
        uint256 keyIndex = map._indexes[key];
        require(keyIndex != 0, errorMessage); // Equivalent to contains(map, key)
        return map._entries[keyIndex - 1]._value; // All indexes are 1-based
    }

    // UintToAddressMap

    struct UintToAddressMap {
        Map _inner;
    }

    /**
     * @dev Adds a key-value pair to a map, or updates the value for an existing
     * key. O(1).
     *
     * Returns true if the key was added to the map, that is if it was not
     * already present.
     */
    function set(UintToAddressMap storage map, uint256 key, address value) internal returns (bool) {
        return _set(map._inner, bytes32(key), bytes32(uint256(uint160(value))));
    }

    /**
     * @dev Removes a value from a set. O(1).
     *
     * Returns true if the key was removed from the map, that is if it was present.
     */
    function remove(UintToAddressMap storage map, uint256 key) internal returns (bool) {
        return _remove(map._inner, bytes32(key));
    }

    /**
     * @dev Returns true if the key is in the map. O(1).
     */
    function contains(UintToAddressMap storage map, uint256 key) internal view returns (bool) {
        return _contains(map._inner, bytes32(key));
    }

    /**
     * @dev Returns the number of elements in the map. O(1).
     */
    function length(UintToAddressMap storage map) internal view returns (uint256) {
        return _length(map._inner);
    }

   /**
    * @dev Returns the element stored at position `index` in the set. O(1).
    * Note that there are no guarantees on the ordering of values inside the
    * array, and it may change when more values are added or removed.
    *
    * Requirements:
    *
    * - `index` must be strictly less than {length}.
    */
    function at(UintToAddressMap storage map, uint256 index) internal view returns (uint256, address) {
        (bytes32 key, bytes32 value) = _at(map._inner, index);
        return (uint256(key), address(uint160(uint256(value))));
    }

    /**
     * @dev Tries to returns the value associated with `key`.  O(1).
     * Does not revert if `key` is not in the map.
     *
     * _Available since v3.4._
     */
    function tryGet(UintToAddressMap storage map, uint256 key) internal view returns (bool, address) {
        (bool success, bytes32 value) = _tryGet(map._inner, bytes32(key));
        return (success, address(uint160(uint256(value))));
    }

    /**
     * @dev Returns the value associated with `key`.  O(1).
     *
     * Requirements:
     *
     * - `key` must be in the map.
     */
    function get(UintToAddressMap storage map, uint256 key) internal view returns (address) {
        return address(uint160(uint256(_get(map._inner, bytes32(key)))));
    }

    /**
     * @dev Same as {get}, with a custom error message when `key` is not in the map.
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {tryGet}.
     */
    function get(UintToAddressMap storage map, uint256 key, string memory errorMessage) internal view returns (address) {
        return address(uint160(uint256(_get(map._inner, bytes32(key), errorMessage))));
    }
}

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



pragma solidity >=0.6.0 <0.8.0;

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

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

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



pragma solidity >=0.6.0 <0.8.0;

/**
 * @title ERC721 Non-Fungible Token Standard basic implementation
 * @dev see https://eips.ethereum.org/EIPS/eip-721
 */
 
contract ERC721 is Context, ERC165, IERC721, IERC721Metadata, IERC721Enumerable {
    using SafeMath for uint256;
    using Address for address;
    using EnumerableSet for EnumerableSet.UintSet;
    using EnumerableMap for EnumerableMap.UintToAddressMap;
    using Strings for uint256;

    // Equals to `bytes4(keccak256("onERC721Received(address,address,uint256,bytes)"))`
    // which can be also obtained as `IERC721Receiver(0).onERC721Received.selector`
    bytes4 private constant _ERC721_RECEIVED = 0x150b7a02;

    // Mapping from holder address to their (enumerable) set of owned tokens
    mapping (address => EnumerableSet.UintSet) private _holderTokens;

    // Enumerable mapping from token ids to their owners
    EnumerableMap.UintToAddressMap private _tokenOwners;

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

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

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Optional mapping for token URIs
    mapping (uint256 => string) private _tokenURIs;

    // Base URI
    string private _baseURI;

    /*
     *     bytes4(keccak256('balanceOf(address)')) == 0x70a08231
     *     bytes4(keccak256('ownerOf(uint256)')) == 0x6352211e
     *     bytes4(keccak256('approve(address,uint256)')) == 0x095ea7b3
     *     bytes4(keccak256('getApproved(uint256)')) == 0x081812fc
     *     bytes4(keccak256('setApprovalForAll(address,bool)')) == 0xa22cb465
     *     bytes4(keccak256('isApprovedForAll(address,address)')) == 0xe985e9c5
     *     bytes4(keccak256('transferFrom(address,address,uint256)')) == 0x23b872dd
     *     bytes4(keccak256('safeTransferFrom(address,address,uint256)')) == 0x42842e0e
     *     bytes4(keccak256('safeTransferFrom(address,address,uint256,bytes)')) == 0xb88d4fde
     *
     *     => 0x70a08231 ^ 0x6352211e ^ 0x095ea7b3 ^ 0x081812fc ^
     *        0xa22cb465 ^ 0xe985e9c5 ^ 0x23b872dd ^ 0x42842e0e ^ 0xb88d4fde == 0x80ac58cd
     */
    bytes4 private constant _INTERFACE_ID_ERC721 = 0x80ac58cd;

    /*
     *     bytes4(keccak256('name()')) == 0x06fdde03
     *     bytes4(keccak256('symbol()')) == 0x95d89b41
     *     bytes4(keccak256('tokenURI(uint256)')) == 0xc87b56dd
     *
     *     => 0x06fdde03 ^ 0x95d89b41 ^ 0xc87b56dd == 0x5b5e139f
     */
    bytes4 private constant _INTERFACE_ID_ERC721_METADATA = 0x5b5e139f;

    /*
     *     bytes4(keccak256('totalSupply()')) == 0x18160ddd
     *     bytes4(keccak256('tokenOfOwnerByIndex(address,uint256)')) == 0x2f745c59
     *     bytes4(keccak256('tokenByIndex(uint256)')) == 0x4f6ccce7
     *
     *     => 0x18160ddd ^ 0x2f745c59 ^ 0x4f6ccce7 == 0x780e9d63
     */
    bytes4 private constant _INTERFACE_ID_ERC721_ENUMERABLE = 0x780e9d63;

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

        // register the supported interfaces to conform to ERC721 via ERC165
        _registerInterface(_INTERFACE_ID_ERC721);
        _registerInterface(_INTERFACE_ID_ERC721_METADATA);
        _registerInterface(_INTERFACE_ID_ERC721_ENUMERABLE);
    }

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

    /**
     * @dev See {IERC721-ownerOf}.
     */
    function ownerOf(uint256 tokenId) public view virtual override returns (address) {
        return _tokenOwners.get(tokenId, "ERC721: owner query for nonexistent token");
    }

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

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

    /**
     * @dev See {IERC721Metadata-tokenURI}.
     */
    function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
        require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");

        string memory _tokenURI = _tokenURIs[tokenId];
        string memory base = baseURI();

        // If there is no base URI, return the token URI.
        if (bytes(base).length == 0) {
            return _tokenURI;
        }
        // If both are set, concatenate the baseURI and tokenURI (via abi.encodePacked).
        if (bytes(_tokenURI).length > 0) {
            return string(abi.encodePacked(base, _tokenURI));
        }
        // If there is a baseURI but no tokenURI, concatenate the tokenID to the baseURI.
        return string(abi.encodePacked(base, tokenId.toString()));
    }

    /**
    * @dev Returns the base URI set via {_setBaseURI}. This will be
    * automatically added as a prefix in {tokenURI} to each token's URI, or
    * to the token ID if no specific URI is set for that token ID.
    */
    function baseURI() public view virtual returns (string memory) {
        return _baseURI;
    }

    /**
     * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) {
        return _holderTokens[owner].at(index);
    }

    /**
     * @dev See {IERC721Enumerable-totalSupply}.
     */
    function totalSupply() public view virtual override returns (uint256) {
        // _tokenOwners are indexed by tokenIds, so .length() returns the number of tokenIds
        return _tokenOwners.length();
    }

    /**
     * @dev See {IERC721Enumerable-tokenByIndex}.
     */
    function tokenByIndex(uint256 index) public view virtual override returns (uint256) {
        (uint256 tokenId, ) = _tokenOwners.at(index);
        return tokenId;
    }

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

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

        _approve(to, tokenId);
    }

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

        return _tokenApprovals[tokenId];
    }

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

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

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

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

        _transfer(from, to, tokenId);
    }

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

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

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

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

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

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

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

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

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

        _holderTokens[to].add(tokenId);

        _tokenOwners.set(tokenId, to);

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

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

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

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

        // Clear metadata (if any)
        if (bytes(_tokenURIs[tokenId]).length != 0) {
            delete _tokenURIs[tokenId];
        }

        _holderTokens[owner].remove(tokenId);

        _tokenOwners.remove(tokenId);

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

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

        _beforeTokenTransfer(from, to, tokenId);

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

        _holderTokens[from].remove(tokenId);
        _holderTokens[to].add(tokenId);

        _tokenOwners.set(tokenId, to);

        emit Transfer(from, to, tokenId);
    }

    /**
     * @dev Sets `_tokenURI` as the tokenURI of `tokenId`.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function _setTokenURI(uint256 tokenId, string memory _tokenURI) internal virtual {
        require(_exists(tokenId), "ERC721Metadata: URI set of nonexistent token");
        _tokenURIs[tokenId] = _tokenURI;
    }

    /**
     * @dev Internal function to set the base URI for all token IDs. It is
     * automatically added as a prefix to the value returned in {tokenURI},
     * or to the token ID if {tokenURI} is empty.
     */
    function _setBaseURI(string memory baseURI_) internal virtual {
        _baseURI = baseURI_;
    }

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

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

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

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



pragma solidity >=0.6.0 <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 () internal {
        address msgSender = _msgSender();
        _owner = msgSender;
        emit OwnershipTransferred(address(0), msgSender);
    }

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

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

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

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


// I hope you find what you're looking for fren


// yourfren.eth


pragma solidity ^0.7.0;
pragma abicoder v2;

contract HackerFrensContractNew is ERC721, Ownable {
    
    using SafeMath for uint256;
    
    string public LICENSE_TEXT = ""; // IT IS WHAT IT SAYS

    uint256 public constant frenPrice = 35000000000000000; // 0.035 ETH

    uint public constant maxFrenPurchase = 10;

    uint256 public constant MAX_FRENS = 5000;

    bool public saleIsActive = false;
    
    mapping(uint => string) public frenNames;
    
    // Reserve 100 Frens for team - Giveaways/Prizes etc, don't try 100 at a time you'll exceed gas limit
    uint public frenReserve = 100;
    
    event frenNameChange(address _by, uint _tokenId, string _name);

    constructor() ERC721("Hacker Frens", "HACKERFRENS") { }
    
    function withdraw() public onlyOwner {
        uint balance = address(this).balance;
        msg.sender.transfer(balance);
    }
    
    function reserveFRENS(address _to, uint256 _reserveAmount) public onlyOwner {        
        uint supply = totalSupply();
        require(_reserveAmount > 0 && _reserveAmount <= frenReserve, "Not enough reserve left for team");
        for (uint i = 0; i < _reserveAmount; i++) {
            _safeMint(_to, supply + i);
        }
        frenReserve = frenReserve.sub(_reserveAmount);
    }

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


    function flipSaleState() public onlyOwner {
        saleIsActive = !saleIsActive;
    }
    
    
    function tokensOfOwner(address _owner) external view returns(uint256[] memory ) {
        uint256 tokenCount = balanceOf(_owner);
        if (tokenCount == 0) {
            // Return an empty array
            return new uint256[](0);
        } else {
            uint256[] memory result = new uint256[](tokenCount);
            uint256 index;
            for (index = 0; index < tokenCount; index++) {
                result[index] = tokenOfOwnerByIndex(_owner, index);
            }
            return result;
        }
    }
    
    // Returns the license for tokens
    function tokenLicense(uint _id) public view returns(string memory) {
        require(_id < totalSupply(), "CHOOSE A FREN WITHIN RANGE");
        return LICENSE_TEXT;
    }
    
    // Change the license
    function changeLicense(string memory _license) public onlyOwner {
        LICENSE_TEXT = _license;
    }
    
    
    function mintMyFren(uint numberOfTokens) public payable {
        require(saleIsActive, "Sale must be active to mint fren");
        require(numberOfTokens > 0 && numberOfTokens <= maxFrenPurchase, "Can only mint 10 tokens at a time");
        require(totalSupply().add(numberOfTokens) <= MAX_FRENS, "Purchase would exceed max supply of frens");
        require(msg.value >= frenPrice.mul(numberOfTokens), "Ether value sent is not correct");
        
        for(uint i = 0; i < numberOfTokens; i++) {
            uint mintIndex = totalSupply();
            if (totalSupply() < MAX_FRENS) {
                _safeMint(msg.sender, mintIndex);
            }
        }

    }
     
    function changeFrenName(uint _tokenId, string memory _name) public {
        require(ownerOf(_tokenId) == msg.sender, "Hey, your wallet doesn't own this fren!");
        require(sha256(bytes(_name)) != sha256(bytes(frenNames[_tokenId])), "New name is same as the current one");
        frenNames[_tokenId] = _name;
        
        emit frenNameChange(msg.sender, _tokenId, _name);
        
    }
    
    function viewFrenName(uint _tokenId) public view returns( string memory ){
        require( _tokenId < totalSupply(), "Choose a fren within range" );
        return frenNames[_tokenId];
    }
    
    
    // GET ALL FRENS OF A WALLET
    function frenNamesOfOwner(address _owner) external view returns(string[] memory ) {
        uint256 tokenCount = balanceOf(_owner);
        if (tokenCount == 0) {
            // Return an empty array
            return new string[](0);
        } else {
            string[] memory result = new string[](tokenCount);
            uint256 index;
            for (index = 0; index < tokenCount; index++) {
                result[index] = frenNames[ tokenOfOwnerByIndex(_owner, index) ] ;
            }
            return result;
        }
    }
    
}

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_by","type":"address"},{"indexed":false,"internalType":"uint256","name":"_tokenId","type":"uint256"},{"indexed":false,"internalType":"string","name":"_name","type":"string"}],"name":"frenNameChange","type":"event"},{"inputs":[],"name":"LICENSE_TEXT","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_FRENS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"string","name":"_name","type":"string"}],"name":"changeFrenName","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_license","type":"string"}],"name":"changeLicense","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"flipSaleState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"frenNames","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"}],"name":"frenNamesOfOwner","outputs":[{"internalType":"string[]","name":"","type":"string[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"frenPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"frenReserve","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxFrenPurchase","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfTokens","type":"uint256"}],"name":"mintMyFren","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_reserveAmount","type":"uint256"}],"name":"reserveFRENS","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"saleIsActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"baseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"}],"name":"tokenLicense","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"}],"name":"tokensOfOwner","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"viewFrenName","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]

60a06040819052600060808190526200001b91600b91620001b5565b50600c805460ff191690556064600e553480156200003857600080fd5b50604080518082018252600c81526b4861636b6572204672656e7360a01b6020808301919091528251808401909352600b83526a4841434b45524652454e5360a81b9083015290620000916301ffc9a760e01b62000156565b8151620000a6906006906020850190620001b5565b508051620000bc906007906020840190620001b5565b50620000cf6380ac58cd60e01b62000156565b620000e1635b5e139f60e01b62000156565b620000f363780e9d6360e01b62000156565b506000905062000102620001b1565b600a80546001600160a01b0319166001600160a01b038316908117909155604051919250906000907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a35062000298565b6001600160e01b031980821614156200018c5760405162461bcd60e51b8152600401620001839062000261565b60405180910390fd5b6001600160e01b0319166000908152602081905260409020805460ff19166001179055565b3390565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282620001ed576000855562000238565b82601f106200020857805160ff191683800117855562000238565b8280016001018555821562000238579182015b82811115620002385782518255916020019190600101906200021b565b50620002469291506200024a565b5090565b5b808211156200024657600081556001016200024b565b6020808252601c908201527f4552433136353a20696e76616c696420696e7465726661636520696400000000604082015260600190565b612ec980620002a86000396000f3fe6080604052600436106102255760003560e01c80637202be8311610123578063b88d4fde116100ab578063d9b137b21161006f578063d9b137b214610612578063df70f31114610632578063e985e9c514610647578063eb8d244414610667578063f2fde38b1461067c57610225565b8063b88d4fde14610595578063ba90029e146105b5578063c87b56dd146105c8578063ce272ca0146105e8578063d7e38c34146105fd57610225565b80639c3e72bd116100f25780639c3e72bd146104fe578063a22cb46514610513578063af2aec2a14610533578063b09904b514610560578063b3b553ba1461058057610225565b80637202be83146104875780638462151c146104a75780638da5cb5b146104d457806395d89b41146104e957610225565b806334918dfd116101b157806355f804b31161017557806355f804b3146103fd5780636352211e1461041d5780636c0360eb1461043d57806370a0823114610452578063715018a61461047257610225565b806334918dfd146103735780633ccfd60b1461038857806342842e0e1461039d5780634889e50b146103bd5780634f6ccce7146103dd57610225565b80630fdfc028116101f85780630fdfc028146102d157806318160ddd146102f157806323b872dd146103135780632cddaf7f146103335780632f745c591461035357610225565b806301ffc9a71461022a57806306fdde0314610260578063081812fc14610282578063095ea7b3146102af575b600080fd5b34801561023657600080fd5b5061024a610245366004612351565b61069c565b6040516102579190612625565b60405180910390f35b34801561026c57600080fd5b506102756106bf565b6040516102579190612630565b34801561028e57600080fd5b506102a261029d3660046123bc565b610755565b6040516102579190612500565b3480156102bb57600080fd5b506102cf6102ca366004612310565b6107a1565b005b3480156102dd57600080fd5b506102756102ec3660046123bc565b610839565b3480156102fd57600080fd5b50610306610900565b6040516102579190612ded565b34801561031f57600080fd5b506102cf61032e366004612222565b610911565b34801561033f57600080fd5b506102cf61034e366004612310565b610949565b34801561035f57600080fd5b5061030661036e366004612310565b6109f7565b34801561037f57600080fd5b506102cf610a22565b34801561039457600080fd5b506102cf610a75565b3480156103a957600080fd5b506102cf6103b8366004612222565b610ae7565b3480156103c957600080fd5b506102756103d83660046123bc565b610b02565b3480156103e957600080fd5b506103066103f83660046123bc565b610b9d565b34801561040957600080fd5b506102cf610418366004612389565b610bb3565b34801561042957600080fd5b506102a26104383660046123bc565b610bfe565b34801561044957600080fd5b50610275610c26565b34801561045e57600080fd5b5061030661046d3660046121d6565b610c87565b34801561047e57600080fd5b506102cf610cd0565b34801561049357600080fd5b506102cf6104a23660046123d4565b610d59565b3480156104b357600080fd5b506104c76104c23660046121d6565b610eb4565b60405161025791906125e1565b3480156104e057600080fd5b506102a2610f67565b3480156104f557600080fd5b50610275610f76565b34801561050a57600080fd5b50610275610fd7565b34801561051f57600080fd5b506102cf61052e3660046122d6565b611032565b34801561053f57600080fd5b5061055361054e3660046121d6565b611100565b6040516102579190612581565b34801561056c57600080fd5b506102cf61057b366004612389565b611269565b34801561058c57600080fd5b506103066112bb565b3480156105a157600080fd5b506102cf6105b036600461225d565b6112c6565b6102cf6105c33660046123bc565b611305565b3480156105d457600080fd5b506102756105e33660046123bc565b6113f2565b3480156105f457600080fd5b50610306611536565b34801561060957600080fd5b5061030661153c565b34801561061e57600080fd5b5061027561062d3660046123bc565b611542565b34801561063e57600080fd5b506103066115c5565b34801561065357600080fd5b5061024a6106623660046121f0565b6115ca565b34801561067357600080fd5b5061024a6115f8565b34801561068857600080fd5b506102cf6106973660046121d6565b611601565b6001600160e01b0319811660009081526020819052604090205460ff165b919050565b60068054604080516020601f600260001961010060018816150201909516949094049384018190048102820181019092528281526060939092909183018282801561074b5780601f106107205761010080835404028352916020019161074b565b820191906000526020600020905b81548152906001019060200180831161072e57829003601f168201915b5050505050905090565b6000610760826116c2565b6107855760405162461bcd60e51b815260040161077c90612b52565b60405180910390fd5b506000908152600460205260409020546001600160a01b031690565b60006107ac82610bfe565b9050806001600160a01b0316836001600160a01b031614156107e05760405162461bcd60e51b815260040161077c90612cae565b806001600160a01b03166107f26116cf565b6001600160a01b0316148061080e575061080e816106626116cf565b61082a5760405162461bcd60e51b815260040161077c906129f3565b61083483836116d3565b505050565b6060610843610900565b82106108615760405162461bcd60e51b815260040161077c90612754565b6000828152600d602090815260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156108f45780601f106108c9576101008083540402835291602001916108f4565b820191906000526020600020905b8154815290600101906020018083116108d757829003601f168201915b50505050509050919050565b600061090c6002611741565b905090565b61092261091c6116cf565b8261174c565b61093e5760405162461bcd60e51b815260040161077c90612cef565b6108348383836117d1565b6109516116cf565b6001600160a01b0316610962610f67565b6001600160a01b0316146109885760405162461bcd60e51b815260040161077c90612b9e565b6000610992610900565b90506000821180156109a65750600e548211155b6109c25760405162461bcd60e51b815260040161077c906127c2565b60005b828110156109e1576109d9848284016118df565b6001016109c5565b50600e546109ef90836118f9565b600e55505050565b6001600160a01b0382166000908152600160205260408120610a199083611921565b90505b92915050565b610a2a6116cf565b6001600160a01b0316610a3b610f67565b6001600160a01b031614610a615760405162461bcd60e51b815260040161077c90612b9e565b600c805460ff19811660ff90911615179055565b610a7d6116cf565b6001600160a01b0316610a8e610f67565b6001600160a01b031614610ab45760405162461bcd60e51b815260040161077c90612b9e565b6040514790339082156108fc029083906000818181858888f19350505050158015610ae3573d6000803e3d6000fd5b5050565b610834838383604051806020016040528060008152506112c6565b600d6020908152600091825260409182902080548351601f600260001961010060018616150201909316929092049182018490048402810184019094528084529091830182828015610b955780601f10610b6a57610100808354040283529160200191610b95565b820191906000526020600020905b815481529060010190602001808311610b7857829003601f168201915b505050505081565b600080610bab60028461192d565b509392505050565b610bbb6116cf565b6001600160a01b0316610bcc610f67565b6001600160a01b031614610bf25760405162461bcd60e51b815260040161077c90612b9e565b610bfb81611949565b50565b6000610a1c82604051806060016040528060298152602001612e6b602991396002919061195c565b60098054604080516020601f600260001961010060018816150201909516949094049384018190048102820181019092528281526060939092909183018282801561074b5780601f106107205761010080835404028352916020019161074b565b60006001600160a01b038216610caf5760405162461bcd60e51b815260040161077c90612a50565b6001600160a01b0382166000908152600160205260409020610a1c90611741565b610cd86116cf565b6001600160a01b0316610ce9610f67565b6001600160a01b031614610d0f5760405162461bcd60e51b815260040161077c90612b9e565b600a546040516000916001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600a80546001600160a01b0319169055565b33610d6383610bfe565b6001600160a01b031614610d895760405162461bcd60e51b815260040161077c90612963565b6000828152600d6020526040908190209051600291610da791612461565b602060405180830381855afa158015610dc4573d6000803e3d6000fd5b5050506040513d601f19601f82011682018060405250810190610de79190612339565b600282604051610df79190612445565b602060405180830381855afa158015610e14573d6000803e3d6000fd5b5050506040513d601f19601f82011682018060405250810190610e379190612339565b1415610e555760405162461bcd60e51b815260040161077c90612c6b565b6000828152600d602090815260409091208251610e749284019061209b565b507f3d41307667220ddef7b24dfded10aeece54d8ea6a68bc69d2a50550e75ec9ca4338383604051610ea893929190612551565b60405180910390a15050565b60606000610ec183610c87565b905080610ee35760408051600080825260208201909252905b509150506106ba565b60008167ffffffffffffffff81118015610efc57600080fd5b50604051908082528060200260200182016040528015610f26578160200160208202803683370190505b50905060005b82811015610f5d57610f3e85826109f7565b828281518110610f4a57fe5b6020908102919091010152600101610f2c565b5091506106ba9050565b600a546001600160a01b031690565b60078054604080516020601f600260001961010060018816150201909516949094049384018190048102820181019092528281526060939092909183018282801561074b5780601f106107205761010080835404028352916020019161074b565b600b805460408051602060026001851615610100026000190190941693909304601f81018490048402820184019092528181529291830182828015610b955780601f10610b6a57610100808354040283529160200191610b95565b61103a6116cf565b6001600160a01b0316826001600160a01b0316141561106b5760405162461bcd60e51b815260040161077c9061283b565b80600560006110786116cf565b6001600160a01b03908116825260208083019390935260409182016000908120918716808252919093529120805460ff1916921515929092179091556110bc6116cf565b6001600160a01b03167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31836040516110f49190612625565b60405180910390a35050565b6060600061110d83610c87565b905080611147576040805160008082526020820190925290610eda565b606081526020019060019003908161112a579050509150506106ba565b60008167ffffffffffffffff8111801561116057600080fd5b5060405190808252806020026020018201604052801561119457816020015b606081526020019060019003908161117f5790505b50905060005b82811015610f5d57600d60006111b087846109f7565b815260208082019290925260409081016000208054825160026001831615610100026000190190921691909104601f8101859004850282018501909352828152929091908301828280156112455780601f1061121a57610100808354040283529160200191611245565b820191906000526020600020905b81548152906001019060200180831161122857829003601f168201915b505050505082828151811061125657fe5b602090810291909101015260010161119a565b6112716116cf565b6001600160a01b0316611282610f67565b6001600160a01b0316146112a85760405162461bcd60e51b815260040161077c90612b9e565b8051610ae390600b90602084019061209b565b667c58508723800081565b6112d76112d16116cf565b8361174c565b6112f35760405162461bcd60e51b815260040161077c90612cef565b6112ff84848484611973565b50505050565b600c5460ff166113275760405162461bcd60e51b815260040161077c90612d77565b6000811180156113385750600a8111155b6113545760405162461bcd60e51b815260040161077c90612dac565b61138861136982611363610900565b906119a6565b11156113875760405162461bcd60e51b815260040161077c906129aa565b611398667c585087238000826119cb565b3410156113b75760405162461bcd60e51b815260040161077c906128a9565b60005b81811015610ae35760006113cc610900565b90506113886113d9610900565b10156113e9576113e933826118df565b506001016113ba565b60606113fd826116c2565b6114195760405162461bcd60e51b815260040161077c90612c1c565b60008281526008602090815260408083208054825160026001831615610100026000190190921691909104601f8101859004850282018501909352828152929091908301828280156114ac5780601f10611481576101008083540402835291602001916114ac565b820191906000526020600020905b81548152906001019060200180831161148f57829003601f168201915b5050505050905060006114bd610c26565b90508051600014156114d1575090506106ba565b8151156115035780826040516020016114eb9291906124d1565b604051602081830303815290604052925050506106ba565b8061150d85611a05565b60405160200161151e9291906124d1565b60405160208183030381529060405292505050919050565b61138881565b600e5481565b606061154c610900565b821061156a5760405162461bcd60e51b815260040161077c90612872565b600b805460408051602060026001851615610100026000190190941693909304601f810184900484028201840190925281815292918301828280156108f45780601f106108c9576101008083540402835291602001916108f4565b600a81565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b600c5460ff1681565b6116096116cf565b6001600160a01b031661161a610f67565b6001600160a01b0316146116405760405162461bcd60e51b815260040161077c90612b9e565b6001600160a01b0381166116665760405162461bcd60e51b815260040161077c906126d7565b600a546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600a80546001600160a01b0319166001600160a01b0392909216919091179055565b6000610a1c600283611ae0565b3390565b600081815260046020526040902080546001600160a01b0319166001600160a01b038416908117909155819061170882610bfe565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b6000610a1c82611aec565b6000611757826116c2565b6117735760405162461bcd60e51b815260040161077c90612917565b600061177e83610bfe565b9050806001600160a01b0316846001600160a01b031614806117b95750836001600160a01b03166117ae84610755565b6001600160a01b0316145b806117c957506117c981856115ca565b949350505050565b826001600160a01b03166117e482610bfe565b6001600160a01b03161461180a5760405162461bcd60e51b815260040161077c90612bd3565b6001600160a01b0382166118305760405162461bcd60e51b815260040161077c906127f7565b61183b838383610834565b6118466000826116d3565b6001600160a01b03831660009081526001602052604090206118689082611af0565b506001600160a01b038216600090815260016020526040902061188b9082611afc565b5061189860028284611b08565b5080826001600160a01b0316846001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4505050565b610ae3828260405180602001604052806000815250611b1e565b60008282111561191b5760405162461bcd60e51b815260040161077c906128e0565b50900390565b6000610a198383611b51565b600080808061193c8686611b96565b9097909650945050505050565b8051610ae390600990602084019061209b565b6000611969848484611bf2565b90505b9392505050565b61197e8484846117d1565b61198a84848484611c51565b6112ff5760405162461bcd60e51b815260040161077c90612685565b600082820183811015610a195760405162461bcd60e51b815260040161077c9061278b565b6000826119da57506000610a1c565b828202828482816119e757fe5b0414610a195760405162461bcd60e51b815260040161077c90612b11565b606081611a2a57506040805180820190915260018152600360fc1b60208201526106ba565b8160005b8115611a4257600101600a82049150611a2e565b60008167ffffffffffffffff81118015611a5b57600080fd5b506040519080825280601f01601f191660200182016040528015611a86576020820181803683370190505b50859350905060001982015b8315611ad757600a840660300160f81b82828060019003935081518110611ab557fe5b60200101906001600160f81b031916908160001a905350600a84049350611a92565b50949350505050565b6000610a198383611d30565b5490565b6000610a198383611d48565b6000610a198383611e0e565b600061196984846001600160a01b038516611e58565b611b288383611eef565b611b356000848484611c51565b6108345760405162461bcd60e51b815260040161077c90612685565b81546000908210611b745760405162461bcd60e51b815260040161077c90612643565b826000018281548110611b8357fe5b9060005260206000200154905092915050565b815460009081908310611bbb5760405162461bcd60e51b815260040161077c90612a9a565b6000846000018481548110611bcc57fe5b906000526020600020906002020190508060000154816001015492509250509250929050565b60008281526001840160205260408120548281611c225760405162461bcd60e51b815260040161077c9190612630565b50846000016001820381548110611c3557fe5b9060005260206000209060020201600101549150509392505050565b6000611c65846001600160a01b0316611fb3565b611c71575060016117c9565b6000611cf9630a85bd0160e11b611c866116cf565b888787604051602401611c9c9493929190612514565b604051602081830303815290604052906001600160e01b0319166020820180516001600160e01b038381831617835250505050604051806060016040528060328152602001612e39603291396001600160a01b0388169190611fb9565b9050600081806020019051810190611d11919061236d565b6001600160e01b031916630a85bd0160e11b1492505050949350505050565b60009081526001919091016020526040902054151590565b60008181526001830160205260408120548015611e045783546000198083019190810190600090879083908110611d7b57fe5b9060005260206000200154905080876000018481548110611d9857fe5b600091825260208083209091019290925582815260018981019092526040902090840190558654879080611dc857fe5b60019003818190600052602060002001600090559055866001016000878152602001908152602001600020600090556001945050505050610a1c565b6000915050610a1c565b6000611e1a8383611d30565b611e5057508154600181810184556000848152602080822090930184905584548482528286019093526040902091909155610a1c565b506000610a1c565b600082815260018401602052604081205480611ebd57505060408051808201825283815260208082018481528654600181810189556000898152848120955160029093029095019182559151908201558654868452818801909252929091205561196c565b82856000016001830381548110611ed057fe5b906000526020600020906002020160010181905550600091505061196c565b6001600160a01b038216611f155760405162461bcd60e51b815260040161077c90612adc565b611f1e816116c2565b15611f3b5760405162461bcd60e51b815260040161077c9061271d565b611f4760008383610834565b6001600160a01b0382166000908152600160205260409020611f699082611afc565b50611f7660028284611b08565b5060405181906001600160a01b038416906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b3b151590565b6060611969848460008585611fcd85611fb3565b611fe95760405162461bcd60e51b815260040161077c90612d40565b600080866001600160a01b031685876040516120059190612445565b60006040518083038185875af1925050503d8060008114612042576040519150601f19603f3d011682016040523d82523d6000602084013e612047565b606091505b5091509150612057828286612062565b979650505050505050565b6060831561207157508161196c565b8251156120815782518084602001fd5b8160405162461bcd60e51b815260040161077c9190612630565b828054600181600116156101000203166002900490600052602060002090601f0160209004810192826120d15760008555612117565b82601f106120ea57805160ff1916838001178555612117565b82800160010185558215612117579182015b828111156121175782518255916020019190600101906120fc565b50612123929150612127565b5090565b5b808211156121235760008155600101612128565b600067ffffffffffffffff8084111561215157fe5b604051601f8501601f19168101602001828111828210171561216f57fe5b60405284815291508183850186101561218757600080fd5b8484602083013760006020868301015250509392505050565b80356001600160a01b03811681146106ba57600080fd5b600082601f8301126121c7578081fd5b610a198383356020850161213c565b6000602082840312156121e7578081fd5b610a19826121a0565b60008060408385031215612202578081fd5b61220b836121a0565b9150612219602084016121a0565b90509250929050565b600080600060608486031215612236578081fd5b61223f846121a0565b925061224d602085016121a0565b9150604084013590509250925092565b60008060008060808587031215612272578081fd5b61227b856121a0565b9350612289602086016121a0565b925060408501359150606085013567ffffffffffffffff8111156122ab578182fd5b8501601f810187136122bb578182fd5b6122ca8782356020840161213c565b91505092959194509250565b600080604083850312156122e8578182fd5b6122f1836121a0565b915060208301358015158114612305578182fd5b809150509250929050565b60008060408385031215612322578182fd5b61232b836121a0565b946020939093013593505050565b60006020828403121561234a578081fd5b5051919050565b600060208284031215612362578081fd5b8135610a1981612e22565b60006020828403121561237e578081fd5b8151610a1981612e22565b60006020828403121561239a578081fd5b813567ffffffffffffffff8111156123b0578182fd5b6117c9848285016121b7565b6000602082840312156123cd578081fd5b5035919050565b600080604083850312156123e6578182fd5b82359150602083013567ffffffffffffffff811115612403578182fd5b61240f858286016121b7565b9150509250929050565b60008151808452612431816020860160208601612df6565b601f01601f19169290920160200192915050565b60008251612457818460208701612df6565b9190910192915050565b60008083546001808216600081146124805760018114612497576124c6565b60ff198316865260028304607f16860193506124c6565b600283048786526020808720875b838110156124be5781548a8201529085019082016124a5565b505050860193505b509195945050505050565b600083516124e3818460208801612df6565b8351908301906124f7818360208801612df6565b01949350505050565b6001600160a01b0391909116815260200190565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061254790830184612419565b9695505050505050565b600060018060a01b0385168252836020830152606060408301526125786060830184612419565b95945050505050565b6000602080830181845280855180835260408601915060408482028701019250838701855b828110156125d457603f198886030184526125c2858351612419565b945092850192908501906001016125a6565b5092979650505050505050565b6020808252825182820181905260009190848201906040850190845b81811015612619578351835292840192918401916001016125fd565b50909695505050505050565b901515815260200190565b600060208252610a196020830184612419565b60208082526022908201527f456e756d657261626c655365743a20696e646578206f7574206f6620626f756e604082015261647360f01b606082015260800190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526026908201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160408201526564647265737360d01b606082015260800190565b6020808252601c908201527f4552433732313a20746f6b656e20616c7265616479206d696e74656400000000604082015260600190565b6020808252601a908201527f43686f6f73652061206672656e2077697468696e2072616e6765000000000000604082015260600190565b6020808252601b908201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604082015260600190565b6020808252818101527f4e6f7420656e6f7567682072657365727665206c65667420666f72207465616d604082015260600190565b60208082526024908201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646040820152637265737360e01b606082015260800190565b60208082526019908201527f4552433732313a20617070726f766520746f2063616c6c657200000000000000604082015260600190565b6020808252601a908201527f43484f4f53452041204652454e2057495448494e2052414e4745000000000000604082015260600190565b6020808252601f908201527f45746865722076616c75652073656e74206973206e6f7420636f727265637400604082015260600190565b6020808252601e908201527f536166654d6174683a207375627472616374696f6e206f766572666c6f770000604082015260600190565b6020808252602c908201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860408201526b34b9ba32b73a103a37b5b2b760a11b606082015260800190565b60208082526027908201527f4865792c20796f75722077616c6c657420646f65736e2774206f776e2074686960408201526673206672656e2160c81b606082015260800190565b60208082526029908201527f507572636861736520776f756c6420657863656564206d617820737570706c79604082015268206f66206672656e7360b81b606082015260800190565b60208082526038908201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760408201527f6e6572206e6f7220617070726f76656420666f7220616c6c0000000000000000606082015260800190565b6020808252602a908201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604082015269726f206164647265737360b01b606082015260800190565b60208082526022908201527f456e756d657261626c654d61703a20696e646578206f7574206f6620626f756e604082015261647360f01b606082015260800190565b6020808252818101527f4552433732313a206d696e7420746f20746865207a65726f2061646472657373604082015260600190565b60208082526021908201527f536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f6040820152607760f81b606082015260800190565b6020808252602c908201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860408201526b34b9ba32b73a103a37b5b2b760a11b606082015260800190565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208082526029908201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960408201526839903737ba1037bbb760b91b606082015260800190565b6020808252602f908201527f4552433732314d657461646174613a2055524920717565727920666f72206e6f60408201526e3732bc34b9ba32b73a103a37b5b2b760891b606082015260800190565b60208082526023908201527f4e6577206e616d652069732073616d65206173207468652063757272656e74206040820152626f6e6560e81b606082015260800190565b60208082526021908201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656040820152603960f91b606082015260800190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b6020808252601d908201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604082015260600190565b6020808252818101527f53616c65206d7573742062652061637469766520746f206d696e74206672656e604082015260600190565b60208082526021908201527f43616e206f6e6c79206d696e7420313020746f6b656e7320617420612074696d6040820152606560f81b606082015260800190565b90815260200190565b60005b83811015612e11578181015183820152602001612df9565b838111156112ff5750506000910152565b6001600160e01b031981168114610bfb57600080fdfe4552433732313a207472616e7366657220746f206e6f6e20455243373231526563656976657220696d706c656d656e7465724552433732313a206f776e657220717565727920666f72206e6f6e6578697374656e7420746f6b656ea2646970667358221220dfcb5ee0a17c4b1cb2a70a74890351068c8d584bedc46fbab659ce82da9b912464736f6c63430007060033

Deployed Bytecode

0x6080604052600436106102255760003560e01c80637202be8311610123578063b88d4fde116100ab578063d9b137b21161006f578063d9b137b214610612578063df70f31114610632578063e985e9c514610647578063eb8d244414610667578063f2fde38b1461067c57610225565b8063b88d4fde14610595578063ba90029e146105b5578063c87b56dd146105c8578063ce272ca0146105e8578063d7e38c34146105fd57610225565b80639c3e72bd116100f25780639c3e72bd146104fe578063a22cb46514610513578063af2aec2a14610533578063b09904b514610560578063b3b553ba1461058057610225565b80637202be83146104875780638462151c146104a75780638da5cb5b146104d457806395d89b41146104e957610225565b806334918dfd116101b157806355f804b31161017557806355f804b3146103fd5780636352211e1461041d5780636c0360eb1461043d57806370a0823114610452578063715018a61461047257610225565b806334918dfd146103735780633ccfd60b1461038857806342842e0e1461039d5780634889e50b146103bd5780634f6ccce7146103dd57610225565b80630fdfc028116101f85780630fdfc028146102d157806318160ddd146102f157806323b872dd146103135780632cddaf7f146103335780632f745c591461035357610225565b806301ffc9a71461022a57806306fdde0314610260578063081812fc14610282578063095ea7b3146102af575b600080fd5b34801561023657600080fd5b5061024a610245366004612351565b61069c565b6040516102579190612625565b60405180910390f35b34801561026c57600080fd5b506102756106bf565b6040516102579190612630565b34801561028e57600080fd5b506102a261029d3660046123bc565b610755565b6040516102579190612500565b3480156102bb57600080fd5b506102cf6102ca366004612310565b6107a1565b005b3480156102dd57600080fd5b506102756102ec3660046123bc565b610839565b3480156102fd57600080fd5b50610306610900565b6040516102579190612ded565b34801561031f57600080fd5b506102cf61032e366004612222565b610911565b34801561033f57600080fd5b506102cf61034e366004612310565b610949565b34801561035f57600080fd5b5061030661036e366004612310565b6109f7565b34801561037f57600080fd5b506102cf610a22565b34801561039457600080fd5b506102cf610a75565b3480156103a957600080fd5b506102cf6103b8366004612222565b610ae7565b3480156103c957600080fd5b506102756103d83660046123bc565b610b02565b3480156103e957600080fd5b506103066103f83660046123bc565b610b9d565b34801561040957600080fd5b506102cf610418366004612389565b610bb3565b34801561042957600080fd5b506102a26104383660046123bc565b610bfe565b34801561044957600080fd5b50610275610c26565b34801561045e57600080fd5b5061030661046d3660046121d6565b610c87565b34801561047e57600080fd5b506102cf610cd0565b34801561049357600080fd5b506102cf6104a23660046123d4565b610d59565b3480156104b357600080fd5b506104c76104c23660046121d6565b610eb4565b60405161025791906125e1565b3480156104e057600080fd5b506102a2610f67565b3480156104f557600080fd5b50610275610f76565b34801561050a57600080fd5b50610275610fd7565b34801561051f57600080fd5b506102cf61052e3660046122d6565b611032565b34801561053f57600080fd5b5061055361054e3660046121d6565b611100565b6040516102579190612581565b34801561056c57600080fd5b506102cf61057b366004612389565b611269565b34801561058c57600080fd5b506103066112bb565b3480156105a157600080fd5b506102cf6105b036600461225d565b6112c6565b6102cf6105c33660046123bc565b611305565b3480156105d457600080fd5b506102756105e33660046123bc565b6113f2565b3480156105f457600080fd5b50610306611536565b34801561060957600080fd5b5061030661153c565b34801561061e57600080fd5b5061027561062d3660046123bc565b611542565b34801561063e57600080fd5b506103066115c5565b34801561065357600080fd5b5061024a6106623660046121f0565b6115ca565b34801561067357600080fd5b5061024a6115f8565b34801561068857600080fd5b506102cf6106973660046121d6565b611601565b6001600160e01b0319811660009081526020819052604090205460ff165b919050565b60068054604080516020601f600260001961010060018816150201909516949094049384018190048102820181019092528281526060939092909183018282801561074b5780601f106107205761010080835404028352916020019161074b565b820191906000526020600020905b81548152906001019060200180831161072e57829003601f168201915b5050505050905090565b6000610760826116c2565b6107855760405162461bcd60e51b815260040161077c90612b52565b60405180910390fd5b506000908152600460205260409020546001600160a01b031690565b60006107ac82610bfe565b9050806001600160a01b0316836001600160a01b031614156107e05760405162461bcd60e51b815260040161077c90612cae565b806001600160a01b03166107f26116cf565b6001600160a01b0316148061080e575061080e816106626116cf565b61082a5760405162461bcd60e51b815260040161077c906129f3565b61083483836116d3565b505050565b6060610843610900565b82106108615760405162461bcd60e51b815260040161077c90612754565b6000828152600d602090815260409182902080548351601f6002600019610100600186161502019093169290920491820184900484028101840190945280845290918301828280156108f45780601f106108c9576101008083540402835291602001916108f4565b820191906000526020600020905b8154815290600101906020018083116108d757829003601f168201915b50505050509050919050565b600061090c6002611741565b905090565b61092261091c6116cf565b8261174c565b61093e5760405162461bcd60e51b815260040161077c90612cef565b6108348383836117d1565b6109516116cf565b6001600160a01b0316610962610f67565b6001600160a01b0316146109885760405162461bcd60e51b815260040161077c90612b9e565b6000610992610900565b90506000821180156109a65750600e548211155b6109c25760405162461bcd60e51b815260040161077c906127c2565b60005b828110156109e1576109d9848284016118df565b6001016109c5565b50600e546109ef90836118f9565b600e55505050565b6001600160a01b0382166000908152600160205260408120610a199083611921565b90505b92915050565b610a2a6116cf565b6001600160a01b0316610a3b610f67565b6001600160a01b031614610a615760405162461bcd60e51b815260040161077c90612b9e565b600c805460ff19811660ff90911615179055565b610a7d6116cf565b6001600160a01b0316610a8e610f67565b6001600160a01b031614610ab45760405162461bcd60e51b815260040161077c90612b9e565b6040514790339082156108fc029083906000818181858888f19350505050158015610ae3573d6000803e3d6000fd5b5050565b610834838383604051806020016040528060008152506112c6565b600d6020908152600091825260409182902080548351601f600260001961010060018616150201909316929092049182018490048402810184019094528084529091830182828015610b955780601f10610b6a57610100808354040283529160200191610b95565b820191906000526020600020905b815481529060010190602001808311610b7857829003601f168201915b505050505081565b600080610bab60028461192d565b509392505050565b610bbb6116cf565b6001600160a01b0316610bcc610f67565b6001600160a01b031614610bf25760405162461bcd60e51b815260040161077c90612b9e565b610bfb81611949565b50565b6000610a1c82604051806060016040528060298152602001612e6b602991396002919061195c565b60098054604080516020601f600260001961010060018816150201909516949094049384018190048102820181019092528281526060939092909183018282801561074b5780601f106107205761010080835404028352916020019161074b565b60006001600160a01b038216610caf5760405162461bcd60e51b815260040161077c90612a50565b6001600160a01b0382166000908152600160205260409020610a1c90611741565b610cd86116cf565b6001600160a01b0316610ce9610f67565b6001600160a01b031614610d0f5760405162461bcd60e51b815260040161077c90612b9e565b600a546040516000916001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600a80546001600160a01b0319169055565b33610d6383610bfe565b6001600160a01b031614610d895760405162461bcd60e51b815260040161077c90612963565b6000828152600d6020526040908190209051600291610da791612461565b602060405180830381855afa158015610dc4573d6000803e3d6000fd5b5050506040513d601f19601f82011682018060405250810190610de79190612339565b600282604051610df79190612445565b602060405180830381855afa158015610e14573d6000803e3d6000fd5b5050506040513d601f19601f82011682018060405250810190610e379190612339565b1415610e555760405162461bcd60e51b815260040161077c90612c6b565b6000828152600d602090815260409091208251610e749284019061209b565b507f3d41307667220ddef7b24dfded10aeece54d8ea6a68bc69d2a50550e75ec9ca4338383604051610ea893929190612551565b60405180910390a15050565b60606000610ec183610c87565b905080610ee35760408051600080825260208201909252905b509150506106ba565b60008167ffffffffffffffff81118015610efc57600080fd5b50604051908082528060200260200182016040528015610f26578160200160208202803683370190505b50905060005b82811015610f5d57610f3e85826109f7565b828281518110610f4a57fe5b6020908102919091010152600101610f2c565b5091506106ba9050565b600a546001600160a01b031690565b60078054604080516020601f600260001961010060018816150201909516949094049384018190048102820181019092528281526060939092909183018282801561074b5780601f106107205761010080835404028352916020019161074b565b600b805460408051602060026001851615610100026000190190941693909304601f81018490048402820184019092528181529291830182828015610b955780601f10610b6a57610100808354040283529160200191610b95565b61103a6116cf565b6001600160a01b0316826001600160a01b0316141561106b5760405162461bcd60e51b815260040161077c9061283b565b80600560006110786116cf565b6001600160a01b03908116825260208083019390935260409182016000908120918716808252919093529120805460ff1916921515929092179091556110bc6116cf565b6001600160a01b03167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31836040516110f49190612625565b60405180910390a35050565b6060600061110d83610c87565b905080611147576040805160008082526020820190925290610eda565b606081526020019060019003908161112a579050509150506106ba565b60008167ffffffffffffffff8111801561116057600080fd5b5060405190808252806020026020018201604052801561119457816020015b606081526020019060019003908161117f5790505b50905060005b82811015610f5d57600d60006111b087846109f7565b815260208082019290925260409081016000208054825160026001831615610100026000190190921691909104601f8101859004850282018501909352828152929091908301828280156112455780601f1061121a57610100808354040283529160200191611245565b820191906000526020600020905b81548152906001019060200180831161122857829003601f168201915b505050505082828151811061125657fe5b602090810291909101015260010161119a565b6112716116cf565b6001600160a01b0316611282610f67565b6001600160a01b0316146112a85760405162461bcd60e51b815260040161077c90612b9e565b8051610ae390600b90602084019061209b565b667c58508723800081565b6112d76112d16116cf565b8361174c565b6112f35760405162461bcd60e51b815260040161077c90612cef565b6112ff84848484611973565b50505050565b600c5460ff166113275760405162461bcd60e51b815260040161077c90612d77565b6000811180156113385750600a8111155b6113545760405162461bcd60e51b815260040161077c90612dac565b61138861136982611363610900565b906119a6565b11156113875760405162461bcd60e51b815260040161077c906129aa565b611398667c585087238000826119cb565b3410156113b75760405162461bcd60e51b815260040161077c906128a9565b60005b81811015610ae35760006113cc610900565b90506113886113d9610900565b10156113e9576113e933826118df565b506001016113ba565b60606113fd826116c2565b6114195760405162461bcd60e51b815260040161077c90612c1c565b60008281526008602090815260408083208054825160026001831615610100026000190190921691909104601f8101859004850282018501909352828152929091908301828280156114ac5780601f10611481576101008083540402835291602001916114ac565b820191906000526020600020905b81548152906001019060200180831161148f57829003601f168201915b5050505050905060006114bd610c26565b90508051600014156114d1575090506106ba565b8151156115035780826040516020016114eb9291906124d1565b604051602081830303815290604052925050506106ba565b8061150d85611a05565b60405160200161151e9291906124d1565b60405160208183030381529060405292505050919050565b61138881565b600e5481565b606061154c610900565b821061156a5760405162461bcd60e51b815260040161077c90612872565b600b805460408051602060026001851615610100026000190190941693909304601f810184900484028201840190925281815292918301828280156108f45780601f106108c9576101008083540402835291602001916108f4565b600a81565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b600c5460ff1681565b6116096116cf565b6001600160a01b031661161a610f67565b6001600160a01b0316146116405760405162461bcd60e51b815260040161077c90612b9e565b6001600160a01b0381166116665760405162461bcd60e51b815260040161077c906126d7565b600a546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600a80546001600160a01b0319166001600160a01b0392909216919091179055565b6000610a1c600283611ae0565b3390565b600081815260046020526040902080546001600160a01b0319166001600160a01b038416908117909155819061170882610bfe565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b6000610a1c82611aec565b6000611757826116c2565b6117735760405162461bcd60e51b815260040161077c90612917565b600061177e83610bfe565b9050806001600160a01b0316846001600160a01b031614806117b95750836001600160a01b03166117ae84610755565b6001600160a01b0316145b806117c957506117c981856115ca565b949350505050565b826001600160a01b03166117e482610bfe565b6001600160a01b03161461180a5760405162461bcd60e51b815260040161077c90612bd3565b6001600160a01b0382166118305760405162461bcd60e51b815260040161077c906127f7565b61183b838383610834565b6118466000826116d3565b6001600160a01b03831660009081526001602052604090206118689082611af0565b506001600160a01b038216600090815260016020526040902061188b9082611afc565b5061189860028284611b08565b5080826001600160a01b0316846001600160a01b03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4505050565b610ae3828260405180602001604052806000815250611b1e565b60008282111561191b5760405162461bcd60e51b815260040161077c906128e0565b50900390565b6000610a198383611b51565b600080808061193c8686611b96565b9097909650945050505050565b8051610ae390600990602084019061209b565b6000611969848484611bf2565b90505b9392505050565b61197e8484846117d1565b61198a84848484611c51565b6112ff5760405162461bcd60e51b815260040161077c90612685565b600082820183811015610a195760405162461bcd60e51b815260040161077c9061278b565b6000826119da57506000610a1c565b828202828482816119e757fe5b0414610a195760405162461bcd60e51b815260040161077c90612b11565b606081611a2a57506040805180820190915260018152600360fc1b60208201526106ba565b8160005b8115611a4257600101600a82049150611a2e565b60008167ffffffffffffffff81118015611a5b57600080fd5b506040519080825280601f01601f191660200182016040528015611a86576020820181803683370190505b50859350905060001982015b8315611ad757600a840660300160f81b82828060019003935081518110611ab557fe5b60200101906001600160f81b031916908160001a905350600a84049350611a92565b50949350505050565b6000610a198383611d30565b5490565b6000610a198383611d48565b6000610a198383611e0e565b600061196984846001600160a01b038516611e58565b611b288383611eef565b611b356000848484611c51565b6108345760405162461bcd60e51b815260040161077c90612685565b81546000908210611b745760405162461bcd60e51b815260040161077c90612643565b826000018281548110611b8357fe5b9060005260206000200154905092915050565b815460009081908310611bbb5760405162461bcd60e51b815260040161077c90612a9a565b6000846000018481548110611bcc57fe5b906000526020600020906002020190508060000154816001015492509250509250929050565b60008281526001840160205260408120548281611c225760405162461bcd60e51b815260040161077c9190612630565b50846000016001820381548110611c3557fe5b9060005260206000209060020201600101549150509392505050565b6000611c65846001600160a01b0316611fb3565b611c71575060016117c9565b6000611cf9630a85bd0160e11b611c866116cf565b888787604051602401611c9c9493929190612514565b604051602081830303815290604052906001600160e01b0319166020820180516001600160e01b038381831617835250505050604051806060016040528060328152602001612e39603291396001600160a01b0388169190611fb9565b9050600081806020019051810190611d11919061236d565b6001600160e01b031916630a85bd0160e11b1492505050949350505050565b60009081526001919091016020526040902054151590565b60008181526001830160205260408120548015611e045783546000198083019190810190600090879083908110611d7b57fe5b9060005260206000200154905080876000018481548110611d9857fe5b600091825260208083209091019290925582815260018981019092526040902090840190558654879080611dc857fe5b60019003818190600052602060002001600090559055866001016000878152602001908152602001600020600090556001945050505050610a1c565b6000915050610a1c565b6000611e1a8383611d30565b611e5057508154600181810184556000848152602080822090930184905584548482528286019093526040902091909155610a1c565b506000610a1c565b600082815260018401602052604081205480611ebd57505060408051808201825283815260208082018481528654600181810189556000898152848120955160029093029095019182559151908201558654868452818801909252929091205561196c565b82856000016001830381548110611ed057fe5b906000526020600020906002020160010181905550600091505061196c565b6001600160a01b038216611f155760405162461bcd60e51b815260040161077c90612adc565b611f1e816116c2565b15611f3b5760405162461bcd60e51b815260040161077c9061271d565b611f4760008383610834565b6001600160a01b0382166000908152600160205260409020611f699082611afc565b50611f7660028284611b08565b5060405181906001600160a01b038416906000907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b3b151590565b6060611969848460008585611fcd85611fb3565b611fe95760405162461bcd60e51b815260040161077c90612d40565b600080866001600160a01b031685876040516120059190612445565b60006040518083038185875af1925050503d8060008114612042576040519150601f19603f3d011682016040523d82523d6000602084013e612047565b606091505b5091509150612057828286612062565b979650505050505050565b6060831561207157508161196c565b8251156120815782518084602001fd5b8160405162461bcd60e51b815260040161077c9190612630565b828054600181600116156101000203166002900490600052602060002090601f0160209004810192826120d15760008555612117565b82601f106120ea57805160ff1916838001178555612117565b82800160010185558215612117579182015b828111156121175782518255916020019190600101906120fc565b50612123929150612127565b5090565b5b808211156121235760008155600101612128565b600067ffffffffffffffff8084111561215157fe5b604051601f8501601f19168101602001828111828210171561216f57fe5b60405284815291508183850186101561218757600080fd5b8484602083013760006020868301015250509392505050565b80356001600160a01b03811681146106ba57600080fd5b600082601f8301126121c7578081fd5b610a198383356020850161213c565b6000602082840312156121e7578081fd5b610a19826121a0565b60008060408385031215612202578081fd5b61220b836121a0565b9150612219602084016121a0565b90509250929050565b600080600060608486031215612236578081fd5b61223f846121a0565b925061224d602085016121a0565b9150604084013590509250925092565b60008060008060808587031215612272578081fd5b61227b856121a0565b9350612289602086016121a0565b925060408501359150606085013567ffffffffffffffff8111156122ab578182fd5b8501601f810187136122bb578182fd5b6122ca8782356020840161213c565b91505092959194509250565b600080604083850312156122e8578182fd5b6122f1836121a0565b915060208301358015158114612305578182fd5b809150509250929050565b60008060408385031215612322578182fd5b61232b836121a0565b946020939093013593505050565b60006020828403121561234a578081fd5b5051919050565b600060208284031215612362578081fd5b8135610a1981612e22565b60006020828403121561237e578081fd5b8151610a1981612e22565b60006020828403121561239a578081fd5b813567ffffffffffffffff8111156123b0578182fd5b6117c9848285016121b7565b6000602082840312156123cd578081fd5b5035919050565b600080604083850312156123e6578182fd5b82359150602083013567ffffffffffffffff811115612403578182fd5b61240f858286016121b7565b9150509250929050565b60008151808452612431816020860160208601612df6565b601f01601f19169290920160200192915050565b60008251612457818460208701612df6565b9190910192915050565b60008083546001808216600081146124805760018114612497576124c6565b60ff198316865260028304607f16860193506124c6565b600283048786526020808720875b838110156124be5781548a8201529085019082016124a5565b505050860193505b509195945050505050565b600083516124e3818460208801612df6565b8351908301906124f7818360208801612df6565b01949350505050565b6001600160a01b0391909116815260200190565b6001600160a01b038581168252841660208201526040810183905260806060820181905260009061254790830184612419565b9695505050505050565b600060018060a01b0385168252836020830152606060408301526125786060830184612419565b95945050505050565b6000602080830181845280855180835260408601915060408482028701019250838701855b828110156125d457603f198886030184526125c2858351612419565b945092850192908501906001016125a6565b5092979650505050505050565b6020808252825182820181905260009190848201906040850190845b81811015612619578351835292840192918401916001016125fd565b50909695505050505050565b901515815260200190565b600060208252610a196020830184612419565b60208082526022908201527f456e756d657261626c655365743a20696e646578206f7574206f6620626f756e604082015261647360f01b606082015260800190565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b60208082526026908201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160408201526564647265737360d01b606082015260800190565b6020808252601c908201527f4552433732313a20746f6b656e20616c7265616479206d696e74656400000000604082015260600190565b6020808252601a908201527f43686f6f73652061206672656e2077697468696e2072616e6765000000000000604082015260600190565b6020808252601b908201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604082015260600190565b6020808252818101527f4e6f7420656e6f7567682072657365727665206c65667420666f72207465616d604082015260600190565b60208082526024908201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646040820152637265737360e01b606082015260800190565b60208082526019908201527f4552433732313a20617070726f766520746f2063616c6c657200000000000000604082015260600190565b6020808252601a908201527f43484f4f53452041204652454e2057495448494e2052414e4745000000000000604082015260600190565b6020808252601f908201527f45746865722076616c75652073656e74206973206e6f7420636f727265637400604082015260600190565b6020808252601e908201527f536166654d6174683a207375627472616374696f6e206f766572666c6f770000604082015260600190565b6020808252602c908201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860408201526b34b9ba32b73a103a37b5b2b760a11b606082015260800190565b60208082526027908201527f4865792c20796f75722077616c6c657420646f65736e2774206f776e2074686960408201526673206672656e2160c81b606082015260800190565b60208082526029908201527f507572636861736520776f756c6420657863656564206d617820737570706c79604082015268206f66206672656e7360b81b606082015260800190565b60208082526038908201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760408201527f6e6572206e6f7220617070726f76656420666f7220616c6c0000000000000000606082015260800190565b6020808252602a908201527f4552433732313a2062616c616e636520717565727920666f7220746865207a65604082015269726f206164647265737360b01b606082015260800190565b60208082526022908201527f456e756d657261626c654d61703a20696e646578206f7574206f6620626f756e604082015261647360f01b606082015260800190565b6020808252818101527f4552433732313a206d696e7420746f20746865207a65726f2061646472657373604082015260600190565b60208082526021908201527f536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f6040820152607760f81b606082015260800190565b6020808252602c908201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860408201526b34b9ba32b73a103a37b5b2b760a11b606082015260800190565b6020808252818101527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604082015260600190565b60208082526029908201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960408201526839903737ba1037bbb760b91b606082015260800190565b6020808252602f908201527f4552433732314d657461646174613a2055524920717565727920666f72206e6f60408201526e3732bc34b9ba32b73a103a37b5b2b760891b606082015260800190565b60208082526023908201527f4e6577206e616d652069732073616d65206173207468652063757272656e74206040820152626f6e6560e81b606082015260800190565b60208082526021908201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656040820152603960f91b606082015260800190565b60208082526031908201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f6040820152701ddb995c881b9bdc88185c1c1c9bdd9959607a1b606082015260800190565b6020808252601d908201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e7472616374000000604082015260600190565b6020808252818101527f53616c65206d7573742062652061637469766520746f206d696e74206672656e604082015260600190565b60208082526021908201527f43616e206f6e6c79206d696e7420313020746f6b656e7320617420612074696d6040820152606560f81b606082015260800190565b90815260200190565b60005b83811015612e11578181015183820152602001612df9565b838111156112ff5750506000910152565b6001600160e01b031981168114610bfb57600080fdfe4552433732313a207472616e7366657220746f206e6f6e20455243373231526563656976657220696d706c656d656e7465724552433732313a206f776e657220717565727920666f72206e6f6e6578697374656e7420746f6b656ea2646970667358221220dfcb5ee0a17c4b1cb2a70a74890351068c8d584bedc46fbab659ce82da9b912464736f6c63430007060033

Deployed Bytecode Sourcemap

66910:4337:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;10236:150;;;;;;;;;;-1:-1:-1;10236:150:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;51521:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;54307:221::-;;;;;;;;;;-1:-1:-1;54307:221:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;53837:404::-;;;;;;;;;;-1:-1:-1;53837:404:0;;;;;:::i;:::-;;:::i;:::-;;70439:194;;;;;;;;;;-1:-1:-1;70439:194:0;;;;;:::i;:::-;;:::i;53315:211::-;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;55197:305::-;;;;;;;;;;-1:-1:-1;55197:305:0;;;;;:::i;:::-;;:::i;67777:398::-;;;;;;;;;;-1:-1:-1;67777:398:0;;;;;:::i;:::-;;:::i;53077:162::-;;;;;;;;;;-1:-1:-1;53077:162:0;;;;;:::i;:::-;;:::i;68292:89::-;;;;;;;;;;;;;:::i;67634:131::-;;;;;;;;;;;;;:::i;55573:151::-;;;;;;;;;;-1:-1:-1;55573:151:0;;;;;:::i;:::-;;:::i;67294:40::-;;;;;;;;;;-1:-1:-1;67294:40:0;;;;;:::i;:::-;;:::i;53603:172::-;;;;;;;;;;-1:-1:-1;53603:172:0;;;;;:::i;:::-;;:::i;68183:99::-;;;;;;;;;;-1:-1:-1;68183:99:0;;;;;:::i;:::-;;:::i;51277:177::-;;;;;;;;;;-1:-1:-1;51277:177:0;;;;;:::i;:::-;;:::i;52896:97::-;;;;;;;;;;;;;:::i;50994:221::-;;;;;;;;;;-1:-1:-1;50994:221:0;;;;;:::i;:::-;;:::i;66232:148::-;;;;;;;;;;;;;:::i;70024:403::-;;;;;;;;;;-1:-1:-1;70024:403:0;;;;;:::i;:::-;;:::i;68399:540::-;;;;;;;;;;-1:-1:-1;68399:540:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;65581:87::-;;;;;;;;;;;;;:::i;51690:104::-;;;;;;;;;;;;;:::i;67013:31::-;;;;;;;;;;;;;:::i;54600:295::-;;;;;;;;;;-1:-1:-1;54600:295:0;;;;;:::i;:::-;;:::i;70685:553::-;;;;;;;;;;-1:-1:-1;70685:553:0;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;69203:106::-;;;;;;;;;;-1:-1:-1;69203:106:0;;;;;:::i;:::-;;:::i;67075:53::-;;;;;;;;;;;;;:::i;55795:285::-;;;;;;;;;;-1:-1:-1;55795:285:0;;;;;:::i;:::-;;:::i;69327:684::-;;;;;;:::i;:::-;;:::i;51865:792::-;;;;;;;;;;-1:-1:-1;51865:792:0;;;;;:::i;:::-;;:::i;67200:40::-;;;;;;;;;;;;;:::i;67454:29::-;;;;;;;;;;;;;:::i;68990:174::-;;;;;;;;;;-1:-1:-1;68990:174:0;;;;;:::i;:::-;;:::i;67150:41::-;;;;;;;;;;;;;:::i;54966:164::-;;;;;;;;;;-1:-1:-1;54966:164:0;;;;;:::i;:::-;;:::i;67249:32::-;;;;;;;;;;;;;:::i;66535:244::-;;;;;;;;;;-1:-1:-1;66535:244:0;;;;;:::i;:::-;;:::i;10236:150::-;-1:-1:-1;;;;;;10345:33:0;;10321:4;10345:33;;;;;;;;;;;;;10236:150;;;;:::o;51521:100::-;51608:5;51601:12;;;;;;;;-1:-1:-1;;51601:12:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;51575:13;;51601:12;;51608:5;;51601:12;;51608:5;51601:12;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;51521:100;:::o;54307:221::-;54383:7;54411:16;54419:7;54411;:16::i;:::-;54403:73;;;;-1:-1:-1;;;54403:73:0;;;;;;;:::i;:::-;;;;;;;;;-1:-1:-1;54496:24:0;;;;:15;:24;;;;;;-1:-1:-1;;;;;54496:24:0;;54307:221::o;53837:404::-;53918:13;53934:23;53949:7;53934:14;:23::i;:::-;53918:39;;53982:5;-1:-1:-1;;;;;53976:11:0;:2;-1:-1:-1;;;;;53976:11:0;;;53968:57;;;;-1:-1:-1;;;53968:57:0;;;;;;;:::i;:::-;54062:5;-1:-1:-1;;;;;54046:21:0;:12;:10;:12::i;:::-;-1:-1:-1;;;;;54046:21:0;;:69;;;;54071:44;54095:5;54102:12;:10;:12::i;54071:44::-;54038:161;;;;-1:-1:-1;;;54038:161:0;;;;;;;:::i;:::-;54212:21;54221:2;54225:7;54212:8;:21::i;:::-;53837:404;;;:::o;70439:194::-;70497:13;70543;:11;:13::i;:::-;70532:8;:24;70523:65;;;;-1:-1:-1;;;70523:65:0;;;;;;;:::i;:::-;70606:19;;;;:9;:19;;;;;;;;;70599:26;;;;;;-1:-1:-1;;70599:26:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70606:19;;70599:26;;70606:19;70599:26;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70439:194;;;:::o;53315:211::-;53376:7;53497:21;:12;:19;:21::i;:::-;53490:28;;53315:211;:::o;55197:305::-;55358:41;55377:12;:10;:12::i;:::-;55391:7;55358:18;:41::i;:::-;55350:103;;;;-1:-1:-1;;;55350:103:0;;;;;;;:::i;:::-;55466:28;55476:4;55482:2;55486:7;55466:9;:28::i;67777:398::-;65812:12;:10;:12::i;:::-;-1:-1:-1;;;;;65801:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;65801:23:0;;65793:68;;;;-1:-1:-1;;;65793:68:0;;;;;;;:::i;:::-;67872:11:::1;67886:13;:11;:13::i;:::-;67872:27;;67935:1;67918:14;:18;:51;;;;;67958:11;;67940:14;:29;;67918:51;67910:96;;;;-1:-1:-1::0;;;67910:96:0::1;;;;;;;:::i;:::-;68022:6;68017:95;68038:14;68034:1;:18;68017:95;;;68074:26;68084:3;68098:1;68089:6;:10;68074:9;:26::i;:::-;68054:3;;68017:95;;;-1:-1:-1::0;68136:11:0::1;::::0;:31:::1;::::0;68152:14;68136:15:::1;:31::i;:::-;68122:11;:45:::0;-1:-1:-1;;;67777:398:0:o;53077:162::-;-1:-1:-1;;;;;53201:20:0;;53174:7;53201:20;;;:13;:20;;;;;:30;;53225:5;53201:23;:30::i;:::-;53194:37;;53077:162;;;;;:::o;68292:89::-;65812:12;:10;:12::i;:::-;-1:-1:-1;;;;;65801:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;65801:23:0;;65793:68;;;;-1:-1:-1;;;65793:68:0;;;;;;;:::i;:::-;68361:12:::1;::::0;;-1:-1:-1;;68345:28:0;::::1;68361:12;::::0;;::::1;68360:13;68345:28;::::0;;68292:89::o;67634:131::-;65812:12;:10;:12::i;:::-;-1:-1:-1;;;;;65801:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;65801:23:0;;65793:68;;;;-1:-1:-1;;;65793:68:0;;;;;;;:::i;:::-;67729:28:::1;::::0;67697:21:::1;::::0;67729:10:::1;::::0;:28;::::1;;;::::0;67697:21;;67682:12:::1;67729:28:::0;67682:12;67729:28;67697:21;67729:10;:28;::::1;;;;;;;;;;;;;::::0;::::1;;;;;;65872:1;67634:131::o:0;55573:151::-;55677:39;55694:4;55700:2;55704:7;55677:39;;;;;;;;;;;;:16;:39::i;67294:40::-;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;67294:40:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;53603:172::-;53678:7;;53720:22;:12;53736:5;53720:15;:22::i;:::-;-1:-1:-1;53698:44:0;53603:172;-1:-1:-1;;;53603:172:0:o;68183:99::-;65812:12;:10;:12::i;:::-;-1:-1:-1;;;;;65801:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;65801:23:0;;65793:68;;;;-1:-1:-1;;;65793:68:0;;;;;;;:::i;:::-;68254:20:::1;68266:7;68254:11;:20::i;:::-;68183:99:::0;:::o;51277:177::-;51349:7;51376:70;51393:7;51376:70;;;;;;;;;;;;;;;;;:12;;:70;:16;:70::i;52896:97::-;52977:8;52970:15;;;;;;;;-1:-1:-1;;52970:15:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;52944:13;;52970:15;;52977:8;;52970:15;;52977:8;52970:15;;;;;;;;;;;;;;;;;;;;;;;;50994:221;51066:7;-1:-1:-1;;;;;51094:19:0;;51086:74;;;;-1:-1:-1;;;51086:74:0;;;;;;;:::i;:::-;-1:-1:-1;;;;;51178:20:0;;;;;;:13;:20;;;;;:29;;:27;:29::i;66232:148::-;65812:12;:10;:12::i;:::-;-1:-1:-1;;;;;65801:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;65801:23:0;;65793:68;;;;-1:-1:-1;;;65793:68:0;;;;;;;:::i;:::-;66323:6:::1;::::0;66302:40:::1;::::0;66339:1:::1;::::0;-1:-1:-1;;;;;66323:6:0::1;::::0;66302:40:::1;::::0;66339:1;;66302:40:::1;66353:6;:19:::0;;-1:-1:-1;;;;;;66353:19:0::1;::::0;;66232:148::o;70024:403::-;70131:10;70110:17;70118:8;70110:7;:17::i;:::-;-1:-1:-1;;;;;70110:31:0;;70102:83;;;;-1:-1:-1;;;70102:83:0;;;;;;;:::i;:::-;70241:19;;;;:9;:19;;;;;;;70228:34;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;70204:20;70217:5;70204:20;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:58;;70196:106;;;;-1:-1:-1;;;70196:106:0;;;;;;;:::i;:::-;70313:19;;;;:9;:19;;;;;;;;:27;;;;;;;;:::i;:::-;;70366:43;70381:10;70393:8;70403:5;70366:43;;;;;;;;:::i;:::-;;;;;;;;70024:403;;:::o;68399:540::-;68460:16;68490:18;68511:17;68521:6;68511:9;:17::i;:::-;68490:38;-1:-1:-1;68543:15:0;68539:393;;68620:16;;;68634:1;68620:16;;;;;;;;;;;;68613:23;;;;;68539:393;68669:23;68709:10;68695:25;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;68695:25:0;;68669:51;;68735:13;68763:130;68787:10;68779:5;:18;68763:130;;;68843:34;68863:6;68871:5;68843:19;:34::i;:::-;68827:6;68834:5;68827:13;;;;;;;;;;;;;;;;;:50;68799:7;;68763:130;;;-1:-1:-1;68914:6:0;-1:-1:-1;68907:13:0;;-1:-1:-1;68907:13:0;65581:87;65654:6;;-1:-1:-1;;;;;65654:6:0;65581:87;:::o;51690:104::-;51779:7;51772:14;;;;;;;;-1:-1:-1;;51772:14:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;51746:13;;51772:14;;51779:7;;51772:14;;51779:7;51772:14;;;;;;;;;;;;;;;;;;;;;;;;67013:31;;;;;;;;;;;;;;;-1:-1:-1;;67013:31:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;54600:295;54715:12;:10;:12::i;:::-;-1:-1:-1;;;;;54703:24:0;:8;-1:-1:-1;;;;;54703:24:0;;;54695:62;;;;-1:-1:-1;;;54695:62:0;;;;;;;:::i;:::-;54815:8;54770:18;:32;54789:12;:10;:12::i;:::-;-1:-1:-1;;;;;54770:32:0;;;;;;;;;;;;;;;;;-1:-1:-1;54770:32:0;;;:42;;;;;;;;;;;;:53;;-1:-1:-1;;54770:53:0;;;;;;;;;;;54854:12;:10;:12::i;:::-;-1:-1:-1;;;;;54839:48:0;;54878:8;54839:48;;;;;;:::i;:::-;;;;;;;;54600:295;;:::o;70685:553::-;70749:15;70778:18;70799:17;70809:6;70799:9;:17::i;:::-;70778:38;-1:-1:-1;70831:15:0;70827:404;;70908:15;;;70921:1;70908:15;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70901:22;;;;;70827:404;70956:22;70994:10;70981:24;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70956:49;;71020:13;71048:144;71072:10;71064:5;:18;71048:144;;;71128:9;:47;71139:34;71159:6;71167:5;71139:19;:34::i;:::-;71128:47;;;;;;;;;;;;;;-1:-1:-1;71128:47:0;71112:63;;;;;;;;;;;-1:-1:-1;;71112:63:0;;;;;;;;;;;;;;;;;;;;;;;;;;;71128:47;;71112:63;;;71128:47;71112:63;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:6;71119:5;71112:13;;;;;;;;;;;;;;;;;:63;71084:7;;71048:144;;69203:106;65812:12;:10;:12::i;:::-;-1:-1:-1;;;;;65801:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;65801:23:0;;65793:68;;;;-1:-1:-1;;;65793:68:0;;;;;;;:::i;:::-;69278:23;;::::1;::::0;:12:::1;::::0;:23:::1;::::0;::::1;::::0;::::1;:::i;67075:53::-:0;67111:17;67075:53;:::o;55795:285::-;55927:41;55946:12;:10;:12::i;:::-;55960:7;55927:18;:41::i;:::-;55919:103;;;;-1:-1:-1;;;55919:103:0;;;;;;;:::i;:::-;56033:39;56047:4;56053:2;56057:7;56066:5;56033:13;:39::i;:::-;55795:285;;;;:::o;69327:684::-;69402:12;;;;69394:57;;;;-1:-1:-1;;;69394:57:0;;;;;;;:::i;:::-;69487:1;69470:14;:18;:55;;;;;67189:2;69492:14;:33;;69470:55;69462:101;;;;-1:-1:-1;;;69462:101:0;;;;;;;:::i;:::-;67236:4;69582:33;69600:14;69582:13;:11;:13::i;:::-;:17;;:33::i;:::-;:46;;69574:100;;;;-1:-1:-1;;;69574:100:0;;;;;;;:::i;:::-;69706:29;67111:17;69720:14;69706:13;:29::i;:::-;69693:9;:42;;69685:86;;;;-1:-1:-1;;;69685:86:0;;;;;;;:::i;:::-;69796:6;69792:210;69812:14;69808:1;:18;69792:210;;;69848:14;69865:13;:11;:13::i;:::-;69848:30;;67236:4;69897:13;:11;:13::i;:::-;:25;69893:98;;;69943:32;69953:10;69965:9;69943;:32::i;:::-;-1:-1:-1;69828:3:0;;69792:210;;51865:792;51938:13;51972:16;51980:7;51972;:16::i;:::-;51964:76;;;;-1:-1:-1;;;51964:76:0;;;;;;;:::i;:::-;52053:23;52079:19;;;:10;:19;;;;;;;;52053:45;;;;;;;;;;;-1:-1:-1;;52053:45:0;;;;;;;;;;;;;;;;;;;;;;;;;;;52079:19;;52053:45;;;52079:19;52053:45;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;52109:18;52130:9;:7;:9::i;:::-;52109:30;;52221:4;52215:18;52237:1;52215:23;52211:72;;;-1:-1:-1;52262:9:0;-1:-1:-1;52255:16:0;;52211:72;52387:23;;:27;52383:108;;52462:4;52468:9;52445:33;;;;;;;;;:::i;:::-;;;;;;;;;;;;;52431:48;;;;;;52383:108;52623:4;52629:18;:7;:16;:18::i;:::-;52606:42;;;;;;;;;:::i;:::-;;;;;;;;;;;;;52592:57;;;;51865:792;;;:::o;67200:40::-;67236:4;67200:40;:::o;67454:29::-;;;;:::o;68990:174::-;69042:13;69082;:11;:13::i;:::-;69076:3;:19;69068:58;;;;-1:-1:-1;;;69068:58:0;;;;;;;:::i;:::-;69144:12;69137:19;;;;;;;;;;;;;-1:-1:-1;;69137:19:0;;;;;;;;;;;;;;;;;;;;;;;;;;;69144:12;69137:19;;69144:12;69137:19;;;;;;;;;;;;;;;;;;;;;;;;67150:41;67189:2;67150:41;:::o;54966:164::-;-1:-1:-1;;;;;55087:25:0;;;55063:4;55087:25;;;:18;:25;;;;;;;;:35;;;;;;;;;;;;;;;54966:164::o;67249:32::-;;;;;;:::o;66535:244::-;65812:12;:10;:12::i;:::-;-1:-1:-1;;;;;65801:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;65801:23:0;;65793:68;;;;-1:-1:-1;;;65793:68:0;;;;;;;:::i;:::-;-1:-1:-1;;;;;66624:22:0;::::1;66616:73;;;;-1:-1:-1::0;;;66616:73:0::1;;;;;;;:::i;:::-;66726:6;::::0;66705:38:::1;::::0;-1:-1:-1;;;;;66705:38:0;;::::1;::::0;66726:6:::1;::::0;66705:38:::1;::::0;66726:6:::1;::::0;66705:38:::1;66754:6;:17:::0;;-1:-1:-1;;;;;;66754:17:0::1;-1:-1:-1::0;;;;;66754:17:0;;;::::1;::::0;;;::::1;::::0;;66535:244::o;57547:127::-;57612:4;57636:30;:12;57658:7;57636:21;:30::i;668:106::-;756:10;668:106;:::o;63565:192::-;63640:24;;;;:15;:24;;;;;:29;;-1:-1:-1;;;;;;63640:29:0;-1:-1:-1;;;;;63640:29:0;;;;;;;;:24;;63694:23;63640:24;63694:14;:23::i;:::-;-1:-1:-1;;;;;63685:46:0;;;;;;;;;;;63565:192;;:::o;44279:123::-;44348:7;44375:19;44383:3;44375:7;:19::i;57841:355::-;57934:4;57959:16;57967:7;57959;:16::i;:::-;57951:73;;;;-1:-1:-1;;;57951:73:0;;;;;;;:::i;:::-;58035:13;58051:23;58066:7;58051:14;:23::i;:::-;58035:39;;58104:5;-1:-1:-1;;;;;58093:16:0;:7;-1:-1:-1;;;;;58093:16:0;;:51;;;;58137:7;-1:-1:-1;;;;;58113:31:0;:20;58125:7;58113:11;:20::i;:::-;-1:-1:-1;;;;;58113:31:0;;58093:51;:94;;;;58148:39;58172:5;58179:7;58148:23;:39::i;:::-;58085:103;57841:355;-1:-1:-1;;;;57841:355:0:o;60977:599::-;61102:4;-1:-1:-1;;;;;61075:31:0;:23;61090:7;61075:14;:23::i;:::-;-1:-1:-1;;;;;61075:31:0;;61067:85;;;;-1:-1:-1;;;61067:85:0;;;;;;;:::i;:::-;-1:-1:-1;;;;;61189:16:0;;61181:65;;;;-1:-1:-1;;;61181:65:0;;;;;;;:::i;:::-;61259:39;61280:4;61286:2;61290:7;61259:20;:39::i;:::-;61363:29;61380:1;61384:7;61363:8;:29::i;:::-;-1:-1:-1;;;;;61405:19:0;;;;;;:13;:19;;;;;:35;;61432:7;61405:26;:35::i;:::-;-1:-1:-1;;;;;;61451:17:0;;;;;;:13;:17;;;;;:30;;61473:7;61451:21;:30::i;:::-;-1:-1:-1;61494:29:0;:12;61511:7;61520:2;61494:16;:29::i;:::-;;61560:7;61556:2;-1:-1:-1;;;;;61541:27:0;61550:4;-1:-1:-1;;;;;61541:27:0;;;;;;;;;;;60977:599;;;:::o;58539:110::-;58615:26;58625:2;58629:7;58615:26;;;;;;;;;;;;:9;:26::i;14260:158::-;14318:7;14351:1;14346;:6;;14338:49;;;;-1:-1:-1;;;14338:49:0;;;;;;;:::i;:::-;-1:-1:-1;14405:5:0;;;14260:158::o;36070:137::-;36141:7;36176:22;36180:3;36192:5;36176:3;:22::i;44741:236::-;44821:7;;;;44881:22;44885:3;44897:5;44881:3;:22::i;:::-;44850:53;;;;-1:-1:-1;44741:236:0;-1:-1:-1;;;;;44741:236:0:o;62177:100::-;62250:19;;;;:8;;:19;;;;;:::i;46027:213::-;46134:7;46185:44;46190:3;46210;46216:12;46185:4;:44::i;:::-;46177:53;-1:-1:-1;46027:213:0;;;;;;:::o;56962:272::-;57076:28;57086:4;57092:2;57096:7;57076:9;:28::i;:::-;57123:48;57146:4;57152:2;57156:7;57165:5;57123:22;:48::i;:::-;57115:111;;;;-1:-1:-1;;;57115:111:0;;;;;;;:::i;13798:179::-;13856:7;13888:5;;;13912:6;;;;13904:46;;;;-1:-1:-1;;;13904:46:0;;;;;;;:::i;14677:220::-;14735:7;14759:6;14755:20;;-1:-1:-1;14774:1:0;14767:8;;14755:20;14798:5;;;14802:1;14798;:5;:1;14822:5;;;;;:10;14814:56;;;;-1:-1:-1;;;14814:56:0;;;;;;;:::i;46491:746::-;46547:13;46768:10;46764:53;;-1:-1:-1;46795:10:0;;;;;;;;;;;;-1:-1:-1;;;46795:10:0;;;;;;46764:53;46842:5;46827:12;46883:78;46890:9;;46883:78;;46916:8;;46947:2;46939:10;;;;46883:78;;;46971:19;47003:6;46993:17;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;46993:17:0;-1:-1:-1;47065:5:0;;-1:-1:-1;46971:39:0;-1:-1:-1;;;47037:10:0;;47081:117;47088:9;;47081:117;;47157:2;47150:4;:9;47145:2;:14;47132:29;;47114:6;47121:7;;;;;;;47114:15;;;;;;;;;;;:47;-1:-1:-1;;;;;47114:47:0;;;;;;;;-1:-1:-1;47184:2:0;47176:10;;;;47081:117;;;-1:-1:-1;47222:6:0;46491:746;-1:-1:-1;;;;46491:746:0:o;44040:151::-;44124:4;44148:35;44158:3;44178;44148:9;:35::i;40858:110::-;40941:19;;40858:110::o;35157:137::-;35227:4;35251:35;35259:3;35279:5;35251:7;:35::i;34850:131::-;34917:4;34941:32;34946:3;34966:5;34941:4;:32::i;43463:185::-;43552:4;43576:64;43581:3;43601;-1:-1:-1;;;;;43615:23:0;;43576:4;:64::i;58876:250::-;58972:18;58978:2;58982:7;58972:5;:18::i;:::-;59009:54;59040:1;59044:2;59048:7;59057:5;59009:22;:54::i;:::-;59001:117;;;;-1:-1:-1;;;59001:117:0;;;;;;;:::i;31108:204::-;31203:18;;31175:7;;31203:26;-1:-1:-1;31195:73:0;;;;-1:-1:-1;;;31195:73:0;;;;;;;:::i;:::-;31286:3;:11;;31298:5;31286:18;;;;;;;;;;;;;;;;31279:25;;31108:204;;;;:::o;41323:279::-;41427:19;;41390:7;;;;41427:27;-1:-1:-1;41419:74:0;;;;-1:-1:-1;;;41419:74:0;;;;;;;:::i;:::-;41506:22;41531:3;:12;;41544:5;41531:19;;;;;;;;;;;;;;;;;;41506:44;;41569:5;:10;;;41581:5;:12;;;41561:33;;;;;41323:279;;;;;:::o;42820:319::-;42914:7;42953:17;;;:12;;;:17;;;;;;43004:12;42989:13;42981:36;;;;-1:-1:-1;;;42981:36:0;;;;;;;;:::i;:::-;;43071:3;:12;;43095:1;43084:8;:12;43071:26;;;;;;;;;;;;;;;;;;:33;;;43064:40;;;42820:319;;;;;:::o;62842:604::-;62963:4;62990:15;:2;-1:-1:-1;;;;;62990:13:0;;:15::i;:::-;62985:60;;-1:-1:-1;63029:4:0;63022:11;;62985:60;63055:23;63081:252;-1:-1:-1;;;63194:12:0;:10;:12::i;:::-;63221:4;63240:7;63262:5;63097:181;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;-1:-1:-1;;;;;63097:181:0;;;;;;;-1:-1:-1;;;;;63097:181:0;;;;;;;;;;;63081:252;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;63081:15:0;;;:252;:15;:252::i;:::-;63055:278;;63344:13;63371:10;63360:32;;;;;;;;;;;;:::i;:::-;-1:-1:-1;;;;;;63411:26:0;-1:-1:-1;;;63411:26:0;;-1:-1:-1;;;62842:604:0;;;;;;:::o;40638:125::-;40709:4;40733:17;;;:12;;;;;:17;;;;;;:22;;;40638:125::o;28810:1544::-;28876:4;29015:19;;;:12;;;:19;;;;;;29051:15;;29047:1300;;29486:18;;-1:-1:-1;;29437:14:0;;;;29486:22;;;;29413:21;;29486:3;;:22;;29773;;;;;;;;;;;;;;29753:42;;29919:9;29890:3;:11;;29902:13;29890:26;;;;;;;;;;;;;;;;;;;:38;;;;29996:23;;;30038:1;29996:12;;;:23;;;;;;30022:17;;;29996:43;;30148:17;;29996:3;;30148:17;;;;;;;;;;;;;;;;;;;;;;30243:3;:12;;:19;30256:5;30243:19;;;;;;;;;;;30236:26;;;30286:4;30279:11;;;;;;;;29047:1300;30330:5;30323:12;;;;;28220:414;28283:4;28305:21;28315:3;28320:5;28305:9;:21::i;:::-;28300:327;;-1:-1:-1;28343:23:0;;;;;;;;:11;:23;;;;;;;;;;;;;28526:18;;28504:19;;;:12;;;:19;;;;;;:40;;;;28559:11;;28300:327;-1:-1:-1;28610:5:0;28603:12;;38138:692;38214:4;38349:17;;;:12;;;:17;;;;;;38383:13;38379:444;;-1:-1:-1;;38468:38:0;;;;;;;;;;;;;;;;;;38450:57;;;;;;;;:12;:57;;;;;;;;;;;;;;;;;;;;;;;;38665:19;;38645:17;;;:12;;;:17;;;;;;;:39;38699:11;;38379:444;38779:5;38743:3;:12;;38767:1;38756:8;:12;38743:26;;;;;;;;;;;;;;;;;;:33;;:41;;;;38806:5;38799:12;;;;;59462:404;-1:-1:-1;;;;;59542:16:0;;59534:61;;;;-1:-1:-1;;;59534:61:0;;;;;;;:::i;:::-;59615:16;59623:7;59615;:16::i;:::-;59614:17;59606:58;;;;-1:-1:-1;;;59606:58:0;;;;;;;:::i;:::-;59677:45;59706:1;59710:2;59714:7;59677:20;:45::i;:::-;-1:-1:-1;;;;;59735:17:0;;;;;;:13;:17;;;;;:30;;59757:7;59735:21;:30::i;:::-;-1:-1:-1;59778:29:0;:12;59795:7;59804:2;59778:16;:29::i;:::-;-1:-1:-1;59825:33:0;;59850:7;;-1:-1:-1;;;;;59825:33:0;;;59842:1;;59825:33;;59842:1;;59825:33;59462:404;;:::o;19239:422::-;19606:20;19645:8;;;19239:422::o;22157:195::-;22260:12;22292:52;22314:6;22322:4;22328:1;22331:12;22260;23461:18;23472:6;23461:10;:18::i;:::-;23453:60;;;;-1:-1:-1;;;23453:60:0;;;;;;;:::i;:::-;23587:12;23601:23;23628:6;-1:-1:-1;;;;;23628:11:0;23648:5;23656:4;23628:33;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;23586:75;;;;23679:52;23697:7;23706:10;23718:12;23679:17;:52::i;:::-;23672:59;23209:530;-1:-1:-1;;;;;;;23209:530:0:o;25749:742::-;25864:12;25893:7;25889:595;;;-1:-1:-1;25924:10:0;25917:17;;25889:595;26038:17;;:21;26034:439;;26301:10;26295:17;26362:15;26349:10;26345:2;26341:19;26334:44;26249:148;26444:12;26437:20;;-1:-1:-1;;;26437:20:0;;;;;;;;:::i;-1:-1:-1:-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;14:589:1;;110:18;151:2;143:6;140:14;137:2;;;157:9;137:2;197;191:9;270:2;247:17;;-1:-1:-1;;243:31:1;231:44;;277:4;227:55;297:18;;;317:22;;;294:46;291:2;;;343:9;291:2;370;363:22;418;;;403:6;-1:-1:-1;403:6:1;455:16;;;452:25;-1:-1:-1;449:2:1;;;490:1;487;480:12;449:2;540:6;535:3;528:4;520:6;516:17;503:44;595:1;588:4;579:6;571;567:19;563:30;556:41;;;90:513;;;;;:::o;608:175::-;678:20;;-1:-1:-1;;;;;727:31:1;;717:42;;707:2;;773:1;770;763:12;788:233;;886:3;879:4;871:6;867:17;863:27;853:2;;908:5;901;894:20;853:2;934:81;1011:3;1002:6;989:20;982:4;974:6;970:17;934:81;:::i;1026:198::-;;1138:2;1126:9;1117:7;1113:23;1109:32;1106:2;;;1159:6;1151;1144:22;1106:2;1187:31;1208:9;1187:31;:::i;1229:274::-;;;1358:2;1346:9;1337:7;1333:23;1329:32;1326:2;;;1379:6;1371;1364:22;1326:2;1407:31;1428:9;1407:31;:::i;:::-;1397:41;;1457:40;1493:2;1482:9;1478:18;1457:40;:::i;:::-;1447:50;;1316:187;;;;;:::o;1508:342::-;;;;1654:2;1642:9;1633:7;1629:23;1625:32;1622:2;;;1675:6;1667;1660:22;1622:2;1703:31;1724:9;1703:31;:::i;:::-;1693:41;;1753:40;1789:2;1778:9;1774:18;1753:40;:::i;:::-;1743:50;;1840:2;1829:9;1825:18;1812:32;1802:42;;1612:238;;;;;:::o;1855:702::-;;;;;2027:3;2015:9;2006:7;2002:23;1998:33;1995:2;;;2049:6;2041;2034:22;1995:2;2077:31;2098:9;2077:31;:::i;:::-;2067:41;;2127:40;2163:2;2152:9;2148:18;2127:40;:::i;:::-;2117:50;;2214:2;2203:9;2199:18;2186:32;2176:42;;2269:2;2258:9;2254:18;2241:32;2296:18;2288:6;2285:30;2282:2;;;2333:6;2325;2318:22;2282:2;2361:22;;2414:4;2406:13;;2402:27;-1:-1:-1;2392:2:1;;2448:6;2440;2433:22;2392:2;2476:75;2543:7;2538:2;2525:16;2520:2;2516;2512:11;2476:75;:::i;:::-;2466:85;;;1985:572;;;;;;;:::o;2562:369::-;;;2688:2;2676:9;2667:7;2663:23;2659:32;2656:2;;;2709:6;2701;2694:22;2656:2;2737:31;2758:9;2737:31;:::i;:::-;2727:41;;2818:2;2807:9;2803:18;2790:32;2865:5;2858:13;2851:21;2844:5;2841:32;2831:2;;2892:6;2884;2877:22;2831:2;2920:5;2910:15;;;2646:285;;;;;:::o;2936:266::-;;;3065:2;3053:9;3044:7;3040:23;3036:32;3033:2;;;3086:6;3078;3071:22;3033:2;3114:31;3135:9;3114:31;:::i;:::-;3104:41;3192:2;3177:18;;;;3164:32;;-1:-1:-1;;;3023:179:1:o;3207:194::-;;3330:2;3318:9;3309:7;3305:23;3301:32;3298:2;;;3351:6;3343;3336:22;3298:2;-1:-1:-1;3379:16:1;;3288:113;-1:-1:-1;3288:113:1:o;3406:257::-;;3517:2;3505:9;3496:7;3492:23;3488:32;3485:2;;;3538:6;3530;3523:22;3485:2;3582:9;3569:23;3601:32;3627:5;3601:32;:::i;3668:261::-;;3790:2;3778:9;3769:7;3765:23;3761:32;3758:2;;;3811:6;3803;3796:22;3758:2;3848:9;3842:16;3867:32;3893:5;3867:32;:::i;3934:344::-;;4056:2;4044:9;4035:7;4031:23;4027:32;4024:2;;;4077:6;4069;4062:22;4024:2;4122:9;4109:23;4155:18;4147:6;4144:30;4141:2;;;4192:6;4184;4177:22;4141:2;4220:52;4264:7;4255:6;4244:9;4240:22;4220:52;:::i;4283:190::-;;4395:2;4383:9;4374:7;4370:23;4366:32;4363:2;;;4416:6;4408;4401:22;4363:2;-1:-1:-1;4444:23:1;;4353:120;-1:-1:-1;4353:120:1:o;4478:412::-;;;4617:2;4605:9;4596:7;4592:23;4588:32;4585:2;;;4638:6;4630;4623:22;4585:2;4679:9;4666:23;4656:33;;4740:2;4729:9;4725:18;4712:32;4767:18;4759:6;4756:30;4753:2;;;4804:6;4796;4789:22;4753:2;4832:52;4876:7;4867:6;4856:9;4852:22;4832:52;:::i;:::-;4822:62;;;4575:315;;;;;:::o;4895:259::-;;4976:5;4970:12;5003:6;4998:3;4991:19;5019:63;5075:6;5068:4;5063:3;5059:14;5052:4;5045:5;5041:16;5019:63;:::i;:::-;5136:2;5115:15;-1:-1:-1;;5111:29:1;5102:39;;;;5143:4;5098:50;;4946:208;-1:-1:-1;;4946:208:1:o;5159:274::-;;5326:6;5320:13;5342:53;5388:6;5383:3;5376:4;5368:6;5364:17;5342:53;:::i;:::-;5411:16;;;;;5296:137;-1:-1:-1;;5296:137:1:o;5438:830::-;;5597:3;5632:6;5626:13;5658:1;5690:2;5679:9;5675:18;5707:1;5702:126;;;;5842:1;5837:406;;;;5668:575;;5702:126;-1:-1:-1;;5735:24:1;;5723:37;;5808:1;5793:17;;5812:4;5789:28;5780:38;;;-1:-1:-1;5702:126:1;;5837:406;5887:1;5876:9;5872:17;5914:6;5909:3;5902:19;5944:4;5991:2;5986:3;5976:18;6016:3;6032:165;6046:6;6043:1;6040:13;6032:165;;;6124:14;;6111:11;;;6104:35;6167:16;;;;6061:10;;6032:165;;;-1:-1:-1;;;6217:16:1;;;-1:-1:-1;5668:575:1;-1:-1:-1;6259:3:1;;5576:692;-1:-1:-1;;;;;5576:692:1:o;6273:470::-;;6490:6;6484:13;6506:53;6552:6;6547:3;6540:4;6532:6;6528:17;6506:53;:::i;:::-;6622:13;;6581:16;;;;6644:57;6622:13;6581:16;6678:4;6666:17;;6644:57;:::i;:::-;6717:20;;6460:283;-1:-1:-1;;;;6460:283:1:o;6748:203::-;-1:-1:-1;;;;;6912:32:1;;;;6894:51;;6882:2;6867:18;;6849:102::o;6956:506::-;-1:-1:-1;;;;;7241:15:1;;;7223:34;;7293:15;;7288:2;7273:18;;7266:43;7340:2;7325:18;;7318:34;;;7388:3;7383:2;7368:18;;7361:31;;;6956:506;;7409:47;;7436:19;;7428:6;7409:47;:::i;:::-;7401:55;7175:287;-1:-1:-1;;;;;;7175:287:1:o;7467:397::-;;7709:1;7705;7700:3;7696:11;7692:19;7684:6;7680:32;7669:9;7662:51;7749:6;7744:2;7733:9;7729:18;7722:34;7792:2;7787;7776:9;7772:18;7765:30;7812:46;7854:2;7843:9;7839:18;7831:6;7812:46;:::i;:::-;7804:54;7652:212;-1:-1:-1;;;;;7652:212:1:o;7869:808::-;;8060:2;8100;8089:9;8085:18;8130:2;8119:9;8112:21;8153:6;8188;8182:13;8219:6;8211;8204:22;8257:2;8246:9;8242:18;8235:25;;8320:2;8314;8306:6;8302:15;8291:9;8287:31;8283:40;8269:54;;8358:2;8350:6;8346:15;8379:4;8392:256;8406:6;8403:1;8400:13;8392:256;;;8499:2;8495:7;8483:9;8475:6;8471:22;8467:36;8462:3;8455:49;8527:41;8561:6;8552;8546:13;8527:41;:::i;:::-;8517:51;-1:-1:-1;8626:12:1;;;;8591:15;;;;8428:1;8421:9;8392:256;;;-1:-1:-1;8665:6:1;;8040:637;-1:-1:-1;;;;;;;8040:637:1:o;8682:635::-;8853:2;8905:21;;;8975:13;;8878:18;;;8997:22;;;8682:635;;8853:2;9076:15;;;;9050:2;9035:18;;;8682:635;9122:169;9136:6;9133:1;9130:13;9122:169;;;9197:13;;9185:26;;9266:15;;;;9231:12;;;;9158:1;9151:9;9122:169;;;-1:-1:-1;9308:3:1;;8833:484;-1:-1:-1;;;;;;8833:484:1:o;9322:187::-;9487:14;;9480:22;9462:41;;9450:2;9435:18;;9417:92::o;9514:221::-;;9663:2;9652:9;9645:21;9683:46;9725:2;9714:9;9710:18;9702:6;9683:46;:::i;9740:398::-;9942:2;9924:21;;;9981:2;9961:18;;;9954:30;10020:34;10015:2;10000:18;;9993:62;-1:-1:-1;;;10086:2:1;10071:18;;10064:32;10128:3;10113:19;;9914:224::o;10143:414::-;10345:2;10327:21;;;10384:2;10364:18;;;10357:30;10423:34;10418:2;10403:18;;10396:62;-1:-1:-1;;;10489:2:1;10474:18;;10467:48;10547:3;10532:19;;10317:240::o;10562:402::-;10764:2;10746:21;;;10803:2;10783:18;;;10776:30;10842:34;10837:2;10822:18;;10815:62;-1:-1:-1;;;10908:2:1;10893:18;;10886:36;10954:3;10939:19;;10736:228::o;10969:352::-;11171:2;11153:21;;;11210:2;11190:18;;;11183:30;11249;11244:2;11229:18;;11222:58;11312:2;11297:18;;11143:178::o;11326:350::-;11528:2;11510:21;;;11567:2;11547:18;;;11540:30;11606:28;11601:2;11586:18;;11579:56;11667:2;11652:18;;11500:176::o;11681:351::-;11883:2;11865:21;;;11922:2;11902:18;;;11895:30;11961:29;11956:2;11941:18;;11934:57;12023:2;12008:18;;11855:177::o;12037:356::-;12239:2;12221:21;;;12258:18;;;12251:30;12317:34;12312:2;12297:18;;12290:62;12384:2;12369:18;;12211:182::o;12398:400::-;12600:2;12582:21;;;12639:2;12619:18;;;12612:30;12678:34;12673:2;12658:18;;12651:62;-1:-1:-1;;;12744:2:1;12729:18;;12722:34;12788:3;12773:19;;12572:226::o;12803:349::-;13005:2;12987:21;;;13044:2;13024:18;;;13017:30;13083:27;13078:2;13063:18;;13056:55;13143:2;13128:18;;12977:175::o;13157:350::-;13359:2;13341:21;;;13398:2;13378:18;;;13371:30;13437:28;13432:2;13417:18;;13410:56;13498:2;13483:18;;13331:176::o;13512:355::-;13714:2;13696:21;;;13753:2;13733:18;;;13726:30;13792:33;13787:2;13772:18;;13765:61;13858:2;13843:18;;13686:181::o;13872:354::-;14074:2;14056:21;;;14113:2;14093:18;;;14086:30;14152:32;14147:2;14132:18;;14125:60;14217:2;14202:18;;14046:180::o;14638:408::-;14840:2;14822:21;;;14879:2;14859:18;;;14852:30;14918:34;14913:2;14898:18;;14891:62;-1:-1:-1;;;14984:2:1;14969:18;;14962:42;15036:3;15021:19;;14812:234::o;15051:403::-;15253:2;15235:21;;;15292:2;15272:18;;;15265:30;15331:34;15326:2;15311:18;;15304:62;-1:-1:-1;;;15397:2:1;15382:18;;15375:37;15444:3;15429:19;;15225:229::o;15459:405::-;15661:2;15643:21;;;15700:2;15680:18;;;15673:30;15739:34;15734:2;15719:18;;15712:62;-1:-1:-1;;;15805:2:1;15790:18;;15783:39;15854:3;15839:19;;15633:231::o;15869:420::-;16071:2;16053:21;;;16110:2;16090:18;;;16083:30;16149:34;16144:2;16129:18;;16122:62;16220:26;16215:2;16200:18;;16193:54;16279:3;16264:19;;16043:246::o;16294:406::-;16496:2;16478:21;;;16535:2;16515:18;;;16508:30;16574:34;16569:2;16554:18;;16547:62;-1:-1:-1;;;16640:2:1;16625:18;;16618:40;16690:3;16675:19;;16468:232::o;16705:398::-;16907:2;16889:21;;;16946:2;16926:18;;;16919:30;16985:34;16980:2;16965:18;;16958:62;-1:-1:-1;;;17051:2:1;17036:18;;17029:32;17093:3;17078:19;;16879:224::o;17108:356::-;17310:2;17292:21;;;17329:18;;;17322:30;17388:34;17383:2;17368:18;;17361:62;17455:2;17440:18;;17282:182::o;17469:397::-;17671:2;17653:21;;;17710:2;17690:18;;;17683:30;17749:34;17744:2;17729:18;;17722:62;-1:-1:-1;;;17815:2:1;17800:18;;17793:31;17856:3;17841:19;;17643:223::o;17871:408::-;18073:2;18055:21;;;18112:2;18092:18;;;18085:30;18151:34;18146:2;18131:18;;18124:62;-1:-1:-1;;;18217:2:1;18202:18;;18195:42;18269:3;18254:19;;18045:234::o;18284:356::-;18486:2;18468:21;;;18505:18;;;18498:30;18564:34;18559:2;18544:18;;18537:62;18631:2;18616:18;;18458:182::o;18645:405::-;18847:2;18829:21;;;18886:2;18866:18;;;18859:30;18925:34;18920:2;18905:18;;18898:62;-1:-1:-1;;;18991:2:1;18976:18;;18969:39;19040:3;19025:19;;18819:231::o;19055:411::-;19257:2;19239:21;;;19296:2;19276:18;;;19269:30;19335:34;19330:2;19315:18;;19308:62;-1:-1:-1;;;19401:2:1;19386:18;;19379:45;19456:3;19441:19;;19229:237::o;19471:399::-;19673:2;19655:21;;;19712:2;19692:18;;;19685:30;19751:34;19746:2;19731:18;;19724:62;-1:-1:-1;;;19817:2:1;19802:18;;19795:33;19860:3;19845:19;;19645:225::o;19875:397::-;20077:2;20059:21;;;20116:2;20096:18;;;20089:30;20155:34;20150:2;20135:18;;20128:62;-1:-1:-1;;;20221:2:1;20206:18;;20199:31;20262:3;20247:19;;20049:223::o;20277:413::-;20479:2;20461:21;;;20518:2;20498:18;;;20491:30;20557:34;20552:2;20537:18;;20530:62;-1:-1:-1;;;20623:2:1;20608:18;;20601:47;20680:3;20665:19;;20451:239::o;20695:353::-;20897:2;20879:21;;;20936:2;20916:18;;;20909:30;20975:31;20970:2;20955:18;;20948:59;21039:2;21024:18;;20869:179::o;21053:356::-;21255:2;21237:21;;;21274:18;;;21267:30;21333:34;21328:2;21313:18;;21306:62;21400:2;21385:18;;21227:182::o;21414:397::-;21616:2;21598:21;;;21655:2;21635:18;;;21628:30;21694:34;21689:2;21674:18;;21667:62;-1:-1:-1;;;21760:2:1;21745:18;;21738:31;21801:3;21786:19;;21588:223::o;21816:177::-;21962:25;;;21950:2;21935:18;;21917:76::o;21998:258::-;22070:1;22080:113;22094:6;22091:1;22088:13;22080:113;;;22170:11;;;22164:18;22151:11;;;22144:39;22116:2;22109:10;22080:113;;;22211:6;22208:1;22205:13;22202:2;;;-1:-1:-1;;22246:1:1;22228:16;;22221:27;22051:205::o;22261:133::-;-1:-1:-1;;;;;;22337:32:1;;22327:43;;22317:2;;22384:1;22381;22374:12

Swarm Source

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