Address Details
contract
proxy

0x52D9f580945E74F604Cce57016c0d1fd36820269

Contract Name
RandomNFTMarket
Creator
0x8b2f36–66963a at 0x956aa0–0642df
Implementation
RandomMarketPlaceV2 | 0x01e833861eb2ed9e8f4f88e3feee555d8a993640
Balance
0 CELO ( )
Locked CELO Balance
0.00 CELO
Voting CELO Balance
0.00 CELO
Pending Unlocked Gold
0.00 CELO
Tokens
Fetching tokens...
Transactions
0 Transactions
Transfers
0 Transfers
Gas Used
Fetching gas used...
Last Balance Update
15819370
This contract has been verified via Sourcify. View contract in Sourcify repository
Contract name:
RandomNFTMarket




Optimization enabled
false
Compiler version
v0.8.3+commit.8d00100c




EVM Version
istanbul




Verified at
2022-12-01T16:50:20.183465Z

contracts/RamdomNFTMarket/RandomNFTMarket.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/utils/math/SafeMath.sol";
import "hardhat/console.sol";

import "./RandomCyberBoxMarketManager.sol";

contract RandomNFTMarket is RandomCyberBoxMarketManager {

    address ERC20Address_celo = address(0x471EcE3750Da237f93B8E339c536989b8978a438);
    address ERC20Address_test_celo = address(0xF194afDf50B03e69Bd7D057c1Aa9e10c9954E4C9);

    constructor(
        address _ownerAddress,
        address _devAddress,
        address _maketPlaceAddress
    ) public {
        ownerAddress = _ownerAddress;
        devAddress = _devAddress;
        implementation = _maketPlaceAddress;
    }

    function addNewRandomeNFT(
        string memory _nftName, 
        string memory _nftSymbol,
        address nftAddress
    ) public {
        addERC721Token(
            _nftName,
            _nftSymbol,
            nftAddress,
            "celo",
            "celo",
            ERC20Address_celo
        );
    }

    function listToken(
        address _nftAddress,
        uint256 tokenId,
        uint256 value
    ) external  {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        (
        address fromAddress, 
        uint256 minValue
        ) = getMarketPlaceToken().listToken(msg.sender, tokenId, value);
        emit RandomMarketTokenListed(
            nftToken.nftToken.tokenAddress,
            tokenId,
            fromAddress,
            minValue
        );
    }

    function delistToken(
        address _nftAddress,
        uint256 tokenId
    ) external {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        getMarketPlaceToken().delistToken(msg.sender, tokenId);
        emit RandomMarketTokenDelisted(
            nftToken.nftToken.tokenAddress,
            tokenId
        );
    }

    function cleanAllInvalidListings() external {
        for(uint256 nftId = 0; nftId < supportNFTs.length; nftId ++){
            address _nftAddress = supportNFTs[nftId];
            NFTToken memory nftToken =  getNFTToken(_nftAddress);
            if(nftToken.marketPlaceAddress != address(0)){
                selectNFT(_nftAddress);
                (uint256[] memory idList) = getMarketPlaceToken().cleanAllInvalidListings();
                for (uint256 i = 0; i < idList.length; i++) {
                    uint256 contract_id = idList[i];
                    emit RandomMarketCleanList(
                        nftToken.nftToken.tokenAddress,
                        contract_id
                    );
                }
                getMarketPlaceToken().deleteTempTokenIdStorage();
            }
        }
    }

    function buyToken(
        address _nftAddress,
        uint256 tokenId,
        uint256 value
    ) external payable{
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        (
            address sellerAddress, 
            address devAddress,
            address createrAddress,
            address producerAddress
            ) = getMarketPlaceToken().buyTokenPrepare(msg.sender, tokenId, value);

            uint256 payment_value = value;
            if(isEtherToken(getSelectedERC20Address()) == true){
                payment_value = msg.value;
            }
            address nftAddress = _nftAddress;
            uint256 sellerFee = getMarketPlaceToken().calculateSellerFee(payment_value);
            uint256 devFee = getMarketPlaceToken().calculateDevFee(payment_value);
            uint256 createrFee = getMarketPlaceToken().calculateCreaterFee(payment_value);
            uint256 producerFee = getMarketPlaceToken().calculateProducerFee(payment_value);
            sendERC20(getSelectedERC20Token(), msg.sender, sellerAddress, sellerFee);
            emit RandomMarketPayment(msg.sender, sellerAddress, tokenId, nftAddress, 0, sellerFee);
            sendERC20(getSelectedERC20Token(), msg.sender, devAddress, devFee);
            emit RandomMarketPayment(msg.sender, devAddress, tokenId, nftAddress, 1, devFee);
            if(createrAddress != address(0) && createrFee > 0){
                sendERC20(getSelectedERC20Token(), msg.sender, createrAddress, createrFee);
                emit RandomMarketPayment(msg.sender, createrAddress, tokenId, nftAddress, 2, createrFee);
            }
            if(producerAddress != address(0) && producerFee > 0){
                sendERC20(getSelectedERC20Token(), msg.sender, producerAddress, producerFee);
                emit RandomMarketPayment(msg.sender, producerAddress, tokenId, nftAddress, 3, producerFee);
            }
            getMarketPlaceToken().buyTokenComplete(msg.sender, tokenId);
            emit RandomMarketTokenBought(
                 nftAddress,
                 tokenId,
                 sellerAddress,
                 msg.sender,
                 payment_value,
                 sellerFee,
                 devFee + createrFee + producerFee
            );
    }

    function transfer(
        address _nftAddress,
        address to,
        uint256 tokenId
    )external {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        getMarketPlaceToken().transfer(msg.sender, to, tokenId);
        emit RandomMarketTokenTransfered(
            _nftAddress,
            tokenId,
            msg.sender,
            to,
            0
        );
    }

    function changePrice(address _nftAddress, uint256 tokenId, uint256 newPrice) external {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        getMarketPlaceToken().changePrice(msg.sender, tokenId, newPrice);
        emit RandomMarketTokenPriceChanged(
            _nftAddress,
            tokenId,
            newPrice
        );
    }

    function getTokenListing(
        uint256 tokenId
    ) public view 
    returns (InterfaceV2.Listing memory) {
        return getMarketPlaceToken().getTokenListing(tokenId);
    }

    function getAllTokenListings() external view returns (InterfaceV2.Listing[] memory) {
        return getMarketPlaceToken().getAllTokenListings();
    }

    function withdrawCelo(address receiver) external onlyDev {
        require(receiver != address(0), "transfer address must not 0x0");
        uint256 contractBalance = address(this).balance;
        require(contractBalance > 0, "contract must have celo");
        payable(receiver).transfer(contractBalance);
    }
    function withdrawERC20(address address20, address receiver) external onlyDev {
        require(receiver != address(0), "transfer address must not 0x0");
        IERC20 _paymentToken = IERC20(address20);
        uint256 contractBalance = _paymentToken.balanceOf(address(this));
        require(contractBalance > 0, "contract must have celo");
        _paymentToken.transfer(receiver, contractBalance);
    }
}

        

/_openzeppelin/contracts/access/Ownable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)

pragma solidity ^0.8.0;

import "../utils/Context.sol";

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

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

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

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

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

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

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

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

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

/_openzeppelin/contracts/security/ReentrancyGuard.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol)

pragma solidity ^0.8.0;

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

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

    uint256 private _status;

    constructor() {
        _status = _NOT_ENTERED;
    }

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

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

        _;

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

/_openzeppelin/contracts/token/ERC1155/IERC1155.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC1155/IERC1155.sol)

pragma solidity ^0.8.0;

import "../../utils/introspection/IERC165.sol";

/**
 * @dev Required interface of an ERC1155 compliant contract, as defined in the
 * https://eips.ethereum.org/EIPS/eip-1155[EIP].
 *
 * _Available since v3.1._
 */
interface IERC1155 is IERC165 {
    /**
     * @dev Emitted when `value` tokens of token type `id` are transferred from `from` to `to` by `operator`.
     */
    event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);

    /**
     * @dev Equivalent to multiple {TransferSingle} events, where `operator`, `from` and `to` are the same for all
     * transfers.
     */
    event TransferBatch(
        address indexed operator,
        address indexed from,
        address indexed to,
        uint256[] ids,
        uint256[] values
    );

    /**
     * @dev Emitted when `account` grants or revokes permission to `operator` to transfer their tokens, according to
     * `approved`.
     */
    event ApprovalForAll(address indexed account, address indexed operator, bool approved);

    /**
     * @dev Emitted when the URI for token type `id` changes to `value`, if it is a non-programmatic URI.
     *
     * If an {URI} event was emitted for `id`, the standard
     * https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[guarantees] that `value` will equal the value
     * returned by {IERC1155MetadataURI-uri}.
     */
    event URI(string value, uint256 indexed id);

    /**
     * @dev Returns the amount of tokens of token type `id` owned by `account`.
     *
     * Requirements:
     *
     * - `account` cannot be the zero address.
     */
    function balanceOf(address account, uint256 id) external view returns (uint256);

    /**
     * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {balanceOf}.
     *
     * Requirements:
     *
     * - `accounts` and `ids` must have the same length.
     */
    function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids)
        external
        view
        returns (uint256[] memory);

    /**
     * @dev Grants or revokes permission to `operator` to transfer the caller's tokens, according to `approved`,
     *
     * Emits an {ApprovalForAll} event.
     *
     * Requirements:
     *
     * - `operator` cannot be the caller.
     */
    function setApprovalForAll(address operator, bool approved) external;

    /**
     * @dev Returns true if `operator` is approved to transfer ``account``'s tokens.
     *
     * See {setApprovalForAll}.
     */
    function isApprovedForAll(address account, address operator) external view returns (bool);

    /**
     * @dev Transfers `amount` tokens of token type `id` from `from` to `to`.
     *
     * Emits a {TransferSingle} event.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - If the caller is not `from`, it must have been approved to spend ``from``'s tokens via {setApprovalForAll}.
     * - `from` must have a balance of tokens of type `id` of at least `amount`.
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
     * acceptance magic value.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 id,
        uint256 amount,
        bytes calldata data
    ) external;

    /**
     * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {safeTransferFrom}.
     *
     * Emits a {TransferBatch} event.
     *
     * Requirements:
     *
     * - `ids` and `amounts` must have the same length.
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
     * acceptance magic value.
     */
    function safeBatchTransferFrom(
        address from,
        address to,
        uint256[] calldata ids,
        uint256[] calldata amounts,
        bytes calldata data
    ) external;
}
          

/_openzeppelin/contracts/token/ERC20/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);
}
          

/_openzeppelin/contracts/token/ERC721/IERC721.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol)

pragma solidity ^0.8.0;

import "../../utils/introspection/IERC165.sol";

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

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

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

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

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

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

    /**
     * @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 have been allowed to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external;

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

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

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

/_openzeppelin/contracts/utils/Address.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)

pragma solidity ^0.8.1;

/**
 * @dev Collection of functions related to the address type
 */
library Address {
    /**
     * @dev Returns true if `account` is a contract.
     *
     * [IMPORTANT]
     * ====
     * It is unsafe to assume that an address for which this function returns
     * false is an externally-owned account (EOA) and not a contract.
     *
     * Among others, `isContract` will return false for the following
     * types of addresses:
     *
     *  - an externally-owned account
     *  - a contract in construction
     *  - an address where a contract will be created
     *  - an address where a contract lived, but was destroyed
     * ====
     *
     * [IMPORTANT]
     * ====
     * You shouldn't rely on `isContract` to protect against flash loan attacks!
     *
     * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
     * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
     * constructor.
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize/address.code.length, which returns 0
        // for contracts in construction, since the code is only stored at the end
        // of the constructor execution.

        return account.code.length > 0;
    }

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

        (bool success, ) = recipient.call{value: amount}("");
        require(success, "Address: unable to send value, recipient may have reverted");
    }

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

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

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

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

        (bool success, bytes memory returndata) = target.call{value: value}(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

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

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

        (bool success, bytes memory returndata) = target.staticcall(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

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

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

        (bool success, bytes memory returndata) = target.delegatecall(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
     * revert reason using the provided one.
     *
     * _Available since v4.3._
     */
    function verifyCallResult(
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) internal pure returns (bytes memory) {
        if (success) {
            return returndata;
        } else {
            // Look for revert reason and bubble it up if present
            if (returndata.length > 0) {
                // The easiest way to bubble the revert reason is using memory via assembly
                /// @solidity memory-safe-assembly
                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}
          

/_openzeppelin/contracts/utils/Context.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)

pragma solidity ^0.8.0;

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

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

/_openzeppelin/contracts/utils/Strings.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)

pragma solidity ^0.8.0;

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

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

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

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
     */
    function toHexString(uint256 value) internal pure returns (string memory) {
        if (value == 0) {
            return "0x00";
        }
        uint256 temp = value;
        uint256 length = 0;
        while (temp != 0) {
            length++;
            temp >>= 8;
        }
        return toHexString(value, length);
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
     */
    function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
        bytes memory buffer = new bytes(2 * length + 2);
        buffer[0] = "0";
        buffer[1] = "x";
        for (uint256 i = 2 * length + 1; i > 1; --i) {
            buffer[i] = _HEX_SYMBOLS[value & 0xf];
            value >>= 4;
        }
        require(value == 0, "Strings: hex length insufficient");
        return string(buffer);
    }

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

/_openzeppelin/contracts/utils/introspection/IERC165.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)

pragma solidity ^0.8.0;

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

/_openzeppelin/contracts/utils/math/SafeMath.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.6.0) (utils/math/SafeMath.sol)

pragma solidity ^0.8.0;

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

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

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

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

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

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

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

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

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

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

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

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

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

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

/_openzeppelin/contracts/utils/structs/EnumerableSet.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (utils/structs/EnumerableSet.sol)

pragma solidity ^0.8.0;

/**
 * @dev Library for managing
 * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive
 * types.
 *
 * Sets have the following properties:
 *
 * - Elements are added, removed, and checked for existence in constant time
 * (O(1)).
 * - Elements are enumerated in O(n). No guarantees are made on the ordering.
 *
 * ```
 * contract Example {
 *     // Add the library methods
 *     using EnumerableSet for EnumerableSet.AddressSet;
 *
 *     // Declare a set state variable
 *     EnumerableSet.AddressSet private mySet;
 * }
 * ```
 *
 * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`)
 * and `uint256` (`UintSet`) are supported.
 *
 * [WARNING]
 * ====
 *  Trying to delete such a structure from storage will likely result in data corruption, rendering the structure unusable.
 *  See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info.
 *
 *  In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an array of EnumerableSet.
 * ====
 */
library EnumerableSet {
    // To implement this library for multiple types with as little code
    // repetition as possible, we write it in terms of a generic Set type with
    // bytes32 values.
    // The Set implementation uses private functions, and user-facing
    // implementations (such as AddressSet) are just wrappers around the
    // underlying Set.
    // This means that we can only create new EnumerableSets for types that fit
    // in bytes32.

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

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

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

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

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

            if (lastIndex != toDeleteIndex) {
                bytes32 lastValue = set._values[lastIndex];

                // Move the last value to the index where the value to delete is
                set._values[toDeleteIndex] = lastValue;
                // Update the index for the moved value
                set._indexes[lastValue] = valueIndex; // Replace lastValue's index to valueIndex
            }

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

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

            return true;
        } else {
            return false;
        }
    }

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

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

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

    /**
     * @dev Return the entire set in an array
     *
     * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
     * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
     * this function has an unbounded cost, and using it as part of a state-changing function may render the function
     * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
     */
    function _values(Set storage set) private view returns (bytes32[] memory) {
        return set._values;
    }

    // Bytes32Set

    struct Bytes32Set {
        Set _inner;
    }

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

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

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

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

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

    /**
     * @dev Return the entire set in an array
     *
     * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
     * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
     * this function has an unbounded cost, and using it as part of a state-changing function may render the function
     * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
     */
    function values(Bytes32Set storage set) internal view returns (bytes32[] memory) {
        return _values(set._inner);
    }

    // AddressSet

    struct AddressSet {
        Set _inner;
    }

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

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

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

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

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

    /**
     * @dev Return the entire set in an array
     *
     * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
     * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
     * this function has an unbounded cost, and using it as part of a state-changing function may render the function
     * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
     */
    function values(AddressSet storage set) internal view returns (address[] memory) {
        bytes32[] memory store = _values(set._inner);
        address[] memory result;

        /// @solidity memory-safe-assembly
        assembly {
            result := store
        }

        return result;
    }

    // UintSet

    struct UintSet {
        Set _inner;
    }

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

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

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

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

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

    /**
     * @dev Return the entire set in an array
     *
     * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed
     * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that
     * this function has an unbounded cost, and using it as part of a state-changing function may render the function
     * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block.
     */
    function values(UintSet storage set) internal view returns (uint256[] memory) {
        bytes32[] memory store = _values(set._inner);
        uint256[] memory result;

        /// @solidity memory-safe-assembly
        assembly {
            result := store
        }

        return result;
    }
}
          

/_openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)

pragma solidity ^0.8.0;

import "../utils/ContextUpgradeable.sol";
import "../proxy/utils/Initializable.sol";

/**
 * @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 OwnableUpgradeable is Initializable, ContextUpgradeable {
    address private _owner;

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

    /**
     * @dev Initializes the contract setting the deployer as the initial owner.
     */
    function __Ownable_init() internal onlyInitializing {
        __Ownable_init_unchained();
    }

    function __Ownable_init_unchained() internal onlyInitializing {
        _transferOwnership(_msgSender());
    }

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

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

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

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

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

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

    /**
     * @dev This empty reserved space is put in place to allow future versions to add new
     * variables without shifting down storage in the inheritance chain.
     * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
     */
    uint256[49] private __gap;
}
          

/_openzeppelin/contracts-upgradeable/proxy/ClonesUpgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (proxy/Clones.sol)

pragma solidity ^0.8.0;

/**
 * @dev https://eips.ethereum.org/EIPS/eip-1167[EIP 1167] is a standard for
 * deploying minimal proxy contracts, also known as "clones".
 *
 * > To simply and cheaply clone contract functionality in an immutable way, this standard specifies
 * > a minimal bytecode implementation that delegates all calls to a known, fixed address.
 *
 * The library includes functions to deploy a proxy using either `create` (traditional deployment) or `create2`
 * (salted deterministic deployment). It also includes functions to predict the addresses of clones deployed using the
 * deterministic method.
 *
 * _Available since v3.4._
 */
library ClonesUpgradeable {
    /**
     * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`.
     *
     * This function uses the create opcode, which should never revert.
     */
    function clone(address implementation) internal returns (address instance) {
        /// @solidity memory-safe-assembly
        assembly {
            let ptr := mload(0x40)
            mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)
            mstore(add(ptr, 0x14), shl(0x60, implementation))
            mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000)
            instance := create(0, ptr, 0x37)
        }
        require(instance != address(0), "ERC1167: create failed");
    }

    /**
     * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`.
     *
     * This function uses the create2 opcode and a `salt` to deterministically deploy
     * the clone. Using the same `implementation` and `salt` multiple time will revert, since
     * the clones cannot be deployed twice at the same address.
     */
    function cloneDeterministic(address implementation, bytes32 salt) internal returns (address instance) {
        /// @solidity memory-safe-assembly
        assembly {
            let ptr := mload(0x40)
            mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)
            mstore(add(ptr, 0x14), shl(0x60, implementation))
            mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000)
            instance := create2(0, ptr, 0x37, salt)
        }
        require(instance != address(0), "ERC1167: create2 failed");
    }

    /**
     * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}.
     */
    function predictDeterministicAddress(
        address implementation,
        bytes32 salt,
        address deployer
    ) internal pure returns (address predicted) {
        /// @solidity memory-safe-assembly
        assembly {
            let ptr := mload(0x40)
            mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000)
            mstore(add(ptr, 0x14), shl(0x60, implementation))
            mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf3ff00000000000000000000000000000000)
            mstore(add(ptr, 0x38), shl(0x60, deployer))
            mstore(add(ptr, 0x4c), salt)
            mstore(add(ptr, 0x6c), keccak256(ptr, 0x37))
            predicted := keccak256(add(ptr, 0x37), 0x55)
        }
    }

    /**
     * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}.
     */
    function predictDeterministicAddress(address implementation, bytes32 salt)
        internal
        view
        returns (address predicted)
    {
        return predictDeterministicAddress(implementation, salt, address(this));
    }
}
          

/_openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (proxy/utils/Initializable.sol)

pragma solidity ^0.8.2;

import "../../utils/AddressUpgradeable.sol";

/**
 * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed
 * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an
 * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer
 * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect.
 *
 * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be
 * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in
 * case an upgrade adds a module that needs to be initialized.
 *
 * For example:
 *
 * [.hljs-theme-light.nopadding]
 * ```
 * contract MyToken is ERC20Upgradeable {
 *     function initialize() initializer public {
 *         __ERC20_init("MyToken", "MTK");
 *     }
 * }
 * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable {
 *     function initializeV2() reinitializer(2) public {
 *         __ERC20Permit_init("MyToken");
 *     }
 * }
 * ```
 *
 * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as
 * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}.
 *
 * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure
 * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity.
 *
 * [CAUTION]
 * ====
 * Avoid leaving a contract uninitialized.
 *
 * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation
 * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke
 * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed:
 *
 * [.hljs-theme-light.nopadding]
 * ```
 * /// @custom:oz-upgrades-unsafe-allow constructor
 * constructor() {
 *     _disableInitializers();
 * }
 * ```
 * ====
 */
abstract contract Initializable {
    /**
     * @dev Indicates that the contract has been initialized.
     * @custom:oz-retyped-from bool
     */
    uint8 private _initialized;

    /**
     * @dev Indicates that the contract is in the process of being initialized.
     */
    bool private _initializing;

    /**
     * @dev Triggered when the contract has been initialized or reinitialized.
     */
    event Initialized(uint8 version);

    /**
     * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope,
     * `onlyInitializing` functions can be used to initialize parent contracts. Equivalent to `reinitializer(1)`.
     */
    modifier initializer() {
        bool isTopLevelCall = !_initializing;
        require(
            (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1),
            "Initializable: contract is already initialized"
        );
        _initialized = 1;
        if (isTopLevelCall) {
            _initializing = true;
        }
        _;
        if (isTopLevelCall) {
            _initializing = false;
            emit Initialized(1);
        }
    }

    /**
     * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the
     * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be
     * used to initialize parent contracts.
     *
     * `initializer` is equivalent to `reinitializer(1)`, so a reinitializer may be used after the original
     * initialization step. This is essential to configure modules that are added through upgrades and that require
     * initialization.
     *
     * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in
     * a contract, executing them in the right order is up to the developer or operator.
     */
    modifier reinitializer(uint8 version) {
        require(!_initializing && _initialized < version, "Initializable: contract is already initialized");
        _initialized = version;
        _initializing = true;
        _;
        _initializing = false;
        emit Initialized(version);
    }

    /**
     * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the
     * {initializer} and {reinitializer} modifiers, directly or indirectly.
     */
    modifier onlyInitializing() {
        require(_initializing, "Initializable: contract is not initializing");
        _;
    }

    /**
     * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call.
     * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized
     * to any version. It is recommended to use this to lock implementation contracts that are designed to be called
     * through proxies.
     */
    function _disableInitializers() internal virtual {
        require(!_initializing, "Initializable: contract is initializing");
        if (_initialized < type(uint8).max) {
            _initialized = type(uint8).max;
            emit Initialized(type(uint8).max);
        }
    }
}
          

