ETH Price: $3,333.28 (-1.23%)

Token

Primate Social Society (PSS)
 

Overview

Max Total Supply

10,000 PSS

Holders

2,387

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A
Filtered by Token Holder
bcaster.eth
Balance
1 PSS
0x4d56cde71a622c6ff74e114844ee53ec0d3418bc
Loading...
Loading
Loading...
Loading
Loading...
Loading

OVERVIEW

The first deflationary collection of Ape Avatars. 2075 burned bringing the total supply down to 7925 (12/18/2021). PSS NFTs are a membership card to the Society, and grant access to members-only benefits including free kitten companions, gorillas, breeding, and future RPG. S...

# Exchange Pair Price  24H Volume % Volume

Contract Source Code Verified (Exact Match)

Contract Name:
PrimateSocialSocietyNFT

Compiler Version
v0.8.7+commit.e28d00a7

Optimization Enabled:
Yes with 999999 runs

Other Settings:
default evmVersion, MIT license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2021-09-18
*/

// Sources flattened with hardhat v2.6.2 https://hardhat.org

// File @openzeppelin/contracts/utils/[email protected]

// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

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

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

// File @openzeppelin/contracts/access/[email protected]

pragma solidity ^0.8.0;

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

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

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

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

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

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

    function _setOwner(address newOwner) private {
        address oldOwner = _owner;
        _owner = newOwner;
        emit OwnershipTransferred(oldOwner, newOwner);
    }
}

// File @openzeppelin/contracts/utils/introspection/[email protected]

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

// File @openzeppelin/contracts/token/ERC721/[email protected]

pragma solidity ^0.8.0;

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

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

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

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

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

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

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

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

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

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

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

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

// File @openzeppelin/contracts/token/ERC721/[email protected]

pragma solidity ^0.8.0;

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

// File @openzeppelin/contracts/token/ERC721/extensions/[email protected]

pragma solidity ^0.8.0;

/**
 * @title ERC-721 Non-Fungible Token Standard, optional metadata extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Metadata is IERC721 {
    /**
     * @dev Returns the token collection name.
     */
    function name() external view returns (string memory);

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

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

// File @openzeppelin/contracts/utils/[email protected]

pragma solidity ^0.8.0;

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

        uint256 size;
        assembly {
            size := extcodesize(account)
        }
        return size > 0;
    }

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

        (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

                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}

// File @openzeppelin/contracts/utils/[email protected]

pragma solidity ^0.8.0;

/**
 * @dev String operations.
 */
library Strings {
    bytes16 private constant _HEX_SYMBOLS = '0123456789abcdef';

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

// File @openzeppelin/contracts/utils/introspection/[email protected]

pragma solidity ^0.8.0;

/**
 * @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 ERC165 is IERC165 {
    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId)
        public
        view
        virtual
        override
        returns (bool)
    {
        return interfaceId == type(IERC165).interfaceId;
    }
}

// File @openzeppelin/contracts/token/ERC721/[email protected]

pragma solidity ^0.8.0;

/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
 * the Metadata extension, but not including the Enumerable extension, which is available separately as
 * {ERC721Enumerable}.
 */
contract ERC721 is Context, ERC165, IERC721, IERC721Metadata {
    using Address for address;
    using Strings for uint256;

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to owner address
    mapping(uint256 => address) private _owners;

    // Mapping owner address to token count
    mapping(address => uint256) private _balances;

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

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

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

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

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

    /**
     * @dev See {IERC721-ownerOf}.
     */
    function ownerOf(uint256 tokenId)
        public
        view
        virtual
        override
        returns (address)
    {
        address owner = _owners[tokenId];
        require(
            owner != address(0),
            'ERC721: owner query for nonexistent token'
        );
        return owner;
    }

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

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

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

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

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

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

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

        _approve(to, tokenId);
    }

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

        return _tokenApprovals[tokenId];
    }

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

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

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

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

        _transfer(from, to, tokenId);
    }

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

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

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

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

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

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

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

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

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

        _balances[to] += 1;
        _owners[tokenId] = to;

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

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

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

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

        _balances[owner] -= 1;
        delete _owners[tokenId];

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

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

        _beforeTokenTransfer(from, to, tokenId);

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

        _balances[from] -= 1;
        _balances[to] += 1;
        _owners[tokenId] = to;

        emit Transfer(from, to, tokenId);
    }

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

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
     * The call is not executed if the target address is not a contract.
     *
     * @param from address representing the previous owner of the given token ID
     * @param to target address that will receive the tokens
     * @param tokenId uint256 ID of the token to be transferred
     * @param _data bytes optional data to send along with the call
     * @return bool whether the call correctly returned the expected magic value
     */
    function _checkOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory _data
    ) private returns (bool) {
        if (to.isContract()) {
            try
                IERC721Receiver(to).onERC721Received(
                    _msgSender(),
                    from,
                    tokenId,
                    _data
                )
            returns (bytes4 retval) {
                return retval == IERC721Receiver.onERC721Received.selector;
            } catch (bytes memory reason) {
                if (reason.length == 0) {
                    revert(
                        'ERC721: transfer to non ERC721Receiver implementer'
                    );
                } else {
                    assembly {
                        revert(add(32, reason), mload(reason))
                    }
                }
            }
        } else {
            return true;
        }
    }

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

// File @openzeppelin/contracts/token/ERC721/extensions/[email protected]

pragma solidity ^0.8.0;

/**
 * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Enumerable is IERC721 {
    /**
     * @dev Returns the total amount of tokens stored by the contract.
     */
    function totalSupply() external view returns (uint256);

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

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

// File @openzeppelin/contracts/token/ERC721/extensions/[email protected]

pragma solidity ^0.8.0;

/**
 * @dev This implements an optional extension of {ERC721} defined in the EIP that adds
 * enumerability of all the token ids in the contract as well as all token ids owned by each
 * account.
 */
abstract contract ERC721Enumerable is ERC721, IERC721Enumerable {
    // Mapping from owner to list of owned token IDs
    mapping(address => mapping(uint256 => uint256)) private _ownedTokens;

    // Mapping from token ID to index of the owner tokens list
    mapping(uint256 => uint256) private _ownedTokensIndex;

    // Array with all token ids, used for enumeration
    uint256[] private _allTokens;

    // Mapping from token id to position in the allTokens array
    mapping(uint256 => uint256) private _allTokensIndex;

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

    /**
     * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index)
        public
        view
        virtual
        override
        returns (uint256)
    {
        require(
            index < ERC721.balanceOf(owner),
            'ERC721Enumerable: owner index out of bounds'
        );
        return _ownedTokens[owner][index];
    }

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

    /**
     * @dev See {IERC721Enumerable-tokenByIndex}.
     */
    function tokenByIndex(uint256 index)
        public
        view
        virtual
        override
        returns (uint256)
    {
        require(
            index < ERC721Enumerable.totalSupply(),
            'ERC721Enumerable: global index out of bounds'
        );
        return _allTokens[index];
    }

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

        if (from == address(0)) {
            _addTokenToAllTokensEnumeration(tokenId);
        } else if (from != to) {
            _removeTokenFromOwnerEnumeration(from, tokenId);
        }
        if (to == address(0)) {
            _removeTokenFromAllTokensEnumeration(tokenId);
        } else if (to != from) {
            _addTokenToOwnerEnumeration(to, tokenId);
        }
    }

    /**
     * @dev Private function to add a token to this extension's ownership-tracking data structures.
     * @param to address representing the new owner of the given token ID
     * @param tokenId uint256 ID of the token to be added to the tokens list of the given address
     */
    function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private {
        uint256 length = ERC721.balanceOf(to);
        _ownedTokens[to][length] = tokenId;
        _ownedTokensIndex[tokenId] = length;
    }

    /**
     * @dev Private function to add a token to this extension's token tracking data structures.
     * @param tokenId uint256 ID of the token to be added to the tokens list
     */
    function _addTokenToAllTokensEnumeration(uint256 tokenId) private {
        _allTokensIndex[tokenId] = _allTokens.length;
        _allTokens.push(tokenId);
    }

    /**
     * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that
     * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for
     * gas optimizations e.g. when performing a transfer operation (avoiding double writes).
     * This has O(1) time complexity, but alters the order of the _ownedTokens array.
     * @param from address representing the previous owner of the given token ID
     * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address
     */
    function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId)
        private
    {
        // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and
        // then delete the last slot (swap and pop).

        uint256 lastTokenIndex = ERC721.balanceOf(from) - 1;
        uint256 tokenIndex = _ownedTokensIndex[tokenId];

        // When the token to delete is the last token, the swap operation is unnecessary
        if (tokenIndex != lastTokenIndex) {
            uint256 lastTokenId = _ownedTokens[from][lastTokenIndex];

            _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
            _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
        }

        // This also deletes the contents at the last position of the array
        delete _ownedTokensIndex[tokenId];
        delete _ownedTokens[from][lastTokenIndex];
    }

    /**
     * @dev Private function to remove a token from this extension's token tracking data structures.
     * This has O(1) time complexity, but alters the order of the _allTokens array.
     * @param tokenId uint256 ID of the token to be removed from the tokens list
     */
    function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private {
        // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and
        // then delete the last slot (swap and pop).

        uint256 lastTokenIndex = _allTokens.length - 1;
        uint256 tokenIndex = _allTokensIndex[tokenId];

        // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so
        // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding
        // an 'if' statement (like in _removeTokenFromOwnerEnumeration)
        uint256 lastTokenId = _allTokens[lastTokenIndex];

        _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
        _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index

        // This also deletes the contents at the last position of the array
        delete _allTokensIndex[tokenId];
        _allTokens.pop();
    }
}

// File contracts/OpenSeaTradableNFT.sol

pragma solidity ^0.8.0;

contract OpenSeaOwnableDelegateProxy {}

contract OpenSeaProxyRegistry {
    mapping(address => OpenSeaOwnableDelegateProxy) public proxies;
}

contract OpenSeaTradableNFT {
    address openSeaProxyRegistryAddress;

    function _setProxyRegistryAddress(address _openSeaProxyRegistryAddress)
        internal
    {
        openSeaProxyRegistryAddress = _openSeaProxyRegistryAddress;
    }

    function isApprovedForAll(address owner, address operator)
        public
        view
        virtual
        returns (bool)
    {
        // Whitelist OpenSea proxy contract for easy trading.
        OpenSeaProxyRegistry openSeaProxyRegistry = OpenSeaProxyRegistry(
            openSeaProxyRegistryAddress
        );
        if (address(openSeaProxyRegistry.proxies(owner)) == operator) {
            return true;
        }

        return false;
    }
}

// File @openzeppelin/contracts/utils/math/[email protected]

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 no longer needed starting with Solidity 0.8. 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 substraction 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;
        }
    }
}

// File contracts/PrimateSocialSocietyNFT.sol

pragma solidity ^0.8.0;

