ETH Price: $3,475.11 (-1.22%)
Gas: 4 Gwei

Sudoge (SUDOGE)
 

Overview

TokenID

85

Total Transfers

-

Market

Onchain Market Cap

$0.00

Circulating Supply Market Cap

-
Loading...
Loading
Loading...
Loading
Loading...
Loading

Click here to update the token information / general information
# Exchange Pair Price  24H Volume % Volume
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.

Contract Source Code Verified (Exact Match)

Contract Name:
SuDoge

Compiler Version
v0.8.9+commit.e5eed63a

Optimization Enabled:
No with 200 runs

Other Settings:
default evmVersion, None license

Contract Source Code (Solidity Multiple files format)

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

/// 1000 SuDoges
/// Sudoswap We stealth, we yaff yaff
/// Twitter: https://twitter.com/sudogenfts
/// Telegram: https://t.me/sudogenfts
/// We're only live on Sudoswap, we won't work on any other platforms! Sudoge NFTs are only moveable on Sudoswap.

pragma solidity ^0.8.9;

import "./ERC721A.sol";
import "./Ownable.sol";
import "./LSSVMPairCloner.sol";
import "./IERC20.sol";

contract SuDoge is ERC721A, Ownable {

    bool _mintingEnabled = true;
    bool _sudoswapOnly;
    string baseURI;

    address constant sudoswapRouter = 0x2B2e8cDA09bBA9660dCA5cB6233787738Ad68329;
    address constant sudoswapFactory = 0xb16c1342E617A5B6E4b631EB114483FDB289c0A4;
    address constant missingEnumerableETHTemplate = 0xCd80C916B1194beB48aBF007D0b79a7238436D56;
    address constant missingEnumerableERC20Template = 0x92de3a1511EF22AbCf3526c302159882a4755B22;

    function mintingEnable() external view returns (bool) {
        return _mintingEnabled;
    }

    function sudoswapOnly() external view returns (bool) {
        return _sudoswapOnly;
    }

    constructor() ERC721A("Sudoge", "SUDOGE") {
        baseURI = "ipfs://QmZLd5vAovekfnsTQTCngcckUWLsLu7Qmfeq7TDw6aXeA5/";
    }

    function ownerMintTo(address receipient, uint256 quantity) external onlyOwner() {
        require(_mintingEnabled, 'Minting has been disabled');
        _safeMint(receipient, quantity);
    }

    function disableMinting() external onlyOwner() {
        _mintingEnabled = false;
    }

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

    function setBaseURI(string memory newURI) external onlyOwner() {
        baseURI = newURI;
    }

    function setSudoswapOnly(bool value) external onlyOwner() {
        _sudoswapOnly = value;
    }

    function withdraw() external onlyOwner() {
        payable(owner()).transfer(address(this).balance);
    }

    function recoverERC20(address tokenAddress, uint256 tokenAmount) external onlyOwner() {
        IERC20(tokenAddress).transfer(owner(), tokenAmount);
    }

    function _beforeTokenTransfers(
        address from,
        address to,
        uint256 startTokenId,
        uint256 quantity
    ) internal view override {
        if(!_sudoswapOnly)
            return;
        if(_msgSenderERC721A() == sudoswapRouter || _msgSenderERC721A() == sudoswapFactory)
            return;
        if(!isContract(_msgSenderERC721A()))
            return;
        if(isPair(_msgSenderERC721A(), PairVariant.MISSING_ENUMERABLE_ETH) || isPair(_msgSenderERC721A(), PairVariant.MISSING_ENUMERABLE_ERC20))
            return;
        
        require(false, 'Can only be swapped via SudoSwap');
    }

    enum PairVariant {
        ENUMERABLE_ETH,
        MISSING_ENUMERABLE_ETH,
        ENUMERABLE_ERC20,
        MISSING_ENUMERABLE_ERC20
    }

    function isPair(address potentialPair, PairVariant variant) private view returns (bool)
    {
        if (variant == PairVariant.MISSING_ENUMERABLE_ERC20) {
            return
                LSSVMPairCloner.isERC20PairClone(
                    sudoswapFactory,
                    missingEnumerableERC20Template,
                    potentialPair
                );
        } else if (variant == PairVariant.MISSING_ENUMERABLE_ETH) {
            return
                LSSVMPairCloner.isETHPairClone(
                    sudoswapFactory,
                    missingEnumerableETHTemplate,
                    potentialPair
                );
        } else {
            // invalid input
            return false;
        }
    }

    function isContract(address addr) internal view returns (bool) {
        uint size;
        assembly { size := extcodesize(addr) }
        return size > 0;
    }
}


File 1 of 6: ERC721A.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.4;

import "./IERC721A.sol";

/**
 * @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 {
    // Reference type for token approval.
    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 1;
    }

    /**
     * @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();
        baseURI = string(abi.encodePacked(baseURI, _toString(tokenId)));
        return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, ".json")) : '';
    }

    /**
     * @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 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 {
        if (operator == _msgSenderERC721A()) revert ApproveToCaller();

        _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 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 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 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`.
                )

                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 0x80 bytes to keep the free memory pointer 32-byte word aligned.
            // We will need 1 32-byte word to store the length,
            // and 3 32-byte words to store a maximum of 78 digits. Total: 0x20 + 3 * 0x20 = 0x80.
            str := add(mload(0x40), 0x80)
            // Update the free memory pointer to allocate.
            mstore(0x40, str)

            // 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 2 of 6: IERC20.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)

pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC20 standard as defined in the EIP.
 */
interface IERC20 {
    /**
     * @dev Emitted when `value` tokens are moved from one account (`from`) to
     * another (`to`).
     *
     * Note that `value` may be zero.
     */
    event Transfer(address indexed from, address indexed to, uint256 value);

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

    /**
     * @dev Returns the amount of tokens in existence.
     */
    function totalSupply() external view returns (uint256);

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

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

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

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

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

File 3 of 6: IERC721A.sol
// SPDX-License-Identifier: MIT

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

    /**
     * The caller cannot approve to their own address.
     */
    error ApproveToCaller();

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

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

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

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

    /**
     * @dev 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 4 of 6: LSSVMPairCloner.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

library LSSVMPairCloner {

    /**
     * @notice Checks if a contract is a clone of a LSSVMPairETH.
     * @dev Only checks the runtime bytecode, does not check the extra data.
     * @param factory the factory that deployed the clone
     * @param implementation the LSSVMPairETH implementation contract
     * @param query the contract to check
     * @return result True if the contract is a clone, false otherwise
     */
    function isETHPairClone(
        address factory,
        address implementation,
        address query
    ) internal view returns (bool result) {
        // solhint-disable-next-line no-inline-assembly
        assembly {
            let ptr := mload(0x40)
            mstore(
                ptr,
                hex"3d_3d_3d_3d_36_3d_3d_37_60_3d_60_35_36_39_36_60_3d_01_3d_73_00_00_00_00_00_00_00_00_00_00_00_00"
            )
            mstore(add(ptr, 0x14), shl(0x60, implementation))
            mstore(
                add(ptr, 0x28),
                hex"5a_f4_3d_3d_93_80_3e_60_33_57_fd_5b_f3_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00"
            )
            mstore(add(ptr, 0x35), shl(0x60, factory))

            // compare expected bytecode with that of the queried contract
            let other := add(ptr, 0x49)
            extcodecopy(query, other, 0, 0x49)
            result := and(
                eq(mload(ptr), mload(other)),
                and(
                    eq(mload(add(ptr, 0x20)), mload(add(other, 0x20))),
                    eq(mload(add(ptr, 0x29)), mload(add(other, 0x29)))
                )
            )
        }
    }

    /**
     * @notice Checks if a contract is a clone of a LSSVMPairERC20.
     * @dev Only checks the runtime bytecode, does not check the extra data.
     * @param implementation the LSSVMPairERC20 implementation contract
     * @param query the contract to check
     * @return result True if the contract is a clone, false otherwise
     */
    function isERC20PairClone(
        address factory,
        address implementation,
        address query
    ) internal view returns (bool result) {
        // solhint-disable-next-line no-inline-assembly
        assembly {
            let ptr := mload(0x40)
            mstore(
                ptr,
                hex"3d_3d_3d_3d_36_3d_3d_37_60_51_60_35_36_39_36_60_51_01_3d_73_00_00_00_00_00_00_00_00_00_00_00_00"
            )
            mstore(add(ptr, 0x14), shl(0x60, implementation))
            mstore(
                add(ptr, 0x28),
                hex"5a_f4_3d_3d_93_80_3e_60_33_57_fd_5b_f3_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00_00"
            )
            mstore(add(ptr, 0x35), shl(0x60, factory))

            // compare expected bytecode with that of the queried contract
            let other := add(ptr, 0x49)
            extcodecopy(query, other, 0, 0x49)
            result := and(
                eq(mload(ptr), mload(other)),
                and(
                    eq(mload(add(ptr, 0x20)), mload(add(other, 0x20))),
                    eq(mload(add(ptr, 0x29)), mload(add(other, 0x29)))
                )
            )
        }
    }
}

File 5 of 6: Ownable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (access/Ownable.sol)

pragma solidity ^0.8.0;

abstract contract Context {
    function _msgSender() internal view virtual returns (address) {
        return msg.sender;
    }

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

/**
 * @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 Returns the address of the current owner.
     */
    function owner() public view virtual returns (address) {
        return _owner;
    }

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

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

Contract Security Audit

Contract ABI

[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"ApproveToCaller","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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"disableMinting","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":[],"name":"mintingEnable","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","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":"address","name":"receipient","type":"address"},{"internalType":"uint256","name":"quantity","type":"uint256"}],"name":"ownerMintTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenAddress","type":"address"},{"internalType":"uint256","name":"tokenAmount","type":"uint256"}],"name":"recoverERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"value","type":"bool"}],"name":"setSudoswapOnly","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"sudoswapOnly","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","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":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]

