ERC-721
Overview
Max Total Supply
3,333 TN
Holders
1,711
Market
Volume (24H)
N/A
Min Price (24H)
N/A
Max Price (24H)
N/A
Other Info
Token Contract
Balance
1 TNLoading...
Loading
Loading...
Loading
Loading...
Loading
# | Exchange | Pair | Price | 24H Volume | % Volume |
---|
Contract Source Code Verified (Exact Match)
Contract Name:
Nakedz
Compiler Version
v0.8.13+commit.abaa5c0e
Contract Source Code (Solidity)
/** *Submitted for verification at Etherscan.io on 2022-08-06 */ // SPDX-License-Identifier: MIT // File: @openzeppelin/contracts/utils/cryptography/MerkleProof.sol // OpenZeppelin Contracts (last updated v4.7.0) (utils/cryptography/MerkleProof.sol) pragma solidity ^0.8.0; /* @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@::;:;;:;;::;::;;:;::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@::::::::::::::::::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@::::::::::::::::::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@r@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@+:.::::::::::::::::::::.:+@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@::::::::::,,,,,,,,,,,,::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@::::::::::,,,,,,,,,,,,::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@::::::::::,,,,,,,,,,,,::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@;::::::::::,,,,,,,,,,,,,,,,:::::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@:::::::::::,,,,,,,,,,,,,,,,,,,:::;@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@h@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@a@@@@@@@@@@ @@:::::::::::,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@::::@@@@@@ @@::::::::::,,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@::::::@@@@@::::::@@@@@@::::::::@@@@@@::::::@@@@:::::::::::::::::::::@::::::::::::@@@@@@@@::::::::::@@@ @@:::::::::::,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@:::::::@@@@::::::@@@@@:::::::::@@@@@@::::::@@@;::::::.::::::::::::::@::::::::::::::@@@@@::::::::::::@@ @@:::::::::::,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@:::::::;@@@::::::@@@@@:::::::::@@@@@@::::::@@.::::::.@::::::::::::::@:::::::::::::::@@@::::::::::::::@ @@::::::::::,,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@::::::::@@@::::::@@@@@::::::::::@@@@@::::::@@:::::::@@::::::::::::::@:::::::::::::::.@:::::::::::::::: @@:::::::::::,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@:::::::::@@::::::@@@@:::::::::::@@@@@::::::@:::::::@@@:::::::::::::;.::::::::::::::::@:::::::;:::::::: @@:::::::::::,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@:::::::::;@::::::@@@@:::::::::::@@@@@:::::::::::::@@@@::::::*@@@@@@@.::::::@@@:::::::@:::::::@@@:::::: @@:::::::::::,,,,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@::::::::::@::::::@@@@::::::::::::@@@@::::::::::::+@@@@:::::::::::::@.::::::@@@@::::::.::::::::::;@@@@@ @@::::::::::::::,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@:::::::::::::::::@@@::::::@::::::@@@@::::::::::::@@@@@:::::::::::::@.::::::@@@@::::::::::::::::::::;@@ @@@:::::::::::::,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@:::::::::::::::::@@@::::::@::::::@@@@::::::::::::@@@@@:::::::::::::@.::::::@@@@:::::::@::::::::::::::@ ::::::::::::::::,,,,,,,,,,,,,,,:::::@@@@@@@@@@@@:::::::::::::::::@@@::::::@;:::::;@@@::::::::::::.@@@@:::::::::::::@.::::::@@@@:::::::@@:::::::::::::: :::,,,,:::::::::,,,,,,,,,,,,::::::::@@@@@@@@@@@@:::::::::::::::::@@:::::::::::::::@@@:::::::::::::@@@@:::::::::::::@.::::::@@@@:::::::@@@+:::::::::::: :::,,,,:::::::::,,,,,,,,,,,,::::::::@@@@@@@@@@@@:::::::;:::::::::@@:::::::::::::::@@@:::::::::::::;@@@:::::::@@@@@@@.::::::@@@@::::::...@@@@@::::::::: :::,,,,:::::::::,,,,,,,,,,,,::::::::@@@@@@@@@@@@:::::::@:::::::::@@::::::::::::::::@@::::::::::::::@@@:::::::@@@@@@@.::::::@@;:::::::@::::::@@@@:::::: ::::,,::::::::::,,,,,,,,,,,,::::::::@@@@@@@@@@@@:::::::@@::::::::@:::::::::::::::::@@::::::@*:::::::@@:::::::::::::;.::::::::::::::::@:::::::::::::::: ::::,:::::::::::,,,,,,,,,,,,::::::::@@@@@@@@@@@@:::::::@@::::::::@:::::::::::::::::@@::::::@@:::::::;@::::::::::::::@:::::::::::::::@@:::::::::::::::: ::::::::::::::::,,,,,,,,,,,,::::::::@@@@@@@@@@@@:::::::@@@:::::::@::::::@@@@@:::::::@::::::@@@:::::::@::::::::::::::@::::::::::::::@@@@::::::::::::::@ ::::::::::::::::,,,,,,,,,,,,,,,:::::@@@@@@@@@@@@:::::::@@@@:::::::::::::@@@@@:::::::@::::::@@@::::::::::::::::::::::@:::::::::::::@@@@@@::::::::::::@@ @@,:::::::::::::,,,,,,,,,,,,,,,,::::@@@@@@@@@@@@;::::::@@@@*::::::::::::@@@@@+::::::@::::::@@@@:::::::::::::::::::::@:::::::::::@@@@@@@@@:::::::::;@@@ @@::::::::::::::o,,,,,,,,,,,,,,,::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@.::::::::::::::,:,,,,,,,,,:::::::.@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@:::::::::::::,,,,,,,,,::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@r@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@:::::::::::::,,,,,,,,,::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@::::::::::::::::::::::::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@::::::::::::::::::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@::::::::::::::::::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@:::::::::::::s::::::@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ */ /** * @dev These functions deal with verification of Merkle Tree proofs. * * The proofs can be generated using the JavaScript library * https://github.com/miguelmota/merkletreejs[merkletreejs]. * Note: the hashing algorithm should be keccak256 and pair sorting should be enabled. * * See `test/utils/cryptography/MerkleProof.test.js` for some examples. * * WARNING: You should avoid using leaf values that are 64 bytes long prior to * hashing, or use a hash function other than keccak256 for hashing leaves. * This is because the concatenation of a sorted pair of internal nodes in * the merkle tree could be reinterpreted as a leaf value. */ library MerkleProof { /** * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree * defined by `root`. For this, a `proof` must be provided, containing * sibling hashes on the branch from the leaf to the root of the tree. Each * pair of leaves and each pair of pre-images are assumed to be sorted. */ function verify( bytes32[] memory proof, bytes32 root, bytes32 leaf ) internal pure returns (bool) { return processProof(proof, leaf) == root; } /** * @dev Calldata version of {verify} * * _Available since v4.7._ */ function verifyCalldata( bytes32[] calldata proof, bytes32 root, bytes32 leaf ) internal pure returns (bool) { return processProofCalldata(proof, leaf) == root; } /** * @dev Returns the rebuilt hash obtained by traversing a Merkle tree up * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt * hash matches the root of the tree. When processing the proof, the pairs * of leafs & pre-images are assumed to be sorted. * * _Available since v4.4._ */ function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Calldata version of {processProof} * * _Available since v4.7._ */ function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Returns true if the `leaves` can be proved to be a part of a Merkle tree defined by * `root`, according to `proof` and `proofFlags` as described in {processMultiProof}. * * _Available since v4.7._ */ function multiProofVerify( bytes32[] memory proof, bool[] memory proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProof(proof, proofFlags, leaves) == root; } /** * @dev Calldata version of {multiProofVerify} * * _Available since v4.7._ */ function multiProofVerifyCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProofCalldata(proof, proofFlags, leaves) == root; } /** * @dev Returns the root of a tree reconstructed from `leaves` and the sibling nodes in `proof`, * consuming from one or the other at each step according to the instructions given by * `proofFlags`. * * _Available since v4.7._ */ function processMultiProof( bytes32[] memory proof, bool[] memory proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the merkle tree. uint256 leavesLen = leaves.length; uint256 totalHashes = proofFlags.length; // Check proof validity. require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof"); // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value for the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { return hashes[totalHashes - 1]; } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } /** * @dev Calldata version of {processMultiProof} * * _Available since v4.7._ */ function processMultiProofCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the merkle tree. uint256 leavesLen = leaves.length; uint256 totalHashes = proofFlags.length; // Check proof validity. require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof"); // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value for the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { return hashes[totalHashes - 1]; } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) { return a < b ? _efficientHash(a, b) : _efficientHash(b, a); } function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) { /// @solidity memory-safe-assembly assembly { mstore(0x00, a) mstore(0x20, b) value := keccak256(0x00, 0x40) } } } // File: contracts/Nakeds.sol // HI OS 01.06.2022 pragma solidity ^0.8.4; /** * @dev Interface of an ERC721A compliant contract. */ interface IERC721A { /** * The caller must own the token or be an approved operator. */ error ApprovalCallerNotOwnerNorApproved(); /** * The token does not exist. */ error ApprovalQueryForNonexistentToken(); /** * The caller cannot approve to their own address. */ error ApproveToCaller(); /** * The caller cannot approve to the current owner. */ error ApprovalToCurrentOwner(); /** * Cannot query the balance for the zero address. */ error BalanceQueryForZeroAddress(); /** * Cannot mint to the zero address. */ error MintToZeroAddress(); /** * The quantity of tokens minted must be more than zero. */ error MintZeroQuantity(); /** * The token does not exist. */ error OwnerQueryForNonexistentToken(); /** * The caller must own the token or be an approved operator. */ error TransferCallerNotOwnerNorApproved(); /** * The token must be owned by `from`. */ error TransferFromIncorrectOwner(); /** * Cannot safely transfer to a contract that does not implement the ERC721Receiver interface. */ error TransferToNonERC721ReceiverImplementer(); /** * Cannot transfer to the zero address. */ error TransferToZeroAddress(); /** * The token does not exist. */ error URIQueryForNonexistentToken(); struct TokenOwnership { // The address of the owner. address addr; // Keeps track of the start time of ownership with minimal overhead for tokenomics. uint64 startTimestamp; // Whether the token has been burned. bool burned; } /** * @dev Returns the total amount of tokens stored by the contract. * * Burned tokens are calculated here, use `_totalMinted()` if you want to count just minted tokens. */ function totalSupply() external view returns (uint256); // ============================== // 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); // ============================== // IERC721 // ============================== /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in ``owner``'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external; /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must 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 Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll} */ function isApprovedForAll(address owner, address operator) external view returns (bool); // ============================== // IERC721Metadata // ============================== /** * @dev Returns the token collection name. */ function name() external view returns (string memory); /** * @dev Returns the token collection symbol. */ function symbol() external view returns (string memory); /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) external view returns (string memory); } // File: https://github.com/chiru-labs/ERC721A/blob/main/contracts/ERC721A.sol // ERC721A Contracts v3.3.0 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @dev ERC721 token receiver interface. */ interface ERC721A__IERC721Receiver { function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } /** * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including * the Metadata extension. Built to optimize for lower gas during batch mints. * * Assumes serials are sequentially minted starting at _startTokenId() (defaults to 0, e.g. 0, 1, 2, 3..). * * Assumes that an owner cannot have more than 2**64 - 1 (max value of uint64) of supply. * * Assumes that the maximum token id cannot exceed 2**256 - 1 (max value of uint256). */ contract ERC721A is IERC721A { // Mask of an entry in packed address data. uint256 private constant BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1; // The bit position of `numberMinted` in packed address data. uint256 private constant BITPOS_NUMBER_MINTED = 64; // The bit position of `numberBurned` in packed address data. uint256 private constant BITPOS_NUMBER_BURNED = 128; // The bit position of `aux` in packed address data. uint256 private constant BITPOS_AUX = 192; // Mask of all 256 bits in packed address data except the 64 bits for `aux`. uint256 private constant BITMASK_AUX_COMPLEMENT = (1 << 192) - 1; // The bit position of `startTimestamp` in packed ownership. uint256 private constant BITPOS_START_TIMESTAMP = 160; // The bit mask of the `burned` bit in packed ownership. uint256 private constant BITMASK_BURNED = 1 << 224; // The bit position of the `nextInitialized` bit in packed ownership. uint256 private constant BITPOS_NEXT_INITIALIZED = 225; // The bit mask of the `nextInitialized` bit in packed ownership. uint256 private constant BITMASK_NEXT_INITIALIZED = 1 << 225; // The tokenId of the next token to be minted. uint256 private _currentIndex; // The number of tokens burned. uint256 private _burnCounter; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to ownership details // An empty struct value does not necessarily mean the token is unowned. // See `_packedOwnershipOf` implementation for details. // // Bits Layout: // - [0..159] `addr` // - [160..223] `startTimestamp` // - [224] `burned` // - [225] `nextInitialized` mapping(uint256 => uint256) private _packedOwnerships; // Mapping owner address to address data. // // Bits Layout: // - [0..63] `balance` // - [64..127] `numberMinted` // - [128..191] `numberBurned` // - [192..255] `aux` mapping(address => uint256) private _packedAddressData; // Mapping from token ID to approved address. mapping(uint256 => address) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; _currentIndex = _startTokenId(); } /** * @dev Returns the starting token ID. * To change the starting token ID, please override this function. */ function _startTokenId() internal view virtual returns (uint256) { return 0; } /** * @dev Returns the next token ID to be minted. */ function _nextTokenId() internal view returns (uint256) { return _currentIndex; } /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see `_totalMinted`. */ function totalSupply() public view override returns (uint256) { // Counter underflow is impossible as _burnCounter cannot be incremented // more than `_currentIndex - _startTokenId()` times. unchecked { return _currentIndex - _burnCounter - _startTokenId(); } } /** * @dev Returns the total amount of tokens minted in the contract. */ function _totalMinted() internal view returns (uint256) { // Counter underflow is impossible as _currentIndex does not decrement, // and it is initialized to `_startTokenId()` unchecked { return _currentIndex - _startTokenId(); } } /** * @dev Returns the total number of tokens burned. */ function _totalBurned() internal view returns (uint256) { return _burnCounter; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { // The interface IDs are constants representing the first 4 bytes of the XOR of // all function selectors in the interface. See: https://eips.ethereum.org/EIPS/eip-165 // e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)` return interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165. interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721. interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata. } /** * @dev See {IERC721-balanceOf}. */ function balanceOf(address owner) public view override returns (uint256) { if (owner == address(0)) revert BalanceQueryForZeroAddress(); return _packedAddressData[owner] & BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens minted by `owner`. */ function _numberMinted(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> BITPOS_NUMBER_MINTED) & BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens burned by or on behalf of `owner`. */ function _numberBurned(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> BITPOS_NUMBER_BURNED) & BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the auxillary data for `owner`. (e.g. number of whitelist mint slots used). */ function _getAux(address owner) internal view returns (uint64) { return uint64(_packedAddressData[owner] >> BITPOS_AUX); } /** * Sets the auxillary data for `owner`. (e.g. number of whitelist mint slots used). * If there are multiple variables, please pack them into a uint64. */ function _setAux(address owner, uint64 aux) internal { uint256 packed = _packedAddressData[owner]; uint256 auxCasted; assembly { // Cast aux without masking. auxCasted := aux } packed = (packed & BITMASK_AUX_COMPLEMENT) | (auxCasted << BITPOS_AUX); _packedAddressData[owner] = packed; } /** * Returns the packed ownership data of `tokenId`. */ function _packedOwnershipOf(uint256 tokenId) private view returns (uint256) { uint256 curr = tokenId; unchecked { if (_startTokenId() <= curr) if (curr < _currentIndex) { uint256 packed = _packedOwnerships[curr]; // If not burned. if (packed & BITMASK_BURNED == 0) { // Invariant: // There will always be an ownership that has an address and is not burned // before an ownership that does not have an address and is not burned. // Hence, curr will not underflow. // // We can directly compare the packed value. // If the address is zero, packed is zero. while (packed == 0) { packed = _packedOwnerships[--curr]; } return packed; } } } revert OwnerQueryForNonexistentToken(); } /** * Returns the unpacked `TokenOwnership` struct from `packed`. */ function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) { ownership.addr = address(uint160(packed)); ownership.startTimestamp = uint64(packed >> BITPOS_START_TIMESTAMP); ownership.burned = packed & BITMASK_BURNED != 0; } /** * Returns the unpacked `TokenOwnership` struct at `index`. */ function _ownershipAt(uint256 index) internal view returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnerships[index]); } /** * @dev Initializes the ownership slot minted at `index` for efficiency purposes. */ function _initializeOwnershipAt(uint256 index) internal { if (_packedOwnerships[index] == 0) { _packedOwnerships[index] = _packedOwnershipOf(index); } } /** * Gas spent here starts off proportional to the maximum mint batch size. * It gradually moves to O(1) as tokens get transferred around in the collection over time. */ function _ownershipOf(uint256 tokenId) internal view returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnershipOf(tokenId)); } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view override returns (address) { return address(uint160(_packedOwnershipOf(tokenId))); } /** * @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) { if (!_exists(tokenId)) revert URIQueryForNonexistentToken(); string memory baseURI = _baseURI(); return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, _toString(tokenId))) : ''; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, can be overriden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ''; } /** * @dev Casts the address to uint256 without masking. */ function _addressToUint256(address value) private pure returns (uint256 result) { assembly { result := value } } /** * @dev Casts the boolean to uint256 without branching. */ function _boolToUint256(bool value) private pure returns (uint256 result) { assembly { result := value } } /** * @dev See {IERC721-approve}. */ function approve(address to, uint256 tokenId) public override { address owner = address(uint160(_packedOwnershipOf(tokenId))); if (to == owner) revert ApprovalToCurrentOwner(); if (_msgSenderERC721A() != owner) if (!isApprovedForAll(owner, _msgSenderERC721A())) { revert ApprovalCallerNotOwnerNorApproved(); } _tokenApprovals[tokenId] = to; emit Approval(owner, to, tokenId); } /** * @dev See {IERC721-getApproved}. */ function getApproved(uint256 tokenId) public view override returns (address) { if (!_exists(tokenId)) revert ApprovalQueryForNonexistentToken(); return _tokenApprovals[tokenId]; } /** * @dev See {IERC721-setApprovalForAll}. */ function setApprovalForAll(address operator, bool approved) public virtual override { if (operator == _msgSenderERC721A()) revert ApproveToCaller(); _operatorApprovals[_msgSenderERC721A()][operator] = approved; emit ApprovalForAll(_msgSenderERC721A(), 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 { _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 { _transfer(from, to, tokenId); if (to.code.length != 0) if (!_checkContractOnERC721Received(from, to, tokenId, _data)) { revert TransferToNonERC721ReceiverImplementer(); } } /** * @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`), */ function _exists(uint256 tokenId) internal view returns (bool) { return _startTokenId() <= tokenId && tokenId < _currentIndex && // If within bounds, _packedOwnerships[tokenId] & BITMASK_BURNED == 0; // and not burned. } /** * @dev Equivalent to `_safeMint(to, quantity, '')`. */ function _safeMint(address to, uint256 quantity) internal { _safeMint(to, quantity, ''); } /** * @dev Safely mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called for each safe transfer. * - `quantity` must be greater than 0. * * Emits a {Transfer} event. */ function _safeMint( address to, uint256 quantity, bytes memory _data ) internal { uint256 startTokenId = _currentIndex; if (to == address(0)) revert MintToZeroAddress(); if (quantity == 0) revert MintZeroQuantity(); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are incredibly unrealistic. // balance or numberMinted overflow if current value of either + quantity > 1.8e19 (2**64) - 1 // updatedIndex overflows if _currentIndex + quantity > 1.2e77 (2**256) - 1 unchecked { // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the balance and number minted. _packedAddressData[to] += quantity * ((1 << BITPOS_NUMBER_MINTED) | 1); // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _addressToUint256(to) | (block.timestamp << BITPOS_START_TIMESTAMP) | (_boolToUint256(quantity == 1) << BITPOS_NEXT_INITIALIZED); uint256 updatedIndex = startTokenId; uint256 end = updatedIndex + quantity; if (to.code.length != 0) { do { emit Transfer(address(0), to, updatedIndex); if (!_checkContractOnERC721Received(address(0), to, updatedIndex++, _data)) { revert TransferToNonERC721ReceiverImplementer(); } } while (updatedIndex < end); // Reentrancy protection if (_currentIndex != startTokenId) revert(); } else { do { emit Transfer(address(0), to, updatedIndex++); } while (updatedIndex < end); } _currentIndex = updatedIndex; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {Transfer} event. */ function _mint(address to, uint256 quantity) internal { uint256 startTokenId = _currentIndex; if (to == address(0)) revert MintToZeroAddress(); if (quantity == 0) revert MintZeroQuantity(); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are incredibly unrealistic. // balance or numberMinted overflow if current value of either + quantity > 1.8e19 (2**64) - 1 // updatedIndex overflows if _currentIndex + quantity > 1.2e77 (2**256) - 1 unchecked { // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the balance and number minted. _packedAddressData[to] += quantity * ((1 << BITPOS_NUMBER_MINTED) | 1); // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _addressToUint256(to) | (block.timestamp << BITPOS_START_TIMESTAMP) | (_boolToUint256(quantity == 1) << BITPOS_NEXT_INITIALIZED); uint256 updatedIndex = startTokenId; uint256 end = updatedIndex + quantity; do { emit Transfer(address(0), to, updatedIndex++); } while (updatedIndex < end); _currentIndex = updatedIndex; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Transfers `tokenId` from `from` to `to`. * * 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 ) private { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); if (address(uint160(prevOwnershipPacked)) != from) revert TransferFromIncorrectOwner(); bool isApprovedOrOwner = (_msgSenderERC721A() == from || isApprovedForAll(from, _msgSenderERC721A()) || getApproved(tokenId) == _msgSenderERC721A()); if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved(); if (to == address(0)) revert TransferToZeroAddress(); _beforeTokenTransfers(from, to, tokenId, 1); // Clear approvals from the previous owner. delete _tokenApprovals[tokenId]; // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as tokenId would have to be 2**256. unchecked { // We can directly increment and decrement the balances. --_packedAddressData[from]; // Updates: `balance -= 1`. ++_packedAddressData[to]; // Updates: `balance += 1`. // Updates: // - `address` to the next owner. // - `startTimestamp` to the timestamp of transfering. // - `burned` to `false`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _addressToUint256(to) | (block.timestamp << BITPOS_START_TIMESTAMP) | BITMASK_NEXT_INITIALIZED; // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } emit Transfer(from, to, tokenId); _afterTokenTransfers(from, to, tokenId, 1); } /** * @dev Equivalent to `_burn(tokenId, false)`. */ function _burn(uint256 tokenId) internal virtual { _burn(tokenId, false); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId, bool approvalCheck) internal virtual { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); address from = address(uint160(prevOwnershipPacked)); if (approvalCheck) { bool isApprovedOrOwner = (_msgSenderERC721A() == from || isApprovedForAll(from, _msgSenderERC721A()) || getApproved(tokenId) == _msgSenderERC721A()); if (!isApprovedOrOwner) revert TransferCallerNotOwnerNorApproved(); } _beforeTokenTransfers(from, address(0), tokenId, 1); // Clear approvals from the previous owner. delete _tokenApprovals[tokenId]; // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as tokenId would have to be 2**256. unchecked { // Updates: // - `balance -= 1`. // - `numberBurned += 1`. // // We can directly decrement the balance, and increment the number burned. // This is equivalent to `packed -= 1; packed += 1 << BITPOS_NUMBER_BURNED;`. _packedAddressData[from] += (1 << BITPOS_NUMBER_BURNED) - 1; // Updates: // - `address` to the last owner. // - `startTimestamp` to the timestamp of burning. // - `burned` to `true`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _addressToUint256(from) | (block.timestamp << BITPOS_START_TIMESTAMP) | BITMASK_BURNED | BITMASK_NEXT_INITIALIZED; // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } emit Transfer(from, address(0), tokenId); _afterTokenTransfers(from, address(0), tokenId, 1); // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times. unchecked { _burnCounter++; } } /** * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target 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 _checkContractOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns ( bytes4 retval ) { return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert TransferToNonERC721ReceiverImplementer(); } else { assembly { revert(add(32, reason), mload(reason)) } } } } /** * @dev Hook that is called before a set of serially-ordered token ids are about to be transferred. This includes minting. * And also called before burning one token. * * startTokenId - the first token id to be transferred * quantity - the amount to be transferred * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _beforeTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Hook that is called after a set of serially-ordered token ids have been transferred. This includes * minting. * And also called after one token has been burned. * * startTokenId - the first token id to be transferred * quantity - the amount to be transferred * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been * transferred to `to`. * - When `from` is zero, `tokenId` has been minted for `to`. * - When `to` is zero, `tokenId` has been burned by `from`. * - `from` and `to` are never both zero. */ function _afterTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Returns the message sender (defaults to `msg.sender`). * * If you are writing GSN compatible contracts, you need to override this function. */ function _msgSenderERC721A() internal view virtual returns (address) { return msg.sender; } /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function _toString(uint256 value) internal pure returns (string memory ptr) { assembly { // The maximum value of a uint256 contains 78 digits (1 byte per digit), // but we allocate 128 bytes to keep the free memory pointer 32-byte word aliged. // We will need 1 32-byte word to store the length, // and 3 32-byte words to store a maximum of 78 digits. Total: 32 + 3 * 32 = 128. ptr := add(mload(0x40), 128) // Update the free memory pointer to allocate. mstore(0x40, ptr) // Cache the end of the memory to calculate the length later. let end := ptr // We write the string from the rightmost digit to the leftmost digit. // The following is essentially a do-while loop that also handles the zero case. // Costs a bit more than early returning for the zero case, // but cheaper in terms of deployment and overall runtime costs. for { // Initialize and perform the first pass without check. let temp := value // Move the pointer 1 byte leftwards to point to an empty character slot. ptr := sub(ptr, 1) // Write the character to the pointer. 48 is the ASCII index of '0'. mstore8(ptr, add(48, mod(temp, 10))) temp := div(temp, 10) } temp { // Keep dividing `temp` until zero. temp := div(temp, 10) } { // Body of the for loop. ptr := sub(ptr, 1) mstore8(ptr, add(48, mod(temp, 10))) } let length := sub(end, ptr) // Move the pointer 32 bytes leftwards to make room for the length. ptr := sub(ptr, 32) // Store the length. mstore(ptr, length) } } } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/Strings.sol // OpenZeppelin Contracts v4.4.1 (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/Context.sol // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/access/Ownable.sol // OpenZeppelin Contracts v4.4.1 (access/Ownable.sol) pragma solidity ^0.8.0; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(owner() == _msgSender(), "Ownable: caller is not the owner"); _; } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/Address.sol // OpenZeppelin Contracts (last updated v4.5.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC721/IERC721Receiver.sol // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol) 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 `IERC721Receiver.onERC721Received.selector`. */ function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/introspection/IERC165.sol // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/introspection/ERC165.sol // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol) 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: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC721/IERC721.sol // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721.sol) 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`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external; /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Transfers `tokenId` token from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external; /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll} */ function isApprovedForAll(address owner, address operator) external view returns (bool); } // File: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC721/extensions/IERC721Metadata.sol // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol) 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: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/token/ERC721/ERC721.sol // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/ERC721.sol) 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: address zero is not a valid owner"); 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 overridden 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 { _setApprovalForAll(_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 || isApprovedForAll(owner, spender) || getApproved(tokenId) == 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); _afterTokenTransfer(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); _afterTokenTransfer(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 from incorrect owner"); require(to != address(0), "ERC721: transfer to the zero address"); _beforeTokenTransfer(from, to, tokenId); // Clear approvals from the previous owner _approve(address(0), tokenId); _balances[from] -= 1; _balances[to] += 1; _owners[tokenId] = to; emit Transfer(from, to, tokenId); _afterTokenTransfer(from, to, tokenId); } /** * @dev Approve `to` to operate on `tokenId` * * Emits an {Approval} event. */ function _approve(address to, uint256 tokenId) internal virtual { _tokenApprovals[tokenId] = to; emit Approval(ERC721.ownerOf(tokenId), to, tokenId); } /** * @dev Approve `operator` to operate on all of `owner` tokens * * Emits an {ApprovalForAll} event. */ function _setApprovalForAll( address owner, address operator, bool approved ) internal virtual { require(owner != operator, "ERC721: approve to caller"); _operatorApprovals[owner][operator] = approved; emit ApprovalForAll(owner, operator, approved); } /** * @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; } } //sf852022 /** * @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 {} /** * @dev Hook that is called after any transfer of tokens. This includes * minting and burning. * * Calling conditions: * * - when `from` and `to` are both non-zero. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _afterTokenTransfer( address from, address to, uint256 tokenId ) internal virtual {} } pragma solidity ^0.8.0; contract Nakedz is ERC721A, Ownable { using Strings for uint256; string private baseURI; uint256 public price = 0.0055 ether; uint256 public maxPerTx = 5; uint256 public maxPerWallet = 5; string public hiddenURI="ipfs://bafybeih5jcdvmsiut4v6wmctjp5c2csh3pploohgya7kcarh563ixr7lci/hidden.json"; bool public revealed = false; uint256 public maxRaffle = 1; uint256 public maxFree = 1; uint256 public maxWL = 2; uint256 public maxSupply = 5555; bool public nakedList = false; bool public allowList = false; bool public publicEnabled = false; bytes32 whitelistRoot; bytes32 raffleRoot; mapping(address => uint256 ) private _mintedFreeAmount; constructor() ERC721A("Nakeds", "TN") { } function nakedMint(bytes32[] calldata _merkleProof,uint256 count) external payable { uint256 cost = price; bytes32 leaf = keccak256(abi.encodePacked(msg.sender)); if(_mintedFreeAmount[msg.sender]<maxFree && count==1){ cost=0; } require(MerkleProof.verify(_merkleProof, whitelistRoot, leaf),"Incorrect Whitelist Proof"); require(msg.value >= cost, "Please send the exact amount."); require(numberMinted(msg.sender)+ count <=maxWL,"You cant mint anymore"); require(totalSupply() + count <= maxSupply , "No more"); require(count>0,"Please enter a number"); require(nakedList, "Minting is not live yet"); _mintedFreeAmount[msg.sender]+=count; _safeMint(msg.sender, count); } function allowListMint(bytes32[] calldata _merkleProof) external payable { bytes32 leaf = keccak256(abi.encodePacked(msg.sender)); require(MerkleProof.verify(_merkleProof, raffleRoot, leaf),"Incorrect Whitelist Proof"); require(msg.value >= price, "Please send the exact amount."); require(numberMinted(msg.sender)+1 <=maxRaffle,"You cant mint anymore"); require(totalSupply()+1<= maxSupply , "No more"); require(allowList, "Minting is not live yet"); _safeMint(msg.sender, 1); } function publicMint(uint256 count) external payable { require(msg.value >= count * price, "Please send the exact amount."); require(numberMinted(msg.sender)+ count <=maxPerWallet,"You cant mint anymore"); require(totalSupply() + count <= maxSupply , "No more"); require(count>0,"Please enter a number"); require(publicEnabled, "Minting is not live yet"); require(count<=maxPerTx,"Max per tx exceeded"); _safeMint(msg.sender, count); } function _baseURI() internal view virtual override returns (string memory) { return baseURI; } function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { require( _exists(tokenId), "ERC721AMetadata: URI query for nonexistent token" ); if (revealed == false) { return hiddenURI; } string memory currentBaseURI = _baseURI(); return bytes(currentBaseURI).length > 0 ? string(abi.encodePacked(currentBaseURI, tokenId.toString(), ".json")) : ""; } function setBaseURI(string memory uri) public onlyOwner { baseURI = uri; } function setHiddenURI(string memory uri) public onlyOwner { hiddenURI = uri; } function setMaxWL(uint256 amount) external onlyOwner { maxWL = amount; } function setFree(uint256 amount) external onlyOwner { maxFree = amount; } function setMaxPerWallet(uint256 amount) external onlyOwner { maxPerWallet = amount; } function setPrice(uint256 _newPrice) external onlyOwner { price = _newPrice; } function flipSale(bool status) external onlyOwner { nakedList = status; allowList= status; } function flipNakedList(bool status) external onlyOwner { nakedList = status; } function flipallowList(bool status) external onlyOwner { allowList=status; } function flipPublic(bool status) external onlyOwner { publicEnabled = status; } function reveal() external onlyOwner { revealed = !revealed; } function setPreSaleRoot(bytes32 _presaleRoot_1, bytes32 _presaleRoot_2) external onlyOwner { whitelistRoot = _presaleRoot_1; raffleRoot = _presaleRoot_2; } function numberMinted(address owner) public view returns (uint256) { return _numberMinted(owner); } function batchmint(uint256 _mintAmount, address destination) public onlyOwner { require(_mintAmount > 0, "need to mint at least 1 NFT"); uint256 supply = totalSupply(); require(supply + _mintAmount <= maxSupply, "max NFT limit exceeded"); _safeMint(destination, _mintAmount); } function withdraw() external onlyOwner { (bool success, ) = payable(msg.sender).call{ value: address(this).balance }(""); require(success, "Transfer failed."); } }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"ApprovalToCurrentOwner","type":"error"},{"inputs":[],"name":"ApproveToCaller","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"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":"allowList","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32[]","name":"_merkleProof","type":"bytes32[]"}],"name":"allowListMint","outputs":[],"stateMutability":"payable","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":[{"internalType":"uint256","name":"_mintAmount","type":"uint256"},{"internalType":"address","name":"destination","type":"address"}],"name":"batchmint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"status","type":"bool"}],"name":"flipNakedList","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"status","type":"bool"}],"name":"flipPublic","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"status","type":"bool"}],"name":"flipSale","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"status","type":"bool"}],"name":"flipallowList","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":[],"name":"hiddenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"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":"maxFree","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxPerTx","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxPerWallet","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxRaffle","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxWL","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"nakedList","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32[]","name":"_merkleProof","type":"bytes32[]"},{"internalType":"uint256","name":"count","type":"uint256"}],"name":"nakedMint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"numberMinted","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"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":"price","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"count","type":"uint256"}],"name":"publicMint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"reveal","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"revealed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"uri","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"setFree","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"uri","type":"string"}],"name":"setHiddenURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"setMaxPerWallet","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"setMaxWL","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"_presaleRoot_1","type":"bytes32"},{"internalType":"bytes32","name":"_presaleRoot_2","type":"bytes32"}],"name":"setPreSaleRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newPrice","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":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
608060405266138a388a43c000600a556005600b556005600c556040518060800160405280604e815260200162004aae604e9139600d90805190602001906200004a92919062000289565b506000600e60006101000a81548160ff0219169083151502179055506001600f55600160105560026011556115b36012556000601360006101000a81548160ff0219169083151502179055506000601360016101000a81548160ff0219169083151502179055506000601360026101000a81548160ff021916908315150217905550348015620000d957600080fd5b506040518060400160405280600681526020017f4e616b65647300000000000000000000000000000000000000000000000000008152506040518060400160405280600281526020017f544e00000000000000000000000000000000000000000000000000000000000081525081600290805190602001906200015e92919062000289565b5080600390805190602001906200017792919062000289565b5062000188620001b660201b60201c565b6000819055505050620001b0620001a4620001bb60201b60201c565b620001c360201b60201c565b6200039d565b600090565b600033905090565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b828054620002979062000368565b90600052602060002090601f016020900481019282620002bb576000855562000307565b82601f10620002d657805160ff191683800117855562000307565b8280016001018555821562000307579182015b8281111562000306578251825591602001919060010190620002e9565b5b5090506200031691906200031a565b5090565b5b80821115620003355760008160009055506001016200031b565b5090565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b600060028204905060018216806200038157607f821691505b60208210810362000397576200039662000339565b5b50919050565b61470180620003ad6000396000f3fe6080604052600436106102885760003560e01c8063783786a91161015a578063a475b5dd116100c1578063dc33e6811161007a578063dc33e68114610943578063e268e4d314610980578063e985e9c5146109a9578063f2fde38b146109e6578063f55e7fc714610a0f578063f968adbe14610a3857610288565b8063a475b5dd14610847578063b88d4fde1461085e578063bbaac02f14610887578063c87b56dd146108b0578063cc2f10d4146108ed578063d5abeb011461091857610288565b806395d89b411161011357806395d89b41146107495780639a7ddf50146107745780639b001f451461079d578063a035b1fe146107c8578063a1eaa87a146107f3578063a22cb4651461081e57610288565b8063783786a91461065a5780637bcf36ae1461067657806387b9d25c1461069f5780638cc54e7f146106ca5780638da5cb5b146106f557806391b7f5ed1461072057610288565b8063453c2310116101fe5780636352211e116101b75780636352211e1461054c5780637040d73f1461058957806370a08231146105b2578063715018a6146105ef5780637220ccc41461060657806377aeead91461063157610288565b8063453c23101461045d578063485a68a31461048857806351830227146104b357806355f804b3146104de5780635841a030146105075780635cfe6b841461053057610288565b806323b872dd1161025057806323b872dd146103865780632db11544146103af5780633031e7c7146103cb57806331461a41146103f45780633ccfd60b1461041d57806342842e0e1461043457610288565b806301ffc9a71461028d57806306fdde03146102ca578063081812fc146102f5578063095ea7b31461033257806318160ddd1461035b575b600080fd5b34801561029957600080fd5b506102b460048036038101906102af9190613414565b610a63565b6040516102c1919061345c565b60405180910390f35b3480156102d657600080fd5b506102df610af5565b6040516102ec9190613510565b60405180910390f35b34801561030157600080fd5b5061031c60048036038101906103179190613568565b610b87565b60405161032991906135d6565b60405180910390f35b34801561033e57600080fd5b506103596004803603810190610354919061361d565b610c03565b005b34801561036757600080fd5b50610370610da9565b60405161037d919061366c565b60405180910390f35b34801561039257600080fd5b506103ad60048036038101906103a89190613687565b610dc0565b005b6103c960048036038101906103c49190613568565b610dd0565b005b3480156103d757600080fd5b506103f260048036038101906103ed9190613706565b610fb3565b005b34801561040057600080fd5b5061041b60048036038101906104169190613706565b611066565b005b34801561042957600080fd5b506104326110ff565b005b34801561044057600080fd5b5061045b60048036038101906104569190613687565b61122a565b005b34801561046957600080fd5b5061047261124a565b60405161047f919061366c565b60405180910390f35b34801561049457600080fd5b5061049d611250565b6040516104aa919061366c565b60405180910390f35b3480156104bf57600080fd5b506104c8611256565b6040516104d5919061345c565b60405180910390f35b3480156104ea57600080fd5b5061050560048036038101906105009190613868565b611269565b005b34801561051357600080fd5b5061052e60048036038101906105299190613568565b6112ff565b005b61054a60048036038101906105459190613911565b611385565b005b34801561055857600080fd5b50610573600480360381019061056e9190613568565b611592565b60405161058091906135d6565b60405180910390f35b34801561059557600080fd5b506105b060048036038101906105ab9190613568565b6115a4565b005b3480156105be57600080fd5b506105d960048036038101906105d4919061395e565b61162a565b6040516105e6919061366c565b60405180910390f35b3480156105fb57600080fd5b506106046116e2565b005b34801561061257600080fd5b5061061b61176a565b604051610628919061345c565b60405180910390f35b34801561063d57600080fd5b50610658600480360381019061065391906139c1565b61177d565b005b610674600480360381019061066f9190613a01565b61180b565b005b34801561068257600080fd5b5061069d60048036038101906106989190613a61565b611b0f565b005b3480156106ab57600080fd5b506106b4611c39565b6040516106c1919061345c565b60405180910390f35b3480156106d657600080fd5b506106df611c4c565b6040516106ec9190613510565b60405180910390f35b34801561070157600080fd5b5061070a611cda565b60405161071791906135d6565b60405180910390f35b34801561072c57600080fd5b5061074760048036038101906107429190613568565b611d04565b005b34801561075557600080fd5b5061075e611d8a565b60405161076b9190613510565b60405180910390f35b34801561078057600080fd5b5061079b60048036038101906107969190613706565b611e1c565b005b3480156107a957600080fd5b506107b2611eb5565b6040516107bf919061345c565b60405180910390f35b3480156107d457600080fd5b506107dd611ec8565b6040516107ea919061366c565b60405180910390f35b3480156107ff57600080fd5b50610808611ece565b604051610815919061366c565b60405180910390f35b34801561082a57600080fd5b5061084560048036038101906108409190613aa1565b611ed4565b005b34801561085357600080fd5b5061085c61204b565b005b34801561086a57600080fd5b5061088560048036038101906108809190613b82565b6120f3565b005b34801561089357600080fd5b506108ae60048036038101906108a99190613868565b612166565b005b3480156108bc57600080fd5b506108d760048036038101906108d29190613568565b6121fc565b6040516108e49190613510565b60405180910390f35b3480156108f957600080fd5b50610902612351565b60405161090f919061366c565b60405180910390f35b34801561092457600080fd5b5061092d612357565b60405161093a919061366c565b60405180910390f35b34801561094f57600080fd5b5061096a6004803603810190610965919061395e565b61235d565b604051610977919061366c565b60405180910390f35b34801561098c57600080fd5b506109a760048036038101906109a29190613568565b61236f565b005b3480156109b557600080fd5b506109d060048036038101906109cb9190613c05565b6123f5565b6040516109dd919061345c565b60405180910390f35b3480156109f257600080fd5b50610a0d6004803603810190610a08919061395e565b612489565b005b348015610a1b57600080fd5b50610a366004803603810190610a319190613706565b612580565b005b348015610a4457600080fd5b50610a4d612619565b604051610a5a919061366c565b60405180910390f35b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480610abe57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80610aee5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b606060028054610b0490613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054610b3090613c74565b8015610b7d5780601f10610b5257610100808354040283529160200191610b7d565b820191906000526020600020905b815481529060010190602001808311610b6057829003601f168201915b5050505050905090565b6000610b928261261f565b610bc8576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b6000610c0e8261267e565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603610c75576040517f943f7b8c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff16610c9461274a565b73ffffffffffffffffffffffffffffffffffffffff1614610cf757610cc081610cbb61274a565b6123f5565b610cf6576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826006600084815260200190815260200160002060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6000610db3612752565b6001546000540303905090565b610dcb838383612757565b505050565b600a5481610dde9190613cd4565b341015610e20576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e1790613d7a565b60405180910390fd5b600c5481610e2d3361235d565b610e379190613d9a565b1115610e78576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e6f90613e3c565b60405180910390fd5b60125481610e84610da9565b610e8e9190613d9a565b1115610ecf576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ec690613ea8565b60405180910390fd5b60008111610f12576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f0990613f14565b60405180910390fd5b601360029054906101000a900460ff16610f61576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f5890613f80565b60405180910390fd5b600b54811115610fa6576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f9d90613fec565b60405180910390fd5b610fb03382612afe565b50565b610fbb612b1c565b73ffffffffffffffffffffffffffffffffffffffff16610fd9611cda565b73ffffffffffffffffffffffffffffffffffffffff161461102f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161102690614058565b60405180910390fd5b80601360006101000a81548160ff02191690831515021790555080601360016101000a81548160ff02191690831515021790555050565b61106e612b1c565b73ffffffffffffffffffffffffffffffffffffffff1661108c611cda565b73ffffffffffffffffffffffffffffffffffffffff16146110e2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016110d990614058565b60405180910390fd5b80601360016101000a81548160ff02191690831515021790555050565b611107612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611125611cda565b73ffffffffffffffffffffffffffffffffffffffff161461117b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161117290614058565b60405180910390fd5b60003373ffffffffffffffffffffffffffffffffffffffff16476040516111a1906140a9565b60006040518083038185875af1925050503d80600081146111de576040519150601f19603f3d011682016040523d82523d6000602084013e6111e3565b606091505b5050905080611227576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161121e9061410a565b60405180910390fd5b50565b611245838383604051806020016040528060008152506120f3565b505050565b600c5481565b60105481565b600e60009054906101000a900460ff1681565b611271612b1c565b73ffffffffffffffffffffffffffffffffffffffff1661128f611cda565b73ffffffffffffffffffffffffffffffffffffffff16146112e5576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016112dc90614058565b60405180910390fd5b80600990805190602001906112fb929190613305565b5050565b611307612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611325611cda565b73ffffffffffffffffffffffffffffffffffffffff161461137b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161137290614058565b60405180910390fd5b8060118190555050565b6000336040516020016113989190614172565b6040516020818303038152906040528051906020012090506113fe838380806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f8201169050808301925050505050505060155483612b24565b61143d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611434906141d9565b60405180910390fd5b600a54341015611482576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161147990613d7a565b60405180910390fd5b600f5460016114903361235d565b61149a9190613d9a565b11156114db576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016114d290613e3c565b60405180910390fd5b60125460016114e8610da9565b6114f29190613d9a565b1115611533576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161152a90613ea8565b60405180910390fd5b601360019054906101000a900460ff16611582576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161157990613f80565b60405180910390fd5b61158d336001612afe565b505050565b600061159d8261267e565b9050919050565b6115ac612b1c565b73ffffffffffffffffffffffffffffffffffffffff166115ca611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611620576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161161790614058565b60405180910390fd5b8060108190555050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611691576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b6116ea612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611708611cda565b73ffffffffffffffffffffffffffffffffffffffff161461175e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161175590614058565b60405180910390fd5b6117686000612b3b565b565b601360009054906101000a900460ff1681565b611785612b1c565b73ffffffffffffffffffffffffffffffffffffffff166117a3611cda565b73ffffffffffffffffffffffffffffffffffffffff16146117f9576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016117f090614058565b60405180910390fd5b81601481905550806015819055505050565b6000600a5490506000336040516020016118259190614172565b604051602081830303815290604052805190602001209050601054601660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205410801561188d5750600183145b1561189757600091505b6118e5858580806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f8201169050808301925050505050505060145483612b24565b611924576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161191b906141d9565b60405180910390fd5b81341015611967576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161195e90613d7a565b60405180910390fd5b601154836119743361235d565b61197e9190613d9a565b11156119bf576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016119b690613e3c565b60405180910390fd5b601254836119cb610da9565b6119d59190613d9a565b1115611a16576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a0d90613ea8565b60405180910390fd5b60008311611a59576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a5090613f14565b60405180910390fd5b601360009054906101000a900460ff16611aa8576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a9f90613f80565b60405180910390fd5b82601660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254611af79190613d9a565b92505081905550611b083384612afe565b5050505050565b611b17612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611b35611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611b8b576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611b8290614058565b60405180910390fd5b60008211611bce576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611bc590614245565b60405180910390fd5b6000611bd8610da9565b90506012548382611be99190613d9a565b1115611c2a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611c21906142b1565b60405180910390fd5b611c348284612afe565b505050565b601360019054906101000a900460ff1681565b600d8054611c5990613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054611c8590613c74565b8015611cd25780601f10611ca757610100808354040283529160200191611cd2565b820191906000526020600020905b815481529060010190602001808311611cb557829003601f168201915b505050505081565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611d0c612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611d2a611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611d80576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d7790614058565b60405180910390fd5b80600a8190555050565b606060038054611d9990613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054611dc590613c74565b8015611e125780601f10611de757610100808354040283529160200191611e12565b820191906000526020600020905b815481529060010190602001808311611df557829003601f168201915b5050505050905090565b611e24612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611e42611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611e98576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611e8f90614058565b60405180910390fd5b80601360006101000a81548160ff02191690831515021790555050565b601360029054906101000a900460ff1681565b600a5481565b600f5481565b611edc61274a565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611f40576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060076000611f4d61274a565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611ffa61274a565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161203f919061345c565b60405180910390a35050565b612053612b1c565b73ffffffffffffffffffffffffffffffffffffffff16612071611cda565b73ffffffffffffffffffffffffffffffffffffffff16146120c7576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016120be90614058565b60405180910390fd5b600e60009054906101000a900460ff1615600e60006101000a81548160ff021916908315150217905550565b6120fe848484612757565b60008373ffffffffffffffffffffffffffffffffffffffff163b146121605761212984848484612c01565b61215f576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b61216e612b1c565b73ffffffffffffffffffffffffffffffffffffffff1661218c611cda565b73ffffffffffffffffffffffffffffffffffffffff16146121e2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016121d990614058565b60405180910390fd5b80600d90805190602001906121f8929190613305565b5050565b60606122078261261f565b612246576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161223d90614343565b60405180910390fd5b60001515600e60009054906101000a900460ff161515036122f357600d805461226e90613c74565b80601f016020809104026020016040519081016040528092919081815260200182805461229a90613c74565b80156122e75780601f106122bc576101008083540402835291602001916122e7565b820191906000526020600020905b8154815290600101906020018083116122ca57829003601f168201915b5050505050905061234c565b60006122fd612d51565b9050600081511161231d5760405180602001604052806000815250612348565b8061232784612de3565b6040516020016123389291906143eb565b6040516020818303038152906040525b9150505b919050565b60115481565b60125481565b600061236882612f43565b9050919050565b612377612b1c565b73ffffffffffffffffffffffffffffffffffffffff16612395611cda565b73ffffffffffffffffffffffffffffffffffffffff16146123eb576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016123e290614058565b60405180910390fd5b80600c8190555050565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b612491612b1c565b73ffffffffffffffffffffffffffffffffffffffff166124af611cda565b73ffffffffffffffffffffffffffffffffffffffff1614612505576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016124fc90614058565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603612574576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161256b9061448c565b60405180910390fd5b61257d81612b3b565b50565b612588612b1c565b73ffffffffffffffffffffffffffffffffffffffff166125a6611cda565b73ffffffffffffffffffffffffffffffffffffffff16146125fc576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016125f390614058565b60405180910390fd5b80601360026101000a81548160ff02191690831515021790555050565b600b5481565b60008161262a612752565b11158015612639575060005482105b8015612677575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b6000808290508061268d612752565b11612713576000548110156127125760006004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821603612710575b600081036127065760046000836001900393508381526020019081526020016000205490506126dc565b8092505050612745565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b600033905090565b600090565b60006127628261267e565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146127c9576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008473ffffffffffffffffffffffffffffffffffffffff166127ea61274a565b73ffffffffffffffffffffffffffffffffffffffff16148061281957506128188561281361274a565b6123f5565b5b8061285e575061282761274a565b73ffffffffffffffffffffffffffffffffffffffff1661284684610b87565b73ffffffffffffffffffffffffffffffffffffffff16145b905080612897576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff16036128fd576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61290a8585856001612f9a565b6006600084815260200190815260200160002060006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008154600101919050819055507c020000000000000000000000000000000000000000000000000000000060a042901b612a0786612fa0565b1717600460008581526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000831603612a8f5760006001840190506000600460008381526020019081526020016000205403612a8d576000548114612a8c578260046000838152602001908152602001600020819055505b5b505b828473ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612af78585856001612faa565b5050505050565b612b18828260405180602001604052806000815250612fb0565b5050565b600033905090565b600082612b318584613263565b1490509392505050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02612c2761274a565b8786866040518563ffffffff1660e01b8152600401612c499493929190614501565b6020604051808303816000875af1925050508015612c8557506040513d601f19601f82011682018060405250810190612c829190614562565b60015b612cfe573d8060008114612cb5576040519150601f19603f3d011682016040523d82523d6000602084013e612cba565b606091505b506000815103612cf6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b606060098054612d6090613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054612d8c90613c74565b8015612dd95780601f10612dae57610100808354040283529160200191612dd9565b820191906000526020600020905b815481529060010190602001808311612dbc57829003601f168201915b5050505050905090565b606060008203612e2a576040518060400160405280600181526020017f30000000000000000000000000000000000000000000000000000000000000008152509050612f3e565b600082905060005b60008214612e5c578080612e459061458f565b915050600a82612e559190614606565b9150612e32565b60008167ffffffffffffffff811115612e7857612e7761373d565b5b6040519080825280601f01601f191660200182016040528015612eaa5781602001600182028036833780820191505090505b5090505b60008514612f3757600182612ec39190614637565b9150600a85612ed2919061466b565b6030612ede9190613d9a565b60f81b818381518110612ef457612ef361469c565b5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350600a85612f309190614606565b9450612eae565b8093505050505b919050565b600067ffffffffffffffff6040600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054901c169050919050565b50505050565b6000819050919050565b50505050565b600080549050600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff160361301c576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008303613056576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6130636000858386612f9a565b600160406001901b178302600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254019250508190555060e16130c8600185146132b9565b901b60a042901b6130d886612fa0565b1717600460008381526020019081526020016000208190555060008190506000848201905060008673ffffffffffffffffffffffffffffffffffffffff163b146131dc575b818673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a461318c6000878480600101955087612c01565b6131c2576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b80821061311d5782600054146131d757600080fd5b613247565b5b818060010192508673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a48082106131dd575b81600081905550505061325d6000858386612faa565b50505050565b60008082905060005b84518110156132ae576132998286838151811061328c5761328b61469c565b5b60200260200101516132c3565b915080806132a69061458f565b91505061326c565b508091505092915050565b6000819050919050565b60008183106132db576132d682846132ee565b6132e6565b6132e583836132ee565b5b905092915050565b600082600052816020526040600020905092915050565b82805461331190613c74565b90600052602060002090601f016020900481019282613333576000855561337a565b82601f1061334c57805160ff191683800117855561337a565b8280016001018555821561337a579182015b8281111561337957825182559160200191906001019061335e565b5b509050613387919061338b565b5090565b5b808211156133a457600081600090555060010161338c565b5090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b6133f1816133bc565b81146133fc57600080fd5b50565b60008135905061340e816133e8565b92915050565b60006020828403121561342a576134296133b2565b5b6000613438848285016133ff565b91505092915050565b60008115159050919050565b61345681613441565b82525050565b6000602082019050613471600083018461344d565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b838110156134b1578082015181840152602081019050613496565b838111156134c0576000848401525b50505050565b6000601f19601f8301169050919050565b60006134e282613477565b6134ec8185613482565b93506134fc818560208601613493565b613505816134c6565b840191505092915050565b6000602082019050818103600083015261352a81846134d7565b905092915050565b6000819050919050565b61354581613532565b811461355057600080fd5b50565b6000813590506135628161353c565b92915050565b60006020828403121561357e5761357d6133b2565b5b600061358c84828501613553565b91505092915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b60006135c082613595565b9050919050565b6135d0816135b5565b82525050565b60006020820190506135eb60008301846135c7565b92915050565b6135fa816135b5565b811461360557600080fd5b50565b600081359050613617816135f1565b92915050565b60008060408385031215613634576136336133b2565b5b600061364285828601613608565b925050602061365385828601613553565b9150509250929050565b61366681613532565b82525050565b6000602082019050613681600083018461365d565b92915050565b6000806000606084860312156136a05761369f6133b2565b5b60006136ae86828701613608565b93505060206136bf86828701613608565b92505060406136d086828701613553565b9150509250925092565b6136e381613441565b81146136ee57600080fd5b50565b600081359050613700816136da565b92915050565b60006020828403121561371c5761371b6133b2565b5b600061372a848285016136f1565b91505092915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b613775826134c6565b810181811067ffffffffffffffff821117156137945761379361373d565b5b80604052505050565b60006137a76133a8565b90506137b3828261376c565b919050565b600067ffffffffffffffff8211156137d3576137d261373d565b5b6137dc826134c6565b9050602081019050919050565b82818337600083830152505050565b600061380b613806846137b8565b61379d565b90508281526020810184848401111561382757613826613738565b5b6138328482856137e9565b509392505050565b600082601f83011261384f5761384e613733565b5b813561385f8482602086016137f8565b91505092915050565b60006020828403121561387e5761387d6133b2565b5b600082013567ffffffffffffffff81111561389c5761389b6133b7565b5b6138a88482850161383a565b91505092915050565b600080fd5b600080fd5b60008083601f8401126138d1576138d0613733565b5b8235905067ffffffffffffffff8111156138ee576138ed6138b1565b5b60208301915083602082028301111561390a576139096138b6565b5b9250929050565b60008060208385031215613928576139276133b2565b5b600083013567ffffffffffffffff811115613946576139456133b7565b5b613952858286016138bb565b92509250509250929050565b600060208284031215613974576139736133b2565b5b600061398284828501613608565b91505092915050565b6000819050919050565b61399e8161398b565b81146139a957600080fd5b50565b6000813590506139bb81613995565b92915050565b600080604083850312156139d8576139d76133b2565b5b60006139e6858286016139ac565b92505060206139f7858286016139ac565b9150509250929050565b600080600060408486031215613a1a57613a196133b2565b5b600084013567ffffffffffffffff811115613a3857613a376133b7565b5b613a44868287016138bb565b93509350506020613a5786828701613553565b9150509250925092565b60008060408385031215613a7857613a776133b2565b5b6000613a8685828601613553565b9250506020613a9785828601613608565b9150509250929050565b60008060408385031215613ab857613ab76133b2565b5b6000613ac685828601613608565b9250506020613ad7858286016136f1565b9150509250929050565b600067ffffffffffffffff821115613afc57613afb61373d565b5b613b05826134c6565b9050602081019050919050565b6000613b25613b2084613ae1565b61379d565b905082815260208101848484011115613b4157613b40613738565b5b613b4c8482856137e9565b509392505050565b600082601f830112613b6957613b68613733565b5b8135613b79848260208601613b12565b91505092915050565b60008060008060808587031215613b9c57613b9b6133b2565b5b6000613baa87828801613608565b9450506020613bbb87828801613608565b9350506040613bcc87828801613553565b925050606085013567ffffffffffffffff811115613bed57613bec6133b7565b5b613bf987828801613b54565b91505092959194509250565b60008060408385031215613c1c57613c1b6133b2565b5b6000613c2a85828601613608565b9250506020613c3b85828601613608565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680613c8c57607f821691505b602082108103613c9f57613c9e613c45565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000613cdf82613532565b9150613cea83613532565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0483118215151615613d2357613d22613ca5565b5b828202905092915050565b7f506c656173652073656e642074686520657861637420616d6f756e742e000000600082015250565b6000613d64601d83613482565b9150613d6f82613d2e565b602082019050919050565b60006020820190508181036000830152613d9381613d57565b9050919050565b6000613da582613532565b9150613db083613532565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115613de557613de4613ca5565b5b828201905092915050565b7f596f752063616e74206d696e7420616e796d6f72650000000000000000000000600082015250565b6000613e26601583613482565b9150613e3182613df0565b602082019050919050565b60006020820190508181036000830152613e5581613e19565b9050919050565b7f4e6f206d6f726500000000000000000000000000000000000000000000000000600082015250565b6000613e92600783613482565b9150613e9d82613e5c565b602082019050919050565b60006020820190508181036000830152613ec181613e85565b9050919050565b7f506c6561736520656e7465722061206e756d6265720000000000000000000000600082015250565b6000613efe601583613482565b9150613f0982613ec8565b602082019050919050565b60006020820190508181036000830152613f2d81613ef1565b9050919050565b7f4d696e74696e67206973206e6f74206c69766520796574000000000000000000600082015250565b6000613f6a601783613482565b9150613f7582613f34565b602082019050919050565b60006020820190508181036000830152613f9981613f5d565b9050919050565b7f4d61782070657220747820657863656564656400000000000000000000000000600082015250565b6000613fd6601383613482565b9150613fe182613fa0565b602082019050919050565b6000602082019050818103600083015261400581613fc9565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b6000614042602083613482565b915061404d8261400c565b602082019050919050565b6000602082019050818103600083015261407181614035565b9050919050565b600081905092915050565b50565b6000614093600083614078565b915061409e82614083565b600082019050919050565b60006140b482614086565b9150819050919050565b7f5472616e73666572206661696c65642e00000000000000000000000000000000600082015250565b60006140f4601083613482565b91506140ff826140be565b602082019050919050565b60006020820190508181036000830152614123816140e7565b9050919050565b60008160601b9050919050565b60006141428261412a565b9050919050565b600061415482614137565b9050919050565b61416c614167826135b5565b614149565b82525050565b600061417e828461415b565b60148201915081905092915050565b7f496e636f72726563742057686974656c6973742050726f6f6600000000000000600082015250565b60006141c3601983613482565b91506141ce8261418d565b602082019050919050565b600060208201905081810360008301526141f2816141b6565b9050919050565b7f6e65656420746f206d696e74206174206c656173742031204e46540000000000600082015250565b600061422f601b83613482565b915061423a826141f9565b602082019050919050565b6000602082019050818103600083015261425e81614222565b9050919050565b7f6d6178204e4654206c696d697420657863656564656400000000000000000000600082015250565b600061429b601683613482565b91506142a682614265565b602082019050919050565b600060208201905081810360008301526142ca8161428e565b9050919050565b7f455243373231414d657461646174613a2055524920717565727920666f72206e60008201527f6f6e6578697374656e7420746f6b656e00000000000000000000000000000000602082015250565b600061432d603083613482565b9150614338826142d1565b604082019050919050565b6000602082019050818103600083015261435c81614320565b9050919050565b600081905092915050565b600061437982613477565b6143838185614363565b9350614393818560208601613493565b80840191505092915050565b7f2e6a736f6e000000000000000000000000000000000000000000000000000000600082015250565b60006143d5600583614363565b91506143e08261439f565b600582019050919050565b60006143f7828561436e565b9150614403828461436e565b915061440e826143c8565b91508190509392505050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b6000614476602683613482565b91506144818261441a565b604082019050919050565b600060208201905081810360008301526144a581614469565b9050919050565b600081519050919050565b600082825260208201905092915050565b60006144d3826144ac565b6144dd81856144b7565b93506144ed818560208601613493565b6144f6816134c6565b840191505092915050565b600060808201905061451660008301876135c7565b61452360208301866135c7565b614530604083018561365d565b818103606083015261454281846144c8565b905095945050505050565b60008151905061455c816133e8565b92915050565b600060208284031215614578576145776133b2565b5b60006145868482850161454d565b91505092915050565b600061459a82613532565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82036145cc576145cb613ca5565b5b600182019050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b600061461182613532565b915061461c83613532565b92508261462c5761462b6145d7565b5b828204905092915050565b600061464282613532565b915061464d83613532565b9250828210156146605761465f613ca5565b5b828203905092915050565b600061467682613532565b915061468183613532565b925082614691576146906145d7565b5b828206905092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fdfea2646970667358221220124521a7a189103aef94266d980080a72d870838b4e6e13d49ad84dec7f3c1b064736f6c634300080d0033697066733a2f2f6261667962656968356a6364766d73697574347636776d63746a703563326373683370706c6f6f68677961376b63617268353633697872376c63692f68696464656e2e6a736f6e
Deployed Bytecode
0x6080604052600436106102885760003560e01c8063783786a91161015a578063a475b5dd116100c1578063dc33e6811161007a578063dc33e68114610943578063e268e4d314610980578063e985e9c5146109a9578063f2fde38b146109e6578063f55e7fc714610a0f578063f968adbe14610a3857610288565b8063a475b5dd14610847578063b88d4fde1461085e578063bbaac02f14610887578063c87b56dd146108b0578063cc2f10d4146108ed578063d5abeb011461091857610288565b806395d89b411161011357806395d89b41146107495780639a7ddf50146107745780639b001f451461079d578063a035b1fe146107c8578063a1eaa87a146107f3578063a22cb4651461081e57610288565b8063783786a91461065a5780637bcf36ae1461067657806387b9d25c1461069f5780638cc54e7f146106ca5780638da5cb5b146106f557806391b7f5ed1461072057610288565b8063453c2310116101fe5780636352211e116101b75780636352211e1461054c5780637040d73f1461058957806370a08231146105b2578063715018a6146105ef5780637220ccc41461060657806377aeead91461063157610288565b8063453c23101461045d578063485a68a31461048857806351830227146104b357806355f804b3146104de5780635841a030146105075780635cfe6b841461053057610288565b806323b872dd1161025057806323b872dd146103865780632db11544146103af5780633031e7c7146103cb57806331461a41146103f45780633ccfd60b1461041d57806342842e0e1461043457610288565b806301ffc9a71461028d57806306fdde03146102ca578063081812fc146102f5578063095ea7b31461033257806318160ddd1461035b575b600080fd5b34801561029957600080fd5b506102b460048036038101906102af9190613414565b610a63565b6040516102c1919061345c565b60405180910390f35b3480156102d657600080fd5b506102df610af5565b6040516102ec9190613510565b60405180910390f35b34801561030157600080fd5b5061031c60048036038101906103179190613568565b610b87565b60405161032991906135d6565b60405180910390f35b34801561033e57600080fd5b506103596004803603810190610354919061361d565b610c03565b005b34801561036757600080fd5b50610370610da9565b60405161037d919061366c565b60405180910390f35b34801561039257600080fd5b506103ad60048036038101906103a89190613687565b610dc0565b005b6103c960048036038101906103c49190613568565b610dd0565b005b3480156103d757600080fd5b506103f260048036038101906103ed9190613706565b610fb3565b005b34801561040057600080fd5b5061041b60048036038101906104169190613706565b611066565b005b34801561042957600080fd5b506104326110ff565b005b34801561044057600080fd5b5061045b60048036038101906104569190613687565b61122a565b005b34801561046957600080fd5b5061047261124a565b60405161047f919061366c565b60405180910390f35b34801561049457600080fd5b5061049d611250565b6040516104aa919061366c565b60405180910390f35b3480156104bf57600080fd5b506104c8611256565b6040516104d5919061345c565b60405180910390f35b3480156104ea57600080fd5b5061050560048036038101906105009190613868565b611269565b005b34801561051357600080fd5b5061052e60048036038101906105299190613568565b6112ff565b005b61054a60048036038101906105459190613911565b611385565b005b34801561055857600080fd5b50610573600480360381019061056e9190613568565b611592565b60405161058091906135d6565b60405180910390f35b34801561059557600080fd5b506105b060048036038101906105ab9190613568565b6115a4565b005b3480156105be57600080fd5b506105d960048036038101906105d4919061395e565b61162a565b6040516105e6919061366c565b60405180910390f35b3480156105fb57600080fd5b506106046116e2565b005b34801561061257600080fd5b5061061b61176a565b604051610628919061345c565b60405180910390f35b34801561063d57600080fd5b50610658600480360381019061065391906139c1565b61177d565b005b610674600480360381019061066f9190613a01565b61180b565b005b34801561068257600080fd5b5061069d60048036038101906106989190613a61565b611b0f565b005b3480156106ab57600080fd5b506106b4611c39565b6040516106c1919061345c565b60405180910390f35b3480156106d657600080fd5b506106df611c4c565b6040516106ec9190613510565b60405180910390f35b34801561070157600080fd5b5061070a611cda565b60405161071791906135d6565b60405180910390f35b34801561072c57600080fd5b5061074760048036038101906107429190613568565b611d04565b005b34801561075557600080fd5b5061075e611d8a565b60405161076b9190613510565b60405180910390f35b34801561078057600080fd5b5061079b60048036038101906107969190613706565b611e1c565b005b3480156107a957600080fd5b506107b2611eb5565b6040516107bf919061345c565b60405180910390f35b3480156107d457600080fd5b506107dd611ec8565b6040516107ea919061366c565b60405180910390f35b3480156107ff57600080fd5b50610808611ece565b604051610815919061366c565b60405180910390f35b34801561082a57600080fd5b5061084560048036038101906108409190613aa1565b611ed4565b005b34801561085357600080fd5b5061085c61204b565b005b34801561086a57600080fd5b5061088560048036038101906108809190613b82565b6120f3565b005b34801561089357600080fd5b506108ae60048036038101906108a99190613868565b612166565b005b3480156108bc57600080fd5b506108d760048036038101906108d29190613568565b6121fc565b6040516108e49190613510565b60405180910390f35b3480156108f957600080fd5b50610902612351565b60405161090f919061366c565b60405180910390f35b34801561092457600080fd5b5061092d612357565b60405161093a919061366c565b60405180910390f35b34801561094f57600080fd5b5061096a6004803603810190610965919061395e565b61235d565b604051610977919061366c565b60405180910390f35b34801561098c57600080fd5b506109a760048036038101906109a29190613568565b61236f565b005b3480156109b557600080fd5b506109d060048036038101906109cb9190613c05565b6123f5565b6040516109dd919061345c565b60405180910390f35b3480156109f257600080fd5b50610a0d6004803603810190610a08919061395e565b612489565b005b348015610a1b57600080fd5b50610a366004803603810190610a319190613706565b612580565b005b348015610a4457600080fd5b50610a4d612619565b604051610a5a919061366c565b60405180910390f35b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480610abe57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80610aee5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b606060028054610b0490613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054610b3090613c74565b8015610b7d5780601f10610b5257610100808354040283529160200191610b7d565b820191906000526020600020905b815481529060010190602001808311610b6057829003601f168201915b5050505050905090565b6000610b928261261f565b610bc8576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b6000610c0e8261267e565b90508073ffffffffffffffffffffffffffffffffffffffff168373ffffffffffffffffffffffffffffffffffffffff1603610c75576040517f943f7b8c00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8073ffffffffffffffffffffffffffffffffffffffff16610c9461274a565b73ffffffffffffffffffffffffffffffffffffffff1614610cf757610cc081610cbb61274a565b6123f5565b610cf6576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b826006600084815260200190815260200160002060006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550818373ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a4505050565b6000610db3612752565b6001546000540303905090565b610dcb838383612757565b505050565b600a5481610dde9190613cd4565b341015610e20576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e1790613d7a565b60405180910390fd5b600c5481610e2d3361235d565b610e379190613d9a565b1115610e78576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e6f90613e3c565b60405180910390fd5b60125481610e84610da9565b610e8e9190613d9a565b1115610ecf576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ec690613ea8565b60405180910390fd5b60008111610f12576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f0990613f14565b60405180910390fd5b601360029054906101000a900460ff16610f61576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f5890613f80565b60405180910390fd5b600b54811115610fa6576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610f9d90613fec565b60405180910390fd5b610fb03382612afe565b50565b610fbb612b1c565b73ffffffffffffffffffffffffffffffffffffffff16610fd9611cda565b73ffffffffffffffffffffffffffffffffffffffff161461102f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161102690614058565b60405180910390fd5b80601360006101000a81548160ff02191690831515021790555080601360016101000a81548160ff02191690831515021790555050565b61106e612b1c565b73ffffffffffffffffffffffffffffffffffffffff1661108c611cda565b73ffffffffffffffffffffffffffffffffffffffff16146110e2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016110d990614058565b60405180910390fd5b80601360016101000a81548160ff02191690831515021790555050565b611107612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611125611cda565b73ffffffffffffffffffffffffffffffffffffffff161461117b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161117290614058565b60405180910390fd5b60003373ffffffffffffffffffffffffffffffffffffffff16476040516111a1906140a9565b60006040518083038185875af1925050503d80600081146111de576040519150601f19603f3d011682016040523d82523d6000602084013e6111e3565b606091505b5050905080611227576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161121e9061410a565b60405180910390fd5b50565b611245838383604051806020016040528060008152506120f3565b505050565b600c5481565b60105481565b600e60009054906101000a900460ff1681565b611271612b1c565b73ffffffffffffffffffffffffffffffffffffffff1661128f611cda565b73ffffffffffffffffffffffffffffffffffffffff16146112e5576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016112dc90614058565b60405180910390fd5b80600990805190602001906112fb929190613305565b5050565b611307612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611325611cda565b73ffffffffffffffffffffffffffffffffffffffff161461137b576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161137290614058565b60405180910390fd5b8060118190555050565b6000336040516020016113989190614172565b6040516020818303038152906040528051906020012090506113fe838380806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f8201169050808301925050505050505060155483612b24565b61143d576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611434906141d9565b60405180910390fd5b600a54341015611482576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161147990613d7a565b60405180910390fd5b600f5460016114903361235d565b61149a9190613d9a565b11156114db576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016114d290613e3c565b60405180910390fd5b60125460016114e8610da9565b6114f29190613d9a565b1115611533576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161152a90613ea8565b60405180910390fd5b601360019054906101000a900460ff16611582576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161157990613f80565b60405180910390fd5b61158d336001612afe565b505050565b600061159d8261267e565b9050919050565b6115ac612b1c565b73ffffffffffffffffffffffffffffffffffffffff166115ca611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611620576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161161790614058565b60405180910390fd5b8060108190555050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611691576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b6116ea612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611708611cda565b73ffffffffffffffffffffffffffffffffffffffff161461175e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161175590614058565b60405180910390fd5b6117686000612b3b565b565b601360009054906101000a900460ff1681565b611785612b1c565b73ffffffffffffffffffffffffffffffffffffffff166117a3611cda565b73ffffffffffffffffffffffffffffffffffffffff16146117f9576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016117f090614058565b60405180910390fd5b81601481905550806015819055505050565b6000600a5490506000336040516020016118259190614172565b604051602081830303815290604052805190602001209050601054601660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205410801561188d5750600183145b1561189757600091505b6118e5858580806020026020016040519081016040528093929190818152602001838360200280828437600081840152601f19601f8201169050808301925050505050505060145483612b24565b611924576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161191b906141d9565b60405180910390fd5b81341015611967576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161195e90613d7a565b60405180910390fd5b601154836119743361235d565b61197e9190613d9a565b11156119bf576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016119b690613e3c565b60405180910390fd5b601254836119cb610da9565b6119d59190613d9a565b1115611a16576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a0d90613ea8565b60405180910390fd5b60008311611a59576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a5090613f14565b60405180910390fd5b601360009054906101000a900460ff16611aa8576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a9f90613f80565b60405180910390fd5b82601660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254611af79190613d9a565b92505081905550611b083384612afe565b5050505050565b611b17612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611b35611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611b8b576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611b8290614058565b60405180910390fd5b60008211611bce576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611bc590614245565b60405180910390fd5b6000611bd8610da9565b90506012548382611be99190613d9a565b1115611c2a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611c21906142b1565b60405180910390fd5b611c348284612afe565b505050565b601360019054906101000a900460ff1681565b600d8054611c5990613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054611c8590613c74565b8015611cd25780601f10611ca757610100808354040283529160200191611cd2565b820191906000526020600020905b815481529060010190602001808311611cb557829003601f168201915b505050505081565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b611d0c612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611d2a611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611d80576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611d7790614058565b60405180910390fd5b80600a8190555050565b606060038054611d9990613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054611dc590613c74565b8015611e125780601f10611de757610100808354040283529160200191611e12565b820191906000526020600020905b815481529060010190602001808311611df557829003601f168201915b5050505050905090565b611e24612b1c565b73ffffffffffffffffffffffffffffffffffffffff16611e42611cda565b73ffffffffffffffffffffffffffffffffffffffff1614611e98576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611e8f90614058565b60405180910390fd5b80601360006101000a81548160ff02191690831515021790555050565b601360029054906101000a900460ff1681565b600a5481565b600f5481565b611edc61274a565b73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611f40576040517fb06307db00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060076000611f4d61274a565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611ffa61274a565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c318360405161203f919061345c565b60405180910390a35050565b612053612b1c565b73ffffffffffffffffffffffffffffffffffffffff16612071611cda565b73ffffffffffffffffffffffffffffffffffffffff16146120c7576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016120be90614058565b60405180910390fd5b600e60009054906101000a900460ff1615600e60006101000a81548160ff021916908315150217905550565b6120fe848484612757565b60008373ffffffffffffffffffffffffffffffffffffffff163b146121605761212984848484612c01565b61215f576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b61216e612b1c565b73ffffffffffffffffffffffffffffffffffffffff1661218c611cda565b73ffffffffffffffffffffffffffffffffffffffff16146121e2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016121d990614058565b60405180910390fd5b80600d90805190602001906121f8929190613305565b5050565b60606122078261261f565b612246576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161223d90614343565b60405180910390fd5b60001515600e60009054906101000a900460ff161515036122f357600d805461226e90613c74565b80601f016020809104026020016040519081016040528092919081815260200182805461229a90613c74565b80156122e75780601f106122bc576101008083540402835291602001916122e7565b820191906000526020600020905b8154815290600101906020018083116122ca57829003601f168201915b5050505050905061234c565b60006122fd612d51565b9050600081511161231d5760405180602001604052806000815250612348565b8061232784612de3565b6040516020016123389291906143eb565b6040516020818303038152906040525b9150505b919050565b60115481565b60125481565b600061236882612f43565b9050919050565b612377612b1c565b73ffffffffffffffffffffffffffffffffffffffff16612395611cda565b73ffffffffffffffffffffffffffffffffffffffff16146123eb576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016123e290614058565b60405180910390fd5b80600c8190555050565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b612491612b1c565b73ffffffffffffffffffffffffffffffffffffffff166124af611cda565b73ffffffffffffffffffffffffffffffffffffffff1614612505576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016124fc90614058565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603612574576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161256b9061448c565b60405180910390fd5b61257d81612b3b565b50565b612588612b1c565b73ffffffffffffffffffffffffffffffffffffffff166125a6611cda565b73ffffffffffffffffffffffffffffffffffffffff16146125fc576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016125f390614058565b60405180910390fd5b80601360026101000a81548160ff02191690831515021790555050565b600b5481565b60008161262a612752565b11158015612639575060005482105b8015612677575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b6000808290508061268d612752565b11612713576000548110156127125760006004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821603612710575b600081036127065760046000836001900393508381526020019081526020016000205490506126dc565b8092505050612745565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b600033905090565b600090565b60006127628261267e565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146127c9576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008473ffffffffffffffffffffffffffffffffffffffff166127ea61274a565b73ffffffffffffffffffffffffffffffffffffffff16148061281957506128188561281361274a565b6123f5565b5b8061285e575061282761274a565b73ffffffffffffffffffffffffffffffffffffffff1661284684610b87565b73ffffffffffffffffffffffffffffffffffffffff16145b905080612897576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff16036128fd576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61290a8585856001612f9a565b6006600084815260200190815260200160002060006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008154600101919050819055507c020000000000000000000000000000000000000000000000000000000060a042901b612a0786612fa0565b1717600460008581526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000831603612a8f5760006001840190506000600460008381526020019081526020016000205403612a8d576000548114612a8c578260046000838152602001908152602001600020819055505b5b505b828473ffffffffffffffffffffffffffffffffffffffff168673ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612af78585856001612faa565b5050505050565b612b18828260405180602001604052806000815250612fb0565b5050565b600033905090565b600082612b318584613263565b1490509392505050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02612c2761274a565b8786866040518563ffffffff1660e01b8152600401612c499493929190614501565b6020604051808303816000875af1925050508015612c8557506040513d601f19601f82011682018060405250810190612c829190614562565b60015b612cfe573d8060008114612cb5576040519150601f19603f3d011682016040523d82523d6000602084013e612cba565b606091505b506000815103612cf6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b606060098054612d6090613c74565b80601f0160208091040260200160405190810160405280929190818152602001828054612d8c90613c74565b8015612dd95780601f10612dae57610100808354040283529160200191612dd9565b820191906000526020600020905b815481529060010190602001808311612dbc57829003601f168201915b5050505050905090565b606060008203612e2a576040518060400160405280600181526020017f30000000000000000000000000000000000000000000000000000000000000008152509050612f3e565b600082905060005b60008214612e5c578080612e459061458f565b915050600a82612e559190614606565b9150612e32565b60008167ffffffffffffffff811115612e7857612e7761373d565b5b6040519080825280601f01601f191660200182016040528015612eaa5781602001600182028036833780820191505090505b5090505b60008514612f3757600182612ec39190614637565b9150600a85612ed2919061466b565b6030612ede9190613d9a565b60f81b818381518110612ef457612ef361469c565b5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a905350600a85612f309190614606565b9450612eae565b8093505050505b919050565b600067ffffffffffffffff6040600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054901c169050919050565b50505050565b6000819050919050565b50505050565b600080549050600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff160361301c576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008303613056576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6130636000858386612f9a565b600160406001901b178302600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254019250508190555060e16130c8600185146132b9565b901b60a042901b6130d886612fa0565b1717600460008381526020019081526020016000208190555060008190506000848201905060008673ffffffffffffffffffffffffffffffffffffffff163b146131dc575b818673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a461318c6000878480600101955087612c01565b6131c2576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b80821061311d5782600054146131d757600080fd5b613247565b5b818060010192508673ffffffffffffffffffffffffffffffffffffffff16600073ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a48082106131dd575b81600081905550505061325d6000858386612faa565b50505050565b60008082905060005b84518110156132ae576132998286838151811061328c5761328b61469c565b5b60200260200101516132c3565b915080806132a69061458f565b91505061326c565b508091505092915050565b6000819050919050565b60008183106132db576132d682846132ee565b6132e6565b6132e583836132ee565b5b905092915050565b600082600052816020526040600020905092915050565b82805461331190613c74565b90600052602060002090601f016020900481019282613333576000855561337a565b82601f1061334c57805160ff191683800117855561337a565b8280016001018555821561337a579182015b8281111561337957825182559160200191906001019061335e565b5b509050613387919061338b565b5090565b5b808211156133a457600081600090555060010161338c565b5090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b6133f1816133bc565b81146133fc57600080fd5b50565b60008135905061340e816133e8565b92915050565b60006020828403121561342a576134296133b2565b5b6000613438848285016133ff565b91505092915050565b60008115159050919050565b61345681613441565b82525050565b6000602082019050613471600083018461344d565b92915050565b600081519050919050565b600082825260208201905092915050565b60005b838110156134b1578082015181840152602081019050613496565b838111156134c0576000848401525b50505050565b6000601f19601f8301169050919050565b60006134e282613477565b6134ec8185613482565b93506134fc818560208601613493565b613505816134c6565b840191505092915050565b6000602082019050818103600083015261352a81846134d7565b905092915050565b6000819050919050565b61354581613532565b811461355057600080fd5b50565b6000813590506135628161353c565b92915050565b60006020828403121561357e5761357d6133b2565b5b600061358c84828501613553565b91505092915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b60006135c082613595565b9050919050565b6135d0816135b5565b82525050565b60006020820190506135eb60008301846135c7565b92915050565b6135fa816135b5565b811461360557600080fd5b50565b600081359050613617816135f1565b92915050565b60008060408385031215613634576136336133b2565b5b600061364285828601613608565b925050602061365385828601613553565b9150509250929050565b61366681613532565b82525050565b6000602082019050613681600083018461365d565b92915050565b6000806000606084860312156136a05761369f6133b2565b5b60006136ae86828701613608565b93505060206136bf86828701613608565b92505060406136d086828701613553565b9150509250925092565b6136e381613441565b81146136ee57600080fd5b50565b600081359050613700816136da565b92915050565b60006020828403121561371c5761371b6133b2565b5b600061372a848285016136f1565b91505092915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b613775826134c6565b810181811067ffffffffffffffff821117156137945761379361373d565b5b80604052505050565b60006137a76133a8565b90506137b3828261376c565b919050565b600067ffffffffffffffff8211156137d3576137d261373d565b5b6137dc826134c6565b9050602081019050919050565b82818337600083830152505050565b600061380b613806846137b8565b61379d565b90508281526020810184848401111561382757613826613738565b5b6138328482856137e9565b509392505050565b600082601f83011261384f5761384e613733565b5b813561385f8482602086016137f8565b91505092915050565b60006020828403121561387e5761387d6133b2565b5b600082013567ffffffffffffffff81111561389c5761389b6133b7565b5b6138a88482850161383a565b91505092915050565b600080fd5b600080fd5b60008083601f8401126138d1576138d0613733565b5b8235905067ffffffffffffffff8111156138ee576138ed6138b1565b5b60208301915083602082028301111561390a576139096138b6565b5b9250929050565b60008060208385031215613928576139276133b2565b5b600083013567ffffffffffffffff811115613946576139456133b7565b5b613952858286016138bb565b92509250509250929050565b600060208284031215613974576139736133b2565b5b600061398284828501613608565b91505092915050565b6000819050919050565b61399e8161398b565b81146139a957600080fd5b50565b6000813590506139bb81613995565b92915050565b600080604083850312156139d8576139d76133b2565b5b60006139e6858286016139ac565b92505060206139f7858286016139ac565b9150509250929050565b600080600060408486031215613a1a57613a196133b2565b5b600084013567ffffffffffffffff811115613a3857613a376133b7565b5b613a44868287016138bb565b93509350506020613a5786828701613553565b9150509250925092565b60008060408385031215613a7857613a776133b2565b5b6000613a8685828601613553565b9250506020613a9785828601613608565b9150509250929050565b60008060408385031215613ab857613ab76133b2565b5b6000613ac685828601613608565b9250506020613ad7858286016136f1565b9150509250929050565b600067ffffffffffffffff821115613afc57613afb61373d565b5b613b05826134c6565b9050602081019050919050565b6000613b25613b2084613ae1565b61379d565b905082815260208101848484011115613b4157613b40613738565b5b613b4c8482856137e9565b509392505050565b600082601f830112613b6957613b68613733565b5b8135613b79848260208601613b12565b91505092915050565b60008060008060808587031215613b9c57613b9b6133b2565b5b6000613baa87828801613608565b9450506020613bbb87828801613608565b9350506040613bcc87828801613553565b925050606085013567ffffffffffffffff811115613bed57613bec6133b7565b5b613bf987828801613b54565b91505092959194509250565b60008060408385031215613c1c57613c1b6133b2565b5b6000613c2a85828601613608565b9250506020613c3b85828601613608565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680613c8c57607f821691505b602082108103613c9f57613c9e613c45565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000613cdf82613532565b9150613cea83613532565b9250817fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0483118215151615613d2357613d22613ca5565b5b828202905092915050565b7f506c656173652073656e642074686520657861637420616d6f756e742e000000600082015250565b6000613d64601d83613482565b9150613d6f82613d2e565b602082019050919050565b60006020820190508181036000830152613d9381613d57565b9050919050565b6000613da582613532565b9150613db083613532565b9250827fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff03821115613de557613de4613ca5565b5b828201905092915050565b7f596f752063616e74206d696e7420616e796d6f72650000000000000000000000600082015250565b6000613e26601583613482565b9150613e3182613df0565b602082019050919050565b60006020820190508181036000830152613e5581613e19565b9050919050565b7f4e6f206d6f726500000000000000000000000000000000000000000000000000600082015250565b6000613e92600783613482565b9150613e9d82613e5c565b602082019050919050565b60006020820190508181036000830152613ec181613e85565b9050919050565b7f506c6561736520656e7465722061206e756d6265720000000000000000000000600082015250565b6000613efe601583613482565b9150613f0982613ec8565b602082019050919050565b60006020820190508181036000830152613f2d81613ef1565b9050919050565b7f4d696e74696e67206973206e6f74206c69766520796574000000000000000000600082015250565b6000613f6a601783613482565b9150613f7582613f34565b602082019050919050565b60006020820190508181036000830152613f9981613f5d565b9050919050565b7f4d61782070657220747820657863656564656400000000000000000000000000600082015250565b6000613fd6601383613482565b9150613fe182613fa0565b602082019050919050565b6000602082019050818103600083015261400581613fc9565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b6000614042602083613482565b915061404d8261400c565b602082019050919050565b6000602082019050818103600083015261407181614035565b9050919050565b600081905092915050565b50565b6000614093600083614078565b915061409e82614083565b600082019050919050565b60006140b482614086565b9150819050919050565b7f5472616e73666572206661696c65642e00000000000000000000000000000000600082015250565b60006140f4601083613482565b91506140ff826140be565b602082019050919050565b60006020820190508181036000830152614123816140e7565b9050919050565b60008160601b9050919050565b60006141428261412a565b9050919050565b600061415482614137565b9050919050565b61416c614167826135b5565b614149565b82525050565b600061417e828461415b565b60148201915081905092915050565b7f496e636f72726563742057686974656c6973742050726f6f6600000000000000600082015250565b60006141c3601983613482565b91506141ce8261418d565b602082019050919050565b600060208201905081810360008301526141f2816141b6565b9050919050565b7f6e65656420746f206d696e74206174206c656173742031204e46540000000000600082015250565b600061422f601b83613482565b915061423a826141f9565b602082019050919050565b6000602082019050818103600083015261425e81614222565b9050919050565b7f6d6178204e4654206c696d697420657863656564656400000000000000000000600082015250565b600061429b601683613482565b91506142a682614265565b602082019050919050565b600060208201905081810360008301526142ca8161428e565b9050919050565b7f455243373231414d657461646174613a2055524920717565727920666f72206e60008201527f6f6e6578697374656e7420746f6b656e00000000000000000000000000000000602082015250565b600061432d603083613482565b9150614338826142d1565b604082019050919050565b6000602082019050818103600083015261435c81614320565b9050919050565b600081905092915050565b600061437982613477565b6143838185614363565b9350614393818560208601613493565b80840191505092915050565b7f2e6a736f6e000000000000000000000000000000000000000000000000000000600082015250565b60006143d5600583614363565b91506143e08261439f565b600582019050919050565b60006143f7828561436e565b9150614403828461436e565b915061440e826143c8565b91508190509392505050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b6000614476602683613482565b91506144818261441a565b604082019050919050565b600060208201905081810360008301526144a581614469565b9050919050565b600081519050919050565b600082825260208201905092915050565b60006144d3826144ac565b6144dd81856144b7565b93506144ed818560208601613493565b6144f6816134c6565b840191505092915050565b600060808201905061451660008301876135c7565b61452360208301866135c7565b614530604083018561365d565b818103606083015261454281846144c8565b905095945050505050565b60008151905061455c816133e8565b92915050565b600060208284031215614578576145776133b2565b5b60006145868482850161454d565b91505092915050565b600061459a82613532565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82036145cc576145cb613ca5565b5b600182019050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b600061461182613532565b915061461c83613532565b92508261462c5761462b6145d7565b5b828204905092915050565b600061464282613532565b915061464d83613532565b9250828210156146605761465f613ca5565b5b828203905092915050565b600061467682613532565b915061468183613532565b925082614691576146906145d7565b5b828206905092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fdfea2646970667358221220124521a7a189103aef94266d980080a72d870838b4e6e13d49ad84dec7f3c1b064736f6c634300080d0033
Deployed Bytecode Sourcemap
91229:5359:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;27764:615;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;32777:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;34845:204;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;34305:474;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;26818:315;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;35731:170;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;93478:536;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;95220:115;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;95440:100;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;96379:206;;;;;;;;;;;;;:::i;:::-;;35972:185;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;91417:31;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91638:26;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91566:28;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;94633:88;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;94832:86;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;92888:588;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;32566:144;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;94924:87;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;28443:224;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;58285:103;;;;;;;;;;;;;:::i;:::-;;91751:29;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;95737:203;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;92056:830;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;96063:310;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;91787:29;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91455:104;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;57634:87;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;95123:92;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;32946:104;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;95341:92;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;91823:33;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91337:35;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91601:28;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;35121:308;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;95655:75;;;;;;;;;;;;;:::i;:::-;;36228:396;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;94727:92;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;94140:481;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91671:24;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;91704:31;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;95947:113;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;95017:100;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;35500:164;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;58543:201;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;95546:103;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;91381:27;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;27764:615;27849:4;28164:10;28149:25;;:11;:25;;;;:102;;;;28241:10;28226:25;;:11;:25;;;;28149:102;:179;;;;28318:10;28303:25;;:11;:25;;;;28149:179;28129:199;;27764:615;;;:::o;32777:100::-;32831:13;32864:5;32857:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;32777:100;:::o;34845:204::-;34913:7;34938:16;34946:7;34938;:16::i;:::-;34933:64;;34963:34;;;;;;;;;;;;;;34933:64;35017:15;:24;35033:7;35017:24;;;;;;;;;;;;;;;;;;;;;35010:31;;34845:204;;;:::o;34305:474::-;34378:13;34410:27;34429:7;34410:18;:27::i;:::-;34378:61;;34460:5;34454:11;;:2;:11;;;34450:48;;34474:24;;;;;;;;;;;;;;34450:48;34538:5;34515:28;;:19;:17;:19::i;:::-;:28;;;34511:175;;34563:44;34580:5;34587:19;:17;:19::i;:::-;34563:16;:44::i;:::-;34558:128;;34635:35;;;;;;;;;;;;;;34558:128;34511:175;34725:2;34698:15;:24;34714:7;34698:24;;;;;;;;;;;;:29;;;;;;;;;;;;;;;;;;34763:7;34759:2;34743:28;;34752:5;34743:28;;;;;;;;;;;;34367:412;34305:474;;:::o;26818:315::-;26871:7;27099:15;:13;:15::i;:::-;27084:12;;27068:13;;:28;:46;27061:53;;26818:315;:::o;35731:170::-;35865:28;35875:4;35881:2;35885:7;35865:9;:28::i;:::-;35731:170;;;:::o;93478:536::-;93581:5;;93573;:13;;;;:::i;:::-;93560:9;:26;;93552:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;93673:12;;93665:5;93639:24;93652:10;93639:12;:24::i;:::-;:31;;;;:::i;:::-;:46;;93631:79;;;;;;;;;;;;:::i;:::-;;;;;;;;;93754:9;;93745:5;93729:13;:11;:13::i;:::-;:21;;;;:::i;:::-;:34;;93721:55;;;;;;;;;;;;:::i;:::-;;;;;;;;;93801:1;93795:5;:7;93787:40;;;;;;;;;;;;:::i;:::-;;;;;;;;;93846:13;;;;;;;;;;;93838:49;;;;;;;;;;;;:::i;:::-;;;;;;;;;93913:8;;93906:5;:15;;93898:46;;;;;;;;;;;;:::i;:::-;;;;;;;;;93978:28;93988:10;94000:5;93978:9;:28::i;:::-;93478:536;:::o;95220:115::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95293:6:::1;95281:9;;:18;;;;;;;;;;;;;;;;;;95321:6;95310:9;;:17;;;;;;;;;;;;;;;;;;95220:115:::0;:::o;95440:100::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95526:6:::1;95516:9;;:16;;;;;;;;;;;;;;;;;;95440:100:::0;:::o;96379:206::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;96430:12:::1;96456:10;96448:24;;96494:21;96448:82;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;96429:101;;;96549:7;96541:36;;;;;;;;;;;;:::i;:::-;;;;;;;;;96418:167;96379:206::o:0;35972:185::-;36110:39;36127:4;36133:2;36137:7;36110:39;;;;;;;;;;;;:16;:39::i;:::-;35972:185;;;:::o;91417:31::-;;;;:::o;91638:26::-;;;;:::o;91566:28::-;;;;;;;;;;;;;:::o;94633:88::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;94710:3:::1;94700:7;:13;;;;;;;;;;;;:::i;:::-;;94633:88:::0;:::o;94832:86::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;94904:6:::1;94896:5;:14;;;;94832:86:::0;:::o;92888:588::-;92991:12;93033:10;93016:28;;;;;;;;:::i;:::-;;;;;;;;;;;;;93006:39;;;;;;92991:54;;93074:50;93093:12;;93074:50;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;93107:10;;93119:4;93074:18;:50::i;:::-;93066:87;;;;;;;;;;;;:::i;:::-;;;;;;;;;93186:5;;93172:9;:19;;93164:61;;;;;;;;;;;;:::i;:::-;;;;;;;;;93273:9;;93269:1;93244:24;93257:10;93244:12;:24::i;:::-;:26;;;;:::i;:::-;:38;;93236:71;;;;;;;;;;;;:::i;:::-;;;;;;;;;93344:9;;93340:1;93326:13;:11;:13::i;:::-;:15;;;;:::i;:::-;:27;;93318:48;;;;;;;;;;;;:::i;:::-;;;;;;;;;93385:9;;;;;;;;;;;93377:45;;;;;;;;;;;;:::i;:::-;;;;;;;;;93444:24;93454:10;93466:1;93444:9;:24::i;:::-;92961:515;92888:588;;:::o;32566:144::-;32630:7;32673:27;32692:7;32673:18;:27::i;:::-;32650:52;;32566:144;;;:::o;94924:87::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;94997:6:::1;94987:7;:16;;;;94924:87:::0;:::o;28443:224::-;28507:7;28548:1;28531:19;;:5;:19;;;28527:60;;28559:28;;;;;;;;;;;;;;28527:60;23782:13;28605:18;:25;28624:5;28605:25;;;;;;;;;;;;;;;;:54;28598:61;;28443:224;;;:::o;58285:103::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;58350:30:::1;58377:1;58350:18;:30::i;:::-;58285:103::o:0;91751:29::-;;;;;;;;;;;;;:::o;95737:203::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95880:14:::1;95864:13;:30;;;;95918:14;95905:10;:27;;;;95737:203:::0;;:::o;92056:830::-;92150:12;92165:5;;92150:20;;92189:12;92231:10;92214:28;;;;;;;;:::i;:::-;;;;;;;;;;;;;92204:39;;;;;;92189:54;;92289:7;;92259:17;:29;92277:10;92259:29;;;;;;;;;;;;;;;;:37;:49;;;;;92307:1;92300:5;:8;92259:49;92256:86;;;92329:1;92324:6;;92256:86;92370:53;92389:12;;92370:53;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;92403:13;;92418:4;92370:18;:53::i;:::-;92362:90;;;;;;;;;;;;:::i;:::-;;;;;;;;;92484:4;92471:9;:17;;92463:59;;;;;;;;;;;;:::i;:::-;;;;;;;;;92575:5;;92567;92541:24;92554:10;92541:12;:24::i;:::-;:31;;;;:::i;:::-;:39;;92533:72;;;;;;;;;;;;:::i;:::-;;;;;;;;;92649:9;;92640:5;92624:13;:11;:13::i;:::-;:21;;;;:::i;:::-;:34;;92616:55;;;;;;;;;;;;:::i;:::-;;;;;;;;;92696:1;92690:5;:7;92682:40;;;;;;;;;;;;:::i;:::-;;;;;;;;;92741:9;;;;;;;;;;;92733:45;;;;;;;;;;;;:::i;:::-;;;;;;;;;92832:5;92801:17;:29;92819:10;92801:29;;;;;;;;;;;;;;;;:36;;;;;;;:::i;:::-;;;;;;;;92850:28;92860:10;92872:5;92850:9;:28::i;:::-;92139:747;;92056:830;;;:::o;96063:310::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;96170:1:::1;96156:11;:15;96148:55;;;;;;;;;;;;:::i;:::-;;;;;;;;;96210:14;96227:13;:11;:13::i;:::-;96210:30;;96279:9;;96264:11;96255:6;:20;;;;:::i;:::-;:33;;96247:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;96326:35;96336:11;96349;96326:9;:35::i;:::-;96141:232;96063:310:::0;;:::o;91787:29::-;;;;;;;;;;;;;:::o;91455:104::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;57634:87::-;57680:7;57707:6;;;;;;;;;;;57700:13;;57634:87;:::o;95123:92::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95198:9:::1;95190:5;:17;;;;95123:92:::0;:::o;32946:104::-;33002:13;33035:7;33028:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;32946:104;:::o;95341:92::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95419:6:::1;95407:9;;:18;;;;;;;;;;;;;;;;;;95341:92:::0;:::o;91823:33::-;;;;;;;;;;;;;:::o;91337:35::-;;;;:::o;91601:28::-;;;;:::o;35121:308::-;35232:19;:17;:19::i;:::-;35220:31;;:8;:31;;;35216:61;;35260:17;;;;;;;;;;;;;;35216:61;35342:8;35290:18;:39;35309:19;:17;:19::i;:::-;35290:39;;;;;;;;;;;;;;;:49;35330:8;35290:49;;;;;;;;;;;;;;;;:60;;;;;;;;;;;;;;;;;;35402:8;35366:55;;35381:19;:17;:19::i;:::-;35366:55;;;35412:8;35366:55;;;;;;:::i;:::-;;;;;;;;35121:308;;:::o;95655:75::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95711:8:::1;;;;;;;;;;;95710:9;95699:8;;:20;;;;;;;;;;;;;;;;;;95655:75::o:0;36228:396::-;36395:28;36405:4;36411:2;36415:7;36395:9;:28::i;:::-;36456:1;36438:2;:14;;;:19;36434:183;;36477:56;36508:4;36514:2;36518:7;36527:5;36477:30;:56::i;:::-;36472:145;;36561:40;;;;;;;;;;;;;;36472:145;36434:183;36228:396;;;;:::o;94727:92::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;94808:3:::1;94796:9;:15;;;;;;;;;;;;:::i;:::-;;94727:92:::0;:::o;94140:481::-;94238:13;94279:16;94287:7;94279;:16::i;:::-;94263:98;;;;;;;;;;;;:::i;:::-;;;;;;;;;94385:5;94373:17;;:8;;;;;;;;;;;:17;;;94369:56;;94408:9;94401:16;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;94369:56;94433:28;94464:10;:8;:10::i;:::-;94433:41;;94519:1;94494:14;94488:28;:32;:127;;;;;;;;;;;;;;;;;94556:14;94572:18;:7;:16;:18::i;:::-;94539:61;;;;;;;;;:::i;:::-;;;;;;;;;;;;;94488:127;94481:134;;;94140:481;;;;:::o;91671:24::-;;;;:::o;91704:31::-;;;;:::o;95947:113::-;96005:7;96032:20;96046:5;96032:13;:20::i;:::-;96025:27;;95947:113;;;:::o;95017:100::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95103:6:::1;95088:12;:21;;;;95017:100:::0;:::o;35500:164::-;35597:4;35621:18;:25;35640:5;35621:25;;;;;;;;;;;;;;;:35;35647:8;35621:35;;;;;;;;;;;;;;;;;;;;;;;;;35614:42;;35500:164;;;;:::o;58543:201::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;58652:1:::1;58632:22;;:8;:22;;::::0;58624:73:::1;;;;;;;;;;;;:::i;:::-;;;;;;;;;58708:28;58727:8;58708:18;:28::i;:::-;58543:201:::0;:::o;95546:103::-;57865:12;:10;:12::i;:::-;57854:23;;:7;:5;:7::i;:::-;:23;;;57846:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;95635:6:::1;95619:13;;:22;;;;;;;;;;;;;;;;;;95546:103:::0;:::o;91381:27::-;;;;:::o;36879:273::-;36936:4;36992:7;36973:15;:13;:15::i;:::-;:26;;:66;;;;;37026:13;;37016:7;:23;36973:66;:152;;;;;37124:1;24552:8;37077:17;:26;37095:7;37077:26;;;;;;;;;;;;:43;:48;36973:152;36953:172;;36879:273;;;:::o;30081:1129::-;30148:7;30168:12;30183:7;30168:22;;30251:4;30232:15;:13;:15::i;:::-;:23;30228:915;;30285:13;;30278:4;:20;30274:869;;;30323:14;30340:17;:23;30358:4;30340:23;;;;;;;;;;;;30323:40;;30456:1;24552:8;30429:6;:23;:28;30425:699;;30948:113;30965:1;30955:6;:11;30948:113;;31008:17;:25;31026:6;;;;;;;31008:25;;;;;;;;;;;;30999:34;;30948:113;;;31094:6;31087:13;;;;;;30425:699;30300:843;30274:869;30228:915;31171:31;;;;;;;;;;;;;;30081:1129;;;;:::o;50861:105::-;50921:7;50948:10;50941:17;;50861:105;:::o;26341:92::-;26397:7;26341:92;:::o;42118:2515::-;42233:27;42263;42282:7;42263:18;:27::i;:::-;42233:57;;42348:4;42307:45;;42323:19;42307:45;;;42303:86;;42361:28;;;;;;;;;;;;;;42303:86;42402:22;42451:4;42428:27;;:19;:17;:19::i;:::-;:27;;;:87;;;;42472:43;42489:4;42495:19;:17;:19::i;:::-;42472:16;:43::i;:::-;42428:87;:147;;;;42556:19;:17;:19::i;:::-;42532:43;;:20;42544:7;42532:11;:20::i;:::-;:43;;;42428:147;42402:174;;42594:17;42589:66;;42620:35;;;;;;;;;;;;;;42589:66;42684:1;42670:16;;:2;:16;;;42666:52;;42695:23;;;;;;;;;;;;;;42666:52;42731:43;42753:4;42759:2;42763:7;42772:1;42731:21;:43::i;:::-;42847:15;:24;42863:7;42847:24;;;;;;;;;;;;42840:31;;;;;;;;;;;43239:18;:24;43258:4;43239:24;;;;;;;;;;;;;;;;43237:26;;;;;;;;;;;;43308:18;:22;43327:2;43308:22;;;;;;;;;;;;;;;;43306:24;;;;;;;;;;;24834:8;24436:3;43689:15;:41;;43647:21;43665:2;43647:17;:21::i;:::-;:84;:128;43601:17;:26;43619:7;43601:26;;;;;;;;;;;:174;;;;43945:1;24834:8;43895:19;:46;:51;43891:626;;43967:19;43999:1;43989:7;:11;43967:33;;44156:1;44122:17;:30;44140:11;44122:30;;;;;;;;;;;;:35;44118:384;;44260:13;;44245:11;:28;44241:242;;44440:19;44407:17;:30;44425:11;44407:30;;;;;;;;;;;:52;;;;44241:242;44118:384;43948:569;43891:626;44564:7;44560:2;44545:27;;44554:4;44545:27;;;;;;;;;;;;44583:42;44604:4;44610:2;44614:7;44623:1;44583:20;:42::i;:::-;42222:2411;;42118:2515;;;:::o;37236:104::-;37305:27;37315:2;37319:8;37305:27;;;;;;;;;;;;:9;:27::i;:::-;37236:104;;:::o;56305:98::-;56358:7;56385:10;56378:17;;56305:98;:::o;7187:190::-;7312:4;7365;7336:25;7349:5;7356:4;7336:12;:25::i;:::-;:33;7329:40;;7187:190;;;;;:::o;58904:191::-;58978:16;58997:6;;;;;;;;;;;58978:25;;59023:8;59014:6;;:17;;;;;;;;;;;;;;;;;;59078:8;59047:40;;59068:8;59047:40;;;;;;;;;;;;58967:128;58904:191;:::o;48330:716::-;48493:4;48539:2;48514:45;;;48560:19;:17;:19::i;:::-;48581:4;48587:7;48596:5;48514:88;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;48510:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;48814:1;48797:6;:13;:18;48793:235;;48843:40;;;;;;;;;;;;;;48793:235;48986:6;48980:13;48971:6;48967:2;48963:15;48956:38;48510:529;48683:54;;;48673:64;;;:6;:64;;;;48666:71;;;48330:716;;;;;;:::o;94020:108::-;94080:13;94113:7;94106:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;94020:108;:::o;53506:723::-;53562:13;53792:1;53783:5;:10;53779:53;;53810:10;;;;;;;;;;;;;;;;;;;;;53779:53;53842:12;53857:5;53842:20;;53873:14;53898:78;53913:1;53905:4;:9;53898:78;;53931:8;;;;;:::i;:::-;;;;53962:2;53954:10;;;;;:::i;:::-;;;53898:78;;;53986:19;54018:6;54008:17;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;53986:39;;54036:154;54052:1;54043:5;:10;54036:154;;54080:1;54070:11;;;;;:::i;:::-;;;54147:2;54139:5;:10;;;;:::i;:::-;54126:2;:24;;;;:::i;:::-;54113:39;;54096:6;54103;54096:14;;;;;;;;:::i;:::-;;;;;:56;;;;;;;;;;;54176:2;54167:11;;;;;:::i;:::-;;;54036:154;;;54214:6;54200:21;;;;;53506:723;;;;:::o;28749:176::-;28810:7;23782:13;23919:2;28838:18;:25;28857:5;28838:25;;;;;;;;;;;;;;;;:49;;28837:80;28830:87;;28749:176;;;:::o;49694:159::-;;;;;:::o;33866:148::-;33930:14;33991:5;33981:15;;33866:148;;;:::o;50512:158::-;;;;;:::o;37713:2236::-;37836:20;37859:13;;37836:36;;37901:1;37887:16;;:2;:16;;;37883:48;;37912:19;;;;;;;;;;;;;;37883:48;37958:1;37946:8;:13;37942:44;;37968:18;;;;;;;;;;;;;;37942:44;37999:61;38029:1;38033:2;38037:12;38051:8;37999:21;:61::i;:::-;38603:1;23919:2;38574:1;:25;;38573:31;38561:8;:44;38535:18;:22;38554:2;38535:22;;;;;;;;;;;;;;;;:70;;;;;;;;;;;24699:3;39004:29;39031:1;39019:8;:13;39004:14;:29::i;:::-;:56;;24436:3;38941:15;:41;;38899:21;38917:2;38899:17;:21::i;:::-;:84;:162;38848:17;:31;38866:12;38848:31;;;;;;;;;;;:213;;;;39078:20;39101:12;39078:35;;39128:11;39157:8;39142:12;:23;39128:37;;39204:1;39186:2;:14;;;:19;39182:635;;39226:313;39282:12;39278:2;39257:38;;39274:1;39257:38;;;;;;;;;;;;39323:69;39362:1;39366:2;39370:14;;;;;;39386:5;39323:30;:69::i;:::-;39318:174;;39428:40;;;;;;;;;;;;;;39318:174;39534:3;39519:12;:18;39226:313;;39620:12;39603:13;;:29;39599:43;;39634:8;;;39599:43;39182:635;;;39683:119;39739:14;;;;;;39735:2;39714:40;;39731:1;39714:40;;;;;;;;;;;;39797:3;39782:12;:18;39683:119;;39182:635;39847:12;39831:13;:28;;;;38312:1559;;39881:60;39910:1;39914:2;39918:12;39932:8;39881:20;:60::i;:::-;37825:2124;37713:2236;;;:::o;8054:296::-;8137:7;8157:20;8180:4;8157:27;;8200:9;8195:118;8219:5;:12;8215:1;:16;8195:118;;;8268:33;8278:12;8292:5;8298:1;8292:8;;;;;;;;:::i;:::-;;;;;;;;8268:9;:33::i;:::-;8253:48;;8233:3;;;;;:::i;:::-;;;;8195:118;;;;8330:12;8323:19;;;8054:296;;;;:::o;34101:142::-;34159:14;34220:5;34210:15;;34101:142;;;:::o;14261:149::-;14324:7;14355:1;14351;:5;:51;;14382:20;14397:1;14400;14382:14;:20::i;:::-;14351:51;;;14359:20;14374:1;14377;14359:14;:20::i;:::-;14351:51;14344:58;;14261:149;;;;:::o;14418:268::-;14486:13;14593:1;14587:4;14580:15;14622:1;14616:4;14609:15;14663:4;14657;14647:21;14638:30;;14418:268;;;;:::o;-1:-1:-1:-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;:::o;:::-;;;;;;;;;;;;;;;;;;;;;:::o;7:75:1:-;40:6;73:2;67:9;57:19;;7:75;:::o;88:117::-;197:1;194;187:12;211:117;320:1;317;310:12;334:149;370:7;410:66;403:5;399:78;388:89;;334:149;;;:::o;489:120::-;561:23;578:5;561:23;:::i;:::-;554:5;551:34;541:62;;599:1;596;589:12;541:62;489:120;:::o;615:137::-;660:5;698:6;685:20;676:29;;714:32;740:5;714:32;:::i;:::-;615:137;;;;:::o;758:327::-;816:6;865:2;853:9;844:7;840:23;836:32;833:119;;;871:79;;:::i;:::-;833:119;991:1;1016:52;1060:7;1051:6;1040:9;1036:22;1016:52;:::i;:::-;1006:62;;962:116;758:327;;;;:::o;1091:90::-;1125:7;1168:5;1161:13;1154:21;1143:32;;1091:90;;;:::o;1187:109::-;1268:21;1283:5;1268:21;:::i;:::-;1263:3;1256:34;1187:109;;:::o;1302:210::-;1389:4;1427:2;1416:9;1412:18;1404:26;;1440:65;1502:1;1491:9;1487:17;1478:6;1440:65;:::i;:::-;1302:210;;;;:::o;1518:99::-;1570:6;1604:5;1598:12;1588:22;;1518:99;;;:::o;1623:169::-;1707:11;1741:6;1736:3;1729:19;1781:4;1776:3;1772:14;1757:29;;1623:169;;;;:::o;1798:307::-;1866:1;1876:113;1890:6;1887:1;1884:13;1876:113;;;1975:1;1970:3;1966:11;1960:18;1956:1;1951:3;1947:11;1940:39;1912:2;1909:1;1905:10;1900:15;;1876:113;;;2007:6;2004:1;2001:13;1998:101;;;2087:1;2078:6;2073:3;2069:16;2062:27;1998:101;1847:258;1798:307;;;:::o;2111:102::-;2152:6;2203:2;2199:7;2194:2;2187:5;2183:14;2179:28;2169:38;;2111:102;;;:::o;2219:364::-;2307:3;2335:39;2368:5;2335:39;:::i;:::-;2390:71;2454:6;2449:3;2390:71;:::i;:::-;2383:78;;2470:52;2515:6;2510:3;2503:4;2496:5;2492:16;2470:52;:::i;:::-;2547:29;2569:6;2547:29;:::i;:::-;2542:3;2538:39;2531:46;;2311:272;2219:364;;;;:::o;2589:313::-;2702:4;2740:2;2729:9;2725:18;2717:26;;2789:9;2783:4;2779:20;2775:1;2764:9;2760:17;2753:47;2817:78;2890:4;2881:6;2817:78;:::i;:::-;2809:86;;2589:313;;;;:::o;2908:77::-;2945:7;2974:5;2963:16;;2908:77;;;:::o;2991:122::-;3064:24;3082:5;3064:24;:::i;:::-;3057:5;3054:35;3044:63;;3103:1;3100;3093:12;3044:63;2991:122;:::o;3119:139::-;3165:5;3203:6;3190:20;3181:29;;3219:33;3246:5;3219:33;:::i;:::-;3119:139;;;;:::o;3264:329::-;3323:6;3372:2;3360:9;3351:7;3347:23;3343:32;3340:119;;;3378:79;;:::i;:::-;3340:119;3498:1;3523:53;3568:7;3559:6;3548:9;3544:22;3523:53;:::i;:::-;3513:63;;3469:117;3264:329;;;;:::o;3599:126::-;3636:7;3676:42;3669:5;3665:54;3654:65;;3599:126;;;:::o;3731:96::-;3768:7;3797:24;3815:5;3797:24;:::i;:::-;3786:35;;3731:96;;;:::o;3833:118::-;3920:24;3938:5;3920:24;:::i;:::-;3915:3;3908:37;3833:118;;:::o;3957:222::-;4050:4;4088:2;4077:9;4073:18;4065:26;;4101:71;4169:1;4158:9;4154:17;4145:6;4101:71;:::i;:::-;3957:222;;;;:::o;4185:122::-;4258:24;4276:5;4258:24;:::i;:::-;4251:5;4248:35;4238:63;;4297:1;4294;4287:12;4238:63;4185:122;:::o;4313:139::-;4359:5;4397:6;4384:20;4375:29;;4413:33;4440:5;4413:33;:::i;:::-;4313:139;;;;:::o;4458:474::-;4526:6;4534;4583:2;4571:9;4562:7;4558:23;4554:32;4551:119;;;4589:79;;:::i;:::-;4551:119;4709:1;4734:53;4779:7;4770:6;4759:9;4755:22;4734:53;:::i;:::-;4724:63;;4680:117;4836:2;4862:53;4907:7;4898:6;4887:9;4883:22;4862:53;:::i;:::-;4852:63;;4807:118;4458:474;;;;;:::o;4938:118::-;5025:24;5043:5;5025:24;:::i;:::-;5020:3;5013:37;4938:118;;:::o;5062:222::-;5155:4;5193:2;5182:9;5178:18;5170:26;;5206:71;5274:1;5263:9;5259:17;5250:6;5206:71;:::i;:::-;5062:222;;;;:::o;5290:619::-;5367:6;5375;5383;5432:2;5420:9;5411:7;5407:23;5403:32;5400:119;;;5438:79;;:::i;:::-;5400:119;5558:1;5583:53;5628:7;5619:6;5608:9;5604:22;5583:53;:::i;:::-;5573:63;;5529:117;5685:2;5711:53;5756:7;5747:6;5736:9;5732:22;5711:53;:::i;:::-;5701:63;;5656:118;5813:2;5839:53;5884:7;5875:6;5864:9;5860:22;5839:53;:::i;:::-;5829:63;;5784:118;5290:619;;;;;:::o;5915:116::-;5985:21;6000:5;5985:21;:::i;:::-;5978:5;5975:32;5965:60;;6021:1;6018;6011:12;5965:60;5915:116;:::o;6037:133::-;6080:5;6118:6;6105:20;6096:29;;6134:30;6158:5;6134:30;:::i;:::-;6037:133;;;;:::o;6176:323::-;6232:6;6281:2;6269:9;6260:7;6256:23;6252:32;6249:119;;;6287:79;;:::i;:::-;6249:119;6407:1;6432:50;6474:7;6465:6;6454:9;6450:22;6432:50;:::i;:::-;6422:60;;6378:114;6176:323;;;;:::o;6505:117::-;6614:1;6611;6604:12;6628:117;6737:1;6734;6727:12;6751:180;6799:77;6796:1;6789:88;6896:4;6893:1;6886:15;6920:4;6917:1;6910:15;6937:281;7020:27;7042:4;7020:27;:::i;:::-;7012:6;7008:40;7150:6;7138:10;7135:22;7114:18;7102:10;7099:34;7096:62;7093:88;;;7161:18;;:::i;:::-;7093:88;7201:10;7197:2;7190:22;6980:238;6937:281;;:::o;7224:129::-;7258:6;7285:20;;:::i;:::-;7275:30;;7314:33;7342:4;7334:6;7314:33;:::i;:::-;7224:129;;;:::o;7359:308::-;7421:4;7511:18;7503:6;7500:30;7497:56;;;7533:18;;:::i;:::-;7497:56;7571:29;7593:6;7571:29;:::i;:::-;7563:37;;7655:4;7649;7645:15;7637:23;;7359:308;;;:::o;7673:154::-;7757:6;7752:3;7747;7734:30;7819:1;7810:6;7805:3;7801:16;7794:27;7673:154;;;:::o;7833:412::-;7911:5;7936:66;7952:49;7994:6;7952:49;:::i;:::-;7936:66;:::i;:::-;7927:75;;8025:6;8018:5;8011:21;8063:4;8056:5;8052:16;8101:3;8092:6;8087:3;8083:16;8080:25;8077:112;;;8108:79;;:::i;:::-;8077:112;8198:41;8232:6;8227:3;8222;8198:41;:::i;:::-;7917:328;7833:412;;;;;:::o;8265:340::-;8321:5;8370:3;8363:4;8355:6;8351:17;8347:27;8337:122;;8378:79;;:::i;:::-;8337:122;8495:6;8482:20;8520:79;8595:3;8587:6;8580:4;8572:6;8568:17;8520:79;:::i;:::-;8511:88;;8327:278;8265:340;;;;:::o;8611:509::-;8680:6;8729:2;8717:9;8708:7;8704:23;8700:32;8697:119;;;8735:79;;:::i;:::-;8697:119;8883:1;8872:9;8868:17;8855:31;8913:18;8905:6;8902:30;8899:117;;;8935:79;;:::i;:::-;8899:117;9040:63;9095:7;9086:6;9075:9;9071:22;9040:63;:::i;:::-;9030:73;;8826:287;8611:509;;;;:::o;9126:117::-;9235:1;9232;9225:12;9249:117;9358:1;9355;9348:12;9389:568;9462:8;9472:6;9522:3;9515:4;9507:6;9503:17;9499:27;9489:122;;9530:79;;:::i;:::-;9489:122;9643:6;9630:20;9620:30;;9673:18;9665:6;9662:30;9659:117;;;9695:79;;:::i;:::-;9659:117;9809:4;9801:6;9797:17;9785:29;;9863:3;9855:4;9847:6;9843:17;9833:8;9829:32;9826:41;9823:128;;;9870:79;;:::i;:::-;9823:128;9389:568;;;;;:::o;9963:559::-;10049:6;10057;10106:2;10094:9;10085:7;10081:23;10077:32;10074:119;;;10112:79;;:::i;:::-;10074:119;10260:1;10249:9;10245:17;10232:31;10290:18;10282:6;10279:30;10276:117;;;10312:79;;:::i;:::-;10276:117;10425:80;10497:7;10488:6;10477:9;10473:22;10425:80;:::i;:::-;10407:98;;;;10203:312;9963:559;;;;;:::o;10528:329::-;10587:6;10636:2;10624:9;10615:7;10611:23;10607:32;10604:119;;;10642:79;;:::i;:::-;10604:119;10762:1;10787:53;10832:7;10823:6;10812:9;10808:22;10787:53;:::i;:::-;10777:63;;10733:117;10528:329;;;;:::o;10863:77::-;10900:7;10929:5;10918:16;;10863:77;;;:::o;10946:122::-;11019:24;11037:5;11019:24;:::i;:::-;11012:5;11009:35;10999:63;;11058:1;11055;11048:12;10999:63;10946:122;:::o;11074:139::-;11120:5;11158:6;11145:20;11136:29;;11174:33;11201:5;11174:33;:::i;:::-;11074:139;;;;:::o;11219:474::-;11287:6;11295;11344:2;11332:9;11323:7;11319:23;11315:32;11312:119;;;11350:79;;:::i;:::-;11312:119;11470:1;11495:53;11540:7;11531:6;11520:9;11516:22;11495:53;:::i;:::-;11485:63;;11441:117;11597:2;11623:53;11668:7;11659:6;11648:9;11644:22;11623:53;:::i;:::-;11613:63;;11568:118;11219:474;;;;;:::o;11699:704::-;11794:6;11802;11810;11859:2;11847:9;11838:7;11834:23;11830:32;11827:119;;;11865:79;;:::i;:::-;11827:119;12013:1;12002:9;11998:17;11985:31;12043:18;12035:6;12032:30;12029:117;;;12065:79;;:::i;:::-;12029:117;12178:80;12250:7;12241:6;12230:9;12226:22;12178:80;:::i;:::-;12160:98;;;;11956:312;12307:2;12333:53;12378:7;12369:6;12358:9;12354:22;12333:53;:::i;:::-;12323:63;;12278:118;11699:704;;;;;:::o;12409:474::-;12477:6;12485;12534:2;12522:9;12513:7;12509:23;12505:32;12502:119;;;12540:79;;:::i;:::-;12502:119;12660:1;12685:53;12730:7;12721:6;12710:9;12706:22;12685:53;:::i;:::-;12675:63;;12631:117;12787:2;12813:53;12858:7;12849:6;12838:9;12834:22;12813:53;:::i;:::-;12803:63;;12758:118;12409:474;;;;;:::o;12889:468::-;12954:6;12962;13011:2;12999:9;12990:7;12986:23;12982:32;12979:119;;;13017:79;;:::i;:::-;12979:119;13137:1;13162:53;13207:7;13198:6;13187:9;13183:22;13162:53;:::i;:::-;13152:63;;13108:117;13264:2;13290:50;13332:7;13323:6;13312:9;13308:22;13290:50;:::i;:::-;13280:60;;13235:115;12889:468;;;;;:::o;13363:307::-;13424:4;13514:18;13506:6;13503:30;13500:56;;;13536:18;;:::i;:::-;13500:56;13574:29;13596:6;13574:29;:::i;:::-;13566:37;;13658:4;13652;13648:15;13640:23;;13363:307;;;:::o;13676:410::-;13753:5;13778:65;13794:48;13835:6;13794:48;:::i;:::-;13778:65;:::i;:::-;13769:74;;13866:6;13859:5;13852:21;13904:4;13897:5;13893:16;13942:3;13933:6;13928:3;13924:16;13921:25;13918:112;;;13949:79;;:::i;:::-;13918:112;14039:41;14073:6;14068:3;14063;14039:41;:::i;:::-;13759:327;13676:410;;;;;:::o;14105:338::-;14160:5;14209:3;14202:4;14194:6;14190:17;14186:27;14176:122;;14217:79;;:::i;:::-;14176:122;14334:6;14321:20;14359:78;14433:3;14425:6;14418:4;14410:6;14406:17;14359:78;:::i;:::-;14350:87;;14166:277;14105:338;;;;:::o;14449:943::-;14544:6;14552;14560;14568;14617:3;14605:9;14596:7;14592:23;14588:33;14585:120;;;14624:79;;:::i;:::-;14585:120;14744:1;14769:53;14814:7;14805:6;14794:9;14790:22;14769:53;:::i;:::-;14759:63;;14715:117;14871:2;14897:53;14942:7;14933:6;14922:9;14918:22;14897:53;:::i;:::-;14887:63;;14842:118;14999:2;15025:53;15070:7;15061:6;15050:9;15046:22;15025:53;:::i;:::-;15015:63;;14970:118;15155:2;15144:9;15140:18;15127:32;15186:18;15178:6;15175:30;15172:117;;;15208:79;;:::i;:::-;15172:117;15313:62;15367:7;15358:6;15347:9;15343:22;15313:62;:::i;:::-;15303:72;;15098:287;14449:943;;;;;;;:::o;15398:474::-;15466:6;15474;15523:2;15511:9;15502:7;15498:23;15494:32;15491:119;;;15529:79;;:::i;:::-;15491:119;15649:1;15674:53;15719:7;15710:6;15699:9;15695:22;15674:53;:::i;:::-;15664:63;;15620:117;15776:2;15802:53;15847:7;15838:6;15827:9;15823:22;15802:53;:::i;:::-;15792:63;;15747:118;15398:474;;;;;:::o;15878:180::-;15926:77;15923:1;15916:88;16023:4;16020:1;16013:15;16047:4;16044:1;16037:15;16064:320;16108:6;16145:1;16139:4;16135:12;16125:22;;16192:1;16186:4;16182:12;16213:18;16203:81;;16269:4;16261:6;16257:17;16247:27;;16203:81;16331:2;16323:6;16320:14;16300:18;16297:38;16294:84;;16350:18;;:::i;:::-;16294:84;16115:269;16064:320;;;:::o;16390:180::-;16438:77;16435:1;16428:88;16535:4;16532:1;16525:15;16559:4;16556:1;16549:15;16576:348;16616:7;16639:20;16657:1;16639:20;:::i;:::-;16634:25;;16673:20;16691:1;16673:20;:::i;:::-;16668:25;;16861:1;16793:66;16789:74;16786:1;16783:81;16778:1;16771:9;16764:17;16760:105;16757:131;;;16868:18;;:::i;:::-;16757:131;16916:1;16913;16909:9;16898:20;;16576:348;;;;:::o;16930:179::-;17070:31;17066:1;17058:6;17054:14;17047:55;16930:179;:::o;17115:366::-;17257:3;17278:67;17342:2;17337:3;17278:67;:::i;:::-;17271:74;;17354:93;17443:3;17354:93;:::i;:::-;17472:2;17467:3;17463:12;17456:19;;17115:366;;;:::o;17487:419::-;17653:4;17691:2;17680:9;17676:18;17668:26;;17740:9;17734:4;17730:20;17726:1;17715:9;17711:17;17704:47;17768:131;17894:4;17768:131;:::i;:::-;17760:139;;17487:419;;;:::o;17912:305::-;17952:3;17971:20;17989:1;17971:20;:::i;:::-;17966:25;;18005:20;18023:1;18005:20;:::i;:::-;18000:25;;18159:1;18091:66;18087:74;18084:1;18081:81;18078:107;;;18165:18;;:::i;:::-;18078:107;18209:1;18206;18202:9;18195:16;;17912:305;;;;:::o;18223:171::-;18363:23;18359:1;18351:6;18347:14;18340:47;18223:171;:::o;18400:366::-;18542:3;18563:67;18627:2;18622:3;18563:67;:::i;:::-;18556:74;;18639:93;18728:3;18639:93;:::i;:::-;18757:2;18752:3;18748:12;18741:19;;18400:366;;;:::o;18772:419::-;18938:4;18976:2;18965:9;18961:18;18953:26;;19025:9;19019:4;19015:20;19011:1;19000:9;18996:17;18989:47;19053:131;19179:4;19053:131;:::i;:::-;19045:139;;18772:419;;;:::o;19197:157::-;19337:9;19333:1;19325:6;19321:14;19314:33;19197:157;:::o;19360:365::-;19502:3;19523:66;19587:1;19582:3;19523:66;:::i;:::-;19516:73;;19598:93;19687:3;19598:93;:::i;:::-;19716:2;19711:3;19707:12;19700:19;;19360:365;;;:::o;19731:419::-;19897:4;19935:2;19924:9;19920:18;19912:26;;19984:9;19978:4;19974:20;19970:1;19959:9;19955:17;19948:47;20012:131;20138:4;20012:131;:::i;:::-;20004:139;;19731:419;;;:::o;20156:171::-;20296:23;20292:1;20284:6;20280:14;20273:47;20156:171;:::o;20333:366::-;20475:3;20496:67;20560:2;20555:3;20496:67;:::i;:::-;20489:74;;20572:93;20661:3;20572:93;:::i;:::-;20690:2;20685:3;20681:12;20674:19;;20333:366;;;:::o;20705:419::-;20871:4;20909:2;20898:9;20894:18;20886:26;;20958:9;20952:4;20948:20;20944:1;20933:9;20929:17;20922:47;20986:131;21112:4;20986:131;:::i;:::-;20978:139;;20705:419;;;:::o;21130:173::-;21270:25;21266:1;21258:6;21254:14;21247:49;21130:173;:::o;21309:366::-;21451:3;21472:67;21536:2;21531:3;21472:67;:::i;:::-;21465:74;;21548:93;21637:3;21548:93;:::i;:::-;21666:2;21661:3;21657:12;21650:19;;21309:366;;;:::o;21681:419::-;21847:4;21885:2;21874:9;21870:18;21862:26;;21934:9;21928:4;21924:20;21920:1;21909:9;21905:17;21898:47;21962:131;22088:4;21962:131;:::i;:::-;21954:139;;21681:419;;;:::o;22106:169::-;22246:21;22242:1;22234:6;22230:14;22223:45;22106:169;:::o;22281:366::-;22423:3;22444:67;22508:2;22503:3;22444:67;:::i;:::-;22437:74;;22520:93;22609:3;22520:93;:::i;:::-;22638:2;22633:3;22629:12;22622:19;;22281:366;;;:::o;22653:419::-;22819:4;22857:2;22846:9;22842:18;22834:26;;22906:9;22900:4;22896:20;22892:1;22881:9;22877:17;22870:47;22934:131;23060:4;22934:131;:::i;:::-;22926:139;;22653:419;;;:::o;23078:182::-;23218:34;23214:1;23206:6;23202:14;23195:58;23078:182;:::o;23266:366::-;23408:3;23429:67;23493:2;23488:3;23429:67;:::i;:::-;23422:74;;23505:93;23594:3;23505:93;:::i;:::-;23623:2;23618:3;23614:12;23607:19;;23266:366;;;:::o;23638:419::-;23804:4;23842:2;23831:9;23827:18;23819:26;;23891:9;23885:4;23881:20;23877:1;23866:9;23862:17;23855:47;23919:131;24045:4;23919:131;:::i;:::-;23911:139;;23638:419;;;:::o;24063:147::-;24164:11;24201:3;24186:18;;24063:147;;;;:::o;24216:114::-;;:::o;24336:398::-;24495:3;24516:83;24597:1;24592:3;24516:83;:::i;:::-;24509:90;;24608:93;24697:3;24608:93;:::i;:::-;24726:1;24721:3;24717:11;24710:18;;24336:398;;;:::o;24740:379::-;24924:3;24946:147;25089:3;24946:147;:::i;:::-;24939:154;;25110:3;25103:10;;24740:379;;;:::o;25125:166::-;25265:18;25261:1;25253:6;25249:14;25242:42;25125:166;:::o;25297:366::-;25439:3;25460:67;25524:2;25519:3;25460:67;:::i;:::-;25453:74;;25536:93;25625:3;25536:93;:::i;:::-;25654:2;25649:3;25645:12;25638:19;;25297:366;;;:::o;25669:419::-;25835:4;25873:2;25862:9;25858:18;25850:26;;25922:9;25916:4;25912:20;25908:1;25897:9;25893:17;25886:47;25950:131;26076:4;25950:131;:::i;:::-;25942:139;;25669:419;;;:::o;26094:94::-;26127:8;26175:5;26171:2;26167:14;26146:35;;26094:94;;;:::o;26194:::-;26233:7;26262:20;26276:5;26262:20;:::i;:::-;26251:31;;26194:94;;;:::o;26294:100::-;26333:7;26362:26;26382:5;26362:26;:::i;:::-;26351:37;;26294:100;;;:::o;26400:157::-;26505:45;26525:24;26543:5;26525:24;:::i;:::-;26505:45;:::i;:::-;26500:3;26493:58;26400:157;;:::o;26563:256::-;26675:3;26690:75;26761:3;26752:6;26690:75;:::i;:::-;26790:2;26785:3;26781:12;26774:19;;26810:3;26803:10;;26563:256;;;;:::o;26825:175::-;26965:27;26961:1;26953:6;26949:14;26942:51;26825:175;:::o;27006:366::-;27148:3;27169:67;27233:2;27228:3;27169:67;:::i;:::-;27162:74;;27245:93;27334:3;27245:93;:::i;:::-;27363:2;27358:3;27354:12;27347:19;;27006:366;;;:::o;27378:419::-;27544:4;27582:2;27571:9;27567:18;27559:26;;27631:9;27625:4;27621:20;27617:1;27606:9;27602:17;27595:47;27659:131;27785:4;27659:131;:::i;:::-;27651:139;;27378:419;;;:::o;27803:177::-;27943:29;27939:1;27931:6;27927:14;27920:53;27803:177;:::o;27986:366::-;28128:3;28149:67;28213:2;28208:3;28149:67;:::i;:::-;28142:74;;28225:93;28314:3;28225:93;:::i;:::-;28343:2;28338:3;28334:12;28327:19;;27986:366;;;:::o;28358:419::-;28524:4;28562:2;28551:9;28547:18;28539:26;;28611:9;28605:4;28601:20;28597:1;28586:9;28582:17;28575:47;28639:131;28765:4;28639:131;:::i;:::-;28631:139;;28358:419;;;:::o;28783:172::-;28923:24;28919:1;28911:6;28907:14;28900:48;28783:172;:::o;28961:366::-;29103:3;29124:67;29188:2;29183:3;29124:67;:::i;:::-;29117:74;;29200:93;29289:3;29200:93;:::i;:::-;29318:2;29313:3;29309:12;29302:19;;28961:366;;;:::o;29333:419::-;29499:4;29537:2;29526:9;29522:18;29514:26;;29586:9;29580:4;29576:20;29572:1;29561:9;29557:17;29550:47;29614:131;29740:4;29614:131;:::i;:::-;29606:139;;29333:419;;;:::o;29758:235::-;29898:34;29894:1;29886:6;29882:14;29875:58;29967:18;29962:2;29954:6;29950:15;29943:43;29758:235;:::o;29999:366::-;30141:3;30162:67;30226:2;30221:3;30162:67;:::i;:::-;30155:74;;30238:93;30327:3;30238:93;:::i;:::-;30356:2;30351:3;30347:12;30340:19;;29999:366;;;:::o;30371:419::-;30537:4;30575:2;30564:9;30560:18;30552:26;;30624:9;30618:4;30614:20;30610:1;30599:9;30595:17;30588:47;30652:131;30778:4;30652:131;:::i;:::-;30644:139;;30371:419;;;:::o;30796:148::-;30898:11;30935:3;30920:18;;30796:148;;;;:::o;30950:377::-;31056:3;31084:39;31117:5;31084:39;:::i;:::-;31139:89;31221:6;31216:3;31139:89;:::i;:::-;31132:96;;31237:52;31282:6;31277:3;31270:4;31263:5;31259:16;31237:52;:::i;:::-;31314:6;31309:3;31305:16;31298:23;;31060:267;30950:377;;;;:::o;31333:155::-;31473:7;31469:1;31461:6;31457:14;31450:31;31333:155;:::o;31494:400::-;31654:3;31675:84;31757:1;31752:3;31675:84;:::i;:::-;31668:91;;31768:93;31857:3;31768:93;:::i;:::-;31886:1;31881:3;31877:11;31870:18;;31494:400;;;:::o;31900:701::-;32181:3;32203:95;32294:3;32285:6;32203:95;:::i;:::-;32196:102;;32315:95;32406:3;32397:6;32315:95;:::i;:::-;32308:102;;32427:148;32571:3;32427:148;:::i;:::-;32420:155;;32592:3;32585:10;;31900:701;;;;;:::o;32607:225::-;32747:34;32743:1;32735:6;32731:14;32724:58;32816:8;32811:2;32803:6;32799:15;32792:33;32607:225;:::o;32838:366::-;32980:3;33001:67;33065:2;33060:3;33001:67;:::i;:::-;32994:74;;33077:93;33166:3;33077:93;:::i;:::-;33195:2;33190:3;33186:12;33179:19;;32838:366;;;:::o;33210:419::-;33376:4;33414:2;33403:9;33399:18;33391:26;;33463:9;33457:4;33453:20;33449:1;33438:9;33434:17;33427:47;33491:131;33617:4;33491:131;:::i;:::-;33483:139;;33210:419;;;:::o;33635:98::-;33686:6;33720:5;33714:12;33704:22;;33635:98;;;:::o;33739:168::-;33822:11;33856:6;33851:3;33844:19;33896:4;33891:3;33887:14;33872:29;;33739:168;;;;:::o;33913:360::-;33999:3;34027:38;34059:5;34027:38;:::i;:::-;34081:70;34144:6;34139:3;34081:70;:::i;:::-;34074:77;;34160:52;34205:6;34200:3;34193:4;34186:5;34182:16;34160:52;:::i;:::-;34237:29;34259:6;34237:29;:::i;:::-;34232:3;34228:39;34221:46;;34003:270;33913:360;;;;:::o;34279:640::-;34474:4;34512:3;34501:9;34497:19;34489:27;;34526:71;34594:1;34583:9;34579:17;34570:6;34526:71;:::i;:::-;34607:72;34675:2;34664:9;34660:18;34651:6;34607:72;:::i;:::-;34689;34757:2;34746:9;34742:18;34733:6;34689:72;:::i;:::-;34808:9;34802:4;34798:20;34793:2;34782:9;34778:18;34771:48;34836:76;34907:4;34898:6;34836:76;:::i;:::-;34828:84;;34279:640;;;;;;;:::o;34925:141::-;34981:5;35012:6;35006:13;34997:22;;35028:32;35054:5;35028:32;:::i;:::-;34925:141;;;;:::o;35072:349::-;35141:6;35190:2;35178:9;35169:7;35165:23;35161:32;35158:119;;;35196:79;;:::i;:::-;35158:119;35316:1;35341:63;35396:7;35387:6;35376:9;35372:22;35341:63;:::i;:::-;35331:73;;35287:127;35072:349;;;;:::o;35427:233::-;35466:3;35489:24;35507:5;35489:24;:::i;:::-;35480:33;;35535:66;35528:5;35525:77;35522:103;;35605:18;;:::i;:::-;35522:103;35652:1;35645:5;35641:13;35634:20;;35427:233;;;:::o;35666:180::-;35714:77;35711:1;35704:88;35811:4;35808:1;35801:15;35835:4;35832:1;35825:15;35852:185;35892:1;35909:20;35927:1;35909:20;:::i;:::-;35904:25;;35943:20;35961:1;35943:20;:::i;:::-;35938:25;;35982:1;35972:35;;35987:18;;:::i;:::-;35972:35;36029:1;36026;36022:9;36017:14;;35852:185;;;;:::o;36043:191::-;36083:4;36103:20;36121:1;36103:20;:::i;:::-;36098:25;;36137:20;36155:1;36137:20;:::i;:::-;36132:25;;36176:1;36173;36170:8;36167:34;;;36181:18;;:::i;:::-;36167:34;36226:1;36223;36219:9;36211:17;;36043:191;;;;:::o;36240:176::-;36272:1;36289:20;36307:1;36289:20;:::i;:::-;36284:25;;36323:20;36341:1;36323:20;:::i;:::-;36318:25;;36362:1;36352:35;;36367:18;;:::i;:::-;36352:35;36408:1;36405;36401:9;36396:14;;36240:176;;;;:::o;36422:180::-;36470:77;36467:1;36460:88;36567:4;36564:1;36557:15;36591:4;36588:1;36581:15
Swarm Source
ipfs://124521a7a189103aef94266d980080a72d870838b4e6e13d49ad84dec7f3c1b0
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.