ETH Price: $3,630.44 (+1.28%)

Token

ERC-20: DegentLottery (DL)
 

Overview

Max Total Supply

935 DL

Holders

33

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Filtered by Token Holder
quietstorm.eth
Balance
49 DL
0x5F9896bDEB3aBacAA17c29C4029e1CCe174fEE67
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:
DegentLottery

Compiler Version
v0.8.7+commit.e28d00a7

Optimization Enabled:
No with 200 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2022-12-22
*/

// File: erc721a/contracts/IERC721A.sol


// ERC721A Contracts v4.2.3
// Creator: Chiru Labs

pragma solidity ^0.8.4;

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

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

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

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

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

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

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

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

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

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

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

    /**
     * The `quantity` minted with ERC2309 exceeds the safety limit.
     */
    error MintERC2309QuantityExceedsLimit();

    /**
     * The `extraData` cannot be set on an unintialized ownership slot.
     */
    error OwnershipNotInitializedForExtraData();

    // =============================================================
    //                            STRUCTS
    // =============================================================

    struct TokenOwnership {
        // The address of the owner.
        address addr;
        // Stores the start time of ownership with minimal overhead for tokenomics.
        uint64 startTimestamp;
        // Whether the token has been burned.
        bool burned;
        // Arbitrary data similar to `startTimestamp` that can be set via {_extraData}.
        uint24 extraData;
    }

    // =============================================================
    //                         TOKEN COUNTERS
    // =============================================================

    /**
     * @dev Returns the total number of tokens in existence.
     * Burned tokens will reduce the count.
     * To get the total number of tokens minted, please see {_totalMinted}.
     */
    function totalSupply() external view returns (uint256);

    // =============================================================
    //                            IERC165
    // =============================================================

    /**
     * @dev Returns true if this contract implements the interface defined by
     * `interfaceId`. See the corresponding
     * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified)
     * to learn more about how these ids are created.
     *
     * This function call must use less than 30000 gas.
     */
    function supportsInterface(bytes4 interfaceId) external view returns (bool);

    // =============================================================
    //                            IERC721
    // =============================================================

    /**
     * @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,
        bytes calldata data
    ) external payable;

    /**
     * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external payable;

    /**
     * @dev Transfers `tokenId` 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 payable;

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

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

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

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

    // =============================================================
    //                        IERC721Metadata
    // =============================================================

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

    // =============================================================
    //                           IERC2309
    // =============================================================

    /**
     * @dev Emitted when tokens in `fromTokenId` to `toTokenId`
     * (inclusive) is transferred from `from` to `to`, as defined in the
     * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309) standard.
     *
     * See {_mintERC2309} for more details.
     */
    event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to);
}

// File: erc721a/contracts/ERC721A.sol


// ERC721A Contracts v4.2.3
// Creator: Chiru Labs

pragma solidity ^0.8.4;


/**
 * @dev Interface of ERC721 token receiver.
 */
interface ERC721A__IERC721Receiver {
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external returns (bytes4);
}

/**
 * @title ERC721A
 *
 * @dev Implementation of the [ERC721](https://eips.ethereum.org/EIPS/eip-721)
 * Non-Fungible Token Standard, including the Metadata extension.
 * Optimized for lower gas during batch mints.
 *
 * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...)
 * starting from `_startTokenId()`.
 *
 * Assumptions:
 *
 * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply.
 * - The maximum token ID cannot exceed 2**256 - 1 (max value of uint256).
 */