/_openzeppelin/contracts-upgradeable/token/ERC1155/ERC1155Upgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC1155/ERC1155.sol)

pragma solidity ^0.8.0;

import "./IERC1155Upgradeable.sol";
import "./IERC1155ReceiverUpgradeable.sol";
import "./extensions/IERC1155MetadataURIUpgradeable.sol";
import "../../utils/AddressUpgradeable.sol";
import "../../utils/ContextUpgradeable.sol";
import "../../utils/introspection/ERC165Upgradeable.sol";
import "../../proxy/utils/Initializable.sol";

/**
 * @dev Implementation of the basic standard multi-token.
 * See https://eips.ethereum.org/EIPS/eip-1155
 * Originally based on code by Enjin: https://github.com/enjin/erc-1155
 *
 * _Available since v3.1._
 */
contract ERC1155Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC1155Upgradeable, IERC1155MetadataURIUpgradeable {
    using AddressUpgradeable for address;

    // Mapping from token ID to account balances
    mapping(uint256 => mapping(address => uint256)) private _balances;

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

    // Used as the URI for all token types by relying on ID substitution, e.g. https://token-cdn-domain/{id}.json
    string private _uri;

    /**
     * @dev See {_setURI}.
     */
    function __ERC1155_init(string memory uri_) internal onlyInitializing {
        __ERC1155_init_unchained(uri_);
    }

    function __ERC1155_init_unchained(string memory uri_) internal onlyInitializing {
        _setURI(uri_);
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) {
        return
            interfaceId == type(IERC1155Upgradeable).interfaceId ||
            interfaceId == type(IERC1155MetadataURIUpgradeable).interfaceId ||
            super.supportsInterface(interfaceId);
    }

    /**
     * @dev See {IERC1155MetadataURI-uri}.
     *
     * This implementation returns the same URI for *all* token types. It relies
     * on the token type ID substitution mechanism
     * https://eips.ethereum.org/EIPS/eip-1155#metadata[defined in the EIP].
     *
     * Clients calling this function must replace the `\{id\}` substring with the
     * actual token type ID.
     */
    function uri(uint256) public view virtual override returns (string memory) {
        return _uri;
    }

    /**
     * @dev See {IERC1155-balanceOf}.
     *
     * Requirements:
     *
     * - `account` cannot be the zero address.
     */
    function balanceOf(address account, uint256 id) public view virtual override returns (uint256) {
        require(account != address(0), "ERC1155: address zero is not a valid owner");
        return _balances[id][account];
    }

    /**
     * @dev See {IERC1155-balanceOfBatch}.
     *
     * Requirements:
     *
     * - `accounts` and `ids` must have the same length.
     */
    function balanceOfBatch(address[] memory accounts, uint256[] memory ids)
        public
        view
        virtual
        override
        returns (uint256[] memory)
    {
        require(accounts.length == ids.length, "ERC1155: accounts and ids length mismatch");

        uint256[] memory batchBalances = new uint256[](accounts.length);

        for (uint256 i = 0; i < accounts.length; ++i) {
            batchBalances[i] = balanceOf(accounts[i], ids[i]);
        }

        return batchBalances;
    }

    /**
     * @dev See {IERC1155-setApprovalForAll}.
     */
    function setApprovalForAll(address operator, bool approved) public virtual override {
        _setApprovalForAll(_msgSender(), operator, approved);
    }

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

    /**
     * @dev See {IERC1155-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 id,
        uint256 amount,
        bytes memory data
    ) public virtual override {
        require(
            from == _msgSender() || isApprovedForAll(from, _msgSender()),
            "ERC1155: caller is not token owner nor approved"
        );
        _safeTransferFrom(from, to, id, amount, data);
    }

    /**
     * @dev See {IERC1155-safeBatchTransferFrom}.
     */
    function safeBatchTransferFrom(
        address from,
        address to,
        uint256[] memory ids,
        uint256[] memory amounts,
        bytes memory data
    ) public virtual override {
        require(
            from == _msgSender() || isApprovedForAll(from, _msgSender()),
            "ERC1155: caller is not token owner nor approved"
        );
        _safeBatchTransferFrom(from, to, ids, amounts, data);
    }

    /**
     * @dev Transfers `amount` tokens of token type `id` from `from` to `to`.
     *
     * Emits a {TransferSingle} event.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `from` must have a balance of tokens of type `id` of at least `amount`.
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
     * acceptance magic value.
     */
    function _safeTransferFrom(
        address from,
        address to,
        uint256 id,
        uint256 amount,
        bytes memory data
    ) internal virtual {
        require(to != address(0), "ERC1155: transfer to the zero address");

        address operator = _msgSender();
        uint256[] memory ids = _asSingletonArray(id);
        uint256[] memory amounts = _asSingletonArray(amount);

        _beforeTokenTransfer(operator, from, to, ids, amounts, data);

        uint256 fromBalance = _balances[id][from];
        require(fromBalance >= amount, "ERC1155: insufficient balance for transfer");
        unchecked {
            _balances[id][from] = fromBalance - amount;
        }
        _balances[id][to] += amount;

        emit TransferSingle(operator, from, to, id, amount);

        _afterTokenTransfer(operator, from, to, ids, amounts, data);

        _doSafeTransferAcceptanceCheck(operator, from, to, id, amount, data);
    }

    /**
     * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {_safeTransferFrom}.
     *
     * Emits a {TransferBatch} event.
     *
     * Requirements:
     *
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
     * acceptance magic value.
     */
    function _safeBatchTransferFrom(
        address from,
        address to,
        uint256[] memory ids,
        uint256[] memory amounts,
        bytes memory data
    ) internal virtual {
        require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");
        require(to != address(0), "ERC1155: transfer to the zero address");

        address operator = _msgSender();

        _beforeTokenTransfer(operator, from, to, ids, amounts, data);

        for (uint256 i = 0; i < ids.length; ++i) {
            uint256 id = ids[i];
            uint256 amount = amounts[i];

            uint256 fromBalance = _balances[id][from];
            require(fromBalance >= amount, "ERC1155: insufficient balance for transfer");
            unchecked {
                _balances[id][from] = fromBalance - amount;
            }
            _balances[id][to] += amount;
        }

        emit TransferBatch(operator, from, to, ids, amounts);

        _afterTokenTransfer(operator, from, to, ids, amounts, data);

        _doSafeBatchTransferAcceptanceCheck(operator, from, to, ids, amounts, data);
    }

    /**
     * @dev Sets a new URI for all token types, by relying on the token type ID
     * substitution mechanism
     * https://eips.ethereum.org/EIPS/eip-1155#metadata[defined in the EIP].
     *
     * By this mechanism, any occurrence of the `\{id\}` substring in either the
     * URI or any of the amounts in the JSON file at said URI will be replaced by
     * clients with the token type ID.
     *
     * For example, the `https://token-cdn-domain/\{id\}.json` URI would be
     * interpreted by clients as
     * `https://token-cdn-domain/000000000000000000000000000000000000000000000000000000000004cce0.json`
     * for token type ID 0x4cce0.
     *
     * See {uri}.
     *
     * Because these URIs cannot be meaningfully represented by the {URI} event,
     * this function emits no events.
     */
    function _setURI(string memory newuri) internal virtual {
        _uri = newuri;
    }

    /**
     * @dev Creates `amount` tokens of token type `id`, and assigns them to `to`.
     *
     * Emits a {TransferSingle} event.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
     * acceptance magic value.
     */
    function _mint(
        address to,
        uint256 id,
        uint256 amount,
        bytes memory data
    ) internal virtual {
        require(to != address(0), "ERC1155: mint to the zero address");

        address operator = _msgSender();
        uint256[] memory ids = _asSingletonArray(id);
        uint256[] memory amounts = _asSingletonArray(amount);

        _beforeTokenTransfer(operator, address(0), to, ids, amounts, data);

        _balances[id][to] += amount;
        emit TransferSingle(operator, address(0), to, id, amount);

        _afterTokenTransfer(operator, address(0), to, ids, amounts, data);

        _doSafeTransferAcceptanceCheck(operator, address(0), to, id, amount, data);
    }

    /**
     * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {_mint}.
     *
     * Emits a {TransferBatch} event.
     *
     * Requirements:
     *
     * - `ids` and `amounts` must have the same length.
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
     * acceptance magic value.
     */
    function _mintBatch(
        address to,
        uint256[] memory ids,
        uint256[] memory amounts,
        bytes memory data
    ) internal virtual {
        require(to != address(0), "ERC1155: mint to the zero address");
        require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");

        address operator = _msgSender();

        _beforeTokenTransfer(operator, address(0), to, ids, amounts, data);

        for (uint256 i = 0; i < ids.length; i++) {
            _balances[ids[i]][to] += amounts[i];
        }

        emit TransferBatch(operator, address(0), to, ids, amounts);

        _afterTokenTransfer(operator, address(0), to, ids, amounts, data);

        _doSafeBatchTransferAcceptanceCheck(operator, address(0), to, ids, amounts, data);
    }

    /**
     * @dev Destroys `amount` tokens of token type `id` from `from`
     *
     * Emits a {TransferSingle} event.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `from` must have at least `amount` tokens of token type `id`.
     */
    function _burn(
        address from,
        uint256 id,
        uint256 amount
    ) internal virtual {
        require(from != address(0), "ERC1155: burn from the zero address");

        address operator = _msgSender();
        uint256[] memory ids = _asSingletonArray(id);
        uint256[] memory amounts = _asSingletonArray(amount);

        _beforeTokenTransfer(operator, from, address(0), ids, amounts, "");

        uint256 fromBalance = _balances[id][from];
        require(fromBalance >= amount, "ERC1155: burn amount exceeds balance");
        unchecked {
            _balances[id][from] = fromBalance - amount;
        }

        emit TransferSingle(operator, from, address(0), id, amount);

        _afterTokenTransfer(operator, from, address(0), ids, amounts, "");
    }

    /**
     * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {_burn}.
     *
     * Emits a {TransferBatch} event.
     *
     * Requirements:
     *
     * - `ids` and `amounts` must have the same length.
     */
    function _burnBatch(
        address from,
        uint256[] memory ids,
        uint256[] memory amounts
    ) internal virtual {
        require(from != address(0), "ERC1155: burn from the zero address");
        require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");

        address operator = _msgSender();

        _beforeTokenTransfer(operator, from, address(0), ids, amounts, "");

        for (uint256 i = 0; i < ids.length; i++) {
            uint256 id = ids[i];
            uint256 amount = amounts[i];

            uint256 fromBalance = _balances[id][from];
            require(fromBalance >= amount, "ERC1155: burn amount exceeds balance");
            unchecked {
                _balances[id][from] = fromBalance - amount;
            }
        }

        emit TransferBatch(operator, from, address(0), ids, amounts);

        _afterTokenTransfer(operator, from, address(0), ids, amounts, "");
    }

    /**
     * @dev Approve `operator` to operate on all of `owner` tokens
     *
     * Emits an {ApprovalForAll} event.
     */
    function _setApprovalForAll(
        address owner,
        address operator,
        bool approved
    ) internal virtual {
        require(owner != operator, "ERC1155: setting approval status for self");
        _operatorApprovals[owner][operator] = approved;
        emit ApprovalForAll(owner, operator, approved);
    }

    /**
     * @dev Hook that is called before any token transfer. This includes minting
     * and burning, as well as batched variants.
     *
     * The same hook is called on both single and batched variants. For single
     * transfers, the length of the `ids` and `amounts` arrays will be 1.
     *
     * Calling conditions (for each `id` and `amount` pair):
     *
     * - When `from` and `to` are both non-zero, `amount` of ``from``'s tokens
     * of token type `id` will be  transferred to `to`.
     * - When `from` is zero, `amount` tokens of token type `id` will be minted
     * for `to`.
     * - when `to` is zero, `amount` of ``from``'s tokens of token type `id`
     * will be burned.
     * - `from` and `to` are never both zero.
     * - `ids` and `amounts` have the same, non-zero length.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _beforeTokenTransfer(
        address operator,
        address from,
        address to,
        uint256[] memory ids,
        uint256[] memory amounts,
        bytes memory data
    ) internal virtual {}

    /**
     * @dev Hook that is called after any token transfer. This includes minting
     * and burning, as well as batched variants.
     *
     * The same hook is called on both single and batched variants. For single
     * transfers, the length of the `id` and `amount` arrays will be 1.
     *
     * Calling conditions (for each `id` and `amount` pair):
     *
     * - When `from` and `to` are both non-zero, `amount` of ``from``'s tokens
     * of token type `id` will be  transferred to `to`.
     * - When `from` is zero, `amount` tokens of token type `id` will be minted
     * for `to`.
     * - when `to` is zero, `amount` of ``from``'s tokens of token type `id`
     * will be burned.
     * - `from` and `to` are never both zero.
     * - `ids` and `amounts` have the same, non-zero length.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _afterTokenTransfer(
        address operator,
        address from,
        address to,
        uint256[] memory ids,
        uint256[] memory amounts,
        bytes memory data
    ) internal virtual {}

    function _doSafeTransferAcceptanceCheck(
        address operator,
        address from,
        address to,
        uint256 id,
        uint256 amount,
        bytes memory data
    ) private {
        if (to.isContract()) {
            try IERC1155ReceiverUpgradeable(to).onERC1155Received(operator, from, id, amount, data) returns (bytes4 response) {
                if (response != IERC1155ReceiverUpgradeable.onERC1155Received.selector) {
                    revert("ERC1155: ERC1155Receiver rejected tokens");
                }
            } catch Error(string memory reason) {
                revert(reason);
            } catch {
                revert("ERC1155: transfer to non ERC1155Receiver implementer");
            }
        }
    }

    function _doSafeBatchTransferAcceptanceCheck(
        address operator,
        address from,
        address to,
        uint256[] memory ids,
        uint256[] memory amounts,
        bytes memory data
    ) private {
        if (to.isContract()) {
            try IERC1155ReceiverUpgradeable(to).onERC1155BatchReceived(operator, from, ids, amounts, data) returns (
                bytes4 response
            ) {
                if (response != IERC1155ReceiverUpgradeable.onERC1155BatchReceived.selector) {
                    revert("ERC1155: ERC1155Receiver rejected tokens");
                }
            } catch Error(string memory reason) {
                revert(reason);
            } catch {
                revert("ERC1155: transfer to non ERC1155Receiver implementer");
            }
        }
    }

    function _asSingletonArray(uint256 element) private pure returns (uint256[] memory) {
        uint256[] memory array = new uint256[](1);
        array[0] = element;

        return array;
    }

    /**
     * @dev This empty reserved space is put in place to allow future versions to add new
     * variables without shifting down storage in the inheritance chain.
     * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
     */
    uint256[47] private __gap;
}
          

/_openzeppelin/contracts-upgradeable/token/ERC1155/IERC1155ReceiverUpgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.5.0) (token/ERC1155/IERC1155Receiver.sol)

pragma solidity ^0.8.0;

import "../../utils/introspection/IERC165Upgradeable.sol";

/**
 * @dev _Available since v3.1._
 */
interface IERC1155ReceiverUpgradeable is IERC165Upgradeable {
    /**
     * @dev Handles the receipt of a single ERC1155 token type. This function is
     * called at the end of a `safeTransferFrom` after the balance has been updated.
     *
     * NOTE: To accept the transfer, this must return
     * `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))`
     * (i.e. 0xf23a6e61, or its own function selector).
     *
     * @param operator The address which initiated the transfer (i.e. msg.sender)
     * @param from The address which previously owned the token
     * @param id The ID of the token being transferred
     * @param value The amount of tokens being transferred
     * @param data Additional data with no specified format
     * @return `bytes4(keccak256("onERC1155Received(address,address,uint256,uint256,bytes)"))` if transfer is allowed
     */
    function onERC1155Received(
        address operator,
        address from,
        uint256 id,
        uint256 value,
        bytes calldata data
    ) external returns (bytes4);

    /**
     * @dev Handles the receipt of a multiple ERC1155 token types. This function
     * is called at the end of a `safeBatchTransferFrom` after the balances have
     * been updated.
     *
     * NOTE: To accept the transfer(s), this must return
     * `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))`
     * (i.e. 0xbc197c81, or its own function selector).
     *
     * @param operator The address which initiated the batch transfer (i.e. msg.sender)
     * @param from The address which previously owned the token
     * @param ids An array containing ids of each token being transferred (order and length must match values array)
     * @param values An array containing amounts of each token being transferred (order and length must match ids array)
     * @param data Additional data with no specified format
     * @return `bytes4(keccak256("onERC1155BatchReceived(address,address,uint256[],uint256[],bytes)"))` if transfer is allowed
     */
    function onERC1155BatchReceived(
        address operator,
        address from,
        uint256[] calldata ids,
        uint256[] calldata values,
        bytes calldata data
    ) external returns (bytes4);
}
          

/_openzeppelin/contracts-upgradeable/token/ERC1155/IERC1155Upgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC1155/IERC1155.sol)

pragma solidity ^0.8.0;

import "../../utils/introspection/IERC165Upgradeable.sol";

/**
 * @dev Required interface of an ERC1155 compliant contract, as defined in the
 * https://eips.ethereum.org/EIPS/eip-1155[EIP].
 *
 * _Available since v3.1._
 */
interface IERC1155Upgradeable is IERC165Upgradeable {
    /**
     * @dev Emitted when `value` tokens of token type `id` are transferred from `from` to `to` by `operator`.
     */
    event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value);

    /**
     * @dev Equivalent to multiple {TransferSingle} events, where `operator`, `from` and `to` are the same for all
     * transfers.
     */
    event TransferBatch(
        address indexed operator,
        address indexed from,
        address indexed to,
        uint256[] ids,
        uint256[] values
    );

    /**
     * @dev Emitted when `account` grants or revokes permission to `operator` to transfer their tokens, according to
     * `approved`.
     */
    event ApprovalForAll(address indexed account, address indexed operator, bool approved);

    /**
     * @dev Emitted when the URI for token type `id` changes to `value`, if it is a non-programmatic URI.
     *
     * If an {URI} event was emitted for `id`, the standard
     * https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[guarantees] that `value` will equal the value
     * returned by {IERC1155MetadataURI-uri}.
     */
    event URI(string value, uint256 indexed id);

    /**
     * @dev Returns the amount of tokens of token type `id` owned by `account`.
     *
     * Requirements:
     *
     * - `account` cannot be the zero address.
     */
    function balanceOf(address account, uint256 id) external view returns (uint256);

    /**
     * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {balanceOf}.
     *
     * Requirements:
     *
     * - `accounts` and `ids` must have the same length.
     */
    function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids)
        external
        view
        returns (uint256[] memory);

    /**
     * @dev Grants or revokes permission to `operator` to transfer the caller's tokens, according to `approved`,
     *
     * Emits an {ApprovalForAll} event.
     *
     * Requirements:
     *
     * - `operator` cannot be the caller.
     */
    function setApprovalForAll(address operator, bool approved) external;

    /**
     * @dev Returns true if `operator` is approved to transfer ``account``'s tokens.
     *
     * See {setApprovalForAll}.
     */
    function isApprovedForAll(address account, address operator) external view returns (bool);

    /**
     * @dev Transfers `amount` tokens of token type `id` from `from` to `to`.
     *
     * Emits a {TransferSingle} event.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - If the caller is not `from`, it must have been approved to spend ``from``'s tokens via {setApprovalForAll}.
     * - `from` must have a balance of tokens of type `id` of at least `amount`.
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
     * acceptance magic value.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 id,
        uint256 amount,
        bytes calldata data
    ) external;

    /**
     * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {safeTransferFrom}.
     *
     * Emits a {TransferBatch} event.
     *
     * Requirements:
     *
     * - `ids` and `amounts` must have the same length.
     * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
     * acceptance magic value.
     */
    function safeBatchTransferFrom(
        address from,
        address to,
        uint256[] calldata ids,
        uint256[] calldata amounts,
        bytes calldata data
    ) external;
}
          

/_openzeppelin/contracts-upgradeable/token/ERC1155/extensions/IERC1155MetadataURIUpgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC1155/extensions/IERC1155MetadataURI.sol)

pragma solidity ^0.8.0;

import "../IERC1155Upgradeable.sol";

/**
 * @dev Interface of the optional ERC1155MetadataExtension interface, as defined
 * in the https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[EIP].
 *
 * _Available since v3.1._
 */
interface IERC1155MetadataURIUpgradeable is IERC1155Upgradeable {
    /**
     * @dev Returns the URI for token type `id`.
     *
     * If the `\{id\}` substring is present in the URI, it must be replaced by
     * clients with the actual token type ID.
     */
    function uri(uint256 id) external view returns (string memory);
}
          

/_openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)

pragma solidity ^0.8.1;

/**
 * @dev Collection of functions related to the address type
 */
library AddressUpgradeable {
    /**
     * @dev Returns true if `account` is a contract.
     *
     * [IMPORTANT]
     * ====
     * It is unsafe to assume that an address for which this function returns
     * false is an externally-owned account (EOA) and not a contract.
     *
     * Among others, `isContract` will return false for the following
     * types of addresses:
     *
     *  - an externally-owned account
     *  - a contract in construction
     *  - an address where a contract will be created
     *  - an address where a contract lived, but was destroyed
     * ====
     *
     * [IMPORTANT]
     * ====
     * You shouldn't rely on `isContract` to protect against flash loan attacks!
     *
     * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
     * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
     * constructor.
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize/address.code.length, which returns 0
        // for contracts in construction, since the code is only stored at the end
        // of the constructor execution.

        return account.code.length > 0;
    }

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

        (bool success, ) = recipient.call{value: amount}("");
        require(success, "Address: unable to send value, recipient may have reverted");
    }

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

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

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

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

        (bool success, bytes memory returndata) = target.call{value: value}(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

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

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

        (bool success, bytes memory returndata) = target.staticcall(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
     * revert reason using the provided one.
     *
     * _Available since v4.3._
     */
    function verifyCallResult(
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) internal pure returns (bytes memory) {
        if (success) {
            return returndata;
        } else {
            // Look for revert reason and bubble it up if present
            if (returndata.length > 0) {
                // The easiest way to bubble the revert reason is using memory via assembly
                /// @solidity memory-safe-assembly
                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}
          

/_openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)

pragma solidity ^0.8.0;
import "../proxy/utils/Initializable.sol";

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

    function __Context_init_unchained() internal onlyInitializing {
    }
    function _msgSender() internal view virtual returns (address) {
        return msg.sender;
    }

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

    /**
     * @dev This empty reserved space is put in place to allow future versions to add new
     * variables without shifting down storage in the inheritance chain.
     * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
     */
    uint256[50] private __gap;
}
          

/_openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Counters.sol)

pragma solidity ^0.8.0;

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

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

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

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

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

/_openzeppelin/contracts-upgradeable/utils/introspection/ERC165Upgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)

pragma solidity ^0.8.0;