60806040526001600860146101000a81548160ff0219169083151502179055503480156200002c57600080fd5b506040518060400160405280600681526020017f5375646f676500000000000000000000000000000000000000000000000000008152506040518060400160405280600681526020017f5355444f474500000000000000000000000000000000000000000000000000008152508160029080519060200190620000b192919062000212565b508060039080519060200190620000ca92919062000212565b50620000db6200013b60201b60201c565b600081905550505062000103620000f76200014460201b60201c565b6200014c60201b60201c565b6040518060600160405280603681526020016200303260369139600990805190602001906200013492919062000212565b5062000327565b60006001905090565b600033905090565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b8280546200022090620002f1565b90600052602060002090601f01602090048101928262000244576000855562000290565b82601f106200025f57805160ff191683800117855562000290565b8280016001018555821562000290579182015b828111156200028f57825182559160200191906001019062000272565b5b5090506200029f9190620002a3565b5090565b5b80821115620002be576000816000905550600101620002a4565b5090565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b600060028204905060018216806200030a57607f821691505b60208210811415620003215762000320620002c2565b5b50919050565b612cfb80620003376000396000f3fe608060405234801561001057600080fd5b50600436106101735760003560e01c8063715018a6116100de578063a22cb46511610097578063c87b56dd11610071578063c87b56dd146103e8578063e56b1c6814610418578063e985e9c514610436578063f2fde38b1461046657610173565b8063a22cb46514610392578063b88d4fde146103ae578063ba7e60bd146103ca57610173565b8063715018a61461030a5780637e5cd5c1146103145780638980f11f1461031e5780638da5cb5b1461033a57806395d89b4114610358578063a0eba25f1461037657610173565b80632b96980c116101305780632b96980c1461024c5780633ccfd60b1461026857806342842e0e1461027257806355f804b31461028e5780636352211e146102aa57806370a08231146102da57610173565b806301ffc9a71461017857806306fdde03146101a8578063081812fc146101c6578063095ea7b3146101f657806318160ddd1461021257806323b872dd14610230575b600080fd5b610192600480360381019061018d91906121d5565b610482565b60405161019f919061221d565b60405180910390f35b6101b0610514565b6040516101bd91906122d1565b60405180910390f35b6101e060048036038101906101db9190612329565b6105a6565b6040516101ed9190612397565b60405180910390f35b610210600480360381019061020b91906123de565b610625565b005b61021a610769565b604051610227919061242d565b60405180910390f35b61024a60048036038101906102459190612448565b610780565b005b610266600480360381019061026191906124c7565b610aa5565b005b610270610b3e565b005b61028c60048036038101906102879190612448565b610c0a565b005b6102a860048036038101906102a39190612629565b610c2a565b005b6102c460048036038101906102bf9190612329565b610cc0565b6040516102d19190612397565b60405180910390f35b6102f460048036038101906102ef9190612672565b610cd2565b604051610301919061242d565b60405180910390f35b610312610d8b565b005b61031c610e13565b005b610338600480360381019061033391906123de565b610eac565b005b610342610fc1565b60405161034f9190612397565b60405180910390f35b610360610feb565b60405161036d91906122d1565b60405180910390f35b610390600480360381019061038b91906123de565b61107d565b005b6103ac60048036038101906103a7919061269f565b611156565b005b6103c860048036038101906103c39190612780565b6112ce565b005b6103d2611341565b6040516103df919061221d565b60405180910390f35b61040260048036038101906103fd9190612329565b611358565b60405161040f91906122d1565b60405180910390f35b610420611419565b60405161042d919061221d565b60405180910390f35b610450600480360381019061044b9190612803565b611430565b60405161045d919061221d565b60405180910390f35b610480600480360381019061047b9190612672565b6114c4565b005b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614806104dd57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b8061050d5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b60606002805461052390612872565b80601f016020809104026020016040519081016040528092919081815260200182805461054f90612872565b801561059c5780601f106105715761010080835404028352916020019161059c565b820191906000526020600020905b81548152906001019060200180831161057f57829003601f168201915b5050505050905090565b60006105b1826115bc565b6105e7576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b600061063082610cc0565b90508073ffffffffffffffffffffffffffffffffffffffff1661065161161b565b73ffffffffffffffffffffffffffffffffffffffff16146106b45761067d8161067861161b565b611430565b6106b3576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826006600084815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6000610773611623565b6001546000540303905090565b600061078b8261162c565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146107f2576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000806107fe846116fa565b91509150610814818761080f61161b565b611721565b610860576108298661082461161b565b611430565b61085f576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1614156108c7576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6108d48686866001611765565b80156108df57600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008154600101919050819055506109ad856109898888876118b9565b7c0200000000000000000000000000000000000000000000000000000000176118e1565b600460008681526020019081526020016000208190555060007c020000000000000000000000000000000000000000000000000000000084161415610a35576000600185019050600060046000838152602001908152602001600020541415610a33576000548114610a32578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4610a9d868686600161190c565b505050505050565b610aad611912565b73ffffffffffffffffffffffffffffffffffffffff16610acb610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610b21576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b18906128f0565b60405180910390fd5b80600860156101000a81548160ff02191690831515021790555050565b610b46611912565b73ffffffffffffffffffffffffffffffffffffffff16610b64610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610bba576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610bb1906128f0565b60405180910390fd5b610bc2610fc1565b73ffffffffffffffffffffffffffffffffffffffff166108fc479081150290604051600060405180830381858888f19350505050158015610c07573d6000803e3d6000fd5b50565b610c25838383604051806020016040528060008152506112ce565b505050565b610c32611912565b73ffffffffffffffffffffffffffffffffffffffff16610c50610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610ca6576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c9d906128f0565b60405180910390fd5b8060099080519060200190610cbc9291906120c6565b5050565b6000610ccb8261162c565b9050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415610d3a576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b610d93611912565b73ffffffffffffffffffffffffffffffffffffffff16610db1610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610e07576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610dfe906128f0565b60405180910390fd5b610e11600061191a565b565b610e1b611912565b73ffffffffffffffffffffffffffffffffffffffff16610e39610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610e8f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e86906128f0565b60405180910390fd5b6000600860146101000a81548160ff021916908315150217905550565b610eb4611912565b73ffffffffffffffffffffffffffffffffffffffff16610ed2610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610f28576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f1f906128f0565b60405180910390fd5b8173ffffffffffffffffffffffffffffffffffffffff1663a9059cbb610f4c610fc1565b836040518363ffffffff1660e01b8152600401610f6a929190612910565b602060405180830381600087803b158015610f8457600080fd5b505af1158015610f98573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610fbc919061294e565b505050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b606060038054610ffa90612872565b80601f016020809104026020016040519081016040528092919081815260200182805461102690612872565b80156110735780601f1061104857610100808354040283529160200191611073565b820191906000526020600020905b81548152906001019060200180831161105657829003601f168201915b5050505050905090565b611085611912565b73ffffffffffffffffffffffffffffffffffffffff166110a3610fc1565b73ffffffffffffffffffffffffffffffffffffffff16146110f9576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016110f0906128f0565b60405180910390fd5b600860149054906101000a900460ff16611148576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161113f906129c7565b60405180910390fd5b61115282826119e0565b5050565b61115e61161b565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614156111c3576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b80600760006111d061161b565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff1661127d61161b565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31836040516112c2919061221d565b60405180910390a35050565b6112d9848484610780565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461133b57611304848484846119fe565b61133a576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b6000600860159054906101000a900460ff16905090565b6060611363826115bc565b611399576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60006113a3611b5e565b9050806113af84611bf0565b6040516020016113c0929190612a23565b60405160208183030381529060405290506000815114156113f05760405180602001604052806000815250611411565b806040516020016114019190612a93565b6040516020818303038152906040525b915050919050565b6000600860149054906101000a900460ff16905090565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b6114cc611912565b73ffffffffffffffffffffffffffffffffffffffff166114ea610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614611540576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611537906128f0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614156115b0576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115a790612b27565b60405180910390fd5b6115b98161191a565b50565b6000816115c7611623565b111580156115d6575060005482105b8015611614575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b600033905090565b60006001905090565b6000808290508061163b611623565b116116c3576000548110156116c25760006004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821614156116c0575b60008114156116b657600460008360019003935083815260200190815260200160002054905061168b565b80925050506116f5565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b600860159054906101000a900460ff1661177e576118b3565b732b2e8cda09bba9660dca5cb6233787738ad6832973ffffffffffffffffffffffffffffffffffffffff166117b161161b565b73ffffffffffffffffffffffffffffffffffffffff161480611819575073b16c1342e617a5b6e4b631eb114483fdb289c0a473ffffffffffffffffffffffffffffffffffffffff1661180161161b565b73ffffffffffffffffffffffffffffffffffffffff16145b15611823576118b3565b61183361182e61161b565b611c40565b61183c576118b3565b61184e61184761161b565b6001611c53565b80611867575061186661185f61161b565b6003611c53565b5b15611871576118b3565b60006118b2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016118a990612b93565b60405180910390fd5b5b50505050565b60008060e883901c905060e86118d0868684611d2d565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b600033905090565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b6119fa828260405180602001604052806000815250611d36565b5050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02611a2461161b565b8786866040518563ffffffff1660e01b8152600401611a469493929190612c08565b602060405180830381600087803b158015611a6057600080fd5b505af1925050508015611a9157506040513d601f19601f82011682018060405250810190611a8e9190612c69565b60015b611b0b573d8060008114611ac1576040519150601f19603f3d011682016040523d82523d6000602084013e611ac6565b606091505b50600081511415611b03576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b606060098054611b6d90612872565b80601f0160208091040260200160405190810160405280929190818152602001828054611b9990612872565b8015611be65780601f10611bbb57610100808354040283529160200191611be6565b820191906000526020600020905b815481529060010190602001808311611bc957829003601f168201915b5050505050905090565b606060806040510190508060405280825b600115611c2c57600183039250600a81066030018353600a8104905080611c2757611c2c565b611c01565b508181036020830392508083525050919050565b600080823b905060008111915050919050565b6000600380811115611c6857611c67612c96565b5b826003811115611c7b57611c7a612c96565b5b1415611cbb57611cb473b16c1342e617a5b6e4b631eb114483fdb289c0a47392de3a1511ef22abcf3526c302159882a4755b2285611dd3565b9050611d27565b60016003811115611ccf57611cce612c96565b5b826003811115611ce257611ce1612c96565b5b1415611d2257611d1b73b16c1342e617a5b6e4b631eb114483fdb289c0a473cd80c916b1194beb48abf007d0b79a7238436d5685611e66565b9050611d27565b600090505b92915050565b60009392505050565b611d408383611ef9565b60008373ffffffffffffffffffffffffffffffffffffffff163b14611dce57600080549050600083820390505b611d8060008683806001019450866119fe565b611db6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b818110611d6d578160005414611dcb57600080fd5b50505b505050565b60006040517f3d3d3d3d363d3d37605160353639366051013d7300000000000000000000000081528360601b60148201527f5af43d3d93803e603357fd5bf30000000000000000000000000000000000000060288201528460601b6035820152604981016049600082863c6029810151602983015114602082015160208401511416815183511416925050509392505050565b60006040517f3d3d3d3d363d3d37603d6035363936603d013d7300000000000000000000000081528360601b60148201527f5af43d3d93803e603357fd5bf30000000000000000000000000000000000000060288201528460601b6035820152604981016049600082863c6029810151602983015114602082015160208401511416815183511416925050509392505050565b6000805490506000821415611f3a576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b611f476000848385611765565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550611fbe83611faf60008660006118b9565b611fb8856120b6565b176118e1565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b81811461205f57808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050612024565b50600082141561209b576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060008190555050506120b1600084838561190c565b505050565b60006001821460e11b9050919050565b8280546120d290612872565b90600052602060002090601f0160209004810192826120f4576000855561213b565b82601f1061210d57805160ff191683800117855561213b565b8280016001018555821561213b579182015b8281111561213a57825182559160200191906001019061211f565b5b509050612148919061214c565b5090565b5b8082111561216557600081600090555060010161214d565b5090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b6121b28161217d565b81146121bd57600080fd5b50565b6000813590506121cf816121a9565b92915050565b6000602082840312156121eb576121ea612173565b5b60006121f9848285016121c0565b91505092915050565b60008115159050919050565b61221781612202565b82525050565b6000602082019050612232600083018461220e565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b83811015612272578082015181840152602081019050612257565b83811115612281576000848401525b50505050565b6000601f19601f8301169050919050565b60006122a382612238565b6122ad8185612243565b93506122bd818560208601612254565b6122c681612287565b840191505092915050565b600060208201905081810360008301526122eb8184612298565b905092915050565b6000819050919050565b612306816122f3565b811461231157600080fd5b50565b600081359050612323816122fd565b92915050565b60006020828403121561233f5761233e612173565b5b600061234d84828501612314565b91505092915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b600061238182612356565b9050919050565b61239181612376565b82525050565b60006020820190506123ac6000830184612388565b92915050565b6123bb81612376565b81146123c657600080fd5b50565b6000813590506123d8816123b2565b92915050565b600080604083850312156123f5576123f4612173565b5b6000612403858286016123c9565b925050602061241485828601612314565b9150509250929050565b612427816122f3565b82525050565b6000602082019050612442600083018461241e565b92915050565b60008060006060848603121561246157612460612173565b5b600061246f868287016123c9565b9350506020612480868287016123c9565b925050604061249186828701612314565b9150509250925092565b6124a481612202565b81146124af57600080fd5b50565b6000813590506124c18161249b565b92915050565b6000602082840312156124dd576124dc612173565b5b60006124eb848285016124b2565b91505092915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b61253682612287565b810181811067ffffffffffffffff82111715612555576125546124fe565b5b80604052505050565b6000612568612169565b9050612574828261252d565b919050565b600067ffffffffffffffff821115612594576125936124fe565b5b61259d82612287565b9050602081019050919050565b82818337600083830152505050565b60006125cc6125c784612579565b61255e565b9050828152602081018484840111156125e8576125e76124f9565b5b6125f38482856125aa565b509392505050565b600082601f8301126126105761260f6124f4565b5b81356126208482602086016125b9565b91505092915050565b60006020828403121561263f5761263e612173565b5b600082013567ffffffffffffffff81111561265d5761265c612178565b5b612669848285016125fb565b91505092915050565b60006020828403121561268857612687612173565b5b6000612696848285016123c9565b91505092915050565b600080604083850312156126b6576126b5612173565b5b60006126c4858286016123c9565b92505060206126d5858286016124b2565b9150509250929050565b600067ffffffffffffffff8211156126fa576126f96124fe565b5b61270382612287565b9050602081019050919050565b600061272361271e846126df565b61255e565b90508281526020810184848401111561273f5761273e6124f9565b5b61274a8482856125aa565b509392505050565b600082601f830112612767576127666124f4565b5b8135612777848260208601612710565b91505092915050565b6000806000806080858703121561279a57612799612173565b5b60006127a8878288016123c9565b94505060206127b9878288016123c9565b93505060406127ca87828801612314565b925050606085013567ffffffffffffffff8111156127eb576127ea612178565b5b6127f787828801612752565b91505092959194509250565b6000806040838503121561281a57612819612173565b5b6000612828858286016123c9565b9250506020612839858286016123c9565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b6000600282049050600182168061288a57607f821691505b6020821081141561289e5761289d612843565b5b50919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006128da602083612243565b91506128e5826128a4565b602082019050919050565b60006020820190508181036000830152612909816128cd565b9050919050565b60006040820190506129256000830185612388565b612932602083018461241e565b9392505050565b6000815190506129488161249b565b92915050565b60006020828403121561296457612963612173565b5b600061297284828501612939565b91505092915050565b7f4d696e74696e6720686173206265656e2064697361626c656400000000000000600082015250565b60006129b1601983612243565b91506129bc8261297b565b602082019050919050565b600060208201905081810360008301526129e0816129a4565b9050919050565b600081905092915050565b60006129fd82612238565b612a0781856129e7565b9350612a17818560208601612254565b80840191505092915050565b6000612a2f82856129f2565b9150612a3b82846129f2565b91508190509392505050565b7f2e6a736f6e000000000000000000000000000000000000000000000000000000600082015250565b6000612a7d6005836129e7565b9150612a8882612a47565b600582019050919050565b6000612a9f82846129f2565b9150612aaa82612a70565b915081905092915050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b6000612b11602683612243565b9150612b1c82612ab5565b604082019050919050565b60006020820190508181036000830152612b4081612b04565b9050919050565b7f43616e206f6e6c79206265207377617070656420766961205375646f53776170600082015250565b6000612b7d602083612243565b9150612b8882612b47565b602082019050919050565b60006020820190508181036000830152612bac81612b70565b9050919050565b600081519050919050565b600082825260208201905092915050565b6000612bda82612bb3565b612be48185612bbe565b9350612bf4818560208601612254565b612bfd81612287565b840191505092915050565b6000608082019050612c1d6000830187612388565b612c2a6020830186612388565b612c37604083018561241e565b8181036060830152612c498184612bcf565b905095945050505050565b600081519050612c63816121a9565b92915050565b600060208284031215612c7f57612c7e612173565b5b6000612c8d84828501612c54565b91505092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fdfea264697066735822122038dedcb5a444c10fcaae8469acef54b6f4682e26aeaf2bc375591b7d6772129f64736f6c63430008090033697066733a2f2f516d5a4c643576416f76656b666e73545154436e6763636b55574c734c7537516d666571375444773661586541352f