contract ERC721A is IERC721A {
    // Bypass for a `--via-ir` bug (https://github.com/chiru-labs/ERC721A/pull/364).
    struct TokenApprovalRef {
        address value;
    }

    // =============================================================
    //                           CONSTANTS
    // =============================================================

    // Mask of an entry in packed address data.
    uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1;

    // The bit position of `numberMinted` in packed address data.
    uint256 private constant _BITPOS_NUMBER_MINTED = 64;

    // The bit position of `numberBurned` in packed address data.
    uint256 private constant _BITPOS_NUMBER_BURNED = 128;

    // The bit position of `aux` in packed address data.
    uint256 private constant _BITPOS_AUX = 192;

    // Mask of all 256 bits in packed address data except the 64 bits for `aux`.
    uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1;

    // The bit position of `startTimestamp` in packed ownership.
    uint256 private constant _BITPOS_START_TIMESTAMP = 160;

    // The bit mask of the `burned` bit in packed ownership.
    uint256 private constant _BITMASK_BURNED = 1 << 224;

    // The bit position of the `nextInitialized` bit in packed ownership.
    uint256 private constant _BITPOS_NEXT_INITIALIZED = 225;

    // The bit mask of the `nextInitialized` bit in packed ownership.
    uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225;

    // The bit position of `extraData` in packed ownership.
    uint256 private constant _BITPOS_EXTRA_DATA = 232;

    // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`.
    uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1;

    // The mask of the lower 160 bits for addresses.
    uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1;

    // The maximum `quantity` that can be minted with {_mintERC2309}.
    // This limit is to prevent overflows on the address data entries.
    // For a limit of 5000, a total of 3.689e15 calls to {_mintERC2309}
    // is required to cause an overflow, which is unrealistic.
    uint256 private constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000;

    // The `Transfer` event signature is given by:
    // `keccak256(bytes("Transfer(address,address,uint256)"))`.
    bytes32 private constant _TRANSFER_EVENT_SIGNATURE =
        0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef;

    // =============================================================
    //                            STORAGE
    // =============================================================

    // The next token ID to be minted.
    uint256 private _currentIndex;

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

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to ownership details
    // An empty struct value does not necessarily mean the token is unowned.
    // See {_packedOwnershipOf} implementation for details.
    //
    // Bits Layout:
    // - [0..159]   `addr`
    // - [160..223] `startTimestamp`
    // - [224]      `burned`
    // - [225]      `nextInitialized`
    // - [232..255] `extraData`
    mapping(uint256 => uint256) private _packedOwnerships;

    // Mapping owner address to address data.
    //
    // Bits Layout:
    // - [0..63]    `balance`
    // - [64..127]  `numberMinted`
    // - [128..191] `numberBurned`
    // - [192..255] `aux`
    mapping(address => uint256) private _packedAddressData;

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

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

    // =============================================================
    //                          CONSTRUCTOR
    // =============================================================

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

    // =============================================================
    //                   TOKEN COUNTING OPERATIONS
    // =============================================================

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

    /**
     * @dev Returns the next token ID to be minted.
     */
    function _nextTokenId() internal view virtual returns (uint256) {
        return _currentIndex;
    }

    /**
     * @dev Returns the total number of tokens in existence.
     * Burned tokens will reduce the count.
     * To get the total number of tokens minted, please see {_totalMinted}.
     */
    function totalSupply() public view virtual override returns (uint256) {
        // Counter underflow is impossible as _burnCounter cannot be incremented
        // more than `_currentIndex - _startTokenId()` times.
        unchecked {
            return _currentIndex - _burnCounter - _startTokenId();
        }
    }

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

    /**
     * @dev Returns the total number of tokens burned.
     */
    function _totalBurned() internal view virtual returns (uint256) {
        return _burnCounter;
    }

    // =============================================================
    //                    ADDRESS DATA OPERATIONS
    // =============================================================

    /**
     * @dev Returns the number of tokens in `owner`'s account.
     */
    function balanceOf(address owner) public view virtual override returns (uint256) {
        if (owner == address(0)) revert BalanceQueryForZeroAddress();
        return _packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY;
    }

    /**
     * Returns the number of tokens minted by `owner`.
     */
    function _numberMinted(address owner) internal view returns (uint256) {
        return (_packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY;
    }

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

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

    /**
     * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used).
     * If there are multiple variables, please pack them into a uint64.
     */
    function _setAux(address owner, uint64 aux) internal virtual {
        uint256 packed = _packedAddressData[owner];
        uint256 auxCasted;
        // Cast `aux` with assembly to avoid redundant masking.
        assembly {
            auxCasted := aux
        }
        packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX);
        _packedAddressData[owner] = packed;
    }

    // =============================================================
    //                            IERC165
    // =============================================================

    /**
     * @dev Returns true if this contract implements the interface defined by
     * `interfaceId`. See the corresponding
     * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified)
     * to learn more about how these ids are created.
     *
     * This function call must use less than 30000 gas.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        // The interface IDs are constants representing the first 4 bytes
        // of the XOR of all function selectors in the interface.
        // See: [ERC165](https://eips.ethereum.org/EIPS/eip-165)
        // (e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`)
        return
            interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165.
            interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721.
            interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata.
    }

    // =============================================================
    //                        IERC721Metadata
    // =============================================================

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

    /**
     * @dev Returns the token collection symbol.
     */
    function symbol() public view virtual override returns (string memory) {
        return _symbol;
    }

    /**
     * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.
     */
    function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
        if (!_exists(tokenId)) revert URIQueryForNonexistentToken();

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

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

    // =============================================================
    //                     OWNERSHIPS OPERATIONS
    // =============================================================

    /**
     * @dev Returns the owner of the `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function ownerOf(uint256 tokenId) public view virtual override returns (address) {
        return address(uint160(_packedOwnershipOf(tokenId)));
    }

    /**
     * @dev Gas spent here starts off proportional to the maximum mint batch size.
     * It gradually moves to O(1) as tokens get transferred around over time.
     */
    function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) {
        return _unpackedOwnership(_packedOwnershipOf(tokenId));
    }

    /**
     * @dev Returns the unpacked `TokenOwnership` struct at `index`.
     */
    function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) {
        return _unpackedOwnership(_packedOwnerships[index]);
    }

    /**
     * @dev Initializes the ownership slot minted at `index` for efficiency purposes.
     */
    function _initializeOwnershipAt(uint256 index) internal virtual {
        if (_packedOwnerships[index] == 0) {
            _packedOwnerships[index] = _packedOwnershipOf(index);
        }
    }

    /**
     * Returns the packed ownership data of `tokenId`.
     */
    function _packedOwnershipOf(uint256 tokenId) private view returns (uint256) {
        uint256 curr = tokenId;

        unchecked {
            if (_startTokenId() <= curr)
                if (curr < _currentIndex) {
                    uint256 packed = _packedOwnerships[curr];
                    // If not burned.
                    if (packed & _BITMASK_BURNED == 0) {
                        // Invariant:
                        // There will always be an initialized ownership slot
                        // (i.e. `ownership.addr != address(0) && ownership.burned == false`)
                        // before an unintialized ownership slot
                        // (i.e. `ownership.addr == address(0) && ownership.burned == false`)
                        // Hence, `curr` will not underflow.
                        //
                        // We can directly compare the packed value.
                        // If the address is zero, packed will be zero.
                        while (packed == 0) {
                            packed = _packedOwnerships[--curr];
                        }
                        return packed;
                    }
                }
        }
        revert OwnerQueryForNonexistentToken();
    }

    /**
     * @dev Returns the unpacked `TokenOwnership` struct from `packed`.
     */
    function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) {
        ownership.addr = address(uint160(packed));
        ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP);
        ownership.burned = packed & _BITMASK_BURNED != 0;
        ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA);
    }

    /**
     * @dev Packs ownership data into a single uint256.
     */
    function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) {
        assembly {
            // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean.
            owner := and(owner, _BITMASK_ADDRESS)
            // `owner | (block.timestamp << _BITPOS_START_TIMESTAMP) | flags`.
            result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags))
        }
    }

    /**
     * @dev Returns the `nextInitialized` flag set if `quantity` equals 1.
     */
    function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) {
        // For branchless setting of the `nextInitialized` flag.
        assembly {
            // `(quantity == 1) << _BITPOS_NEXT_INITIALIZED`.
            result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1))
        }
    }

    // =============================================================
    //                      APPROVAL OPERATIONS
    // =============================================================

    /**
     * @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) public payable virtual override {
        address owner = ownerOf(tokenId);

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

        _tokenApprovals[tokenId].value = to;
        emit Approval(owner, to, tokenId);
    }

    /**
     * @dev Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) public view virtual override returns (address) {
        if (!_exists(tokenId)) revert ApprovalQueryForNonexistentToken();

        return _tokenApprovals[tokenId].value;
    }

    /**
     * @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) public virtual override {
        _operatorApprovals[_msgSenderERC721A()][operator] = approved;
        emit ApprovalForAll(_msgSenderERC721A(), operator, approved);
    }

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

    /**
     * @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. See {_mint}.
     */
    function _exists(uint256 tokenId) internal view virtual returns (bool) {
        return
            _startTokenId() <= tokenId &&
            tokenId < _currentIndex && // If within bounds,
            _packedOwnerships[tokenId] & _BITMASK_BURNED == 0; // and not burned.
    }

    /**
     * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`.
     */
    function _isSenderApprovedOrOwner(
        address approvedAddress,
        address owner,
        address msgSender
    ) private pure returns (bool result) {
        assembly {
            // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean.
            owner := and(owner, _BITMASK_ADDRESS)
            // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean.
            msgSender := and(msgSender, _BITMASK_ADDRESS)
            // `msgSender == owner || msgSender == approvedAddress`.
            result := or(eq(msgSender, owner), eq(msgSender, approvedAddress))
        }
    }

    /**
     * @dev Returns the storage slot and value for the approved address of `tokenId`.
     */
    function _getApprovedSlotAndAddress(uint256 tokenId)
        private
        view
        returns (uint256 approvedAddressSlot, address approvedAddress)
    {
        TokenApprovalRef storage tokenApproval = _tokenApprovals[tokenId];
        // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId].value`.
        assembly {
            approvedAddressSlot := tokenApproval.slot
            approvedAddress := sload(approvedAddressSlot)
        }
    }

    // =============================================================
    //                      TRANSFER OPERATIONS
    // =============================================================

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *
     * 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
    ) public payable virtual override {
        uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId);

        if (address(uint160(prevOwnershipPacked)) != from) revert TransferFromIncorrectOwner();

        (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId);

        // The nested ifs save around 20+ gas over a compound boolean condition.
        if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A()))
            if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();

        if (to == address(0)) revert TransferToZeroAddress();

        _beforeTokenTransfers(from, to, tokenId, 1);

        // Clear approvals from the previous owner.
        assembly {
            if approvedAddress {
                // This is equivalent to `delete _tokenApprovals[tokenId]`.
                sstore(approvedAddressSlot, 0)
            }
        }

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

            // Updates:
            // - `address` to the next owner.
            // - `startTimestamp` to the timestamp of transfering.
            // - `burned` to `false`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] = _packOwnershipData(
                to,
                _BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked)
            );

            // If the next slot may not have been initialized (i.e. `nextInitialized == false`) .
            if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) {
                uint256 nextTokenId = tokenId + 1;
                // If the next slot's address is zero and not burned (i.e. packed value is zero).
                if (_packedOwnerships[nextTokenId] == 0) {
                    // If the next slot is within bounds.
                    if (nextTokenId != _currentIndex) {
                        // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`.
                        _packedOwnerships[nextTokenId] = prevOwnershipPacked;
                    }
                }
            }
        }

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

    /**
     * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public payable virtual override {
        safeTransferFrom(from, to, tokenId, '');
    }

    /**
     * @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 memory _data
    ) public payable virtual override {
        transferFrom(from, to, tokenId);
        if (to.code.length != 0)
            if (!_checkContractOnERC721Received(from, to, tokenId, _data)) {
                revert TransferToNonERC721ReceiverImplementer();
            }
    }

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

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

    /**
     * @dev Private function to invoke {IERC721Receiver-onERC721Received} on a target contract.
     *
     * `from` - Previous owner of the given token ID.
     * `to` - Target address that will receive the token.
     * `tokenId` - Token ID to be transferred.
     * `_data` - Optional data to send along with the call.
     *
     * Returns whether the call correctly returned the expected magic value.
     */
    function _checkContractOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns (
            bytes4 retval
        ) {
            return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector;
        } catch (bytes memory reason) {
            if (reason.length == 0) {
                revert TransferToNonERC721ReceiverImplementer();
            } else {
                assembly {
                    revert(add(32, reason), mload(reason))
                }
            }
        }
    }

    // =============================================================
    //                        MINT OPERATIONS
    // =============================================================

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

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

        // Overflows are incredibly unrealistic.
        // `balance` and `numberMinted` have a maximum limit of 2**64.
        // `tokenId` has a maximum limit of 2**256.
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the `balance` and `numberMinted`.
            _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 1`.
            _packedOwnerships[startTokenId] = _packOwnershipData(
                to,
                _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0)
            );

            uint256 toMasked;
            uint256 end = startTokenId + quantity;

            // Use assembly to loop and emit the `Transfer` event for gas savings.
            // The duplicated `log4` removes an extra check and reduces stack juggling.
            // The assembly, together with the surrounding Solidity code, have been
            // delicately arranged to nudge the compiler into producing optimized opcodes.
            assembly {
                // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean.
                toMasked := and(to, _BITMASK_ADDRESS)
                // Emit the `Transfer` event.
                log4(
                    0, // Start of data (0, since no data).
                    0, // End of data (0, since no data).
                    _TRANSFER_EVENT_SIGNATURE, // Signature.
                    0, // `address(0)`.
                    toMasked, // `to`.
                    startTokenId // `tokenId`.
                )

                // The `iszero(eq(,))` check ensures that large values of `quantity`
                // that overflows uint256 will make the loop run out of gas.
                // The compiler will optimize the `iszero` away for performance.
                for {
                    let tokenId := add(startTokenId, 1)
                } iszero(eq(tokenId, end)) {
                    tokenId := add(tokenId, 1)
                } {
                    // Emit the `Transfer` event. Similar to above.
                    log4(0, 0, _TRANSFER_EVENT_SIGNATURE, 0, toMasked, tokenId)
                }
            }
            if (toMasked == 0) revert MintToZeroAddress();

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

    /**
     * @dev Mints `quantity` tokens and transfers them to `to`.
     *
     * This function is intended for efficient minting only during contract creation.
     *
     * It emits only one {ConsecutiveTransfer} as defined in
     * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309),
     * instead of a sequence of {Transfer} event(s).
     *
     * Calling this function outside of contract creation WILL make your contract
     * non-compliant with the ERC721 standard.
     * For full ERC721 compliance, substituting ERC721 {Transfer} event(s) with the ERC2309
     * {ConsecutiveTransfer} event is only permissible during contract creation.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `quantity` must be greater than 0.
     *
     * Emits a {ConsecutiveTransfer} event.
     */
    function _mintERC2309(address to, uint256 quantity) internal virtual {
        uint256 startTokenId = _currentIndex;
        if (to == address(0)) revert MintToZeroAddress();
        if (quantity == 0) revert MintZeroQuantity();
        if (quantity > _MAX_MINT_ERC2309_QUANTITY_LIMIT) revert MintERC2309QuantityExceedsLimit();

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

        // Overflows are unrealistic due to the above check for `quantity` to be below the limit.
        unchecked {
            // Updates:
            // - `balance += quantity`.
            // - `numberMinted += quantity`.
            //
            // We can directly add to the `balance` and `numberMinted`.
            _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1);

            // Updates:
            // - `address` to the owner.
            // - `startTimestamp` to the timestamp of minting.
            // - `burned` to `false`.
            // - `nextInitialized` to `quantity == 1`.
            _packedOwnerships[startTokenId] = _packOwnershipData(
                to,
                _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0)
            );

            emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to);

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

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

        unchecked {
            if (to.code.length != 0) {
                uint256 end = _currentIndex;
                uint256 index = end - quantity;
                do {
                    if (!_checkContractOnERC721Received(address(0), to, index++, _data)) {
                        revert TransferToNonERC721ReceiverImplementer();
                    }
                } while (index < end);
                // Reentrancy protection.
                if (_currentIndex != end) revert();
            }
        }
    }

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

    // =============================================================
    //                        BURN OPERATIONS
    // =============================================================

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

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

        address from = address(uint160(prevOwnershipPacked));

        (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId);

        if (approvalCheck) {
            // The nested ifs save around 20+ gas over a compound boolean condition.
            if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A()))
                if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved();
        }

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

        // Clear approvals from the previous owner.
        assembly {
            if approvedAddress {
                // This is equivalent to `delete _tokenApprovals[tokenId]`.
                sstore(approvedAddressSlot, 0)
            }
        }

        // Underflow of the sender's balance is impossible because we check for
        // ownership above and the recipient's balance can't realistically overflow.
        // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256.
        unchecked {
            // Updates:
            // - `balance -= 1`.
            // - `numberBurned += 1`.
            //
            // We can directly decrement the balance, and increment the number burned.
            // This is equivalent to `packed -= 1; packed += 1 << _BITPOS_NUMBER_BURNED;`.
            _packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1;

            // Updates:
            // - `address` to the last owner.
            // - `startTimestamp` to the timestamp of burning.
            // - `burned` to `true`.
            // - `nextInitialized` to `true`.
            _packedOwnerships[tokenId] = _packOwnershipData(
                from,
                (_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked)
            );

            // If the next slot may not have been initialized (i.e. `nextInitialized == false`) .
            if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) {
                uint256 nextTokenId = tokenId + 1;
                // If the next slot's address is zero and not burned (i.e. packed value is zero).
                if (_packedOwnerships[nextTokenId] == 0) {
                    // If the next slot is within bounds.
                    if (nextTokenId != _currentIndex) {
                        // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`.
                        _packedOwnerships[nextTokenId] = prevOwnershipPacked;
                    }
                }
            }
        }

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

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

    // =============================================================
    //                     EXTRA DATA OPERATIONS
    // =============================================================

    /**
     * @dev Directly sets the extra data for the ownership data `index`.
     */
    function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual {
        uint256 packed = _packedOwnerships[index];
        if (packed == 0) revert OwnershipNotInitializedForExtraData();
        uint256 extraDataCasted;
        // Cast `extraData` with assembly to avoid redundant masking.
        assembly {
            extraDataCasted := extraData
        }
        packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA);
        _packedOwnerships[index] = packed;
    }

    /**
     * @dev Called during each token transfer to set the 24bit `extraData` field.
     * Intended to be overridden by the cosumer contract.
     *
     * `previousExtraData` - the value of `extraData` before transfer.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, `tokenId` will be burned by `from`.
     * - `from` and `to` are never both zero.
     */
    function _extraData(
        address from,
        address to,
        uint24 previousExtraData
    ) internal view virtual returns (uint24) {}

    /**
     * @dev Returns the next extra data for the packed ownership data.
     * The returned result is shifted into position.
     */
    function _nextExtraData(
        address from,
        address to,
        uint256 prevOwnershipPacked
    ) private view returns (uint256) {
        uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA);
        return uint256(_extraData(from, to, extraData)) << _BITPOS_EXTRA_DATA;
    }

    // =============================================================
    //                       OTHER OPERATIONS
    // =============================================================

    /**
     * @dev Returns the message sender (defaults to `msg.sender`).
     *
     * If you are writing GSN compatible contracts, you need to override this function.
     */
    function _msgSenderERC721A() internal view virtual returns (address) {
        return msg.sender;
    }

    /**
     * @dev Converts a uint256 to its ASCII string decimal representation.
     */
    function _toString(uint256 value) internal pure virtual returns (string memory str) {
        assembly {
            // The maximum value of a uint256 contains 78 digits (1 byte per digit), but
            // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned.
            // We will need 1 word for the trailing zeros padding, 1 word for the length,
            // and 3 words for a maximum of 78 digits. Total: 5 * 0x20 = 0xa0.
            let m := add(mload(0x40), 0xa0)
            // Update the free memory pointer to allocate.
            mstore(0x40, m)
            // Assign the `str` to the end.
            str := sub(m, 0x20)
            // Zeroize the slot after the string.
            mstore(str, 0)

            // Cache the end of the memory to calculate the length later.
            let end := str

            // We write the string from rightmost digit to leftmost digit.
            // The following is essentially a do-while loop that also handles the zero case.
            // prettier-ignore
            for { let temp := value } 1 {} {
                str := sub(str, 1)
                // Write the character to the pointer.
                // The ASCII index of the '0' character is 48.
                mstore8(str, add(48, mod(temp, 10)))
                // Keep dividing `temp` until zero.
                temp := div(temp, 10)
                // prettier-ignore
                if iszero(temp) { break }
            }

            let length := sub(end, str)
            // Move the pointer 32 bytes leftwards to make room for the length.
            str := sub(str, 0x20)
            // Store the length.
            mstore(str, length)
        }
    }
}

// File: @openzeppelin/contracts/security/ReentrancyGuard.sol


// OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol)

pragma solidity ^0.8.0;

/**
 * @dev Contract module that helps prevent reentrant calls to a function.
 *
 * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
 * available, which can be applied to functions to make sure there are no nested
 * (reentrant) calls to them.
 *
 * Note that because there is a single `nonReentrant` guard, functions marked as
 * `nonReentrant` may not call one another. This can be worked around by making
 * those functions `private`, and then adding `external` `nonReentrant` entry
 * points to them.
 *
 * TIP: If you would like to learn more about reentrancy and alternative ways
 * to protect against it, check out our blog post
 * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
 */
abstract contract ReentrancyGuard {
    // Booleans are more expensive than uint256 or any type that takes up a full
    // word because each write operation emits an extra SLOAD to first read the
    // slot's contents, replace the bits taken up by the boolean, and then write
    // back. This is the compiler's defense against contract upgrades and
    // pointer aliasing, and it cannot be disabled.

    // The values being non-zero value makes deployment a bit more expensive,
    // but in exchange the refund on every call to nonReentrant will be lower in
    // amount. Since refunds are capped to a percentage of the total
    // transaction's gas, it is best to keep them low in cases like this one, to
    // increase the likelihood of the full refund coming into effect.
    uint256 private constant _NOT_ENTERED = 1;
    uint256 private constant _ENTERED = 2;

    uint256 private _status;

    constructor() {
        _status = _NOT_ENTERED;
    }

    /**
     * @dev Prevents a contract from calling itself, directly or indirectly.
     * Calling a `nonReentrant` function from another `nonReentrant`
     * function is not supported. It is possible to prevent this from happening
     * by making the `nonReentrant` function external, and making it call a
     * `private` function that does the actual work.
     */
    modifier nonReentrant() {
        // On the first call to nonReentrant, _notEntered will be true
        require(_status != _ENTERED, "ReentrancyGuard: reentrant call");

        // Any calls to nonReentrant after this point will fail
        _status = _ENTERED;

        _;

        // By storing the original value once again, a refund is triggered (see
        // https://eips.ethereum.org/EIPS/eip-2200)
        _status = _NOT_ENTERED;
    }
}

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


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

pragma solidity ^0.8.0;

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

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

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


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

pragma solidity ^0.8.0;


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

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

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

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

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

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

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

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

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

// File: DegentLottery.sol


pragma solidity ^0.8.7;