contract PrimateSocialSocietyNFT is
    Ownable,
    ERC721Enumerable,
    OpenSeaTradableNFT
{
    using SafeMath for uint256;

    string public baseURI = '';

    uint256 public price = 0.06 ether;
    uint256 public preSaleBuyInPrice = 0.05 ether;

    uint256 public MAX_PRIMATES = 10000;
    uint256 public MAX_PRIMATE_PURCHASE = 20;
    uint256 public MAX_PRESALE_PRIMATE_PURCHASE = 10;
    uint256 public PURCHASEABLE_PRESALES = 250;
    uint256 public PRIMATE_RESERVE = 500;

    bool public preSaleIsActive = false;
    bool public saleIsActive = false;

    mapping(address => bool) public whitelist;
    mapping(address => uint256) public presalePurchases;

    constructor(
        string memory _initialBaseURI,
        address _openSeaProxyRegistryAddress
    ) ERC721('Primate Social Society', 'PSS') {
        baseURI = _initialBaseURI;
        OpenSeaTradableNFT._setProxyRegistryAddress(
            _openSeaProxyRegistryAddress
        );
    }

    function isApprovedForAll(address owner, address operator)
        public
        view
        override(ERC721, OpenSeaTradableNFT)
        returns (bool)
    {
        if (OpenSeaTradableNFT.isApprovedForAll(owner, operator)) {
            return true;
        }

        return ERC721.isApprovedForAll(owner, operator);
    }

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

    function maxAvailableForSell() public view returns (uint256) {
        return MAX_PRIMATES.sub(PRIMATE_RESERVE).sub(totalSupply());
    }

    function mintPreSalePrimates(uint256 numberOfTokens) external payable {
        require(whitelist[msg.sender], 'Whitelist required for pre sale');
        require(preSaleIsActive, 'Pre sale must be active to mint Primates');
        require(
            presalePurchases[msg.sender].add(numberOfTokens) <=
                MAX_PRESALE_PRIMATE_PURCHASE,
            'Can only mint MAX_PRESALE_PRIMATE_PURCHASE tokens during the presale'
        );
        require(
            msg.value >= price.mul(numberOfTokens),
            'Ether value sent is not correct'
        );

        presalePurchases[msg.sender] = presalePurchases[msg.sender].add(
            numberOfTokens
        );

        for (uint256 i = 0; i < numberOfTokens; i++) {
            uint256 mintIndex = totalSupply();
            if (maxAvailableForSell() > 0) {
                _safeMint(msg.sender, mintIndex);
            }
        }
    }

    function mintPrimates(uint256 numberOfTokens) external payable {
        require(saleIsActive, 'Sale must be active to mint Primates');
        require(
            numberOfTokens <= MAX_PRIMATE_PURCHASE,
            'Can only mint MAX_PRIMATE_PURCHASE tokens at a time'
        );
        require(
            numberOfTokens <= maxAvailableForSell(),
            'Purchase would exceed max supply of Primates'
        );
        require(
            msg.value >= price.mul(numberOfTokens),
            'Ether value sent is not correct'
        );

        for (uint256 i = 0; i < numberOfTokens; i++) {
            uint256 mintIndex = totalSupply();
            if (maxAvailableForSell() > 0) {
                _safeMint(msg.sender, mintIndex);
            }
        }
    }

    function buyIntoPreSale() external payable {
        require(!whitelist[msg.sender], 'Already in whitelist');
        require(!saleIsActive, 'Regular sale is already active');
        require(PURCHASEABLE_PRESALES > 0, 'Pre sale sold out');
        require(
            msg.value >= preSaleBuyInPrice,
            'Ether value sent is not correct'
        );

        PURCHASEABLE_PRESALES = PURCHASEABLE_PRESALES - 1;
        whitelist[msg.sender] = true;
    }

    function whitelistAddresses(address[] memory addresses) external onlyOwner {
        for (uint256 i = 0; i < addresses.length; i++) {
            whitelist[addresses[i]] = true;
        }
    }

    function removeWhitelistAddresses(address[] memory addresses)
        external
        onlyOwner
    {
        for (uint256 i = 0; i < addresses.length; i++) {
            whitelist[addresses[i]] = false;
        }
    }

    function withdraw() external onlyOwner {
        uint256 balance = address(this).balance;
        payable(msg.sender).transfer(balance);
    }

    function flipPreSaleState() external onlyOwner {
        preSaleIsActive = !preSaleIsActive;
    }

    function flipSaleState() external onlyOwner {
        saleIsActive = !saleIsActive;
    }

    function setPrice(uint256 _price) external onlyOwner {
        price = _price;
    }

    function setPreSaleBuyInPrice(uint256 _price) external onlyOwner {
        preSaleBuyInPrice = _price;
    }

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

    function reservePrimates(uint256 numberOfTokens) external onlyOwner {
        require(
            numberOfTokens > 0 && numberOfTokens <= PRIMATE_RESERVE,
            'Not enough reserve left for team'
        );

        for (uint256 i = 0; i < numberOfTokens; i++) {
            uint256 mintIndex = totalSupply();
            _safeMint(msg.sender, mintIndex);
        }

        PRIMATE_RESERVE = PRIMATE_RESERVE.sub(numberOfTokens);
    }
}

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"string","name":"_initialBaseURI","type":"string"},{"internalType":"address","name":"_openSeaProxyRegistryAddress","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"MAX_PRESALE_PRIMATE_PURCHASE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_PRIMATES","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_PRIMATE_PURCHASE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PRIMATE_RESERVE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PURCHASEABLE_PRESALES","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"buyIntoPreSale","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"flipPreSaleState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"flipSaleState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxAvailableForSell","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfTokens","type":"uint256"}],"name":"mintPreSalePrimates","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfTokens","type":"uint256"}],"name":"mintPrimates","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"preSaleBuyInPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"preSaleIsActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"presalePurchases","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"price","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"addresses","type":"address[]"}],"name":"removeWhitelistAddresses","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfTokens","type":"uint256"}],"name":"reservePrimates","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"saleIsActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_newBaseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"setPreSaleBuyInPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"}],"name":"setPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"whitelist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"addresses","type":"address[]"}],"name":"whitelistAddresses","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]