import "./IERC165Upgradeable.sol";
import "../../proxy/utils/Initializable.sol";

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

    function __ERC165_init_unchained() internal onlyInitializing {
    }
    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        return interfaceId == type(IERC165Upgradeable).interfaceId;
    }

    /**
     * @dev This empty reserved space is put in place to allow future versions to add new
     * variables without shifting down storage in the inheritance chain.
     * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
     */
    uint256[50] private __gap;
}
          

/_openzeppelin/contracts-upgradeable/utils/introspection/IERC165Upgradeable.sol

// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)

pragma solidity ^0.8.0;

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

/contracts/MarketPlace/InterfaceV2.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

interface InterfaceV2 {
    struct Bid {
        uint256 tokenId;
        uint256 bidPrice;
        address bidder;
    }

    struct Listing {
        uint256 tokenId;
        uint256 listingPrice;
        address seller;
    }

    event TokenListed(
        uint256 indexed tokenId, 
        address indexed fromAddress, 
        uint256 minValue, 
        string nft);
    event TokenDelisted(uint256 indexed tokenId, address indexed fromAddress, string nft);
    event TokenBidEntered(uint256 indexed tokenId, address indexed fromAddress, uint256 value, string nft);
    event TokenBidWithdrawn(uint256 indexed tokenId, address indexed fromAddress, uint256 value, string nft);
    event TokenBought(
        uint256 indexed tokenId,
        address indexed fromAddress,
        address indexed toAddress,
        uint256 total,
        uint256 value,
        uint256 fees,
        string nft
    );
    event TokenBidAccepted(
        uint256 indexed tokenId,
        address indexed owner,
        address indexed bidder,
        uint256 total,
        uint256 value,
        uint256 fees,
        string nft
    );

    event TokenTransfered(
        uint256 indexed tokenId,
        address indexed from,
        address indexed to,
        uint256 total,
        string nft
    );

    event TokenFeeChanged(
        address nftAddress,
        string nft
    );
}
          

/contracts/MarketPlace/MarketPlaceFeeAPI.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

import "@openzeppelin/contracts/utils/math/SafeMath.sol";

contract MarketPlaceFeeAPI {

    using SafeMath for uint256;

    uint256 private _baseFeeTokenSeller;
    uint256 private _baseFeeTokenProducer;
    uint256 private _baseFeeTokenCreater;
    uint256 private _baseFeeTokenDev;
    uint256 private _baseFeeFraction;
    uint256 private _baseFeeTokenBase;

    address public maketPlaceFeeAddress;
    address public nftCreaterAddress;
    address public nftProducerAddress;

    function initializeFee(
        address _owner
    ) public {
        _baseFeeTokenSeller = 975;
        _baseFeeTokenProducer = 0;
        _baseFeeTokenCreater = 0;
        _baseFeeTokenDev = 25;
        _baseFeeFraction = 25;
        _baseFeeTokenBase = 1000;

        maketPlaceFeeAddress = _owner;
    }

    function calculateSellerFee(uint256 value) public returns(uint256){
        return value.sub(value.mul(_baseFeeFraction).div(_baseFeeTokenBase));
    }
    function calculateDevFee(uint256 value) public returns(uint256){
        return value.mul(_baseFeeTokenDev).div(_baseFeeTokenBase);
    }
    function calculateCreaterFee(uint256 value) public returns(uint256){
        return value.mul(_baseFeeTokenCreater).div(_baseFeeTokenBase);
    }
    function calculateProducerFee(uint256 value) public returns(uint256){
        return value.mul(_baseFeeTokenProducer).div(_baseFeeTokenBase);
    }
    function serviceFee(address nftAddress) external view returns (uint256, uint256, uint256, uint256, uint256, uint256) {
        require(_baseFeeTokenBase == 1000, "This token is not registed");
        return (
            _baseFeeTokenSeller, 
            _baseFeeTokenProducer, 
            _baseFeeTokenCreater, 
            _baseFeeTokenDev, 
            _baseFeeFraction, 
            _baseFeeTokenBase
        );
    }
    function setNFTFees(
        address _nftAddress,
        uint256 _feeCreater,
        uint256 _feeProducer
        )
        external
    {
        require(
            _feeCreater == 0 || nftCreaterAddress != address(0), "This token don't set creater address"
        );
        require(
            _feeProducer == 0 || nftProducerAddress != address(0), "This token don't set producer address"
        );

        _baseFeeTokenCreater = _feeCreater;
        _baseFeeTokenProducer = _feeProducer;
        _baseFeeTokenSeller = _baseFeeTokenBase - _baseFeeTokenCreater - _baseFeeTokenDev - _baseFeeTokenProducer;
        _baseFeeFraction = _baseFeeTokenCreater + _baseFeeTokenDev + _baseFeeTokenProducer;
    }

    function setMaketPlaceAddressAndDevFee(
        address _nftAddress,
        address _maketPlaceFeeAddress, 
        uint256 _maketPlaceFeePercentage)
        external
    {
        require(
            _maketPlaceFeePercentage > 0 && _maketPlaceFeePercentage <= 1000,
            "Allowed percentage range is 1 to 1000"
        );
        maketPlaceFeeAddress = _maketPlaceFeeAddress;
        require(
            1000 == _baseFeeTokenBase, "This token is not registed"
        );

        _baseFeeTokenDev = _maketPlaceFeePercentage;
        _baseFeeTokenSeller = _baseFeeTokenBase - _baseFeeTokenDev - _baseFeeTokenCreater - _baseFeeTokenProducer; 
        _baseFeeFraction = _baseFeeTokenDev + _baseFeeTokenCreater - _baseFeeTokenProducer;
    }

    function setTokenCreaterAddress(
        address _nftAddress,
        address _tokenCreaterAddress)
        external
    {
        require(_tokenCreaterAddress != address(0), "Can't set to address 0x0");
        require(
            1000 == _baseFeeTokenBase, "This token is not registed"
        );
        nftCreaterAddress = _tokenCreaterAddress;
    }

    function setTokenProducerAddress(
        address _nftAddress,
        address _tokenProducerAddress)
        external
    {
        require(_tokenProducerAddress != address(0), "Can't set to address 0x0");
        require(
            1000 == _baseFeeTokenBase, "This token is not registed"
        );
        nftProducerAddress = _tokenProducerAddress;
    }

    function changeTokenCreaterAddress(
        address _nftAddress,
        address _tokenCreaterAddress) external {
        nftCreaterAddress = _tokenCreaterAddress;
    }

}
          

/contracts/MarketPlace/MarketPlaceNFTAPI.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol";
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/utils/Address.sol";

import "hardhat/console.sol";

contract MarketPlaceNFTAPI {
    address public dev; // developer address

    string private _erc721Name;
    IERC721 private _erc721;
    string private _erc1155Name;
    IERC1155 private _erc1155;
    address private _selectedNftAddress;
    address private _selectedERC20Address;
    IERC20 private _paymentToken;

    bool private isSupport1555;

    modifier onlyDev() {
        require(msg.sender == dev, "auction: wrong developer");
        _;
    }
    function changeDev(address _newDev) public onlyDev {
        dev  = _newDev;
    }
    
    function initializeNFTWithERC721(
        string memory erc721Name_,
        address _erc721Address,
        address _paymentTokenAddress
    ) public {
        _erc721Name = erc721Name_;
        _erc721 = IERC721(_erc721Address);
        _selectedNftAddress = _erc721Address;
        _paymentToken = IERC20(_paymentTokenAddress);
        _selectedERC20Address = _paymentTokenAddress;

        dev = msg.sender;
        isSupport1555 = false;
    }
    function initializeNFTWithERC1155(
        string memory erc1155Name_,
        address _erc1155Address,
        address _paymentTokenAddress
    ) public {
        _erc1155Name = erc1155Name_;
        _erc1155 = IERC1155(_erc1155Address);
        _selectedNftAddress = _erc1155Address;
        _paymentToken = IERC20(_paymentTokenAddress);
        _selectedERC20Address = _paymentTokenAddress;

        dev = msg.sender;
        isSupport1555 = true;
    }
    /**
     * @dev check if the account is the owner of this erc721 token
     */
    function _isTokenOwner(uint256 tokenId, address account) public view returns (bool) {
        if(isSupport1555 == false){
            try _erc721.ownerOf(tokenId) returns (address tokenOwner) {
                return tokenOwner == account;
            } catch {
                return false;
            }
        }else{
            return _erc1155.balanceOf(account, tokenId) != 0;
        }
   }
   /**
     * @dev check if this contract has approved to transfer this erc721 token
     */
    function _isTokenApproved(uint256 tokenId) public view returns (bool) {
        if(isSupport1555 == false){
            try _erc721.getApproved(tokenId) returns (address tokenOperator) {
                return tokenOperator == address(this);
            } catch {
                return false;
            }
        }else{
            return true;
        }
    }
    /**
     * @dev check if this contract has approved to all of this owner's erc721 tokens
     */
    function _isAllTokenApproved(address owner) public view returns (bool) {
        if(isSupport1555 == false){
            return _erc721.isApprovedForAll(owner, address(this));
        }else{
            return _erc1155.isApprovedForAll(owner, address(this));
        }
    }

    /**
     * @dev See {INFTKEYMarketPlaceV1-tokenAddress}.
     */
    function nftAddress() external view returns (address) {
        return _selectedNftAddress;
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-paymentTokenAddress}.
     */
    function paymentTokenAddress() external view returns (address) {
        return _selectedERC20Address;
    }

    /**
     * @dev Transfer token to Other
     * Must be owner of this token
     * Must have approved this contract to transfer token
     * Must have a valid existing bid that matches the bidder address
    */
    function transfer(
        address sender,
        address to,
        uint256 tokenId
    ) external onlyDev {
        require(_isTokenOwner(tokenId, sender), "Only token owner can accept bid of token");
        require(
            _isTokenApproved(tokenId) || _isAllTokenApproved(sender),
            "The token is not approved to transfer by the contract"
        );
        if(isSupport1555 == false){
            _erc721.safeTransferFrom(sender, to, tokenId);
        }else{
            bytes memory data = '0x0';
            _erc1155.safeTransferFrom(sender, to, tokenId, 1, data);
        }
        
    }

    function nftTransferFrom(
        address sender,
        address to,
        uint256 tokenId
    ) public onlyDev {
        if(isSupport1555 == false){
            _erc721.safeTransferFrom(sender, to, tokenId);
        }else{
            bytes memory data = '0x0';
            _erc1155.safeTransferFrom(sender, to, tokenId, 1, data);
        }
    }

}
          

/contracts/MarketPlace/RandomMarketPlaceV2.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/utils/Address.sol";
import "@openzeppelin/contracts/utils/Strings.sol";

import "hardhat/console.sol";

import "./InterfaceV2.sol";
import "./MarketPlaceFeeAPI.sol";
import "./MarketPlaceNFTAPI.sol";