contract DegentLottery is Ownable, ERC721A, ReentrancyGuard {
    
    bool public mPublicSaleIsActive = false;
    uint256 public mMaxTokenSupply = 35000;
    string public mBaseURI = "";

    // Pricing
    uint256[5] public mAllowedQuantities = [1, 3, 5, 20, 50];
    mapping(uint256 => uint256) public mPriceMap;

    constructor() ERC721A("DegentLottery", "DL") {
        mPriceMap[1] = 0.00777 ether;
        mPriceMap[3] = 0.0233 ether;
        mPriceMap[5] = 0.0388 ether;
        mPriceMap[20] = 0.090 ether;
        mPriceMap[50] = 0.150 ether;
    }

    /// Functions for minting
    function mint(uint256 quantity) external payable {
        require(mPublicSaleIsActive, "Public sale must be active to mint");
        require(quantity + totalSupply() <= mMaxTokenSupply, "Purchase would exceed max supply");
        bool isQuantityMapped = false;
        for(uint8 i = 0; i < mAllowedQuantities.length; i++)
        {
            if(quantity == mAllowedQuantities[i]){
                isQuantityMapped = true;
                break;
            }
        }
        require(isQuantityMapped, "Quantity not accepted");
        require(msg.value >= mPriceMap[quantity], "Ether value sent was not enough");

        _safeMint(msg.sender, quantity);
    }

    function reserveMint(address[] calldata recipients, uint256[] calldata quantities) external onlyOwner nonReentrant {
        require(recipients.length == quantities.length, "Array lengths must match");
        
        // Ensure total quantities doesn't exceed supply
        uint256 totalQuantity = 0;
        for(uint256 i = 0; i < quantities.length; i++){
            totalQuantity += quantities[i];
        }
        require(totalQuantity + totalSupply() <= mMaxTokenSupply, "Reserve mint would exceed max supply");
        
        // Mint for each address
        for (uint256 i = 0; i < recipients.length; i++) {
            _safeMint(recipients[i], quantities[i]);
        }
    }

    /// Functions for managing token metadata
    function setBaseURI(string calldata baseURI) external onlyOwner {
        mBaseURI = baseURI;
    }

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

    /// Turns sale state on/off - ONLY OWNER
    function flipPublicSaleState() external onlyOwner {
         mPublicSaleIsActive = !mPublicSaleIsActive;
    }

    /// Ether withdrawal - ONLY OWNER
    function forceWithdraw(uint256 amount, address payable to) external onlyOwner nonReentrant {
        (bool success, ) = payable(to).call{value: amount}("");
        require(success, "Transfer failed.");
    }
}

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"fromTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"toTokenId","type":"uint256"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"ConsecutiveTransfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"flipPublicSaleState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address payable","name":"to","type":"address"}],"name":"forceWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"mAllowedQuantities","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mBaseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mMaxTokenSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"mPriceMap","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"mPublicSaleIsActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"},{"internalType":"uint256[]","name":"quantities","type":"uint256[]"}],"name":"reserveMint","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":"payable","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":"payable","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":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"}]

60806040526000600a60006101000a81548160ff0219169083151502179055506188b8600b5560405180602001604052806000815250600c90805190602001906200004c92919062000300565b506040518060a00160405280600160ff168152602001600360ff168152602001600560ff168152602001601460ff168152602001603260ff16815250600d9060056200009a92919062000391565b50348015620000a857600080fd5b506040518060400160405280600d81526020017f446567656e744c6f7474657279000000000000000000000000000000000000008152506040518060400160405280600281526020017f444c00000000000000000000000000000000000000000000000000000000000081525062000135620001296200022f60201b60201c565b6200023760201b60201c565b81600390805190602001906200014d92919062000300565b5080600490805190602001906200016692919062000300565b5062000177620002fb60201b60201c565b60018190555050506001600981905550661b9ac619e7a0006012600060018152602001908152602001600020819055506652c739ff4440006012600060038152602001908152602001600020819055506689d864f949000060126000600581526020019081526020016000208190555067013fbe85edc90000601260006014815260200190815260200160002081905550670214e8348c4f00006012600060328152602001908152602001600020819055506200045f565b600033905090565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050816000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600090565b8280546200030e90620003fa565b90600052602060002090601f0160209004810192826200033257600085556200037e565b82601f106200034d57805160ff19168380011785556200037e565b828001600101855582156200037e579182015b828111156200037d57825182559160200191906001019062000360565b5b5090506200038d9190620003db565b5090565b8260058101928215620003c8579160200282015b82811115620003c7578251829060ff16905591602001919060010190620003a5565b5b509050620003d79190620003db565b5090565b5b80821115620003f6576000816000905550600101620003dc565b5090565b600060028204905060018216806200041357607f821691505b602082108114156200042a576200042962000430565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b612fc3806200046f6000396000f3fe60806040526004361061019c5760003560e01c8063715018a6116100ec578063a22cb4651161008a578063c87b56dd11610064578063c87b56dd1461055d578063e985e9c51461059a578063f2fde38b146105d7578063faabbb06146106005761019c565b8063a22cb465146104ed578063b88d4fde14610516578063c0a58fa9146105325761019c565b806395d89b41116100c657806395d89b4114610466578063963565e114610491578063a0712d68146104ba578063a10866ef146104d65761019c565b8063715018a6146103f95780638734c6ea146104105780638da5cb5b1461043b5761019c565b806323b872dd1161015957806355f804b31161013357806355f804b3146103195780636352211e146103425780636ebb7f201461037f57806370a08231146103bc5761019c565b806323b872dd146102b857806342842e0e146102d45780634ef020b2146102f05761019c565b806301ffc9a7146101a157806306fdde03146101de578063081812fc14610209578063095ea7b31461024657806318160ddd146102625780632149b8ed1461028d575b600080fd5b3480156101ad57600080fd5b506101c860048036038101906101c391906123de565b61063d565b6040516101d591906127e3565b60405180910390f35b3480156101ea57600080fd5b506101f36106cf565b60405161020091906127fe565b60405180910390f35b34801561021557600080fd5b50610230600480360381019061022b9190612485565b610761565b60405161023d919061277c565b60405180910390f35b610260600480360381019061025b919061231d565b6107e0565b005b34801561026e57600080fd5b50610277610924565b6040516102849190612960565b60405180910390f35b34801561029957600080fd5b506102a261093b565b6040516102af91906127fe565b60405180910390f35b6102d260048036038101906102cd9190612207565b6109c9565b005b6102ee60048036038101906102e99190612207565b610cee565b005b3480156102fc57600080fd5b50610317600480360381019061031291906124b2565b610d0e565b005b34801561032557600080fd5b50610340600480360381019061033b9190612438565b610e1d565b005b34801561034e57600080fd5b5061036960048036038101906103649190612485565b610e3b565b604051610376919061277c565b60405180910390f35b34801561038b57600080fd5b506103a660048036038101906103a19190612485565b610e4d565b6040516103b39190612960565b60405180910390f35b3480156103c857600080fd5b506103e360048036038101906103de919061219a565b610e65565b6040516103f09190612960565b60405180910390f35b34801561040557600080fd5b5061040e610f1e565b005b34801561041c57600080fd5b50610425610f32565b60405161043291906127e3565b60405180910390f35b34801561044757600080fd5b50610450610f45565b60405161045d919061277c565b60405180910390f35b34801561047257600080fd5b5061047b610f6e565b60405161048891906127fe565b60405180910390f35b34801561049d57600080fd5b506104b860048036038101906104b3919061235d565b611000565b005b6104d460048036038101906104cf9190612485565b6111ba565b005b3480156104e257600080fd5b506104eb611351565b005b3480156104f957600080fd5b50610514600480360381019061050f91906122dd565b611385565b005b610530600480360381019061052b919061225a565b611490565b005b34801561053e57600080fd5b50610547611503565b6040516105549190612960565b60405180910390f35b34801561056957600080fd5b50610584600480360381019061057f9190612485565b611509565b60405161059191906127fe565b60405180910390f35b3480156105a657600080fd5b506105c160048036038101906105bc91906121c7565b6115a8565b6040516105ce91906127e3565b60405180910390f35b3480156105e357600080fd5b506105fe60048036038101906105f9919061219a565b61163c565b005b34801561060c57600080fd5b5061062760048036038101906106229190612485565b6116c0565b6040516106349190612960565b60405180910390f35b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061069857506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b806106c85750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b6060600380546106de90612b4a565b80601f016020809104026020016040519081016040528092919081815260200182805461070a90612b4a565b80156107575780601f1061072c57610100808354040283529160200191610757565b820191906000526020600020905b81548152906001019060200180831161073a57829003601f168201915b5050505050905090565b600061076c826116db565b6107a2576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6007600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b60006107eb82610e3b565b90508073ffffffffffffffffffffffffffffffffffffffff1661080c61173a565b73ffffffffffffffffffffffffffffffffffffffff161461086f576108388161083361173a565b6115a8565b61086e576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826007600084815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b600061092e611742565b6002546001540303905090565b600c805461094890612b4a565b80601f016020809104026020016040519081016040528092919081815260200182805461097490612b4a565b80156109c15780601f10610996576101008083540402835291602001916109c1565b820191906000526020600020905b8154815290600101906020018083116109a457829003601f168201915b505050505081565b60006109d482611747565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614610a3b576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080610a4784611815565b91509150610a5d8187610a5861173a565b61183c565b610aa957610a7286610a6d61173a565b6115a8565b610aa8576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff161415610b10576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610b1d8686866001611880565b8015610b2857600082555b600660008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600660008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000815460010191905081905550610bf685610bd2888887611886565b7c0200000000000000000000000000000000000000000000000000000000176118ae565b600560008681526020019081526020016000208190555060007c020000000000000000000000000000000000000000000000000000000084161415610c7e576000600185019050600060056000838152602001908152602001600020541415610c7c576001548114610c7b578360056000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4610ce686868660016118d9565b505050505050565b610d0983838360405180602001604052806000815250611490565b505050565b610d166118df565b60026009541415610d5c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d5390612940565b60405180910390fd5b600260098190555060008173ffffffffffffffffffffffffffffffffffffffff1683604051610d8a90612767565b60006040518083038185875af1925050503d8060008114610dc7576040519150601f19603f3d011682016040523d82523d6000602084013e610dcc565b606091505b5050905080610e10576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e0790612920565b60405180910390fd5b5060016009819055505050565b610e256118df565b8181600c9190610e36929190611f07565b505050565b6000610e4682611747565b9050919050565b60126020528060005260406000206000915090505481565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415610ecd576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600660008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b610f266118df565b610f30600061195d565b565b600a60009054906101000a900460ff1681565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b606060048054610f7d90612b4a565b80601f0160208091040260200160405190810160405280929190818152602001828054610fa990612b4a565b8015610ff65780601f10610fcb57610100808354040283529160200191610ff6565b820191906000526020600020905b815481529060010190602001808311610fd957829003601f168201915b5050505050905090565b6110086118df565b6002600954141561104e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161104590612940565b60405180910390fd5b600260098190555081819050848490501461109e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161109590612900565b60405180910390fd5b6000805b838390508110156110e7578383828181106110c0576110bf612c7e565b5b90506020020135826110d29190612a1f565b915080806110df90612bad565b9150506110a2565b50600b546110f3610924565b826110fe9190612a1f565b111561113f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611136906128c0565b60405180910390fd5b60005b858590508110156111aa5761119786868381811061116357611162612c7e565b5b9050602002016020810190611178919061219a565b85858481811061118b5761118a612c7e565b5b90506020020135611a21565b80806111a290612bad565b915050611142565b5050600160098190555050505050565b600a60009054906101000a900460ff16611209576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161120090612820565b60405180910390fd5b600b54611214610924565b8261121f9190612a1f565b1115611260576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611257906128a0565b60405180910390fd5b6000805b60058160ff1610156112ac57600d8160ff166005811061128757611286612c7e565b5b015483141561129957600191506112ac565b80806112a490612bf6565b915050611264565b50806112ed576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016112e490612860565b60405180910390fd5b6012600083815260200190815260200160002054341015611343576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161133a90612880565b60405180910390fd5b61134d3383611a21565b5050565b6113596118df565b600a60009054906101000a900460ff1615600a60006101000a81548160ff021916908315150217905550565b806008600061139261173a565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff1661143f61173a565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161148491906127e3565b60405180910390a35050565b61149b8484846109c9565b60008373ffffffffffffffffffffffffffffffffffffffff163b146114fd576114c684848484611a3f565b6114fc576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b600b5481565b6060611514826116db565b61154a576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000611554611b9f565b905060008151141561157557604051806020016040528060008152506115a0565b8061157f84611c31565b604051602001611590929190612743565b6040516020818303038152906040525b915050919050565b6000600860008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b6116446118df565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614156116b4576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116ab90612840565b60405180910390fd5b6116bd8161195d565b50565b600d81600581106116d057600080fd5b016000915090505481565b6000816116e6611742565b111580156116f5575060015482105b8015611733575060007c0100000000000000000000000000000000000000000000000000000000600560008581526020019081526020016000205416145b9050919050565b600033905090565b600090565b60008082905080611756611742565b116117de576001548110156117dd5760006005600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821614156117db575b60008114156117d15760056000836001900393508381526020019081526020016000205490506117a6565b8092505050611810565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b60008060006007600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b50505050565b60008060e883901c905060e861189d868684611c8a565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b6118e7611c93565b73ffffffffffffffffffffffffffffffffffffffff16611905610f45565b73ffffffffffffffffffffffffffffffffffffffff161461195b576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611952906128e0565b60405180910390fd5b565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050816000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b611a3b828260405180602001604052806000815250611c9b565b5050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02611a6561173a565b8786866040518563ffffffff1660e01b8152600401611a879493929190612797565b602060405180830381600087803b158015611aa157600080fd5b505af1925050508015611ad257506040513d601f19601f82011682018060405250810190611acf919061240b565b60015b611b4c573d8060008114611b02576040519150601f19603f3d011682016040523d82523d6000602084013e611b07565b606091505b50600081511415611b44576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b6060600c8054611bae90612b4a565b80601f0160208091040260200160405190810160405280929190818152602001828054611bda90612b4a565b8015611c275780601f10611bfc57610100808354040283529160200191611c27565b820191906000526020600020905b815481529060010190602001808311611c0a57829003601f168201915b5050505050905090565b606060a060405101806040526020810391506000825281835b600115611c7557600184039350600a81066030018453600a8104905080611c7057611c75565b611c4a565b50828103602084039350808452505050919050565b60009392505050565b600033905090565b611ca58383611d39565b60008373ffffffffffffffffffffffffffffffffffffffff163b14611d345760006001549050600083820390505b611ce66000868380600101945086611a3f565b611d1c576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b818110611cd3578160015414611d3157600080fd5b50505b505050565b600060015490506000821415611d7b576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b611d886000848385611880565b600160406001901b178202600660008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550611dff83611df06000866000611886565b611df985611ef7565b176118ae565b6005600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b818114611ea057808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050611e65565b506000821415611edc576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b806001819055505050611ef260008483856118d9565b505050565b60006001821460e11b9050919050565b828054611f1390612b4a565b90600052602060002090601f016020900481019282611f355760008555611f7c565b82601f10611f4e57803560ff1916838001178555611f7c565b82800160010185558215611f7c579182015b82811115611f7b578235825591602001919060010190611f60565b5b509050611f899190611f8d565b5090565b5b80821115611fa6576000816000905550600101611f8e565b5090565b6000611fbd611fb8846129a0565b61297b565b905082815260208101848484011115611fd957611fd8612ceb565b5b611fe4848285612b08565b509392505050565b600081359050611ffb81612f1a565b92915050565b60008135905061201081612f31565b92915050565b60008083601f84011261202c5761202b612ce1565b5b8235905067ffffffffffffffff81111561204957612048612cdc565b5b60208301915083602082028301111561206557612064612ce6565b5b9250929050565b60008083601f84011261208257612081612ce1565b5b8235905067ffffffffffffffff81111561209f5761209e612cdc565b5b6020830191508360208202830111156120bb576120ba612ce6565b5b9250929050565b6000813590506120d181612f48565b92915050565b6000813590506120e681612f5f565b92915050565b6000815190506120fb81612f5f565b92915050565b600082601f83011261211657612115612ce1565b5b8135612126848260208601611faa565b91505092915050565b60008083601f84011261214557612144612ce1565b5b8235905067ffffffffffffffff81111561216257612161612cdc565b5b60208301915083600182028301111561217e5761217d612ce6565b5b9250929050565b60008135905061219481612f76565b92915050565b6000602082840312156121b0576121af612cf5565b5b60006121be84828501611fec565b91505092915050565b600080604083850312156121de576121dd612cf5565b5b60006121ec85828601611fec565b92505060206121fd85828601611fec565b9150509250929050565b6000806000606084860312156122205761221f612cf5565b5b600061222e86828701611fec565b935050602061223f86828701611fec565b925050604061225086828701612185565b9150509250925092565b6000806000806080858703121561227457612273612cf5565b5b600061228287828801611fec565b945050602061229387828801611fec565b93505060406122a487828801612185565b925050606085013567ffffffffffffffff8111156122c5576122c4612cf0565b5b6122d187828801612101565b91505092959194509250565b600080604083850312156122f4576122f3612cf5565b5b600061230285828601611fec565b9250506020612313858286016120c2565b9150509250929050565b6000806040838503121561233457612333612cf5565b5b600061234285828601611fec565b925050602061235385828601612185565b9150509250929050565b6000806000806040858703121561237757612376612cf5565b5b600085013567ffffffffffffffff81111561239557612394612cf0565b5b6123a187828801612016565b9450945050602085013567ffffffffffffffff8111156123c4576123c3612cf0565b5b6123d08782880161206c565b925092505092959194509250565b6000602082840312156123f4576123f3612cf5565b5b6000612402848285016120d7565b91505092915050565b60006020828403121561242157612420612cf5565b5b600061242f848285016120ec565b91505092915050565b6000806020838503121561244f5761244e612cf5565b5b600083013567ffffffffffffffff81111561246d5761246c612cf0565b5b6124798582860161212f565b92509250509250929050565b60006020828403121561249b5761249a612cf5565b5b60006124a984828501612185565b91505092915050565b600080604083850312156124c9576124c8612cf5565b5b60006124d785828601612185565b92505060206124e885828601612001565b9150509250929050565b6124fb81612a75565b82525050565b61250a81612a99565b82525050565b600061251b826129d1565b61252581856129e7565b9350612535818560208601612b17565b61253e81612cfa565b840191505092915050565b6000612554826129dc565b61255e8185612a03565b935061256e818560208601612b17565b61257781612cfa565b840191505092915050565b600061258d826129dc565b6125978185612a14565b93506125a7818560208601612b17565b80840191505092915050565b60006125c0602283612a03565b91506125cb82612d0b565b604082019050919050565b60006125e3602683612a03565b91506125ee82612d5a565b604082019050919050565b6000612606601583612a03565b915061261182612da9565b602082019050919050565b6000612629601f83612a03565b915061263482612dd2565b602082019050919050565b600061264c602083612a03565b915061265782612dfb565b602082019050919050565b600061266f602483612a03565b915061267a82612e24565b604082019050919050565b6000612692602083612a03565b915061269d82612e73565b602082019050919050565b60006126b5601883612a03565b91506126c082612e9c565b602082019050919050565b60006126d86000836129f8565b91506126e382612ec5565b600082019050919050565b60006126fb601083612a03565b915061270682612ec8565b602082019050919050565b600061271e601f83612a03565b915061272982612ef1565b602082019050919050565b61273d81612af1565b82525050565b600061274f8285612582565b915061275b8284612582565b91508190509392505050565b6000612772826126cb565b9150819050919050565b600060208201905061279160008301846124f2565b92915050565b60006080820190506127ac60008301876124f2565b6127b960208301866124f2565b6127c66040830185612734565b81810360608301526127d88184612510565b905095945050505050565b60006020820190506127f86000830184612501565b92915050565b600060208201905081810360008301526128188184612549565b905092915050565b60006020820190508181036000830152612839816125b3565b9050919050565b60006020820190508181036000830152612859816125d6565b9050919050565b60006020820190508181036000830152612879816125f9565b9050919050565b600060208201905081810360008301526128998161261c565b9050919050565b600060208201905081810360008301526128b98161263f565b9050919050565b600060208201905081810360008301526128d981612662565b9050919050565b600060208201905081810360008301526128f981612685565b9050919050565b60006020820190508181036000830152612919816126a8565b9050919050565b60006020820190508181036000830152612939816126ee565b9050919050565b6000602082019050818103600083015261295981612711565b9050919050565b60006020820190506129756000830184612734565b92915050565b6000612985612996565b90506129918282612b7c565b919050565b6000604051905090565b600067ffffffffffffffff8211156129bb576129ba612cad565b5b6129c482612cfa565b9050602081019050919050565b600081519050919050565b600081519050919050565b600082825260208201905092915050565b600081905092915050565b600082825260208201905092915050565b600081905092915050565b6000612a2a82612af1565b9150612a3583612af1565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115612a6a57612a69612c20565b5b828201905092915050565b6000612a8082612ad1565b9050919050565b6000612a9282612ad1565b9050919050565b60008115159050919050565b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b600060ff82169050919050565b82818337600083830152505050565b60005b83811015612b35578082015181840152602081019050612b1a565b83811115612b44576000848401525b50505050565b60006002820490506001821680612b6257607f821691505b60208210811415612b7657612b75612c4f565b5b50919050565b612b8582612cfa565b810181811067ffffffffffffffff82111715612ba457612ba3612cad565b5b80604052505050565b6000612bb882612af1565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff821415612beb57612bea612c20565b5b600182019050919050565b6000612c0182612afb565b915060ff821415612c1557612c14612c20565b5b600182019050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b6000601f19601f8301169050919050565b7f5075626c69632073616c65206d7573742062652061637469766520746f206d6960008201527f6e74000000000000000000000000000000000000000000000000000000000000602082015250565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f5175616e74697479206e6f742061636365707465640000000000000000000000600082015250565b7f45746865722076616c75652073656e7420776173206e6f7420656e6f75676800600082015250565b7f507572636861736520776f756c6420657863656564206d617820737570706c79600082015250565b7f52657365727665206d696e7420776f756c6420657863656564206d617820737560008201527f70706c7900000000000000000000000000000000000000000000000000000000602082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f4172726179206c656e67746873206d757374206d617463680000000000000000600082015250565b50565b7f5472616e73666572206661696c65642e00000000000000000000000000000000600082015250565b7f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00600082015250565b612f2381612a75565b8114612f2e57600080fd5b50565b612f3a81612a87565b8114612f4557600080fd5b50565b612f5181612a99565b8114612f5c57600080fd5b50565b612f6881612aa5565b8114612f7357600080fd5b50565b612f7f81612af1565b8114612f8a57600080fd5b5056fea2646970667358221220ddd984e0ef6cca9a044dd506ad3b167de1624a3b0d0dfc8219bad2c007e124b264736f6c63430008070033