60a06040819052600060808190526200001b91600c91620001cc565b5066d529ae9e860000600d5566b1a2bc2ec50000600e55612710600f5560146010819055600a60115560fa6012556101f4601355805461ffff191690553480156200006557600080fd5b5060405162003f2038038062003f2083398101604081905262000088916200028f565b6040518060400160405280601681526020017f5072696d61746520536f6369616c20536f6369657479000000000000000000008152506040518060400160405280600381526020016250535360e81b815250620000f4620000ee6200015660201b60201c565b6200015a565b815162000109906001906020850190620001cc565b5080516200011f906002906020840190620001cc565b50508251620001379150600c906020850190620001cc565b506200014e81620001aa60201b620023461760201c565b5050620003d3565b3390565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b600b80546001600160a01b0319166001600160a01b0392909216919091179055565b828054620001da9062000380565b90600052602060002090601f016020900481019282620001fe576000855562000249565b82601f106200021957805160ff191683800117855562000249565b8280016001018555821562000249579182015b82811115620002495782518255916020019190600101906200022c565b50620002579291506200025b565b5090565b5b808211156200025757600081556001016200025c565b80516001600160a01b03811681146200028a57600080fd5b919050565b60008060408385031215620002a357600080fd5b82516001600160401b0380821115620002bb57600080fd5b818501915085601f830112620002d057600080fd5b815181811115620002e557620002e5620003bd565b604051601f8201601f19908116603f01168101908382118183101715620003105762000310620003bd565b816040528281526020935088848487010111156200032d57600080fd5b600091505b8282101562000351578482018401518183018501529083019062000332565b82821115620003635760008484830101525b95506200037591505085820162000272565b925050509250929050565b600181811c908216806200039557607f821691505b60208210811415620003b757634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052604160045260246000fd5b613b3d80620003e36000396000f3fe6080604052600436106102e75760003560e01c806391b7f5ed11610184578063b88d4fde116100d6578063eb60962c1161008a578063f1a1d39111610064578063f1a1d391146107af578063f2fde38b146107cf578063fa60d42a146107ef57600080fd5b8063eb60962c14610765578063eb8d24441461077b578063f03255491461079a57600080fd5b8063ccd53e97116100bb578063ccd53e971461071c578063e92e32a61461072f578063e985e9c51461074557600080fd5b8063b88d4fde146106dc578063c87b56dd146106fc57600080fd5b8063a113482c11610138578063a68574d311610112578063a68574d31461066f578063aa2652de1461069c578063b83921a6146106bc57600080fd5b8063a113482c14610626578063a22cb46514610639578063a4af09411461065957600080fd5b80639a843c70116101695780639a843c70146105ca5780639b19251a146105e0578063a035b1fe1461061057600080fd5b806391b7f5ed1461059557806395d89b41146105b557600080fd5b80633ccfd60b1161023d5780636352211e116101f1578063715018a6116101cb578063715018a61461053f5780638da5cb5b146105545780638f18ab061461057f57600080fd5b80636352211e146104ea5780636c0360eb1461050a57806370a082311461051f57600080fd5b80634eee3c41116102225780634eee3c41146104955780634f6ccce7146104aa57806355f804b3146104ca57600080fd5b80633ccfd60b1461046057806342842e0e1461047557600080fd5b806318160ddd1161029f5780632bf04304116102795780632bf043041461040b5780632f745c591461042b57806334918dfd1461044b57600080fd5b806318160ddd146103b25780631f0234d8146103d157806323b872dd146103eb57600080fd5b8063081812fc116102d0578063081812fc14610343578063095ea7b3146103885780631057f4f4146103aa57600080fd5b806301ffc9a7146102ec57806306fdde0314610321575b600080fd5b3480156102f857600080fd5b5061030c6103073660046136a2565b610805565b60405190151581526020015b60405180910390f35b34801561032d57600080fd5b50610336610861565b604051610318919061381d565b34801561034f57600080fd5b5061036361035e366004613742565b6108f3565b60405173ffffffffffffffffffffffffffffffffffffffff9091168152602001610318565b34801561039457600080fd5b506103a86103a33660046135bd565b6109d2565b005b6103a8610b5f565b3480156103be57600080fd5b506009545b604051908152602001610318565b3480156103dd57600080fd5b5060145461030c9060ff1681565b3480156103f757600080fd5b506103a86104063660046134c9565b610d6f565b34801561041757600080fd5b506103a86104263660046135e9565b610e10565b34801561043757600080fd5b506103c36104463660046135bd565b610f28565b34801561045757600080fd5b506103a8610ff7565b34801561046c57600080fd5b506103a86110b2565b34801561048157600080fd5b506103a86104903660046134c9565b611162565b3480156104a157600080fd5b506103c361117d565b3480156104b657600080fd5b506103c36104c5366004613742565b6111a5565b3480156104d657600080fd5b506103a86104e53660046136f9565b611263565b3480156104f657600080fd5b50610363610505366004613742565b6112f7565b34801561051657600080fd5b506103366113a9565b34801561052b57600080fd5b506103c361053a366004613473565b611437565b34801561054b57600080fd5b506103a8611505565b34801561056057600080fd5b5060005473ffffffffffffffffffffffffffffffffffffffff16610363565b34801561058b57600080fd5b506103c360105481565b3480156105a157600080fd5b506103a86105b0366004613742565b611592565b3480156105c157600080fd5b50610336611618565b3480156105d657600080fd5b506103c360125481565b3480156105ec57600080fd5b5061030c6105fb366004613473565b60156020526000908152604090205460ff1681565b34801561061c57600080fd5b506103c3600d5481565b6103a8610634366004613742565b611627565b34801561064557600080fd5b506103a861065436600461358a565b6118a1565b34801561066557600080fd5b506103c3600e5481565b34801561067b57600080fd5b506103c361068a366004613473565b60166020526000908152604090205481565b3480156106a857600080fd5b506103a86106b7366004613742565b6119b8565b3480156106c857600080fd5b506103a86106d73660046135e9565b611afa565b3480156106e857600080fd5b506103a86106f736600461350a565b611c0e565b34801561070857600080fd5b50610336610717366004613742565b611cb6565b6103a861072a366004613742565b611dc6565b34801561073b57600080fd5b506103c3600f5481565b34801561075157600080fd5b5061030c610760366004613490565b612089565b34801561077157600080fd5b506103c360115481565b34801561078757600080fd5b5060145461030c90610100900460ff1681565b3480156107a657600080fd5b506103a86120dd565b3480156107bb57600080fd5b506103a86107ca366004613742565b612190565b3480156107db57600080fd5b506103a86107ea366004613473565b612216565b3480156107fb57600080fd5b506103c360135481565b60007fffffffff0000000000000000000000000000000000000000000000000000000082167f780e9d6300000000000000000000000000000000000000000000000000000000148061085b575061085b8261238d565b92915050565b6060600180546108709061392b565b80601f016020809104026020016040519081016040528092919081815260200182805461089c9061392b565b80156108e95780601f106108be576101008083540402835291602001916108e9565b820191906000526020600020905b8154815290600101906020018083116108cc57829003601f168201915b5050505050905090565b60008181526003602052604081205473ffffffffffffffffffffffffffffffffffffffff166109a9576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201527f697374656e7420746f6b656e000000000000000000000000000000000000000060648201526084015b60405180910390fd5b5060009081526005602052604090205473ffffffffffffffffffffffffffffffffffffffff1690565b60006109dd826112f7565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610a9b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e6560448201527f720000000000000000000000000000000000000000000000000000000000000060648201526084016109a0565b3373ffffffffffffffffffffffffffffffffffffffff82161480610ac45750610ac48133612089565b610b50576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c000000000000000060648201526084016109a0565b610b5a8383612470565b505050565b3360009081526015602052604090205460ff1615610bd9576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f416c726561647920696e2077686974656c69737400000000000000000000000060448201526064016109a0565b601454610100900460ff1615610c4b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601e60248201527f526567756c61722073616c6520697320616c726561647920616374697665000060448201526064016109a0565b600060125411610cb7576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601160248201527f5072652073616c6520736f6c64206f757400000000000000000000000000000060448201526064016109a0565b600e54341015610d23576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f45746865722076616c75652073656e74206973206e6f7420636f72726563740060448201526064016109a0565b6001601254610d3291906138e8565b60125533600090815260156020526040902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00166001179055565b610d793382612510565b610e05576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016109a0565b610b5a838383612653565b60005473ffffffffffffffffffffffffffffffffffffffff163314610e91576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b60005b8151811015610f2457600160156000848481518110610eb557610eb5613a59565b60209081029190910181015173ffffffffffffffffffffffffffffffffffffffff16825281019190915260400160002080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001691151591909117905580610f1c8161397f565b915050610e94565b5050565b6000610f3383611437565b8210610fc1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201527f74206f6620626f756e647300000000000000000000000000000000000000000060648201526084016109a0565b5073ffffffffffffffffffffffffffffffffffffffff919091166000908152600760209081526040808320938352929052205490565b60005473ffffffffffffffffffffffffffffffffffffffff163314611078576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b601480547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff81166101009182900460ff1615909102179055565b60005473ffffffffffffffffffffffffffffffffffffffff163314611133576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b6040514790339082156108fc029083906000818181858888f19350505050158015610f24573d6000803e3d6000fd5b610b5a83838360405180602001604052806000815250611c0e565b60006111a061118b60095490565b601354600f5461119a916128c5565b906128c5565b905090565b60006111b060095490565b821061123e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201527f7574206f6620626f756e6473000000000000000000000000000000000000000060648201526084016109a0565b6009828154811061125157611251613a59565b90600052602060002001549050919050565b60005473ffffffffffffffffffffffffffffffffffffffff1633146112e4576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b8051610f2490600c906020840190613364565b60008181526003602052604081205473ffffffffffffffffffffffffffffffffffffffff168061085b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201527f656e7420746f6b656e000000000000000000000000000000000000000000000060648201526084016109a0565b600c80546113b69061392b565b80601f01602080910402602001604051908101604052809291908181526020018280546113e29061392b565b801561142f5780601f106114045761010080835404028352916020019161142f565b820191906000526020600020905b81548152906001019060200180831161141257829003601f168201915b505050505081565b600073ffffffffffffffffffffffffffffffffffffffff82166114dc576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a6560448201527f726f20616464726573730000000000000000000000000000000000000000000060648201526084016109a0565b5073ffffffffffffffffffffffffffffffffffffffff1660009081526004602052604090205490565b60005473ffffffffffffffffffffffffffffffffffffffff163314611586576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b61159060006128d1565b565b60005473ffffffffffffffffffffffffffffffffffffffff163314611613576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b600d55565b6060600280546108709061392b565b601454610100900460ff166116bd576040517f08c379a0000000000000000000000000000000000000000000000000000000008152602060048201526024808201527f53616c65206d7573742062652061637469766520746f206d696e74205072696d60448201527f617465730000000000000000000000000000000000000000000000000000000060648201526084016109a0565b60105481111561174f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603360248201527f43616e206f6e6c79206d696e74204d41585f5052494d4154455f50555243484160448201527f534520746f6b656e7320617420612074696d650000000000000000000000000060648201526084016109a0565b61175761117d565b8111156117e6576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f507572636861736520776f756c6420657863656564206d617820737570706c7960448201527f206f66205072696d61746573000000000000000000000000000000000000000060648201526084016109a0565b600d546117f39082612946565b34101561185c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f45746865722076616c75652073656e74206973206e6f7420636f72726563740060448201526064016109a0565b60005b81811015610f2457600061187260095490565b9050600061187e61117d565b111561188e5761188e3382612952565b50806118998161397f565b91505061185f565b73ffffffffffffffffffffffffffffffffffffffff8216331415611921576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c65720000000000000060448201526064016109a0565b33600081815260066020908152604080832073ffffffffffffffffffffffffffffffffffffffff87168085529083529281902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b60005473ffffffffffffffffffffffffffffffffffffffff163314611a39576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b600081118015611a4b57506013548111155b611ab1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4e6f7420656e6f7567682072657365727665206c65667420666f72207465616d60448201526064016109a0565b60005b81811015611ae6576000611ac760095490565b9050611ad33382612952565b5080611ade8161397f565b915050611ab4565b50601354611af490826128c5565b60135550565b60005473ffffffffffffffffffffffffffffffffffffffff163314611b7b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b60005b8151811015610f2457600060156000848481518110611b9f57611b9f613a59565b60209081029190910181015173ffffffffffffffffffffffffffffffffffffffff16825281019190915260400160002080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001691151591909117905580611c068161397f565b915050611b7e565b611c183383612510565b611ca4576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016109a0565b611cb08484848461296c565b50505050565b60008181526003602052604090205460609073ffffffffffffffffffffffffffffffffffffffff16611d6a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602f60248201527f4552433732314d657461646174613a2055524920717565727920666f72206e6f60448201527f6e6578697374656e7420746f6b656e000000000000000000000000000000000060648201526084016109a0565b6000611d74612a0f565b90506000815111611d945760405180602001604052806000815250611dbf565b80611d9e84612a1e565b604051602001611daf9291906137a5565b6040516020818303038152906040525b9392505050565b3360009081526015602052604090205460ff16611e3f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f57686974656c69737420726571756972656420666f72207072652073616c650060448201526064016109a0565b60145460ff16611ed1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602860248201527f5072652073616c65206d7573742062652061637469766520746f206d696e742060448201527f5072696d6174657300000000000000000000000000000000000000000000000060648201526084016109a0565b60115433600090815260166020526040902054611eee9083612b50565b1115611fa3576040517f08c379a0000000000000000000000000000000000000000000000000000000008152602060048201526044602482018190527f43616e206f6e6c79206d696e74204d41585f50524553414c455f5052494d4154908201527f455f505552434841534520746f6b656e7320647572696e67207468652070726560648201527f73616c6500000000000000000000000000000000000000000000000000000000608482015260a4016109a0565b600d54611fb09082612946565b341015612019576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f45746865722076616c75652073656e74206973206e6f7420636f72726563740060448201526064016109a0565b336000908152601660205260409020546120339082612b50565b336000908152601660205260408120919091555b81811015610f2457600061205a60095490565b9050600061206661117d565b1115612076576120763382612952565b50806120818161397f565b915050612047565b60006120958383612b5c565b156120a25750600161085b565b73ffffffffffffffffffffffffffffffffffffffff80841660009081526006602090815260408083209386168352929052205460ff16611dbf565b60005473ffffffffffffffffffffffffffffffffffffffff16331461215e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b601480547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00811660ff90911615179055565b60005473ffffffffffffffffffffffffffffffffffffffff163314612211576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b600e55565b60005473ffffffffffffffffffffffffffffffffffffffff163314612297576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b73ffffffffffffffffffffffffffffffffffffffff811661233a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016109a0565b612343816128d1565b50565b600b80547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b60007fffffffff0000000000000000000000000000000000000000000000000000000082167f80ac58cd00000000000000000000000000000000000000000000000000000000148061242057507fffffffff0000000000000000000000000000000000000000000000000000000082167f5b5e139f00000000000000000000000000000000000000000000000000000000145b8061085b57507f01ffc9a7000000000000000000000000000000000000000000000000000000007fffffffff0000000000000000000000000000000000000000000000000000000083161461085b565b600081815260056020526040902080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff841690811790915581906124ca826112f7565b73ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b60008181526003602052604081205473ffffffffffffffffffffffffffffffffffffffff166125c1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201527f697374656e7420746f6b656e000000000000000000000000000000000000000060648201526084016109a0565b60006125cc836112f7565b90508073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff16148061263b57508373ffffffffffffffffffffffffffffffffffffffff16612623846108f3565b73ffffffffffffffffffffffffffffffffffffffff16145b8061264b575061264b8185612089565b949350505050565b8273ffffffffffffffffffffffffffffffffffffffff16612673826112f7565b73ffffffffffffffffffffffffffffffffffffffff1614612716576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201527f73206e6f74206f776e000000000000000000000000000000000000000000000060648201526084016109a0565b73ffffffffffffffffffffffffffffffffffffffff82166127b8576040517f08c379a0000000000000000000000000000000000000000000000000000000008152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f2061646460448201527f726573730000000000000000000000000000000000000000000000000000000060648201526084016109a0565b6127c3838383612c37565b6127ce600082612470565b73ffffffffffffffffffffffffffffffffffffffff831660009081526004602052604081208054600192906128049084906138e8565b909155505073ffffffffffffffffffffffffffffffffffffffff8216600090815260046020526040812080546001929061283f90849061387f565b909155505060008181526003602052604080822080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff86811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b6000611dbf82846138e8565b6000805473ffffffffffffffffffffffffffffffffffffffff8381167fffffffffffffffffffffffff0000000000000000000000000000000000000000831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b6000611dbf82846138ab565b610f24828260405180602001604052806000815250612d3d565b612977848484612653565b61298384848484612de0565b611cb0576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016109a0565b6060600c80546108709061392b565b606081612a5e57505060408051808201909152600181527f3000000000000000000000000000000000000000000000000000000000000000602082015290565b8160005b8115612a885780612a728161397f565b9150612a819050600a83613897565b9150612a62565b60008167ffffffffffffffff811115612aa357612aa3613a88565b6040519080825280601f01601f191660200182016040528015612acd576020820181803683370190505b5090505b841561264b57612ae26001836138e8565b9150612aef600a866139b8565b612afa90603061387f565b60f81b818381518110612b0f57612b0f613a59565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350612b49600a86613897565b9450612ad1565b6000611dbf828461387f565b600b546040517fc455279100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff848116600483015260009281169190841690829063c45527919060240160206040518083038186803b158015612bcf57600080fd5b505afa158015612be3573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612c0791906136dc565b73ffffffffffffffffffffffffffffffffffffffff161415612c2d57600191505061085b565b5060009392505050565b73ffffffffffffffffffffffffffffffffffffffff8316612c9f57612c9a81600980546000838152600a60205260408120829055600182018355919091527f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af0155565b612cdc565b8173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1614612cdc57612cdc8382612fdf565b73ffffffffffffffffffffffffffffffffffffffff8216612d0057610b5a81613096565b8273ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614610b5a57610b5a8282613145565b612d478383613196565b612d546000848484612de0565b610b5a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016109a0565b600073ffffffffffffffffffffffffffffffffffffffff84163b15612fd4576040517f150b7a0200000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff85169063150b7a0290612e579033908990889088906004016137d4565b602060405180830381600087803b158015612e7157600080fd5b505af1925050508015612ebf575060408051601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0168201909252612ebc918101906136bf565b60015b612f89573d808015612eed576040519150601f19603f3d011682016040523d82523d6000602084013e612ef2565b606091505b508051612f81576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016109a0565b805181602001fd5b7fffffffff00000000000000000000000000000000000000000000000000000000167f150b7a020000000000000000000000000000000000000000000000000000000014905061264b565b506001949350505050565b60006001612fec84611437565b612ff691906138e8565b6000838152600860205260409020549091508082146130565773ffffffffffffffffffffffffffffffffffffffff841660009081526007602090815260408083208584528252808320548484528184208190558352600890915290208190555b50600091825260086020908152604080842084905573ffffffffffffffffffffffffffffffffffffffff9094168352600781528383209183525290812055565b6009546000906130a8906001906138e8565b6000838152600a6020526040812054600980549394509092849081106130d0576130d0613a59565b9060005260206000200154905080600983815481106130f1576130f1613a59565b6000918252602080832090910192909255828152600a9091526040808220849055858252812055600980548061312957613129613a2a565b6001900381819060005260206000200160009055905550505050565b600061315083611437565b73ffffffffffffffffffffffffffffffffffffffff9093166000908152600760209081526040808320868452825280832085905593825260089052919091209190915550565b73ffffffffffffffffffffffffffffffffffffffff8216613213576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f206164647265737360448201526064016109a0565b60008181526003602052604090205473ffffffffffffffffffffffffffffffffffffffff161561329f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e7465640000000060448201526064016109a0565b6132ab60008383612c37565b73ffffffffffffffffffffffffffffffffffffffff821660009081526004602052604081208054600192906132e190849061387f565b909155505060008181526003602052604080822080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b8280546133709061392b565b90600052602060002090601f01602090048101928261339257600085556133d8565b82601f106133ab57805160ff19168380011785556133d8565b828001600101855582156133d8579182015b828111156133d85782518255916020019190600101906133bd565b506133e49291506133e8565b5090565b5b808211156133e457600081556001016133e9565b600067ffffffffffffffff83111561341757613417613a88565b61344860207fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f86011601613830565b905082815283838301111561345c57600080fd5b828260208301376000602084830101529392505050565b60006020828403121561348557600080fd5b8135611dbf81613ab7565b600080604083850312156134a357600080fd5b82356134ae81613ab7565b915060208301356134be81613ab7565b809150509250929050565b6000806000606084860312156134de57600080fd5b83356134e981613ab7565b925060208401356134f981613ab7565b929592945050506040919091013590565b6000806000806080858703121561352057600080fd5b843561352b81613ab7565b9350602085013561353b81613ab7565b925060408501359150606085013567ffffffffffffffff81111561355e57600080fd5b8501601f8101871361356f57600080fd5b61357e878235602084016133fd565b91505092959194509250565b6000806040838503121561359d57600080fd5b82356135a881613ab7565b9150602083013580151581146134be57600080fd5b600080604083850312156135d057600080fd5b82356135db81613ab7565b946020939093013593505050565b600060208083850312156135fc57600080fd5b823567ffffffffffffffff8082111561361457600080fd5b818501915085601f83011261362857600080fd5b81358181111561363a5761363a613a88565b8060051b915061364b848301613830565b8181528481019084860184860187018a101561366657600080fd5b600095505b83861015613695578035945061368085613ab7565b8483526001959095019491860191860161366b565b5098975050505050505050565b6000602082840312156136b457600080fd5b8135611dbf81613ad9565b6000602082840312156136d157600080fd5b8151611dbf81613ad9565b6000602082840312156136ee57600080fd5b8151611dbf81613ab7565b60006020828403121561370b57600080fd5b813567ffffffffffffffff81111561372257600080fd5b8201601f8101841361373357600080fd5b61264b848235602084016133fd565b60006020828403121561375457600080fd5b5035919050565b600081518084526137738160208601602086016138ff565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0169290920160200192915050565b600083516137b78184602088016138ff565b8351908301906137cb8183602088016138ff565b01949350505050565b600073ffffffffffffffffffffffffffffffffffffffff808716835280861660208401525083604083015260806060830152613813608083018461375b565b9695505050505050565b602081526000611dbf602083018461375b565b604051601f82017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016810167ffffffffffffffff8111828210171561387757613877613a88565b604052919050565b60008219821115613892576138926139cc565b500190565b6000826138a6576138a66139fb565b500490565b6000817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff04831182151516156138e3576138e36139cc565b500290565b6000828210156138fa576138fa6139cc565b500390565b60005b8381101561391a578181015183820152602001613902565b83811115611cb05750506000910152565b600181811c9082168061393f57607f821691505b60208210811415613979577f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b50919050565b60007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8214156139b1576139b16139cc565b5060010190565b6000826139c7576139c76139fb565b500690565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b73ffffffffffffffffffffffffffffffffffffffff8116811461234357600080fd5b7fffffffff000000000000000000000000000000000000000000000000000000008116811461234357600080fdfea26469706673582212202f5e99d7387fb45f3f3c9871e5b1b1b4871095d87c791d7859dfd95b5aa1c64764736f6c634300080700330000000000000000000000000000000000000000000000000000000000000040000000000000000000000000a5409ec958c83c3f309868babaca7c86dcb077c1000000000000000000000000000000000000000000000000000000000000002c68747470733a2f2f6170692e7072696d617465736f6369616c736f63696574792e636f6d2f746f6b656e732f0000000000000000000000000000000000000000