Deployed Bytecode

0x608060405234801561001057600080fd5b50600436106101735760003560e01c8063715018a6116100de578063a22cb46511610097578063c87b56dd11610071578063c87b56dd146103e8578063e56b1c6814610418578063e985e9c514610436578063f2fde38b1461046657610173565b8063a22cb46514610392578063b88d4fde146103ae578063ba7e60bd146103ca57610173565b8063715018a61461030a5780637e5cd5c1146103145780638980f11f1461031e5780638da5cb5b1461033a57806395d89b4114610358578063a0eba25f1461037657610173565b80632b96980c116101305780632b96980c1461024c5780633ccfd60b1461026857806342842e0e1461027257806355f804b31461028e5780636352211e146102aa57806370a08231146102da57610173565b806301ffc9a71461017857806306fdde03146101a8578063081812fc146101c6578063095ea7b3146101f657806318160ddd1461021257806323b872dd14610230575b600080fd5b610192600480360381019061018d91906121d5565b610482565b60405161019f919061221d565b60405180910390f35b6101b0610514565b6040516101bd91906122d1565b60405180910390f35b6101e060048036038101906101db9190612329565b6105a6565b6040516101ed9190612397565b60405180910390f35b610210600480360381019061020b91906123de565b610625565b005b61021a610769565b604051610227919061242d565b60405180910390f35b61024a60048036038101906102459190612448565b610780565b005b610266600480360381019061026191906124c7565b610aa5565b005b610270610b3e565b005b61028c60048036038101906102879190612448565b610c0a565b005b6102a860048036038101906102a39190612629565b610c2a565b005b6102c460048036038101906102bf9190612329565b610cc0565b6040516102d19190612397565b60405180910390f35b6102f460048036038101906102ef9190612672565b610cd2565b604051610301919061242d565b60405180910390f35b610312610d8b565b005b61031c610e13565b005b610338600480360381019061033391906123de565b610eac565b005b610342610fc1565b60405161034f9190612397565b60405180910390f35b610360610feb565b60405161036d91906122d1565b60405180910390f35b610390600480360381019061038b91906123de565b61107d565b005b6103ac60048036038101906103a7919061269f565b611156565b005b6103c860048036038101906103c39190612780565b6112ce565b005b6103d2611341565b6040516103df919061221d565b60405180910390f35b61040260048036038101906103fd9190612329565b611358565b60405161040f91906122d1565b60405180910390f35b610420611419565b60405161042d919061221d565b60405180910390f35b610450600480360381019061044b9190612803565b611430565b60405161045d919061221d565b60405180910390f35b610480600480360381019061047b9190612672565b6114c4565b005b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614806104dd57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b8061050d5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b60606002805461052390612872565b80601f016020809104026020016040519081016040528092919081815260200182805461054f90612872565b801561059c5780601f106105715761010080835404028352916020019161059c565b820191906000526020600020905b81548152906001019060200180831161057f57829003601f168201915b5050505050905090565b60006105b1826115bc565b6105e7576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b600061063082610cc0565b90508073ffffffffffffffffffffffffffffffffffffffff1661065161161b565b73ffffffffffffffffffffffffffffffffffffffff16146106b45761067d8161067861161b565b611430565b6106b3576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826006600084815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6000610773611623565b6001546000540303905090565b600061078b8261162c565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146107f2576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000806107fe846116fa565b91509150610814818761080f61161b565b611721565b610860576108298661082461161b565b611430565b61085f576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1614156108c7576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6108d48686866001611765565b80156108df57600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008154600101919050819055506109ad856109898888876118b9565b7c0200000000000000000000000000000000000000000000000000000000176118e1565b600460008681526020019081526020016000208190555060007c020000000000000000000000000000000000000000000000000000000084161415610a35576000600185019050600060046000838152602001908152602001600020541415610a33576000548114610a32578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4610a9d868686600161190c565b505050505050565b610aad611912565b73ffffffffffffffffffffffffffffffffffffffff16610acb610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610b21576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610b18906128f0565b60405180910390fd5b80600860156101000a81548160ff02191690831515021790555050565b610b46611912565b73ffffffffffffffffffffffffffffffffffffffff16610b64610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610bba576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610bb1906128f0565b60405180910390fd5b610bc2610fc1565b73ffffffffffffffffffffffffffffffffffffffff166108fc479081150290604051600060405180830381858888f19350505050158015610c07573d6000803e3d6000fd5b50565b610c25838383604051806020016040528060008152506112ce565b505050565b610c32611912565b73ffffffffffffffffffffffffffffffffffffffff16610c50610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610ca6576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c9d906128f0565b60405180910390fd5b8060099080519060200190610cbc9291906120c6565b5050565b6000610ccb8261162c565b9050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415610d3a576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b610d93611912565b73ffffffffffffffffffffffffffffffffffffffff16610db1610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610e07576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610dfe906128f0565b60405180910390fd5b610e11600061191a565b565b610e1b611912565b73ffffffffffffffffffffffffffffffffffffffff16610e39610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610e8f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e86906128f0565b60405180910390fd5b6000600860146101000a81548160ff021916908315150217905550565b610eb4611912565b73ffffffffffffffffffffffffffffffffffffffff16610ed2610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614610f28576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f1f906128f0565b60405180910390fd5b8173ffffffffffffffffffffffffffffffffffffffff1663a9059cbb610f4c610fc1565b836040518363ffffffff1660e01b8152600401610f6a929190612910565b602060405180830381600087803b158015610f8457600080fd5b505af1158015610f98573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610fbc919061294e565b505050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b606060038054610ffa90612872565b80601f016020809104026020016040519081016040528092919081815260200182805461102690612872565b80156110735780601f1061104857610100808354040283529160200191611073565b820191906000526020600020905b81548152906001019060200180831161105657829003601f168201915b5050505050905090565b611085611912565b73ffffffffffffffffffffffffffffffffffffffff166110a3610fc1565b73ffffffffffffffffffffffffffffffffffffffff16146110f9576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016110f0906128f0565b60405180910390fd5b600860149054906101000a900460ff16611148576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161113f906129c7565b60405180910390fd5b61115282826119e0565b5050565b61115e61161b565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614156111c3576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b80600760006111d061161b565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff1661127d61161b565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31836040516112c2919061221d565b60405180910390a35050565b6112d9848484610780565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461133b57611304848484846119fe565b61133a576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b6000600860159054906101000a900460ff16905090565b6060611363826115bc565b611399576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60006113a3611b5e565b9050806113af84611bf0565b6040516020016113c0929190612a23565b60405160208183030381529060405290506000815114156113f05760405180602001604052806000815250611411565b806040516020016114019190612a93565b6040516020818303038152906040525b915050919050565b6000600860149054906101000a900460ff16905090565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b6114cc611912565b73ffffffffffffffffffffffffffffffffffffffff166114ea610fc1565b73ffffffffffffffffffffffffffffffffffffffff1614611540576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611537906128f0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614156115b0576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115a790612b27565b60405180910390fd5b6115b98161191a565b50565b6000816115c7611623565b111580156115d6575060005482105b8015611614575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b600033905090565b60006001905090565b6000808290508061163b611623565b116116c3576000548110156116c25760006004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821614156116c0575b60008114156116b657600460008360019003935083815260200190815260200160002054905061168b565b80925050506116f5565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b600860159054906101000a900460ff1661177e576118b3565b732b2e8cda09bba9660dca5cb6233787738ad6832973ffffffffffffffffffffffffffffffffffffffff166117b161161b565b73ffffffffffffffffffffffffffffffffffffffff161480611819575073b16c1342e617a5b6e4b631eb114483fdb289c0a473ffffffffffffffffffffffffffffffffffffffff1661180161161b565b73ffffffffffffffffffffffffffffffffffffffff16145b15611823576118b3565b61183361182e61161b565b611c40565b61183c576118b3565b61184e61184761161b565b6001611c53565b80611867575061186661185f61161b565b6003611c53565b5b15611871576118b3565b60006118b2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016118a990612b93565b60405180910390fd5b5b50505050565b60008060e883901c905060e86118d0868684611d2d565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b600033905090565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b6119fa828260405180602001604052806000815250611d36565b5050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02611a2461161b565b8786866040518563ffffffff1660e01b8152600401611a469493929190612c08565b602060405180830381600087803b158015611a6057600080fd5b505af1925050508015611a9157506040513d601f19601f82011682018060405250810190611a8e9190612c69565b60015b611b0b573d8060008114611ac1576040519150601f19603f3d011682016040523d82523d6000602084013e611ac6565b606091505b50600081511415611b03576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b606060098054611b6d90612872565b80601f0160208091040260200160405190810160405280929190818152602001828054611b9990612872565b8015611be65780601f10611bbb57610100808354040283529160200191611be6565b820191906000526020600020905b815481529060010190602001808311611bc957829003601f168201915b5050505050905090565b606060806040510190508060405280825b600115611c2c57600183039250600a81066030018353600a8104905080611c2757611c2c565b611c01565b508181036020830392508083525050919050565b600080823b905060008111915050919050565b6000600380811115611c6857611c67612c96565b5b826003811115611c7b57611c7a612c96565b5b1415611cbb57611cb473b16c1342e617a5b6e4b631eb114483fdb289c0a47392de3a1511ef22abcf3526c302159882a4755b2285611dd3565b9050611d27565b60016003811115611ccf57611cce612c96565b5b826003811115611ce257611ce1612c96565b5b1415611d2257611d1b73b16c1342e617a5b6e4b631eb114483fdb289c0a473cd80c916b1194beb48abf007d0b79a7238436d5685611e66565b9050611d27565b600090505b92915050565b60009392505050565b611d408383611ef9565b60008373ffffffffffffffffffffffffffffffffffffffff163b14611dce57600080549050600083820390505b611d8060008683806001019450866119fe565b611db6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b818110611d6d578160005414611dcb57600080fd5b50505b505050565b60006040517f3d3d3d3d363d3d37605160353639366051013d7300000000000000000000000081528360601b60148201527f5af43d3d93803e603357fd5bf30000000000000000000000000000000000000060288201528460601b6035820152604981016049600082863c6029810151602983015114602082015160208401511416815183511416925050509392505050565b60006040517f3d3d3d3d363d3d37603d6035363936603d013d7300000000000000000000000081528360601b60148201527f5af43d3d93803e603357fd5bf30000000000000000000000000000000000000060288201528460601b6035820152604981016049600082863c6029810151602983015114602082015160208401511416815183511416925050509392505050565b6000805490506000821415611f3a576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b611f476000848385611765565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550611fbe83611faf60008660006118b9565b611fb8856120b6565b176118e1565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b81811461205f57808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050612024565b50600082141561209b576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060008190555050506120b1600084838561190c565b505050565b60006001821460e11b9050919050565b8280546120d290612872565b90600052602060002090601f0160209004810192826120f4576000855561213b565b82601f1061210d57805160ff191683800117855561213b565b8280016001018555821561213b579182015b8281111561213a57825182559160200191906001019061211f565b5b509050612148919061214c565b5090565b5b8082111561216557600081600090555060010161214d565b5090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b6121b28161217d565b81146121bd57600080fd5b50565b6000813590506121cf816121a9565b92915050565b6000602082840312156121eb576121ea612173565b5b60006121f9848285016121c0565b91505092915050565b60008115159050919050565b61221781612202565b82525050565b6000602082019050612232600083018461220e565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b83811015612272578082015181840152602081019050612257565b83811115612281576000848401525b50505050565b6000601f19601f8301169050919050565b60006122a382612238565b6122ad8185612243565b93506122bd818560208601612254565b6122c681612287565b840191505092915050565b600060208201905081810360008301526122eb8184612298565b905092915050565b6000819050919050565b612306816122f3565b811461231157600080fd5b50565b600081359050612323816122fd565b92915050565b60006020828403121561233f5761233e612173565b5b600061234d84828501612314565b91505092915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b600061238182612356565b9050919050565b61239181612376565b82525050565b60006020820190506123ac6000830184612388565b92915050565b6123bb81612376565b81146123c657600080fd5b50565b6000813590506123d8816123b2565b92915050565b600080604083850312156123f5576123f4612173565b5b6000612403858286016123c9565b925050602061241485828601612314565b9150509250929050565b612427816122f3565b82525050565b6000602082019050612442600083018461241e565b92915050565b60008060006060848603121561246157612460612173565b5b600061246f868287016123c9565b9350506020612480868287016123c9565b925050604061249186828701612314565b9150509250925092565b6124a481612202565b81146124af57600080fd5b50565b6000813590506124c18161249b565b92915050565b6000602082840312156124dd576124dc612173565b5b60006124eb848285016124b2565b91505092915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b61253682612287565b810181811067ffffffffffffffff82111715612555576125546124fe565b5b80604052505050565b6000612568612169565b9050612574828261252d565b919050565b600067ffffffffffffffff821115612594576125936124fe565b5b61259d82612287565b9050602081019050919050565b82818337600083830152505050565b60006125cc6125c784612579565b61255e565b9050828152602081018484840111156125e8576125e76124f9565b5b6125f38482856125aa565b509392505050565b600082601f8301126126105761260f6124f4565b5b81356126208482602086016125b9565b91505092915050565b60006020828403121561263f5761263e612173565b5b600082013567ffffffffffffffff81111561265d5761265c612178565b5b612669848285016125fb565b91505092915050565b60006020828403121561268857612687612173565b5b6000612696848285016123c9565b91505092915050565b600080604083850312156126b6576126b5612173565b5b60006126c4858286016123c9565b92505060206126d5858286016124b2565b9150509250929050565b600067ffffffffffffffff8211156126fa576126f96124fe565b5b61270382612287565b9050602081019050919050565b600061272361271e846126df565b61255e565b90508281526020810184848401111561273f5761273e6124f9565b5b61274a8482856125aa565b509392505050565b600082601f830112612767576127666124f4565b5b8135612777848260208601612710565b91505092915050565b6000806000806080858703121561279a57612799612173565b5b60006127a8878288016123c9565b94505060206127b9878288016123c9565b93505060406127ca87828801612314565b925050606085013567ffffffffffffffff8111156127eb576127ea612178565b5b6127f787828801612752565b91505092959194509250565b6000806040838503121561281a57612819612173565b5b6000612828858286016123c9565b9250506020612839858286016123c9565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b6000600282049050600182168061288a57607f821691505b6020821081141561289e5761289d612843565b5b50919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006128da602083612243565b91506128e5826128a4565b602082019050919050565b60006020820190508181036000830152612909816128cd565b9050919050565b60006040820190506129256000830185612388565b612932602083018461241e565b9392505050565b6000815190506129488161249b565b92915050565b60006020828403121561296457612963612173565b5b600061297284828501612939565b91505092915050565b7f4d696e74696e6720686173206265656e2064697361626c656400000000000000600082015250565b60006129b1601983612243565b91506129bc8261297b565b602082019050919050565b600060208201905081810360008301526129e0816129a4565b9050919050565b600081905092915050565b60006129fd82612238565b612a0781856129e7565b9350612a17818560208601612254565b80840191505092915050565b6000612a2f82856129f2565b9150612a3b82846129f2565b91508190509392505050565b7f2e6a736f6e000000000000000000000000000000000000000000000000000000600082015250565b6000612a7d6005836129e7565b9150612a8882612a47565b600582019050919050565b6000612a9f82846129f2565b9150612aaa82612a70565b915081905092915050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b6000612b11602683612243565b9150612b1c82612ab5565b604082019050919050565b60006020820190508181036000830152612b4081612b04565b9050919050565b7f43616e206f6e6c79206265207377617070656420766961205375646f53776170600082015250565b6000612b7d602083612243565b9150612b8882612b47565b602082019050919050565b60006020820190508181036000830152612bac81612b70565b9050919050565b600081519050919050565b600082825260208201905092915050565b6000612bda82612bb3565b612be48185612bbe565b9350612bf4818560208601612254565b612bfd81612287565b840191505092915050565b6000608082019050612c1d6000830187612388565b612c2a6020830186612388565b612c37604083018561241e565b8181036060830152612c498184612bcf565b905095945050505050565b600081519050612c63816121a9565b92915050565b600060208284031215612c7f57612c7e612173565b5b6000612c8d84828501612c54565b91505092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fdfea264697066735822122038dedcb5a444c10fcaae8469acef54b6f4682e26aeaf2bc375591b7d6772129f64736f6c63430008090033