Deployed Bytecode

0x60806040526004361061019c5760003560e01c8063715018a6116100ec578063a22cb4651161008a578063c87b56dd11610064578063c87b56dd1461055d578063e985e9c51461059a578063f2fde38b146105d7578063faabbb06146106005761019c565b8063a22cb465146104ed578063b88d4fde14610516578063c0a58fa9146105325761019c565b806395d89b41116100c657806395d89b4114610466578063963565e114610491578063a0712d68146104ba578063a10866ef146104d65761019c565b8063715018a6146103f95780638734c6ea146104105780638da5cb5b1461043b5761019c565b806323b872dd1161015957806355f804b31161013357806355f804b3146103195780636352211e146103425780636ebb7f201461037f57806370a08231146103bc5761019c565b806323b872dd146102b857806342842e0e146102d45780634ef020b2146102f05761019c565b806301ffc9a7146101a157806306fdde03146101de578063081812fc14610209578063095ea7b31461024657806318160ddd146102625780632149b8ed1461028d575b600080fd5b3480156101ad57600080fd5b506101c860048036038101906101c391906123de565b61063d565b6040516101d591906127e3565b60405180910390f35b3480156101ea57600080fd5b506101f36106cf565b60405161020091906127fe565b60405180910390f35b34801561021557600080fd5b50610230600480360381019061022b9190612485565b610761565b60405161023d919061277c565b60405180910390f35b610260600480360381019061025b919061231d565b6107e0565b005b34801561026e57600080fd5b50610277610924565b6040516102849190612960565b60405180910390f35b34801561029957600080fd5b506102a261093b565b6040516102af91906127fe565b60405180910390f35b6102d260048036038101906102cd9190612207565b6109c9565b005b6102ee60048036038101906102e99190612207565b610cee565b005b3480156102fc57600080fd5b50610317600480360381019061031291906124b2565b610d0e565b005b34801561032557600080fd5b50610340600480360381019061033b9190612438565b610e1d565b005b34801561034e57600080fd5b5061036960048036038101906103649190612485565b610e3b565b604051610376919061277c565b60405180910390f35b34801561038b57600080fd5b506103a660048036038101906103a19190612485565b610e4d565b6040516103b39190612960565b60405180910390f35b3480156103c857600080fd5b506103e360048036038101906103de919061219a565b610e65565b6040516103f09190612960565b60405180910390f35b34801561040557600080fd5b5061040e610f1e565b005b34801561041c57600080fd5b50610425610f32565b60405161043291906127e3565b60405180910390f35b34801561044757600080fd5b50610450610f45565b60405161045d919061277c565b60405180910390f35b34801561047257600080fd5b5061047b610f6e565b60405161048891906127fe565b60405180910390f35b34801561049d57600080fd5b506104b860048036038101906104b3919061235d565b611000565b005b6104d460048036038101906104cf9190612485565b6111ba565b005b3480156104e257600080fd5b506104eb611351565b005b3480156104f957600080fd5b50610514600480360381019061050f91906122dd565b611385565b005b610530600480360381019061052b919061225a565b611490565b005b34801561053e57600080fd5b50610547611503565b6040516105549190612960565b60405180910390f35b34801561056957600080fd5b50610584600480360381019061057f9190612485565b611509565b60405161059191906127fe565b60405180910390f35b3480156105a657600080fd5b506105c160048036038101906105bc91906121c7565b6115a8565b6040516105ce91906127e3565b60405180910390f35b3480156105e357600080fd5b506105fe60048036038101906105f9919061219a565b61163c565b005b34801561060c57600080fd5b5061062760048036038101906106229190612485565b6116c0565b6040516106349190612960565b60405180910390f35b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061069857506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b806106c85750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b6060600380546106de90612b4a565b80601f016020809104026020016040519081016040528092919081815260200182805461070a90612b4a565b80156107575780601f1061072c57610100808354040283529160200191610757565b820191906000526020600020905b81548152906001019060200180831161073a57829003601f168201915b5050505050905090565b600061076c826116db565b6107a2576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6007600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b60006107eb82610e3b565b90508073ffffffffffffffffffffffffffffffffffffffff1661080c61173a565b73ffffffffffffffffffffffffffffffffffffffff161461086f576108388161083361173a565b6115a8565b61086e576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826007600084815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b600061092e611742565b6002546001540303905090565b600c805461094890612b4a565b80601f016020809104026020016040519081016040528092919081815260200182805461097490612b4a565b80156109c15780601f10610996576101008083540402835291602001916109c1565b820191906000526020600020905b8154815290600101906020018083116109a457829003601f168201915b505050505081565b60006109d482611747565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614610a3b576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080610a4784611815565b91509150610a5d8187610a5861173a565b61183c565b610aa957610a7286610a6d61173a565b6115a8565b610aa8576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff161415610b10576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b610b1d8686866001611880565b8015610b2857600082555b600660008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600660008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000815460010191905081905550610bf685610bd2888887611886565b7c0200000000000000000000000000000000000000000000000000000000176118ae565b600560008681526020019081526020016000208190555060007c020000000000000000000000000000000000000000000000000000000084161415610c7e576000600185019050600060056000838152602001908152602001600020541415610c7c576001548114610c7b578360056000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4610ce686868660016118d9565b505050505050565b610d0983838360405180602001604052806000815250611490565b505050565b610d166118df565b60026009541415610d5c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d5390612940565b60405180910390fd5b600260098190555060008173ffffffffffffffffffffffffffffffffffffffff1683604051610d8a90612767565b60006040518083038185875af1925050503d8060008114610dc7576040519150601f19603f3d011682016040523d82523d6000602084013e610dcc565b606091505b5050905080610e10576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e0790612920565b60405180910390fd5b5060016009819055505050565b610e256118df565b8181600c9190610e36929190611f07565b505050565b6000610e4682611747565b9050919050565b60126020528060005260406000206000915090505481565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415610ecd576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600660008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b610f266118df565b610f30600061195d565b565b600a60009054906101000a900460ff1681565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b606060048054610f7d90612b4a565b80601f0160208091040260200160405190810160405280929190818152602001828054610fa990612b4a565b8015610ff65780601f10610fcb57610100808354040283529160200191610ff6565b820191906000526020600020905b815481529060010190602001808311610fd957829003601f168201915b5050505050905090565b6110086118df565b6002600954141561104e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161104590612940565b60405180910390fd5b600260098190555081819050848490501461109e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161109590612900565b60405180910390fd5b6000805b838390508110156110e7578383828181106110c0576110bf612c7e565b5b90506020020135826110d29190612a1f565b915080806110df90612bad565b9150506110a2565b50600b546110f3610924565b826110fe9190612a1f565b111561113f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611136906128c0565b60405180910390fd5b60005b858590508110156111aa5761119786868381811061116357611162612c7e565b5b9050602002016020810190611178919061219a565b85858481811061118b5761118a612c7e565b5b90506020020135611a21565b80806111a290612bad565b915050611142565b5050600160098190555050505050565b600a60009054906101000a900460ff16611209576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161120090612820565b60405180910390fd5b600b54611214610924565b8261121f9190612a1f565b1115611260576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611257906128a0565b60405180910390fd5b6000805b60058160ff1610156112ac57600d8160ff166005811061128757611286612c7e565b5b015483141561129957600191506112ac565b80806112a490612bf6565b915050611264565b50806112ed576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016112e490612860565b60405180910390fd5b6012600083815260200190815260200160002054341015611343576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161133a90612880565b60405180910390fd5b61134d3383611a21565b5050565b6113596118df565b600a60009054906101000a900460ff1615600a60006101000a81548160ff021916908315150217905550565b806008600061139261173a565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff1661143f61173a565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161148491906127e3565b60405180910390a35050565b61149b8484846109c9565b60008373ffffffffffffffffffffffffffffffffffffffff163b146114fd576114c684848484611a3f565b6114fc576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b600b5481565b6060611514826116db565b61154a576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000611554611b9f565b905060008151141561157557604051806020016040528060008152506115a0565b8061157f84611c31565b604051602001611590929190612743565b6040516020818303038152906040525b915050919050565b6000600860008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b6116446118df565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614156116b4576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016116ab90612840565b60405180910390fd5b6116bd8161195d565b50565b600d81600581106116d057600080fd5b016000915090505481565b6000816116e6611742565b111580156116f5575060015482105b8015611733575060007c0100000000000000000000000000000000000000000000000000000000600560008581526020019081526020016000205416145b9050919050565b600033905090565b600090565b60008082905080611756611742565b116117de576001548110156117dd5760006005600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821614156117db575b60008114156117d15760056000836001900393508381526020019081526020016000205490506117a6565b8092505050611810565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b60008060006007600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b50505050565b60008060e883901c905060e861189d868684611c8a565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b6118e7611c93565b73ffffffffffffffffffffffffffffffffffffffff16611905610f45565b73ffffffffffffffffffffffffffffffffffffffff161461195b576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611952906128e0565b60405180910390fd5b565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050816000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b611a3b828260405180602001604052806000815250611c9b565b5050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02611a6561173a565b8786866040518563ffffffff1660e01b8152600401611a879493929190612797565b602060405180830381600087803b158015611aa157600080fd5b505af1925050508015611ad257506040513d601f19601f82011682018060405250810190611acf919061240b565b60015b611b4c573d8060008114611b02576040519150601f19603f3d011682016040523d82523d6000602084013e611b07565b606091505b50600081511415611b44576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b6060600c8054611bae90612b4a565b80601f0160208091040260200160405190810160405280929190818152602001828054611bda90612b4a565b8015611c275780601f10611bfc57610100808354040283529160200191611c27565b820191906000526020600020905b815481529060010190602001808311611c0a57829003601f168201915b5050505050905090565b606060a060405101806040526020810391506000825281835b600115611c7557600184039350600a81066030018453600a8104905080611c7057611c75565b611c4a565b50828103602084039350808452505050919050565b60009392505050565b600033905090565b611ca58383611d39565b60008373ffffffffffffffffffffffffffffffffffffffff163b14611d345760006001549050600083820390505b611ce66000868380600101945086611a3f565b611d1c576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b818110611cd3578160015414611d3157600080fd5b50505b505050565b600060015490506000821415611d7b576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b611d886000848385611880565b600160406001901b178202600660008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550611dff83611df06000866000611886565b611df985611ef7565b176118ae565b6005600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b818114611ea057808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050611e65565b506000821415611edc576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b806001819055505050611ef260008483856118d9565b505050565b60006001821460e11b9050919050565b828054611f1390612b4a565b90600052602060002090601f016020900481019282611f355760008555611f7c565b82601f10611f4e57803560ff1916838001178555611f7c565b82800160010185558215611f7c579182015b82811115611f7b578235825591602001919060010190611f60565b5b509050611f899190611f8d565b5090565b5b80821115611fa6576000816000905550600101611f8e565b5090565b6000611fbd611fb8846129a0565b61297b565b905082815260208101848484011115611fd957611fd8612ceb565b5b611fe4848285612b08565b509392505050565b600081359050611ffb81612f1a565b92915050565b60008135905061201081612f31565b92915050565b60008083601f84011261202c5761202b612ce1565b5b8235905067ffffffffffffffff81111561204957612048612cdc565b5b60208301915083602082028301111561206557612064612ce6565b5b9250929050565b60008083601f84011261208257612081612ce1565b5b8235905067ffffffffffffffff81111561209f5761209e612cdc565b5b6020830191508360208202830111156120bb576120ba612ce6565b5b9250929050565b6000813590506120d181612f48565b92915050565b6000813590506120e681612f5f565b92915050565b6000815190506120fb81612f5f565b92915050565b600082601f83011261211657612115612ce1565b5b8135612126848260208601611faa565b91505092915050565b60008083601f84011261214557612144612ce1565b5b8235905067ffffffffffffffff81111561216257612161612cdc565b5b60208301915083600182028301111561217e5761217d612ce6565b5b9250929050565b60008135905061219481612f76565b92915050565b6000602082840312156121b0576121af612cf5565b5b60006121be84828501611fec565b91505092915050565b600080604083850312156121de576121dd612cf5565b5b60006121ec85828601611fec565b92505060206121fd85828601611fec565b9150509250929050565b6000806000606084860312156122205761221f612cf5565b5b600061222e86828701611fec565b935050602061223f86828701611fec565b925050604061225086828701612185565b9150509250925092565b6000806000806080858703121561227457612273612cf5565b5b600061228287828801611fec565b945050602061229387828801611fec565b93505060406122a487828801612185565b925050606085013567ffffffffffffffff8111156122c5576122c4612cf0565b5b6122d187828801612101565b91505092959194509250565b600080604083850312156122f4576122f3612cf5565b5b600061230285828601611fec565b9250506020612313858286016120c2565b9150509250929050565b6000806040838503121561233457612333612cf5565b5b600061234285828601611fec565b925050602061235385828601612185565b9150509250929050565b6000806000806040858703121561237757612376612cf5565b5b600085013567ffffffffffffffff81111561239557612394612cf0565b5b6123a187828801612016565b9450945050602085013567ffffffffffffffff8111156123c4576123c3612cf0565b5b6123d08782880161206c565b925092505092959194509250565b6000602082840312156123f4576123f3612cf5565b5b6000612402848285016120d7565b91505092915050565b60006020828403121561242157612420612cf5565b5b600061242f848285016120ec565b91505092915050565b6000806020838503121561244f5761244e612cf5565b5b600083013567ffffffffffffffff81111561246d5761246c612cf0565b5b6124798582860161212f565b92509250509250929050565b60006020828403121561249b5761249a612cf5565b5b60006124a984828501612185565b91505092915050565b600080604083850312156124c9576124c8612cf5565b5b60006124d785828601612185565b92505060206124e885828601612001565b9150509250929050565b6124fb81612a75565b82525050565b61250a81612a99565b82525050565b600061251b826129d1565b61252581856129e7565b9350612535818560208601612b17565b61253e81612cfa565b840191505092915050565b6000612554826129dc565b61255e8185612a03565b935061256e818560208601612b17565b61257781612cfa565b840191505092915050565b600061258d826129dc565b6125978185612a14565b93506125a7818560208601612b17565b80840191505092915050565b60006125c0602283612a03565b91506125cb82612d0b565b604082019050919050565b60006125e3602683612a03565b91506125ee82612d5a565b604082019050919050565b6000612606601583612a03565b915061261182612da9565b602082019050919050565b6000612629601f83612a03565b915061263482612dd2565b602082019050919050565b600061264c602083612a03565b915061265782612dfb565b602082019050919050565b600061266f602483612a03565b915061267a82612e24565b604082019050919050565b6000612692602083612a03565b915061269d82612e73565b602082019050919050565b60006126b5601883612a03565b91506126c082612e9c565b602082019050919050565b60006126d86000836129f8565b91506126e382612ec5565b600082019050919050565b60006126fb601083612a03565b915061270682612ec8565b602082019050919050565b600061271e601f83612a03565b915061272982612ef1565b602082019050919050565b61273d81612af1565b82525050565b600061274f8285612582565b915061275b8284612582565b91508190509392505050565b6000612772826126cb565b9150819050919050565b600060208201905061279160008301846124f2565b92915050565b60006080820190506127ac60008301876124f2565b6127b960208301866124f2565b6127c66040830185612734565b81810360608301526127d88184612510565b905095945050505050565b60006020820190506127f86000830184612501565b92915050565b600060208201905081810360008301526128188184612549565b905092915050565b60006020820190508181036000830152612839816125b3565b9050919050565b60006020820190508181036000830152612859816125d6565b9050919050565b60006020820190508181036000830152612879816125f9565b9050919050565b600060208201905081810360008301526128998161261c565b9050919050565b600060208201905081810360008301526128b98161263f565b9050919050565b600060208201905081810360008301526128d981612662565b9050919050565b600060208201905081810360008301526128f981612685565b9050919050565b60006020820190508181036000830152612919816126a8565b9050919050565b60006020820190508181036000830152612939816126ee565b9050919050565b6000602082019050818103600083015261295981612711565b9050919050565b60006020820190506129756000830184612734565b92915050565b6000612985612996565b90506129918282612b7c565b919050565b6000604051905090565b600067ffffffffffffffff8211156129bb576129ba612cad565b5b6129c482612cfa565b9050602081019050919050565b600081519050919050565b600081519050919050565b600082825260208201905092915050565b600081905092915050565b600082825260208201905092915050565b600081905092915050565b6000612a2a82612af1565b9150612a3583612af1565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115612a6a57612a69612c20565b5b828201905092915050565b6000612a8082612ad1565b9050919050565b6000612a9282612ad1565b9050919050565b60008115159050919050565b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b600060ff82169050919050565b82818337600083830152505050565b60005b83811015612b35578082015181840152602081019050612b1a565b83811115612b44576000848401525b50505050565b60006002820490506001821680612b6257607f821691505b60208210811415612b7657612b75612c4f565b5b50919050565b612b8582612cfa565b810181811067ffffffffffffffff82111715612ba457612ba3612cad565b5b80604052505050565b6000612bb882612af1565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff821415612beb57612bea612c20565b5b600182019050919050565b6000612c0182612afb565b915060ff821415612c1557612c14612c20565b5b600182019050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b600080fd5b6000601f19601f8301169050919050565b7f5075626c69632073616c65206d7573742062652061637469766520746f206d6960008201527f6e74000000000000000000000000000000000000000000000000000000000000602082015250565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f5175616e74697479206e6f742061636365707465640000000000000000000000600082015250565b7f45746865722076616c75652073656e7420776173206e6f7420656e6f75676800600082015250565b7f507572636861736520776f756c6420657863656564206d617820737570706c79600082015250565b7f52657365727665206d696e7420776f756c6420657863656564206d617820737560008201527f70706c7900000000000000000000000000000000000000000000000000000000602082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f4172726179206c656e67746873206d757374206d617463680000000000000000600082015250565b50565b7f5472616e73666572206661696c65642e00000000000000000000000000000000600082015250565b7f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00600082015250565b612f2381612a75565b8114612f2e57600080fd5b50565b612f3a81612a87565b8114612f4557600080fd5b50565b612f5181612a99565b8114612f5c57600080fd5b50565b612f6881612aa5565b8114612f7357600080fd5b50565b612f7f81612af1565b8114612f8a57600080fd5b5056fea2646970667358221220ddd984e0ef6cca9a044dd506ad3b167de1624a3b0d0dfc8219bad2c007e124b264736f6c63430008070033