Deployed Bytecode

0x6080604052600436106102e75760003560e01c806391b7f5ed11610184578063b88d4fde116100d6578063eb60962c1161008a578063f1a1d39111610064578063f1a1d391146107af578063f2fde38b146107cf578063fa60d42a146107ef57600080fd5b8063eb60962c14610765578063eb8d24441461077b578063f03255491461079a57600080fd5b8063ccd53e97116100bb578063ccd53e971461071c578063e92e32a61461072f578063e985e9c51461074557600080fd5b8063b88d4fde146106dc578063c87b56dd146106fc57600080fd5b8063a113482c11610138578063a68574d311610112578063a68574d31461066f578063aa2652de1461069c578063b83921a6146106bc57600080fd5b8063a113482c14610626578063a22cb46514610639578063a4af09411461065957600080fd5b80639a843c70116101695780639a843c70146105ca5780639b19251a146105e0578063a035b1fe1461061057600080fd5b806391b7f5ed1461059557806395d89b41146105b557600080fd5b80633ccfd60b1161023d5780636352211e116101f1578063715018a6116101cb578063715018a61461053f5780638da5cb5b146105545780638f18ab061461057f57600080fd5b80636352211e146104ea5780636c0360eb1461050a57806370a082311461051f57600080fd5b80634eee3c41116102225780634eee3c41146104955780634f6ccce7146104aa57806355f804b3146104ca57600080fd5b80633ccfd60b1461046057806342842e0e1461047557600080fd5b806318160ddd1161029f5780632bf04304116102795780632bf043041461040b5780632f745c591461042b57806334918dfd1461044b57600080fd5b806318160ddd146103b25780631f0234d8146103d157806323b872dd146103eb57600080fd5b8063081812fc116102d0578063081812fc14610343578063095ea7b3146103885780631057f4f4146103aa57600080fd5b806301ffc9a7146102ec57806306fdde0314610321575b600080fd5b3480156102f857600080fd5b5061030c6103073660046136a2565b610805565b60405190151581526020015b60405180910390f35b34801561032d57600080fd5b50610336610861565b604051610318919061381d565b34801561034f57600080fd5b5061036361035e366004613742565b6108f3565b60405173ffffffffffffffffffffffffffffffffffffffff9091168152602001610318565b34801561039457600080fd5b506103a86103a33660046135bd565b6109d2565b005b6103a8610b5f565b3480156103be57600080fd5b506009545b604051908152602001610318565b3480156103dd57600080fd5b5060145461030c9060ff1681565b3480156103f757600080fd5b506103a86104063660046134c9565b610d6f565b34801561041757600080fd5b506103a86104263660046135e9565b610e10565b34801561043757600080fd5b506103c36104463660046135bd565b610f28565b34801561045757600080fd5b506103a8610ff7565b34801561046c57600080fd5b506103a86110b2565b34801561048157600080fd5b506103a86104903660046134c9565b611162565b3480156104a157600080fd5b506103c361117d565b3480156104b657600080fd5b506103c36104c5366004613742565b6111a5565b3480156104d657600080fd5b506103a86104e53660046136f9565b611263565b3480156104f657600080fd5b50610363610505366004613742565b6112f7565b34801561051657600080fd5b506103366113a9565b34801561052b57600080fd5b506103c361053a366004613473565b611437565b34801561054b57600080fd5b506103a8611505565b34801561056057600080fd5b5060005473ffffffffffffffffffffffffffffffffffffffff16610363565b34801561058b57600080fd5b506103c360105481565b3480156105a157600080fd5b506103a86105b0366004613742565b611592565b3480156105c157600080fd5b50610336611618565b3480156105d657600080fd5b506103c360125481565b3480156105ec57600080fd5b5061030c6105fb366004613473565b60156020526000908152604090205460ff1681565b34801561061c57600080fd5b506103c3600d5481565b6103a8610634366004613742565b611627565b34801561064557600080fd5b506103a861065436600461358a565b6118a1565b34801561066557600080fd5b506103c3600e5481565b34801561067b57600080fd5b506103c361068a366004613473565b60166020526000908152604090205481565b3480156106a857600080fd5b506103a86106b7366004613742565b6119b8565b3480156106c857600080fd5b506103a86106d73660046135e9565b611afa565b3480156106e857600080fd5b506103a86106f736600461350a565b611c0e565b34801561070857600080fd5b50610336610717366004613742565b611cb6565b6103a861072a366004613742565b611dc6565b34801561073b57600080fd5b506103c3600f5481565b34801561075157600080fd5b5061030c610760366004613490565b612089565b34801561077157600080fd5b506103c360115481565b34801561078757600080fd5b5060145461030c90610100900460ff1681565b3480156107a657600080fd5b506103a86120dd565b3480156107bb57600080fd5b506103a86107ca366004613742565b612190565b3480156107db57600080fd5b506103a86107ea366004613473565b612216565b3480156107fb57600080fd5b506103c360135481565b60007fffffffff0000000000000000000000000000000000000000000000000000000082167f780e9d6300000000000000000000000000000000000000000000000000000000148061085b575061085b8261238d565b92915050565b6060600180546108709061392b565b80601f016020809104026020016040519081016040528092919081815260200182805461089c9061392b565b80156108e95780601f106108be576101008083540402835291602001916108e9565b820191906000526020600020905b8154815290600101906020018083116108cc57829003601f168201915b5050505050905090565b60008181526003602052604081205473ffffffffffffffffffffffffffffffffffffffff166109a9576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201527f697374656e7420746f6b656e000000000000000000000000000000000000000060648201526084015b60405180910390fd5b5060009081526005602052604090205473ffffffffffffffffffffffffffffffffffffffff1690565b60006109dd826112f7565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff161415610a9b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e6560448201527f720000000000000000000000000000000000000000000000000000000000000060648201526084016109a0565b3373ffffffffffffffffffffffffffffffffffffffff82161480610ac45750610ac48133612089565b610b50576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c000000000000000060648201526084016109a0565b610b5a8383612470565b505050565b3360009081526015602052604090205460ff1615610bd9576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f416c726561647920696e2077686974656c69737400000000000000000000000060448201526064016109a0565b601454610100900460ff1615610c4b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601e60248201527f526567756c61722073616c6520697320616c726561647920616374697665000060448201526064016109a0565b600060125411610cb7576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601160248201527f5072652073616c6520736f6c64206f757400000000000000000000000000000060448201526064016109a0565b600e54341015610d23576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f45746865722076616c75652073656e74206973206e6f7420636f72726563740060448201526064016109a0565b6001601254610d3291906138e8565b60125533600090815260156020526040902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00166001179055565b610d793382612510565b610e05576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016109a0565b610b5a838383612653565b60005473ffffffffffffffffffffffffffffffffffffffff163314610e91576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b60005b8151811015610f2457600160156000848481518110610eb557610eb5613a59565b60209081029190910181015173ffffffffffffffffffffffffffffffffffffffff16825281019190915260400160002080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001691151591909117905580610f1c8161397f565b915050610e94565b5050565b6000610f3383611437565b8210610fc1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201527f74206f6620626f756e647300000000000000000000000000000000000000000060648201526084016109a0565b5073ffffffffffffffffffffffffffffffffffffffff919091166000908152600760209081526040808320938352929052205490565b60005473ffffffffffffffffffffffffffffffffffffffff163314611078576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b601480547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff81166101009182900460ff1615909102179055565b60005473ffffffffffffffffffffffffffffffffffffffff163314611133576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b6040514790339082156108fc029083906000818181858888f19350505050158015610f24573d6000803e3d6000fd5b610b5a83838360405180602001604052806000815250611c0e565b60006111a061118b60095490565b601354600f5461119a916128c5565b906128c5565b905090565b60006111b060095490565b821061123e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201527f7574206f6620626f756e6473000000000000000000000000000000000000000060648201526084016109a0565b6009828154811061125157611251613a59565b90600052602060002001549050919050565b60005473ffffffffffffffffffffffffffffffffffffffff1633146112e4576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b8051610f2490600c906020840190613364565b60008181526003602052604081205473ffffffffffffffffffffffffffffffffffffffff168061085b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201527f656e7420746f6b656e000000000000000000000000000000000000000000000060648201526084016109a0565b600c80546113b69061392b565b80601f01602080910402602001604051908101604052809291908181526020018280546113e29061392b565b801561142f5780601f106114045761010080835404028352916020019161142f565b820191906000526020600020905b81548152906001019060200180831161141257829003601f168201915b505050505081565b600073ffffffffffffffffffffffffffffffffffffffff82166114dc576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a6560448201527f726f20616464726573730000000000000000000000000000000000000000000060648201526084016109a0565b5073ffffffffffffffffffffffffffffffffffffffff1660009081526004602052604090205490565b60005473ffffffffffffffffffffffffffffffffffffffff163314611586576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b61159060006128d1565b565b60005473ffffffffffffffffffffffffffffffffffffffff163314611613576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b600d55565b6060600280546108709061392b565b601454610100900460ff166116bd576040517f08c379a0000000000000000000000000000000000000000000000000000000008152602060048201526024808201527f53616c65206d7573742062652061637469766520746f206d696e74205072696d60448201527f617465730000000000000000000000000000000000000000000000000000000060648201526084016109a0565b60105481111561174f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603360248201527f43616e206f6e6c79206d696e74204d41585f5052494d4154455f50555243484160448201527f534520746f6b656e7320617420612074696d650000000000000000000000000060648201526084016109a0565b61175761117d565b8111156117e6576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f507572636861736520776f756c6420657863656564206d617820737570706c7960448201527f206f66205072696d61746573000000000000000000000000000000000000000060648201526084016109a0565b600d546117f39082612946565b34101561185c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f45746865722076616c75652073656e74206973206e6f7420636f72726563740060448201526064016109a0565b60005b81811015610f2457600061187260095490565b9050600061187e61117d565b111561188e5761188e3382612952565b50806118998161397f565b91505061185f565b73ffffffffffffffffffffffffffffffffffffffff8216331415611921576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c65720000000000000060448201526064016109a0565b33600081815260066020908152604080832073ffffffffffffffffffffffffffffffffffffffff87168085529083529281902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001686151590811790915590519081529192917f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a35050565b60005473ffffffffffffffffffffffffffffffffffffffff163314611a39576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b600081118015611a4b57506013548111155b611ab1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4e6f7420656e6f7567682072657365727665206c65667420666f72207465616d60448201526064016109a0565b60005b81811015611ae6576000611ac760095490565b9050611ad33382612952565b5080611ade8161397f565b915050611ab4565b50601354611af490826128c5565b60135550565b60005473ffffffffffffffffffffffffffffffffffffffff163314611b7b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b60005b8151811015610f2457600060156000848481518110611b9f57611b9f613a59565b60209081029190910181015173ffffffffffffffffffffffffffffffffffffffff16825281019190915260400160002080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff001691151591909117905580611c068161397f565b915050611b7e565b611c183383612510565b611ca4576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016109a0565b611cb08484848461296c565b50505050565b60008181526003602052604090205460609073ffffffffffffffffffffffffffffffffffffffff16611d6a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602f60248201527f4552433732314d657461646174613a2055524920717565727920666f72206e6f60448201527f6e6578697374656e7420746f6b656e000000000000000000000000000000000060648201526084016109a0565b6000611d74612a0f565b90506000815111611d945760405180602001604052806000815250611dbf565b80611d9e84612a1e565b604051602001611daf9291906137a5565b6040516020818303038152906040525b9392505050565b3360009081526015602052604090205460ff16611e3f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f57686974656c69737420726571756972656420666f72207072652073616c650060448201526064016109a0565b60145460ff16611ed1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602860248201527f5072652073616c65206d7573742062652061637469766520746f206d696e742060448201527f5072696d6174657300000000000000000000000000000000000000000000000060648201526084016109a0565b60115433600090815260166020526040902054611eee9083612b50565b1115611fa3576040517f08c379a0000000000000000000000000000000000000000000000000000000008152602060048201526044602482018190527f43616e206f6e6c79206d696e74204d41585f50524553414c455f5052494d4154908201527f455f505552434841534520746f6b656e7320647572696e67207468652070726560648201527f73616c6500000000000000000000000000000000000000000000000000000000608482015260a4016109a0565b600d54611fb09082612946565b341015612019576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f45746865722076616c75652073656e74206973206e6f7420636f72726563740060448201526064016109a0565b336000908152601660205260409020546120339082612b50565b336000908152601660205260408120919091555b81811015610f2457600061205a60095490565b9050600061206661117d565b1115612076576120763382612952565b50806120818161397f565b915050612047565b60006120958383612b5c565b156120a25750600161085b565b73ffffffffffffffffffffffffffffffffffffffff80841660009081526006602090815260408083209386168352929052205460ff16611dbf565b60005473ffffffffffffffffffffffffffffffffffffffff16331461215e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b601480547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00811660ff90911615179055565b60005473ffffffffffffffffffffffffffffffffffffffff163314612211576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b600e55565b60005473ffffffffffffffffffffffffffffffffffffffff163314612297576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016109a0565b73ffffffffffffffffffffffffffffffffffffffff811661233a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016109a0565b612343816128d1565b50565b600b80547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff92909216919091179055565b60007fffffffff0000000000000000000000000000000000000000000000000000000082167f80ac58cd00000000000000000000000000000000000000000000000000000000148061242057507fffffffff0000000000000000000000000000000000000000000000000000000082167f5b5e139f00000000000000000000000000000000000000000000000000000000145b8061085b57507f01ffc9a7000000000000000000000000000000000000000000000000000000007fffffffff0000000000000000000000000000000000000000000000000000000083161461085b565b600081815260056020526040902080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff841690811790915581906124ca826112f7565b73ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b60008181526003602052604081205473ffffffffffffffffffffffffffffffffffffffff166125c1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201527f697374656e7420746f6b656e000000000000000000000000000000000000000060648201526084016109a0565b60006125cc836112f7565b90508073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff16148061263b57508373ffffffffffffffffffffffffffffffffffffffff16612623846108f3565b73ffffffffffffffffffffffffffffffffffffffff16145b8061264b575061264b8185612089565b949350505050565b8273ffffffffffffffffffffffffffffffffffffffff16612673826112f7565b73ffffffffffffffffffffffffffffffffffffffff1614612716576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602960248201527f4552433732313a207472616e73666572206f6620746f6b656e2074686174206960448201527f73206e6f74206f776e000000000000000000000000000000000000000000000060648201526084016109a0565b73ffffffffffffffffffffffffffffffffffffffff82166127b8576040517f08c379a0000000000000000000000000000000000000000000000000000000008152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f2061646460448201527f726573730000000000000000000000000000000000000000000000000000000060648201526084016109a0565b6127c3838383612c37565b6127ce600082612470565b73ffffffffffffffffffffffffffffffffffffffff831660009081526004602052604081208054600192906128049084906138e8565b909155505073ffffffffffffffffffffffffffffffffffffffff8216600090815260046020526040812080546001929061283f90849061387f565b909155505060008181526003602052604080822080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff86811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b6000611dbf82846138e8565b6000805473ffffffffffffffffffffffffffffffffffffffff8381167fffffffffffffffffffffffff0000000000000000000000000000000000000000831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b6000611dbf82846138ab565b610f24828260405180602001604052806000815250612d3d565b612977848484612653565b61298384848484612de0565b611cb0576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016109a0565b6060600c80546108709061392b565b606081612a5e57505060408051808201909152600181527f3000000000000000000000000000000000000000000000000000000000000000602082015290565b8160005b8115612a885780612a728161397f565b9150612a819050600a83613897565b9150612a62565b60008167ffffffffffffffff811115612aa357612aa3613a88565b6040519080825280601f01601f191660200182016040528015612acd576020820181803683370190505b5090505b841561264b57612ae26001836138e8565b9150612aef600a866139b8565b612afa90603061387f565b60f81b818381518110612b0f57612b0f613a59565b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350612b49600a86613897565b9450612ad1565b6000611dbf828461387f565b600b546040517fc455279100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff848116600483015260009281169190841690829063c45527919060240160206040518083038186803b158015612bcf57600080fd5b505afa158015612be3573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612c0791906136dc565b73ffffffffffffffffffffffffffffffffffffffff161415612c2d57600191505061085b565b5060009392505050565b73ffffffffffffffffffffffffffffffffffffffff8316612c9f57612c9a81600980546000838152600a60205260408120829055600182018355919091527f6e1540171b6c0c960b71a7020d9f60077f6af931a8bbf590da0223dacf75c7af0155565b612cdc565b8173ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1614612cdc57612cdc8382612fdf565b73ffffffffffffffffffffffffffffffffffffffff8216612d0057610b5a81613096565b8273ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614610b5a57610b5a8282613145565b612d478383613196565b612d546000848484612de0565b610b5a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016109a0565b600073ffffffffffffffffffffffffffffffffffffffff84163b15612fd4576040517f150b7a0200000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff85169063150b7a0290612e579033908990889088906004016137d4565b602060405180830381600087803b158015612e7157600080fd5b505af1925050508015612ebf575060408051601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0168201909252612ebc918101906136bf565b60015b612f89573d808015612eed576040519150601f19603f3d011682016040523d82523d6000602084013e612ef2565b606091505b508051612f81576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016109a0565b805181602001fd5b7fffffffff00000000000000000000000000000000000000000000000000000000167f150b7a020000000000000000000000000000000000000000000000000000000014905061264b565b506001949350505050565b60006001612fec84611437565b612ff691906138e8565b6000838152600860205260409020549091508082146130565773ffffffffffffffffffffffffffffffffffffffff841660009081526007602090815260408083208584528252808320548484528184208190558352600890915290208190555b50600091825260086020908152604080842084905573ffffffffffffffffffffffffffffffffffffffff9094168352600781528383209183525290812055565b6009546000906130a8906001906138e8565b6000838152600a6020526040812054600980549394509092849081106130d0576130d0613a59565b9060005260206000200154905080600983815481106130f1576130f1613a59565b6000918252602080832090910192909255828152600a9091526040808220849055858252812055600980548061312957613129613a2a565b6001900381819060005260206000200160009055905550505050565b600061315083611437565b73ffffffffffffffffffffffffffffffffffffffff9093166000908152600760209081526040808320868452825280832085905593825260089052919091209190915550565b73ffffffffffffffffffffffffffffffffffffffff8216613213576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f206164647265737360448201526064016109a0565b60008181526003602052604090205473ffffffffffffffffffffffffffffffffffffffff161561329f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e7465640000000060448201526064016109a0565b6132ab60008383612c37565b73ffffffffffffffffffffffffffffffffffffffff821660009081526004602052604081208054600192906132e190849061387f565b909155505060008181526003602052604080822080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b8280546133709061392b565b90600052602060002090601f01602090048101928261339257600085556133d8565b82601f106133ab57805160ff19168380011785556133d8565b828001600101855582156133d8579182015b828111156133d85782518255916020019190600101906133bd565b506133e49291506133e8565b5090565b5b808211156133e457600081556001016133e9565b600067ffffffffffffffff83111561341757613417613a88565b61344860207fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f86011601613830565b905082815283838301111561345c57600080fd5b828260208301376000602084830101529392505050565b60006020828403121561348557600080fd5b8135611dbf81613ab7565b600080604083850312156134a357600080fd5b82356134ae81613ab7565b915060208301356134be81613ab7565b809150509250929050565b6000806000606084860312156134de57600080fd5b83356134e981613ab7565b925060208401356134f981613ab7565b929592945050506040919091013590565b6000806000806080858703121561352057600080fd5b843561352b81613ab7565b9350602085013561353b81613ab7565b925060408501359150606085013567ffffffffffffffff81111561355e57600080fd5b8501601f8101871361356f57600080fd5b61357e878235602084016133fd565b91505092959194509250565b6000806040838503121561359d57600080fd5b82356135a881613ab7565b9150602083013580151581146134be57600080fd5b600080604083850312156135d057600080fd5b82356135db81613ab7565b946020939093013593505050565b600060208083850312156135fc57600080fd5b823567ffffffffffffffff8082111561361457600080fd5b818501915085601f83011261362857600080fd5b81358181111561363a5761363a613a88565b8060051b915061364b848301613830565b8181528481019084860184860187018a101561366657600080fd5b600095505b83861015613695578035945061368085613ab7565b8483526001959095019491860191860161366b565b5098975050505050505050565b6000602082840312156136b457600080fd5b8135611dbf81613ad9565b6000602082840312156136d157600080fd5b8151611dbf81613ad9565b6000602082840312156136ee57600080fd5b8151611dbf81613ab7565b60006020828403121561370b57600080fd5b813567ffffffffffffffff81111561372257600080fd5b8201601f8101841361373357600080fd5b61264b848235602084016133fd565b60006020828403121561375457600080fd5b5035919050565b600081518084526137738160208601602086016138ff565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0169290920160200192915050565b600083516137b78184602088016138ff565b8351908301906137cb8183602088016138ff565b01949350505050565b600073ffffffffffffffffffffffffffffffffffffffff808716835280861660208401525083604083015260806060830152613813608083018461375b565b9695505050505050565b602081526000611dbf602083018461375b565b604051601f82017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016810167ffffffffffffffff8111828210171561387757613877613a88565b604052919050565b60008219821115613892576138926139cc565b500190565b6000826138a6576138a66139fb565b500490565b6000817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff04831182151516156138e3576138e36139cc565b500290565b6000828210156138fa576138fa6139cc565b500390565b60005b8381101561391a578181015183820152602001613902565b83811115611cb05750506000910152565b600181811c9082168061393f57607f821691505b60208210811415613979577f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b50919050565b60007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8214156139b1576139b16139cc565b5060010190565b6000826139c7576139c76139fb565b500690565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b73ffffffffffffffffffffffffffffffffffffffff8116811461234357600080fd5b7fffffffff000000000000000000000000000000000000000000000000000000008116811461234357600080fdfea26469706673582212202f5e99d7387fb45f3f3c9871e5b1b1b4871095d87c791d7859dfd95b5aa1c64764736f6c63430008070033

Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)