import {ERC1155Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC1155/ERC1155Upgradeable.sol";
import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol";

contract RandomMarketPlaceV2 is InterfaceV2, MarketPlaceFeeAPI, MarketPlaceNFTAPI, OwnableUpgradeable, ReentrancyGuard, ERC1155Upgradeable {
    using SafeMath for uint256;
    using Address for address;
    using EnumerableSet for EnumerableSet.UintSet;
    using EnumerableSet for EnumerableSet.AddressSet;

    struct TokenBid {
        EnumerableSet.AddressSet bidders;
        mapping(address => Bid) bids;
    }

    bool private _isListingAndBidEnabled = true;

    mapping(uint256 => Listing) private _tokenListings;
    EnumerableSet.UintSet private _tokenIdWithListing;

    mapping(uint256 => TokenBid) private _tokenBids;
    EnumerableSet.UintSet private _tokenIdWithBid;

    EnumerableSet.AddressSet private _emptyBidders;
    uint256[] private _tempTokenIdStorage; // Storage to assist cleaning
    address[] private _tempBidderStorage; // Storage to assist cleaning bids

    constructor() public {
    }

    function initializeWithERC721(
        string memory erc721Name_,
        address _erc721Address,
        address _paymentTokenAddress,
        address _owner
    ) public initializer {
        __ERC1155_init("");
        initializeNFTWithERC721(erc721Name_, _erc721Address, _paymentTokenAddress);
        initializeFee(_owner);
        _isListingAndBidEnabled = true;
    }

    function initializeWithERC1155(
        string memory erc1155Name_,
        address _erc1155Address,
        address _paymentTokenAddress,
        address _owner
    ) public initializer {
        __ERC1155_init("");
        initializeNFTWithERC1155(erc1155Name_, _erc1155Address, _paymentTokenAddress);
        initializeFee(_owner);
        _isListingAndBidEnabled = true;
    }

    modifier onlyMarketplaceOpen() {
        require(_isListingAndBidEnabled, "Listing and bid are not enabled");
        _;
    }

    function _isListingValid(Listing memory listing) private view returns (bool) {
        if (
            _isTokenOwner(listing.tokenId, listing.seller) &&
            (_isTokenApproved(listing.tokenId) || _isAllTokenApproved(listing.seller)) &&
            listing.listingPrice > 0) {
            return true;
        }
    }
    function getTokenListing(uint256 tokenId) public view returns (Listing memory) {
        Listing memory listing = _tokenListings[tokenId];
        if (_isListingValid(listing)) {
            return listing;
        }
    }
    function getTokenListings(uint256 from, uint256 size)
        public
        view
        returns (Listing[] memory)
    {
        if (from < _tokenIdWithListing.length() && size > 0) {
            uint256 querySize = size;
            if ((from + size) > _tokenIdWithListing.length()) {
                querySize = _tokenIdWithListing.length() - from;
            }
            Listing[] memory listings = new Listing[](querySize);
            for (uint256 i = 0; i < querySize; i++) {
                Listing memory listing = _tokenListings[_tokenIdWithListing.at(i + from)];
                if (_isListingValid(listing)) {
                    listings[i] = listing;
                }
            }
            return listings;
        }
    }
    function getAllTokenListings() external view returns (Listing[] memory) {
        return getTokenListings(0, _tokenIdWithListing.length());
    }
    function _delistToken(uint256 tokenId) private {
        if (_tokenIdWithListing.contains(tokenId)) {
            delete _tokenListings[tokenId];
            _tokenIdWithListing.remove(tokenId);
        }
    }
    /**
     * @dev List token for sale
     * @param tokenId erc721 token Id
     * @param value min price to sell the token
     */
    function listToken(
        address sender,
        uint256 tokenId,
        uint256 value
    ) external onlyMarketplaceOpen
     returns (
        address, 
        uint256
    ) {
        console.log("sender:", sender);
        require(value > 0, "Please list for more than 0 or use the transfer function");
        require(_isTokenOwner(tokenId, sender), "Only token owner can list token");
        require(
            _isTokenApproved(tokenId) || _isAllTokenApproved(sender),
            "This token is not allowed to transfer by this contract"
        );
        _tokenListings[tokenId] = Listing(tokenId, value, sender);
        _tokenIdWithListing.add(tokenId);
        return ( sender, value);
    }
    /**
     * @dev change price for already listed token.l
     
     * Must have a valid listing
     * msg.sender must not the owner of token
     */
    function changePrice(address sender,uint256 tokenId, uint256 newPrice) external nonReentrant {
        Listing memory listing = getTokenListing(tokenId); // Get valid listing
        require(_isTokenOwner(tokenId, sender), "Only token owner can change price of token");
        require(listing.seller != address(0), "Token is not for sale"); // Listing not valid
        require(
            newPrice >= 0,
            "The value send is below zero"
        );
        _tokenListings[tokenId].listingPrice = newPrice;
    }
     /**
     * @dev See {INFTKEYMarketPlaceV1-delistToken}.
     * msg.sender must be the seller of the listing record
     */
    function delistToken(address sender, uint256 tokenId) external {
        require(_tokenListings[tokenId].seller == sender, "Only token seller can delist token");
        _delistToken(tokenId);
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-buyToken}.
     * Must have a valid listing
     * msg.sender must not the owner of token
     * msg.value must be at least sell price plus fees
     */
    function buyTokenPrepare(address sender, uint256 tokenId, uint256 value) external onlyDev returns (
        address,
        address, 
        address,
        address
    ) {
        Listing memory listing = getTokenListing(tokenId); // Get valid listing
        require(listing.seller != address(0), "Token is not for sale"); // Listing not valid
        require(!_isTokenOwner(tokenId, sender), "Token owner can't buy their own token");
        require(
            value >= listing.listingPrice,
            
            "The value send is below sale price plus fees"
        );
        return (
            listing.seller,
            maketPlaceFeeAddress,
            nftCreaterAddress,
            nftProducerAddress
        );
    }
    function buyTokenComplete(address sender, uint256 tokenId) external onlyDev {


        
        Listing memory listing = getTokenListing(tokenId); // Get valid listing
         nftTransferFrom(listing.seller, sender, tokenId);
         // Remove token listing
        _delistToken(tokenId);
        _removeBidOfBidder(tokenId, sender);
    }

    /**
     * @dev Check if an bid is valid or not
     * Bidder must not be the owner
     * Bidder must give the contract allowance same or more than bid price
     * Bid price must > 0
     * Bid mustn't been expired
     */
    function _isBidValid(Bid memory bid) private view returns (bool) {
        if (
            !_isTokenOwner(bid.tokenId, bid.bidder) &&
            bid.bidPrice > 0) {
            return true;
        }
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-getBidderTokenBid}.
     */
    function getBidderTokenBid(uint256 tokenId, address bidder)
        public
        view
        returns (Bid memory)
    {
        Bid memory bid = _tokenBids[tokenId].bids[bidder];
        if (_isBidValid(bid)) {
            return bid;
        }
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-getTokenBids}.
     */
    function getTokenBids(uint256 tokenId) external view returns (Bid[] memory) {
        Bid[] memory bids = new Bid[](_tokenBids[tokenId].bidders.length());
        for (uint256 i; i < _tokenBids[tokenId].bidders.length(); i++) {
            address bidder = _tokenBids[tokenId].bidders.at(i);
            Bid memory bid = _tokenBids[tokenId].bids[bidder];
            if (_isBidValid(bid)) {
                bids[i] = bid;
            }
        }
        return bids;
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-getTokenHighestBid}.
     */
    function getTokenHighestBid(uint256 tokenId) public view returns (Bid memory) {
        Bid memory highestBid = Bid(tokenId, 0, address(0));
        for (uint256 i; i < _tokenBids[tokenId].bidders.length(); i++) {
            address bidder = _tokenBids[tokenId].bidders.at(i);
            Bid memory bid = _tokenBids[tokenId].bids[bidder];
            if (_isBidValid(bid) && bid.bidPrice > highestBid.bidPrice) {
                highestBid = bid;
            }
        }
        return highestBid;
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-getTokenHighestBids}.
     */
    function getTokenHighestBids(uint256 from, uint256 size)
        public
        view
        returns (Bid[] memory)
    {
        if (from < _tokenIdWithBid.length() && size > 0) {
            uint256 querySize = size;
            if ((from + size) > _tokenIdWithBid.length()) {
                querySize = _tokenIdWithBid.length() - from;
            }
            Bid[] memory highestBids = new Bid[](querySize);
            for (uint256 i = 0; i < querySize; i++) {
                highestBids[i] = getTokenHighestBid(_tokenIdWithBid.at(i + from));
            }
            return highestBids;
        }
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-getAllTokenHighestBids}.
     */
    function getAllTokenHighestBids() external view returns (Bid[] memory) {
        return getTokenHighestBids(0, _tokenIdWithBid.length());
    }

    /**
     * @dev remove a bid of a bidder
     * @param tokenId erc721 token Id
     * @param bidder bidder address
     */
    function _removeBidOfBidder(uint256 tokenId, address bidder) private {
        if (_tokenBids[tokenId].bidders.contains(bidder)) {
            // Step 1: delete the bid and the address
            delete _tokenBids[tokenId].bids[bidder];
            _tokenBids[tokenId].bidders.remove(bidder);

            // Step 2: if no bid left
            if (_tokenBids[tokenId].bidders.length() == 0) {
                _tokenIdWithBid.remove(tokenId);
            }
        }
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-enterBidForToken}.
     * People can only enter bid if bid is allowed
     * The timestamp set needs to be in the allowed range
     * bid price > 0
     * must not be token owner
     * must allow this contract to spend enough payment token
     */
    function enterBidForToken(
        address sender,
        uint256 tokenId,
        uint256 bidPrice
    ) external onlyMarketplaceOpen {
        require(bidPrice > 0, "Please bid for more than 0");
        require(!_isTokenOwner(tokenId, sender), "This Token belongs to this address");
      
        Bid memory bid = Bid(tokenId, bidPrice, sender);
        if (!_tokenIdWithBid.contains(tokenId)) {
            _tokenIdWithBid.add(tokenId);
        }
        _tokenBids[tokenId].bidders.add(sender);
        _tokenBids[tokenId].bids[sender] = bid;
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-acceptBidForToken}.
     * Must be owner of this token
     * Must have approved this contract to transfer token
     * Must have a valid existing bid that matches the bidder address
     */
     /**
     * @dev See {INFTKEYMarketPlaceV1-withdrawBidForToken}.
     * There must be a bid exists
     * remove this bid record
     */
    function withdrawBidForToken(address sender, uint256 tokenId) external returns (address, uint256 bidPrice) {
        Bid memory bid = _tokenBids[tokenId].bids[sender];
        require(bid.bidder == sender, "This address doesn't have bid on this token");
        _removeBidOfBidder(tokenId, sender);
        return (bid.bidder, bid.bidPrice);
    }
    function acceptBidForTokenPrepare(address sender, uint256 tokenId, address bidder) external  onlyDev returns (
        address, 
        address,
        address,
        uint256
    ) {
        require(_isTokenOwner(tokenId, sender), "Only token owner can accept bid of token");
        require(
            _isTokenApproved(tokenId) || _isAllTokenApproved(sender),
            "The token is not approved to transfer by the contract"
        );

        Bid memory existingBid = getBidderTokenBid(tokenId, bidder);
        require(
            existingBid.bidPrice > 0 && existingBid.bidder == bidder,
            "This token doesn't have a matching bid"
        );
        return (
            maketPlaceFeeAddress,
            nftCreaterAddress,
            nftProducerAddress,
            existingBid.bidPrice
        );
    }
    function acceptBidForTokenComplete(address sender, uint256 tokenId, address bidder) external  onlyDev{
        nftTransferFrom(sender, bidder, tokenId);
        // Remove token listing
        _delistToken(tokenId);
        _removeBidOfBidder(tokenId, bidder);
    }

    /**
     * @dev See {INFTKEYMarketPlaceV1-getInvalidListingCount}.
     */
    function getInvalidListingCount() external view returns (uint256) {
        uint256 count = 0;
        for (uint256 i = 0; i < _tokenIdWithListing.length(); i++) {
            if (!_isListingValid(_tokenListings[_tokenIdWithListing.at(i)])) {
                count = count.add(1);
            }
        }
        return count;
    }

    /**
     * @dev Count how many bid records of a token are invalid now
     */
    function _getInvalidBidOfTokenCount(uint256 tokenId) private view returns (uint256) {
        uint256 count = 0;
        for (uint256 i = 0; i < _tokenBids[tokenId].bidders.length(); i++) {
            address bidder = _tokenBids[tokenId].bidders.at(i);
            Bid memory bid = _tokenBids[tokenId].bids[bidder];
            if (!_isBidValid(bid)) {
                count = count.add(1);
            }
        }
        return count;
    }

    /**
     * @dev See {INFTKEYMarketPlaceV1-getInvalidBidCount}.
     */
    function getInvalidBidCount() external view returns (uint256) {
        uint256 count = 0;
        for (uint256 i = 0; i < _tokenIdWithBid.length(); i++) {
            count = count.add(_getInvalidBidOfTokenCount(_tokenIdWithBid.at(i)));
        }
        return count;
    }
    /**
     * @dev See {INFTKEYMarketPlaceV1-cleanAllInvalidListings}.
     */
    function cleanAllInvalidListings() external returns(uint256[] memory){
        for (uint256 i = 0; i < _tokenIdWithListing.length(); i++) {
            uint256 tokenId = _tokenIdWithListing.at(i);
            if (!_isListingValid(_tokenListings[tokenId])) {
                _tempTokenIdStorage.push(tokenId);
            }
        }
        for (uint256 i = 0; i < _tempTokenIdStorage.length; i++) {
            _delistToken(_tempTokenIdStorage[i]);
        }
        
        return (_tempTokenIdStorage);
    }

    function deleteTempTokenIdStorage() external onlyDev {
        delete _tempTokenIdStorage;
    }

    /**
     * @dev remove invalid bids of a token
     * @param tokenId erc721 token Id
     */
    function _cleanInvalidBidsOfToken(uint256 tokenId) private {
        for (uint256 i = 0; i < _tokenBids[tokenId].bidders.length(); i++) {
            address bidder = _tokenBids[tokenId].bidders.at(i);
            Bid memory bid = _tokenBids[tokenId].bids[bidder];
            if (!_isBidValid(bid)) {
                _tempBidderStorage.push(_tokenBids[tokenId].bidders.at(i));
            }
        }
        for (uint256 i = 0; i < _tempBidderStorage.length; i++) {
            address bidder = _tempBidderStorage[i];
            _removeBidOfBidder(tokenId, bidder);
        }
        delete _tempBidderStorage;
    }

    /**
     * @dev See {INFTKEYMarketPlaceV1-cleanAllInvalidBids}.
     */
    function cleanAllInvalidBids() external returns(uint256[] memory){
        for (uint256 i = 0; i < _tokenIdWithBid.length(); i++) {
            uint256 tokenId = _tokenIdWithBid.at(i);
            uint256 invalidCount = _getInvalidBidOfTokenCount(tokenId);
            if (invalidCount > 0) {
                _tempTokenIdStorage.push(tokenId);
            }
        }
        for (uint256 i = 0; i < _tempTokenIdStorage.length; i++) {
            _cleanInvalidBidsOfToken(_tempTokenIdStorage[i]);
        }
        return (_tempTokenIdStorage);
    }

    /**
     * @dev See {INFTKEYMarketPlaceV1-isListingAndBidEnabled}.
     */
    function isListingAndBidEnabled() external view returns (bool) {
        return _isListingAndBidEnabled;
    }
    /**
     * @dev Enable to disable Bids and Listing
     */
    function changeMarketplaceStatus(bool enabled) external onlyDev {
        _isListingAndBidEnabled = enabled;
    }
    
}
          

/contracts/RamdomNFTMarket/CyberBoxMarketPaymentAPI.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol";
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/utils/Address.sol";

contract CyberBoxMarketPaymentAPI {
    using Address for address;

    constructor() public {}

    /**
     * @dev payment from sender to receiver with Custom ERC20 or Celo
     * @param paymentToken payment ERC20
     * @param sender payment from address
     * @param receiver payment to address
     * @param value payment amount
    */
    function sendERC20(
        IERC20 paymentToken,
        address sender,
        address receiver,
        uint256 value
        )  public {
        
        if(isEtherToken(address(paymentToken)) == true){
            transferCelo(sender, receiver, value);
        }else{
            transferERC20(paymentToken, sender, receiver, value);
        }
    }

    /**
     * @dev returns ERC20 is celo token or not
     * @param paymentToken payment token address
    */
    function isEtherToken(address paymentToken) public returns (bool){
        if(paymentToken == address(0x471EcE3750Da237f93B8E339c536989b8978a438) ||
        paymentToken == address(0xF194afDf50B03e69Bd7D057c1Aa9e10c9954E4C9)){
            return true;
        }
        return false;
    }
    /**
     * @dev payment from sender to receiver with Custom ERC20
     * @param paymentToken payment ERC20
     * @param sender payment from address
     * @param receiver payment to address
     * @param value payment amount
    */
    function transferERC20(IERC20 paymentToken, address sender, address receiver, uint256 value) private{
        require(sender != address(0), "transfer address must not 0x0");
        require(receiver != address(0), "transfer address must not 0x0");
        require(value > 0, "transfer amount must large than 0");
        if(sender == address(this)){
            paymentToken.transfer(receiver, value);
        }else{
            paymentToken.transferFrom(sender, receiver, value);
        }
    }
    /**
     * @dev payment from sender to receiver with Celo
     * @param sender payment from address
     * @param receiver payment to address
     * @param value payment amount
    */
    function transferCelo(address sender, address receiver, uint256 value)  private{
        require(sender != address(0), "transfer address must not 0x0");
        require(receiver != address(0), "transfer address must not 0x0");
        require(value > 0, "transfer amount must large than 0");
        if(sender == address(this)){
            payable(receiver).transfer(value);
        }else{
            Address.sendValue(payable(receiver), value);
        }
    }
}
          

/contracts/RamdomNFTMarket/RandomCyberBoxMarketManager.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/utils/math/SafeMath.sol";
import "hardhat/console.sol";

import "./RandomNFTMarketInterface.sol";
import "./RandomNFTMarketAPI.sol";
import "./CyberBoxMarketPaymentAPI.sol";

contract RandomCyberBoxMarketManager is Ownable, RandomNFTMarketInterface, RandomNFTMarketAPI, CyberBoxMarketPaymentAPI {
    constructor() public {}

    function setMaketPlaceAddressAndDevFee(
        address _nftAddress,
        address _maketPlaceFeeAddress, 
        uint256 _maketPlaceFeePercentage)
        external
        onlyDev
    {

        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        getMarketPlaceToken().setMaketPlaceAddressAndDevFee(
            _nftAddress,
            _maketPlaceFeeAddress,
            _maketPlaceFeePercentage
        );
        emit RandomMarketDevFeeChanged(
            _nftAddress,
            _maketPlaceFeeAddress,
            _maketPlaceFeePercentage
        );
    }
    function setTokenCreaterAddress(
        address _nftAddress,
        address _tokenCreaterAddress)
        external
        onlyDev
    {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        getMarketPlaceToken().setTokenCreaterAddress(
            _nftAddress,
            _tokenCreaterAddress
        );
        emit RandomMarketTokenCreaterChanged(
            _nftAddress,
            _tokenCreaterAddress
        );
    }
    function setTokenProducerAddress(
        address _nftAddress,
        address _tokenProducerAddress)
        external
        onlyDev
    {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        getMarketPlaceToken().setTokenProducerAddress(
            _nftAddress,
            _tokenProducerAddress
        );
        emit RandomMarketTokenProducerChanged(
            _nftAddress,
            _tokenProducerAddress
        );
    }
    function getServiceFee(address _nftAddress) external 
    returns (uint256, uint256, uint256, uint256, uint256, uint256) 
    {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        return getMarketPlaceToken().serviceFee(_nftAddress);
    }
    function setNFTFees(
        address _nftAddress,
        uint256 _feeCreater,
        uint256 _feeProducer
    )
    external
    onlyDev
    {
        NFTToken memory nftToken =  getNFTToken(_nftAddress);
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        selectNFT(_nftAddress);
        getMarketPlaceToken().setNFTFees(
            _nftAddress,
            _feeCreater,
            _feeProducer
        );
        emit RandomMarketFeeChanged(
            _nftAddress,
            _feeCreater,
            _feeProducer
        );
    }

}
          

/contracts/RamdomNFTMarket/RandomNFTMarketAPI.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol";
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/utils/Address.sol";
import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol";
import "@openzeppelin/contracts/utils/math/SafeMath.sol";

import {ClonesUpgradeable} from "@openzeppelin/contracts-upgradeable/proxy/ClonesUpgradeable.sol";
import {CountersUpgradeable} from "@openzeppelin/contracts-upgradeable/utils/CountersUpgradeable.sol";

import "../MarketPlace/RandomMarketPlaceV2.sol";
import "./RandomNFTMarketInterface.sol";

contract RandomNFTMarketAPI is RandomNFTMarketInterface {

    using SafeMath for uint256;
    using EnumerableSet for EnumerableSet.UintSet;
    using EnumerableSet for EnumerableSet.AddressSet;
    using CountersUpgradeable for CountersUpgradeable.Counter;
    CountersUpgradeable.Counter private atContract;

    address public implementation;
    address public ownerAddress; // developer address
    address public devAddress; // developer address

    struct Token {
        string tokenName;
        string tokenSymbol;
        address tokenAddress;
    }

    struct NFTToken {
        Token nftToken;
        Token paymentToken;
        address marketPlaceAddress;
        bool is721;
    }

    mapping(address => NFTToken) private _nftManager;
    address private _selectedNftAddress;
    IERC721 private _erc721;
    address private _selectedPayTokenAddress;
    IERC20 private _paymentToken;
    RandomMarketPlaceV2 private _selectedMarketPlaceToken;
    address[] public supportNFTs;

    constructor() public {}

    modifier onlyDev() {
        require(msg.sender == devAddress, "auction: wrong developer");
        _;
    }

    function changeDev(address _newDev) public onlyDev {
        devAddress  = _newDev;
    }
    function getNFTToken(address _nftAddress) public view returns (NFTToken memory){
        return _nftManager[_nftAddress];
    }

    function setNFTToken(address _nftAddress, NFTToken memory nftToken) public {
        _nftManager[_nftAddress] = nftToken;
    }

    function getMarketPlaceToken() public view returns (RandomMarketPlaceV2){
        return _selectedMarketPlaceToken;
    }
    function getSelectedNFTAddress() public view returns (address){
        return _selectedNftAddress;
    }
    function getSelectedERC20Token() public view returns (IERC20){
        return _paymentToken;
    }
    function getSelectedERC20Address() public view returns (address){
        return _selectedPayTokenAddress;
    }
    function getSupportNFTToken(
        address _nftAddress
    ) external view returns (Token memory) {
        NFTToken memory nftToken =  _nftManager[_nftAddress];
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        return nftToken.nftToken;
    }
    function getSupportPaymentToken(
        address _nftAddress
    ) external view returns (Token memory) {
        NFTToken memory nftToken =  _nftManager[_nftAddress];
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        return nftToken.paymentToken;
    }
    function getSupportMarketPlaceToken(
        address _nftAddress
    ) external view returns (address) {
        NFTToken memory nftToken =  _nftManager[_nftAddress];
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        return nftToken.marketPlaceAddress;
    }
    function setNewMarketPlaceAddress(
        address _newAddress
    ) public onlyDev {
        implementation  = _newAddress;
    }

    function addERC721Token(
        string memory _nftName, 
        string memory _nftSymbol, 
        address _nftAddress,
        string memory _erc20Name, 
        string memory _erc20Symbol, 
        address _erc20Address
    ) public {
        
        if(_nftManager[_nftAddress].marketPlaceAddress == address(0)){
            supportNFTs.push(_nftAddress);
        }
        _nftManager[_nftAddress].nftToken = Token(_nftName, _nftSymbol, _nftAddress);
        _nftManager[_nftAddress].paymentToken = Token(_erc20Name, _erc20Symbol, _erc20Address);
        _nftManager[_nftAddress].is721 = true;
        createNewMarketPlaceToken(_nftName, _nftAddress, _erc20Address, devAddress, true);
        NFTToken memory nftToken =  _nftManager[_nftAddress];
        selectNFT(_nftAddress);
        emit RandomMarketTokenAdded(
                _nftName,
                _nftSymbol,
                _nftAddress,
                _erc20Name,
                _erc20Symbol,
                _erc20Address,
                nftToken.marketPlaceAddress
        );
    }
    function selectNFT(
        address _nftAddress
    ) public {
        NFTToken memory nftToken =  _nftManager[_nftAddress];
        require(
            nftToken.marketPlaceAddress != address(0), "This token still not registed."
        );
        Token memory nft = nftToken.nftToken;
        if(nftToken.is721){
            selectNFT721Token(nft.tokenAddress);
        }else{
        }
        Token memory erc20 = nftToken.paymentToken;
        selectPaymentToken(erc20.tokenAddress);
        address marketToken = nftToken.marketPlaceAddress;
        _selectedMarketPlaceToken = RandomMarketPlaceV2(marketToken);
    }
    function selectNFT721Token(
        address _nftAddress
    ) private {
        if(_nftAddress != _selectedNftAddress){
            _erc721 = IERC721(_nftAddress);
            _selectedNftAddress = _nftAddress;
        }
    }
    function selectPaymentToken(
        address _erc20Address
    ) private {
        if(_erc20Address != _selectedPayTokenAddress){
            _paymentToken = IERC20(_erc20Address);
            _selectedPayTokenAddress = _erc20Address;
        }
    }
    function createNewMarketPlaceToken(
        string memory _nftName,
        address _nftAddress,
        address _erc20Address,
        address _owner,
        bool isSupportERC721
        ) private {
        
        uint256 newId = atContract.current();
        address newContract = ClonesUpgradeable.cloneDeterministic(
            implementation,
            toBytes(newId)
        );
        _nftManager[_nftAddress].marketPlaceAddress = newContract;
        if(isSupportERC721 == true){
            RandomMarketPlaceV2(newContract).initializeWithERC721(
                _nftName,
                _nftAddress,
                _erc20Address,
                _owner
            );
        }else{
            RandomMarketPlaceV2(newContract).initializeWithERC1155(
                _nftName,
                _nftAddress,
                _erc20Address,
                _owner
            );
        }
        atContract.increment();
    }

    function toBytes(uint256 x)
        private
        view 
        returns (bytes32 b) {
		return bytes32(keccak256(abi.encodePacked(x)));
	}
}
          

/contracts/RamdomNFTMarket/RandomNFTMarketInterface.sol

// SPDX-License-Identifier: AGPL-3.0-or-later

pragma solidity 0.8.3;
pragma experimental ABIEncoderV2;

interface RandomNFTMarketInterface {
    event RandomMarketTokenAdded(
        string nftName, 
        string nftSymbol,  
        address nftAddress, 
        string erc20Name,
        string erc20Symbol,
        address erc20Address,
        address marketPlaceAddress
    );
    event RandomMarketDevFeeChanged(
        address nftAddress,
        address marketPlaceFeeAddress,
        uint256 marketFee
    );
    event RandomMarketTokenCreaterChanged(
        address nftAddress,
        address tokenCreaterAddress
    );
    event RandomMarketTokenProducerChanged(
        address nftAddress,
        address tokenCreaterAddress
    );
    event RandomMarketFeeChanged(
        address nftAddress,
        uint256 createrFee,
        uint256 producerFee
    );

    event RandomMarketTokenListed(
        address nftAddress,
        uint256 indexed tokenId, 
        address indexed fromAddress, 
        uint256 minValue
    );

    event RandomMarketTokenDelisted(
        address nftAddress,
        uint256 indexed tokenId
    );

    event RandomMarketCleanList(
        address nftAddress,
        uint256 tokenId
    );

    event RandomMarketPayment(
        address fromAddress,
        address toAddress,
        uint256 tokenId,
        address nftAddress,
        uint256 paymentType,
        uint256 amount
    );

    event RandomMarketTokenBought(
        address nftAddress,
        uint256 tokenId,
        address fromAddress, 
        address toAddress,
        uint256 total,
        uint256 value,
        uint256 fees
    );

    event RandomMarketTokenTransfered(
        address nftAddress,
        uint256 indexed tokenId,
        address indexed from,
        address indexed to,
        uint256 total
    );

    event RandomMarketTokenPriceChanged(
        address nftAddress,
        uint256 indexed tokenId,
        uint256 price
    );
}
          

/hardhat/console.sol

// SPDX-License-Identifier: MIT
pragma solidity >= 0.4.22 <0.9.0;

library console {
	address constant CONSOLE_ADDRESS = address(0x000000000000000000636F6e736F6c652e6c6f67);

	function _sendLogPayload(bytes memory payload) private view {
		uint256 payloadLength = payload.length;
		address consoleAddress = CONSOLE_ADDRESS;
		assembly {
			let payloadStart := add(payload, 32)
			let r := staticcall(gas(), consoleAddress, payloadStart, payloadLength, 0, 0)
		}
	}

	function log() internal view {
		_sendLogPayload(abi.encodeWithSignature("log()"));
	}

	function logInt(int256 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(int256)", p0));
	}

	function logUint(uint256 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256)", p0));
	}

	function logString(string memory p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string)", p0));
	}

	function logBool(bool p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool)", p0));
	}

	function logAddress(address p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address)", p0));
	}

	function logBytes(bytes memory p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes)", p0));
	}

	function logBytes1(bytes1 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes1)", p0));
	}

	function logBytes2(bytes2 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes2)", p0));
	}

	function logBytes3(bytes3 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes3)", p0));
	}

	function logBytes4(bytes4 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes4)", p0));
	}

	function logBytes5(bytes5 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes5)", p0));
	}

	function logBytes6(bytes6 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes6)", p0));
	}

	function logBytes7(bytes7 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes7)", p0));
	}

	function logBytes8(bytes8 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes8)", p0));
	}

	function logBytes9(bytes9 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes9)", p0));
	}

	function logBytes10(bytes10 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes10)", p0));
	}

	function logBytes11(bytes11 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes11)", p0));
	}

	function logBytes12(bytes12 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes12)", p0));
	}

	function logBytes13(bytes13 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes13)", p0));
	}

	function logBytes14(bytes14 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes14)", p0));
	}

	function logBytes15(bytes15 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes15)", p0));
	}

	function logBytes16(bytes16 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes16)", p0));
	}

	function logBytes17(bytes17 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes17)", p0));
	}

	function logBytes18(bytes18 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes18)", p0));
	}

	function logBytes19(bytes19 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes19)", p0));
	}

	function logBytes20(bytes20 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes20)", p0));
	}

	function logBytes21(bytes21 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes21)", p0));
	}

	function logBytes22(bytes22 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes22)", p0));
	}

	function logBytes23(bytes23 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes23)", p0));
	}

	function logBytes24(bytes24 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes24)", p0));
	}

	function logBytes25(bytes25 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes25)", p0));
	}

	function logBytes26(bytes26 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes26)", p0));
	}

	function logBytes27(bytes27 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes27)", p0));
	}

	function logBytes28(bytes28 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes28)", p0));
	}

	function logBytes29(bytes29 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes29)", p0));
	}

	function logBytes30(bytes30 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes30)", p0));
	}

	function logBytes31(bytes31 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes31)", p0));
	}

	function logBytes32(bytes32 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bytes32)", p0));
	}

	function log(uint256 p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256)", p0));
	}

	function log(string memory p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string)", p0));
	}

	function log(bool p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool)", p0));
	}

	function log(address p0) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address)", p0));
	}

	function log(uint256 p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256)", p0, p1));
	}

	function log(uint256 p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string)", p0, p1));
	}

	function log(uint256 p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool)", p0, p1));
	}

	function log(uint256 p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address)", p0, p1));
	}

	function log(string memory p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256)", p0, p1));
	}

	function log(string memory p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string)", p0, p1));
	}

	function log(string memory p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool)", p0, p1));
	}

	function log(string memory p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address)", p0, p1));
	}

	function log(bool p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256)", p0, p1));
	}

	function log(bool p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string)", p0, p1));
	}

	function log(bool p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool)", p0, p1));
	}

	function log(bool p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address)", p0, p1));
	}

	function log(address p0, uint256 p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256)", p0, p1));
	}

	function log(address p0, string memory p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string)", p0, p1));
	}

	function log(address p0, bool p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool)", p0, p1));
	}

	function log(address p0, address p1) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address)", p0, p1));
	}

	function log(uint256 p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool)", p0, p1, p2));
	}

	function log(uint256 p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool)", p0, p1, p2));
	}

	function log(uint256 p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool)", p0, p1, p2));
	}

	function log(uint256 p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool)", p0, p1, p2));
	}

	function log(string memory p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool)", p0, p1, p2));
	}

	function log(string memory p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool)", p0, p1, p2));
	}

	function log(string memory p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address)", p0, p1, p2));
	}

	function log(string memory p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256)", p0, p1, p2));
	}

	function log(string memory p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string)", p0, p1, p2));
	}

	function log(string memory p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool)", p0, p1, p2));
	}

	function log(string memory p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool)", p0, p1, p2));
	}

	function log(bool p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool)", p0, p1, p2));
	}

	function log(bool p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address)", p0, p1, p2));
	}

	function log(bool p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256)", p0, p1, p2));
	}

	function log(bool p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string)", p0, p1, p2));
	}

	function log(bool p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool)", p0, p1, p2));
	}

	function log(bool p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address)", p0, p1, p2));
	}

	function log(bool p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256)", p0, p1, p2));
	}

	function log(bool p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string)", p0, p1, p2));
	}

	function log(bool p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool)", p0, p1, p2));
	}

	function log(bool p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool)", p0, p1, p2));
	}

	function log(address p0, uint256 p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address)", p0, p1, p2));
	}

	function log(address p0, string memory p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256)", p0, p1, p2));
	}

	function log(address p0, string memory p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string)", p0, p1, p2));
	}

	function log(address p0, string memory p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool)", p0, p1, p2));
	}

	function log(address p0, string memory p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address)", p0, p1, p2));
	}

	function log(address p0, bool p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256)", p0, p1, p2));
	}

	function log(address p0, bool p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string)", p0, p1, p2));
	}

	function log(address p0, bool p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool)", p0, p1, p2));
	}

	function log(address p0, bool p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address)", p0, p1, p2));
	}

	function log(address p0, address p1, uint256 p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256)", p0, p1, p2));
	}

	function log(address p0, address p1, string memory p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string)", p0, p1, p2));
	}

	function log(address p0, address p1, bool p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool)", p0, p1, p2));
	}

	function log(address p0, address p1, address p2) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address)", p0, p1, p2));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,string,address,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,bool,address,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,string,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,bool,address)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,string)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,bool)", p0, p1, p2, p3));
	}

	function log(uint256 p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(uint256,address,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,string,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,bool,address,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,string,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,bool,address)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,string)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,bool)", p0, p1, p2, p3));
	}

	function log(string memory p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(string,address,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,string,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,bool,address,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,string,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,bool,address)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,string)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,bool)", p0, p1, p2, p3));
	}

	function log(bool p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(bool,address,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, uint256 p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,uint256,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, string memory p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,string,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, bool p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,bool,address,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, uint256 p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,uint256,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, string memory p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,string,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, bool p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,bool,address)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, uint256 p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,uint256)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, string memory p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,string)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, bool p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,bool)", p0, p1, p2, p3));
	}

	function log(address p0, address p1, address p2, address p3) internal view {
		_sendLogPayload(abi.encodeWithSignature("log(address,address,address,address)", p0, p1, p2, p3));
	}

}
          