Deployed Bytecode Sourcemap

57852:2707:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;18404:639;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;19306:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;25797:218;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;25230:408;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;15057:323;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;58016:27;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;29436:2825;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;32357:193;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;60345:211;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;59914:101;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;20699:152;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;58131:44;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;16241:233;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;56970:103;;;;;;;;;;;;;:::i;:::-;;57925:39;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;56322:87;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;19482:104;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;59157:702;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;58467:682;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;60186:112;;;;;;;;;;;;;:::i;:::-;;26355:234;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;33148:407;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;57971:38;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;19692:318;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;26746:164;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57228:201;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;58068:56;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;18404:639;18489:4;18828:10;18813:25;;:11;:25;;;;:102;;;;18905:10;18890:25;;:11;:25;;;;18813:102;:179;;;;18982:10;18967:25;;:11;:25;;;;18813:179;18793:199;;18404:639;;;:::o;19306:100::-;19360:13;19393:5;19386:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;19306:100;:::o;25797:218::-;25873:7;25898:16;25906:7;25898;:16::i;:::-;25893:64;;25923:34;;;;;;;;;;;;;;25893:64;25977:15;:24;25993:7;25977:24;;;;;;;;;;;:30;;;;;;;;;;;;25970:37;;25797:218;;;:::o;25230:408::-;25319:13;25335:16;25343:7;25335;:16::i;:::-;25319:32;;25391:5;25368:28;;:19;:17;:19::i;:::-;:28;;;25364:175;;25416:44;25433:5;25440:19;:17;:19::i;:::-;25416:16;:44::i;:::-;25411:128;;25488:35;;;;;;;;;;;;;;25411:128;25364:175;25584:2;25551:15;:24;25567:7;25551:24;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;25622:7;25618:2;25602:28;;25611:5;25602:28;;;;;;;;;;;;25308:330;25230:408;;:::o;15057:323::-;15118:7;15346:15;:13;:15::i;:::-;15331:12;;15315:13;;:28;:46;15308:53;;15057:323;:::o;58016:27::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;29436:2825::-;29578:27;29608;29627:7;29608:18;:27::i;:::-;29578:57;;29693:4;29652:45;;29668:19;29652:45;;;29648:86;;29706:28;;;;;;;;;;;;;;29648:86;29748:27;29777:23;29804:35;29831:7;29804:26;:35::i;:::-;29747:92;;;;29939:68;29964:15;29981:4;29987:19;:17;:19::i;:::-;29939:24;:68::i;:::-;29934:180;;30027:43;30044:4;30050:19;:17;:19::i;:::-;30027:16;:43::i;:::-;30022:92;;30079:35;;;;;;;;;;;;;;30022:92;29934:180;30145:1;30131:16;;:2;:16;;;30127:52;;;30156:23;;;;;;;;;;;;;;30127:52;30192:43;30214:4;30220:2;30224:7;30233:1;30192:21;:43::i;:::-;30328:15;30325:160;;;30468:1;30447:19;30440:30;30325:160;30865:18;:24;30884:4;30865:24;;;;;;;;;;;;;;;;30863:26;;;;;;;;;;;;30934:18;:22;30953:2;30934:22;;;;;;;;;;;;;;;;30932:24;;;;;;;;;;;31256:146;31293:2;31342:45;31357:4;31363:2;31367:19;31342:14;:45::i;:::-;11456:8;31314:73;31256:18;:146::i;:::-;31227:17;:26;31245:7;31227:26;;;;;;;;;;;:175;;;;31573:1;11456:8;31522:19;:47;:52;31518:627;;;31595:19;31627:1;31617:7;:11;31595:33;;31784:1;31750:17;:30;31768:11;31750:30;;;;;;;;;;;;:35;31746:384;;;31888:13;;31873:11;:28;31869:242;;32068:19;32035:17;:30;32053:11;32035:30;;;;;;;;;;;:52;;;;31869:242;31746:384;31576:569;31518:627;32192:7;32188:2;32173:27;;32182:4;32173:27;;;;;;;;;;;;32211:42;32232:4;32238:2;32242:7;32251:1;32211:20;:42::i;:::-;29567:2694;;;29436:2825;;;:::o;32357:193::-;32503:39;32520:4;32526:2;32530:7;32503:39;;;;;;;;;;;;:16;:39::i;:::-;32357:193;;;:::o;60345:211::-;56208:13;:11;:13::i;:::-;53247:1:::1;53845:7;;:19;;53837:63;;;;;;;;;;;;:::i;:::-;;;;;;;;;53247:1;53978:7;:18;;;;60448:12:::2;60474:2;60466:16;;60490:6;60466:35;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60447:54;;;60520:7;60512:36;;;;;;;;;;;;:::i;:::-;;;;;;;;;60436:120;53203:1:::1;54157:7;:22;;;;60345:211:::0;;:::o;59914:101::-;56208:13;:11;:13::i;:::-;60000:7:::1;;59989:8;:18;;;;;;;:::i;:::-;;59914:101:::0;;:::o;20699:152::-;20771:7;20814:27;20833:7;20814:18;:27::i;:::-;20791:52;;20699:152;;;:::o;58131:44::-;;;;;;;;;;;;;;;;;:::o;16241:233::-;16313:7;16354:1;16337:19;;:5;:19;;;16333:60;;;16365:28;;;;;;;;;;;;;;16333:60;10400:13;16411:18;:25;16430:5;16411:25;;;;;;;;;;;;;;;;:55;16404:62;;16241:233;;;:::o;56970:103::-;56208:13;:11;:13::i;:::-;57035:30:::1;57062:1;57035:18;:30::i;:::-;56970:103::o:0;57925:39::-;;;;;;;;;;;;;:::o;56322:87::-;56368:7;56395:6;;;;;;;;;;;56388:13;;56322:87;:::o;19482:104::-;19538:13;19571:7;19564:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;19482:104;:::o;59157:702::-;56208:13;:11;:13::i;:::-;53247:1:::1;53845:7;;:19;;53837:63;;;;;;;;;;;;:::i;:::-;;;;;;;;;53247:1;53978:7;:18;;;;59312:10:::2;;:17;;59291:10;;:17;;:38;59283:75;;;;;;;;;;;;:::i;:::-;;;;;;;;;59437:21;59477:9:::0;59473:103:::2;59496:10;;:17;;59492:1;:21;59473:103;;;59551:10;;59562:1;59551:13;;;;;;;:::i;:::-;;;;;;;;59534:30;;;;;:::i;:::-;;;59515:3;;;;;:::i;:::-;;;;59473:103;;;;59627:15;;59610:13;:11;:13::i;:::-;59594;:29;;;;:::i;:::-;:48;;59586:97;;;;;;;;;;;;:::i;:::-;;;;;;;;;59743:9;59738:114;59762:10;;:17;;59758:1;:21;59738:114;;;59801:39;59811:10;;59822:1;59811:13;;;;;;;:::i;:::-;;;;;;;;;;;;;;;:::i;:::-;59826:10;;59837:1;59826:13;;;;;;;:::i;:::-;;;;;;;;59801:9;:39::i;:::-;59781:3;;;;;:::i;:::-;;;;59738:114;;;;59272:587;53203:1:::1;54157:7;:22;;;;59157:702:::0;;;;:::o;58467:682::-;58535:19;;;;;;;;;;;58527:66;;;;;;;;;;;;:::i;:::-;;;;;;;;;58640:15;;58623:13;:11;:13::i;:::-;58612:8;:24;;;;:::i;:::-;:43;;58604:88;;;;;;;;;;;;:::i;:::-;;;;;;;;;58703:21;58747:7;58743:207;58764:25;58760:1;:29;;;58743:207;;;58835:18;58854:1;58835:21;;;;;;;;;:::i;:::-;;;;58823:8;:33;58820:119;;;58895:4;58876:23;;58918:5;;58820:119;58791:3;;;;;:::i;:::-;;;;58743:207;;;;58968:16;58960:50;;;;;;;;;;;;:::i;:::-;;;;;;;;;59042:9;:19;59052:8;59042:19;;;;;;;;;;;;59029:9;:32;;59021:76;;;;;;;;;;;;:::i;:::-;;;;;;;;;59110:31;59120:10;59132:8;59110:9;:31::i;:::-;58516:633;58467:682;:::o;60186:112::-;56208:13;:11;:13::i;:::-;60271:19:::1;;;;;;;;;;;60270:20;60248:19;;:42;;;;;;;;;;;;;;;;;;60186:112::o:0;26355:234::-;26502:8;26450:18;:39;26469:19;:17;:19::i;:::-;26450:39;;;;;;;;;;;;;;;:49;26490:8;26450:49;;;;;;;;;;;;;;;;:60;;;;;;;;;;;;;;;;;;26562:8;26526:55;;26541:19;:17;:19::i;:::-;26526:55;;;26572:8;26526:55;;;;;;:::i;:::-;;;;;;;;26355:234;;:::o;33148:407::-;33323:31;33336:4;33342:2;33346:7;33323:12;:31::i;:::-;33387:1;33369:2;:14;;;:19;33365:183;;33408:56;33439:4;33445:2;33449:7;33458:5;33408:30;:56::i;:::-;33403:145;;33492:40;;;;;;;;;;;;;;33403:145;33365:183;33148:407;;;;:::o;57971:38::-;;;;:::o;19692:318::-;19765:13;19796:16;19804:7;19796;:16::i;:::-;19791:59;;19821:29;;;;;;;;;;;;;;19791:59;19863:21;19887:10;:8;:10::i;:::-;19863:34;;19940:1;19921:7;19915:21;:26;;:87;;;;;;;;;;;;;;;;;19968:7;19977:18;19987:7;19977:9;:18::i;:::-;19951:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;19915:87;19908:94;;;19692:318;;;:::o;26746:164::-;26843:4;26867:18;:25;26886:5;26867:25;;;;;;;;;;;;;;;:35;26893:8;26867:35;;;;;;;;;;;;;;;;;;;;;;;;;26860:42;;26746:164;;;;:::o;57228:201::-;56208:13;:11;:13::i;:::-;57337:1:::1;57317:22;;:8;:22;;;;57309:73;;;;;;;;;;;;:::i;:::-;;;;;;;;;57393:28;57412:8;57393:18;:28::i;:::-;57228:201:::0;:::o;58068:56::-;;;;;;;;;;;;;;;;;;;;:::o;27168:282::-;27233:4;27289:7;27270:15;:13;:15::i;:::-;:26;;:66;;;;;27323:13;;27313:7;:23;27270:66;:153;;;;;27422:1;11176:8;27374:17;:26;27392:7;27374:26;;;;;;;;;;;;:44;:49;27270:153;27250:173;;27168:282;;;:::o;49476:105::-;49536:7;49563:10;49556:17;;49476:105;:::o;14573:92::-;14629:7;14573:92;:::o;21854:1275::-;21921:7;21941:12;21956:7;21941:22;;22024:4;22005:15;:13;:15::i;:::-;:23;22001:1061;;22058:13;;22051:4;:20;22047:1015;;;22096:14;22113:17;:23;22131:4;22113:23;;;;;;;;;;;;22096:40;;22230:1;11176:8;22202:6;:24;:29;22198:845;;;22867:113;22884:1;22874:6;:11;22867:113;;;22927:17;:25;22945:6;;;;;;;22927:25;;;;;;;;;;;;22918:34;;22867:113;;;23013:6;23006:13;;;;;;22198:845;22073:989;22047:1015;22001:1061;23090:31;;;;;;;;;;;;;;21854:1275;;;;:::o;28331:485::-;28433:27;28462:23;28503:38;28544:15;:24;28560:7;28544:24;;;;;;;;;;;28503:65;;28721:18;28698:41;;28778:19;28772:26;28753:45;;28683:126;28331:485;;;:::o;27559:659::-;27708:11;27873:16;27866:5;27862:28;27853:37;;28033:16;28022:9;28018:32;28005:45;;28183:15;28172:9;28169:30;28161:5;28150:9;28147:20;28144:56;28134:66;;27559:659;;;;;:::o;34217:159::-;;;;;:::o;48785:311::-;48920:7;48940:16;11580:3;48966:19;:41;;48940:68;;11580:3;49034:31;49045:4;49051:2;49055:9;49034:10;:31::i;:::-;49026:40;;:62;;49019:69;;;48785:311;;;;;:::o;23677:450::-;23757:14;23925:16;23918:5;23914:28;23905:37;;24102:5;24088:11;24063:23;24059:41;24056:52;24049:5;24046:63;24036:73;;23677:450;;;;:::o;35041:158::-;;;;;:::o;56487:132::-;56562:12;:10;:12::i;:::-;56551:23;;:7;:5;:7::i;:::-;:23;;;56543:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;56487:132::o;57589:191::-;57663:16;57682:6;;;;;;;;;;;57663:25;;57708:8;57699:6;;:17;;;;;;;;;;;;;;;;;;57763:8;57732:40;;57753:8;57732:40;;;;;;;;;;;;57652:128;57589:191;:::o;43308:112::-;43385:27;43395:2;43399:8;43385:27;;;;;;;;;;;;:9;:27::i;:::-;43308:112;;:::o;35639:716::-;35802:4;35848:2;35823:45;;;35869:19;:17;:19::i;:::-;35890:4;35896:7;35905:5;35823:88;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;35819:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;36123:1;36106:6;:13;:18;36102:235;;;36152:40;;;;;;;;;;;;;;36102:235;36295:6;36289:13;36280:6;36276:2;36272:15;36265:38;35819:529;35992:54;;;35982:64;;;:6;:64;;;;35975:71;;;35639:716;;;;;;:::o;60023:109::-;60083:13;60116:8;60109:15;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60023:109;:::o;49683:1745::-;49748:17;50182:4;50175;50169:11;50165:22;50274:1;50268:4;50261:15;50349:4;50346:1;50342:12;50335:19;;50431:1;50426:3;50419:14;50535:3;50774:5;50756:428;50782:1;50756:428;;;50822:1;50817:3;50813:11;50806:18;;50993:2;50987:4;50983:13;50979:2;50975:22;50970:3;50962:36;51087:2;51081:4;51077:13;51069:21;;51154:4;51144:25;;51162:5;;51144:25;50756:428;;;50760:21;51223:3;51218;51214:13;51338:4;51333:3;51329:14;51322:21;;51403:6;51398:3;51391:19;49787:1634;;;49683:1745;;;:::o;48486:147::-;48623:6;48486:147;;;;;:::o;54873:98::-;54926:7;54953:10;54946:17;;54873:98;:::o;42535:689::-;42666:19;42672:2;42676:8;42666:5;:19::i;:::-;42745:1;42727:2;:14;;;:19;42723:483;;42767:11;42781:13;;42767:27;;42813:13;42835:8;42829:3;:14;42813:30;;42862:233;42893:62;42932:1;42936:2;42940:7;;;;;;42949:5;42893:30;:62::i;:::-;42888:167;;42991:40;;;;;;;;;;;;;;42888:167;43090:3;43082:5;:11;42862:233;;43177:3;43160:13;;:20;43156:34;;43182:8;;;43156:34;42748:458;;42723:483;42535:689;;;:::o;36817:2966::-;36890:20;36913:13;;36890:36;;36953:1;36941:8;:13;36937:44;;;36963:18;;;;;;;;;;;;;;36937:44;36994:61;37024:1;37028:2;37032:12;37046:8;36994:21;:61::i;:::-;37538:1;10538:2;37508:1;:26;;37507:32;37495:8;:45;37469:18;:22;37488:2;37469:22;;;;;;;;;;;;;;;;:71;;;;;;;;;;;37817:139;37854:2;37908:33;37931:1;37935:2;37939:1;37908:14;:33::i;:::-;37875:30;37896:8;37875:20;:30::i;:::-;:66;37817:18;:139::i;:::-;37783:17;:31;37801:12;37783:31;;;;;;;;;;;:173;;;;37973:16;38004:11;38033:8;38018:12;:23;38004:37;;38554:16;38550:2;38546:25;38534:37;;38926:12;38886:8;38845:1;38783:25;38724:1;38663;38636:335;39297:1;39283:12;39279:20;39237:346;39338:3;39329:7;39326:16;39237:346;;39556:7;39546:8;39543:1;39516:25;39513:1;39510;39505:59;39391:1;39382:7;39378:15;39367:26;;39237:346;;;39241:77;39628:1;39616:8;:13;39612:45;;;39638:19;;;;;;;;;;;;;;39612:45;39690:3;39674:13;:19;;;;37243:2462;;39715:60;39744:1;39748:2;39752:12;39766:8;39715:20;:60::i;:::-;36879:2904;36817:2966;;:::o;24229:324::-;24299:14;24532:1;24522:8;24519:15;24493:24;24489:46;24479:56;;24229:324;;;:::o;-1:-1:-1:-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;:::o;7:410:1:-;84:5;109:65;125:48;166:6;125:48;:::i;:::-;109:65;:::i;:::-;100:74;;197:6;190:5;183:21;235:4;228:5;224:16;273:3;264:6;259:3;255:16;252:25;249:112;;;280:79;;:::i;:::-;249:112;370:41;404:6;399:3;394;370:41;:::i;:::-;90:327;7:410;;;;;:::o;423:139::-;469:5;507:6;494:20;485:29;;523:33;550:5;523:33;:::i;:::-;423:139;;;;:::o;568:155::-;622:5;660:6;647:20;638:29;;676:41;711:5;676:41;:::i;:::-;568:155;;;;:::o;746:568::-;819:8;829:6;879:3;872:4;864:6;860:17;856:27;846:122;;887:79;;:::i;:::-;846:122;1000:6;987:20;977:30;;1030:18;1022:6;1019:30;1016:117;;;1052:79;;:::i;:::-;1016:117;1166:4;1158:6;1154:17;1142:29;;1220:3;1212:4;1204:6;1200:17;1190:8;1186:32;1183:41;1180:128;;;1227:79;;:::i;:::-;1180:128;746:568;;;;;:::o;1337:::-;1410:8;1420:6;1470:3;1463:4;1455:6;1451:17;1447:27;1437:122;;1478:79;;:::i;:::-;1437:122;1591:6;1578:20;1568:30;;1621:18;1613:6;1610:30;1607:117;;;1643:79;;:::i;:::-;1607:117;1757:4;1749:6;1745:17;1733:29;;1811:3;1803:4;1795:6;1791:17;1781:8;1777:32;1774:41;1771:128;;;1818:79;;:::i;:::-;1771:128;1337:568;;;;;:::o;1911:133::-;1954:5;1992:6;1979:20;1970:29;;2008:30;2032:5;2008:30;:::i;:::-;1911:133;;;;:::o;2050:137::-;2095:5;2133:6;2120:20;2111:29;;2149:32;2175:5;2149:32;:::i;:::-;2050:137;;;;:::o;2193:141::-;2249:5;2280:6;2274:13;2265:22;;2296:32;2322:5;2296:32;:::i;:::-;2193:141;;;;:::o;2353:338::-;2408:5;2457:3;2450:4;2442:6;2438:17;2434:27;2424:122;;2465:79;;:::i;:::-;2424:122;2582:6;2569:20;2607:78;2681:3;2673:6;2666:4;2658:6;2654:17;2607:78;:::i;:::-;2598:87;;2414:277;2353:338;;;;:::o;2711:553::-;2769:8;2779:6;2829:3;2822:4;2814:6;2810:17;2806:27;2796:122;;2837:79;;:::i;:::-;2796:122;2950:6;2937:20;2927:30;;2980:18;2972:6;2969:30;2966:117;;;3002:79;;:::i;:::-;2966:117;3116:4;3108:6;3104:17;3092:29;;3170:3;3162:4;3154:6;3150:17;3140:8;3136:32;3133:41;3130:128;;;3177:79;;:::i;:::-;3130:128;2711:553;;;;;:::o;3270:139::-;3316:5;3354:6;3341:20;3332:29;;3370:33;3397:5;3370:33;:::i;:::-;3270:139;;;;:::o;3415:329::-;3474:6;3523:2;3511:9;3502:7;3498:23;3494:32;3491:119;;;3529:79;;:::i;:::-;3491:119;3649:1;3674:53;3719:7;3710:6;3699:9;3695:22;3674:53;:::i;:::-;3664:63;;3620:117;3415:329;;;;:::o;3750:474::-;3818:6;3826;3875:2;3863:9;3854:7;3850:23;3846:32;3843:119;;;3881:79;;:::i;:::-;3843:119;4001:1;4026:53;4071:7;4062:6;4051:9;4047:22;4026:53;:::i;:::-;4016:63;;3972:117;4128:2;4154:53;4199:7;4190:6;4179:9;4175:22;4154:53;:::i;:::-;4144:63;;4099:118;3750:474;;;;;:::o;4230:619::-;4307:6;4315;4323;4372:2;4360:9;4351:7;4347:23;4343:32;4340:119;;;4378:79;;:::i;:::-;4340:119;4498:1;4523:53;4568:7;4559:6;4548:9;4544:22;4523:53;:::i;:::-;4513:63;;4469:117;4625:2;4651:53;4696:7;4687:6;4676:9;4672:22;4651:53;:::i;:::-;4641:63;;4596:118;4753:2;4779:53;4824:7;4815:6;4804:9;4800:22;4779:53;:::i;:::-;4769:63;;4724:118;4230:619;;;;;:::o;4855:943::-;4950:6;4958;4966;4974;5023:3;5011:9;5002:7;4998:23;4994:33;4991:120;;;5030:79;;:::i;:::-;4991:120;5150:1;5175:53;5220:7;5211:6;5200:9;5196:22;5175:53;:::i;:::-;5165:63;;5121:117;5277:2;5303:53;5348:7;5339:6;5328:9;5324:22;5303:53;:::i;:::-;5293:63;;5248:118;5405:2;5431:53;5476:7;5467:6;5456:9;5452:22;5431:53;:::i;:::-;5421:63;;5376:118;5561:2;5550:9;5546:18;5533:32;5592:18;5584:6;5581:30;5578:117;;;5614:79;;:::i;:::-;5578:117;5719:62;5773:7;5764:6;5753:9;5749:22;5719:62;:::i;:::-;5709:72;;5504:287;4855:943;;;;;;;:::o;5804:468::-;5869:6;5877;5926:2;5914:9;5905:7;5901:23;5897:32;5894:119;;;5932:79;;:::i;:::-;5894:119;6052:1;6077:53;6122:7;6113:6;6102:9;6098:22;6077:53;:::i;:::-;6067:63;;6023:117;6179:2;6205:50;6247:7;6238:6;6227:9;6223:22;6205:50;:::i;:::-;6195:60;;6150:115;5804:468;;;;;:::o;6278:474::-;6346:6;6354;6403:2;6391:9;6382:7;6378:23;6374:32;6371:119;;;6409:79;;:::i;:::-;6371:119;6529:1;6554:53;6599:7;6590:6;6579:9;6575:22;6554:53;:::i;:::-;6544:63;;6500:117;6656:2;6682:53;6727:7;6718:6;6707:9;6703:22;6682:53;:::i;:::-;6672:63;;6627:118;6278:474;;;;;:::o;6758:934::-;6880:6;6888;6896;6904;6953:2;6941:9;6932:7;6928:23;6924:32;6921:119;;;6959:79;;:::i;:::-;6921:119;7107:1;7096:9;7092:17;7079:31;7137:18;7129:6;7126:30;7123:117;;;7159:79;;:::i;:::-;7123:117;7272:80;7344:7;7335:6;7324:9;7320:22;7272:80;:::i;:::-;7254:98;;;;7050:312;7429:2;7418:9;7414:18;7401:32;7460:18;7452:6;7449:30;7446:117;;;7482:79;;:::i;:::-;7446:117;7595:80;7667:7;7658:6;7647:9;7643:22;7595:80;:::i;:::-;7577:98;;;;7372:313;6758:934;;;;;;;:::o;7698:327::-;7756:6;7805:2;7793:9;7784:7;7780:23;7776:32;7773:119;;;7811:79;;:::i;:::-;7773:119;7931:1;7956:52;8000:7;7991:6;7980:9;7976:22;7956:52;:::i;:::-;7946:62;;7902:116;7698:327;;;;:::o;8031:349::-;8100:6;8149:2;8137:9;8128:7;8124:23;8120:32;8117:119;;;8155:79;;:::i;:::-;8117:119;8275:1;8300:63;8355:7;8346:6;8335:9;8331:22;8300:63;:::i;:::-;8290:73;;8246:127;8031:349;;;;:::o;8386:529::-;8457:6;8465;8514:2;8502:9;8493:7;8489:23;8485:32;8482:119;;;8520:79;;:::i;:::-;8482:119;8668:1;8657:9;8653:17;8640:31;8698:18;8690:6;8687:30;8684:117;;;8720:79;;:::i;:::-;8684:117;8833:65;8890:7;8881:6;8870:9;8866:22;8833:65;:::i;:::-;8815:83;;;;8611:297;8386:529;;;;;:::o;8921:329::-;8980:6;9029:2;9017:9;9008:7;9004:23;9000:32;8997:119;;;9035:79;;:::i;:::-;8997:119;9155:1;9180:53;9225:7;9216:6;9205:9;9201:22;9180:53;:::i;:::-;9170:63;;9126:117;8921:329;;;;:::o;9256:490::-;9332:6;9340;9389:2;9377:9;9368:7;9364:23;9360:32;9357:119;;;9395:79;;:::i;:::-;9357:119;9515:1;9540:53;9585:7;9576:6;9565:9;9561:22;9540:53;:::i;:::-;9530:63;;9486:117;9642:2;9668:61;9721:7;9712:6;9701:9;9697:22;9668:61;:::i;:::-;9658:71;;9613:126;9256:490;;;;;:::o;9752:118::-;9839:24;9857:5;9839:24;:::i;:::-;9834:3;9827:37;9752:118;;:::o;9876:109::-;9957:21;9972:5;9957:21;:::i;:::-;9952:3;9945:34;9876:109;;:::o;9991:360::-;10077:3;10105:38;10137:5;10105:38;:::i;:::-;10159:70;10222:6;10217:3;10159:70;:::i;:::-;10152:77;;10238:52;10283:6;10278:3;10271:4;10264:5;10260:16;10238:52;:::i;:::-;10315:29;10337:6;10315:29;:::i;:::-;10310:3;10306:39;10299:46;;10081:270;9991:360;;;;:::o;10357:364::-;10445:3;10473:39;10506:5;10473:39;:::i;:::-;10528:71;10592:6;10587:3;10528:71;:::i;:::-;10521:78;;10608:52;10653:6;10648:3;10641:4;10634:5;10630:16;10608:52;:::i;:::-;10685:29;10707:6;10685:29;:::i;:::-;10680:3;10676:39;10669:46;;10449:272;10357:364;;;;:::o;10727:377::-;10833:3;10861:39;10894:5;10861:39;:::i;:::-;10916:89;10998:6;10993:3;10916:89;:::i;:::-;10909:96;;11014:52;11059:6;11054:3;11047:4;11040:5;11036:16;11014:52;:::i;:::-;11091:6;11086:3;11082:16;11075:23;;10837:267;10727:377;;;;:::o;11110:366::-;11252:3;11273:67;11337:2;11332:3;11273:67;:::i;:::-;11266:74;;11349:93;11438:3;11349:93;:::i;:::-;11467:2;11462:3;11458:12;11451:19;;11110:366;;;:::o;11482:::-;11624:3;11645:67;11709:2;11704:3;11645:67;:::i;:::-;11638:74;;11721:93;11810:3;11721:93;:::i;:::-;11839:2;11834:3;11830:12;11823:19;;11482:366;;;:::o;11854:::-;11996:3;12017:67;12081:2;12076:3;12017:67;:::i;:::-;12010:74;;12093:93;12182:3;12093:93;:::i;:::-;12211:2;12206:3;12202:12;12195:19;;11854:366;;;:::o;12226:::-;12368:3;12389:67;12453:2;12448:3;12389:67;:::i;:::-;12382:74;;12465:93;12554:3;12465:93;:::i;:::-;12583:2;12578:3;12574:12;12567:19;;12226:366;;;:::o;12598:::-;12740:3;12761:67;12825:2;12820:3;12761:67;:::i;:::-;12754:74;;12837:93;12926:3;12837:93;:::i;:::-;12955:2;12950:3;12946:12;12939:19;;12598:366;;;:::o;12970:::-;13112:3;13133:67;13197:2;13192:3;13133:67;:::i;:::-;13126:74;;13209:93;13298:3;13209:93;:::i;:::-;13327:2;13322:3;13318:12;13311:19;;12970:366;;;:::o;13342:::-;13484:3;13505:67;13569:2;13564:3;13505:67;:::i;:::-;13498:74;;13581:93;13670:3;13581:93;:::i;:::-;13699:2;13694:3;13690:12;13683:19;;13342:366;;;:::o;13714:::-;13856:3;13877:67;13941:2;13936:3;13877:67;:::i;:::-;13870:74;;13953:93;14042:3;13953:93;:::i;:::-;14071:2;14066:3;14062:12;14055:19;;13714:366;;;:::o;14086:398::-;14245:3;14266:83;14347:1;14342:3;14266:83;:::i;:::-;14259:90;;14358:93;14447:3;14358:93;:::i;:::-;14476:1;14471:3;14467:11;14460:18;;14086:398;;;:::o;14490:366::-;14632:3;14653:67;14717:2;14712:3;14653:67;:::i;:::-;14646:74;;14729:93;14818:3;14729:93;:::i;:::-;14847:2;14842:3;14838:12;14831:19;;14490:366;;;:::o;14862:::-;15004:3;15025:67;15089:2;15084:3;15025:67;:::i;:::-;15018:74;;15101:93;15190:3;15101:93;:::i;:::-;15219:2;15214:3;15210:12;15203:19;;14862:366;;;:::o;15234:118::-;15321:24;15339:5;15321:24;:::i;:::-;15316:3;15309:37;15234:118;;:::o;15358:435::-;15538:3;15560:95;15651:3;15642:6;15560:95;:::i;:::-;15553:102;;15672:95;15763:3;15754:6;15672:95;:::i;:::-;15665:102;;15784:3;15777:10;;15358:435;;;;;:::o;15799:379::-;15983:3;16005:147;16148:3;16005:147;:::i;:::-;15998:154;;16169:3;16162:10;;15799:379;;;:::o;16184:222::-;16277:4;16315:2;16304:9;16300:18;16292:26;;16328:71;16396:1;16385:9;16381:17;16372:6;16328:71;:::i;:::-;16184:222;;;;:::o;16412:640::-;16607:4;16645:3;16634:9;16630:19;16622:27;;16659:71;16727:1;16716:9;16712:17;16703:6;16659:71;:::i;:::-;16740:72;16808:2;16797:9;16793:18;16784:6;16740:72;:::i;:::-;16822;16890:2;16879:9;16875:18;16866:6;16822:72;:::i;:::-;16941:9;16935:4;16931:20;16926:2;16915:9;16911:18;16904:48;16969:76;17040:4;17031:6;16969:76;:::i;:::-;16961:84;;16412:640;;;;;;;:::o;17058:210::-;17145:4;17183:2;17172:9;17168:18;17160:26;;17196:65;17258:1;17247:9;17243:17;17234:6;17196:65;:::i;:::-;17058:210;;;;:::o;17274:313::-;17387:4;17425:2;17414:9;17410:18;17402:26;;17474:9;17468:4;17464:20;17460:1;17449:9;17445:17;17438:47;17502:78;17575:4;17566:6;17502:78;:::i;:::-;17494:86;;17274:313;;;;:::o;17593:419::-;17759:4;17797:2;17786:9;17782:18;17774:26;;17846:9;17840:4;17836:20;17832:1;17821:9;17817:17;17810:47;17874:131;18000:4;17874:131;:::i;:::-;17866:139;;17593:419;;;:::o;18018:::-;18184:4;18222:2;18211:9;18207:18;18199:26;;18271:9;18265:4;18261:20;18257:1;18246:9;18242:17;18235:47;18299:131;18425:4;18299:131;:::i;:::-;18291:139;;18018:419;;;:::o;18443:::-;18609:4;18647:2;18636:9;18632:18;18624:26;;18696:9;18690:4;18686:20;18682:1;18671:9;18667:17;18660:47;18724:131;18850:4;18724:131;:::i;:::-;18716:139;;18443:419;;;:::o;18868:::-;19034:4;19072:2;19061:9;19057:18;19049:26;;19121:9;19115:4;19111:20;19107:1;19096:9;19092:17;19085:47;19149:131;19275:4;19149:131;:::i;:::-;19141:139;;18868:419;;;:::o;19293:::-;19459:4;19497:2;19486:9;19482:18;19474:26;;19546:9;19540:4;19536:20;19532:1;19521:9;19517:17;19510:47;19574:131;19700:4;19574:131;:::i;:::-;19566:139;;19293:419;;;:::o;19718:::-;19884:4;19922:2;19911:9;19907:18;19899:26;;19971:9;19965:4;19961:20;19957:1;19946:9;19942:17;19935:47;19999:131;20125:4;19999:131;:::i;:::-;19991:139;;19718:419;;;:::o;20143:::-;20309:4;20347:2;20336:9;20332:18;20324:26;;20396:9;20390:4;20386:20;20382:1;20371:9;20367:17;20360:47;20424:131;20550:4;20424:131;:::i;:::-;20416:139;;20143:419;;;:::o;20568:::-;20734:4;20772:2;20761:9;20757:18;20749:26;;20821:9;20815:4;20811:20;20807:1;20796:9;20792:17;20785:47;20849:131;20975:4;20849:131;:::i;:::-;20841:139;;20568:419;;;:::o;20993:::-;21159:4;21197:2;21186:9;21182:18;21174:26;;21246:9;21240:4;21236:20;21232:1;21221:9;21217:17;21210:47;21274:131;21400:4;21274:131;:::i;:::-;21266:139;;20993:419;;;:::o;21418:::-;21584:4;21622:2;21611:9;21607:18;21599:26;;21671:9;21665:4;21661:20;21657:1;21646:9;21642:17;21635:47;21699:131;21825:4;21699:131;:::i;:::-;21691:139;;21418:419;;;:::o;21843:222::-;21936:4;21974:2;21963:9;21959:18;21951:26;;21987:71;22055:1;22044:9;22040:17;22031:6;21987:71;:::i;:::-;21843:222;;;;:::o;22071:129::-;22105:6;22132:20;;:::i;:::-;22122:30;;22161:33;22189:4;22181:6;22161:33;:::i;:::-;22071:129;;;:::o;22206:75::-;22239:6;22272:2;22266:9;22256:19;;22206:75;:::o;22287:307::-;22348:4;22438:18;22430:6;22427:30;22424:56;;;22460:18;;:::i;:::-;22424:56;22498:29;22520:6;22498:29;:::i;:::-;22490:37;;22582:4;22576;22572:15;22564:23;;22287:307;;;:::o;22600:98::-;22651:6;22685:5;22679:12;22669:22;;22600:98;;;:::o;22704:99::-;22756:6;22790:5;22784:12;22774:22;;22704:99;;;:::o;22809:168::-;22892:11;22926:6;22921:3;22914:19;22966:4;22961:3;22957:14;22942:29;;22809:168;;;;:::o;22983:147::-;23084:11;23121:3;23106:18;;22983:147;;;;:::o;23136:169::-;23220:11;23254:6;23249:3;23242:19;23294:4;23289:3;23285:14;23270:29;;23136:169;;;;:::o;23311:148::-;23413:11;23450:3;23435:18;;23311:148;;;;:::o;23465:305::-;23505:3;23524:20;23542:1;23524:20;:::i;:::-;23519:25;;23558:20;23576:1;23558:20;:::i;:::-;23553:25;;23712:1;23644:66;23640:74;23637:1;23634:81;23631:107;;;23718:18;;:::i;:::-;23631:107;23762:1;23759;23755:9;23748:16;;23465:305;;;;:::o;23776:96::-;23813:7;23842:24;23860:5;23842:24;:::i;:::-;23831:35;;23776:96;;;:::o;23878:104::-;23923:7;23952:24;23970:5;23952:24;:::i;:::-;23941:35;;23878:104;;;:::o;23988:90::-;24022:7;24065:5;24058:13;24051:21;24040:32;;23988:90;;;:::o;24084:149::-;24120:7;24160:66;24153:5;24149:78;24138:89;;24084:149;;;:::o;24239:126::-;24276:7;24316:42;24309:5;24305:54;24294:65;;24239:126;;;:::o;24371:77::-;24408:7;24437:5;24426:16;;24371:77;;;:::o;24454:86::-;24489:7;24529:4;24522:5;24518:16;24507:27;;24454:86;;;:::o;24546:154::-;24630:6;24625:3;24620;24607:30;24692:1;24683:6;24678:3;24674:16;24667:27;24546:154;;;:::o;24706:307::-;24774:1;24784:113;24798:6;24795:1;24792:13;24784:113;;;24883:1;24878:3;24874:11;24868:18;24864:1;24859:3;24855:11;24848:39;24820:2;24817:1;24813:10;24808:15;;24784:113;;;24915:6;24912:1;24909:13;24906:101;;;24995:1;24986:6;24981:3;24977:16;24970:27;24906:101;24755:258;24706:307;;;:::o;25019:320::-;25063:6;25100:1;25094:4;25090:12;25080:22;;25147:1;25141:4;25137:12;25168:18;25158:81;;25224:4;25216:6;25212:17;25202:27;;25158:81;25286:2;25278:6;25275:14;25255:18;25252:38;25249:84;;;25305:18;;:::i;:::-;25249:84;25070:269;25019:320;;;:::o;25345:281::-;25428:27;25450:4;25428:27;:::i;:::-;25420:6;25416:40;25558:6;25546:10;25543:22;25522:18;25510:10;25507:34;25504:62;25501:88;;;25569:18;;:::i;:::-;25501:88;25609:10;25605:2;25598:22;25388:238;25345:281;;:::o;25632:233::-;25671:3;25694:24;25712:5;25694:24;:::i;:::-;25685:33;;25740:66;25733:5;25730:77;25727:103;;;25810:18;;:::i;:::-;25727:103;25857:1;25850:5;25846:13;25839:20;;25632:233;;;:::o;25871:167::-;25908:3;25931:22;25947:5;25931:22;:::i;:::-;25922:31;;25975:4;25968:5;25965:15;25962:41;;;25983:18;;:::i;:::-;25962:41;26030:1;26023:5;26019:13;26012:20;;25871:167;;;:::o;26044:180::-;26092:77;26089:1;26082:88;26189:4;26186:1;26179:15;26213:4;26210:1;26203:15;26230:180;26278:77;26275:1;26268:88;26375:4;26372:1;26365:15;26399:4;26396:1;26389:15;26416:180;26464:77;26461:1;26454:88;26561:4;26558:1;26551:15;26585:4;26582:1;26575:15;26602:180;26650:77;26647:1;26640:88;26747:4;26744:1;26737:15;26771:4;26768:1;26761:15;26788:117;26897:1;26894;26887:12;26911:117;27020:1;27017;27010:12;27034:117;27143:1;27140;27133:12;27157:117;27266:1;27263;27256:12;27280:117;27389:1;27386;27379:12;27403:117;27512:1;27509;27502:12;27526:102;27567:6;27618:2;27614:7;27609:2;27602:5;27598:14;27594:28;27584:38;;27526:102;;;:::o;27634:221::-;27774:34;27770:1;27762:6;27758:14;27751:58;27843:4;27838:2;27830:6;27826:15;27819:29;27634:221;:::o;27861:225::-;28001:34;27997:1;27989:6;27985:14;27978:58;28070:8;28065:2;28057:6;28053:15;28046:33;27861:225;:::o;28092:171::-;28232:23;28228:1;28220:6;28216:14;28209:47;28092:171;:::o;28269:181::-;28409:33;28405:1;28397:6;28393:14;28386:57;28269:181;:::o;28456:182::-;28596:34;28592:1;28584:6;28580:14;28573:58;28456:182;:::o;28644:223::-;28784:34;28780:1;28772:6;28768:14;28761:58;28853:6;28848:2;28840:6;28836:15;28829:31;28644:223;:::o;28873:182::-;29013:34;29009:1;29001:6;28997:14;28990:58;28873:182;:::o;29061:174::-;29201:26;29197:1;29189:6;29185:14;29178:50;29061:174;:::o;29241:114::-;;:::o;29361:166::-;29501:18;29497:1;29489:6;29485:14;29478:42;29361:166;:::o;29533:181::-;29673:33;29669:1;29661:6;29657:14;29650:57;29533:181;:::o;29720:122::-;29793:24;29811:5;29793:24;:::i;:::-;29786:5;29783:35;29773:63;;29832:1;29829;29822:12;29773:63;29720:122;:::o;29848:138::-;29929:32;29955:5;29929:32;:::i;:::-;29922:5;29919:43;29909:71;;29976:1;29973;29966:12;29909:71;29848:138;:::o;29992:116::-;30062:21;30077:5;30062:21;:::i;:::-;30055:5;30052:32;30042:60;;30098:1;30095;30088:12;30042:60;29992:116;:::o;30114:120::-;30186:23;30203:5;30186:23;:::i;:::-;30179:5;30176:34;30166:62;;30224:1;30221;30214:12;30166:62;30114:120;:::o;30240:122::-;30313:24;30331:5;30313:24;:::i;:::-;30306:5;30303:35;30293:63;;30352:1;30349;30342:12;30293:63;30240:122;:::o

Swarm Source

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