Deployed Bytecode Sourcemap

429:3462:5:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;9312:639:0;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;10214:100;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;16760:218;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;16201:400;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;5965:323;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;20473:2817;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;1780:98:5;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;1886:108;;;:::i;:::-;;23386:185:0;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;1674:98:5;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;11670:152:0;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;7149:233;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;1934:103:4;;;:::i;:::-;;1461:89:5;;;:::i;:::-;;2002:156;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;1283:87:4;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;10390:104:0;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;1259:194:5;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;17318:308:0;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;24169:399;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;1024:92:5;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;10600:381:0;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;921:95:5;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;17783:164:0;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;2192:201:4;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;9312:639:0;9397:4;9736:10;9721:25;;:11;:25;;;;:102;;;;9813:10;9798:25;;:11;:25;;;;9721:102;:179;;;;9890:10;9875:25;;:11;:25;;;;9721:179;9701:199;;9312:639;;;:::o;10214:100::-;10268:13;10301:5;10294:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;10214:100;:::o;16760:218::-;16836:7;16861:16;16869:7;16861;:16::i;:::-;16856:64;;16886:34;;;;;;;;;;;;;;16856:64;16940:15;:24;16956:7;16940:24;;;;;;;;;;;:30;;;;;;;;;;;;16933:37;;16760:218;;;:::o;16201:400::-;16282:13;16298:16;16306:7;16298;:16::i;:::-;16282:32;;16354:5;16331:28;;:19;:17;:19::i;:::-;:28;;;16327:175;;16379:44;16396:5;16403:19;:17;:19::i;:::-;16379:16;:44::i;:::-;16374:128;;16451:35;;;;;;;;;;;;;;16374:128;16327:175;16547:2;16514:15;:24;16530:7;16514:24;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;16585:7;16581:2;16565:28;;16574:5;16565:28;;;;;;;;;;;;16271:330;16201:400;;:::o;5965:323::-;6026:7;6254:15;:13;:15::i;:::-;6239:12;;6223:13;;:28;:46;6216:53;;5965:323;:::o;20473:2817::-;20607:27;20637;20656:7;20637:18;:27::i;:::-;20607:57;;20722:4;20681:45;;20697:19;20681:45;;;20677:86;;20735:28;;;;;;;;;;;;;;20677:86;20777:27;20806:23;20833:35;20860:7;20833:26;:35::i;:::-;20776:92;;;;20968:68;20993:15;21010:4;21016:19;:17;:19::i;:::-;20968:24;:68::i;:::-;20963:180;;21056:43;21073:4;21079:19;:17;:19::i;:::-;21056:16;:43::i;:::-;21051:92;;21108:35;;;;;;;;;;;;;;21051:92;20963:180;21174:1;21160:16;;:2;:16;;;21156:52;;;21185:23;;;;;;;;;;;;;;21156:52;21221:43;21243:4;21249:2;21253:7;21262:1;21221:21;:43::i;:::-;21357:15;21354:160;;;21497:1;21476:19;21469:30;21354:160;21894:18;:24;21913:4;21894:24;;;;;;;;;;;;;;;;21892:26;;;;;;;;;;;;21963:18;:22;21982:2;21963:22;;;;;;;;;;;;;;;;21961:24;;;;;;;;;;;22285:146;22322:2;22371:45;22386:4;22392:2;22396:19;22371:14;:45::i;:::-;2364:8;22343:73;22285:18;:146::i;:::-;22256:17;:26;22274:7;22256:26;;;;;;;;;;;:175;;;;22602:1;2364:8;22551:19;:47;:52;22547:627;;;22624:19;22656:1;22646:7;:11;22624:33;;22813:1;22779:17;:30;22797:11;22779:30;;;;;;;;;;;;:35;22775:384;;;22917:13;;22902:11;:28;22898:242;;23097:19;23064:17;:30;23082:11;23064:30;;;;;;;;;;;:52;;;;22898:242;22775:384;22605:569;22547:627;23221:7;23217:2;23202:27;;23211:4;23202:27;;;;;;;;;;;;23240:42;23261:4;23267:2;23271:7;23280:1;23240:20;:42::i;:::-;20596:2694;;;20473:2817;;;:::o;1780:98:5:-;1514:12:4;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;1865:5:5::1;1849:13;;:21;;;;;;;;;;;;;;;;;;1780:98:::0;:::o;1886:108::-;1514:12:4;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;1946:7:5::1;:5;:7::i;:::-;1938:25;;:48;1964:21;1938:48;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;;;;;;1886:108::o:0;23386:185:0:-;23524:39;23541:4;23547:2;23551:7;23524:39;;;;;;;;;;;;:16;:39::i;:::-;23386:185;;;:::o;1674:98:5:-;1514:12:4;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;1758:6:5::1;1748:7;:16;;;;;;;;;;;;:::i;:::-;;1674:98:::0;:::o;11670:152:0:-;11742:7;11785:27;11804:7;11785:18;:27::i;:::-;11762:52;;11670:152;;;:::o;7149:233::-;7221:7;7262:1;7245:19;;:5;:19;;;7241:60;;;7273:28;;;;;;;;;;;;;;7241:60;1308:13;7319:18;:25;7338:5;7319:25;;;;;;;;;;;;;;;;:55;7312:62;;7149:233;;;:::o;1934:103:4:-;1514:12;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;1999:30:::1;2026:1;1999:18;:30::i;:::-;1934:103::o:0;1461:89:5:-;1514:12:4;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;1537:5:5::1;1519:15;;:23;;;;;;;;;;;;;;;;;;1461:89::o:0;2002:156::-;1514:12:4;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;2106:12:5::1;2099:29;;;2129:7;:5;:7::i;:::-;2138:11;2099:51;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;2002:156:::0;;:::o;1283:87:4:-;1329:7;1356:6;;;;;;;;;;;1349:13;;1283:87;:::o;10390:104:0:-;10446:13;10479:7;10472:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;10390:104;:::o;1259:194:5:-;1514:12:4;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;1358:15:5::1;;;;;;;;;;;1350:53;;;;;;;;;;;;:::i;:::-;;;;;;;;;1414:31;1424:10;1436:8;1414:9;:31::i;:::-;1259:194:::0;;:::o;17318:308:0:-;17429:19;:17;:19::i;:::-;17417:31;;:8;:31;;;17413:61;;;17457:17;;;;;;;;;;;;;;17413:61;17539:8;17487:18;:39;17506:19;:17;:19::i;:::-;17487:39;;;;;;;;;;;;;;;:49;17527:8;17487:49;;;;;;;;;;;;;;;;:60;;;;;;;;;;;;;;;;;;17599:8;17563:55;;17578:19;:17;:19::i;:::-;17563:55;;;17609:8;17563:55;;;;;;:::i;:::-;;;;;;;;17318:308;;:::o;24169:399::-;24336:31;24349:4;24355:2;24359:7;24336:12;:31::i;:::-;24400:1;24382:2;:14;;;:19;24378:183;;24421:56;24452:4;24458:2;24462:7;24471:5;24421:30;:56::i;:::-;24416:145;;24505:40;;;;;;;;;;;;;;24416:145;24378:183;24169:399;;;;:::o;1024:92:5:-;1071:4;1095:13;;;;;;;;;;;1088:20;;1024:92;:::o;10600:381:0:-;10673:13;10704:16;10712:7;10704;:16::i;:::-;10699:59;;10729:29;;;;;;;;;;;;;;10699:59;10771:21;10795:10;:8;:10::i;:::-;10771:34;;10850:7;10859:18;10869:7;10859:9;:18::i;:::-;10833:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;10816:63;;10922:1;10903:7;10897:21;:26;;:76;;;;;;;;;;;;;;;;;10950:7;10933:34;;;;;;;;:::i;:::-;;;;;;;;;;;;;10897:76;10890:83;;;10600:381;;;:::o;921:95:5:-;969:4;993:15;;;;;;;;;;;986:22;;921:95;:::o;17783:164:0:-;17880:4;17904:18;:25;17923:5;17904:25;;;;;;;;;;;;;;;:35;17930:8;17904:35;;;;;;;;;;;;;;;;;;;;;;;;;17897:42;;17783:164;;;;:::o;2192:201:4:-;1514:12;:10;:12::i;:::-;1503:23;;:7;:5;:7::i;:::-;:23;;;1495:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;2301:1:::1;2281:22;;:8;:22;;;;2273:73;;;;;;;;;;;;:::i;:::-;;;;;;;;;2357:28;2376:8;2357:18;:28::i;:::-;2192:201:::0;:::o;18205:282:0:-;18270:4;18326:7;18307:15;:13;:15::i;:::-;:26;;:66;;;;;18360:13;;18350:7;:23;18307:66;:153;;;;;18459:1;2084:8;18411:17;:26;18429:7;18411:26;;;;;;;;;;;;:44;:49;18307:153;18287:173;;18205:282;;;:::o;40243:105::-;40303:7;40330:10;40323:17;;40243:105;:::o;5481:92::-;5537:7;5564:1;5557:8;;5481:92;:::o;12825:1275::-;12892:7;12912:12;12927:7;12912:22;;12995:4;12976:15;:13;:15::i;:::-;:23;12972:1061;;13029:13;;13022:4;:20;13018:1015;;;13067:14;13084:17;:23;13102:4;13084:23;;;;;;;;;;;;13067:40;;13201:1;2084:8;13173:6;:24;:29;13169:845;;;13838:113;13855:1;13845:6;:11;13838:113;;;13898:17;:25;13916:6;;;;;;;13898:25;;;;;;;;;;;;13889:34;;13838:113;;;13984:6;13977:13;;;;;;13169:845;13044:989;13018:1015;12972:1061;14061:31;;;;;;;;;;;;;;12825:1275;;;;:::o;19368:485::-;19470:27;19499:23;19540:38;19581:15;:24;19597:7;19581:24;;;;;;;;;;;19540:65;;19758:18;19735:41;;19815:19;19809:26;19790:45;;19720:126;19368:485;;;:::o;18596:659::-;18745:11;18910:16;18903:5;18899:28;18890:37;;19070:16;19059:9;19055:32;19042:45;;19220:15;19209:9;19206:30;19198:5;19187:9;19184:20;19181:56;19171:66;;18596:659;;;;;:::o;2166:639:5:-;2344:13;;;;;;;;;;;2340:39;;2372:7;;2340:39;590:42;2392:37;;:19;:17;:19::i;:::-;:37;;;:79;;;;674:42;2433:38;;:19;:17;:19::i;:::-;:38;;;2392:79;2389:104;;;2486:7;;2389:104;2507:31;2518:19;:17;:19::i;:::-;2507:10;:31::i;:::-;2503:57;;2553:7;;2503:57;2573:63;2580:19;:17;:19::i;:::-;2601:34;2573:6;:63::i;:::-;:132;;;;2640:65;2647:19;:17;:19::i;:::-;2668:36;2640:6;:65::i;:::-;2573:132;2570:157;;;2720:7;;2570:157;2755:5;2747:50;;;;;;;;;;;;:::i;:::-;;;;;;;;;2166:639;;;;;:::o;39552:311:0:-;39687:7;39707:16;2488:3;39733:19;:41;;39707:68;;2488:3;39801:31;39812:4;39818:2;39822:9;39801:10;:31::i;:::-;39793:40;;:62;;39786:69;;;39552:311;;;;;:::o;14648:450::-;14728:14;14896:16;14889:5;14885:28;14876:37;;15073:5;15059:11;15034:23;15030:41;15027:52;15020:5;15017:63;15007:73;;14648:450;;;;:::o;26054:158::-;;;;;:::o;150:98:4:-;203:7;230:10;223:17;;150:98;:::o;2553:191::-;2627:16;2646:6;;;;;;;;;;;2627:25;;2672:8;2663:6;;:17;;;;;;;;;;;;;;;;;;2727:8;2696:40;;2717:8;2696:40;;;;;;;;;;;;2616:128;2553:191;:::o;34075:112:0:-;34152:27;34162:2;34166:8;34152:27;;;;;;;;;;;;:9;:27::i;:::-;34075:112;;:::o;26652:716::-;26815:4;26861:2;26836:45;;;26882:19;:17;:19::i;:::-;26903:4;26909:7;26918:5;26836:88;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;26832:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;27136:1;27119:6;:13;:18;27115:235;;;27165:40;;;;;;;;;;;;;;27115:235;27308:6;27302:13;27293:6;27289:2;27285:15;27278:38;26832:529;27005:54;;;26995:64;;;:6;:64;;;;26988:71;;;26652:716;;;;;;:::o;1558:108:5:-;1618:13;1651:7;1644:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;1558:108;:::o;40450:1582:0:-;40515:17;40941:4;40934;40928:11;40924:22;40917:29;;41033:3;41027:4;41020:17;41139:3;41378:5;41360:428;41386:1;41360:428;;;41426:1;41421:3;41417:11;41410:18;;41597:2;41591:4;41587:13;41583:2;41579:22;41574:3;41566:36;41691:2;41685:4;41681:13;41673:21;;41758:4;41748:25;;41766:5;;41748:25;41360:428;;;41364:21;41827:3;41822;41818:13;41942:4;41937:3;41933:14;41926:21;;42007:6;42002:3;41995:19;40554:1471;;40450:1582;;;:::o;3723:165:5:-;3780:4;3797:9;3848:4;3836:17;3828:25;;3879:1;3872:4;:8;3865:15;;;3723:165;;;:::o;2965:750::-;3047:4;3084:36;3073:47;;;;;;;;:::i;:::-;;:7;:47;;;;;;;;:::i;:::-;;;3069:639;;;3161:178;674:42;870;3307:13;3161:32;:178::i;:::-;3137:202;;;;3069:639;3372:34;3361:45;;;;;;;;:::i;:::-;;:7;:45;;;;;;;;:::i;:::-;;;3357:351;;;3447:174;674:42;771;3589:13;3447:30;:174::i;:::-;3423:198;;;;3357:351;3691:5;3684:12;;2965:750;;;;;:::o;39253:147:0:-;39390:6;39253:147;;;;;:::o;33302:689::-;33433:19;33439:2;33443:8;33433:5;:19::i;:::-;33512:1;33494:2;:14;;;:19;33490:483;;33534:11;33548:13;;33534:27;;33580:13;33602:8;33596:3;:14;33580:30;;33629:233;33660:62;33699:1;33703:2;33707:7;;;;;;33716:5;33660:30;:62::i;:::-;33655:167;;33758:40;;;;;;;;;;;;;;33655:167;33857:3;33849:5;:11;33629:233;;33944:3;33927:13;;:20;33923:34;;33949:8;;;33923:34;33515:458;;33490:483;33302:689;;;:::o;2071:1209:3:-;2210:11;2332:4;2326:11;2398:100;2376:3;2351:162;2560:14;2554:4;2550:25;2543:4;2538:3;2534:14;2527:49;2648:100;2624:4;2619:3;2615:14;2590:173;2810:7;2804:4;2800:18;2793:4;2788:3;2784:14;2777:42;2933:4;2928:3;2924:14;2981:4;2978:1;2971:5;2964;2952:34;3221:4;3214:5;3210:16;3204:23;3196:4;3191:3;3187:14;3181:21;3178:50;3148:4;3141:5;3137:16;3131:23;3123:4;3118:3;3114:14;3108:21;3105:50;3079:168;3053:5;3047:12;3041:3;3035:10;3032:28;3010:252;3000:262;;2300:973;;2071:1209;;;;;:::o;503:1207::-;640:11;762:4;756:11;828:100;806:3;781:162;990:14;984:4;980:25;973:4;968:3;964:14;957:49;1078:100;1054:4;1049:3;1045:14;1020:173;1240:7;1234:4;1230:18;1223:4;1218:3;1214:14;1207:42;1363:4;1358:3;1354:14;1411:4;1408:1;1401:5;1394;1382:34;1651:4;1644:5;1640:16;1634:23;1626:4;1621:3;1617:14;1611:21;1608:50;1578:4;1571:5;1567:16;1561:23;1553:4;1548:3;1544:14;1538:21;1535:50;1509:168;1483:5;1477:12;1471:3;1465:10;1462:28;1440:252;1430:262;;730:973;;503:1207;;;;;:::o;27830:2720:0:-;27903:20;27926:13;;27903:36;;27966:1;27954:8;:13;27950:44;;;27976:18;;;;;;;;;;;;;;27950:44;28007:61;28037:1;28041:2;28045:12;28059:8;28007:21;:61::i;:::-;28551:1;1446:2;28521:1;:26;;28520:32;28508:8;:45;28482:18;:22;28501:2;28482:22;;;;;;;;;;;;;;;;:71;;;;;;;;;;;28830:139;28867:2;28921:33;28944:1;28948:2;28952:1;28921:14;:33::i;:::-;28888:30;28909:8;28888:20;:30::i;:::-;:66;28830:18;:139::i;:::-;28796:17;:31;28814:12;28796:31;;;;;;;;;;;:173;;;;28986:16;29017:11;29046:8;29031:12;:23;29017:37;;29567:16;29563:2;29559:25;29547:37;;29939:12;29899:8;29858:1;29796:25;29737:1;29676;29649:335;30064:1;30050:12;30046:20;30004:346;30105:3;30096:7;30093:16;30004:346;;30323:7;30313:8;30310:1;30283:25;30280:1;30277;30272:59;30158:1;30149:7;30145:15;30134:26;;30004:346;;;30008:77;30395:1;30383:8;:13;30379:45;;;30405:19;;;;;;;;;;;;;;30379:45;30457:3;30441:13;:19;;;;28256:2216;;30482:60;30511:1;30515:2;30519:12;30533:8;30482:20;:60::i;:::-;27892:2658;27830:2720;;:::o;15200:324::-;15270:14;15503:1;15493:8;15490:15;15464:24;15460:46;15450:56;;15200:324;;;:::o;-1:-1:-1:-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;:::o;7:75:6:-;40:6;73:2;67:9;57:19;;7:75;:::o;88:117::-;197:1;194;187:12;211:117;320:1;317;310:12;334:149;370:7;410:66;403:5;399:78;388:89;;334:149;;;:::o;489:120::-;561:23;578:5;561:23;:::i;:::-;554:5;551:34;541:62;;599:1;596;589:12;541:62;489:120;:::o;615:137::-;660:5;698:6;685:20;676:29;;714:32;740:5;714:32;:::i;:::-;615:137;;;;:::o;758:327::-;816:6;865:2;853:9;844:7;840:23;836:32;833:119;;;871:79;;:::i;:::-;833:119;991:1;1016:52;1060:7;1051:6;1040:9;1036:22;1016:52;:::i;:::-;1006:62;;962:116;758:327;;;;:::o;1091:90::-;1125:7;1168:5;1161:13;1154:21;1143:32;;1091:90;;;:::o;1187:109::-;1268:21;1283:5;1268:21;:::i;:::-;1263:3;1256:34;1187:109;;:::o;1302:210::-;1389:4;1427:2;1416:9;1412:18;1404:26;;1440:65;1502:1;1491:9;1487:17;1478:6;1440:65;:::i;:::-;1302:210;;;;:::o;1518:99::-;1570:6;1604:5;1598:12;1588:22;;1518:99;;;:::o;1623:169::-;1707:11;1741:6;1736:3;1729:19;1781:4;1776:3;1772:14;1757:29;;1623:169;;;;:::o;1798:307::-;1866:1;1876:113;1890:6;1887:1;1884:13;1876:113;;;1975:1;1970:3;1966:11;1960:18;1956:1;1951:3;1947:11;1940:39;1912:2;1909:1;1905:10;1900:15;;1876:113;;;2007:6;2004:1;2001:13;1998:101;;;2087:1;2078:6;2073:3;2069:16;2062:27;1998:101;1847:258;1798:307;;;:::o;2111:102::-;2152:6;2203:2;2199:7;2194:2;2187:5;2183:14;2179:28;2169:38;;2111:102;;;:::o;2219:364::-;2307:3;2335:39;2368:5;2335:39;:::i;:::-;2390:71;2454:6;2449:3;2390:71;:::i;:::-;2383:78;;2470:52;2515:6;2510:3;2503:4;2496:5;2492:16;2470:52;:::i;:::-;2547:29;2569:6;2547:29;:::i;:::-;2542:3;2538:39;2531:46;;2311:272;2219:364;;;;:::o;2589:313::-;2702:4;2740:2;2729:9;2725:18;2717:26;;2789:9;2783:4;2779:20;2775:1;2764:9;2760:17;2753:47;2817:78;2890:4;2881:6;2817:78;:::i;:::-;2809:86;;2589:313;;;;:::o;2908:77::-;2945:7;2974:5;2963:16;;2908:77;;;:::o;2991:122::-;3064:24;3082:5;3064:24;:::i;:::-;3057:5;3054:35;3044:63;;3103:1;3100;3093:12;3044:63;2991:122;:::o;3119:139::-;3165:5;3203:6;3190:20;3181:29;;3219:33;3246:5;3219:33;:::i;:::-;3119:139;;;;:::o;3264:329::-;3323:6;3372:2;3360:9;3351:7;3347:23;3343:32;3340:119;;;3378:79;;:::i;:::-;3340:119;3498:1;3523:53;3568:7;3559:6;3548:9;3544:22;3523:53;:::i;:::-;3513:63;;3469:117;3264:329;;;;:::o;3599:126::-;3636:7;3676:42;3669:5;3665:54;3654:65;;3599:126;;;:::o;3731:96::-;3768:7;3797:24;3815:5;3797:24;:::i;:::-;3786:35;;3731:96;;;:::o;3833:118::-;3920:24;3938:5;3920:24;:::i;:::-;3915:3;3908:37;3833:118;;:::o;3957:222::-;4050:4;4088:2;4077:9;4073:18;4065:26;;4101:71;4169:1;4158:9;4154:17;4145:6;4101:71;:::i;:::-;3957:222;;;;:::o;4185:122::-;4258:24;4276:5;4258:24;:::i;:::-;4251:5;4248:35;4238:63;;4297:1;4294;4287:12;4238:63;4185:122;:::o;4313:139::-;4359:5;4397:6;4384:20;4375:29;;4413:33;4440:5;4413:33;:::i;:::-;4313:139;;;;:::o;4458:474::-;4526:6;4534;4583:2;4571:9;4562:7;4558:23;4554:32;4551:119;;;4589:79;;:::i;:::-;4551:119;4709:1;4734:53;4779:7;4770:6;4759:9;4755:22;4734:53;:::i;:::-;4724:63;;4680:117;4836:2;4862:53;4907:7;4898:6;4887:9;4883:22;4862:53;:::i;:::-;4852:63;;4807:118;4458:474;;;;;:::o;4938:118::-;5025:24;5043:5;5025:24;:::i;:::-;5020:3;5013:37;4938:118;;:::o;5062:222::-;5155:4;5193:2;5182:9;5178:18;5170:26;;5206:71;5274:1;5263:9;5259:17;5250:6;5206:71;:::i;:::-;5062:222;;;;:::o;5290:619::-;5367:6;5375;5383;5432:2;5420:9;5411:7;5407:23;5403:32;5400:119;;;5438:79;;:::i;:::-;5400:119;5558:1;5583:53;5628:7;5619:6;5608:9;5604:22;5583:53;:::i;:::-;5573:63;;5529:117;5685:2;5711:53;5756:7;5747:6;5736:9;5732:22;5711:53;:::i;:::-;5701:63;;5656:118;5813:2;5839:53;5884:7;5875:6;5864:9;5860:22;5839:53;:::i;:::-;5829:63;;5784:118;5290:619;;;;;:::o;5915:116::-;5985:21;6000:5;5985:21;:::i;:::-;5978:5;5975:32;5965:60;;6021:1;6018;6011:12;5965:60;5915:116;:::o;6037:133::-;6080:5;6118:6;6105:20;6096:29;;6134:30;6158:5;6134:30;:::i;:::-;6037:133;;;;:::o;6176:323::-;6232:6;6281:2;6269:9;6260:7;6256:23;6252:32;6249:119;;;6287:79;;:::i;:::-;6249:119;6407:1;6432:50;6474:7;6465:6;6454:9;6450:22;6432:50;:::i;:::-;6422:60;;6378:114;6176:323;;;;:::o;6505:117::-;6614:1;6611;6604:12;6628:117;6737:1;6734;6727:12;6751:180;6799:77;6796:1;6789:88;6896:4;6893:1;6886:15;6920:4;6917:1;6910:15;6937:281;7020:27;7042:4;7020:27;:::i;:::-;7012:6;7008:40;7150:6;7138:10;7135:22;7114:18;7102:10;7099:34;7096:62;7093:88;;;7161:18;;:::i;:::-;7093:88;7201:10;7197:2;7190:22;6980:238;6937:281;;:::o;7224:129::-;7258:6;7285:20;;:::i;:::-;7275:30;;7314:33;7342:4;7334:6;7314:33;:::i;:::-;7224:129;;;:::o;7359:308::-;7421:4;7511:18;7503:6;7500:30;7497:56;;;7533:18;;:::i;:::-;7497:56;7571:29;7593:6;7571:29;:::i;:::-;7563:37;;7655:4;7649;7645:15;7637:23;;7359:308;;;:::o;7673:154::-;7757:6;7752:3;7747;7734:30;7819:1;7810:6;7805:3;7801:16;7794:27;7673:154;;;:::o;7833:412::-;7911:5;7936:66;7952:49;7994:6;7952:49;:::i;:::-;7936:66;:::i;:::-;7927:75;;8025:6;8018:5;8011:21;8063:4;8056:5;8052:16;8101:3;8092:6;8087:3;8083:16;8080:25;8077:112;;;8108:79;;:::i;:::-;8077:112;8198:41;8232:6;8227:3;8222;8198:41;:::i;:::-;7917:328;7833:412;;;;;:::o;8265:340::-;8321:5;8370:3;8363:4;8355:6;8351:17;8347:27;8337:122;;8378:79;;:::i;:::-;8337:122;8495:6;8482:20;8520:79;8595:3;8587:6;8580:4;8572:6;8568:17;8520:79;:::i;:::-;8511:88;;8327:278;8265:340;;;;:::o;8611:509::-;8680:6;8729:2;8717:9;8708:7;8704:23;8700:32;8697:119;;;8735:79;;:::i;:::-;8697:119;8883:1;8872:9;8868:17;8855:31;8913:18;8905:6;8902:30;8899:117;;;8935:79;;:::i;:::-;8899:117;9040:63;9095:7;9086:6;9075:9;9071:22;9040:63;:::i;:::-;9030:73;;8826:287;8611:509;;;;:::o;9126:329::-;9185:6;9234:2;9222:9;9213:7;9209:23;9205:32;9202:119;;;9240:79;;:::i;:::-;9202:119;9360:1;9385:53;9430:7;9421:6;9410:9;9406:22;9385:53;:::i;:::-;9375:63;;9331:117;9126:329;;;;:::o;9461:468::-;9526:6;9534;9583:2;9571:9;9562:7;9558:23;9554:32;9551:119;;;9589:79;;:::i;:::-;9551:119;9709:1;9734:53;9779:7;9770:6;9759:9;9755:22;9734:53;:::i;:::-;9724:63;;9680:117;9836:2;9862:50;9904:7;9895:6;9884:9;9880:22;9862:50;:::i;:::-;9852:60;;9807:115;9461:468;;;;;:::o;9935:307::-;9996:4;10086:18;10078:6;10075:30;10072:56;;;10108:18;;:::i;:::-;10072:56;10146:29;10168:6;10146:29;:::i;:::-;10138:37;;10230:4;10224;10220:15;10212:23;;9935:307;;;:::o;10248:410::-;10325:5;10350:65;10366:48;10407:6;10366:48;:::i;:::-;10350:65;:::i;:::-;10341:74;;10438:6;10431:5;10424:21;10476:4;10469:5;10465:16;10514:3;10505:6;10500:3;10496:16;10493:25;10490:112;;;10521:79;;:::i;:::-;10490:112;10611:41;10645:6;10640:3;10635;10611:41;:::i;:::-;10331:327;10248:410;;;;;:::o;10677:338::-;10732:5;10781:3;10774:4;10766:6;10762:17;10758:27;10748:122;;10789:79;;:::i;:::-;10748:122;10906:6;10893:20;10931:78;11005:3;10997:6;10990:4;10982:6;10978:17;10931:78;:::i;:::-;10922:87;;10738:277;10677:338;;;;:::o;11021:943::-;11116:6;11124;11132;11140;11189:3;11177:9;11168:7;11164:23;11160:33;11157:120;;;11196:79;;:::i;:::-;11157:120;11316:1;11341:53;11386:7;11377:6;11366:9;11362:22;11341:53;:::i;:::-;11331:63;;11287:117;11443:2;11469:53;11514:7;11505:6;11494:9;11490:22;11469:53;:::i;:::-;11459:63;;11414:118;11571:2;11597:53;11642:7;11633:6;11622:9;11618:22;11597:53;:::i;:::-;11587:63;;11542:118;11727:2;11716:9;11712:18;11699:32;11758:18;11750:6;11747:30;11744:117;;;11780:79;;:::i;:::-;11744:117;11885:62;11939:7;11930:6;11919:9;11915:22;11885:62;:::i;:::-;11875:72;;11670:287;11021:943;;;;;;;:::o;11970:474::-;12038:6;12046;12095:2;12083:9;12074:7;12070:23;12066:32;12063:119;;;12101:79;;:::i;:::-;12063:119;12221:1;12246:53;12291:7;12282:6;12271:9;12267:22;12246:53;:::i;:::-;12236:63;;12192:117;12348:2;12374:53;12419:7;12410:6;12399:9;12395:22;12374:53;:::i;:::-;12364:63;;12319:118;11970:474;;;;;:::o;12450:180::-;12498:77;12495:1;12488:88;12595:4;12592:1;12585:15;12619:4;12616:1;12609:15;12636:320;12680:6;12717:1;12711:4;12707:12;12697:22;;12764:1;12758:4;12754:12;12785:18;12775:81;;12841:4;12833:6;12829:17;12819:27;;12775:81;12903:2;12895:6;12892:14;12872:18;12869:38;12866:84;;;12922:18;;:::i;:::-;12866:84;12687:269;12636:320;;;:::o;12962:182::-;13102:34;13098:1;13090:6;13086:14;13079:58;12962:182;:::o;13150:366::-;13292:3;13313:67;13377:2;13372:3;13313:67;:::i;:::-;13306:74;;13389:93;13478:3;13389:93;:::i;:::-;13507:2;13502:3;13498:12;13491:19;;13150:366;;;:::o;13522:419::-;13688:4;13726:2;13715:9;13711:18;13703:26;;13775:9;13769:4;13765:20;13761:1;13750:9;13746:17;13739:47;13803:131;13929:4;13803:131;:::i;:::-;13795:139;;13522:419;;;:::o;13947:332::-;14068:4;14106:2;14095:9;14091:18;14083:26;;14119:71;14187:1;14176:9;14172:17;14163:6;14119:71;:::i;:::-;14200:72;14268:2;14257:9;14253:18;14244:6;14200:72;:::i;:::-;13947:332;;;;;:::o;14285:137::-;14339:5;14370:6;14364:13;14355:22;;14386:30;14410:5;14386:30;:::i;:::-;14285:137;;;;:::o;14428:345::-;14495:6;14544:2;14532:9;14523:7;14519:23;14515:32;14512:119;;;14550:79;;:::i;:::-;14512:119;14670:1;14695:61;14748:7;14739:6;14728:9;14724:22;14695:61;:::i;:::-;14685:71;;14641:125;14428:345;;;;:::o;14779:175::-;14919:27;14915:1;14907:6;14903:14;14896:51;14779:175;:::o;14960:366::-;15102:3;15123:67;15187:2;15182:3;15123:67;:::i;:::-;15116:74;;15199:93;15288:3;15199:93;:::i;:::-;15317:2;15312:3;15308:12;15301:19;;14960:366;;;:::o;15332:419::-;15498:4;15536:2;15525:9;15521:18;15513:26;;15585:9;15579:4;15575:20;15571:1;15560:9;15556:17;15549:47;15613:131;15739:4;15613:131;:::i;:::-;15605:139;;15332:419;;;:::o;15757:148::-;15859:11;15896:3;15881:18;;15757:148;;;;:::o;15911:377::-;16017:3;16045:39;16078:5;16045:39;:::i;:::-;16100:89;16182:6;16177:3;16100:89;:::i;:::-;16093:96;;16198:52;16243:6;16238:3;16231:4;16224:5;16220:16;16198:52;:::i;:::-;16275:6;16270:3;16266:16;16259:23;;16021:267;15911:377;;;;:::o;16294:435::-;16474:3;16496:95;16587:3;16578:6;16496:95;:::i;:::-;16489:102;;16608:95;16699:3;16690:6;16608:95;:::i;:::-;16601:102;;16720:3;16713:10;;16294:435;;;;;:::o;16735:155::-;16875:7;16871:1;16863:6;16859:14;16852:31;16735:155;:::o;16896:400::-;17056:3;17077:84;17159:1;17154:3;17077:84;:::i;:::-;17070:91;;17170:93;17259:3;17170:93;:::i;:::-;17288:1;17283:3;17279:11;17272:18;;16896:400;;;:::o;17302:541::-;17535:3;17557:95;17648:3;17639:6;17557:95;:::i;:::-;17550:102;;17669:148;17813:3;17669:148;:::i;:::-;17662:155;;17834:3;17827:10;;17302:541;;;;:::o;17849:225::-;17989:34;17985:1;17977:6;17973:14;17966:58;18058:8;18053:2;18045:6;18041:15;18034:33;17849:225;:::o;18080:366::-;18222:3;18243:67;18307:2;18302:3;18243:67;:::i;:::-;18236:74;;18319:93;18408:3;18319:93;:::i;:::-;18437:2;18432:3;18428:12;18421:19;;18080:366;;;:::o;18452:419::-;18618:4;18656:2;18645:9;18641:18;18633:26;;18705:9;18699:4;18695:20;18691:1;18680:9;18676:17;18669:47;18733:131;18859:4;18733:131;:::i;:::-;18725:139;;18452:419;;;:::o;18877:182::-;19017:34;19013:1;19005:6;19001:14;18994:58;18877:182;:::o;19065:366::-;19207:3;19228:67;19292:2;19287:3;19228:67;:::i;:::-;19221:74;;19304:93;19393:3;19304:93;:::i;:::-;19422:2;19417:3;19413:12;19406:19;;19065:366;;;:::o;19437:419::-;19603:4;19641:2;19630:9;19626:18;19618:26;;19690:9;19684:4;19680:20;19676:1;19665:9;19661:17;19654:47;19718:131;19844:4;19718:131;:::i;:::-;19710:139;;19437:419;;;:::o;19862:98::-;19913:6;19947:5;19941:12;19931:22;;19862:98;;;:::o;19966:168::-;20049:11;20083:6;20078:3;20071:19;20123:4;20118:3;20114:14;20099:29;;19966:168;;;;:::o;20140:360::-;20226:3;20254:38;20286:5;20254:38;:::i;:::-;20308:70;20371:6;20366:3;20308:70;:::i;:::-;20301:77;;20387:52;20432:6;20427:3;20420:4;20413:5;20409:16;20387:52;:::i;:::-;20464:29;20486:6;20464:29;:::i;:::-;20459:3;20455:39;20448:46;;20230:270;20140:360;;;;:::o;20506:640::-;20701:4;20739:3;20728:9;20724:19;20716:27;;20753:71;20821:1;20810:9;20806:17;20797:6;20753:71;:::i;:::-;20834:72;20902:2;20891:9;20887:18;20878:6;20834:72;:::i;:::-;20916;20984:2;20973:9;20969:18;20960:6;20916:72;:::i;:::-;21035:9;21029:4;21025:20;21020:2;21009:9;21005:18;20998:48;21063:76;21134:4;21125:6;21063:76;:::i;:::-;21055:84;;20506:640;;;;;;;:::o;21152:141::-;21208:5;21239:6;21233:13;21224:22;;21255:32;21281:5;21255:32;:::i;:::-;21152:141;;;;:::o;21299:349::-;21368:6;21417:2;21405:9;21396:7;21392:23;21388:32;21385:119;;;21423:79;;:::i;:::-;21385:119;21543:1;21568:63;21623:7;21614:6;21603:9;21599:22;21568:63;:::i;:::-;21558:73;;21514:127;21299:349;;;;:::o;21654:180::-;21702:77;21699:1;21692:88;21799:4;21796:1;21789:15;21823:4;21820:1;21813:15

Swarm Source

ipfs://38dedcb5a444c10fcaae8469acef54b6f4682e26aeaf2bc375591b7d6772129f
Loading...
Loading
Loading...
Loading
[ 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.