Contract ABI

[{"type":"constructor","stateMutability":"nonpayable","inputs":[{"type":"address","name":"_ownerAddress","internalType":"address"},{"type":"address","name":"_devAddress","internalType":"address"},{"type":"address","name":"_maketPlaceAddress","internalType":"address"}]},{"type":"event","name":"OwnershipTransferred","inputs":[{"type":"address","name":"previousOwner","internalType":"address","indexed":true},{"type":"address","name":"newOwner","internalType":"address","indexed":true}],"anonymous":false},{"type":"event","name":"RandomMarketCleanList","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"tokenId","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketDevFeeChanged","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"address","name":"marketPlaceFeeAddress","internalType":"address","indexed":false},{"type":"uint256","name":"marketFee","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketFeeChanged","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"createrFee","internalType":"uint256","indexed":false},{"type":"uint256","name":"producerFee","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketPayment","inputs":[{"type":"address","name":"fromAddress","internalType":"address","indexed":false},{"type":"address","name":"toAddress","internalType":"address","indexed":false},{"type":"uint256","name":"tokenId","internalType":"uint256","indexed":false},{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"paymentType","internalType":"uint256","indexed":false},{"type":"uint256","name":"amount","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketTokenAdded","inputs":[{"type":"string","name":"nftName","internalType":"string","indexed":false},{"type":"string","name":"nftSymbol","internalType":"string","indexed":false},{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"string","name":"erc20Name","internalType":"string","indexed":false},{"type":"string","name":"erc20Symbol","internalType":"string","indexed":false},{"type":"address","name":"erc20Address","internalType":"address","indexed":false},{"type":"address","name":"marketPlaceAddress","internalType":"address","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketTokenBought","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"tokenId","internalType":"uint256","indexed":false},{"type":"address","name":"fromAddress","internalType":"address","indexed":false},{"type":"address","name":"toAddress","internalType":"address","indexed":false},{"type":"uint256","name":"total","internalType":"uint256","indexed":false},{"type":"uint256","name":"value","internalType":"uint256","indexed":false},{"type":"uint256","name":"fees","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketTokenCreaterChanged","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"address","name":"tokenCreaterAddress","internalType":"address","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketTokenDelisted","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"tokenId","internalType":"uint256","indexed":true}],"anonymous":false},{"type":"event","name":"RandomMarketTokenListed","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"tokenId","internalType":"uint256","indexed":true},{"type":"address","name":"fromAddress","internalType":"address","indexed":true},{"type":"uint256","name":"minValue","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketTokenPriceChanged","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"tokenId","internalType":"uint256","indexed":true},{"type":"uint256","name":"price","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketTokenProducerChanged","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"address","name":"tokenCreaterAddress","internalType":"address","indexed":false}],"anonymous":false},{"type":"event","name":"RandomMarketTokenTransfered","inputs":[{"type":"address","name":"nftAddress","internalType":"address","indexed":false},{"type":"uint256","name":"tokenId","internalType":"uint256","indexed":true},{"type":"address","name":"from","internalType":"address","indexed":true},{"type":"address","name":"to","internalType":"address","indexed":true},{"type":"uint256","name":"total","internalType":"uint256","indexed":false}],"anonymous":false},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"addERC721Token","inputs":[{"type":"string","name":"_nftName","internalType":"string"},{"type":"string","name":"_nftSymbol","internalType":"string"},{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"string","name":"_erc20Name","internalType":"string"},{"type":"string","name":"_erc20Symbol","internalType":"string"},{"type":"address","name":"_erc20Address","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"addNewRandomeNFT","inputs":[{"type":"string","name":"_nftName","internalType":"string"},{"type":"string","name":"_nftSymbol","internalType":"string"},{"type":"address","name":"nftAddress","internalType":"address"}]},{"type":"function","stateMutability":"payable","outputs":[],"name":"buyToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"uint256","name":"tokenId","internalType":"uint256"},{"type":"uint256","name":"value","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"changeDev","inputs":[{"type":"address","name":"_newDev","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"changePrice","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"uint256","name":"tokenId","internalType":"uint256"},{"type":"uint256","name":"newPrice","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"cleanAllInvalidListings","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"delistToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"uint256","name":"tokenId","internalType":"uint256"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"devAddress","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple[]","name":"","internalType":"struct InterfaceV2.Listing[]","components":[{"type":"uint256","name":"tokenId","internalType":"uint256"},{"type":"uint256","name":"listingPrice","internalType":"uint256"},{"type":"address","name":"seller","internalType":"address"}]}],"name":"getAllTokenListings","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"contract RandomMarketPlaceV2"}],"name":"getMarketPlaceToken","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct RandomNFTMarketAPI.NFTToken","components":[{"type":"tuple","name":"nftToken","internalType":"struct RandomNFTMarketAPI.Token","components":[{"type":"string","name":"tokenName","internalType":"string"},{"type":"string","name":"tokenSymbol","internalType":"string"},{"type":"address","name":"tokenAddress","internalType":"address"}]},{"type":"tuple","name":"paymentToken","internalType":"struct RandomNFTMarketAPI.Token","components":[{"type":"string","name":"tokenName","internalType":"string"},{"type":"string","name":"tokenSymbol","internalType":"string"},{"type":"address","name":"tokenAddress","internalType":"address"}]},{"type":"address","name":"marketPlaceAddress","internalType":"address"},{"type":"bool","name":"is721","internalType":"bool"}]}],"name":"getNFTToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"getSelectedERC20Address","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"contract IERC20"}],"name":"getSelectedERC20Token","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"getSelectedNFTAddress","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[{"type":"uint256","name":"","internalType":"uint256"},{"type":"uint256","name":"","internalType":"uint256"},{"type":"uint256","name":"","internalType":"uint256"},{"type":"uint256","name":"","internalType":"uint256"},{"type":"uint256","name":"","internalType":"uint256"},{"type":"uint256","name":"","internalType":"uint256"}],"name":"getServiceFee","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"getSupportMarketPlaceToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct RandomNFTMarketAPI.Token","components":[{"type":"string","name":"tokenName","internalType":"string"},{"type":"string","name":"tokenSymbol","internalType":"string"},{"type":"address","name":"tokenAddress","internalType":"address"}]}],"name":"getSupportNFTToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct RandomNFTMarketAPI.Token","components":[{"type":"string","name":"tokenName","internalType":"string"},{"type":"string","name":"tokenSymbol","internalType":"string"},{"type":"address","name":"tokenAddress","internalType":"address"}]}],"name":"getSupportPaymentToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"tuple","name":"","internalType":"struct InterfaceV2.Listing","components":[{"type":"uint256","name":"tokenId","internalType":"uint256"},{"type":"uint256","name":"listingPrice","internalType":"uint256"},{"type":"address","name":"seller","internalType":"address"}]}],"name":"getTokenListing","inputs":[{"type":"uint256","name":"tokenId","internalType":"uint256"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"implementation","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[{"type":"bool","name":"","internalType":"bool"}],"name":"isEtherToken","inputs":[{"type":"address","name":"paymentToken","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"listToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"uint256","name":"tokenId","internalType":"uint256"},{"type":"uint256","name":"value","internalType":"uint256"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"owner","inputs":[]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"ownerAddress","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"renounceOwnership","inputs":[]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"selectNFT","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"sendERC20","inputs":[{"type":"address","name":"paymentToken","internalType":"contract IERC20"},{"type":"address","name":"sender","internalType":"address"},{"type":"address","name":"receiver","internalType":"address"},{"type":"uint256","name":"value","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setMaketPlaceAddressAndDevFee","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"address","name":"_maketPlaceFeeAddress","internalType":"address"},{"type":"uint256","name":"_maketPlaceFeePercentage","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setNFTFees","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"uint256","name":"_feeCreater","internalType":"uint256"},{"type":"uint256","name":"_feeProducer","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setNFTToken","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"tuple","name":"nftToken","internalType":"struct RandomNFTMarketAPI.NFTToken","components":[{"type":"tuple","name":"nftToken","internalType":"struct RandomNFTMarketAPI.Token","components":[{"type":"string","name":"tokenName","internalType":"string"},{"type":"string","name":"tokenSymbol","internalType":"string"},{"type":"address","name":"tokenAddress","internalType":"address"}]},{"type":"tuple","name":"paymentToken","internalType":"struct RandomNFTMarketAPI.Token","components":[{"type":"string","name":"tokenName","internalType":"string"},{"type":"string","name":"tokenSymbol","internalType":"string"},{"type":"address","name":"tokenAddress","internalType":"address"}]},{"type":"address","name":"marketPlaceAddress","internalType":"address"},{"type":"bool","name":"is721","internalType":"bool"}]}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setNewMarketPlaceAddress","inputs":[{"type":"address","name":"_newAddress","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setTokenCreaterAddress","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"address","name":"_tokenCreaterAddress","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"setTokenProducerAddress","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"address","name":"_tokenProducerAddress","internalType":"address"}]},{"type":"function","stateMutability":"view","outputs":[{"type":"address","name":"","internalType":"address"}],"name":"supportNFTs","inputs":[{"type":"uint256","name":"","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"transfer","inputs":[{"type":"address","name":"_nftAddress","internalType":"address"},{"type":"address","name":"to","internalType":"address"},{"type":"uint256","name":"tokenId","internalType":"uint256"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"transferOwnership","inputs":[{"type":"address","name":"newOwner","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"withdrawCelo","inputs":[{"type":"address","name":"receiver","internalType":"address"}]},{"type":"function","stateMutability":"nonpayable","outputs":[],"name":"withdrawERC20","inputs":[{"type":"address","name":"address20","internalType":"address"},{"type":"address","name":"receiver","internalType":"address"}]}]
              

Contract Creation Code

0x608060405273471ece3750da237f93b8e339c536989b8978a438600c60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555073f194afdf50b03e69bd7d057c1aa9e10c9954e4c9600d60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550348015620000bb57600080fd5b5060405162007495380380620074958339818101604052810190620000e19190620002b0565b62000101620000f5620001cd60201b60201c565b620001d560201b60201c565b82600360006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555081600460006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555080600260006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050505062000354565b600033905090565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050816000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600081519050620002aa816200033a565b92915050565b600080600060608486031215620002c657600080fd5b6000620002d68682870162000299565b9350506020620002e98682870162000299565b9250506040620002fc8682870162000299565b9150509250925092565b600062000313826200031a565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b620003458162000306565b81146200035157600080fd5b50565b61713180620003646000396000f3fe6080604052600436106102255760003560e01c806388a8c95c11610123578063b4ed5546116100ab578063e15487021161006f578063e154870214610804578063e671499b1461082f578063e8dbfd5a1461084b578063f2fde38b14610874578063feb884061461089d57610225565b8063b4ed55461461070f578063b99c6fa714610738578063beabacc814610775578063d1715aed1461079e578063e0f04eaf146107db57610225565b806391ad9428116100f257806391ad9428146106065780639456fbcc1461064357806398792eec1461066c578063a77ad2fd146106a9578063ade56423146106e657610225565b806388a8c95c1461055e5780638da5cb5b146105875780638ebf66ff146105b25780638f84aa09146105db57610225565b806369c33a07116101b157806379a35b4b1161017557806379a35b4b1461048f57806379c0a3ea146104b85780637de7589d146104e15780637e07590d1461050c5780638264045a1461053557610225565b806369c33a07146103be5780636d1d7a97146103fb578063715018a614610426578063741cbae41461043d57806374cf93cd1461046657610225565b80633ad10ef6116101f85780633ad10ef6146102d057806348a806b4146102fb5780634b4703ec146103265780635c60da1b14610368578063696db72c1461039357610225565b80631f4f82eb1461022a578063264e5d5a1461025357806334e0955214610290578063383fba25146102b9575b600080fd5b34801561023657600080fd5b50610251600480360381019061024c9190615d52565b6108c6565b005b34801561025f57600080fd5b5061027a60048036038101906102759190615993565b61106e565b60405161028791906169fd565b60405180910390f35b34801561029c57600080fd5b506102b760048036038101906102b29190615a5b565b6114d2565b005b3480156102c557600080fd5b506102ce6116a3565b005b3480156102dc57600080fd5b506102e5611922565b6040516102f2919061643f565b60405180910390f35b34801561030757600080fd5b50610310611948565b60405161031d919061678e565b60405180910390f35b34801561033257600080fd5b5061034d60048036038101906103489190615993565b611972565b60405161035f96959493929190616a3a565b60405180910390f35b34801561037457600080fd5b5061037d611aab565b60405161038a919061643f565b60405180910390f35b34801561039f57600080fd5b506103a8611ad1565b6040516103b5919061643f565b60405180910390f35b3480156103ca57600080fd5b506103e560048036038101906103e09190615993565b611afb565b6040516103f291906169fd565b60405180910390f35b34801561040757600080fd5b50610410611f5f565b60405161041d919061643f565b60405180910390f35b34801561043257600080fd5b5061043b611f89565b005b34801561044957600080fd5b50610464600480360381019061045f9190615b76565b611f9d565b005b34801561047257600080fd5b5061048d60048036038101906104889190615cd3565b6120dd565b005b34801561049b57600080fd5b506104b660048036038101906104b19190615c70565b61217c565b005b3480156104c457600080fd5b506104df60048036038101906104da9190615993565b6121b4565b005b3480156104ed57600080fd5b506104f6612347565b6040516105039190616773565b60405180910390f35b34801561051857600080fd5b50610533600480360381019061052e9190615b76565b612371565b005b34801561054157600080fd5b5061055c60048036038101906105579190615b76565b6124f8565b005b34801561056a57600080fd5b5061058560048036038101906105809190615993565b6126c9565b005b34801561059357600080fd5b5061059c61279d565b6040516105a9919061643f565b60405180910390f35b3480156105be57600080fd5b506105d960048036038101906105d49190615aaa565b6127c6565b005b3480156105e757600080fd5b506105f061298c565b6040516105fd919061643f565b60405180910390f35b34801561061257600080fd5b5061062d60048036038101906106289190615993565b6129b2565b60405161063a9190616758565b60405180910390f35b34801561064f57600080fd5b5061066a600480360381019061066591906159bc565b612a59565b005b34801561067857600080fd5b50610693600480360381019061068e9190615e64565b612cc2565b6040516106a091906169c0565b60405180910390f35b3480156106b557600080fd5b506106d060048036038101906106cb9190615993565b612d61565b6040516106dd919061643f565b60405180910390f35b3480156106f257600080fd5b5061070d60048036038101906107089190615993565b6131be565b005b34801561071b57600080fd5b5061073660048036038101906107319190615993565b61369a565b005b34801561074457600080fd5b5061075f600480360381019061075a9190615e64565b61376e565b60405161076c919061643f565b60405180910390f35b34801561078157600080fd5b5061079c60048036038101906107979190615a5b565b6137ad565b005b3480156107aa57600080fd5b506107c560048036038101906107c09190615993565b61391c565b6040516107d291906169db565b60405180910390f35b3480156107e757600080fd5b5061080260048036038101906107fd91906159bc565b613d02565b005b34801561081057600080fd5b50610819613ece565b6040516108269190616736565b60405180910390f35b61084960048036038101906108449190615b76565b613f5f565b005b34801561085757600080fd5b50610872600480360381019061086d91906159bc565b6145c6565b005b34801561088057600080fd5b5061089b60048036038101906108969190615993565b614792565b005b3480156108a957600080fd5b506108c460048036038101906108bf9190615afe565b614816565b005b600073ffffffffffffffffffffffffffffffffffffffff16600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060060160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156109c157600b849080600181540180825580915050600190039060005260206000200160009091909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505b60405180606001604052808781526020018681526020018573ffffffffffffffffffffffffffffffffffffffff16815250600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000820151816000019080519060200190610a5092919061547b565b506020820151816001019080519060200190610a6d92919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555090505060405180606001604052808481526020018381526020018273ffffffffffffffffffffffffffffffffffffffff16815250600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206003016000820151816000019080519060200190610b4792919061547b565b506020820151816001019080519060200190610b6492919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055509050506001600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060060160146101000a81548160ff021916908315150217905550610c3a868583600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff166001614959565b6000600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060405180608001604052908160008201604051806060016040529081600082018054610ca690616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610cd290616d3c565b8015610d1f5780601f10610cf457610100808354040283529160200191610d1f565b820191906000526020600020905b815481529060010190602001808311610d0257829003601f168201915b50505050508152602001600182018054610d3890616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610d6490616d3c565b8015610db15780601f10610d8657610100808354040283529160200191610db1565b820191906000526020600020905b815481529060010190602001808311610d9457829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054610e3490616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610e6090616d3c565b8015610ead5780601f10610e8257610100808354040283529160200191610ead565b820191906000526020600020905b815481529060010190602001808311610e9057829003601f168201915b50505050508152602001600182018054610ec690616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610ef290616d3c565b8015610f3f5780601f10610f1457610100808354040283529160200191610f3f565b820191906000526020600020905b815481529060010190602001808311610f2257829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff161515151581525050905061101e856131be565b7fe26e9ba2baee73bb13c229f09b48c470f45b9338eec5b098406161ac0d79e1ac878787878787876040015160405161105d97969594939291906167f5565b60405180910390a150505050505050565b611076615501565b6000600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020604051806080016040529081600082016040518060600160405290816000820180546110e290616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461110e90616d3c565b801561115b5780601f106111305761010080835404028352916020019161115b565b820191906000526020600020905b81548152906001019060200180831161113e57829003601f168201915b5050505050815260200160018201805461117490616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546111a090616d3c565b80156111ed5780601f106111c2576101008083540402835291602001916111ed565b820191906000526020600020905b8154815290600101906020018083116111d057829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016003820160405180606001604052908160008201805461127090616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461129c90616d3c565b80156112e95780601f106112be576101008083540402835291602001916112e9565b820191906000526020600020905b8154815290600101906020018083116112cc57829003601f168201915b5050505050815260200160018201805461130290616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461132e90616d3c565b801561137b5780601f106113505761010080835404028352916020019161137b565b820191906000526020600020905b81548152906001019060200180831161135e57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156114c5576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016114bc90616920565b60405180910390fd5b8060000151915050919050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614611562576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611559906169a0565b60405180910390fd5b600061156d8461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156115e3576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115da90616920565b60405180910390fd5b6115ec846131be565b6115f4611948565b73ffffffffffffffffffffffffffffffffffffffff166334e095528585856040518463ffffffff1660e01b815260040161163093929190616483565b600060405180830381600087803b15801561164a57600080fd5b505af115801561165e573d6000803e3d6000fd5b505050507f917840854c773dda3ec519bb6dd82de7389053094e0b4e8442db88b43681bb7884848460405161169593929190616483565b60405180910390a150505050565b60005b600b8054905081101561191f576000600b82815481106116ef577f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b9060005260206000200160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905060006117278261391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161461190a5761176a826131be565b6000611774611948565b73ffffffffffffffffffffffffffffffffffffffff1663383fba256040518163ffffffff1660e01b8152600401600060405180830381600087803b1580156117bb57600080fd5b505af11580156117cf573d6000803e3d6000fd5b505050506040513d6000823e3d601f19601f820116820180604052508101906117f89190615c06565b905060005b81518110156118a0576000828281518110611841577f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b602002602001015190507fc87975e7f19023dc873bd4977d00e5b4f7bd2f1fb6a670849c147bcf1288853384600001516040015182604051611884929190616667565b60405180910390a150808061189890616d9f565b9150506117fd565b506118a9611948565b73ffffffffffffffffffffffffffffffffffffffff1663bbaed38f6040518163ffffffff1660e01b8152600401600060405180830381600087803b1580156118f057600080fd5b505af1158015611904573d6000803e3d6000fd5b50505050505b5050808061191790616d9f565b9150506116a6565b50565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000600a60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b60008060008060008060006119868861391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156119fc576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016119f390616920565b60405180910390fd5b611a05886131be565b611a0d611948565b73ffffffffffffffffffffffffffffffffffffffff1663606df430896040518263ffffffff1660e01b8152600401611a45919061643f565b60c06040518083038186803b158015611a5d57600080fd5b505afa158015611a71573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a959190615eb6565b9650965096509650965096505091939550919395565b600260009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611b03615501565b6000600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060405180608001604052908160008201604051806060016040529081600082018054611b6f90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611b9b90616d3c565b8015611be85780601f10611bbd57610100808354040283529160200191611be8565b820191906000526020600020905b815481529060010190602001808311611bcb57829003601f168201915b50505050508152602001600182018054611c0190616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611c2d90616d3c565b8015611c7a5780601f10611c4f57610100808354040283529160200191611c7a565b820191906000526020600020905b815481529060010190602001808311611c5d57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054611cfd90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611d2990616d3c565b8015611d765780601f10611d4b57610100808354040283529160200191611d76565b820191906000526020600020905b815481529060010190602001808311611d5957829003601f168201915b50505050508152602001600182018054611d8f90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611dbb90616d3c565b8015611e085780601f10611ddd57610100808354040283529160200191611e08565b820191906000526020600020905b815481529060010190602001808311611deb57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415611f52576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611f4990616920565b60405180910390fd5b8060200151915050919050565b6000600660009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611f91614b28565b611f9b6000614ba6565b565b6000611fa88461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff16141561201e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161201590616920565b60405180910390fd5b612027846131be565b61202f611948565b73ffffffffffffffffffffffffffffffffffffffff1663741cbae43385856040518463ffffffff1660e01b815260040161206b939291906166ff565b600060405180830381600087803b15801561208557600080fd5b505af1158015612099573d6000803e3d6000fd5b50505050827fba2254e53d458d2a10902ac4fca53c8e8f2fb623f5716c5f02948b1f7762b81b85846040516120cf929190616667565b60405180910390a250505050565b6121778383836040518060400160405280600481526020017f63656c6f000000000000000000000000000000000000000000000000000000008152506040518060400160405280600481526020017f63656c6f00000000000000000000000000000000000000000000000000000000815250600c60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff166108c6565b505050565b60011515612189856129b2565b151514156121a15761219c838383614c6a565b6121ae565b6121ad84848484614e1d565b5b50505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612244576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161223b906169a0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614156122b4576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016122ab90616940565b60405180910390fd5b6000479050600081116122fc576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016122f390616960565b60405180910390fd5b8173ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f19350505050158015612342573d6000803e3d6000fd5b505050565b6000600960009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b600061237c8461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156123f2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016123e990616920565b60405180910390fd5b6123fb846131be565b600080612406611948565b73ffffffffffffffffffffffffffffffffffffffff16637e07590d3387876040518463ffffffff1660e01b8152600401612442939291906166ff565b6040805180830381600087803b15801561245b57600080fd5b505af115801561246f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906124939190615b3a565b915091508173ffffffffffffffffffffffffffffffffffffffff16857f2a72dbd9400910f5daeea0d38f90dfff5f64b1674015f8666bb35ab548ecf9d5856000015160400151846040516124e8929190616667565b60405180910390a3505050505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612588576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161257f906169a0565b60405180910390fd5b60006125938461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415612609576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161260090616920565b60405180910390fd5b612612846131be565b61261a611948565b73ffffffffffffffffffffffffffffffffffffffff16638264045a8585856040518463ffffffff1660e01b8152600401612656939291906166ff565b600060405180830381600087803b15801561267057600080fd5b505af1158015612684573d6000803e3d6000fd5b505050507fc4cf474fdae931e45f789cbd99027d46dad91d17ed83c00fbc05ba44c23b85948484846040516126bb939291906166ff565b60405180910390a150505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612759576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612750906169a0565b60405180910390fd5b80600460006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b80600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082015181600001600082015181600001908051906020019061282b92919061547b565b50602082015181600101908051906020019061284892919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505060208201518160030160008201518160000190805190602001906128b792919061547b565b5060208201518160010190805190602001906128d492919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505060408201518160060160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060608201518160060160146101000a81548160ff0219169083151502179055509050505050565b600360009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b600073471ece3750da237f93b8e339c536989b8978a43873ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161480612a41575073f194afdf50b03e69bd7d057c1aa9e10c9954e4c973ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16145b15612a4f5760019050612a54565b600090505b919050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612ae9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612ae0906169a0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161415612b59576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612b5090616940565b60405180910390fd5b600082905060008173ffffffffffffffffffffffffffffffffffffffff166370a08231306040518263ffffffff1660e01b8152600401612b99919061643f565b60206040518083038186803b158015612bb157600080fd5b505afa158015612bc5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612be99190615e8d565b905060008111612c2e576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612c2590616960565b60405180910390fd5b8173ffffffffffffffffffffffffffffffffffffffff1663a9059cbb84836040518363ffffffff1660e01b8152600401612c69929190616667565b602060405180830381600087803b158015612c8357600080fd5b505af1158015612c97573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612cbb9190615c47565b5050505050565b612cca615538565b612cd2611948565b73ffffffffffffffffffffffffffffffffffffffff166398792eec836040518263ffffffff1660e01b8152600401612d0a9190616a1f565b60606040518083038186803b158015612d2257600080fd5b505afa158015612d36573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612d5a9190615e3b565b9050919050565b600080600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060405180608001604052908160008201604051806060016040529081600082018054612dce90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054612dfa90616d3c565b8015612e475780601f10612e1c57610100808354040283529160200191612e47565b820191906000526020600020905b815481529060010190602001808311612e2a57829003601f168201915b50505050508152602001600182018054612e6090616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054612e8c90616d3c565b8015612ed95780601f10612eae57610100808354040283529160200191612ed9565b820191906000526020600020905b815481529060010190602001808311612ebc57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054612f5c90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054612f8890616d3c565b8015612fd55780601f10612faa57610100808354040283529160200191612fd5565b820191906000526020600020905b815481529060010190602001808311612fb857829003601f168201915b50505050508152602001600182018054612fee90616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461301a90616d3c565b80156130675780601f1061303c57610100808354040283529160200191613067565b820191906000526020600020905b81548152906001019060200180831161304a57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156131b1576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016131a890616920565b60405180910390fd5b8060400151915050919050565b6000600560008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206040518060800160405290816000820160405180606001604052908160008201805461322a90616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461325690616d3c565b80156132a35780601f10613278576101008083540402835291602001916132a3565b820191906000526020600020905b81548152906001019060200180831161328657829003601f168201915b505050505081526020016001820180546132bc90616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546132e890616d3c565b80156133355780601f1061330a57610100808354040283529160200191613335565b820191906000526020600020905b81548152906001019060200180831161331857829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815250508152602001600382016040518060600160405290816000820180546133b890616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546133e490616d3c565b80156134315780601f1061340657610100808354040283529160200191613431565b820191906000526020600020905b81548152906001019060200180831161341457829003601f168201915b5050505050815260200160018201805461344a90616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461347690616d3c565b80156134c35780601f10613498576101008083540402835291602001916134c3565b820191906000526020600020905b8154815290600101906020018083116134a657829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff16141561360d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161360490616920565b60405180910390fd5b6000816000015190508160600151156136325761362d816040015161509e565b613633565b5b6000826020015190506136498160400151615179565b60008360400151905080600a60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505050505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff161461372a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613721906169a0565b60405180910390fd5b80600260006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b600b818154811061377e57600080fd5b906000526020600020016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b60006137b88461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff16141561382e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161382590616920565b60405180910390fd5b613837846131be565b61383f611948565b73ffffffffffffffffffffffffffffffffffffffff1663beabacc83385856040518463ffffffff1660e01b815260040161387b93929190616483565b600060405180830381600087803b15801561389557600080fd5b505af11580156138a9573d6000803e3d6000fd5b505050508273ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff16837f08110c4334f195060053093980b785caacfc9c5949cb76c8dcc45d3e5b0e090c87600060405161390e92919061663e565b60405180910390a450505050565b61392461556f565b600560008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206040518060800160405290816000820160405180606001604052908160008201805461398e90616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546139ba90616d3c565b8015613a075780601f106139dc57610100808354040283529160200191613a07565b820191906000526020600020905b8154815290600101906020018083116139ea57829003601f168201915b50505050508152602001600182018054613a2090616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054613a4c90616d3c565b8015613a995780601f10613a6e57610100808354040283529160200191613a99565b820191906000526020600020905b815481529060010190602001808311613a7c57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054613b1c90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054613b4890616d3c565b8015613b955780601f10613b6a57610100808354040283529160200191613b95565b820191906000526020600020905b815481529060010190602001808311613b7857829003601f168201915b50505050508152602001600182018054613bae90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054613bda90616d3c565b8015613c275780601f10613bfc57610100808354040283529160200191613c27565b820191906000526020600020905b815481529060010190602001808311613c0a57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050919050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614613d92576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613d89906169a0565b60405180910390fd5b6000613d9d8361391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415613e13576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613e0a90616920565b60405180910390fd5b613e1c836131be565b613e24611948565b73ffffffffffffffffffffffffffffffffffffffff1663e0f04eaf84846040518363ffffffff1660e01b8152600401613e5e92919061645a565b600060405180830381600087803b158015613e7857600080fd5b505af1158015613e8c573d6000803e3d6000fd5b505050507f696d4f988fdab27b5aa3571ddd084be2015d4117ab1401966989ea9a61fc2b308383604051613ec192919061645a565b60405180910390a1505050565b6060613ed8611948565b73ffffffffffffffffffffffffffffffffffffffff1663e15487026040518163ffffffff1660e01b815260040160006040518083038186803b158015613f1d57600080fd5b505afa158015613f31573d6000803e3d6000fd5b505050506040513d6000823e3d601f19601f82011682018060405250810190613f5a9190615bc5565b905090565b6000613f6a8461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415613fe0576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613fd790616920565b60405180910390fd5b613fe9846131be565b600080600080613ff7611948565b73ffffffffffffffffffffffffffffffffffffffff16630ade77023389896040518463ffffffff1660e01b8152600401614033939291906166ff565b608060405180830381600087803b15801561404d57600080fd5b505af1158015614061573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061408591906159f8565b93509350935093506000869050600115156140a66140a1611ad1565b6129b2565b151514156140b2573490505b600089905060006140c1611948565b73ffffffffffffffffffffffffffffffffffffffff1663ae4044d4846040518263ffffffff1660e01b81526004016140f99190616a1f565b602060405180830381600087803b15801561411357600080fd5b505af1158015614127573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061414b9190615e8d565b90506000614157611948565b73ffffffffffffffffffffffffffffffffffffffff16639a8cea82856040518263ffffffff1660e01b815260040161418f9190616a1f565b602060405180830381600087803b1580156141a957600080fd5b505af11580156141bd573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906141e19190615e8d565b905060006141ed611948565b73ffffffffffffffffffffffffffffffffffffffff166307da163b866040518263ffffffff1660e01b81526004016142259190616a1f565b602060405180830381600087803b15801561423f57600080fd5b505af1158015614253573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906142779190615e8d565b90506000614283611948565b73ffffffffffffffffffffffffffffffffffffffff166363c46cdf876040518263ffffffff1660e01b81526004016142bb9190616a1f565b602060405180830381600087803b1580156142d557600080fd5b505af11580156142e9573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061430d9190615e8d565b905061432261431a612347565b338c8761217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a036338b8f8860008960405161435c969594939291906164ba565b60405180910390a161437761436f612347565b338b8661217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a036338a8f886001886040516143b19695949392919061651b565b60405180910390a1600073ffffffffffffffffffffffffffffffffffffffff168873ffffffffffffffffffffffffffffffffffffffff16141580156143f65750600082115b156144515761440e614406612347565b338a8561217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a03633898f886002876040516144489695949392919061657c565b60405180910390a15b600073ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff161415801561448e5750600081115b156144e9576144a661449e612347565b33898461217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a03633888f886003866040516144e0969594939291906165dd565b60405180910390a15b6144f1611948565b73ffffffffffffffffffffffffffffffffffffffff16639912cd8c338f6040518363ffffffff1660e01b815260040161452b929190616667565b600060405180830381600087803b15801561454557600080fd5b505af1158015614559573d6000803e3d6000fd5b505050507fc0f65426ed29bc920c10a95f6f30fac3e73847a39e0507da5c2ce03be0580cdb858e8c338a8987898b6145919190616bba565b61459b9190616bba565b6040516145ae9796959493929190616690565b60405180910390a15050505050505050505050505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614614656576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161464d906169a0565b60405180910390fd5b60006146618361391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156146d7576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016146ce90616920565b60405180910390fd5b6146e0836131be565b6146e8611948565b73ffffffffffffffffffffffffffffffffffffffff1663e8dbfd5a84846040518363ffffffff1660e01b815260040161472292919061645a565b600060405180830381600087803b15801561473c57600080fd5b505af1158015614750573d6000803e3d6000fd5b505050507fdc3f12914a143aaae689b532671e688d8dd80b665e7f504031ce6561a8b8c543838360405161478592919061645a565b60405180910390a1505050565b61479a614b28565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16141561480a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161480190616880565b60405180910390fd5b61481381614ba6565b50565b60006148218361391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415614897576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161488e90616920565b60405180910390fd5b6148a0836131be565b6148a8611948565b73ffffffffffffffffffffffffffffffffffffffff1663feb8840633846040518363ffffffff1660e01b81526004016148e2929190616667565b600060405180830381600087803b1580156148fc57600080fd5b505af1158015614910573d6000803e3d6000fd5b50505050817fec2b87d914a369632a6c6ee381d0d6d70b1e3a14436184876b35f5562493d3df82600001516040015160405161494c919061643f565b60405180910390a2505050565b60006149656001615254565b9050600061499d600260009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1661499884615262565b615292565b905080600560008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060060160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550600115158315151415614aa3578073ffffffffffffffffffffffffffffffffffffffff1663f21723ec888888886040518563ffffffff1660e01b8152600401614a6c94939291906167a9565b600060405180830381600087803b158015614a8657600080fd5b505af1158015614a9a573d6000803e3d6000fd5b50505050614b15565b8073ffffffffffffffffffffffffffffffffffffffff16630c266c80888888886040518563ffffffff1660e01b8152600401614ae294939291906167a9565b600060405180830381600087803b158015614afc57600080fd5b505af1158015614b10573d6000803e3d6000fd5b505050505b614b1f6001615369565b50505050505050565b614b3061537f565b73ffffffffffffffffffffffffffffffffffffffff16614b4e61279d565b73ffffffffffffffffffffffffffffffffffffffff1614614ba4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614b9b90616980565b60405180910390fd5b565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050816000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415614cda576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614cd190616940565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415614d4a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614d4190616940565b60405180910390fd5b60008111614d8d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614d8490616900565b60405180910390fd5b3073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415614e0d578173ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f19350505050158015614e07573d6000803e3d6000fd5b50614e18565b614e178282615387565b5b505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415614e8d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614e8490616940565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415614efd576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614ef490616940565b60405180910390fd5b60008111614f40576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614f3790616900565b60405180910390fd5b3073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415615007578373ffffffffffffffffffffffffffffffffffffffff1663a9059cbb83836040518363ffffffff1660e01b8152600401614faf929190616667565b602060405180830381600087803b158015614fc957600080fd5b505af1158015614fdd573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906150019190615c47565b50615098565b8373ffffffffffffffffffffffffffffffffffffffff166323b872dd8484846040518463ffffffff1660e01b815260040161504493929190616483565b602060405180830381600087803b15801561505e57600080fd5b505af1158015615072573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906150969190615c47565b505b50505050565b600660009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146151765780600760006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555080600660006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505b50565b600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146152515780600960006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555080600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505b50565b600081600001549050919050565b6000816040516020016152759190616424565b604051602081830303815290604052805190602001209050919050565b60006040517f3d602d80600a3d3981f3363d3d373d3d3d363d7300000000000000000000000081528360601b60148201527f5af43d82803e903d91602b57fd5bf300000000000000000000000000000000006028820152826037826000f5915050600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161415615363576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161535a906168a0565b60405180910390fd5b92915050565b6001816000016000828254019250508190555050565b600033905090565b804710156153ca576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016153c1906168e0565b60405180910390fd5b60008273ffffffffffffffffffffffffffffffffffffffff16826040516153f09061640f565b60006040518083038185875af1925050503d806000811461542d576040519150601f19603f3d011682016040523d82523d6000602084013e615432565b606091505b5050905080615476576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161546d906168c0565b60405180910390fd5b505050565b82805461548790616d3c565b90600052602060002090601f0160209004810192826154a957600085556154f0565b82601f106154c257805160ff19168380011785556154f0565b828001600101855582156154f0579182015b828111156154ef5782518255916020019190600101906154d4565b5b5090506154fd91906155bb565b5090565b60405180606001604052806060815260200160608152602001600073ffffffffffffffffffffffffffffffffffffffff1681525090565b60405180606001604052806000815260200160008152602001600073ffffffffffffffffffffffffffffffffffffffff1681525090565b6040518060800160405280615582615501565b815260200161558f615501565b8152602001600073ffffffffffffffffffffffffffffffffffffffff1681526020016000151581525090565b5b808211156155d45760008160009055506001016155bc565b5090565b60006155eb6155e684616ac0565b616a9b565b9050808382526020820190508285606086028201111561560a57600080fd5b60005b8581101561563a578161562088826157d5565b84526020840193506060830192505060018101905061560d565b5050509392505050565b600061565761565284616aec565b616a9b565b9050808382526020820190508285602086028201111561567657600080fd5b60005b858110156156a6578161568c888261597e565b845260208401935060208301925050600181019050615679565b5050509392505050565b60006156c36156be84616b18565b616a9b565b9050828152602081018484840111156156db57600080fd5b6156e6848285616cfa565b509392505050565b6000813590506156fd8161709f565b92915050565b6000815190506157128161709f565b92915050565b600082601f83011261572957600080fd5b81516157398482602086016155d8565b91505092915050565b600082601f83011261575357600080fd5b8151615763848260208601615644565b91505092915050565b60008135905061577b816170b6565b92915050565b600081519050615790816170b6565b92915050565b6000813590506157a5816170cd565b92915050565b600082601f8301126157bc57600080fd5b81356157cc8482602086016156b0565b91505092915050565b6000606082840312156157e757600080fd5b6157f16060616a9b565b905060006158018482850161597e565b60008301525060206158158482850161597e565b602083015250604061582984828501615703565b60408301525092915050565b60006080828403121561584757600080fd5b6158516080616a9b565b9050600082013567ffffffffffffffff81111561586d57600080fd5b615879848285016158d9565b600083015250602082013567ffffffffffffffff81111561589957600080fd5b6158a5848285016158d9565b60208301525060406158b9848285016156ee565b60408301525060606158cd8482850161576c565b60608301525092915050565b6000606082840312156158eb57600080fd5b6158f56060616a9b565b9050600082013567ffffffffffffffff81111561591157600080fd5b61591d848285016157ab565b600083015250602082013567ffffffffffffffff81111561593d57600080fd5b615949848285016157ab565b602083015250604061595d848285016156ee565b60408301525092915050565b600081359050615978816170e4565b92915050565b60008151905061598d816170e4565b92915050565b6000602082840312156159a557600080fd5b60006159b3848285016156ee565b91505092915050565b600080604083850312156159cf57600080fd5b60006159dd858286016156ee565b92505060206159ee858286016156ee565b9150509250929050565b60008060008060808587031215615a0e57600080fd5b6000615a1c87828801615703565b9450506020615a2d87828801615703565b9350506040615a3e87828801615703565b9250506060615a4f87828801615703565b91505092959194509250565b600080600060608486031215615a7057600080fd5b6000615a7e868287016156ee565b9350506020615a8f868287016156ee565b9250506040615aa086828701615969565b9150509250925092565b60008060408385031215615abd57600080fd5b6000615acb858286016156ee565b925050602083013567ffffffffffffffff811115615ae857600080fd5b615af485828601615835565b9150509250929050565b60008060408385031215615b1157600080fd5b6000615b1f858286016156ee565b9250506020615b3085828601615969565b9150509250929050565b60008060408385031215615b4d57600080fd5b6000615b5b85828601615703565b9250506020615b6c8582860161597e565b9150509250929050565b600080600060608486031215615b8b57600080fd5b6000615b99868287016156ee565b9350506020615baa86828701615969565b9250506040615bbb86828701615969565b9150509250925092565b600060208284031215615bd757600080fd5b600082015167ffffffffffffffff811115615bf157600080fd5b615bfd84828501615718565b91505092915050565b600060208284031215615c1857600080fd5b600082015167ffffffffffffffff811115615c3257600080fd5b615c3e84828501615742565b91505092915050565b600060208284031215615c5957600080fd5b6000615c6784828501615781565b91505092915050565b60008060008060808587031215615c8657600080fd5b6000615c9487828801615796565b9450506020615ca5878288016156ee565b9350506040615cb6878288016156ee565b9250506060615cc787828801615969565b91505092959194509250565b600080600060608486031215615ce857600080fd5b600084013567ffffffffffffffff811115615d0257600080fd5b615d0e868287016157ab565b935050602084013567ffffffffffffffff811115615d2b57600080fd5b615d37868287016157ab565b9250506040615d48868287016156ee565b9150509250925092565b60008060008060008060c08789031215615d6b57600080fd5b600087013567ffffffffffffffff811115615d8557600080fd5b615d9189828a016157ab565b965050602087013567ffffffffffffffff811115615dae57600080fd5b615dba89828a016157ab565b9550506040615dcb89828a016156ee565b945050606087013567ffffffffffffffff811115615de857600080fd5b615df489828a016157ab565b935050608087013567ffffffffffffffff811115615e1157600080fd5b615e1d89828a016157ab565b92505060a0615e2e89828a016156ee565b9150509295509295509295565b600060608284031215615e4d57600080fd5b6000615e5b848285016157d5565b91505092915050565b600060208284031215615e7657600080fd5b6000615e8484828501615969565b91505092915050565b600060208284031215615e9f57600080fd5b6000615ead8482850161597e565b91505092915050565b60008060008060008060c08789031215615ecf57600080fd5b6000615edd89828a0161597e565b9650506020615eee89828a0161597e565b9550506040615eff89828a0161597e565b9450506060615f1089828a0161597e565b9350506080615f2189828a0161597e565b92505060a0615f3289828a0161597e565b9150509295509295509295565b6000615f4b838361623e565b60608301905092915050565b615f6081616c10565b82525050565b615f6f81616c10565b82525050565b6000615f8082616b59565b615f8a8185616b7c565b9350615f9583616b49565b8060005b83811015615fc6578151615fad8882615f3f565b9750615fb883616b6f565b925050600181019050615f99565b5085935050505092915050565b615fdc81616c22565b82525050565b615feb81616c22565b82525050565b615ffa81616c6a565b82525050565b61600981616c8e565b82525050565b61601881616cb2565b82525050565b61602781616cc4565b82525050565b61603681616cd6565b82525050565b61604581616ce8565b82525050565b600061605682616b64565b6160608185616b98565b9350616070818560208601616d09565b61607981616e7f565b840191505092915050565b600061608f82616b64565b6160998185616ba9565b93506160a9818560208601616d09565b6160b281616e7f565b840191505092915050565b60006160ca602683616ba9565b91506160d582616e90565b604082019050919050565b60006160ed601783616ba9565b91506160f882616edf565b602082019050919050565b6000616110603a83616ba9565b915061611b82616f08565b604082019050919050565b6000616133601d83616ba9565b915061613e82616f57565b602082019050919050565b6000616156602183616ba9565b915061616182616f80565b604082019050919050565b6000616179601e83616ba9565b915061618482616fcf565b602082019050919050565b600061619c601d83616ba9565b91506161a782616ff8565b602082019050919050565b60006161bf601783616ba9565b91506161ca82617021565b602082019050919050565b60006161e2602083616ba9565b91506161ed8261704a565b602082019050919050565b6000616205601883616ba9565b915061621082617073565b602082019050919050565b6000616228600083616b8d565b91506162338261709c565b600082019050919050565b60608201600082015161625460008501826163da565b50602082015161626760208501826163da565b50604082015161627a6040850182615f57565b50505050565b60608201600082015161629660008501826163da565b5060208201516162a960208501826163da565b5060408201516162bc6040850182615f57565b50505050565b600060808301600083015184820360008601526162df828261632c565b915050602083015184820360208601526162f9828261632c565b915050604083015161630e6040860182615f57565b5060608301516163216060860182615fd3565b508091505092915050565b60006060830160008301518482036000860152616349828261604b565b91505060208301518482036020860152616363828261604b565b91505060408301516163786040860182615f57565b508091505092915050565b600060608301600083015184820360008601526163a0828261604b565b915050602083015184820360208601526163ba828261604b565b91505060408301516163cf6040860182615f57565b508091505092915050565b6163e381616c60565b82525050565b6163f281616c60565b82525050565b61640961640482616c60565b616de8565b82525050565b600061641a8261621b565b9150819050919050565b600061643082846163f8565b60208201915081905092915050565b60006020820190506164546000830184615f66565b92915050565b600060408201905061646f6000830185615f66565b61647c6020830184615f66565b9392505050565b60006060820190506164986000830186615f66565b6164a56020830185615f66565b6164b260408301846163e9565b949350505050565b600060c0820190506164cf6000830189615f66565b6164dc6020830188615f66565b6164e960408301876163e9565b6164f66060830186615f66565b616503608083018561600f565b61651060a08301846163e9565b979650505050505050565b600060c0820190506165306000830189615f66565b61653d6020830188615f66565b61654a60408301876163e9565b6165576060830186615f66565b616564608083018561601e565b61657160a08301846163e9565b979650505050505050565b600060c0820190506165916000830189615f66565b61659e6020830188615f66565b6165ab60408301876163e9565b6165b86060830186615f66565b6165c5608083018561602d565b6165d260a08301846163e9565b979650505050505050565b600060c0820190506165f26000830189615f66565b6165ff6020830188615f66565b61660c60408301876163e9565b6166196060830186615f66565b616626608083018561603c565b61663360a08301846163e9565b979650505050505050565b60006040820190506166536000830185615f66565b616660602083018461600f565b9392505050565b600060408201905061667c6000830185615f66565b61668960208301846163e9565b9392505050565b600060e0820190506166a5600083018a615f66565b6166b260208301896163e9565b6166bf6040830188615f66565b6166cc6060830187615f66565b6166d960808301866163e9565b6166e660a08301856163e9565b6166f360c08301846163e9565b98975050505050505050565b60006060820190506167146000830186615f66565b61672160208301856163e9565b61672e60408301846163e9565b949350505050565b600060208201905081810360008301526167508184615f75565b905092915050565b600060208201905061676d6000830184615fe2565b92915050565b60006020820190506167886000830184615ff1565b92915050565b60006020820190506167a36000830184616000565b92915050565b600060808201905081810360008301526167c38187616084565b90506167d26020830186615f66565b6167df6040830185615f66565b6167ec6060830184615f66565b95945050505050565b600060e082019050818103600083015261680f818a616084565b905081810360208301526168238189616084565b90506168326040830188615f66565b81810360608301526168448187616084565b905081810360808301526168588186616084565b905061686760a0830185615f66565b61687460c0830184615f66565b98975050505050505050565b60006020820190508181036000830152616899816160bd565b9050919050565b600060208201905081810360008301526168b9816160e0565b9050919050565b600060208201905081810360008301526168d981616103565b9050919050565b600060208201905081810360008301526168f981616126565b9050919050565b6000602082019050818103600083015261691981616149565b9050919050565b600060208201905081810360008301526169398161616c565b9050919050565b600060208201905081810360008301526169598161618f565b9050919050565b60006020820190508181036000830152616979816161b2565b9050919050565b60006020820190508181036000830152616999816161d5565b9050919050565b600060208201905081810360008301526169b9816161f8565b9050919050565b60006060820190506169d56000830184616280565b92915050565b600060208201905081810360008301526169f581846162c2565b905092915050565b60006020820190508181036000830152616a178184616383565b905092915050565b6000602082019050616a3460008301846163e9565b92915050565b600060c082019050616a4f60008301896163e9565b616a5c60208301886163e9565b616a6960408301876163e9565b616a7660608301866163e9565b616a8360808301856163e9565b616a9060a08301846163e9565b979650505050505050565b6000616aa5616ab6565b9050616ab18282616d6e565b919050565b6000604051905090565b600067ffffffffffffffff821115616adb57616ada616e50565b5b602082029050602081019050919050565b600067ffffffffffffffff821115616b0757616b06616e50565b5b602082029050602081019050919050565b600067ffffffffffffffff821115616b3357616b32616e50565b5b616b3c82616e7f565b9050602081019050919050565b6000819050602082019050919050565b600081519050919050565b600081519050919050565b6000602082019050919050565b600082825260208201905092915050565b600081905092915050565b600082825260208201905092915050565b600082825260208201905092915050565b6000616bc582616c60565b9150616bd083616c60565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115616c0557616c04616df2565b5b828201905092915050565b6000616c1b82616c40565b9050919050565b60008115159050919050565b6000616c3982616c10565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b6000616c7582616c7c565b9050919050565b6000616c8782616c40565b9050919050565b6000616c9982616ca0565b9050919050565b6000616cab82616c40565b9050919050565b6000616cbd82616c60565b9050919050565b6000616ccf82616c60565b9050919050565b6000616ce182616c60565b9050919050565b6000616cf382616c60565b9050919050565b82818337600083830152505050565b60005b83811015616d27578082015181840152602081019050616d0c565b83811115616d36576000848401525b50505050565b60006002820490506001821680616d5457607f821691505b60208210811415616d6857616d67616e21565b5b50919050565b616d7782616e7f565b810181811067ffffffffffffffff82111715616d9657616d95616e50565b5b80604052505050565b6000616daa82616c60565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff821415616ddd57616ddc616df2565b5b600182019050919050565b6000819050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6000601f19601f8301169050919050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f455243313136373a2063726561746532206661696c6564000000000000000000600082015250565b7f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260008201527f6563697069656e74206d61792068617665207265766572746564000000000000602082015250565b7f416464726573733a20696e73756666696369656e742062616c616e6365000000600082015250565b7f7472616e7366657220616d6f756e74206d757374206c61726765207468616e2060008201527f3000000000000000000000000000000000000000000000000000000000000000602082015250565b7f5468697320746f6b656e207374696c6c206e6f742072656769737465642e0000600082015250565b7f7472616e736665722061646472657373206d757374206e6f7420307830000000600082015250565b7f636f6e7472616374206d75737420686176652063656c6f000000000000000000600082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f61756374696f6e3a2077726f6e6720646576656c6f7065720000000000000000600082015250565b50565b6170a881616c10565b81146170b357600080fd5b50565b6170bf81616c22565b81146170ca57600080fd5b50565b6170d681616c2e565b81146170e157600080fd5b50565b6170ed81616c60565b81146170f857600080fd5b5056fea264697066735822122085c5ba6c8aab14ea02e90bd76fd1a5a63d5a368022414dc628b2f7e72f56bc5664736f6c634300080300330000000000000000000000008b2f369379c6ccfec432e34d435712616666963a0000000000000000000000008b2f369379c6ccfec432e34d435712616666963a00000000000000000000000001e833861eb2ed9e8f4f88e3feee555d8a993640

Deployed ByteCode

0x6080604052600436106102255760003560e01c806388a8c95c11610123578063b4ed5546116100ab578063e15487021161006f578063e154870214610804578063e671499b1461082f578063e8dbfd5a1461084b578063f2fde38b14610874578063feb884061461089d57610225565b8063b4ed55461461070f578063b99c6fa714610738578063beabacc814610775578063d1715aed1461079e578063e0f04eaf146107db57610225565b806391ad9428116100f257806391ad9428146106065780639456fbcc1461064357806398792eec1461066c578063a77ad2fd146106a9578063ade56423146106e657610225565b806388a8c95c1461055e5780638da5cb5b146105875780638ebf66ff146105b25780638f84aa09146105db57610225565b806369c33a07116101b157806379a35b4b1161017557806379a35b4b1461048f57806379c0a3ea146104b85780637de7589d146104e15780637e07590d1461050c5780638264045a1461053557610225565b806369c33a07146103be5780636d1d7a97146103fb578063715018a614610426578063741cbae41461043d57806374cf93cd1461046657610225565b80633ad10ef6116101f85780633ad10ef6146102d057806348a806b4146102fb5780634b4703ec146103265780635c60da1b14610368578063696db72c1461039357610225565b80631f4f82eb1461022a578063264e5d5a1461025357806334e0955214610290578063383fba25146102b9575b600080fd5b34801561023657600080fd5b50610251600480360381019061024c9190615d52565b6108c6565b005b34801561025f57600080fd5b5061027a60048036038101906102759190615993565b61106e565b60405161028791906169fd565b60405180910390f35b34801561029c57600080fd5b506102b760048036038101906102b29190615a5b565b6114d2565b005b3480156102c557600080fd5b506102ce6116a3565b005b3480156102dc57600080fd5b506102e5611922565b6040516102f2919061643f565b60405180910390f35b34801561030757600080fd5b50610310611948565b60405161031d919061678e565b60405180910390f35b34801561033257600080fd5b5061034d60048036038101906103489190615993565b611972565b60405161035f96959493929190616a3a565b60405180910390f35b34801561037457600080fd5b5061037d611aab565b60405161038a919061643f565b60405180910390f35b34801561039f57600080fd5b506103a8611ad1565b6040516103b5919061643f565b60405180910390f35b3480156103ca57600080fd5b506103e560048036038101906103e09190615993565b611afb565b6040516103f291906169fd565b60405180910390f35b34801561040757600080fd5b50610410611f5f565b60405161041d919061643f565b60405180910390f35b34801561043257600080fd5b5061043b611f89565b005b34801561044957600080fd5b50610464600480360381019061045f9190615b76565b611f9d565b005b34801561047257600080fd5b5061048d60048036038101906104889190615cd3565b6120dd565b005b34801561049b57600080fd5b506104b660048036038101906104b19190615c70565b61217c565b005b3480156104c457600080fd5b506104df60048036038101906104da9190615993565b6121b4565b005b3480156104ed57600080fd5b506104f6612347565b6040516105039190616773565b60405180910390f35b34801561051857600080fd5b50610533600480360381019061052e9190615b76565b612371565b005b34801561054157600080fd5b5061055c60048036038101906105579190615b76565b6124f8565b005b34801561056a57600080fd5b5061058560048036038101906105809190615993565b6126c9565b005b34801561059357600080fd5b5061059c61279d565b6040516105a9919061643f565b60405180910390f35b3480156105be57600080fd5b506105d960048036038101906105d49190615aaa565b6127c6565b005b3480156105e757600080fd5b506105f061298c565b6040516105fd919061643f565b60405180910390f35b34801561061257600080fd5b5061062d60048036038101906106289190615993565b6129b2565b60405161063a9190616758565b60405180910390f35b34801561064f57600080fd5b5061066a600480360381019061066591906159bc565b612a59565b005b34801561067857600080fd5b50610693600480360381019061068e9190615e64565b612cc2565b6040516106a091906169c0565b60405180910390f35b3480156106b557600080fd5b506106d060048036038101906106cb9190615993565b612d61565b6040516106dd919061643f565b60405180910390f35b3480156106f257600080fd5b5061070d60048036038101906107089190615993565b6131be565b005b34801561071b57600080fd5b5061073660048036038101906107319190615993565b61369a565b005b34801561074457600080fd5b5061075f600480360381019061075a9190615e64565b61376e565b60405161076c919061643f565b60405180910390f35b34801561078157600080fd5b5061079c60048036038101906107979190615a5b565b6137ad565b005b3480156107aa57600080fd5b506107c560048036038101906107c09190615993565b61391c565b6040516107d291906169db565b60405180910390f35b3480156107e757600080fd5b5061080260048036038101906107fd91906159bc565b613d02565b005b34801561081057600080fd5b50610819613ece565b6040516108269190616736565b60405180910390f35b61084960048036038101906108449190615b76565b613f5f565b005b34801561085757600080fd5b50610872600480360381019061086d91906159bc565b6145c6565b005b34801561088057600080fd5b5061089b60048036038101906108969190615993565b614792565b005b3480156108a957600080fd5b506108c460048036038101906108bf9190615afe565b614816565b005b600073ffffffffffffffffffffffffffffffffffffffff16600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060060160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156109c157600b849080600181540180825580915050600190039060005260206000200160009091909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505b60405180606001604052808781526020018681526020018573ffffffffffffffffffffffffffffffffffffffff16815250600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000820151816000019080519060200190610a5092919061547b565b506020820151816001019080519060200190610a6d92919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555090505060405180606001604052808481526020018381526020018273ffffffffffffffffffffffffffffffffffffffff16815250600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206003016000820151816000019080519060200190610b4792919061547b565b506020820151816001019080519060200190610b6492919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055509050506001600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060060160146101000a81548160ff021916908315150217905550610c3a868583600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff166001614959565b6000600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060405180608001604052908160008201604051806060016040529081600082018054610ca690616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610cd290616d3c565b8015610d1f5780601f10610cf457610100808354040283529160200191610d1f565b820191906000526020600020905b815481529060010190602001808311610d0257829003601f168201915b50505050508152602001600182018054610d3890616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610d6490616d3c565b8015610db15780601f10610d8657610100808354040283529160200191610db1565b820191906000526020600020905b815481529060010190602001808311610d9457829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054610e3490616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610e6090616d3c565b8015610ead5780601f10610e8257610100808354040283529160200191610ead565b820191906000526020600020905b815481529060010190602001808311610e9057829003601f168201915b50505050508152602001600182018054610ec690616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054610ef290616d3c565b8015610f3f5780601f10610f1457610100808354040283529160200191610f3f565b820191906000526020600020905b815481529060010190602001808311610f2257829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff161515151581525050905061101e856131be565b7fe26e9ba2baee73bb13c229f09b48c470f45b9338eec5b098406161ac0d79e1ac878787878787876040015160405161105d97969594939291906167f5565b60405180910390a150505050505050565b611076615501565b6000600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020604051806080016040529081600082016040518060600160405290816000820180546110e290616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461110e90616d3c565b801561115b5780601f106111305761010080835404028352916020019161115b565b820191906000526020600020905b81548152906001019060200180831161113e57829003601f168201915b5050505050815260200160018201805461117490616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546111a090616d3c565b80156111ed5780601f106111c2576101008083540402835291602001916111ed565b820191906000526020600020905b8154815290600101906020018083116111d057829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016003820160405180606001604052908160008201805461127090616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461129c90616d3c565b80156112e95780601f106112be576101008083540402835291602001916112e9565b820191906000526020600020905b8154815290600101906020018083116112cc57829003601f168201915b5050505050815260200160018201805461130290616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461132e90616d3c565b801561137b5780601f106113505761010080835404028352916020019161137b565b820191906000526020600020905b81548152906001019060200180831161135e57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156114c5576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016114bc90616920565b60405180910390fd5b8060000151915050919050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614611562576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611559906169a0565b60405180910390fd5b600061156d8461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156115e3576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016115da90616920565b60405180910390fd5b6115ec846131be565b6115f4611948565b73ffffffffffffffffffffffffffffffffffffffff166334e095528585856040518463ffffffff1660e01b815260040161163093929190616483565b600060405180830381600087803b15801561164a57600080fd5b505af115801561165e573d6000803e3d6000fd5b505050507f917840854c773dda3ec519bb6dd82de7389053094e0b4e8442db88b43681bb7884848460405161169593929190616483565b60405180910390a150505050565b60005b600b8054905081101561191f576000600b82815481106116ef577f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b9060005260206000200160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905060006117278261391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161461190a5761176a826131be565b6000611774611948565b73ffffffffffffffffffffffffffffffffffffffff1663383fba256040518163ffffffff1660e01b8152600401600060405180830381600087803b1580156117bb57600080fd5b505af11580156117cf573d6000803e3d6000fd5b505050506040513d6000823e3d601f19601f820116820180604052508101906117f89190615c06565b905060005b81518110156118a0576000828281518110611841577f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b602002602001015190507fc87975e7f19023dc873bd4977d00e5b4f7bd2f1fb6a670849c147bcf1288853384600001516040015182604051611884929190616667565b60405180910390a150808061189890616d9f565b9150506117fd565b506118a9611948565b73ffffffffffffffffffffffffffffffffffffffff1663bbaed38f6040518163ffffffff1660e01b8152600401600060405180830381600087803b1580156118f057600080fd5b505af1158015611904573d6000803e3d6000fd5b50505050505b5050808061191790616d9f565b9150506116a6565b50565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000600a60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b60008060008060008060006119868861391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156119fc576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016119f390616920565b60405180910390fd5b611a05886131be565b611a0d611948565b73ffffffffffffffffffffffffffffffffffffffff1663606df430896040518263ffffffff1660e01b8152600401611a45919061643f565b60c06040518083038186803b158015611a5d57600080fd5b505afa158015611a71573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611a959190615eb6565b9650965096509650965096505091939550919395565b600260009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611b03615501565b6000600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060405180608001604052908160008201604051806060016040529081600082018054611b6f90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611b9b90616d3c565b8015611be85780601f10611bbd57610100808354040283529160200191611be8565b820191906000526020600020905b815481529060010190602001808311611bcb57829003601f168201915b50505050508152602001600182018054611c0190616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611c2d90616d3c565b8015611c7a5780601f10611c4f57610100808354040283529160200191611c7a565b820191906000526020600020905b815481529060010190602001808311611c5d57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054611cfd90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611d2990616d3c565b8015611d765780601f10611d4b57610100808354040283529160200191611d76565b820191906000526020600020905b815481529060010190602001808311611d5957829003601f168201915b50505050508152602001600182018054611d8f90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054611dbb90616d3c565b8015611e085780601f10611ddd57610100808354040283529160200191611e08565b820191906000526020600020905b815481529060010190602001808311611deb57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415611f52576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611f4990616920565b60405180910390fd5b8060200151915050919050565b6000600660009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611f91614b28565b611f9b6000614ba6565b565b6000611fa88461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff16141561201e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161201590616920565b60405180910390fd5b612027846131be565b61202f611948565b73ffffffffffffffffffffffffffffffffffffffff1663741cbae43385856040518463ffffffff1660e01b815260040161206b939291906166ff565b600060405180830381600087803b15801561208557600080fd5b505af1158015612099573d6000803e3d6000fd5b50505050827fba2254e53d458d2a10902ac4fca53c8e8f2fb623f5716c5f02948b1f7762b81b85846040516120cf929190616667565b60405180910390a250505050565b6121778383836040518060400160405280600481526020017f63656c6f000000000000000000000000000000000000000000000000000000008152506040518060400160405280600481526020017f63656c6f00000000000000000000000000000000000000000000000000000000815250600c60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff166108c6565b505050565b60011515612189856129b2565b151514156121a15761219c838383614c6a565b6121ae565b6121ad84848484614e1d565b5b50505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612244576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161223b906169a0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614156122b4576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016122ab90616940565b60405180910390fd5b6000479050600081116122fc576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016122f390616960565b60405180910390fd5b8173ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f19350505050158015612342573d6000803e3d6000fd5b505050565b6000600960009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b600061237c8461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156123f2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016123e990616920565b60405180910390fd5b6123fb846131be565b600080612406611948565b73ffffffffffffffffffffffffffffffffffffffff16637e07590d3387876040518463ffffffff1660e01b8152600401612442939291906166ff565b6040805180830381600087803b15801561245b57600080fd5b505af115801561246f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906124939190615b3a565b915091508173ffffffffffffffffffffffffffffffffffffffff16857f2a72dbd9400910f5daeea0d38f90dfff5f64b1674015f8666bb35ab548ecf9d5856000015160400151846040516124e8929190616667565b60405180910390a3505050505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612588576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161257f906169a0565b60405180910390fd5b60006125938461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415612609576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161260090616920565b60405180910390fd5b612612846131be565b61261a611948565b73ffffffffffffffffffffffffffffffffffffffff16638264045a8585856040518463ffffffff1660e01b8152600401612656939291906166ff565b600060405180830381600087803b15801561267057600080fd5b505af1158015612684573d6000803e3d6000fd5b505050507fc4cf474fdae931e45f789cbd99027d46dad91d17ed83c00fbc05ba44c23b85948484846040516126bb939291906166ff565b60405180910390a150505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612759576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612750906169a0565b60405180910390fd5b80600460006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b80600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082015181600001600082015181600001908051906020019061282b92919061547b565b50602082015181600101908051906020019061284892919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505060208201518160030160008201518160000190805190602001906128b792919061547b565b5060208201518160010190805190602001906128d492919061547b565b5060408201518160020160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505060408201518160060160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060608201518160060160146101000a81548160ff0219169083151502179055509050505050565b600360009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b600073471ece3750da237f93b8e339c536989b8978a43873ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161480612a41575073f194afdf50b03e69bd7d057c1aa9e10c9954e4c973ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16145b15612a4f5760019050612a54565b600090505b919050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614612ae9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612ae0906169a0565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161415612b59576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612b5090616940565b60405180910390fd5b600082905060008173ffffffffffffffffffffffffffffffffffffffff166370a08231306040518263ffffffff1660e01b8152600401612b99919061643f565b60206040518083038186803b158015612bb157600080fd5b505afa158015612bc5573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612be99190615e8d565b905060008111612c2e576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612c2590616960565b60405180910390fd5b8173ffffffffffffffffffffffffffffffffffffffff1663a9059cbb84836040518363ffffffff1660e01b8152600401612c69929190616667565b602060405180830381600087803b158015612c8357600080fd5b505af1158015612c97573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612cbb9190615c47565b5050505050565b612cca615538565b612cd2611948565b73ffffffffffffffffffffffffffffffffffffffff166398792eec836040518263ffffffff1660e01b8152600401612d0a9190616a1f565b60606040518083038186803b158015612d2257600080fd5b505afa158015612d36573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612d5a9190615e3b565b9050919050565b600080600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060405180608001604052908160008201604051806060016040529081600082018054612dce90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054612dfa90616d3c565b8015612e475780601f10612e1c57610100808354040283529160200191612e47565b820191906000526020600020905b815481529060010190602001808311612e2a57829003601f168201915b50505050508152602001600182018054612e6090616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054612e8c90616d3c565b8015612ed95780601f10612eae57610100808354040283529160200191612ed9565b820191906000526020600020905b815481529060010190602001808311612ebc57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054612f5c90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054612f8890616d3c565b8015612fd55780601f10612faa57610100808354040283529160200191612fd5565b820191906000526020600020905b815481529060010190602001808311612fb857829003601f168201915b50505050508152602001600182018054612fee90616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461301a90616d3c565b80156130675780601f1061303c57610100808354040283529160200191613067565b820191906000526020600020905b81548152906001019060200180831161304a57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156131b1576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016131a890616920565b60405180910390fd5b8060400151915050919050565b6000600560008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206040518060800160405290816000820160405180606001604052908160008201805461322a90616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461325690616d3c565b80156132a35780601f10613278576101008083540402835291602001916132a3565b820191906000526020600020905b81548152906001019060200180831161328657829003601f168201915b505050505081526020016001820180546132bc90616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546132e890616d3c565b80156133355780601f1061330a57610100808354040283529160200191613335565b820191906000526020600020905b81548152906001019060200180831161331857829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815250508152602001600382016040518060600160405290816000820180546133b890616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546133e490616d3c565b80156134315780601f1061340657610100808354040283529160200191613431565b820191906000526020600020905b81548152906001019060200180831161341457829003601f168201915b5050505050815260200160018201805461344a90616d3c565b80601f016020809104026020016040519081016040528092919081815260200182805461347690616d3c565b80156134c35780601f10613498576101008083540402835291602001916134c3565b820191906000526020600020905b8154815290600101906020018083116134a657829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff16141561360d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161360490616920565b60405180910390fd5b6000816000015190508160600151156136325761362d816040015161509e565b613633565b5b6000826020015190506136498160400151615179565b60008360400151905080600a60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505050505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff161461372a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613721906169a0565b60405180910390fd5b80600260006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b600b818154811061377e57600080fd5b906000526020600020016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b60006137b88461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff16141561382e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161382590616920565b60405180910390fd5b613837846131be565b61383f611948565b73ffffffffffffffffffffffffffffffffffffffff1663beabacc83385856040518463ffffffff1660e01b815260040161387b93929190616483565b600060405180830381600087803b15801561389557600080fd5b505af11580156138a9573d6000803e3d6000fd5b505050508273ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff16837f08110c4334f195060053093980b785caacfc9c5949cb76c8dcc45d3e5b0e090c87600060405161390e92919061663e565b60405180910390a450505050565b61392461556f565b600560008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206040518060800160405290816000820160405180606001604052908160008201805461398e90616d3c565b80601f01602080910402602001604051908101604052809291908181526020018280546139ba90616d3c565b8015613a075780601f106139dc57610100808354040283529160200191613a07565b820191906000526020600020905b8154815290600101906020018083116139ea57829003601f168201915b50505050508152602001600182018054613a2090616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054613a4c90616d3c565b8015613a995780601f10613a6e57610100808354040283529160200191613a99565b820191906000526020600020905b815481529060010190602001808311613a7c57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681525050815260200160038201604051806060016040529081600082018054613b1c90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054613b4890616d3c565b8015613b955780601f10613b6a57610100808354040283529160200191613b95565b820191906000526020600020905b815481529060010190602001808311613b7857829003601f168201915b50505050508152602001600182018054613bae90616d3c565b80601f0160208091040260200160405190810160405280929190818152602001828054613bda90616d3c565b8015613c275780601f10613bfc57610100808354040283529160200191613c27565b820191906000526020600020905b815481529060010190602001808311613c0a57829003601f168201915b505050505081526020016002820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152505081526020016006820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016006820160149054906101000a900460ff1615151515815250509050919050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614613d92576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613d89906169a0565b60405180910390fd5b6000613d9d8361391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415613e13576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613e0a90616920565b60405180910390fd5b613e1c836131be565b613e24611948565b73ffffffffffffffffffffffffffffffffffffffff1663e0f04eaf84846040518363ffffffff1660e01b8152600401613e5e92919061645a565b600060405180830381600087803b158015613e7857600080fd5b505af1158015613e8c573d6000803e3d6000fd5b505050507f696d4f988fdab27b5aa3571ddd084be2015d4117ab1401966989ea9a61fc2b308383604051613ec192919061645a565b60405180910390a1505050565b6060613ed8611948565b73ffffffffffffffffffffffffffffffffffffffff1663e15487026040518163ffffffff1660e01b815260040160006040518083038186803b158015613f1d57600080fd5b505afa158015613f31573d6000803e3d6000fd5b505050506040513d6000823e3d601f19601f82011682018060405250810190613f5a9190615bc5565b905090565b6000613f6a8461391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415613fe0576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401613fd790616920565b60405180910390fd5b613fe9846131be565b600080600080613ff7611948565b73ffffffffffffffffffffffffffffffffffffffff16630ade77023389896040518463ffffffff1660e01b8152600401614033939291906166ff565b608060405180830381600087803b15801561404d57600080fd5b505af1158015614061573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061408591906159f8565b93509350935093506000869050600115156140a66140a1611ad1565b6129b2565b151514156140b2573490505b600089905060006140c1611948565b73ffffffffffffffffffffffffffffffffffffffff1663ae4044d4846040518263ffffffff1660e01b81526004016140f99190616a1f565b602060405180830381600087803b15801561411357600080fd5b505af1158015614127573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061414b9190615e8d565b90506000614157611948565b73ffffffffffffffffffffffffffffffffffffffff16639a8cea82856040518263ffffffff1660e01b815260040161418f9190616a1f565b602060405180830381600087803b1580156141a957600080fd5b505af11580156141bd573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906141e19190615e8d565b905060006141ed611948565b73ffffffffffffffffffffffffffffffffffffffff166307da163b866040518263ffffffff1660e01b81526004016142259190616a1f565b602060405180830381600087803b15801561423f57600080fd5b505af1158015614253573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906142779190615e8d565b90506000614283611948565b73ffffffffffffffffffffffffffffffffffffffff166363c46cdf876040518263ffffffff1660e01b81526004016142bb9190616a1f565b602060405180830381600087803b1580156142d557600080fd5b505af11580156142e9573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061430d9190615e8d565b905061432261431a612347565b338c8761217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a036338b8f8860008960405161435c969594939291906164ba565b60405180910390a161437761436f612347565b338b8661217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a036338a8f886001886040516143b19695949392919061651b565b60405180910390a1600073ffffffffffffffffffffffffffffffffffffffff168873ffffffffffffffffffffffffffffffffffffffff16141580156143f65750600082115b156144515761440e614406612347565b338a8561217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a03633898f886002876040516144489695949392919061657c565b60405180910390a15b600073ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff161415801561448e5750600081115b156144e9576144a661449e612347565b33898461217c565b7f1fb03a71d0375135794d21f644ae7dc64b0ad4e30d89bedc889603e15fb0a03633888f886003866040516144e0969594939291906165dd565b60405180910390a15b6144f1611948565b73ffffffffffffffffffffffffffffffffffffffff16639912cd8c338f6040518363ffffffff1660e01b815260040161452b929190616667565b600060405180830381600087803b15801561454557600080fd5b505af1158015614559573d6000803e3d6000fd5b505050507fc0f65426ed29bc920c10a95f6f30fac3e73847a39e0507da5c2ce03be0580cdb858e8c338a8987898b6145919190616bba565b61459b9190616bba565b6040516145ae9796959493929190616690565b60405180910390a15050505050505050505050505050565b600460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff1614614656576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161464d906169a0565b60405180910390fd5b60006146618361391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff1614156146d7576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016146ce90616920565b60405180910390fd5b6146e0836131be565b6146e8611948565b73ffffffffffffffffffffffffffffffffffffffff1663e8dbfd5a84846040518363ffffffff1660e01b815260040161472292919061645a565b600060405180830381600087803b15801561473c57600080fd5b505af1158015614750573d6000803e3d6000fd5b505050507fdc3f12914a143aaae689b532671e688d8dd80b665e7f504031ce6561a8b8c543838360405161478592919061645a565b60405180910390a1505050565b61479a614b28565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16141561480a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161480190616880565b60405180910390fd5b61481381614ba6565b50565b60006148218361391c565b9050600073ffffffffffffffffffffffffffffffffffffffff16816040015173ffffffffffffffffffffffffffffffffffffffff161415614897576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161488e90616920565b60405180910390fd5b6148a0836131be565b6148a8611948565b73ffffffffffffffffffffffffffffffffffffffff1663feb8840633846040518363ffffffff1660e01b81526004016148e2929190616667565b600060405180830381600087803b1580156148fc57600080fd5b505af1158015614910573d6000803e3d6000fd5b50505050817fec2b87d914a369632a6c6ee381d0d6d70b1e3a14436184876b35f5562493d3df82600001516040015160405161494c919061643f565b60405180910390a2505050565b60006149656001615254565b9050600061499d600260009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1661499884615262565b615292565b905080600560008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060060160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550600115158315151415614aa3578073ffffffffffffffffffffffffffffffffffffffff1663f21723ec888888886040518563ffffffff1660e01b8152600401614a6c94939291906167a9565b600060405180830381600087803b158015614a8657600080fd5b505af1158015614a9a573d6000803e3d6000fd5b50505050614b15565b8073ffffffffffffffffffffffffffffffffffffffff16630c266c80888888886040518563ffffffff1660e01b8152600401614ae294939291906167a9565b600060405180830381600087803b158015614afc57600080fd5b505af1158015614b10573d6000803e3d6000fd5b505050505b614b1f6001615369565b50505050505050565b614b3061537f565b73ffffffffffffffffffffffffffffffffffffffff16614b4e61279d565b73ffffffffffffffffffffffffffffffffffffffff1614614ba4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614b9b90616980565b60405180910390fd5b565b60008060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050816000806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415614cda576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614cd190616940565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415614d4a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614d4190616940565b60405180910390fd5b60008111614d8d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614d8490616900565b60405180910390fd5b3073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415614e0d578173ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f19350505050158015614e07573d6000803e3d6000fd5b50614e18565b614e178282615387565b5b505050565b600073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415614e8d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614e8490616940565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff161415614efd576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614ef490616940565b60405180910390fd5b60008111614f40576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401614f3790616900565b60405180910390fd5b3073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415615007578373ffffffffffffffffffffffffffffffffffffffff1663a9059cbb83836040518363ffffffff1660e01b8152600401614faf929190616667565b602060405180830381600087803b158015614fc957600080fd5b505af1158015614fdd573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906150019190615c47565b50615098565b8373ffffffffffffffffffffffffffffffffffffffff166323b872dd8484846040518463ffffffff1660e01b815260040161504493929190616483565b602060405180830381600087803b15801561505e57600080fd5b505af1158015615072573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906150969190615c47565b505b50505050565b600660009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146151765780600760006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555080600660006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505b50565b600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146152515780600960006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555080600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505b50565b600081600001549050919050565b6000816040516020016152759190616424565b604051602081830303815290604052805190602001209050919050565b60006040517f3d602d80600a3d3981f3363d3d373d3d3d363d7300000000000000000000000081528360601b60148201527f5af43d82803e903d91602b57fd5bf300000000000000000000000000000000006028820152826037826000f5915050600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161415615363576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161535a906168a0565b60405180910390fd5b92915050565b6001816000016000828254019250508190555050565b600033905090565b804710156153ca576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016153c1906168e0565b60405180910390fd5b60008273ffffffffffffffffffffffffffffffffffffffff16826040516153f09061640f565b60006040518083038185875af1925050503d806000811461542d576040519150601f19603f3d011682016040523d82523d6000602084013e615432565b606091505b5050905080615476576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161546d906168c0565b60405180910390fd5b505050565b82805461548790616d3c565b90600052602060002090601f0160209004810192826154a957600085556154f0565b82601f106154c257805160ff19168380011785556154f0565b828001600101855582156154f0579182015b828111156154ef5782518255916020019190600101906154d4565b5b5090506154fd91906155bb565b5090565b60405180606001604052806060815260200160608152602001600073ffffffffffffffffffffffffffffffffffffffff1681525090565b60405180606001604052806000815260200160008152602001600073ffffffffffffffffffffffffffffffffffffffff1681525090565b6040518060800160405280615582615501565b815260200161558f615501565b8152602001600073ffffffffffffffffffffffffffffffffffffffff1681526020016000151581525090565b5b808211156155d45760008160009055506001016155bc565b5090565b60006155eb6155e684616ac0565b616a9b565b9050808382526020820190508285606086028201111561560a57600080fd5b60005b8581101561563a578161562088826157d5565b84526020840193506060830192505060018101905061560d565b5050509392505050565b600061565761565284616aec565b616a9b565b9050808382526020820190508285602086028201111561567657600080fd5b60005b858110156156a6578161568c888261597e565b845260208401935060208301925050600181019050615679565b5050509392505050565b60006156c36156be84616b18565b616a9b565b9050828152602081018484840111156156db57600080fd5b6156e6848285616cfa565b509392505050565b6000813590506156fd8161709f565b92915050565b6000815190506157128161709f565b92915050565b600082601f83011261572957600080fd5b81516157398482602086016155d8565b91505092915050565b600082601f83011261575357600080fd5b8151615763848260208601615644565b91505092915050565b60008135905061577b816170b6565b92915050565b600081519050615790816170b6565b92915050565b6000813590506157a5816170cd565b92915050565b600082601f8301126157bc57600080fd5b81356157cc8482602086016156b0565b91505092915050565b6000606082840312156157e757600080fd5b6157f16060616a9b565b905060006158018482850161597e565b60008301525060206158158482850161597e565b602083015250604061582984828501615703565b60408301525092915050565b60006080828403121561584757600080fd5b6158516080616a9b565b9050600082013567ffffffffffffffff81111561586d57600080fd5b615879848285016158d9565b600083015250602082013567ffffffffffffffff81111561589957600080fd5b6158a5848285016158d9565b60208301525060406158b9848285016156ee565b60408301525060606158cd8482850161576c565b60608301525092915050565b6000606082840312156158eb57600080fd5b6158f56060616a9b565b9050600082013567ffffffffffffffff81111561591157600080fd5b61591d848285016157ab565b600083015250602082013567ffffffffffffffff81111561593d57600080fd5b615949848285016157ab565b602083015250604061595d848285016156ee565b60408301525092915050565b600081359050615978816170e4565b92915050565b60008151905061598d816170e4565b92915050565b6000602082840312156159a557600080fd5b60006159b3848285016156ee565b91505092915050565b600080604083850312156159cf57600080fd5b60006159dd858286016156ee565b92505060206159ee858286016156ee565b9150509250929050565b60008060008060808587031215615a0e57600080fd5b6000615a1c87828801615703565b9450506020615a2d87828801615703565b9350506040615a3e87828801615703565b9250506060615a4f87828801615703565b91505092959194509250565b600080600060608486031215615a7057600080fd5b6000615a7e868287016156ee565b9350506020615a8f868287016156ee565b9250506040615aa086828701615969565b9150509250925092565b60008060408385031215615abd57600080fd5b6000615acb858286016156ee565b925050602083013567ffffffffffffffff811115615ae857600080fd5b615af485828601615835565b9150509250929050565b60008060408385031215615b1157600080fd5b6000615b1f858286016156ee565b9250506020615b3085828601615969565b9150509250929050565b60008060408385031215615b4d57600080fd5b6000615b5b85828601615703565b9250506020615b6c8582860161597e565b9150509250929050565b600080600060608486031215615b8b57600080fd5b6000615b99868287016156ee565b9350506020615baa86828701615969565b9250506040615bbb86828701615969565b9150509250925092565b600060208284031215615bd757600080fd5b600082015167ffffffffffffffff811115615bf157600080fd5b615bfd84828501615718565b91505092915050565b600060208284031215615c1857600080fd5b600082015167ffffffffffffffff811115615c3257600080fd5b615c3e84828501615742565b91505092915050565b600060208284031215615c5957600080fd5b6000615c6784828501615781565b91505092915050565b60008060008060808587031215615c8657600080fd5b6000615c9487828801615796565b9450506020615ca5878288016156ee565b9350506040615cb6878288016156ee565b9250506060615cc787828801615969565b91505092959194509250565b600080600060608486031215615ce857600080fd5b600084013567ffffffffffffffff811115615d0257600080fd5b615d0e868287016157ab565b935050602084013567ffffffffffffffff811115615d2b57600080fd5b615d37868287016157ab565b9250506040615d48868287016156ee565b9150509250925092565b60008060008060008060c08789031215615d6b57600080fd5b600087013567ffffffffffffffff811115615d8557600080fd5b615d9189828a016157ab565b965050602087013567ffffffffffffffff811115615dae57600080fd5b615dba89828a016157ab565b9550506040615dcb89828a016156ee565b945050606087013567ffffffffffffffff811115615de857600080fd5b615df489828a016157ab565b935050608087013567ffffffffffffffff811115615e1157600080fd5b615e1d89828a016157ab565b92505060a0615e2e89828a016156ee565b9150509295509295509295565b600060608284031215615e4d57600080fd5b6000615e5b848285016157d5565b91505092915050565b600060208284031215615e7657600080fd5b6000615e8484828501615969565b91505092915050565b600060208284031215615e9f57600080fd5b6000615ead8482850161597e565b91505092915050565b60008060008060008060c08789031215615ecf57600080fd5b6000615edd89828a0161597e565b9650506020615eee89828a0161597e565b9550506040615eff89828a0161597e565b9450506060615f1089828a0161597e565b9350506080615f2189828a0161597e565b92505060a0615f3289828a0161597e565b9150509295509295509295565b6000615f4b838361623e565b60608301905092915050565b615f6081616c10565b82525050565b615f6f81616c10565b82525050565b6000615f8082616b59565b615f8a8185616b7c565b9350615f9583616b49565b8060005b83811015615fc6578151615fad8882615f3f565b9750615fb883616b6f565b925050600181019050615f99565b5085935050505092915050565b615fdc81616c22565b82525050565b615feb81616c22565b82525050565b615ffa81616c6a565b82525050565b61600981616c8e565b82525050565b61601881616cb2565b82525050565b61602781616cc4565b82525050565b61603681616cd6565b82525050565b61604581616ce8565b82525050565b600061605682616b64565b6160608185616b98565b9350616070818560208601616d09565b61607981616e7f565b840191505092915050565b600061608f82616b64565b6160998185616ba9565b93506160a9818560208601616d09565b6160b281616e7f565b840191505092915050565b60006160ca602683616ba9565b91506160d582616e90565b604082019050919050565b60006160ed601783616ba9565b91506160f882616edf565b602082019050919050565b6000616110603a83616ba9565b915061611b82616f08565b604082019050919050565b6000616133601d83616ba9565b915061613e82616f57565b602082019050919050565b6000616156602183616ba9565b915061616182616f80565b604082019050919050565b6000616179601e83616ba9565b915061618482616fcf565b602082019050919050565b600061619c601d83616ba9565b91506161a782616ff8565b602082019050919050565b60006161bf601783616ba9565b91506161ca82617021565b602082019050919050565b60006161e2602083616ba9565b91506161ed8261704a565b602082019050919050565b6000616205601883616ba9565b915061621082617073565b602082019050919050565b6000616228600083616b8d565b91506162338261709c565b600082019050919050565b60608201600082015161625460008501826163da565b50602082015161626760208501826163da565b50604082015161627a6040850182615f57565b50505050565b60608201600082015161629660008501826163da565b5060208201516162a960208501826163da565b5060408201516162bc6040850182615f57565b50505050565b600060808301600083015184820360008601526162df828261632c565b915050602083015184820360208601526162f9828261632c565b915050604083015161630e6040860182615f57565b5060608301516163216060860182615fd3565b508091505092915050565b60006060830160008301518482036000860152616349828261604b565b91505060208301518482036020860152616363828261604b565b91505060408301516163786040860182615f57565b508091505092915050565b600060608301600083015184820360008601526163a0828261604b565b915050602083015184820360208601526163ba828261604b565b91505060408301516163cf6040860182615f57565b508091505092915050565b6163e381616c60565b82525050565b6163f281616c60565b82525050565b61640961640482616c60565b616de8565b82525050565b600061641a8261621b565b9150819050919050565b600061643082846163f8565b60208201915081905092915050565b60006020820190506164546000830184615f66565b92915050565b600060408201905061646f6000830185615f66565b61647c6020830184615f66565b9392505050565b60006060820190506164986000830186615f66565b6164a56020830185615f66565b6164b260408301846163e9565b949350505050565b600060c0820190506164cf6000830189615f66565b6164dc6020830188615f66565b6164e960408301876163e9565b6164f66060830186615f66565b616503608083018561600f565b61651060a08301846163e9565b979650505050505050565b600060c0820190506165306000830189615f66565b61653d6020830188615f66565b61654a60408301876163e9565b6165576060830186615f66565b616564608083018561601e565b61657160a08301846163e9565b979650505050505050565b600060c0820190506165916000830189615f66565b61659e6020830188615f66565b6165ab60408301876163e9565b6165b86060830186615f66565b6165c5608083018561602d565b6165d260a08301846163e9565b979650505050505050565b600060c0820190506165f26000830189615f66565b6165ff6020830188615f66565b61660c60408301876163e9565b6166196060830186615f66565b616626608083018561603c565b61663360a08301846163e9565b979650505050505050565b60006040820190506166536000830185615f66565b616660602083018461600f565b9392505050565b600060408201905061667c6000830185615f66565b61668960208301846163e9565b9392505050565b600060e0820190506166a5600083018a615f66565b6166b260208301896163e9565b6166bf6040830188615f66565b6166cc6060830187615f66565b6166d960808301866163e9565b6166e660a08301856163e9565b6166f360c08301846163e9565b98975050505050505050565b60006060820190506167146000830186615f66565b61672160208301856163e9565b61672e60408301846163e9565b949350505050565b600060208201905081810360008301526167508184615f75565b905092915050565b600060208201905061676d6000830184615fe2565b92915050565b60006020820190506167886000830184615ff1565b92915050565b60006020820190506167a36000830184616000565b92915050565b600060808201905081810360008301526167c38187616084565b90506167d26020830186615f66565b6167df6040830185615f66565b6167ec6060830184615f66565b95945050505050565b600060e082019050818103600083015261680f818a616084565b905081810360208301526168238189616084565b90506168326040830188615f66565b81810360608301526168448187616084565b905081810360808301526168588186616084565b905061686760a0830185615f66565b61687460c0830184615f66565b98975050505050505050565b60006020820190508181036000830152616899816160bd565b9050919050565b600060208201905081810360008301526168b9816160e0565b9050919050565b600060208201905081810360008301526168d981616103565b9050919050565b600060208201905081810360008301526168f981616126565b9050919050565b6000602082019050818103600083015261691981616149565b9050919050565b600060208201905081810360008301526169398161616c565b9050919050565b600060208201905081810360008301526169598161618f565b9050919050565b60006020820190508181036000830152616979816161b2565b9050919050565b60006020820190508181036000830152616999816161d5565b9050919050565b600060208201905081810360008301526169b9816161f8565b9050919050565b60006060820190506169d56000830184616280565b92915050565b600060208201905081810360008301526169f581846162c2565b905092915050565b60006020820190508181036000830152616a178184616383565b905092915050565b6000602082019050616a3460008301846163e9565b92915050565b600060c082019050616a4f60008301896163e9565b616a5c60208301886163e9565b616a6960408301876163e9565b616a7660608301866163e9565b616a8360808301856163e9565b616a9060a08301846163e9565b979650505050505050565b6000616aa5616ab6565b9050616ab18282616d6e565b919050565b6000604051905090565b600067ffffffffffffffff821115616adb57616ada616e50565b5b602082029050602081019050919050565b600067ffffffffffffffff821115616b0757616b06616e50565b5b602082029050602081019050919050565b600067ffffffffffffffff821115616b3357616b32616e50565b5b616b3c82616e7f565b9050602081019050919050565b6000819050602082019050919050565b600081519050919050565b600081519050919050565b6000602082019050919050565b600082825260208201905092915050565b600081905092915050565b600082825260208201905092915050565b600082825260208201905092915050565b6000616bc582616c60565b9150616bd083616c60565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115616c0557616c04616df2565b5b828201905092915050565b6000616c1b82616c40565b9050919050565b60008115159050919050565b6000616c3982616c10565b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000819050919050565b6000616c7582616c7c565b9050919050565b6000616c8782616c40565b9050919050565b6000616c9982616ca0565b9050919050565b6000616cab82616c40565b9050919050565b6000616cbd82616c60565b9050919050565b6000616ccf82616c60565b9050919050565b6000616ce182616c60565b9050919050565b6000616cf382616c60565b9050919050565b82818337600083830152505050565b60005b83811015616d27578082015181840152602081019050616d0c565b83811115616d36576000848401525b50505050565b60006002820490506001821680616d5457607f821691505b60208210811415616d6857616d67616e21565b5b50919050565b616d7782616e7f565b810181811067ffffffffffffffff82111715616d9657616d95616e50565b5b80604052505050565b6000616daa82616c60565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff821415616ddd57616ddc616df2565b5b600182019050919050565b6000819050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6000601f19601f8301169050919050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b7f455243313136373a2063726561746532206661696c6564000000000000000000600082015250565b7f416464726573733a20756e61626c6520746f2073656e642076616c75652c207260008201527f6563697069656e74206d61792068617665207265766572746564000000000000602082015250565b7f416464726573733a20696e73756666696369656e742062616c616e6365000000600082015250565b7f7472616e7366657220616d6f756e74206d757374206c61726765207468616e2060008201527f3000000000000000000000000000000000000000000000000000000000000000602082015250565b7f5468697320746f6b656e207374696c6c206e6f742072656769737465642e0000600082015250565b7f7472616e736665722061646472657373206d757374206e6f7420307830000000600082015250565b7f636f6e7472616374206d75737420686176652063656c6f000000000000000000600082015250565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b7f61756374696f6e3a2077726f6e6720646576656c6f7065720000000000000000600082015250565b50565b6170a881616c10565b81146170b357600080fd5b50565b6170bf81616c22565b81146170ca57600080fd5b50565b6170d681616c2e565b81146170e157600080fd5b50565b6170ed81616c60565b81146170f857600080fd5b5056fea264697066735822122085c5ba6c8aab14ea02e90bd76fd1a5a63d5a368022414dc628b2f7e72f56bc5664736f6c63430008030033