0000000000000000000000000000000000000000000000000000000000000040000000000000000000000000a5409ec958c83c3f309868babaca7c86dcb077c1000000000000000000000000000000000000000000000000000000000000002c68747470733a2f2f6170692e7072696d617465736f6369616c736f63696574792e636f6d2f746f6b656e732f0000000000000000000000000000000000000000

-----Decoded View---------------
Arg [0] : _initialBaseURI (string): https://api.primatesocialsociety.com/tokens/
Arg [1] : _openSeaProxyRegistryAddress (address): 0xa5409ec958C83C3f309868babACA7c86DCB077c1

-----Encoded View---------------
5 Constructor Arguments found :
Arg [0] : 0000000000000000000000000000000000000000000000000000000000000040
Arg [1] : 000000000000000000000000a5409ec958c83c3f309868babaca7c86dcb077c1
Arg [2] : 000000000000000000000000000000000000000000000000000000000000002c
Arg [3] : 68747470733a2f2f6170692e7072696d617465736f6369616c736f6369657479
Arg [4] : 2e636f6d2f746f6b656e732f0000000000000000000000000000000000000000


Deployed Bytecode Sourcemap

53379:5423:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;38816:300;;;;;;;;;;-1:-1:-1;38816:300:0;;;;;:::i;:::-;;:::i;:::-;;;7311:14:1;;7304:22;7286:41;;7274:2;7259:18;38816:300:0;;;;;;;;24956:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;26649:308::-;;;;;;;;;;-1:-1:-1;26649:308:0;;;;;:::i;:::-;;:::i;:::-;;;6575:42:1;6563:55;;;6545:74;;6533:2;6518:18;26649:308:0;6399:226:1;26172:411:0;;;;;;;;;;-1:-1:-1;26172:411:0;;;;;:::i;:::-;;:::i;:::-;;56738:473;;;:::i;39619:113::-;;;;;;;;;;-1:-1:-1;39707:10:0;:17;39619:113;;;19174:25:1;;;19162:2;19147:18;39619:113:0;19028:177:1;53886:35:0;;;;;;;;;;-1:-1:-1;53886:35:0;;;;;;;;27708:376;;;;;;;;;;-1:-1:-1;27708:376:0;;;;;:::i;:::-;;:::i;57219:197::-;;;;;;;;;;-1:-1:-1;57219:197:0;;;;;:::i;:::-;;:::i;39200:343::-;;;;;;;;;;-1:-1:-1;39200:343:0;;;;;:::i;:::-;;:::i;57920:91::-;;;;;;;;;;;;;:::i;57659:145::-;;;;;;;;;;;;;:::i;28155:185::-;;;;;;;;;;-1:-1:-1;28155:185:0;;;;;:::i;:::-;;:::i;54845:139::-;;;;;;;;;;;;;:::i;39809:320::-;;;;;;;;;;-1:-1:-1;39809:320:0;;;;;:::i;:::-;;:::i;58231:106::-;;;;;;;;;;-1:-1:-1;58231:106:0;;;;;:::i;:::-;;:::i;24563:326::-;;;;;;;;;;-1:-1:-1;24563:326:0;;;;;:::i;:::-;;:::i;53519:26::-;;;;;;;;;;;;;:::i;24206:295::-;;;;;;;;;;-1:-1:-1;24206:295:0;;;;;:::i;:::-;;:::i;2614:94::-;;;;;;;;;;;;;:::i;1963:87::-;;;;;;;;;;-1:-1:-1;2009:7:0;2036:6;;;1963:87;;53690:40;;;;;;;;;;;;;;;;58019:86;;;;;;;;;;-1:-1:-1;58019:86:0;;;;;:::i;:::-;;:::i;25125:104::-;;;;;;;;;;;;;:::i;53792:42::-;;;;;;;;;;;;;;;;53969:41;;;;;;;;;;-1:-1:-1;53969:41:0;;;;;:::i;:::-;;;;;;;;;;;;;;;;53554:33;;;;;;;;;;;;;;;;55934:796;;;;;;:::i;:::-;;:::i;27029:327::-;;;;;;;;;;-1:-1:-1;27029:327:0;;;;;:::i;:::-;;:::i;53594:45::-;;;;;;;;;;;;;;;;54017:51;;;;;;;;;;-1:-1:-1;54017:51:0;;;;;:::i;:::-;;;;;;;;;;;;;;58345:454;;;;;;;;;;-1:-1:-1;58345:454:0;;;;;:::i;:::-;;:::i;57424:227::-;;;;;;;;;;-1:-1:-1;57424:227:0;;;;;:::i;:::-;;:::i;28411:365::-;;;;;;;;;;-1:-1:-1;28411:365:0;;;;;:::i;:::-;;:::i;25300:468::-;;;;;;;;;;-1:-1:-1;25300:468:0;;;;;:::i;:::-;;:::i;54992:934::-;;;;;;:::i;:::-;;:::i;53648:35::-;;;;;;;;;;;;;;;;54383:338;;;;;;;;;;-1:-1:-1;54383:338:0;;;;;:::i;:::-;;:::i;53737:48::-;;;;;;;;;;;;;;;;53928:32;;;;;;;;;;-1:-1:-1;53928:32:0;;;;;;;;;;;57812:100;;;;;;;;;;;;;:::i;58113:110::-;;;;;;;;;;-1:-1:-1;58113:110:0;;;;;:::i;:::-;;:::i;2863:229::-;;;;;;;;;;-1:-1:-1;2863:229:0;;;;;:::i;:::-;;:::i;53841:36::-;;;;;;;;;;;;;;;;38816:300;38963:4;39005:50;;;39020:35;39005:50;;:103;;;39072:36;39096:11;39072:23;:36::i;:::-;38985:123;38816:300;-1:-1:-1;;38816:300:0:o;24956:100::-;25010:13;25043:5;25036:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;24956:100;:::o;26649:308::-;26770:7;30412:16;;;:7;:16;;;;;;:30;:16;26795:110;;;;;;;14860:2:1;26795:110:0;;;14842:21:1;14899:2;14879:18;;;14872:30;14938:34;14918:18;;;14911:62;15009:14;14989:18;;;14982:42;15041:19;;26795:110:0;;;;;;;;;-1:-1:-1;26925:24:0;;;;:15;:24;;;;;;;;;26649:308::o;26172:411::-;26253:13;26269:23;26284:7;26269:14;:23::i;:::-;26253:39;;26317:5;26311:11;;:2;:11;;;;26303:57;;;;;;;17232:2:1;26303:57:0;;;17214:21:1;17271:2;17251:18;;;17244:30;17310:34;17290:18;;;17283:62;17381:3;17361:18;;;17354:31;17402:19;;26303:57:0;17030:397:1;26303:57:0;806:10;26395:21;;;;;:62;;-1:-1:-1;26420:37:0;26437:5;806:10;54383:338;:::i;26420:37::-;26373:168;;;;;;;12833:2:1;26373:168:0;;;12815:21:1;12872:2;12852:18;;;12845:30;12911:34;12891:18;;;12884:62;12982:26;12962:18;;;12955:54;13026:19;;26373:168:0;12631:420:1;26373:168:0;26554:21;26563:2;26567:7;26554:8;:21::i;:::-;26242:341;26172:411;;:::o;56738:473::-;56811:10;56801:21;;;;:9;:21;;;;;;;;56800:22;56792:55;;;;;;;11594:2:1;56792:55:0;;;11576:21:1;11633:2;11613:18;;;11606:30;11672:22;11652:18;;;11645:50;11712:18;;56792:55:0;11392:344:1;56792:55:0;56867:12;;;;;;;56866:13;56858:56;;;;;;;16873:2:1;56858:56:0;;;16855:21:1;16912:2;16892:18;;;16885:30;16951:32;16931:18;;;16924:60;17001:18;;56858:56:0;16671:354:1;56858:56:0;56957:1;56933:21;;:25;56925:55;;;;;;;9768:2:1;56925:55:0;;;9750:21:1;9807:2;9787:18;;;9780:30;9846:19;9826:18;;;9819:47;9883:18;;56925:55:0;9566:341:1;56925:55:0;57026:17;;57013:9;:30;;56991:111;;;;;;;11234:2:1;56991:111:0;;;11216:21:1;11273:2;11253:18;;;11246:30;11312:33;11292:18;;;11285:61;11363:18;;56991:111:0;11032:355:1;56991:111:0;57163:1;57139:21;;:25;;;;:::i;:::-;57115:21;:49;57185:10;57175:21;;;;:9;:21;;;;;:28;;;;57199:4;57175:28;;;56738:473::o;27708:376::-;27917:41;806:10;27950:7;27917:18;:41::i;:::-;27895:140;;;;;;;18039:2:1;27895:140:0;;;18021:21:1;18078:2;18058:18;;;18051:30;18117:34;18097:18;;;18090:62;18188:19;18168:18;;;18161:47;18225:19;;27895:140:0;17837:413:1;27895:140:0;28048:28;28058:4;28064:2;28068:7;28048:9;:28::i;57219:197::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;57310:9:::1;57305:104;57329:9;:16;57325:1;:20;57305:104;;;57393:4;57367:9;:23;57377:9;57387:1;57377:12;;;;;;;;:::i;:::-;;::::0;;::::1;::::0;;;;;;;57367:23:::1;;::::0;;;::::1;::::0;;;;;;-1:-1:-1;57367:23:0;:30;;;::::1;::::0;::::1;;::::0;;;::::1;::::0;;57347:3;::::1;::::0;::::1;:::i;:::-;;;;57305:104;;;;57219:197:::0;:::o;39200:343::-;39342:7;39397:23;39414:5;39397:16;:23::i;:::-;39389:5;:31;39367:124;;;;;;;7764:2:1;39367:124:0;;;7746:21:1;7803:2;7783:18;;;7776:30;7842:34;7822:18;;;7815:62;7913:13;7893:18;;;7886:41;7944:19;;39367:124:0;7562:407:1;39367:124:0;-1:-1:-1;39509:19:0;;;;;;;;;:12;:19;;;;;;;;:26;;;;;;;;;39200:343::o;57920:91::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;57991:12:::1;::::0;;57975:28;;::::1;57991:12;::::0;;;::::1;;;57990:13;57975:28:::0;;::::1;;::::0;;57920:91::o;57659:145::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;57759:37:::1;::::0;57727:21:::1;::::0;57767:10:::1;::::0;57759:37;::::1;;;::::0;57727:21;;57709:15:::1;57759:37:::0;57709:15;57759:37;57727:21;57767:10;57759:37;::::1;;;;;;;;;;;;;::::0;::::1;;;;28155:185:::0;28293:39;28310:4;28316:2;28320:7;28293:39;;;;;;;;;;;;:16;:39::i;54845:139::-;54897:7;54924:52;54962:13;39707:10;:17;;39619:113;54962:13;54941:15;;54924:12;;:33;;:16;:33::i;:::-;:37;;:52::i;:::-;54917:59;;54845:139;:::o;39809:320::-;39929:7;39984:30;39707:10;:17;;39619:113;39984:30;39976:5;:38;39954:132;;;;;;;18457:2:1;39954:132:0;;;18439:21:1;18496:2;18476:18;;;18469:30;18535:34;18515:18;;;18508:62;18606:14;18586:18;;;18579:42;18638:19;;39954:132:0;18255:408:1;39954:132:0;40104:10;40115:5;40104:17;;;;;;;;:::i;:::-;;;;;;;;;40097:24;;39809:320;;;:::o;58231:106::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;58308:21;;::::1;::::0;:7:::1;::::0;:21:::1;::::0;::::1;::::0;::::1;:::i;24563:326::-:0;24680:7;24721:16;;;:7;:16;;;;;;;;24770:19;24748:110;;;;;;;14089:2:1;24748:110:0;;;14071:21:1;14128:2;14108:18;;;14101:30;14167:34;14147:18;;;14140:62;14238:11;14218:18;;;14211:39;14267:19;;24748:110:0;13887:405:1;53519:26:0;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;24206:295::-;24323:7;24370:19;;;24348:111;;;;;;;13678:2:1;24348:111:0;;;13660:21:1;13717:2;13697:18;;;13690:30;13756:34;13736:18;;;13729:62;13827:12;13807:18;;;13800:40;13857:19;;24348:111:0;13476:406:1;24348:111:0;-1:-1:-1;24477:16:0;;;;;;:9;:16;;;;;;;24206:295::o;2614:94::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;2679:21:::1;2697:1;2679:9;:21::i;:::-;2614:94::o:0;58019:86::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;58083:5:::1;:14:::0;58019:86::o;25125:104::-;25181:13;25214:7;25207:14;;;;;:::i;55934:796::-;56016:12;;;;;;;56008:61;;;;;;;17634:2:1;56008:61:0;;;17616:21:1;17673:2;17653:18;;;17646:30;17712:34;17692:18;;;17685:62;17783:6;17763:18;;;17756:34;17807:19;;56008:61:0;17432:400:1;56008:61:0;56120:20;;56102:14;:38;;56080:139;;;;;;;13258:2:1;56080:139:0;;;13240:21:1;13297:2;13277:18;;;13270:30;13336:34;13316:18;;;13309:62;13407:21;13387:18;;;13380:49;13446:19;;56080:139:0;13056:415:1;56080:139:0;56270:21;:19;:21::i;:::-;56252:14;:39;;56230:133;;;;;;;15634:2:1;56230:133:0;;;15616:21:1;15673:2;15653:18;;;15646:30;15712:34;15692:18;;;15685:62;15783:14;15763:18;;;15756:42;15815:19;;56230:133:0;15432:408:1;56230:133:0;56409:5;;:25;;56419:14;56409:9;:25::i;:::-;56396:9;:38;;56374:119;;;;;;;11234:2:1;56374:119:0;;;11216:21:1;11273:2;11253:18;;;11246:30;11312:33;11292:18;;;11285:61;11363:18;;56374:119:0;11032:355:1;56374:119:0;56511:9;56506:217;56530:14;56526:1;:18;56506:217;;;56566:17;56586:13;39707:10;:17;;39619:113;56586:13;56566:33;;56642:1;56618:21;:19;:21::i;:::-;:25;56614:98;;;56664:32;56674:10;56686:9;56664;:32::i;:::-;-1:-1:-1;56546:3:0;;;;:::i;:::-;;;;56506:217;;27029:327;27164:24;;;806:10;27164:24;;27156:62;;;;;;;10880:2:1;27156:62:0;;;10862:21:1;10919:2;10899:18;;;10892:30;10958:27;10938:18;;;10931:55;11003:18;;27156:62:0;10678:349:1;27156:62:0;806:10;27231:32;;;;:18;:32;;;;;;;;;:42;;;;;;;;;;;;:53;;;;;;;;;;;;;27300:48;;7286:41:1;;;27231:42:0;;806:10;27300:48;;7259:18:1;27300:48:0;;;;;;;27029:327;;:::o;58345:454::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;58463:1:::1;58446:14;:18;:55;;;;;58486:15;;58468:14;:33;;58446:55;58424:137;;;::::0;::::1;::::0;;10114:2:1;58424:137:0::1;::::0;::::1;10096:21:1::0;;;10133:18;;;10126:30;10192:34;10172:18;;;10165:62;10244:18;;58424:137:0::1;9912:356:1::0;58424:137:0::1;58579:9;58574:152;58598:14;58594:1;:18;58574:152;;;58634:17;58654:13;39707:10:::0;:17;;39619:113;58654:13:::1;58634:33;;58682:32;58692:10;58704:9;58682;:32::i;:::-;-1:-1:-1::0;58614:3:0;::::1;::::0;::::1;:::i;:::-;;;;58574:152;;;-1:-1:-1::0;58756:15:0::1;::::0;:35:::1;::::0;58776:14;58756:19:::1;:35::i;:::-;58738:15;:53:::0;-1:-1:-1;58345:454:0:o;57424:227::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;57544:9:::1;57539:105;57563:9;:16;57559:1;:20;57539:105;;;57627:5;57601:9;:23;57611:9;57621:1;57611:12;;;;;;;;:::i;:::-;;::::0;;::::1;::::0;;;;;;;57601:23:::1;;::::0;;;::::1;::::0;;;;;;-1:-1:-1;57601:23:0;:31;;;::::1;::::0;::::1;;::::0;;;::::1;::::0;;57581:3;::::1;::::0;::::1;:::i;:::-;;;;57539:105;;28411:365:::0;28600:41;806:10;28633:7;28600:18;:41::i;:::-;28578:140;;;;;;;18039:2:1;28578:140:0;;;18021:21:1;18078:2;18058:18;;;18051:30;18117:34;18097:18;;;18090:62;18188:19;18168:18;;;18161:47;18225:19;;28578:140:0;17837:413:1;28578:140:0;28729:39;28743:4;28749:2;28753:7;28762:5;28729:13;:39::i;:::-;28411:365;;;;:::o;25300:468::-;30388:4;30412:16;;;:7;:16;;;;;;25418:13;;30412:30;:16;25449:113;;;;;;;16457:2:1;25449:113:0;;;16439:21:1;16496:2;16476:18;;;16469:30;16535:34;16515:18;;;16508:62;16606:17;16586:18;;;16579:45;16641:19;;25449:113:0;16255:411:1;25449:113:0;25575:21;25599:10;:8;:10::i;:::-;25575:34;;25664:1;25646:7;25640:21;:25;:120;;;;;;;;;;;;;;;;;25709:7;25718:18;:7;:16;:18::i;:::-;25692:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;25640:120;25620:140;25300:468;-1:-1:-1;;;25300:468:0:o;54992:934::-;55091:10;55081:21;;;;:9;:21;;;;;;;;55073:65;;;;;;;18870:2:1;55073:65:0;;;18852:21:1;18909:2;18889:18;;;18882:30;18948:33;18928:18;;;18921:61;18999:18;;55073:65:0;18668:355:1;55073:65:0;55157:15;;;;55149:68;;;;;;;9359:2:1;55149:68:0;;;9341:21:1;9398:2;9378:18;;;9371:30;9437:34;9417:18;;;9410:62;9508:10;9488:18;;;9481:38;9536:19;;55149:68:0;9157:404:1;55149:68:0;55319:28;;55267:10;55250:28;;;;:16;:28;;;;;;:48;;55283:14;55250:32;:48::i;:::-;:97;;55228:215;;;;;;;12356:2:1;55228:215:0;;;12338:21:1;12395:2;12375:18;;;12368:30;;;12434:34;12414:18;;;12407:62;12505:34;12485:18;;;12478:62;12577:6;12556:19;;;12549:35;12601:19;;55228:215:0;12154:472:1;55228:215:0;55489:5;;:25;;55499:14;55489:9;:25::i;:::-;55476:9;:38;;55454:119;;;;;;;11234:2:1;55454:119:0;;;11216:21:1;11273:2;11253:18;;;11246:30;11312:33;11292:18;;;11285:61;11363:18;;55454:119:0;11032:355:1;55454:119:0;55634:10;55617:28;;;;:16;:28;;;;;;:72;;55664:14;55617:32;:72::i;:::-;55603:10;55586:28;;;;:16;:28;;;;;:103;;;;55702:217;55726:14;55722:1;:18;55702:217;;;55762:17;55782:13;39707:10;:17;;39619:113;55782:13;55762:33;;55838:1;55814:21;:19;:21::i;:::-;:25;55810:98;;;55860:32;55870:10;55882:9;55860;:32::i;:::-;-1:-1:-1;55742:3:0;;;;:::i;:::-;;;;55702:217;;54383:338;54536:4;54562:52;54598:5;54605:8;54562:35;:52::i;:::-;54558:96;;;-1:-1:-1;54638:4:0;54631:11;;54558:96;27598:25;;;;27569:4;27598:25;;;:18;:25;;;;;;;;:35;;;;;;;;;;;;54673:40;27427:214;57812:100;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;57889:15:::1;::::0;;57870:34;;::::1;57889:15;::::0;;::::1;57888:16;57870:34;::::0;;57812:100::o;58113:110::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;58189:17:::1;:26:::0;58113:110::o;2863:229::-;2009:7;2036:6;2183:23;2036:6;806:10;2183:23;2175:68;;;;;;;15273:2:1;2175:68:0;;;15255:21:1;;;15292:18;;;15285:30;15351:34;15331:18;;;15324:62;15403:18;;2175:68:0;15071:356:1;2175:68:0;2966:22:::1;::::0;::::1;2944:110;;;::::0;::::1;::::0;;8595:2:1;2944:110:0::1;::::0;::::1;8577:21:1::0;8634:2;8614:18;;;8607:30;8673:34;8653:18;;;8646:62;8744:8;8724:18;;;8717:36;8770:19;;2944:110:0::1;8393:402:1::0;2944:110:0::1;3065:19;3075:8;3065:9;:19::i;:::-;2863:229:::0;:::o;45528:172::-;45634:27;:58;;;;;;;;;;;;;;;45528:172::o;23787:355::-;23934:4;23976:40;;;23991:25;23976:40;;:105;;-1:-1:-1;24033:48:0;;;24048:33;24033:48;23976:105;:158;;;-1:-1:-1;22427:25:0;22412:40;;;;24098:36;22253:207;34446:174;34521:24;;;;:15;:24;;;;;:29;;;;;;;;;;;;;:24;;34575:23;34521:24;34575:14;:23::i;:::-;34566:46;;;;;;;;;;;;34446:174;;:::o;30617:452::-;30746:4;30412:16;;;:7;:16;;;;;;:30;:16;30768:110;;;;;;;11943:2:1;30768:110:0;;;11925:21:1;11982:2;11962:18;;;11955:30;12021:34;12001:18;;;11994:62;12092:14;12072:18;;;12065:42;12124:19;;30768:110:0;11741:408:1;30768:110:0;30889:13;30905:23;30920:7;30905:14;:23::i;:::-;30889:39;;30958:5;30947:16;;:7;:16;;;:64;;;;31004:7;30980:31;;:20;30992:7;30980:11;:20::i;:::-;:31;;;30947:64;:113;;;;31028:32;31045:5;31052:7;31028:16;:32::i;:::-;30939:122;30617:452;-1:-1:-1;;;;30617:452:0:o;33713:615::-;33886:4;33859:31;;:23;33874:7;33859:14;:23::i;:::-;:31;;;33837:122;;;;;;;16047:2:1;33837:122:0;;;16029:21:1;16086:2;16066:18;;;16059:30;16125:34;16105:18;;;16098:62;16196:11;16176:18;;;16169:39;16225:19;;33837:122:0;15845:405:1;33837:122:0;33978:16;;;33970:65;;;;;;;10475:2:1;33970:65:0;;;10457:21:1;10514:2;10494:18;;;10487:30;10553:34;10533:18;;;10526:62;10624:6;10604:18;;;10597:34;10648:19;;33970:65:0;10273:400:1;33970:65:0;34048:39;34069:4;34075:2;34079:7;34048:20;:39::i;:::-;34152:29;34169:1;34173:7;34152:8;:29::i;:::-;34194:15;;;;;;;:9;:15;;;;;:20;;34213:1;;34194:15;:20;;34213:1;;34194:20;:::i;:::-;;;;-1:-1:-1;;34225:13:0;;;;;;;:9;:13;;;;;:18;;34242:1;;34225:13;:18;;34242:1;;34225:18;:::i;:::-;;;;-1:-1:-1;;34254:16:0;;;;:7;:16;;;;;;:21;;;;;;;;;;;;;;34293:27;;34254:16;;34293:27;;;;;;;33713:615;;;:::o;49519:98::-;49577:7;49604:5;49608:1;49604;:5;:::i;3100:173::-;3156:16;3175:6;;;3192:17;;;;;;;;;;3225:40;;3175:6;;;;;;;3225:40;;3156:16;3225:40;3145:128;3100:173;:::o;49876:98::-;49934:7;49961:5;49965:1;49961;:5;:::i;31411:110::-;31487:26;31497:2;31501:7;31487:26;;;;;;;;;;;;:9;:26::i;29658:352::-;29815:28;29825:4;29831:2;29835:7;29815:9;:28::i;:::-;29876:48;29899:4;29905:2;29909:7;29918:5;29876:22;:48::i;:::-;29854:148;;;;;;;8176:2:1;29854:148:0;;;8158:21:1;8215:2;8195:18;;;8188:30;8254:34;8234:18;;;8227:62;8325:20;8305:18;;;8298:48;8363:19;;29854:148:0;7974:414:1;54729:108:0;54789:13;54822:7;54815:14;;;;;:::i;19664:723::-;19720:13;19941:10;19937:53;;-1:-1:-1;;19968:10:0;;;;;;;;;;;;;;;;;;19664:723::o;19937:53::-;20015:5;20000:12;20056:78;20063:9;;20056:78;;20089:8;;;;:::i;:::-;;-1:-1:-1;20112:10:0;;-1:-1:-1;20120:2:0;20112:10;;:::i;:::-;;;20056:78;;;20144:19;20176:6;20166:17;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;20166:17:0;;20144:39;;20194:154;20201:10;;20194:154;;20228:11;20238:1;20228:11;;:::i;:::-;;-1:-1:-1;20297:10:0;20305:2;20297:5;:10;:::i;:::-;20284:24;;:2;:24;:::i;:::-;20271:39;;20254:6;20261;20254:14;;;;;;;;:::i;:::-;;;;:56;;;;;;;;;;-1:-1:-1;20325:11:0;20334:2;20325:11;;:::i;:::-;;;20194:154;;49138:98;49196:7;49223:5;49227:1;49223;:5;:::i;45708:469::-;45996:27;;46057:35;;;;;45996:27;6563:55:1;;;46057:35:0;;;6545:74:1;45832:4:0;;45996:27;;;46049:56;;;;45996:27;;46057:28;;6518:18:1;;46057:35:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;46049:56;;;46045:100;;;46129:4;46122:11;;;;;46045:100;-1:-1:-1;46164:5:0;;45708:469;-1:-1:-1;;;45708:469:0:o;40742:589::-;40948:18;;;40944:187;;40983:40;41015:7;42158:10;:17;;42131:24;;;;:15;:24;;;;;:44;;;42186:24;;;;;;;;;;;;42054:164;40983:40;40944:187;;;41053:2;41045:10;;:4;:10;;;41041:90;;41072:47;41105:4;41111:7;41072:32;:47::i;:::-;41145:16;;;41141:183;;41178:45;41215:7;41178:36;:45::i;41141:183::-;41251:4;41245:10;;:2;:10;;;41241:83;;41272:40;41300:2;41304:7;41272:27;:40::i;31748:321::-;31878:18;31884:2;31888:7;31878:5;:18::i;:::-;31929:54;31960:1;31964:2;31968:7;31977:5;31929:22;:54::i;:::-;31907:154;;;;;;;8176:2:1;31907:154:0;;;8158:21:1;8215:2;8195:18;;;8188:30;8254:34;8234:18;;;8227:62;8325:20;8305:18;;;8298:48;8363:19;;31907:154:0;7974:414:1;35185:980:0;35340:4;35361:13;;;11876:20;11924:8;35357:801;;35414:175;;;;;:36;;;;;;:175;;806:10;;35508:4;;35535:7;;35565:5;;35414:175;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;35414:175:0;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;35393:710;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;35772:13:0;;35768:320;;35815:108;;;;;8176:2:1;35815:108:0;;;8158:21:1;8215:2;8195:18;;;8188:30;8254:34;8234:18;;;8227:62;8325:20;8305:18;;;8298:48;8363:19;;35815:108:0;7974:414:1;35768:320:0;36038:6;36032:13;36023:6;36019:2;36015:15;36008:38;35393:710;35653:51;;35663:41;35653:51;;-1:-1:-1;35646:58:0;;35357:801;-1:-1:-1;36142:4:0;35185:980;;;;;;:::o;42845:1002::-;43125:22;43175:1;43150:22;43167:4;43150:16;:22::i;:::-;:26;;;;:::i;:::-;43187:18;43208:26;;;:17;:26;;;;;;43125:51;;-1:-1:-1;43341:28:0;;;43337:328;;43408:18;;;43386:19;43408:18;;;:12;:18;;;;;;;;:34;;;;;;;;;43459:30;;;;;;:44;;;43576:30;;:17;:30;;;;;:43;;;43337:328;-1:-1:-1;43761:26:0;;;;:17;:26;;;;;;;;43754:33;;;43805:18;;;;;;:12;:18;;;;;:34;;;;;;;43798:41;42845:1002::o;44142:1079::-;44420:10;:17;44395:22;;44420:21;;44440:1;;44420:21;:::i;:::-;44452:18;44473:24;;;:15;:24;;;;;;44846:10;:26;;44395:46;;-1:-1:-1;44473:24:0;;44395:46;;44846:26;;;;;;:::i;:::-;;;;;;;;;44824:48;;44910:11;44885:10;44896;44885:22;;;;;;;;:::i;:::-;;;;;;;;;;;;:36;;;;44990:28;;;:15;:28;;;;;;;:41;;;45162:24;;;;;45155:31;45197:10;:16;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;44213:1008;;;44142:1079;:::o;41632:221::-;41717:14;41734:20;41751:2;41734:16;:20::i;:::-;41765:16;;;;;;;;:12;:16;;;;;;;;:24;;;;;;;;:34;;;41810:26;;;:17;:26;;;;;;:35;;;;-1:-1:-1;41632:221:0:o;32405:382::-;32485:16;;;32477:61;;;;;;;14499:2:1;32477:61:0;;;14481:21:1;;;14518:18;;;14511:30;14577:34;14557:18;;;14550:62;14629:18;;32477:61:0;14297:356:1;32477:61:0;30388:4;30412:16;;;:7;:16;;;;;;:30;:16;:30;32549:58;;;;;;;9002:2:1;32549:58:0;;;8984:21:1;9041:2;9021:18;;;9014:30;9080;9060:18;;;9053:58;9128:18;;32549:58:0;8800:352:1;32549:58:0;32620:45;32649:1;32653:2;32657:7;32620:20;:45::i;:::-;32678:13;;;;;;;:9;:13;;;;;:18;;32695:1;;32678:13;:18;;32695:1;;32678:18;:::i;:::-;;;;-1:-1:-1;;32707:16:0;;;;:7;:16;;;;;;:21;;;;;;;;;;;;;32746:33;;32707:16;;;32746:33;;32707:16;;32746:33;32405:382;;:::o;-1:-1:-1:-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;14:465:1;78:5;112:18;104:6;101:30;98:56;;;134:18;;:::i;:::-;172:116;282:4;213:66;208:2;200:6;196:15;192:88;188:99;172:116;:::i;:::-;163:125;;311:6;304:5;297:21;351:3;342:6;337:3;333:16;330:25;327:45;;;368:1;365;358:12;327:45;417:6;412:3;405:4;398:5;394:16;381:43;471:1;464:4;455:6;448:5;444:18;440:29;433:40;14:465;;;;;:::o;484:247::-;543:6;596:2;584:9;575:7;571:23;567:32;564:52;;;612:1;609;602:12;564:52;651:9;638:23;670:31;695:5;670:31;:::i;736:388::-;804:6;812;865:2;853:9;844:7;840:23;836:32;833:52;;;881:1;878;871:12;833:52;920:9;907:23;939:31;964:5;939:31;:::i;:::-;989:5;-1:-1:-1;1046:2:1;1031:18;;1018:32;1059:33;1018:32;1059:33;:::i;:::-;1111:7;1101:17;;;736:388;;;;;:::o;1129:456::-;1206:6;1214;1222;1275:2;1263:9;1254:7;1250:23;1246:32;1243:52;;;1291:1;1288;1281:12;1243:52;1330:9;1317:23;1349:31;1374:5;1349:31;:::i;:::-;1399:5;-1:-1:-1;1456:2:1;1441:18;;1428:32;1469:33;1428:32;1469:33;:::i;:::-;1129:456;;1521:7;;-1:-1:-1;;;1575:2:1;1560:18;;;;1547:32;;1129:456::o;1590:794::-;1685:6;1693;1701;1709;1762:3;1750:9;1741:7;1737:23;1733:33;1730:53;;;1779:1;1776;1769:12;1730:53;1818:9;1805:23;1837:31;1862:5;1837:31;:::i;:::-;1887:5;-1:-1:-1;1944:2:1;1929:18;;1916:32;1957:33;1916:32;1957:33;:::i;:::-;2009:7;-1:-1:-1;2063:2:1;2048:18;;2035:32;;-1:-1:-1;2118:2:1;2103:18;;2090:32;2145:18;2134:30;;2131:50;;;2177:1;2174;2167:12;2131:50;2200:22;;2253:4;2245:13;;2241:27;-1:-1:-1;2231:55:1;;2282:1;2279;2272:12;2231:55;2305:73;2370:7;2365:2;2352:16;2347:2;2343;2339:11;2305:73;:::i;:::-;2295:83;;;1590:794;;;;;;;:::o;2389:416::-;2454:6;2462;2515:2;2503:9;2494:7;2490:23;2486:32;2483:52;;;2531:1;2528;2521:12;2483:52;2570:9;2557:23;2589:31;2614:5;2589:31;:::i;:::-;2639:5;-1:-1:-1;2696:2:1;2681:18;;2668:32;2738:15;;2731:23;2719:36;;2709:64;;2769:1;2766;2759:12;2810:315;2878:6;2886;2939:2;2927:9;2918:7;2914:23;2910:32;2907:52;;;2955:1;2952;2945:12;2907:52;2994:9;2981:23;3013:31;3038:5;3013:31;:::i;:::-;3063:5;3115:2;3100:18;;;;3087:32;;-1:-1:-1;;;2810:315:1:o;3130:1032::-;3214:6;3245:2;3288;3276:9;3267:7;3263:23;3259:32;3256:52;;;3304:1;3301;3294:12;3256:52;3344:9;3331:23;3373:18;3414:2;3406:6;3403:14;3400:34;;;3430:1;3427;3420:12;3400:34;3468:6;3457:9;3453:22;3443:32;;3513:7;3506:4;3502:2;3498:13;3494:27;3484:55;;3535:1;3532;3525:12;3484:55;3571:2;3558:16;3593:2;3589;3586:10;3583:36;;;3599:18;;:::i;:::-;3645:2;3642:1;3638:10;3628:20;;3668:28;3692:2;3688;3684:11;3668:28;:::i;:::-;3730:15;;;3761:12;;;;3793:11;;;3823;;;3819:20;;3816:33;-1:-1:-1;3813:53:1;;;3862:1;3859;3852:12;3813:53;3884:1;3875:10;;3894:238;3908:2;3905:1;3902:9;3894:238;;;3979:3;3966:17;3953:30;;3996:31;4021:5;3996:31;:::i;:::-;4040:18;;;3926:1;3919:9;;;;;4078:12;;;;4110;;3894:238;;;-1:-1:-1;4151:5:1;3130:1032;-1:-1:-1;;;;;;;;3130:1032:1:o;4167:245::-;4225:6;4278:2;4266:9;4257:7;4253:23;4249:32;4246:52;;;4294:1;4291;4284:12;4246:52;4333:9;4320:23;4352:30;4376:5;4352:30;:::i;4417:249::-;4486:6;4539:2;4527:9;4518:7;4514:23;4510:32;4507:52;;;4555:1;4552;4545:12;4507:52;4587:9;4581:16;4606:30;4630:5;4606:30;:::i;4671:287::-;4777:6;4830:2;4818:9;4809:7;4805:23;4801:32;4798:52;;;4846:1;4843;4836:12;4798:52;4878:9;4872:16;4897:31;4922:5;4897:31;:::i;4963:450::-;5032:6;5085:2;5073:9;5064:7;5060:23;5056:32;5053:52;;;5101:1;5098;5091:12;5053:52;5141:9;5128:23;5174:18;5166:6;5163:30;5160:50;;;5206:1;5203;5196:12;5160:50;5229:22;;5282:4;5274:13;;5270:27;-1:-1:-1;5260:55:1;;5311:1;5308;5301:12;5260:55;5334:73;5399:7;5394:2;5381:16;5376:2;5372;5368:11;5334:73;:::i;5418:180::-;5477:6;5530:2;5518:9;5509:7;5505:23;5501:32;5498:52;;;5546:1;5543;5536:12;5498:52;-1:-1:-1;5569:23:1;;5418:180;-1:-1:-1;5418:180:1:o;5603:316::-;5644:3;5682:5;5676:12;5709:6;5704:3;5697:19;5725:63;5781:6;5774:4;5769:3;5765:14;5758:4;5751:5;5747:16;5725:63;:::i;:::-;5833:2;5821:15;5838:66;5817:88;5808:98;;;;5908:4;5804:109;;5603:316;-1:-1:-1;;5603:316:1:o;5924:470::-;6103:3;6141:6;6135:13;6157:53;6203:6;6198:3;6191:4;6183:6;6179:17;6157:53;:::i;:::-;6273:13;;6232:16;;;;6295:57;6273:13;6232:16;6329:4;6317:17;;6295:57;:::i;:::-;6368:20;;5924:470;-1:-1:-1;;;;5924:470:1:o;6630:511::-;6824:4;6853:42;6934:2;6926:6;6922:15;6911:9;6904:34;6986:2;6978:6;6974:15;6969:2;6958:9;6954:18;6947:43;;7026:6;7021:2;7010:9;7006:18;6999:34;7069:3;7064:2;7053:9;7049:18;7042:31;7090:45;7130:3;7119:9;7115:19;7107:6;7090:45;:::i;:::-;7082:53;6630:511;-1:-1:-1;;;;;;6630:511:1:o;7338:219::-;7487:2;7476:9;7469:21;7450:4;7507:44;7547:2;7536:9;7532:18;7524:6;7507:44;:::i;19210:334::-;19281:2;19275:9;19337:2;19327:13;;19342:66;19323:86;19311:99;;19440:18;19425:34;;19461:22;;;19422:62;19419:88;;;19487:18;;:::i;:::-;19523:2;19516:22;19210:334;;-1:-1:-1;19210:334:1:o;19549:128::-;19589:3;19620:1;19616:6;19613:1;19610:13;19607:39;;;19626:18;;:::i;:::-;-1:-1:-1;19662:9:1;;19549:128::o;19682:120::-;19722:1;19748;19738:35;;19753:18;;:::i;:::-;-1:-1:-1;19787:9:1;;19682:120::o;19807:228::-;19847:7;19973:1;19905:66;19901:74;19898:1;19895:81;19890:1;19883:9;19876:17;19872:105;19869:131;;;19980:18;;:::i;:::-;-1:-1:-1;20020:9:1;;19807:228::o;20040:125::-;20080:4;20108:1;20105;20102:8;20099:34;;;20113:18;;:::i;:::-;-1:-1:-1;20150:9:1;;20040:125::o;20170:258::-;20242:1;20252:113;20266:6;20263:1;20260:13;20252:113;;;20342:11;;;20336:18;20323:11;;;20316:39;20288:2;20281:10;20252:113;;;20383:6;20380:1;20377:13;20374:48;;;-1:-1:-1;;20418:1:1;20400:16;;20393:27;20170:258::o;20433:437::-;20512:1;20508:12;;;;20555;;;20576:61;;20630:4;20622:6;20618:17;20608:27;;20576:61;20683:2;20675:6;20672:14;20652:18;20649:38;20646:218;;;20720:77;20717:1;20710:88;20821:4;20818:1;20811:15;20849:4;20846:1;20839:15;20646:218;;20433:437;;;:::o;20875:195::-;20914:3;20945:66;20938:5;20935:77;20932:103;;;21015:18;;:::i;:::-;-1:-1:-1;21062:1:1;21051:13;;20875:195::o;21075:112::-;21107:1;21133;21123:35;;21138:18;;:::i;:::-;-1:-1:-1;21172:9:1;;21075:112::o;21192:184::-;21244:77;21241:1;21234:88;21341:4;21338:1;21331:15;21365:4;21362:1;21355:15;21381:184;21433:77;21430:1;21423:88;21530:4;21527:1;21520:15;21554:4;21551:1;21544:15;21570:184;21622:77;21619:1;21612:88;21719:4;21716:1;21709:15;21743:4;21740:1;21733:15;21759:184;21811:77;21808:1;21801:88;21908:4;21905:1;21898:15;21932:4;21929:1;21922:15;21948:184;22000:77;21997:1;21990:88;22097:4;22094:1;22087:15;22121:4;22118:1;22111:15;22137:154;22223:42;22216:5;22212:54;22205:5;22202:65;22192:93;;22281:1;22278;22271:12;22296:177;22381:66;22374:5;22370:78;22363:5;22360:89;22350:117;;22463:1;22460;22453:12

Swarm Source

ipfs://2f5e99d7387fb45f3f3c9871e5b1b1b4871095d87c791d7859dfd95b5aa1c647
Loading...
Loading
Loading...
Loading
[ Download: CSV Export  ]
[ Download: CSV Export  ]

A token is a representation of an on-chain or off-chain asset. The token page shows information such as price, total supply, holders, transfers and social links. Learn more about this page in our Knowledge Base.