ERC-721
Overview
Max Total Supply
50 FLURLABS
Holders
41
Market
Volume (24H)
N/A
Min Price (24H)
N/A
Max Price (24H)
N/A
Other Info
Token Contract
Balance
1 FLURLABSLoading...
Loading
Loading...
Loading
Loading...
Loading
# | Exchange | Pair | Price | 24H Volume | % Volume |
---|
Contract Source Code Verified (Exact Match)
Contract Name:
FLURLABS
Compiler Version
v0.8.17+commit.8df45f5f
Contract Source Code (Solidity)
/** *Submitted for verification at Etherscan.io on 2022-12-14 */ // SPDX-License-Identifier: MIT pragma solidity >=0.8.0 <0.9.0; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv( uint256 x, uint256 y, uint256 denominator ) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv( uint256 x, uint256 y, uint256 denominator, Rounding rounding ) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10**64) { value /= 10**64; result += 64; } if (value >= 10**32) { value /= 10**32; result += 32; } if (value >= 10**16) { value /= 10**16; result += 16; } if (value >= 10**8) { value /= 10**8; result += 8; } if (value >= 10**4) { value /= 10**4; result += 4; } if (value >= 10**2) { value /= 10**2; result += 2; } if (value >= 10**1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10**result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result * 8) < value ? 1 : 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); } /** * @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 functionCallWithValue(target, data, 0, "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"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or 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 { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } /** * @dev String operations. */ library Strings { bytes16 private constant _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) { unchecked { uint256 length = Math.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), _SYMBOLS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, Math.log256(value) + 1); } } /** * @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] = _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); } } /** * @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); } interface IOperatorFilterRegistry { function isOperatorAllowed(address registrant, address operator) external view returns (bool); function register(address registrant) external; function registerAndSubscribe(address registrant, address subscription) external; function registerAndCopyEntries(address registrant, address registrantToCopy) external; function unregister(address addr) external; function updateOperator(address registrant, address operator, bool filtered) external; function updateOperators(address registrant, address[] calldata operators, bool filtered) external; function updateCodeHash(address registrant, bytes32 codehash, bool filtered) external; function updateCodeHashes(address registrant, bytes32[] calldata codeHashes, bool filtered) external; function subscribe(address registrant, address registrantToSubscribe) external; function unsubscribe(address registrant, bool copyExistingEntries) external; function subscriptionOf(address addr) external returns (address registrant); function subscribers(address registrant) external returns (address[] memory); function subscriberAt(address registrant, uint256 index) external returns (address); function copyEntriesOf(address registrant, address registrantToCopy) external; function isOperatorFiltered(address registrant, address operator) external returns (bool); function isCodeHashOfFiltered(address registrant, address operatorWithCode) external returns (bool); function isCodeHashFiltered(address registrant, bytes32 codeHash) external returns (bool); function filteredOperators(address addr) external returns (address[] memory); function filteredCodeHashes(address addr) external returns (bytes32[] memory); function filteredOperatorAt(address registrant, uint256 index) external returns (address); function filteredCodeHashAt(address registrant, uint256 index) external returns (bytes32); function isRegistered(address addr) external returns (bool); function codeHashOf(address addr) external returns (bytes32); } /** * @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; } } /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } } /** * @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; } } /** * @dev Interface for the NFT Royalty Standard. * * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal * support for royalty payments across all NFT marketplaces and ecosystem participants. * * _Available since v4.5._ */ interface IERC2981 is IERC165 { /** * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of * exchange. The royalty amount is denominated and should be paid in that same unit of exchange. */ function royaltyInfo(uint256 tokenId, uint256 salePrice) external view returns (address receiver, uint256 royaltyAmount); } /** * @dev Interface of ERC721A. */ interface IERC721A { /** * The caller must own the token or be an approved operator. */ error ApprovalCallerNotOwnerNorApproved(); /** * The token does not exist. */ error ApprovalQueryForNonexistentToken(); /** * Cannot query the balance for the zero address. */ error BalanceQueryForZeroAddress(); /** * Cannot mint to the zero address. */ error MintToZeroAddress(); /** * The quantity of tokens minted must be more than zero. */ error MintZeroQuantity(); /** * The token does not exist. */ error OwnerQueryForNonexistentToken(); /** * The caller must own the token or be an approved operator. */ error TransferCallerNotOwnerNorApproved(); /** * The token must be owned by `from`. */ error TransferFromIncorrectOwner(); /** * Cannot safely transfer to a contract that does not implement the * ERC721Receiver interface. */ error TransferToNonERC721ReceiverImplementer(); /** * Cannot transfer to the zero address. */ error TransferToZeroAddress(); /** * The token does not exist. */ error URIQueryForNonexistentToken(); /** * The `quantity` minted with ERC2309 exceeds the safety limit. */ error MintERC2309QuantityExceedsLimit(); /** * The `extraData` cannot be set on an unintialized ownership slot. */ error OwnershipNotInitializedForExtraData(); // ============================================================= // STRUCTS // ============================================================= struct TokenOwnership { // The address of the owner. address addr; // Stores the start time of ownership with minimal overhead for tokenomics. uint64 startTimestamp; // Whether the token has been burned. bool burned; // Arbitrary data similar to `startTimestamp` that can be set via {_extraData}. uint24 extraData; } // ============================================================= // TOKEN COUNTERS // ============================================================= /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() external view returns (uint256); // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); // ============================================================= // IERC721 // ============================================================= /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables * (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`, * checking first that contract recipients are aware of the ERC721 protocol * to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be have been allowed to move * this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external payable; /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external payable; /** * @dev Transfers `tokenId` from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} * whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) external payable; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the * zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external payable; /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} * for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) external view returns (bool); // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() external view returns (string memory); /** * @dev Returns the token collection symbol. */ function symbol() external view returns (string memory); /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) external view returns (string memory); // ============================================================= // IERC2309 // ============================================================= /** * @dev Emitted when tokens in `fromTokenId` to `toTokenId` * (inclusive) is transferred from `from` to `to`, as defined in the * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309) standard. * * See {_mintERC2309} for more details. */ event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to); } /** * @dev Interface of ERC721ABurnable. */ interface IERC721ABurnable is IERC721A { /** * @dev Burns `tokenId`. See {ERC721A-_burn}. * * Requirements: * * - The caller must own `tokenId` or be an approved operator. */ function burn(uint256 tokenId) external; } /** * @dev Interface of ERC721AQueryable. */ interface IERC721AQueryable is IERC721A { /** * Invalid query range (`start` >= `stop`). */ error InvalidQueryRange(); /** * @dev Returns the `TokenOwnership` struct at `tokenId` without reverting. * * If the `tokenId` is out of bounds: * * - `addr = address(0)` * - `startTimestamp = 0` * - `burned = false` * - `extraData = 0` * * If the `tokenId` is burned: * * - `addr = <Address of owner before token was burned>` * - `startTimestamp = <Timestamp when token was burned>` * - `burned = true` * - `extraData = <Extra data when token was burned>` * * Otherwise: * * - `addr = <Address of owner>` * - `startTimestamp = <Timestamp of start of ownership>` * - `burned = false` * - `extraData = <Extra data at start of ownership>` */ function explicitOwnershipOf(uint256 tokenId) external view returns (TokenOwnership memory); /** * @dev Returns an array of `TokenOwnership` structs at `tokenIds` in order. * See {ERC721AQueryable-explicitOwnershipOf} */ function explicitOwnershipsOf(uint256[] memory tokenIds) external view returns (TokenOwnership[] memory); /** * @dev Returns an array of token IDs owned by `owner`, * in the range [`start`, `stop`) * (i.e. `start <= tokenId < stop`). * * This function allows for tokens to be queried if the collection * grows too big for a single call of {ERC721AQueryable-tokensOfOwner}. * * Requirements: * * - `start < stop` */ function tokensOfOwnerIn( address owner, uint256 start, uint256 stop ) external view returns (uint256[] memory); /** * @dev Returns an array of token IDs owned by `owner`. * * This function scans the ownership mapping and is O(`totalSupply`) in complexity. * It is meant to be called off-chain. * * See {ERC721AQueryable-tokensOfOwnerIn} for splitting the scan into * multiple smaller scans if the collection is large enough to cause * an out-of-gas error (10K collections should be fine). */ function tokensOfOwner(address owner) external view returns (uint256[] memory); } /** * @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: Note that the caller is responsible to confirm that the recipient is capable of receiving ERC721 * or else they may be permanently lost. Usage of {safeTransferFrom} prevents loss, though the caller must * understand this adds an external call which potentially creates a reentrancy vulnerability. * * 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); } /** * @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); } /** * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Enumerable is IERC721 { /** * @dev Returns the total amount of tokens stored by the contract. */ function totalSupply() external view returns (uint256); /** * @dev Returns a token ID owned by `owner` at a given `index` of its token list. * Use along with {balanceOf} to enumerate all of ``owner``'s tokens. */ function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256); /** * @dev Returns a token ID at a given `index` of all the tokens stored by the contract. * Use along with {totalSupply} to enumerate all tokens. */ function tokenByIndex(uint256 index) external view returns (uint256); } /** * @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 = _ownerOf(tokenId); require(owner != address(0), "ERC721: invalid token ID"); 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) { _requireMinted(tokenId); 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 token owner or approved for all" ); _approve(to, tokenId); } /** * @dev See {IERC721-getApproved}. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { _requireMinted(tokenId); 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: caller is not token owner or 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: caller is not token owner or 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 the owner of the `tokenId`. Does NOT revert if token doesn't exist */ function _ownerOf(uint256 tokenId) internal view virtual returns (address) { return _owners[tokenId]; } /** * @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 _ownerOf(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) { 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, 1); // Check that tokenId was not minted by `_beforeTokenTransfer` hook require(!_exists(tokenId), "ERC721: token already minted"); unchecked { // Will not overflow unless all 2**256 token ids are minted to the same owner. // Given that tokens are minted one by one, it is impossible in practice that // this ever happens. Might change if we allow batch minting. // The ERC fails to describe this case. _balances[to] += 1; } _owners[tokenId] = to; emit Transfer(address(0), to, tokenId); _afterTokenTransfer(address(0), to, tokenId, 1); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * This is an internal function that does not check if the sender is authorized to operate on the token. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId) internal virtual { address owner = ERC721.ownerOf(tokenId); _beforeTokenTransfer(owner, address(0), tokenId, 1); // Update ownership in case tokenId was transferred by `_beforeTokenTransfer` hook owner = ERC721.ownerOf(tokenId); // Clear approvals delete _tokenApprovals[tokenId]; unchecked { // Cannot overflow, as that would require more tokens to be burned/transferred // out than the owner initially received through minting and transferring in. _balances[owner] -= 1; } delete _owners[tokenId]; emit Transfer(owner, address(0), tokenId); _afterTokenTransfer(owner, address(0), tokenId, 1); } /** * @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, 1); // Check that tokenId was not transferred by `_beforeTokenTransfer` hook require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner"); // Clear approvals from the previous owner delete _tokenApprovals[tokenId]; unchecked { // `_balances[from]` cannot overflow for the same reason as described in `_burn`: // `from`'s balance is the number of token held, which is at least one before the current // transfer. // `_balances[to]` could overflow in the conditions described in `_mint`. That would require // all 2**256 token ids to be minted, which in practice is impossible. _balances[from] -= 1; _balances[to] += 1; } _owners[tokenId] = to; emit Transfer(from, to, tokenId); _afterTokenTransfer(from, to, tokenId, 1); } /** * @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 Reverts if the `tokenId` has not been minted yet. */ function _requireMinted(uint256 tokenId) internal view virtual { require(_exists(tokenId), "ERC721: invalid token ID"); } /** * @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 { /// @solidity memory-safe-assembly assembly { revert(add(32, reason), mload(reason)) } } } } else { return true; } } /** * @dev Hook that is called before any token transfer. This includes minting and burning. If {ERC721Consecutive} is * used, the hook may be called as part of a consecutive (batch) mint, as indicated by `batchSize` greater than 1. * * Calling conditions: * * - When `from` and `to` are both non-zero, ``from``'s tokens will be transferred to `to`. * - When `from` is zero, the tokens will be minted for `to`. * - When `to` is zero, ``from``'s tokens will be burned. * - `from` and `to` are never both zero. * - `batchSize` is non-zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _beforeTokenTransfer( address from, address to, uint256, /* firstTokenId */ uint256 batchSize ) internal virtual { if (batchSize > 1) { if (from != address(0)) { _balances[from] -= batchSize; } if (to != address(0)) { _balances[to] += batchSize; } } } /** * @dev Hook that is called after any token transfer. This includes minting and burning. If {ERC721Consecutive} is * used, the hook may be called as part of a consecutive (batch) mint, as indicated by `batchSize` greater than 1. * * Calling conditions: * * - When `from` and `to` are both non-zero, ``from``'s tokens were transferred to `to`. * - When `from` is zero, the tokens were minted for `to`. * - When `to` is zero, ``from``'s tokens were burned. * - `from` and `to` are never both zero. * - `batchSize` is non-zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _afterTokenTransfer( address from, address to, uint256 firstTokenId, uint256 batchSize ) internal virtual {} } /** * @dev Implementation of the NFT Royalty Standard, a standardized way to retrieve royalty payment information. * * Royalty information can be specified globally for all token ids via {_setDefaultRoyalty}, and/or individually for * specific token ids via {_setTokenRoyalty}. The latter takes precedence over the first. * * Royalty is specified as a fraction of sale price. {_feeDenominator} is overridable but defaults to 10000, meaning the * fee is specified in basis points by default. * * IMPORTANT: ERC-2981 only specifies a way to signal royalty information and does not enforce its payment. See * https://eips.ethereum.org/EIPS/eip-2981#optional-royalty-payments[Rationale] in the EIP. Marketplaces are expected to * voluntarily pay royalties together with sales, but note that this standard is not yet widely supported. * * _Available since v4.5._ */ abstract contract ERC2981 is IERC2981, ERC165 { struct RoyaltyInfo { address receiver; uint96 royaltyFraction; } RoyaltyInfo private _defaultRoyaltyInfo; mapping(uint256 => RoyaltyInfo) private _tokenRoyaltyInfo; /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC165) returns (bool) { return interfaceId == type(IERC2981).interfaceId || super.supportsInterface(interfaceId); } /** * @inheritdoc IERC2981 */ function royaltyInfo(uint256 _tokenId, uint256 _salePrice) public view virtual override returns (address, uint256) { RoyaltyInfo memory royalty = _tokenRoyaltyInfo[_tokenId]; if (royalty.receiver == address(0)) { royalty = _defaultRoyaltyInfo; } uint256 royaltyAmount = (_salePrice * royalty.royaltyFraction) / _feeDenominator(); return (royalty.receiver, royaltyAmount); } /** * @dev The denominator with which to interpret the fee set in {_setTokenRoyalty} and {_setDefaultRoyalty} as a * fraction of the sale price. Defaults to 10000 so fees are expressed in basis points, but may be customized by an * override. */ function _feeDenominator() internal pure virtual returns (uint96) { return 10000; } /** * @dev Sets the royalty information that all ids in this contract will default to. * * Requirements: * * - `receiver` cannot be the zero address. * - `feeNumerator` cannot be greater than the fee denominator. */ function _setDefaultRoyalty(address receiver, uint96 feeNumerator) internal virtual { require(feeNumerator <= _feeDenominator(), "ERC2981: royalty fee will exceed salePrice"); require(receiver != address(0), "ERC2981: invalid receiver"); _defaultRoyaltyInfo = RoyaltyInfo(receiver, feeNumerator); } /** * @dev Removes default royalty information. */ function _deleteDefaultRoyalty() internal virtual { delete _defaultRoyaltyInfo; } /** * @dev Sets the royalty information for a specific token id, overriding the global default. * * Requirements: * * - `receiver` cannot be the zero address. * - `feeNumerator` cannot be greater than the fee denominator. */ function _setTokenRoyalty( uint256 tokenId, address receiver, uint96 feeNumerator ) internal virtual { require(feeNumerator <= _feeDenominator(), "ERC2981: royalty fee will exceed salePrice"); require(receiver != address(0), "ERC2981: Invalid parameters"); _tokenRoyaltyInfo[tokenId] = RoyaltyInfo(receiver, feeNumerator); } /** * @dev Resets royalty information for the token id back to the global default. */ function _resetTokenRoyalty(uint256 tokenId) internal virtual { delete _tokenRoyaltyInfo[tokenId]; } } /** * @dev Interface of ERC721 token receiver. */ interface ERC721A__IERC721Receiver { function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } /** * @title ERC721A * * @dev Implementation of the [ERC721](https://eips.ethereum.org/EIPS/eip-721) * Non-Fungible Token Standard, including the Metadata extension. * Optimized for lower gas during batch mints. * * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...) * starting from `_startTokenId()`. * * Assumptions: * * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply. * - The maximum token ID cannot exceed 2**256 - 1 (max value of uint256). */ contract ERC721A is IERC721A { // Bypass for a `--via-ir` bug (https://github.com/chiru-labs/ERC721A/pull/364). struct TokenApprovalRef { address value; } // ============================================================= // CONSTANTS // ============================================================= // Mask of an entry in packed address data. uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1; // The bit position of `numberMinted` in packed address data. uint256 private constant _BITPOS_NUMBER_MINTED = 64; // The bit position of `numberBurned` in packed address data. uint256 private constant _BITPOS_NUMBER_BURNED = 128; // The bit position of `aux` in packed address data. uint256 private constant _BITPOS_AUX = 192; // Mask of all 256 bits in packed address data except the 64 bits for `aux`. uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1; // The bit position of `startTimestamp` in packed ownership. uint256 private constant _BITPOS_START_TIMESTAMP = 160; // The bit mask of the `burned` bit in packed ownership. uint256 private constant _BITMASK_BURNED = 1 << 224; // The bit position of the `nextInitialized` bit in packed ownership. uint256 private constant _BITPOS_NEXT_INITIALIZED = 225; // The bit mask of the `nextInitialized` bit in packed ownership. uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225; // The bit position of `extraData` in packed ownership. uint256 private constant _BITPOS_EXTRA_DATA = 232; // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`. uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1; // The mask of the lower 160 bits for addresses. uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1; // The maximum `quantity` that can be minted with {_mintERC2309}. // This limit is to prevent overflows on the address data entries. // For a limit of 5000, a total of 3.689e15 calls to {_mintERC2309} // is required to cause an overflow, which is unrealistic. uint256 private constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000; // The `Transfer` event signature is given by: // `keccak256(bytes("Transfer(address,address,uint256)"))`. bytes32 private constant _TRANSFER_EVENT_SIGNATURE = 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef; // ============================================================= // STORAGE // ============================================================= // The next token ID to be minted. uint256 private _currentIndex; // The number of tokens burned. uint256 private _burnCounter; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to ownership details // An empty struct value does not necessarily mean the token is unowned. // See {_packedOwnershipOf} implementation for details. // // Bits Layout: // - [0..159] `addr` // - [160..223] `startTimestamp` // - [224] `burned` // - [225] `nextInitialized` // - [232..255] `extraData` mapping(uint256 => uint256) private _packedOwnerships; // Mapping owner address to address data. // // Bits Layout: // - [0..63] `balance` // - [64..127] `numberMinted` // - [128..191] `numberBurned` // - [192..255] `aux` mapping(address => uint256) private _packedAddressData; // Mapping from token ID to approved address. mapping(uint256 => TokenApprovalRef) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; // ============================================================= // CONSTRUCTOR // ============================================================= constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; _currentIndex = _startTokenId(); } // ============================================================= // TOKEN COUNTING OPERATIONS // ============================================================= /** * @dev Returns the starting token ID. * To change the starting token ID, please override this function. */ function _startTokenId() internal view virtual returns (uint256) { return 0; } /** * @dev Returns the next token ID to be minted. */ function _nextTokenId() internal view virtual returns (uint256) { return _currentIndex; } /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() public view virtual override returns (uint256) { // Counter underflow is impossible as _burnCounter cannot be incremented // more than `_currentIndex - _startTokenId()` times. unchecked { return _currentIndex - _burnCounter - _startTokenId(); } } /** * @dev Returns the total amount of tokens minted in the contract. */ function _totalMinted() internal view virtual returns (uint256) { // Counter underflow is impossible as `_currentIndex` does not decrement, // and it is initialized to `_startTokenId()`. unchecked { return _currentIndex - _startTokenId(); } } /** * @dev Returns the total number of tokens burned. */ function _totalBurned() internal view virtual returns (uint256) { return _burnCounter; } // ============================================================= // ADDRESS DATA OPERATIONS // ============================================================= /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) public view virtual override returns (uint256) { if (owner == address(0)) revert BalanceQueryForZeroAddress(); return _packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens minted by `owner`. */ function _numberMinted(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens burned by or on behalf of `owner`. */ function _numberBurned(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> _BITPOS_NUMBER_BURNED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). */ function _getAux(address owner) internal view returns (uint64) { return uint64(_packedAddressData[owner] >> _BITPOS_AUX); } /** * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). * If there are multiple variables, please pack them into a uint64. */ function _setAux(address owner, uint64 aux) internal virtual { uint256 packed = _packedAddressData[owner]; uint256 auxCasted; // Cast `aux` with assembly to avoid redundant masking. assembly { auxCasted := aux } packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX); _packedAddressData[owner] = packed; } // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { // The interface IDs are constants representing the first 4 bytes // of the XOR of all function selectors in the interface. // See: [ERC165](https://eips.ethereum.org/EIPS/eip-165) // (e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`) return interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165. interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721. interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata. } // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev Returns the token collection symbol. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) revert URIQueryForNonexistentToken(); string memory baseURI = _baseURI(); return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, _toString(tokenId))) : ''; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, it can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ''; } // ============================================================= // OWNERSHIPS OPERATIONS // ============================================================= /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { return address(uint160(_packedOwnershipOf(tokenId))); } /** * @dev Gas spent here starts off proportional to the maximum mint batch size. * It gradually moves to O(1) as tokens get transferred around over time. */ function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnershipOf(tokenId)); } /** * @dev Returns the unpacked `TokenOwnership` struct at `index`. */ function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnerships[index]); } /** * @dev Initializes the ownership slot minted at `index` for efficiency purposes. */ function _initializeOwnershipAt(uint256 index) internal virtual { if (_packedOwnerships[index] == 0) { _packedOwnerships[index] = _packedOwnershipOf(index); } } /** * Returns the packed ownership data of `tokenId`. */ function _packedOwnershipOf(uint256 tokenId) private view returns (uint256) { uint256 curr = tokenId; unchecked { if (_startTokenId() <= curr) if (curr < _currentIndex) { uint256 packed = _packedOwnerships[curr]; // If not burned. if (packed & _BITMASK_BURNED == 0) { // Invariant: // There will always be an initialized ownership slot // (i.e. `ownership.addr != address(0) && ownership.burned == false`) // before an unintialized ownership slot // (i.e. `ownership.addr == address(0) && ownership.burned == false`) // Hence, `curr` will not underflow. // // We can directly compare the packed value. // If the address is zero, packed will be zero. while (packed == 0) { packed = _packedOwnerships[--curr]; } return packed; } } } revert OwnerQueryForNonexistentToken(); } /** * @dev Returns the unpacked `TokenOwnership` struct from `packed`. */ function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) { ownership.addr = address(uint160(packed)); ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP); ownership.burned = packed & _BITMASK_BURNED != 0; ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA); } /** * @dev Packs ownership data into a single uint256. */ function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // `owner | (block.timestamp << _BITPOS_START_TIMESTAMP) | flags`. result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags)) } } /** * @dev Returns the `nextInitialized` flag set if `quantity` equals 1. */ function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) { // For branchless setting of the `nextInitialized` flag. assembly { // `(quantity == 1) << _BITPOS_NEXT_INITIALIZED`. result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1)) } } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. See {ERC721A-_approve}. * * Requirements: * * - The caller must own the token or be an approved operator. */ function approve(address to, uint256 tokenId) public payable virtual override { _approve(to, tokenId, true); } /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { if (!_exists(tokenId)) revert ApprovalQueryForNonexistentToken(); return _tokenApprovals[tokenId].value; } /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} * for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool approved) public virtual override { _operatorApprovals[_msgSenderERC721A()][operator] = approved; emit ApprovalForAll(_msgSenderERC721A(), operator, approved); } /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev Returns whether `tokenId` exists. * * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}. * * Tokens start existing when they are minted. See {_mint}. */ function _exists(uint256 tokenId) internal view virtual returns (bool) { return _startTokenId() <= tokenId && tokenId < _currentIndex && // If within bounds, _packedOwnerships[tokenId] & _BITMASK_BURNED == 0; // and not burned. } /** * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`. */ function _isSenderApprovedOrOwner( address approvedAddress, address owner, address msgSender ) private pure returns (bool result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean. msgSender := and(msgSender, _BITMASK_ADDRESS) // `msgSender == owner || msgSender == approvedAddress`. result := or(eq(msgSender, owner), eq(msgSender, approvedAddress)) } } /** * @dev Returns the storage slot and value for the approved address of `tokenId`. */ function _getApprovedSlotAndAddress(uint256 tokenId) private view returns (uint256 approvedAddressSlot, address approvedAddress) { TokenApprovalRef storage tokenApproval = _tokenApprovals[tokenId]; // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId].value`. assembly { approvedAddressSlot := tokenApproval.slot approvedAddress := sload(approvedAddressSlot) } } // ============================================================= // TRANSFER OPERATIONS // ============================================================= /** * @dev Transfers `tokenId` from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) public payable virtual override { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); if (address(uint160(prevOwnershipPacked)) != from) revert TransferFromIncorrectOwner(); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved(); if (to == address(0)) revert TransferToZeroAddress(); _beforeTokenTransfers(from, to, tokenId, 1); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // We can directly increment and decrement the balances. --_packedAddressData[from]; // Updates: `balance -= 1`. ++_packedAddressData[to]; // Updates: `balance += 1`. // Updates: // - `address` to the next owner. // - `startTimestamp` to the timestamp of transfering. // - `burned` to `false`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _packOwnershipData( to, _BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } emit Transfer(from, to, tokenId); _afterTokenTransfers(from, to, tokenId, 1); } /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) public payable virtual override { safeTransferFrom(from, to, tokenId, ''); } /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token * by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes memory _data ) public payable virtual override { transferFrom(from, to, tokenId); if (to.code.length != 0) if (!_checkContractOnERC721Received(from, to, tokenId, _data)) { revert TransferToNonERC721ReceiverImplementer(); } } /** * @dev Hook that is called before a set of serially-ordered token IDs * are about to be transferred. This includes minting. * And also called before burning one token. * * `startTokenId` - the first token ID to be transferred. * `quantity` - the amount to be transferred. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _beforeTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Hook that is called after a set of serially-ordered token IDs * have been transferred. This includes minting. * And also called after one token has been burned. * * `startTokenId` - the first token ID to be transferred. * `quantity` - the amount to be transferred. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been * transferred to `to`. * - When `from` is zero, `tokenId` has been minted for `to`. * - When `to` is zero, `tokenId` has been burned by `from`. * - `from` and `to` are never both zero. */ function _afterTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Private function to invoke {IERC721Receiver-onERC721Received} on a target contract. * * `from` - Previous owner of the given token ID. * `to` - Target address that will receive the token. * `tokenId` - Token ID to be transferred. * `_data` - Optional data to send along with the call. * * Returns whether the call correctly returned the expected magic value. */ function _checkContractOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns ( bytes4 retval ) { return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert TransferToNonERC721ReceiverImplementer(); } else { assembly { revert(add(32, reason), mload(reason)) } } } } // ============================================================= // MINT OPERATIONS // ============================================================= /** * @dev Mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {Transfer} event for each mint. */ function _mint(address to, uint256 quantity) internal virtual { uint256 startTokenId = _currentIndex; if (quantity == 0) revert MintZeroQuantity(); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are incredibly unrealistic. // `balance` and `numberMinted` have a maximum limit of 2**64. // `tokenId` has a maximum limit of 2**256. unchecked { // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); uint256 toMasked; uint256 end = startTokenId + quantity; // Use assembly to loop and emit the `Transfer` event for gas savings. // The duplicated `log4` removes an extra check and reduces stack juggling. // The assembly, together with the surrounding Solidity code, have been // delicately arranged to nudge the compiler into producing optimized opcodes. assembly { // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. toMasked := and(to, _BITMASK_ADDRESS) // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. 0, // `address(0)`. toMasked, // `to`. startTokenId // `tokenId`. ) // The `iszero(eq(,))` check ensures that large values of `quantity` // that overflows uint256 will make the loop run out of gas. // The compiler will optimize the `iszero` away for performance. for { let tokenId := add(startTokenId, 1) } iszero(eq(tokenId, end)) { tokenId := add(tokenId, 1) } { // Emit the `Transfer` event. Similar to above. log4(0, 0, _TRANSFER_EVENT_SIGNATURE, 0, toMasked, tokenId) } } if (toMasked == 0) revert MintToZeroAddress(); _currentIndex = end; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Mints `quantity` tokens and transfers them to `to`. * * This function is intended for efficient minting only during contract creation. * * It emits only one {ConsecutiveTransfer} as defined in * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309), * instead of a sequence of {Transfer} event(s). * * Calling this function outside of contract creation WILL make your contract * non-compliant with the ERC721 standard. * For full ERC721 compliance, substituting ERC721 {Transfer} event(s) with the ERC2309 * {ConsecutiveTransfer} event is only permissible during contract creation. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {ConsecutiveTransfer} event. */ function _mintERC2309(address to, uint256 quantity) internal virtual { uint256 startTokenId = _currentIndex; if (to == address(0)) revert MintToZeroAddress(); if (quantity == 0) revert MintZeroQuantity(); if (quantity > _MAX_MINT_ERC2309_QUANTITY_LIMIT) revert MintERC2309QuantityExceedsLimit(); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are unrealistic due to the above check for `quantity` to be below the limit. unchecked { // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to); _currentIndex = startTokenId + quantity; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Safely mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called for each safe transfer. * - `quantity` must be greater than 0. * * See {_mint}. * * Emits a {Transfer} event for each mint. */ function _safeMint( address to, uint256 quantity, bytes memory _data ) internal virtual { _mint(to, quantity); unchecked { if (to.code.length != 0) { uint256 end = _currentIndex; uint256 index = end - quantity; do { if (!_checkContractOnERC721Received(address(0), to, index++, _data)) { revert TransferToNonERC721ReceiverImplementer(); } } while (index < end); // Reentrancy protection. if (_currentIndex != end) revert(); } } } /** * @dev Equivalent to `_safeMint(to, quantity, '')`. */ function _safeMint(address to, uint256 quantity) internal virtual { _safeMint(to, quantity, ''); } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Equivalent to `_approve(to, tokenId, false)`. */ function _approve(address to, uint256 tokenId) internal virtual { _approve(to, tokenId, false); } /** * @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: * * - `tokenId` must exist. * * Emits an {Approval} event. */ function _approve(address to, uint256 tokenId, bool approvalCheck) internal virtual { address owner = ownerOf(tokenId); if (approvalCheck && _msgSenderERC721A() != owner) if (!isApprovedForAll(owner, _msgSenderERC721A())) { revert ApprovalCallerNotOwnerNorApproved(); } _tokenApprovals[tokenId].value = to; emit Approval(owner, to, tokenId); } // ============================================================= // BURN OPERATIONS // ============================================================= /** * @dev Equivalent to `_burn(tokenId, false)`. */ function _burn(uint256 tokenId) internal virtual { _burn(tokenId, false); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId, bool approvalCheck) internal virtual { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); address from = address(uint160(prevOwnershipPacked)); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); if (approvalCheck) { // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) revert TransferCallerNotOwnerNorApproved(); } _beforeTokenTransfers(from, address(0), tokenId, 1); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // Updates: // - `balance -= 1`. // - `numberBurned += 1`. // // We can directly decrement the balance, and increment the number burned. // This is equivalent to `packed -= 1; packed += 1 << _BITPOS_NUMBER_BURNED;`. _packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1; // Updates: // - `address` to the last owner. // - `startTimestamp` to the timestamp of burning. // - `burned` to `true`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _packOwnershipData( from, (_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } emit Transfer(from, address(0), tokenId); _afterTokenTransfers(from, address(0), tokenId, 1); // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times. unchecked { _burnCounter++; } } // ============================================================= // EXTRA DATA OPERATIONS // ============================================================= /** * @dev Directly sets the extra data for the ownership data `index`. */ function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual { uint256 packed = _packedOwnerships[index]; if (packed == 0) revert OwnershipNotInitializedForExtraData(); uint256 extraDataCasted; // Cast `extraData` with assembly to avoid redundant masking. assembly { extraDataCasted := extraData } packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA); _packedOwnerships[index] = packed; } /** * @dev Called during each token transfer to set the 24bit `extraData` field. * Intended to be overridden by the cosumer contract. * * `previousExtraData` - the value of `extraData` before transfer. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _extraData( address from, address to, uint24 previousExtraData ) internal view virtual returns (uint24) {} /** * @dev Returns the next extra data for the packed ownership data. * The returned result is shifted into position. */ function _nextExtraData( address from, address to, uint256 prevOwnershipPacked ) private view returns (uint256) { uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA); return uint256(_extraData(from, to, extraData)) << _BITPOS_EXTRA_DATA; } // ============================================================= // OTHER OPERATIONS // ============================================================= /** * @dev Returns the message sender (defaults to `msg.sender`). * * If you are writing GSN compatible contracts, you need to override this function. */ function _msgSenderERC721A() internal view virtual returns (address) { return msg.sender; } /** * @dev Converts a uint256 to its ASCII string decimal representation. */ function _toString(uint256 value) internal pure virtual returns (string memory str) { assembly { // The maximum value of a uint256 contains 78 digits (1 byte per digit), but // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned. // We will need 1 word for the trailing zeros padding, 1 word for the length, // and 3 words for a maximum of 78 digits. Total: 5 * 0x20 = 0xa0. let m := add(mload(0x40), 0xa0) // Update the free memory pointer to allocate. mstore(0x40, m) // Assign the `str` to the end. str := sub(m, 0x20) // Zeroize the slot after the string. mstore(str, 0) // Cache the end of the memory to calculate the length later. let end := str // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. // prettier-ignore for { let temp := value } 1 {} { str := sub(str, 1) // Write the character to the pointer. // The ASCII index of the '0' character is 48. mstore8(str, add(48, mod(temp, 10))) // Keep dividing `temp` until zero. temp := div(temp, 10) // prettier-ignore if iszero(temp) { break } } let length := sub(end, str) // Move the pointer 32 bytes leftwards to make room for the length. str := sub(str, 0x20) // Store the length. mstore(str, length) } } } /** * @dev This implements an optional extension of {ERC721} defined in the EIP that adds * enumerability of all the token ids in the contract as well as all token ids owned by each * account. */ abstract contract ERC721Enumerable is ERC721, IERC721Enumerable { // Mapping from owner to list of owned token IDs mapping(address => mapping(uint256 => uint256)) private _ownedTokens; // Mapping from token ID to index of the owner tokens list mapping(uint256 => uint256) private _ownedTokensIndex; // Array with all token ids, used for enumeration uint256[] private _allTokens; // Mapping from token id to position in the allTokens array mapping(uint256 => uint256) private _allTokensIndex; /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC721) returns (bool) { return interfaceId == type(IERC721Enumerable).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}. */ function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) { require(index < ERC721.balanceOf(owner), "ERC721Enumerable: owner index out of bounds"); return _ownedTokens[owner][index]; } /** * @dev See {IERC721Enumerable-totalSupply}. */ function totalSupply() public view virtual override returns (uint256) { return _allTokens.length; } /** * @dev See {IERC721Enumerable-tokenByIndex}. */ function tokenByIndex(uint256 index) public view virtual override returns (uint256) { require(index < ERC721Enumerable.totalSupply(), "ERC721Enumerable: global index out of bounds"); return _allTokens[index]; } /** * @dev See {ERC721-_beforeTokenTransfer}. */ function _beforeTokenTransfer( address from, address to, uint256 firstTokenId, uint256 batchSize ) internal virtual override { super._beforeTokenTransfer(from, to, firstTokenId, batchSize); if (batchSize > 1) { // Will only trigger during construction. Batch transferring (minting) is not available afterwards. revert("ERC721Enumerable: consecutive transfers not supported"); } uint256 tokenId = firstTokenId; if (from == address(0)) { _addTokenToAllTokensEnumeration(tokenId); } else if (from != to) { _removeTokenFromOwnerEnumeration(from, tokenId); } if (to == address(0)) { _removeTokenFromAllTokensEnumeration(tokenId); } else if (to != from) { _addTokenToOwnerEnumeration(to, tokenId); } } /** * @dev Private function to add a token to this extension's ownership-tracking data structures. * @param to address representing the new owner of the given token ID * @param tokenId uint256 ID of the token to be added to the tokens list of the given address */ function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private { uint256 length = ERC721.balanceOf(to); _ownedTokens[to][length] = tokenId; _ownedTokensIndex[tokenId] = length; } /** * @dev Private function to add a token to this extension's token tracking data structures. * @param tokenId uint256 ID of the token to be added to the tokens list */ function _addTokenToAllTokensEnumeration(uint256 tokenId) private { _allTokensIndex[tokenId] = _allTokens.length; _allTokens.push(tokenId); } /** * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for * gas optimizations e.g. when performing a transfer operation (avoiding double writes). * This has O(1) time complexity, but alters the order of the _ownedTokens array. * @param from address representing the previous owner of the given token ID * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address */ function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private { // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and // then delete the last slot (swap and pop). uint256 lastTokenIndex = ERC721.balanceOf(from) - 1; uint256 tokenIndex = _ownedTokensIndex[tokenId]; // When the token to delete is the last token, the swap operation is unnecessary if (tokenIndex != lastTokenIndex) { uint256 lastTokenId = _ownedTokens[from][lastTokenIndex]; _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index } // This also deletes the contents at the last position of the array delete _ownedTokensIndex[tokenId]; delete _ownedTokens[from][lastTokenIndex]; } /** * @dev Private function to remove a token from this extension's token tracking data structures. * This has O(1) time complexity, but alters the order of the _allTokens array. * @param tokenId uint256 ID of the token to be removed from the tokens list */ function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private { // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and // then delete the last slot (swap and pop). uint256 lastTokenIndex = _allTokens.length - 1; uint256 tokenIndex = _allTokensIndex[tokenId]; // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding // an 'if' statement (like in _removeTokenFromOwnerEnumeration) uint256 lastTokenId = _allTokens[lastTokenIndex]; _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index // This also deletes the contents at the last position of the array delete _allTokensIndex[tokenId]; _allTokens.pop(); } } /** * @title ERC721AQueryable. * * @dev ERC721A subclass with convenience query functions. */ abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { /** * @dev Returns the `TokenOwnership` struct at `tokenId` without reverting. * * If the `tokenId` is out of bounds: * * - `addr = address(0)` * - `startTimestamp = 0` * - `burned = false` * - `extraData = 0` * * If the `tokenId` is burned: * * - `addr = <Address of owner before token was burned>` * - `startTimestamp = <Timestamp when token was burned>` * - `burned = true` * - `extraData = <Extra data when token was burned>` * * Otherwise: * * - `addr = <Address of owner>` * - `startTimestamp = <Timestamp of start of ownership>` * - `burned = false` * - `extraData = <Extra data at start of ownership>` */ function explicitOwnershipOf(uint256 tokenId) public view virtual override returns (TokenOwnership memory) { TokenOwnership memory ownership; if (tokenId < _startTokenId() || tokenId >= _nextTokenId()) { return ownership; } ownership = _ownershipAt(tokenId); if (ownership.burned) { return ownership; } return _ownershipOf(tokenId); } /** * @dev Returns an array of `TokenOwnership` structs at `tokenIds` in order. * See {ERC721AQueryable-explicitOwnershipOf} */ function explicitOwnershipsOf(uint256[] calldata tokenIds) external view virtual override returns (TokenOwnership[] memory) { unchecked { uint256 tokenIdsLength = tokenIds.length; TokenOwnership[] memory ownerships = new TokenOwnership[](tokenIdsLength); for (uint256 i; i != tokenIdsLength; ++i) { ownerships[i] = explicitOwnershipOf(tokenIds[i]); } return ownerships; } } /** * @dev Returns an array of token IDs owned by `owner`, * in the range [`start`, `stop`) * (i.e. `start <= tokenId < stop`). * * This function allows for tokens to be queried if the collection * grows too big for a single call of {ERC721AQueryable-tokensOfOwner}. * * Requirements: * * - `start < stop` */ function tokensOfOwnerIn( address owner, uint256 start, uint256 stop ) external view virtual override returns (uint256[] memory) { unchecked { if (start >= stop) revert InvalidQueryRange(); uint256 tokenIdsIdx; uint256 stopLimit = _nextTokenId(); // Set `start = max(start, _startTokenId())`. if (start < _startTokenId()) { start = _startTokenId(); } // Set `stop = min(stop, stopLimit)`. if (stop > stopLimit) { stop = stopLimit; } uint256 tokenIdsMaxLength = balanceOf(owner); // Set `tokenIdsMaxLength = min(balanceOf(owner), stop - start)`, // to cater for cases where `balanceOf(owner)` is too big. if (start < stop) { uint256 rangeLength = stop - start; if (rangeLength < tokenIdsMaxLength) { tokenIdsMaxLength = rangeLength; } } else { tokenIdsMaxLength = 0; } uint256[] memory tokenIds = new uint256[](tokenIdsMaxLength); if (tokenIdsMaxLength == 0) { return tokenIds; } // We need to call `explicitOwnershipOf(start)`, // because the slot at `start` may not be initialized. TokenOwnership memory ownership = explicitOwnershipOf(start); address currOwnershipAddr; // If the starting slot exists (i.e. not burned), initialize `currOwnershipAddr`. // `ownership.address` will not be zero, as `start` is clamped to the valid token ID range. if (!ownership.burned) { currOwnershipAddr = ownership.addr; } for (uint256 i = start; i != stop && tokenIdsIdx != tokenIdsMaxLength; ++i) { ownership = _ownershipAt(i); if (ownership.burned) { continue; } if (ownership.addr != address(0)) { currOwnershipAddr = ownership.addr; } if (currOwnershipAddr == owner) { tokenIds[tokenIdsIdx++] = i; } } // Downsize the array to fit. assembly { mstore(tokenIds, tokenIdsIdx) } return tokenIds; } } /** * @dev Returns an array of token IDs owned by `owner`. * * This function scans the ownership mapping and is O(`totalSupply`) in complexity. * It is meant to be called off-chain. * * See {ERC721AQueryable-tokensOfOwnerIn} for splitting the scan into * multiple smaller scans if the collection is large enough to cause * an out-of-gas error (10K collections should be fine). */ function tokensOfOwner(address owner) external view virtual override returns (uint256[] memory) { unchecked { uint256 tokenIdsIdx; address currOwnershipAddr; uint256 tokenIdsLength = balanceOf(owner); uint256[] memory tokenIds = new uint256[](tokenIdsLength); TokenOwnership memory ownership; for (uint256 i = _startTokenId(); tokenIdsIdx != tokenIdsLength; ++i) { ownership = _ownershipAt(i); if (ownership.burned) { continue; } if (ownership.addr != address(0)) { currOwnershipAddr = ownership.addr; } if (currOwnershipAddr == owner) { tokenIds[tokenIdsIdx++] = i; } } return tokenIds; } } } /** * @title ERC721ABurnable. * * @dev ERC721A token that can be irreversibly burned (destroyed). */ abstract contract ERC721ABurnable is ERC721A, IERC721ABurnable { /** * @dev Burns `tokenId`. See {ERC721A-_burn}. * * Requirements: * * - The caller must own `tokenId` or be an approved operator. */ function burn(uint256 tokenId) public virtual override { _burn(tokenId, true); } } abstract contract OperatorFilterer { error OperatorNotAllowed(address operator); IOperatorFilterRegistry constant OPERATOR_FILTER_REGISTRY = IOperatorFilterRegistry(0x000000000000AAeB6D7670E522A718067333cd4E); constructor(address subscriptionOrRegistrantToCopy, bool subscribe) { // If an inheriting token contract is deployed to a network without the registry deployed, the modifier // will not revert, but the contract will need to be registered with the registry once it is deployed in // order for the modifier to filter addresses. if (address(OPERATOR_FILTER_REGISTRY).code.length > 0) { if (subscribe) { OPERATOR_FILTER_REGISTRY.registerAndSubscribe(address(this), subscriptionOrRegistrantToCopy); } else { if (subscriptionOrRegistrantToCopy != address(0)) { OPERATOR_FILTER_REGISTRY.registerAndCopyEntries(address(this), subscriptionOrRegistrantToCopy); } else { OPERATOR_FILTER_REGISTRY.register(address(this)); } } } } modifier onlyAllowedOperator(address from) virtual { // Check registry code length to facilitate testing in environments without a deployed registry. if (address(OPERATOR_FILTER_REGISTRY).code.length > 0) { // Allow spending tokens from addresses with balance // Note that this still allows listings and marketplaces with escrow to transfer tokens if transferred // from an EOA. if (from == msg.sender) { _; return; } if ( !( OPERATOR_FILTER_REGISTRY.isOperatorAllowed(address(this), msg.sender) && OPERATOR_FILTER_REGISTRY.isOperatorAllowed(address(this), from) ) ) { revert OperatorNotAllowed(msg.sender); } } _; } } abstract contract DefaultOperatorFilterer is OperatorFilterer { address constant DEFAULT_SUBSCRIPTION = address(0x3cc6CddA760b79bAfa08dF41ECFA224f810dCeB6); constructor() OperatorFilterer(DEFAULT_SUBSCRIPTION, true) {} } contract FLURLABS is ERC721A, ERC721ABurnable, ERC721AQueryable, Ownable, ERC2981, DefaultOperatorFilterer { using Strings for uint256; uint256 constant maxSupply = 50; uint256 constant mintPrice = 1 ether; uint256 public maxPerAddress = 1; uint256 public maxPerTx = 1; string public baseURI; string public baseExtension = ".json"; string public notRevealedUri; bool public revealed = true; bool public letsgolive = false; mapping(address => uint256) public flurAlphaWhale; constructor() ERC721A("Flur Labs", "FLURLABS") { } function goLive(bool _letsfngo) external onlyOwner { letsgolive = _letsfngo; } function setFlurAlphaWhales(address[] memory wallets, uint256[] memory amounts) external onlyOwner { for(uint256 i = 0; i < wallets.length; i++){ flurAlphaWhale[wallets[i]] = amounts[i]; } } function checkNumberMinted(address _wallet) public view returns (uint256) { return _numberMinted(_wallet); } function checkTotalAvailableToMint(address _wallet) public view returns (uint256) { return flurAlphaWhale[_wallet]; } function mint(uint256 _quantity) external payable { require(letsgolive, "FlurLabs: Mint Not Active"); require(flurAlphaWhale[msg.sender] > 0, "You are NOT a whale"); require(_quantity <= flurAlphaWhale[msg.sender], "FlurLabs: Trying to mint more than Allowed"); require(totalSupply() + _quantity <= maxSupply, "FlurLabs: Mint Supply Exceeded"); require(_numberMinted(msg.sender) + _quantity <= flurAlphaWhale[msg.sender], "FlurLabs: Exceeds Max For Your Whale Wallet"); require(mintPrice * _quantity <= msg.value, "Not enough ETH sent for selected amount"); _safeMint(msg.sender, _quantity); } function reserve(address _address, uint256 _quantity) external onlyOwner { require(totalSupply() + _quantity <= maxSupply, "FlurLabs: Mint Supply Exceeded"); _safeMint(_address, _quantity); } function tokenURI(uint256 tokenId) public view virtual override(ERC721A, IERC721A) returns (string memory) { require( _exists(tokenId), "ERC721Metadata: URI query for nonexistent token" ); if(revealed == false) { return notRevealedUri; } string memory currentBaseURI = _baseURI(); return bytes(currentBaseURI).length > 0 ? string(abi.encodePacked(currentBaseURI, tokenId.toString(), baseExtension)) : ""; } function supportsInterface( bytes4 interfaceId ) public view override(ERC721A, ERC2981, IERC721A) returns (bool) { return ERC2981.supportsInterface(interfaceId) || ERC721A.supportsInterface(interfaceId); } function _startTokenId() internal view virtual override returns (uint256) { return 1; } function setNotRevealedURI(string memory _notRevealedURI) public onlyOwner { notRevealedUri = _notRevealedURI; } function setBaseURI(string memory _newBaseURI) public onlyOwner { baseURI = _newBaseURI; } function setBaseExtension(string memory _newBaseExtension) public onlyOwner { baseExtension = _newBaseExtension; } function _baseURI() internal view virtual override returns (string memory) { return baseURI; } function reveal() public onlyOwner { revealed = true; } function setDefaultRoyalty( address _receiver, uint96 _feeNumerator ) external onlyOwner { _setDefaultRoyalty(_receiver, _feeNumerator); } function deleteDefaultRoyalty() external onlyOwner { _deleteDefaultRoyalty(); } function setTokenRoyalty( uint256 _tokenId, address _receiver, uint96 _feeNumerator ) external onlyOwner { _setTokenRoyalty(_tokenId, _receiver, _feeNumerator); } function resetTokenRoyalty( uint256 tokenId ) external onlyOwner { _resetTokenRoyalty(tokenId); } /* ------------ OpenSea Overrides --------------*/ function transferFrom( address _from, address _to, uint256 _tokenId ) public payable override(ERC721A, IERC721A) onlyAllowedOperator(_from) { super.transferFrom(_from, _to, _tokenId); } function safeTransferFrom( address _from, address _to, uint256 _tokenId ) public payable override(ERC721A, IERC721A) onlyAllowedOperator(_from) { super.safeTransferFrom(_from, _to, _tokenId); } function safeTransferFrom( address _from, address _to, uint256 _tokenId, bytes memory _data ) public payable override(ERC721A, IERC721A) onlyAllowedOperator(_from) { super.safeTransferFrom(_from, _to, _tokenId, _data); } function withdrawMoney() external onlyOwner { (bool success, ) = msg.sender.call{value: address(this).balance}(""); require(success, "Withdraw 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":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"InvalidQueryRange","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[{"internalType":"address","name":"operator","type":"address"}],"name":"OperatorNotAllowed","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"fromTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"toTokenId","type":"uint256"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"ConsecutiveTransfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseExtension","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"checkNumberMinted","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"checkTotalAvailableToMint","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"deleteDefaultRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"explicitOwnershipOf","outputs":[{"components":[{"internalType":"address","name":"addr","type":"address"},{"internalType":"uint64","name":"startTimestamp","type":"uint64"},{"internalType":"bool","name":"burned","type":"bool"},{"internalType":"uint24","name":"extraData","type":"uint24"}],"internalType":"struct IERC721A.TokenOwnership","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"}],"name":"explicitOwnershipsOf","outputs":[{"components":[{"internalType":"address","name":"addr","type":"address"},{"internalType":"uint64","name":"startTimestamp","type":"uint64"},{"internalType":"bool","name":"burned","type":"bool"},{"internalType":"uint24","name":"extraData","type":"uint24"}],"internalType":"struct IERC721A.TokenOwnership[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"flurAlphaWhale","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bool","name":"_letsfngo","type":"bool"}],"name":"goLive","outputs":[],"stateMutability":"nonpayable","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":"letsgolive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxPerAddress","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxPerTx","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_quantity","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"notRevealedUri","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"},{"internalType":"uint256","name":"_quantity","type":"uint256"}],"name":"reserve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"resetTokenRoyalty","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":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"","type":"address"},{"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":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_newBaseExtension","type":"string"}],"name":"setBaseExtension","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_newBaseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_receiver","type":"address"},{"internalType":"uint96","name":"_feeNumerator","type":"uint96"}],"name":"setDefaultRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"wallets","type":"address[]"},{"internalType":"uint256[]","name":"amounts","type":"uint256[]"}],"name":"setFlurAlphaWhales","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_notRevealedURI","type":"string"}],"name":"setNotRevealedURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"address","name":"_receiver","type":"address"},{"internalType":"uint96","name":"_feeNumerator","type":"uint96"}],"name":"setTokenRoyalty","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":[{"internalType":"address","name":"owner","type":"address"}],"name":"tokensOfOwner","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"stop","type":"uint256"}],"name":"tokensOfOwnerIn","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawMoney","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
60806040526001600b556001600c556040518060400160405280600581526020017f2e6a736f6e000000000000000000000000000000000000000000000000000000815250600e9081620000549190620006c6565b506001601060006101000a81548160ff0219169083151502179055506000601060016101000a81548160ff0219169083151502179055503480156200009857600080fd5b50733cc6cdda760b79bafa08df41ecfa224f810dceb660016040518060400160405280600981526020017f466c7572204c61627300000000000000000000000000000000000000000000008152506040518060400160405280600881526020017f464c55524c41425300000000000000000000000000000000000000000000000081525081600290816200012d9190620006c6565b5080600390816200013f9190620006c6565b50620001506200037560201b60201c565b6000819055505050620001786200016c6200037e60201b60201c565b6200038660201b60201c565b60006daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff163b11156200036d57801562000233576daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff16637d3e3dbe30846040518363ffffffff1660e01b8152600401620001f9929190620007f2565b600060405180830381600087803b1580156200021457600080fd5b505af115801562000229573d6000803e3d6000fd5b505050506200036c565b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1614620002ed576daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663a0af290330846040518363ffffffff1660e01b8152600401620002b3929190620007f2565b600060405180830381600087803b158015620002ce57600080fd5b505af1158015620002e3573d6000803e3d6000fd5b505050506200036b565b6daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff16634420e486306040518263ffffffff1660e01b81526004016200033691906200081f565b600060405180830381600087803b1580156200035157600080fd5b505af115801562000366573d6000803e3d6000fd5b505050505b5b5b50506200083c565b60006001905090565b600033905090565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b600081519050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680620004ce57607f821691505b602082108103620004e457620004e362000486565b5b50919050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b6000600883026200054e7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff826200050f565b6200055a86836200050f565b95508019841693508086168417925050509392505050565b6000819050919050565b6000819050919050565b6000620005a7620005a16200059b8462000572565b6200057c565b62000572565b9050919050565b6000819050919050565b620005c38362000586565b620005db620005d282620005ae565b8484546200051c565b825550505050565b600090565b620005f2620005e3565b620005ff818484620005b8565b505050565b5b8181101562000627576200061b600082620005e8565b60018101905062000605565b5050565b601f82111562000676576200064081620004ea565b6200064b84620004ff565b810160208510156200065b578190505b620006736200066a85620004ff565b83018262000604565b50505b505050565b600082821c905092915050565b60006200069b600019846008026200067b565b1980831691505092915050565b6000620006b6838362000688565b9150826002028217905092915050565b620006d1826200044c565b67ffffffffffffffff811115620006ed57620006ec62000457565b5b620006f98254620004b5565b620007068282856200062b565b600060209050601f8311600181146200073e576000841562000729578287015190505b620007358582620006a8565b865550620007a5565b601f1984166200074e86620004ea565b60005b82811015620007785784890151825560018201915060208501945060208101905062000751565b8683101562000798578489015162000794601f89168262000688565b8355505b6001600288020188555050505b505050505050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000620007da82620007ad565b9050919050565b620007ec81620007cd565b82525050565b6000604082019050620008096000830185620007e1565b620008186020830184620007e1565b9392505050565b6000602082019050620008366000830184620007e1565b92915050565b615862806200084c6000396000f3fe60806040526004361061027d5760003560e01c806370a082311161014f578063aa1b103f116100c1578063cc47a40b1161007a578063cc47a40b1461097a578063da3ef23f146109a3578063e985e9c5146109cc578063f2c4ce1e14610a09578063f2fde38b14610a32578063f968adbe14610a5b5761027d565b8063aa1b103f1461088b578063ac446002146108a2578063b88d4fde146108b9578063c23dc68f146108d5578063c668286214610912578063c87b56dd1461093d5761027d565b80639434654b116101135780639434654b1461079e57806395d89b41146107c757806399a2557a146107f2578063a0712d681461082f578063a22cb4651461084b578063a475b5dd146108745761027d565b806370a08231146106b9578063715018a6146106f65780638462151c1461070d5780638a616bc01461074a5780638da5cb5b146107735761027d565b80632a55205a116101f35780635944c753116101ac5780635944c753146105835780635bbb2177146105ac5780636352211e146105e9578063639814e0146106265780636c0360eb14610651578063703cc38a1461067c5761027d565b80632a55205a1461048157806342842e0e146104bf57806342966c68146104db5780634725508e14610504578063518302271461052f57806355f804b31461055a5761027d565b8063095ea7b311610245578063095ea7b31461037b5780630cf845021461039757806315417bd7146103d457806318160ddd146104115780631bac14ce1461043c57806323b872dd146104655761027d565b806301ffc9a71461028257806304634d8d146102bf57806306fdde03146102e8578063081812fc14610313578063081c8c4414610350575b600080fd5b34801561028e57600080fd5b506102a960048036038101906102a49190613d76565b610a86565b6040516102b69190613dbe565b60405180910390f35b3480156102cb57600080fd5b506102e660048036038101906102e19190613e7b565b610aa8565b005b3480156102f457600080fd5b506102fd610abe565b60405161030a9190613f4b565b60405180910390f35b34801561031f57600080fd5b5061033a60048036038101906103359190613fa3565b610b50565b6040516103479190613fdf565b60405180910390f35b34801561035c57600080fd5b50610365610bcf565b6040516103729190613f4b565b60405180910390f35b61039560048036038101906103909190613ffa565b610c5d565b005b3480156103a357600080fd5b506103be60048036038101906103b9919061403a565b610c6d565b6040516103cb9190614076565b60405180910390f35b3480156103e057600080fd5b506103fb60048036038101906103f6919061403a565b610c85565b6040516104089190614076565b60405180910390f35b34801561041d57600080fd5b50610426610c97565b6040516104339190614076565b60405180910390f35b34801561044857600080fd5b50610463600480360381019061045e919061429c565b610cae565b005b61047f600480360381019061047a9190614314565b610d52565b005b34801561048d57600080fd5b506104a860048036038101906104a39190614367565b610f34565b6040516104b69291906143a7565b60405180910390f35b6104d960048036038101906104d49190614314565b61111e565b005b3480156104e757600080fd5b5061050260048036038101906104fd9190613fa3565b611300565b005b34801561051057600080fd5b5061051961130e565b6040516105269190613dbe565b60405180910390f35b34801561053b57600080fd5b50610544611321565b6040516105519190613dbe565b60405180910390f35b34801561056657600080fd5b50610581600480360381019061057c9190614485565b611334565b005b34801561058f57600080fd5b506105aa60048036038101906105a591906144ce565b61134f565b005b3480156105b857600080fd5b506105d360048036038101906105ce919061457c565b611367565b6040516105e0919061472c565b60405180910390f35b3480156105f557600080fd5b50610610600480360381019061060b9190613fa3565b61142a565b60405161061d9190613fdf565b60405180910390f35b34801561063257600080fd5b5061063b61143c565b6040516106489190614076565b60405180910390f35b34801561065d57600080fd5b50610666611442565b6040516106739190613f4b565b60405180910390f35b34801561068857600080fd5b506106a3600480360381019061069e919061403a565b6114d0565b6040516106b09190614076565b60405180910390f35b3480156106c557600080fd5b506106e060048036038101906106db919061403a565b611519565b6040516106ed9190614076565b60405180910390f35b34801561070257600080fd5b5061070b6115d1565b005b34801561071957600080fd5b50610734600480360381019061072f919061403a565b6115e5565b604051610741919061480c565b60405180910390f35b34801561075657600080fd5b50610771600480360381019061076c9190613fa3565b611728565b005b34801561077f57600080fd5b5061078861173c565b6040516107959190613fdf565b60405180910390f35b3480156107aa57600080fd5b506107c560048036038101906107c0919061485a565b611766565b005b3480156107d357600080fd5b506107dc61178b565b6040516107e99190613f4b565b60405180910390f35b3480156107fe57600080fd5b5061081960048036038101906108149190614887565b61181d565b604051610826919061480c565b60405180910390f35b61084960048036038101906108449190613fa3565b611a29565b005b34801561085757600080fd5b50610872600480360381019061086d91906148da565b611cca565b005b34801561088057600080fd5b50610889611dd5565b005b34801561089757600080fd5b506108a0611dfa565b005b3480156108ae57600080fd5b506108b7611e0c565b005b6108d360048036038101906108ce91906149bb565b611ec3565b005b3480156108e157600080fd5b506108fc60048036038101906108f79190613fa3565b6120a8565b6040516109099190614a93565b60405180910390f35b34801561091e57600080fd5b50610927612112565b6040516109349190613f4b565b60405180910390f35b34801561094957600080fd5b50610964600480360381019061095f9190613fa3565b6121a0565b6040516109719190613f4b565b60405180910390f35b34801561098657600080fd5b506109a1600480360381019061099c9190613ffa565b6122f8565b005b3480156109af57600080fd5b506109ca60048036038101906109c59190614485565b612364565b005b3480156109d857600080fd5b506109f360048036038101906109ee9190614aae565b61237f565b604051610a009190613dbe565b60405180910390f35b348015610a1557600080fd5b50610a306004803603810190610a2b9190614485565b612413565b005b348015610a3e57600080fd5b50610a596004803603810190610a54919061403a565b61242e565b005b348015610a6757600080fd5b50610a706124b1565b604051610a7d9190614076565b60405180910390f35b6000610a91826124b7565b80610aa15750610aa082612531565b5b9050919050565b610ab06125c3565b610aba8282612641565b5050565b606060028054610acd90614b1d565b80601f0160208091040260200160405190810160405280929190818152602001828054610af990614b1d565b8015610b465780601f10610b1b57610100808354040283529160200191610b46565b820191906000526020600020905b815481529060010190602001808311610b2957829003601f168201915b5050505050905090565b6000610b5b826127d6565b610b91576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b600f8054610bdc90614b1d565b80601f0160208091040260200160405190810160405280929190818152602001828054610c0890614b1d565b8015610c555780601f10610c2a57610100808354040283529160200191610c55565b820191906000526020600020905b815481529060010190602001808311610c3857829003601f168201915b505050505081565b610c6982826001612835565b5050565b60116020528060005260406000206000915090505481565b6000610c9082612985565b9050919050565b6000610ca16129dc565b6001546000540303905090565b610cb66125c3565b60005b8251811015610d4d57818181518110610cd557610cd4614b4e565b5b602002602001015160116000858481518110610cf457610cf3614b4e565b5b602002602001015173ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055508080610d4590614bac565b915050610cb9565b505050565b8260006daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff163b1115610f22573373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603610dc457610dbf8484846129e5565b610f2e565b6daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430336040518363ffffffff1660e01b8152600401610e0d929190614bf4565b602060405180830381865afa158015610e2a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e4e9190614c32565b8015610ee057506daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430836040518363ffffffff1660e01b8152600401610e9e929190614bf4565b602060405180830381865afa158015610ebb573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610edf9190614c32565b5b610f2157336040517fede71dcc000000000000000000000000000000000000000000000000000000008152600401610f189190613fdf565b60405180910390fd5b5b610f2d8484846129e5565b5b50505050565b6000806000600a60008681526020019081526020016000206040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff16815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff16036110c95760096040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff168152505090505b60006110d3612d07565b6bffffffffffffffffffffffff1682602001516bffffffffffffffffffffffff16866110ff9190614c5f565b6111099190614cd0565b90508160000151819350935050509250929050565b8260006daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff163b11156112ee573373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036111905761118b848484612d11565b6112fa565b6daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430336040518363ffffffff1660e01b81526004016111d9929190614bf4565b602060405180830381865afa1580156111f6573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061121a9190614c32565b80156112ac57506daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430836040518363ffffffff1660e01b815260040161126a929190614bf4565b602060405180830381865afa158015611287573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112ab9190614c32565b5b6112ed57336040517fede71dcc0000000000000000000000000000000000000000000000000000000081526004016112e49190613fdf565b60405180910390fd5b5b6112f9848484612d11565b5b50505050565b61130b816001612d31565b50565b601060019054906101000a900460ff1681565b601060009054906101000a900460ff1681565b61133c6125c3565b80600d908161134b9190614ead565b5050565b6113576125c3565b611362838383612f83565b505050565b6060600083839050905060008167ffffffffffffffff81111561138d5761138c614096565b5b6040519080825280602002602001820160405280156113c657816020015b6113b3613cbb565b8152602001906001900390816113ab5790505b50905060005b82811461141e576113f58686838181106113e9576113e8614b4e565b5b905060200201356120a8565b82828151811061140857611407614b4e565b5b60200260200101819052508060010190506113cc565b50809250505092915050565b60006114358261312a565b9050919050565b600b5481565b600d805461144f90614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461147b90614b1d565b80156114c85780601f1061149d576101008083540402835291602001916114c8565b820191906000526020600020905b8154815290600101906020018083116114ab57829003601f168201915b505050505081565b6000601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020549050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611580576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b6115d96125c3565b6115e360006131f6565b565b606060008060006115f585611519565b905060008167ffffffffffffffff81111561161357611612614096565b5b6040519080825280602002602001820160405280156116415781602001602082028036833780820191505090505b50905061164c613cbb565b60006116566129dc565b90505b83861461171a57611669816132bc565b9150816040015161170f57600073ffffffffffffffffffffffffffffffffffffffff16826000015173ffffffffffffffffffffffffffffffffffffffff16146116b457816000015194505b8773ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff160361170e578083878060010198508151811061170157611700614b4e565b5b6020026020010181815250505b5b806001019050611659565b508195505050505050919050565b6117306125c3565b611739816132e7565b50565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b61176e6125c3565b80601060016101000a81548160ff02191690831515021790555050565b60606003805461179a90614b1d565b80601f01602080910402602001604051908101604052809291908181526020018280546117c690614b1d565b80156118135780601f106117e857610100808354040283529160200191611813565b820191906000526020600020905b8154815290600101906020018083116117f657829003601f168201915b5050505050905090565b6060818310611858576040517f32c1995a00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080611863613346565b905061186d6129dc565b85101561187f5761187c6129dc565b94505b8084111561188b578093505b600061189687611519565b9050848610156118b95760008686039050818110156118b3578091505b506118be565b600090505b60008167ffffffffffffffff8111156118da576118d9614096565b5b6040519080825280602002602001820160405280156119085781602001602082028036833780820191505090505b5090506000820361191f5780945050505050611a22565b600061192a886120a8565b90506000816040015161193f57816000015190505b60008990505b8881141580156119555750848714155b15611a1457611963816132bc565b92508260400151611a0957600073ffffffffffffffffffffffffffffffffffffffff16836000015173ffffffffffffffffffffffffffffffffffffffff16146119ae57826000015191505b8a73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611a0857808488806001019950815181106119fb576119fa614b4e565b5b6020026020010181815250505b5b806001019050611945565b508583528296505050505050505b9392505050565b601060019054906101000a900460ff16611a78576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a6f90614fcb565b60405180910390fd5b6000601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205411611afa576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611af190615037565b60405180910390fd5b601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054811115611b7c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611b73906150c9565b60405180910390fd5b603281611b87610c97565b611b9191906150e9565b1115611bd2576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611bc990615169565b60405180910390fd5b601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205481611c1c33612985565b611c2691906150e9565b1115611c67576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611c5e906151fb565b60405180910390fd5b3481670de0b6b3a7640000611c7c9190614c5f565b1115611cbd576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611cb49061528d565b60405180910390fd5b611cc7338261334f565b50565b8060076000611cd761336d565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611d8461336d565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c3183604051611dc99190613dbe565b60405180910390a35050565b611ddd6125c3565b6001601060006101000a81548160ff021916908315150217905550565b611e026125c3565b611e0a613375565b565b611e146125c3565b60003373ffffffffffffffffffffffffffffffffffffffff1647604051611e3a906152de565b60006040518083038185875af1925050503d8060008114611e77576040519150601f19603f3d011682016040523d82523d6000602084013e611e7c565b606091505b5050905080611ec0576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611eb79061533f565b60405180910390fd5b50565b8360006daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff163b1115612094573373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603611f3657611f31858585856133c2565b6120a1565b6daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430336040518363ffffffff1660e01b8152600401611f7f929190614bf4565b602060405180830381865afa158015611f9c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611fc09190614c32565b801561205257506daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430836040518363ffffffff1660e01b8152600401612010929190614bf4565b602060405180830381865afa15801561202d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120519190614c32565b5b61209357336040517fede71dcc00000000000000000000000000000000000000000000000000000000815260040161208a9190613fdf565b60405180910390fd5b5b6120a0858585856133c2565b5b5050505050565b6120b0613cbb565b6120b8613cbb565b6120c06129dc565b8310806120d457506120d0613346565b8310155b156120e2578091505061210d565b6120eb836132bc565b9050806040015115612100578091505061210d565b61210983613435565b9150505b919050565b600e805461211f90614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461214b90614b1d565b80156121985780601f1061216d57610100808354040283529160200191612198565b820191906000526020600020905b81548152906001019060200180831161217b57829003601f168201915b505050505081565b60606121ab826127d6565b6121ea576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016121e1906153d1565b60405180910390fd5b60001515601060009054906101000a900460ff1615150361229757600f805461221290614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461223e90614b1d565b801561228b5780601f106122605761010080835404028352916020019161228b565b820191906000526020600020905b81548152906001019060200180831161226e57829003601f168201915b505050505090506122f3565b60006122a1613455565b905060008151116122c157604051806020016040528060008152506122ef565b806122cb846134e7565b600e6040516020016122df939291906154b0565b6040516020818303038152906040525b9150505b919050565b6123006125c3565b60328161230b610c97565b61231591906150e9565b1115612356576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161234d90615169565b60405180910390fd5b612360828261334f565b5050565b61236c6125c3565b80600e908161237b9190614ead565b5050565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b61241b6125c3565b80600f908161242a9190614ead565b5050565b6124366125c3565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036124a5576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161249c90615553565b60405180910390fd5b6124ae816131f6565b50565b600c5481565b60007f2a55205a000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061252a5750612529826135b5565b5b9050919050565b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061258c57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b806125bc5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b6125cb61361f565b73ffffffffffffffffffffffffffffffffffffffff166125e961173c565b73ffffffffffffffffffffffffffffffffffffffff161461263f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612636906155bf565b60405180910390fd5b565b612649612d07565b6bffffffffffffffffffffffff16816bffffffffffffffffffffffff1611156126a7576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161269e90615651565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603612716576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161270d906156bd565b60405180910390fd5b60405180604001604052808373ffffffffffffffffffffffffffffffffffffffff168152602001826bffffffffffffffffffffffff16815250600960008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a8154816bffffffffffffffffffffffff02191690836bffffffffffffffffffffffff1602179055509050505050565b6000816127e16129dc565b111580156127f0575060005482105b801561282e575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b60006128408361142a565b905081801561288257508073ffffffffffffffffffffffffffffffffffffffff1661286961336d565b73ffffffffffffffffffffffffffffffffffffffff1614155b156128cf576128988161289361336d565b61237f565b6128ce576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b836006600085815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550828473ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a450505050565b600067ffffffffffffffff6040600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054901c169050919050565b60006001905090565b60006129f08261312a565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614612a57576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080612a6384613627565b91509150612a798187612a7461336d565b61364e565b612ac557612a8e86612a8961336d565b61237f565b612ac4576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1603612b2b576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612b388686866001613692565b8015612b4357600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000815460010191905081905550612c1185612bed888887613698565b7c0200000000000000000000000000000000000000000000000000000000176136c0565b600460008681526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000841603612c975760006001850190506000600460008381526020019081526020016000205403612c95576000548114612c94578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612cff86868660016136eb565b505050505050565b6000612710905090565b612d2c83838360405180602001604052806000815250611ec3565b505050565b6000612d3c8361312a565b90506000819050600080612d4f86613627565b915091508415612db857612d6b8184612d6661336d565b61364e565b612db757612d8083612d7b61336d565b61237f565b612db6576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b5b612dc6836000886001613692565b8015612dd157600082555b600160806001901b03600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550612e7983612e3685600088613698565b7c02000000000000000000000000000000000000000000000000000000007c010000000000000000000000000000000000000000000000000000000017176136c0565b600460008881526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000851603612eff5760006001870190506000600460008381526020019081526020016000205403612efd576000548114612efc578460046000838152602001908152602001600020819055505b5b505b85600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612f698360008860016136eb565b600160008154809291906001019190505550505050505050565b612f8b612d07565b6bffffffffffffffffffffffff16816bffffffffffffffffffffffff161115612fe9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612fe090615651565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603613058576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161304f90615729565b60405180910390fd5b60405180604001604052808373ffffffffffffffffffffffffffffffffffffffff168152602001826bffffffffffffffffffffffff16815250600a600085815260200190815260200160002060008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a8154816bffffffffffffffffffffffff02191690836bffffffffffffffffffffffff160217905550905050505050565b600080829050806131396129dc565b116131bf576000548110156131be5760006004600083815260200190815260200160002054905060007c01000000000000000000000000000000000000000000000000000000008216036131bc575b600081036131b2576004600083600190039350838152602001908152602001600020549050613188565b80925050506131f1565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b6132c4613cbb565b6132e060046000848152602001908152602001600020546136f1565b9050919050565b600a6000828152602001908152602001600020600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a8154906bffffffffffffffffffffffff0219169055505050565b60008054905090565b6133698282604051806020016040528060008152506137a7565b5050565b600033905090565b6009600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a8154906bffffffffffffffffffffffff02191690555050565b6133cd848484610d52565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461342f576133f884848484613844565b61342e576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b61343d613cbb565b61344e6134498361312a565b6136f1565b9050919050565b6060600d805461346490614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461349090614b1d565b80156134dd5780601f106134b2576101008083540402835291602001916134dd565b820191906000526020600020905b8154815290600101906020018083116134c057829003601f168201915b5050505050905090565b6060600060016134f684613994565b01905060008167ffffffffffffffff81111561351557613514614096565b5b6040519080825280601f01601f1916602001820160405280156135475781602001600182028036833780820191505090505b509050600082602001820190505b6001156135aa578080600190039150507f3031323334353637383961626364656600000000000000000000000000000000600a86061a8153600a858161359e5761359d614ca1565b5b04945060008503613555575b819350505050919050565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b600033905090565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b50505050565b60008060e883901c905060e86136af868684613ae7565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b6136f9613cbb565b81816000019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060a082901c816020019067ffffffffffffffff16908167ffffffffffffffff168152505060007c01000000000000000000000000000000000000000000000000000000008316141581604001901515908115158152505060e882901c816060019062ffffff16908162ffffff1681525050919050565b6137b18383613af0565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461383f57600080549050600083820390505b6137f16000868380600101945086613844565b613827576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8181106137de57816000541461383c57600080fd5b50505b505050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a0261386a61336d565b8786866040518563ffffffff1660e01b815260040161388c949392919061579e565b6020604051808303816000875af19250505080156138c857506040513d601f19601f820116820180604052508101906138c591906157ff565b60015b613941573d80600081146138f8576040519150601f19603f3d011682016040523d82523d6000602084013e6138fd565b606091505b506000815103613939576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b600080600090507a184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000083106139f2577a184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000083816139e8576139e7614ca1565b5b0492506040810190505b6d04ee2d6d415b85acef81000000008310613a2f576d04ee2d6d415b85acef81000000008381613a2557613a24614ca1565b5b0492506020810190505b662386f26fc100008310613a5e57662386f26fc100008381613a5457613a53614ca1565b5b0492506010810190505b6305f5e1008310613a87576305f5e1008381613a7d57613a7c614ca1565b5b0492506008810190505b6127108310613aac576127108381613aa257613aa1614ca1565b5b0492506004810190505b60648310613acf5760648381613ac557613ac4614ca1565b5b0492506002810190505b600a8310613ade576001810190505b80915050919050565b60009392505050565b60008054905060008203613b30576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b613b3d6000848385613692565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550613bb483613ba56000866000613698565b613bae85613cab565b176136c0565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b818114613c5557808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050613c1a565b5060008203613c90576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b806000819055505050613ca660008483856136eb565b505050565b60006001821460e11b9050919050565b6040518060800160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff168152602001600015158152602001600062ffffff1681525090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b613d5381613d1e565b8114613d5e57600080fd5b50565b600081359050613d7081613d4a565b92915050565b600060208284031215613d8c57613d8b613d14565b5b6000613d9a84828501613d61565b91505092915050565b60008115159050919050565b613db881613da3565b82525050565b6000602082019050613dd36000830184613daf565b92915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000613e0482613dd9565b9050919050565b613e1481613df9565b8114613e1f57600080fd5b50565b600081359050613e3181613e0b565b92915050565b60006bffffffffffffffffffffffff82169050919050565b613e5881613e37565b8114613e6357600080fd5b50565b600081359050613e7581613e4f565b92915050565b60008060408385031215613e9257613e91613d14565b5b6000613ea085828601613e22565b9250506020613eb185828601613e66565b9150509250929050565b600081519050919050565b600082825260208201905092915050565b60005b83811015613ef5578082015181840152602081019050613eda565b60008484015250505050565b6000601f19601f8301169050919050565b6000613f1d82613ebb565b613f278185613ec6565b9350613f37818560208601613ed7565b613f4081613f01565b840191505092915050565b60006020820190508181036000830152613f658184613f12565b905092915050565b6000819050919050565b613f8081613f6d565b8114613f8b57600080fd5b50565b600081359050613f9d81613f77565b92915050565b600060208284031215613fb957613fb8613d14565b5b6000613fc784828501613f8e565b91505092915050565b613fd981613df9565b82525050565b6000602082019050613ff46000830184613fd0565b92915050565b6000806040838503121561401157614010613d14565b5b600061401f85828601613e22565b925050602061403085828601613f8e565b9150509250929050565b6000602082840312156140505761404f613d14565b5b600061405e84828501613e22565b91505092915050565b61407081613f6d565b82525050565b600060208201905061408b6000830184614067565b92915050565b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6140ce82613f01565b810181811067ffffffffffffffff821117156140ed576140ec614096565b5b80604052505050565b6000614100613d0a565b905061410c82826140c5565b919050565b600067ffffffffffffffff82111561412c5761412b614096565b5b602082029050602081019050919050565b600080fd5b600061415561415084614111565b6140f6565b905080838252602082019050602084028301858111156141785761417761413d565b5b835b818110156141a1578061418d8882613e22565b84526020840193505060208101905061417a565b5050509392505050565b600082601f8301126141c0576141bf614091565b5b81356141d0848260208601614142565b91505092915050565b600067ffffffffffffffff8211156141f4576141f3614096565b5b602082029050602081019050919050565b6000614218614213846141d9565b6140f6565b9050808382526020820190506020840283018581111561423b5761423a61413d565b5b835b8181101561426457806142508882613f8e565b84526020840193505060208101905061423d565b5050509392505050565b600082601f83011261428357614282614091565b5b8135614293848260208601614205565b91505092915050565b600080604083850312156142b3576142b2613d14565b5b600083013567ffffffffffffffff8111156142d1576142d0613d19565b5b6142dd858286016141ab565b925050602083013567ffffffffffffffff8111156142fe576142fd613d19565b5b61430a8582860161426e565b9150509250929050565b60008060006060848603121561432d5761432c613d14565b5b600061433b86828701613e22565b935050602061434c86828701613e22565b925050604061435d86828701613f8e565b9150509250925092565b6000806040838503121561437e5761437d613d14565b5b600061438c85828601613f8e565b925050602061439d85828601613f8e565b9150509250929050565b60006040820190506143bc6000830185613fd0565b6143c96020830184614067565b9392505050565b600080fd5b600067ffffffffffffffff8211156143f0576143ef614096565b5b6143f982613f01565b9050602081019050919050565b82818337600083830152505050565b6000614428614423846143d5565b6140f6565b905082815260208101848484011115614444576144436143d0565b5b61444f848285614406565b509392505050565b600082601f83011261446c5761446b614091565b5b813561447c848260208601614415565b91505092915050565b60006020828403121561449b5761449a613d14565b5b600082013567ffffffffffffffff8111156144b9576144b8613d19565b5b6144c584828501614457565b91505092915050565b6000806000606084860312156144e7576144e6613d14565b5b60006144f586828701613f8e565b935050602061450686828701613e22565b925050604061451786828701613e66565b9150509250925092565b600080fd5b60008083601f84011261453c5761453b614091565b5b8235905067ffffffffffffffff81111561455957614558614521565b5b6020830191508360208202830111156145755761457461413d565b5b9250929050565b6000806020838503121561459357614592613d14565b5b600083013567ffffffffffffffff8111156145b1576145b0613d19565b5b6145bd85828601614526565b92509250509250929050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b6145fe81613df9565b82525050565b600067ffffffffffffffff82169050919050565b61462181614604565b82525050565b61463081613da3565b82525050565b600062ffffff82169050919050565b61464e81614636565b82525050565b60808201600082015161466a60008501826145f5565b50602082015161467d6020850182614618565b5060408201516146906040850182614627565b5060608201516146a36060850182614645565b50505050565b60006146b58383614654565b60808301905092915050565b6000602082019050919050565b60006146d9826145c9565b6146e381856145d4565b93506146ee836145e5565b8060005b8381101561471f57815161470688826146a9565b9750614711836146c1565b9250506001810190506146f2565b5085935050505092915050565b6000602082019050818103600083015261474681846146ce565b905092915050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b61478381613f6d565b82525050565b6000614795838361477a565b60208301905092915050565b6000602082019050919050565b60006147b98261474e565b6147c38185614759565b93506147ce8361476a565b8060005b838110156147ff5781516147e68882614789565b97506147f1836147a1565b9250506001810190506147d2565b5085935050505092915050565b6000602082019050818103600083015261482681846147ae565b905092915050565b61483781613da3565b811461484257600080fd5b50565b6000813590506148548161482e565b92915050565b6000602082840312156148705761486f613d14565b5b600061487e84828501614845565b91505092915050565b6000806000606084860312156148a05761489f613d14565b5b60006148ae86828701613e22565b93505060206148bf86828701613f8e565b92505060406148d086828701613f8e565b9150509250925092565b600080604083850312156148f1576148f0613d14565b5b60006148ff85828601613e22565b925050602061491085828601614845565b9150509250929050565b600067ffffffffffffffff82111561493557614934614096565b5b61493e82613f01565b9050602081019050919050565b600061495e6149598461491a565b6140f6565b90508281526020810184848401111561497a576149796143d0565b5b614985848285614406565b509392505050565b600082601f8301126149a2576149a1614091565b5b81356149b284826020860161494b565b91505092915050565b600080600080608085870312156149d5576149d4613d14565b5b60006149e387828801613e22565b94505060206149f487828801613e22565b9350506040614a0587828801613f8e565b925050606085013567ffffffffffffffff811115614a2657614a25613d19565b5b614a328782880161498d565b91505092959194509250565b608082016000820151614a5460008501826145f5565b506020820151614a676020850182614618565b506040820151614a7a6040850182614627565b506060820151614a8d6060850182614645565b50505050565b6000608082019050614aa86000830184614a3e565b92915050565b60008060408385031215614ac557614ac4613d14565b5b6000614ad385828601613e22565b9250506020614ae485828601613e22565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680614b3557607f821691505b602082108103614b4857614b47614aee565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000614bb782613f6d565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203614be957614be8614b7d565b5b600182019050919050565b6000604082019050614c096000830185613fd0565b614c166020830184613fd0565b9392505050565b600081519050614c2c8161482e565b92915050565b600060208284031215614c4857614c47613d14565b5b6000614c5684828501614c1d565b91505092915050565b6000614c6a82613f6d565b9150614c7583613f6d565b9250828202614c8381613f6d565b91508282048414831517614c9a57614c99614b7d565b5b5092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b6000614cdb82613f6d565b9150614ce683613f6d565b925082614cf657614cf5614ca1565b5b828204905092915050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b600060088302614d637fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82614d26565b614d6d8683614d26565b95508019841693508086168417925050509392505050565b6000819050919050565b6000614daa614da5614da084613f6d565b614d85565b613f6d565b9050919050565b6000819050919050565b614dc483614d8f565b614dd8614dd082614db1565b848454614d33565b825550505050565b600090565b614ded614de0565b614df8818484614dbb565b505050565b5b81811015614e1c57614e11600082614de5565b600181019050614dfe565b5050565b601f821115614e6157614e3281614d01565b614e3b84614d16565b81016020851015614e4a578190505b614e5e614e5685614d16565b830182614dfd565b50505b505050565b600082821c905092915050565b6000614e8460001984600802614e66565b1980831691505092915050565b6000614e9d8383614e73565b9150826002028217905092915050565b614eb682613ebb565b67ffffffffffffffff811115614ecf57614ece614096565b5b614ed98254614b1d565b614ee4828285614e20565b600060209050601f831160018114614f175760008415614f05578287015190505b614f0f8582614e91565b865550614f77565b601f198416614f2586614d01565b60005b82811015614f4d57848901518255600182019150602085019450602081019050614f28565b86831015614f6a5784890151614f66601f891682614e73565b8355505b6001600288020188555050505b505050505050565b7f466c75724c6162733a204d696e74204e6f742041637469766500000000000000600082015250565b6000614fb5601983613ec6565b9150614fc082614f7f565b602082019050919050565b60006020820190508181036000830152614fe481614fa8565b9050919050565b7f596f7520617265204e4f542061207768616c6500000000000000000000000000600082015250565b6000615021601383613ec6565b915061502c82614feb565b602082019050919050565b6000602082019050818103600083015261505081615014565b9050919050565b7f466c75724c6162733a20547279696e6720746f206d696e74206d6f726520746860008201527f616e20416c6c6f77656400000000000000000000000000000000000000000000602082015250565b60006150b3602a83613ec6565b91506150be82615057565b604082019050919050565b600060208201905081810360008301526150e2816150a6565b9050919050565b60006150f482613f6d565b91506150ff83613f6d565b925082820190508082111561511757615116614b7d565b5b92915050565b7f466c75724c6162733a204d696e7420537570706c792045786365656465640000600082015250565b6000615153601e83613ec6565b915061515e8261511d565b602082019050919050565b6000602082019050818103600083015261518281615146565b9050919050565b7f466c75724c6162733a2045786365656473204d617820466f7220596f7572205760008201527f68616c652057616c6c6574000000000000000000000000000000000000000000602082015250565b60006151e5602b83613ec6565b91506151f082615189565b604082019050919050565b60006020820190508181036000830152615214816151d8565b9050919050565b7f4e6f7420656e6f756768204554482073656e7420666f722073656c656374656460008201527f20616d6f756e7400000000000000000000000000000000000000000000000000602082015250565b6000615277602783613ec6565b91506152828261521b565b604082019050919050565b600060208201905081810360008301526152a68161526a565b9050919050565b600081905092915050565b50565b60006152c86000836152ad565b91506152d3826152b8565b600082019050919050565b60006152e9826152bb565b9150819050919050565b7f5769746864726177206661696c65642e00000000000000000000000000000000600082015250565b6000615329601083613ec6565b9150615334826152f3565b602082019050919050565b600060208201905081810360008301526153588161531c565b9050919050565b7f4552433732314d657461646174613a2055524920717565727920666f72206e6f60008201527f6e6578697374656e7420746f6b656e0000000000000000000000000000000000602082015250565b60006153bb602f83613ec6565b91506153c68261535f565b604082019050919050565b600060208201905081810360008301526153ea816153ae565b9050919050565b600081905092915050565b600061540782613ebb565b61541181856153f1565b9350615421818560208601613ed7565b80840191505092915050565b6000815461543a81614b1d565b61544481866153f1565b9450600182166000811461545f5760018114615474576154a7565b60ff19831686528115158202860193506154a7565b61547d85614d01565b60005b8381101561549f57815481890152600182019150602081019050615480565b838801955050505b50505092915050565b60006154bc82866153fc565b91506154c882856153fc565b91506154d4828461542d565b9150819050949350505050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b600061553d602683613ec6565b9150615548826154e1565b604082019050919050565b6000602082019050818103600083015261556c81615530565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006155a9602083613ec6565b91506155b482615573565b602082019050919050565b600060208201905081810360008301526155d88161559c565b9050919050565b7f455243323938313a20726f79616c7479206665652077696c6c2065786365656460008201527f2073616c65507269636500000000000000000000000000000000000000000000602082015250565b600061563b602a83613ec6565b9150615646826155df565b604082019050919050565b6000602082019050818103600083015261566a8161562e565b9050919050565b7f455243323938313a20696e76616c696420726563656976657200000000000000600082015250565b60006156a7601983613ec6565b91506156b282615671565b602082019050919050565b600060208201905081810360008301526156d68161569a565b9050919050565b7f455243323938313a20496e76616c696420706172616d65746572730000000000600082015250565b6000615713601b83613ec6565b915061571e826156dd565b602082019050919050565b6000602082019050818103600083015261574281615706565b9050919050565b600081519050919050565b600082825260208201905092915050565b600061577082615749565b61577a8185615754565b935061578a818560208601613ed7565b61579381613f01565b840191505092915050565b60006080820190506157b36000830187613fd0565b6157c06020830186613fd0565b6157cd6040830185614067565b81810360608301526157df8184615765565b905095945050505050565b6000815190506157f981613d4a565b92915050565b60006020828403121561581557615814613d14565b5b6000615823848285016157ea565b9150509291505056fea2646970667358221220b67668e595bed29b874105bc717d5732ec5e2b63043552ccb0f269e84c38111164736f6c63430008110033
Deployed Bytecode
0x60806040526004361061027d5760003560e01c806370a082311161014f578063aa1b103f116100c1578063cc47a40b1161007a578063cc47a40b1461097a578063da3ef23f146109a3578063e985e9c5146109cc578063f2c4ce1e14610a09578063f2fde38b14610a32578063f968adbe14610a5b5761027d565b8063aa1b103f1461088b578063ac446002146108a2578063b88d4fde146108b9578063c23dc68f146108d5578063c668286214610912578063c87b56dd1461093d5761027d565b80639434654b116101135780639434654b1461079e57806395d89b41146107c757806399a2557a146107f2578063a0712d681461082f578063a22cb4651461084b578063a475b5dd146108745761027d565b806370a08231146106b9578063715018a6146106f65780638462151c1461070d5780638a616bc01461074a5780638da5cb5b146107735761027d565b80632a55205a116101f35780635944c753116101ac5780635944c753146105835780635bbb2177146105ac5780636352211e146105e9578063639814e0146106265780636c0360eb14610651578063703cc38a1461067c5761027d565b80632a55205a1461048157806342842e0e146104bf57806342966c68146104db5780634725508e14610504578063518302271461052f57806355f804b31461055a5761027d565b8063095ea7b311610245578063095ea7b31461037b5780630cf845021461039757806315417bd7146103d457806318160ddd146104115780631bac14ce1461043c57806323b872dd146104655761027d565b806301ffc9a71461028257806304634d8d146102bf57806306fdde03146102e8578063081812fc14610313578063081c8c4414610350575b600080fd5b34801561028e57600080fd5b506102a960048036038101906102a49190613d76565b610a86565b6040516102b69190613dbe565b60405180910390f35b3480156102cb57600080fd5b506102e660048036038101906102e19190613e7b565b610aa8565b005b3480156102f457600080fd5b506102fd610abe565b60405161030a9190613f4b565b60405180910390f35b34801561031f57600080fd5b5061033a60048036038101906103359190613fa3565b610b50565b6040516103479190613fdf565b60405180910390f35b34801561035c57600080fd5b50610365610bcf565b6040516103729190613f4b565b60405180910390f35b61039560048036038101906103909190613ffa565b610c5d565b005b3480156103a357600080fd5b506103be60048036038101906103b9919061403a565b610c6d565b6040516103cb9190614076565b60405180910390f35b3480156103e057600080fd5b506103fb60048036038101906103f6919061403a565b610c85565b6040516104089190614076565b60405180910390f35b34801561041d57600080fd5b50610426610c97565b6040516104339190614076565b60405180910390f35b34801561044857600080fd5b50610463600480360381019061045e919061429c565b610cae565b005b61047f600480360381019061047a9190614314565b610d52565b005b34801561048d57600080fd5b506104a860048036038101906104a39190614367565b610f34565b6040516104b69291906143a7565b60405180910390f35b6104d960048036038101906104d49190614314565b61111e565b005b3480156104e757600080fd5b5061050260048036038101906104fd9190613fa3565b611300565b005b34801561051057600080fd5b5061051961130e565b6040516105269190613dbe565b60405180910390f35b34801561053b57600080fd5b50610544611321565b6040516105519190613dbe565b60405180910390f35b34801561056657600080fd5b50610581600480360381019061057c9190614485565b611334565b005b34801561058f57600080fd5b506105aa60048036038101906105a591906144ce565b61134f565b005b3480156105b857600080fd5b506105d360048036038101906105ce919061457c565b611367565b6040516105e0919061472c565b60405180910390f35b3480156105f557600080fd5b50610610600480360381019061060b9190613fa3565b61142a565b60405161061d9190613fdf565b60405180910390f35b34801561063257600080fd5b5061063b61143c565b6040516106489190614076565b60405180910390f35b34801561065d57600080fd5b50610666611442565b6040516106739190613f4b565b60405180910390f35b34801561068857600080fd5b506106a3600480360381019061069e919061403a565b6114d0565b6040516106b09190614076565b60405180910390f35b3480156106c557600080fd5b506106e060048036038101906106db919061403a565b611519565b6040516106ed9190614076565b60405180910390f35b34801561070257600080fd5b5061070b6115d1565b005b34801561071957600080fd5b50610734600480360381019061072f919061403a565b6115e5565b604051610741919061480c565b60405180910390f35b34801561075657600080fd5b50610771600480360381019061076c9190613fa3565b611728565b005b34801561077f57600080fd5b5061078861173c565b6040516107959190613fdf565b60405180910390f35b3480156107aa57600080fd5b506107c560048036038101906107c0919061485a565b611766565b005b3480156107d357600080fd5b506107dc61178b565b6040516107e99190613f4b565b60405180910390f35b3480156107fe57600080fd5b5061081960048036038101906108149190614887565b61181d565b604051610826919061480c565b60405180910390f35b61084960048036038101906108449190613fa3565b611a29565b005b34801561085757600080fd5b50610872600480360381019061086d91906148da565b611cca565b005b34801561088057600080fd5b50610889611dd5565b005b34801561089757600080fd5b506108a0611dfa565b005b3480156108ae57600080fd5b506108b7611e0c565b005b6108d360048036038101906108ce91906149bb565b611ec3565b005b3480156108e157600080fd5b506108fc60048036038101906108f79190613fa3565b6120a8565b6040516109099190614a93565b60405180910390f35b34801561091e57600080fd5b50610927612112565b6040516109349190613f4b565b60405180910390f35b34801561094957600080fd5b50610964600480360381019061095f9190613fa3565b6121a0565b6040516109719190613f4b565b60405180910390f35b34801561098657600080fd5b506109a1600480360381019061099c9190613ffa565b6122f8565b005b3480156109af57600080fd5b506109ca60048036038101906109c59190614485565b612364565b005b3480156109d857600080fd5b506109f360048036038101906109ee9190614aae565b61237f565b604051610a009190613dbe565b60405180910390f35b348015610a1557600080fd5b50610a306004803603810190610a2b9190614485565b612413565b005b348015610a3e57600080fd5b50610a596004803603810190610a54919061403a565b61242e565b005b348015610a6757600080fd5b50610a706124b1565b604051610a7d9190614076565b60405180910390f35b6000610a91826124b7565b80610aa15750610aa082612531565b5b9050919050565b610ab06125c3565b610aba8282612641565b5050565b606060028054610acd90614b1d565b80601f0160208091040260200160405190810160405280929190818152602001828054610af990614b1d565b8015610b465780601f10610b1b57610100808354040283529160200191610b46565b820191906000526020600020905b815481529060010190602001808311610b2957829003601f168201915b5050505050905090565b6000610b5b826127d6565b610b91576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b600f8054610bdc90614b1d565b80601f0160208091040260200160405190810160405280929190818152602001828054610c0890614b1d565b8015610c555780601f10610c2a57610100808354040283529160200191610c55565b820191906000526020600020905b815481529060010190602001808311610c3857829003601f168201915b505050505081565b610c6982826001612835565b5050565b60116020528060005260406000206000915090505481565b6000610c9082612985565b9050919050565b6000610ca16129dc565b6001546000540303905090565b610cb66125c3565b60005b8251811015610d4d57818181518110610cd557610cd4614b4e565b5b602002602001015160116000858481518110610cf457610cf3614b4e565b5b602002602001015173ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055508080610d4590614bac565b915050610cb9565b505050565b8260006daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff163b1115610f22573373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603610dc457610dbf8484846129e5565b610f2e565b6daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430336040518363ffffffff1660e01b8152600401610e0d929190614bf4565b602060405180830381865afa158015610e2a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e4e9190614c32565b8015610ee057506daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430836040518363ffffffff1660e01b8152600401610e9e929190614bf4565b602060405180830381865afa158015610ebb573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610edf9190614c32565b5b610f2157336040517fede71dcc000000000000000000000000000000000000000000000000000000008152600401610f189190613fdf565b60405180910390fd5b5b610f2d8484846129e5565b5b50505050565b6000806000600a60008681526020019081526020016000206040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff16815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff16036110c95760096040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff168152505090505b60006110d3612d07565b6bffffffffffffffffffffffff1682602001516bffffffffffffffffffffffff16866110ff9190614c5f565b6111099190614cd0565b90508160000151819350935050509250929050565b8260006daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff163b11156112ee573373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036111905761118b848484612d11565b6112fa565b6daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430336040518363ffffffff1660e01b81526004016111d9929190614bf4565b602060405180830381865afa1580156111f6573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061121a9190614c32565b80156112ac57506daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430836040518363ffffffff1660e01b815260040161126a929190614bf4565b602060405180830381865afa158015611287573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112ab9190614c32565b5b6112ed57336040517fede71dcc0000000000000000000000000000000000000000000000000000000081526004016112e49190613fdf565b60405180910390fd5b5b6112f9848484612d11565b5b50505050565b61130b816001612d31565b50565b601060019054906101000a900460ff1681565b601060009054906101000a900460ff1681565b61133c6125c3565b80600d908161134b9190614ead565b5050565b6113576125c3565b611362838383612f83565b505050565b6060600083839050905060008167ffffffffffffffff81111561138d5761138c614096565b5b6040519080825280602002602001820160405280156113c657816020015b6113b3613cbb565b8152602001906001900390816113ab5790505b50905060005b82811461141e576113f58686838181106113e9576113e8614b4e565b5b905060200201356120a8565b82828151811061140857611407614b4e565b5b60200260200101819052508060010190506113cc565b50809250505092915050565b60006114358261312a565b9050919050565b600b5481565b600d805461144f90614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461147b90614b1d565b80156114c85780601f1061149d576101008083540402835291602001916114c8565b820191906000526020600020905b8154815290600101906020018083116114ab57829003601f168201915b505050505081565b6000601160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020549050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611580576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b6115d96125c3565b6115e360006131f6565b565b606060008060006115f585611519565b905060008167ffffffffffffffff81111561161357611612614096565b5b6040519080825280602002602001820160405280156116415781602001602082028036833780820191505090505b50905061164c613cbb565b60006116566129dc565b90505b83861461171a57611669816132bc565b9150816040015161170f57600073ffffffffffffffffffffffffffffffffffffffff16826000015173ffffffffffffffffffffffffffffffffffffffff16146116b457816000015194505b8773ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff160361170e578083878060010198508151811061170157611700614b4e565b5b6020026020010181815250505b5b806001019050611659565b508195505050505050919050565b6117306125c3565b611739816132e7565b50565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905090565b61176e6125c3565b80601060016101000a81548160ff02191690831515021790555050565b60606003805461179a90614b1d565b80601f01602080910402602001604051908101604052809291908181526020018280546117c690614b1d565b80156118135780601f106117e857610100808354040283529160200191611813565b820191906000526020600020905b8154815290600101906020018083116117f657829003601f168201915b5050505050905090565b6060818310611858576040517f32c1995a00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080611863613346565b905061186d6129dc565b85101561187f5761187c6129dc565b94505b8084111561188b578093505b600061189687611519565b9050848610156118b95760008686039050818110156118b3578091505b506118be565b600090505b60008167ffffffffffffffff8111156118da576118d9614096565b5b6040519080825280602002602001820160405280156119085781602001602082028036833780820191505090505b5090506000820361191f5780945050505050611a22565b600061192a886120a8565b90506000816040015161193f57816000015190505b60008990505b8881141580156119555750848714155b15611a1457611963816132bc565b92508260400151611a0957600073ffffffffffffffffffffffffffffffffffffffff16836000015173ffffffffffffffffffffffffffffffffffffffff16146119ae57826000015191505b8a73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611a0857808488806001019950815181106119fb576119fa614b4e565b5b6020026020010181815250505b5b806001019050611945565b508583528296505050505050505b9392505050565b601060019054906101000a900460ff16611a78576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611a6f90614fcb565b60405180910390fd5b6000601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205411611afa576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611af190615037565b60405180910390fd5b601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054811115611b7c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611b73906150c9565b60405180910390fd5b603281611b87610c97565b611b9191906150e9565b1115611bd2576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611bc990615169565b60405180910390fd5b601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205481611c1c33612985565b611c2691906150e9565b1115611c67576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611c5e906151fb565b60405180910390fd5b3481670de0b6b3a7640000611c7c9190614c5f565b1115611cbd576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611cb49061528d565b60405180910390fd5b611cc7338261334f565b50565b8060076000611cd761336d565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff16611d8461336d565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c3183604051611dc99190613dbe565b60405180910390a35050565b611ddd6125c3565b6001601060006101000a81548160ff021916908315150217905550565b611e026125c3565b611e0a613375565b565b611e146125c3565b60003373ffffffffffffffffffffffffffffffffffffffff1647604051611e3a906152de565b60006040518083038185875af1925050503d8060008114611e77576040519150601f19603f3d011682016040523d82523d6000602084013e611e7c565b606091505b5050905080611ec0576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611eb79061533f565b60405180910390fd5b50565b8360006daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff163b1115612094573373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603611f3657611f31858585856133c2565b6120a1565b6daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430336040518363ffffffff1660e01b8152600401611f7f929190614bf4565b602060405180830381865afa158015611f9c573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611fc09190614c32565b801561205257506daaeb6d7670e522a718067333cd4e73ffffffffffffffffffffffffffffffffffffffff1663c617113430836040518363ffffffff1660e01b8152600401612010929190614bf4565b602060405180830381865afa15801561202d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906120519190614c32565b5b61209357336040517fede71dcc00000000000000000000000000000000000000000000000000000000815260040161208a9190613fdf565b60405180910390fd5b5b6120a0858585856133c2565b5b5050505050565b6120b0613cbb565b6120b8613cbb565b6120c06129dc565b8310806120d457506120d0613346565b8310155b156120e2578091505061210d565b6120eb836132bc565b9050806040015115612100578091505061210d565b61210983613435565b9150505b919050565b600e805461211f90614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461214b90614b1d565b80156121985780601f1061216d57610100808354040283529160200191612198565b820191906000526020600020905b81548152906001019060200180831161217b57829003601f168201915b505050505081565b60606121ab826127d6565b6121ea576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016121e1906153d1565b60405180910390fd5b60001515601060009054906101000a900460ff1615150361229757600f805461221290614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461223e90614b1d565b801561228b5780601f106122605761010080835404028352916020019161228b565b820191906000526020600020905b81548152906001019060200180831161226e57829003601f168201915b505050505090506122f3565b60006122a1613455565b905060008151116122c157604051806020016040528060008152506122ef565b806122cb846134e7565b600e6040516020016122df939291906154b0565b6040516020818303038152906040525b9150505b919050565b6123006125c3565b60328161230b610c97565b61231591906150e9565b1115612356576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161234d90615169565b60405180910390fd5b612360828261334f565b5050565b61236c6125c3565b80600e908161237b9190614ead565b5050565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b61241b6125c3565b80600f908161242a9190614ead565b5050565b6124366125c3565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16036124a5576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161249c90615553565b60405180910390fd5b6124ae816131f6565b50565b600c5481565b60007f2a55205a000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061252a5750612529826135b5565b5b9050919050565b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916148061258c57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b806125bc5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b6125cb61361f565b73ffffffffffffffffffffffffffffffffffffffff166125e961173c565b73ffffffffffffffffffffffffffffffffffffffff161461263f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612636906155bf565b60405180910390fd5b565b612649612d07565b6bffffffffffffffffffffffff16816bffffffffffffffffffffffff1611156126a7576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161269e90615651565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603612716576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161270d906156bd565b60405180910390fd5b60405180604001604052808373ffffffffffffffffffffffffffffffffffffffff168152602001826bffffffffffffffffffffffff16815250600960008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a8154816bffffffffffffffffffffffff02191690836bffffffffffffffffffffffff1602179055509050505050565b6000816127e16129dc565b111580156127f0575060005482105b801561282e575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b60006128408361142a565b905081801561288257508073ffffffffffffffffffffffffffffffffffffffff1661286961336d565b73ffffffffffffffffffffffffffffffffffffffff1614155b156128cf576128988161289361336d565b61237f565b6128ce576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b836006600085815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550828473ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a450505050565b600067ffffffffffffffff6040600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054901c169050919050565b60006001905090565b60006129f08261312a565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1614612a57576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b600080612a6384613627565b91509150612a798187612a7461336d565b61364e565b612ac557612a8e86612a8961336d565b61237f565b612ac4576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff1603612b2b576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612b388686866001613692565b8015612b4357600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000815460010191905081905550612c1185612bed888887613698565b7c0200000000000000000000000000000000000000000000000000000000176136c0565b600460008681526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000841603612c975760006001850190506000600460008381526020019081526020016000205403612c95576000548114612c94578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612cff86868660016136eb565b505050505050565b6000612710905090565b612d2c83838360405180602001604052806000815250611ec3565b505050565b6000612d3c8361312a565b90506000819050600080612d4f86613627565b915091508415612db857612d6b8184612d6661336d565b61364e565b612db757612d8083612d7b61336d565b61237f565b612db6576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b5b612dc6836000886001613692565b8015612dd157600082555b600160806001901b03600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550612e7983612e3685600088613698565b7c02000000000000000000000000000000000000000000000000000000007c010000000000000000000000000000000000000000000000000000000017176136c0565b600460008881526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000851603612eff5760006001870190506000600460008381526020019081526020016000205403612efd576000548114612efc578460046000838152602001908152602001600020819055505b5b505b85600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a4612f698360008860016136eb565b600160008154809291906001019190505550505050505050565b612f8b612d07565b6bffffffffffffffffffffffff16816bffffffffffffffffffffffff161115612fe9576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612fe090615651565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603613058576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161304f90615729565b60405180910390fd5b60405180604001604052808373ffffffffffffffffffffffffffffffffffffffff168152602001826bffffffffffffffffffffffff16815250600a600085815260200190815260200160002060008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a8154816bffffffffffffffffffffffff02191690836bffffffffffffffffffffffff160217905550905050505050565b600080829050806131396129dc565b116131bf576000548110156131be5760006004600083815260200190815260200160002054905060007c01000000000000000000000000000000000000000000000000000000008216036131bc575b600081036131b2576004600083600190039350838152602001908152602001600020549050613188565b80925050506131f1565b505b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b6000600860009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16905081600860006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055508173ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e060405160405180910390a35050565b6132c4613cbb565b6132e060046000848152602001908152602001600020546136f1565b9050919050565b600a6000828152602001908152602001600020600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a8154906bffffffffffffffffffffffff0219169055505050565b60008054905090565b6133698282604051806020016040528060008152506137a7565b5050565b600033905090565b6009600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a8154906bffffffffffffffffffffffff02191690555050565b6133cd848484610d52565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461342f576133f884848484613844565b61342e576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b61343d613cbb565b61344e6134498361312a565b6136f1565b9050919050565b6060600d805461346490614b1d565b80601f016020809104026020016040519081016040528092919081815260200182805461349090614b1d565b80156134dd5780601f106134b2576101008083540402835291602001916134dd565b820191906000526020600020905b8154815290600101906020018083116134c057829003601f168201915b5050505050905090565b6060600060016134f684613994565b01905060008167ffffffffffffffff81111561351557613514614096565b5b6040519080825280601f01601f1916602001820160405280156135475781602001600182028036833780820191505090505b509050600082602001820190505b6001156135aa578080600190039150507f3031323334353637383961626364656600000000000000000000000000000000600a86061a8153600a858161359e5761359d614ca1565b5b04945060008503613555575b819350505050919050565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b600033905090565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b50505050565b60008060e883901c905060e86136af868684613ae7565b62ffffff16901b9150509392505050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b6136f9613cbb565b81816000019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060a082901c816020019067ffffffffffffffff16908167ffffffffffffffff168152505060007c01000000000000000000000000000000000000000000000000000000008316141581604001901515908115158152505060e882901c816060019062ffffff16908162ffffff1681525050919050565b6137b18383613af0565b60008373ffffffffffffffffffffffffffffffffffffffff163b1461383f57600080549050600083820390505b6137f16000868380600101945086613844565b613827576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8181106137de57816000541461383c57600080fd5b50505b505050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a0261386a61336d565b8786866040518563ffffffff1660e01b815260040161388c949392919061579e565b6020604051808303816000875af19250505080156138c857506040513d601f19601f820116820180604052508101906138c591906157ff565b60015b613941573d80600081146138f8576040519150601f19603f3d011682016040523d82523d6000602084013e6138fd565b606091505b506000815103613939576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b600080600090507a184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000083106139f2577a184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000083816139e8576139e7614ca1565b5b0492506040810190505b6d04ee2d6d415b85acef81000000008310613a2f576d04ee2d6d415b85acef81000000008381613a2557613a24614ca1565b5b0492506020810190505b662386f26fc100008310613a5e57662386f26fc100008381613a5457613a53614ca1565b5b0492506010810190505b6305f5e1008310613a87576305f5e1008381613a7d57613a7c614ca1565b5b0492506008810190505b6127108310613aac576127108381613aa257613aa1614ca1565b5b0492506004810190505b60648310613acf5760648381613ac557613ac4614ca1565b5b0492506002810190505b600a8310613ade576001810190505b80915050919050565b60009392505050565b60008054905060008203613b30576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b613b3d6000848385613692565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550613bb483613ba56000866000613698565b613bae85613cab565b176136c0565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b818114613c5557808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600181019050613c1a565b5060008203613c90576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b806000819055505050613ca660008483856136eb565b505050565b60006001821460e11b9050919050565b6040518060800160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff168152602001600015158152602001600062ffffff1681525090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b613d5381613d1e565b8114613d5e57600080fd5b50565b600081359050613d7081613d4a565b92915050565b600060208284031215613d8c57613d8b613d14565b5b6000613d9a84828501613d61565b91505092915050565b60008115159050919050565b613db881613da3565b82525050565b6000602082019050613dd36000830184613daf565b92915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b6000613e0482613dd9565b9050919050565b613e1481613df9565b8114613e1f57600080fd5b50565b600081359050613e3181613e0b565b92915050565b60006bffffffffffffffffffffffff82169050919050565b613e5881613e37565b8114613e6357600080fd5b50565b600081359050613e7581613e4f565b92915050565b60008060408385031215613e9257613e91613d14565b5b6000613ea085828601613e22565b9250506020613eb185828601613e66565b9150509250929050565b600081519050919050565b600082825260208201905092915050565b60005b83811015613ef5578082015181840152602081019050613eda565b60008484015250505050565b6000601f19601f8301169050919050565b6000613f1d82613ebb565b613f278185613ec6565b9350613f37818560208601613ed7565b613f4081613f01565b840191505092915050565b60006020820190508181036000830152613f658184613f12565b905092915050565b6000819050919050565b613f8081613f6d565b8114613f8b57600080fd5b50565b600081359050613f9d81613f77565b92915050565b600060208284031215613fb957613fb8613d14565b5b6000613fc784828501613f8e565b91505092915050565b613fd981613df9565b82525050565b6000602082019050613ff46000830184613fd0565b92915050565b6000806040838503121561401157614010613d14565b5b600061401f85828601613e22565b925050602061403085828601613f8e565b9150509250929050565b6000602082840312156140505761404f613d14565b5b600061405e84828501613e22565b91505092915050565b61407081613f6d565b82525050565b600060208201905061408b6000830184614067565b92915050565b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6140ce82613f01565b810181811067ffffffffffffffff821117156140ed576140ec614096565b5b80604052505050565b6000614100613d0a565b905061410c82826140c5565b919050565b600067ffffffffffffffff82111561412c5761412b614096565b5b602082029050602081019050919050565b600080fd5b600061415561415084614111565b6140f6565b905080838252602082019050602084028301858111156141785761417761413d565b5b835b818110156141a1578061418d8882613e22565b84526020840193505060208101905061417a565b5050509392505050565b600082601f8301126141c0576141bf614091565b5b81356141d0848260208601614142565b91505092915050565b600067ffffffffffffffff8211156141f4576141f3614096565b5b602082029050602081019050919050565b6000614218614213846141d9565b6140f6565b9050808382526020820190506020840283018581111561423b5761423a61413d565b5b835b8181101561426457806142508882613f8e565b84526020840193505060208101905061423d565b5050509392505050565b600082601f83011261428357614282614091565b5b8135614293848260208601614205565b91505092915050565b600080604083850312156142b3576142b2613d14565b5b600083013567ffffffffffffffff8111156142d1576142d0613d19565b5b6142dd858286016141ab565b925050602083013567ffffffffffffffff8111156142fe576142fd613d19565b5b61430a8582860161426e565b9150509250929050565b60008060006060848603121561432d5761432c613d14565b5b600061433b86828701613e22565b935050602061434c86828701613e22565b925050604061435d86828701613f8e565b9150509250925092565b6000806040838503121561437e5761437d613d14565b5b600061438c85828601613f8e565b925050602061439d85828601613f8e565b9150509250929050565b60006040820190506143bc6000830185613fd0565b6143c96020830184614067565b9392505050565b600080fd5b600067ffffffffffffffff8211156143f0576143ef614096565b5b6143f982613f01565b9050602081019050919050565b82818337600083830152505050565b6000614428614423846143d5565b6140f6565b905082815260208101848484011115614444576144436143d0565b5b61444f848285614406565b509392505050565b600082601f83011261446c5761446b614091565b5b813561447c848260208601614415565b91505092915050565b60006020828403121561449b5761449a613d14565b5b600082013567ffffffffffffffff8111156144b9576144b8613d19565b5b6144c584828501614457565b91505092915050565b6000806000606084860312156144e7576144e6613d14565b5b60006144f586828701613f8e565b935050602061450686828701613e22565b925050604061451786828701613e66565b9150509250925092565b600080fd5b60008083601f84011261453c5761453b614091565b5b8235905067ffffffffffffffff81111561455957614558614521565b5b6020830191508360208202830111156145755761457461413d565b5b9250929050565b6000806020838503121561459357614592613d14565b5b600083013567ffffffffffffffff8111156145b1576145b0613d19565b5b6145bd85828601614526565b92509250509250929050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b6145fe81613df9565b82525050565b600067ffffffffffffffff82169050919050565b61462181614604565b82525050565b61463081613da3565b82525050565b600062ffffff82169050919050565b61464e81614636565b82525050565b60808201600082015161466a60008501826145f5565b50602082015161467d6020850182614618565b5060408201516146906040850182614627565b5060608201516146a36060850182614645565b50505050565b60006146b58383614654565b60808301905092915050565b6000602082019050919050565b60006146d9826145c9565b6146e381856145d4565b93506146ee836145e5565b8060005b8381101561471f57815161470688826146a9565b9750614711836146c1565b9250506001810190506146f2565b5085935050505092915050565b6000602082019050818103600083015261474681846146ce565b905092915050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b61478381613f6d565b82525050565b6000614795838361477a565b60208301905092915050565b6000602082019050919050565b60006147b98261474e565b6147c38185614759565b93506147ce8361476a565b8060005b838110156147ff5781516147e68882614789565b97506147f1836147a1565b9250506001810190506147d2565b5085935050505092915050565b6000602082019050818103600083015261482681846147ae565b905092915050565b61483781613da3565b811461484257600080fd5b50565b6000813590506148548161482e565b92915050565b6000602082840312156148705761486f613d14565b5b600061487e84828501614845565b91505092915050565b6000806000606084860312156148a05761489f613d14565b5b60006148ae86828701613e22565b93505060206148bf86828701613f8e565b92505060406148d086828701613f8e565b9150509250925092565b600080604083850312156148f1576148f0613d14565b5b60006148ff85828601613e22565b925050602061491085828601614845565b9150509250929050565b600067ffffffffffffffff82111561493557614934614096565b5b61493e82613f01565b9050602081019050919050565b600061495e6149598461491a565b6140f6565b90508281526020810184848401111561497a576149796143d0565b5b614985848285614406565b509392505050565b600082601f8301126149a2576149a1614091565b5b81356149b284826020860161494b565b91505092915050565b600080600080608085870312156149d5576149d4613d14565b5b60006149e387828801613e22565b94505060206149f487828801613e22565b9350506040614a0587828801613f8e565b925050606085013567ffffffffffffffff811115614a2657614a25613d19565b5b614a328782880161498d565b91505092959194509250565b608082016000820151614a5460008501826145f5565b506020820151614a676020850182614618565b506040820151614a7a6040850182614627565b506060820151614a8d6060850182614645565b50505050565b6000608082019050614aa86000830184614a3e565b92915050565b60008060408385031215614ac557614ac4613d14565b5b6000614ad385828601613e22565b9250506020614ae485828601613e22565b9150509250929050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680614b3557607f821691505b602082108103614b4857614b47614aee565b5b50919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000614bb782613f6d565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203614be957614be8614b7d565b5b600182019050919050565b6000604082019050614c096000830185613fd0565b614c166020830184613fd0565b9392505050565b600081519050614c2c8161482e565b92915050565b600060208284031215614c4857614c47613d14565b5b6000614c5684828501614c1d565b91505092915050565b6000614c6a82613f6d565b9150614c7583613f6d565b9250828202614c8381613f6d565b91508282048414831517614c9a57614c99614b7d565b5b5092915050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b6000614cdb82613f6d565b9150614ce683613f6d565b925082614cf657614cf5614ca1565b5b828204905092915050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b600060088302614d637fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82614d26565b614d6d8683614d26565b95508019841693508086168417925050509392505050565b6000819050919050565b6000614daa614da5614da084613f6d565b614d85565b613f6d565b9050919050565b6000819050919050565b614dc483614d8f565b614dd8614dd082614db1565b848454614d33565b825550505050565b600090565b614ded614de0565b614df8818484614dbb565b505050565b5b81811015614e1c57614e11600082614de5565b600181019050614dfe565b5050565b601f821115614e6157614e3281614d01565b614e3b84614d16565b81016020851015614e4a578190505b614e5e614e5685614d16565b830182614dfd565b50505b505050565b600082821c905092915050565b6000614e8460001984600802614e66565b1980831691505092915050565b6000614e9d8383614e73565b9150826002028217905092915050565b614eb682613ebb565b67ffffffffffffffff811115614ecf57614ece614096565b5b614ed98254614b1d565b614ee4828285614e20565b600060209050601f831160018114614f175760008415614f05578287015190505b614f0f8582614e91565b865550614f77565b601f198416614f2586614d01565b60005b82811015614f4d57848901518255600182019150602085019450602081019050614f28565b86831015614f6a5784890151614f66601f891682614e73565b8355505b6001600288020188555050505b505050505050565b7f466c75724c6162733a204d696e74204e6f742041637469766500000000000000600082015250565b6000614fb5601983613ec6565b9150614fc082614f7f565b602082019050919050565b60006020820190508181036000830152614fe481614fa8565b9050919050565b7f596f7520617265204e4f542061207768616c6500000000000000000000000000600082015250565b6000615021601383613ec6565b915061502c82614feb565b602082019050919050565b6000602082019050818103600083015261505081615014565b9050919050565b7f466c75724c6162733a20547279696e6720746f206d696e74206d6f726520746860008201527f616e20416c6c6f77656400000000000000000000000000000000000000000000602082015250565b60006150b3602a83613ec6565b91506150be82615057565b604082019050919050565b600060208201905081810360008301526150e2816150a6565b9050919050565b60006150f482613f6d565b91506150ff83613f6d565b925082820190508082111561511757615116614b7d565b5b92915050565b7f466c75724c6162733a204d696e7420537570706c792045786365656465640000600082015250565b6000615153601e83613ec6565b915061515e8261511d565b602082019050919050565b6000602082019050818103600083015261518281615146565b9050919050565b7f466c75724c6162733a2045786365656473204d617820466f7220596f7572205760008201527f68616c652057616c6c6574000000000000000000000000000000000000000000602082015250565b60006151e5602b83613ec6565b91506151f082615189565b604082019050919050565b60006020820190508181036000830152615214816151d8565b9050919050565b7f4e6f7420656e6f756768204554482073656e7420666f722073656c656374656460008201527f20616d6f756e7400000000000000000000000000000000000000000000000000602082015250565b6000615277602783613ec6565b91506152828261521b565b604082019050919050565b600060208201905081810360008301526152a68161526a565b9050919050565b600081905092915050565b50565b60006152c86000836152ad565b91506152d3826152b8565b600082019050919050565b60006152e9826152bb565b9150819050919050565b7f5769746864726177206661696c65642e00000000000000000000000000000000600082015250565b6000615329601083613ec6565b9150615334826152f3565b602082019050919050565b600060208201905081810360008301526153588161531c565b9050919050565b7f4552433732314d657461646174613a2055524920717565727920666f72206e6f60008201527f6e6578697374656e7420746f6b656e0000000000000000000000000000000000602082015250565b60006153bb602f83613ec6565b91506153c68261535f565b604082019050919050565b600060208201905081810360008301526153ea816153ae565b9050919050565b600081905092915050565b600061540782613ebb565b61541181856153f1565b9350615421818560208601613ed7565b80840191505092915050565b6000815461543a81614b1d565b61544481866153f1565b9450600182166000811461545f5760018114615474576154a7565b60ff19831686528115158202860193506154a7565b61547d85614d01565b60005b8381101561549f57815481890152600182019150602081019050615480565b838801955050505b50505092915050565b60006154bc82866153fc565b91506154c882856153fc565b91506154d4828461542d565b9150819050949350505050565b7f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160008201527f6464726573730000000000000000000000000000000000000000000000000000602082015250565b600061553d602683613ec6565b9150615548826154e1565b604082019050919050565b6000602082019050818103600083015261556c81615530565b9050919050565b7f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572600082015250565b60006155a9602083613ec6565b91506155b482615573565b602082019050919050565b600060208201905081810360008301526155d88161559c565b9050919050565b7f455243323938313a20726f79616c7479206665652077696c6c2065786365656460008201527f2073616c65507269636500000000000000000000000000000000000000000000602082015250565b600061563b602a83613ec6565b9150615646826155df565b604082019050919050565b6000602082019050818103600083015261566a8161562e565b9050919050565b7f455243323938313a20696e76616c696420726563656976657200000000000000600082015250565b60006156a7601983613ec6565b91506156b282615671565b602082019050919050565b600060208201905081810360008301526156d68161569a565b9050919050565b7f455243323938313a20496e76616c696420706172616d65746572730000000000600082015250565b6000615713601b83613ec6565b915061571e826156dd565b602082019050919050565b6000602082019050818103600083015261574281615706565b9050919050565b600081519050919050565b600082825260208201905092915050565b600061577082615749565b61577a8185615754565b935061578a818560208601613ed7565b61579381613f01565b840191505092915050565b60006080820190506157b36000830187613fd0565b6157c06020830186613fd0565b6157cd6040830185614067565b81810360608301526157df8184615765565b905095945050505050565b6000815190506157f981613d4a565b92915050565b60006020828403121561581557615814613d14565b5b6000615823848285016157ea565b9150509291505056fea2646970667358221220b67668e595bed29b874105bc717d5732ec5e2b63043552ccb0f269e84c38111164736f6c63430008110033
Deployed Bytecode Sourcemap
130718:5323:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;133407:301;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;134406:172;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;82068:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;88031:218;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131092:28;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;87748:124;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;131198:49;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131653:122;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;77819:323;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131419:226;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;135083:238;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;69402:442;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;135327:244;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;128313:94;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;131161:30;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131127:27;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;133963:104;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;134690:201;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;123175:528;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;83461:152;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;130947:32;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131020:21;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131783:131;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;79003:233;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;30448:103;;;;;;;;;;;;;:::i;:::-;;127051:900;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;134897:126;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;29800:87;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131319:92;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;82244:104;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;124091:2513;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131922:661;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;88589:234;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;134327:69;;;;;;;;;;;;;:::i;:::-;;134584:100;;;;;;;;;;;;;:::i;:::-;;135860:178;;;;;;;;;;;;;:::i;:::-;;135577:275;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;122588:428;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;131048:37;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;132813:584;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;132591:214;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;134075:128;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;88980:164;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;133829:126;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;30706:201;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;130986:27;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;133407:301;133564:4;133607:38;133633:11;133607:25;:38::i;:::-;:93;;;;133662:38;133688:11;133662:25;:38::i;:::-;133607:93;133587:113;;133407:301;;;:::o;134406:172::-;29686:13;:11;:13::i;:::-;134528:44:::1;134547:9;134558:13;134528:18;:44::i;:::-;134406:172:::0;;:::o;82068:100::-;82122:13;82155:5;82148:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;82068:100;:::o;88031:218::-;88107:7;88132:16;88140:7;88132;:16::i;:::-;88127:64;;88157:34;;;;;;;;;;;;;;88127:64;88211:15;:24;88227:7;88211:24;;;;;;;;;;;:30;;;;;;;;;;;;88204:37;;88031:218;;;:::o;131092:28::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;87748:124::-;87837:27;87846:2;87850:7;87859:4;87837:8;:27::i;:::-;87748:124;;:::o;131198:49::-;;;;;;;;;;;;;;;;;:::o;131653:122::-;131718:7;131745:22;131759:7;131745:13;:22::i;:::-;131738:29;;131653:122;;;:::o;77819:323::-;77880:7;78108:15;:13;:15::i;:::-;78093:12;;78077:13;;:28;:46;78070:53;;77819:323;:::o;131419:226::-;29686:13;:11;:13::i;:::-;131533:9:::1;131529:109;131552:7;:14;131548:1;:18;131529:109;;;131616:7;131624:1;131616:10;;;;;;;;:::i;:::-;;;;;;;;131587:14;:26;131602:7;131610:1;131602:10;;;;;;;;:::i;:::-;;;;;;;;131587:26;;;;;;;;;;;;;;;:39;;;;131568:3;;;;;:::i;:::-;;;;131529:109;;;;131419:226:::0;;:::o;135083:238::-;135256:5;129798:1;128602:42;129750:45;;;:49;129746:705;;;130039:10;130031:18;;:4;:18;;;130027:85;;135275:40:::1;135294:5;135301:3;135306:8;135275:18;:40::i;:::-;130090:7:::0;;130027:85;128602:42;130172;;;130223:4;130230:10;130172:69;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:161;;;;;128602:42;130270;;;130321:4;130328;130270:63;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;130172:161;130126:314;;130413:10;130394:30;;;;;;;;;;;:::i;:::-;;;;;;;;130126:314;129746:705;135275:40:::1;135294:5;135301:3;135306:8;135275:18;:40::i;:::-;135083:238:::0;;;;;:::o;69402:442::-;69499:7;69508;69528:26;69557:17;:27;69575:8;69557:27;;;;;;;;;;;69528:56;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;69629:1;69601:30;;:7;:16;;;:30;;;69597:92;;69658:19;69648:29;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;69597:92;69701:21;69766:17;:15;:17::i;:::-;69725:58;;69739:7;:23;;;69726:36;;:10;:36;;;;:::i;:::-;69725:58;;;;:::i;:::-;69701:82;;69804:7;:16;;;69822:13;69796:40;;;;;;69402:442;;;;;:::o;135327:244::-;135504:5;129798:1;128602:42;129750:45;;;:49;129746:705;;;130039:10;130031:18;;:4;:18;;;130027:85;;135521:44:::1;135544:5;135551:3;135556:8;135521:22;:44::i;:::-;130090:7:::0;;130027:85;128602:42;130172;;;130223:4;130230:10;130172:69;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:161;;;;;128602:42;130270;;;130321:4;130328;130270:63;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;130172:161;130126:314;;130413:10;130394:30;;;;;;;;;;;:::i;:::-;;;;;;;;130126:314;129746:705;135521:44:::1;135544:5;135551:3;135556:8;135521:22;:44::i;:::-;135327:244:::0;;;;;:::o;128313:94::-;128379:20;128385:7;128394:4;128379:5;:20::i;:::-;128313:94;:::o;131161:30::-;;;;;;;;;;;;;:::o;131127:27::-;;;;;;;;;;;;;:::o;133963:104::-;29686:13;:11;:13::i;:::-;134048:11:::1;134038:7;:21;;;;;;:::i;:::-;;133963:104:::0;:::o;134690:201::-;29686:13;:11;:13::i;:::-;134833:52:::1;134850:8;134860:9;134871:13;134833:16;:52::i;:::-;134690:201:::0;;;:::o;123175:528::-;123319:23;123385:22;123410:8;;:15;;123385:40;;123440:34;123498:14;123477:36;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;123440:73;;123533:9;123528:125;123549:14;123544:1;:19;123528:125;;123605:32;123625:8;;123634:1;123625:11;;;;;;;:::i;:::-;;;;;;;;123605:19;:32::i;:::-;123589:10;123600:1;123589:13;;;;;;;;:::i;:::-;;;;;;;:48;;;;123565:3;;;;;123528:125;;;;123674:10;123667:17;;;;123175:528;;;;:::o;83461:152::-;83533:7;83576:27;83595:7;83576:18;:27::i;:::-;83553:52;;83461:152;;;:::o;130947:32::-;;;;:::o;131020:21::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;131783:131::-;131856:7;131883:14;:23;131898:7;131883:23;;;;;;;;;;;;;;;;131876:30;;131783:131;;;:::o;79003:233::-;79075:7;79116:1;79099:19;;:5;:19;;;79095:60;;79127:28;;;;;;;;;;;;;;79095:60;73162:13;79173:18;:25;79192:5;79173:25;;;;;;;;;;;;;;;;:55;79166:62;;79003:233;;;:::o;30448:103::-;29686:13;:11;:13::i;:::-;30513:30:::1;30540:1;30513:18;:30::i;:::-;30448:103::o:0;127051:900::-;127129:16;127183:19;127217:25;127257:22;127282:16;127292:5;127282:9;:16::i;:::-;127257:41;;127313:25;127355:14;127341:29;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;127313:57;;127385:31;;:::i;:::-;127436:9;127448:15;:13;:15::i;:::-;127436:27;;127431:472;127480:14;127465:11;:29;127431:472;;127532:15;127545:1;127532:12;:15::i;:::-;127520:27;;127570:9;:16;;;127611:8;127566:73;127687:1;127661:28;;:9;:14;;;:28;;;127657:111;;127734:9;:14;;;127714:34;;127657:111;127811:5;127790:26;;:17;:26;;;127786:102;;127867:1;127841:8;127850:13;;;;;;127841:23;;;;;;;;:::i;:::-;;;;;;;:27;;;;;127786:102;127431:472;127496:3;;;;;127431:472;;;;127924:8;127917:15;;;;;;;127051:900;;;:::o;134897:126::-;29686:13;:11;:13::i;:::-;134990:27:::1;135009:7;134990:18;:27::i;:::-;134897:126:::0;:::o;29800:87::-;29846:7;29873:6;;;;;;;;;;;29866:13;;29800:87;:::o;131319:92::-;29686:13;:11;:13::i;:::-;131394:9:::1;131381:10;;:22;;;;;;;;;;;;;;;;;;131319:92:::0;:::o;82244:104::-;82300:13;82333:7;82326:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;82244:104;:::o;124091:2513::-;124234:16;124301:4;124292:5;:13;124288:45;;124314:19;;;;;;;;;;;;;;124288:45;124348:19;124382:17;124402:14;:12;:14::i;:::-;124382:34;;124502:15;:13;:15::i;:::-;124494:5;:23;124490:87;;;124546:15;:13;:15::i;:::-;124538:23;;124490:87;124653:9;124646:4;:16;124642:73;;;124690:9;124683:16;;124642:73;124729:25;124757:16;124767:5;124757:9;:16::i;:::-;124729:44;;124951:4;124943:5;:12;124939:278;;;124976:19;125005:5;124998:4;:12;124976:34;;125047:17;125033:11;:31;125029:111;;;125109:11;125089:31;;125029:111;124957:198;124939:278;;;125200:1;125180:21;;124939:278;125231:25;125273:17;125259:32;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;125231:60;;125331:1;125310:17;:22;125306:78;;125360:8;125353:15;;;;;;;;125306:78;125528:31;125562:26;125582:5;125562:19;:26::i;:::-;125528:60;;125603:25;125848:9;:16;;;125843:92;;125905:9;:14;;;125885:34;;125843:92;125954:9;125966:5;125954:17;;125949:478;125978:4;125973:1;:9;;:45;;;;;126001:17;125986:11;:32;;125973:45;125949:478;;;126056:15;126069:1;126056:12;:15::i;:::-;126044:27;;126094:9;:16;;;126135:8;126090:73;126211:1;126185:28;;:9;:14;;;:28;;;126181:111;;126258:9;:14;;;126238:34;;126181:111;126335:5;126314:26;;:17;:26;;;126310:102;;126391:1;126365:8;126374:13;;;;;;126365:23;;;;;;;;:::i;:::-;;;;;;;:27;;;;;126310:102;125949:478;126020:3;;;;;125949:478;;;;126529:11;126519:8;126512:29;126577:8;126570:15;;;;;;;;124091:2513;;;;;;:::o;131922:661::-;131991:10;;;;;;;;;;;131983:48;;;;;;;;;;;;:::i;:::-;;;;;;;;;132079:1;132050:14;:26;132065:10;132050:26;;;;;;;;;;;;;;;;:30;132042:62;;;;;;;;;;;;:::i;:::-;;;;;;;;;132136:14;:26;132151:10;132136:26;;;;;;;;;;;;;;;;132123:9;:39;;132115:94;;;;;;;;;;;;:::i;:::-;;;;;;;;;130895:2;132244:9;132228:13;:11;:13::i;:::-;:25;;;;:::i;:::-;:38;;132220:81;;;;;;;;;;;;:::i;:::-;;;;;;;;;132361:14;:26;132376:10;132361:26;;;;;;;;;;;;;;;;132348:9;132320:25;132334:10;132320:13;:25::i;:::-;:37;;;;:::i;:::-;:67;;132312:123;;;;;;;;;;;;:::i;:::-;;;;;;;;;132479:9;132466;130933:7;132454:21;;;;:::i;:::-;:34;;132446:86;;;;;;;;;;;;:::i;:::-;;;;;;;;;132543:32;132553:10;132565:9;132543;:32::i;:::-;131922:661;:::o;88589:234::-;88736:8;88684:18;:39;88703:19;:17;:19::i;:::-;88684:39;;;;;;;;;;;;;;;:49;88724:8;88684:49;;;;;;;;;;;;;;;;:60;;;;;;;;;;;;;;;;;;88796:8;88760:55;;88775:19;:17;:19::i;:::-;88760:55;;;88806:8;88760:55;;;;;;:::i;:::-;;;;;;;;88589:234;;:::o;134327:69::-;29686:13;:11;:13::i;:::-;134384:4:::1;134373:8;;:15;;;;;;;;;;;;;;;;;;134327:69::o:0;134584:100::-;29686:13;:11;:13::i;:::-;134655:23:::1;:21;:23::i;:::-;134584:100::o:0;135860:178::-;29686:13;:11;:13::i;:::-;135916:12:::1;135934:10;:15;;135957:21;135934:49;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;135915:68;;;136002:7;135994:36;;;;;;;;;;;;:::i;:::-;;;;;;;;;135904:134;135860:178::o:0;135577:275::-;135778:5;129798:1;128602:42;129750:45;;;:49;129746:705;;;130039:10;130031:18;;:4;:18;;;130027:85;;135795:51:::1;135818:5;135825:3;135830:8;135840:5;135795:22;:51::i;:::-;130090:7:::0;;130027:85;128602:42;130172;;;130223:4;130230:10;130172:69;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:161;;;;;128602:42;130270;;;130321:4;130328;130270:63;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;130172:161;130126:314;;130413:10;130394:30;;;;;;;;;;;:::i;:::-;;;;;;;;130126:314;129746:705;135795:51:::1;135818:5;135825:3;135830:8;135840:5;135795:22;:51::i;:::-;135577:275:::0;;;;;;:::o;122588:428::-;122672:21;;:::i;:::-;122706:31;;:::i;:::-;122762:15;:13;:15::i;:::-;122752:7;:25;:54;;;;122792:14;:12;:14::i;:::-;122781:7;:25;;122752:54;122748:103;;;122830:9;122823:16;;;;;122748:103;122873:21;122886:7;122873:12;:21::i;:::-;122861:33;;122909:9;:16;;;122905:65;;;122949:9;122942:16;;;;;122905:65;122987:21;123000:7;122987:12;:21::i;:::-;122980:28;;;122588:428;;;;:::o;131048:37::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;132813:584::-;132950:13;132999:16;133007:7;132999;:16::i;:::-;132981:105;;;;;;;;;;;;:::i;:::-;;;;;;;;;133122:5;133110:17;;:8;;;;;;;;;;;:17;;;133107:70;;133151:14;133144:21;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;133107:70;133189:28;133220:10;:8;:10::i;:::-;133189:41;;133279:1;133254:14;133248:28;:32;:141;;;;;;;;;;;;;;;;;133320:14;133336:18;:7;:16;:18::i;:::-;133356:13;133303:67;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;133248:141;133241:148;;;132813:584;;;;:::o;132591:214::-;29686:13;:11;:13::i;:::-;130895:2:::1;132699:9;132683:13;:11;:13::i;:::-;:25;;;;:::i;:::-;:38;;132675:81;;;;;;;;;;;;:::i;:::-;;;;;;;;;132767:30;132777:8;132787:9;132767;:30::i;:::-;132591:214:::0;;:::o;134075:128::-;29686:13;:11;:13::i;:::-;134178:17:::1;134162:13;:33;;;;;;:::i;:::-;;134075:128:::0;:::o;88980:164::-;89077:4;89101:18;:25;89120:5;89101:25;;;;;;;;;;;;;;;:35;89127:8;89101:35;;;;;;;;;;;;;;;;;;;;;;;;;89094:42;;88980:164;;;;:::o;133829:126::-;29686:13;:11;:13::i;:::-;133932:15:::1;133915:14;:32;;;;;;:::i;:::-;;133829:126:::0;:::o;30706:201::-;29686:13;:11;:13::i;:::-;30815:1:::1;30795:22;;:8;:22;;::::0;30787:73:::1;;;;;;;;;;;;:::i;:::-;;;;;;;;;30871:28;30890:8;30871:18;:28::i;:::-;30706:201:::0;:::o;130986:27::-;;;;:::o;69132:215::-;69234:4;69273:26;69258:41;;;:11;:41;;;;:81;;;;69303:36;69327:11;69303:23;:36::i;:::-;69258:81;69251:88;;69132:215;;;:::o;81166:639::-;81251:4;81590:10;81575:25;;:11;:25;;;;:102;;;;81667:10;81652:25;;:11;:25;;;;81575:102;:179;;;;81744:10;81729:25;;:11;:25;;;;81575:179;81555:199;;81166:639;;;:::o;29965:132::-;30040:12;:10;:12::i;:::-;30029:23;;:7;:5;:7::i;:::-;:23;;;30021:68;;;;;;;;;;;;:::i;:::-;;;;;;;;;29965:132::o;70494:332::-;70613:17;:15;:17::i;:::-;70597:33;;:12;:33;;;;70589:88;;;;;;;;;;;;:::i;:::-;;;;;;;;;70716:1;70696:22;;:8;:22;;;70688:60;;;;;;;;;;;;:::i;:::-;;;;;;;;;70783:35;;;;;;;;70795:8;70783:35;;;;;;70805:12;70783:35;;;;;70761:19;:57;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70494:332;;:::o;89402:282::-;89467:4;89523:7;89504:15;:13;:15::i;:::-;:26;;:66;;;;;89557:13;;89547:7;:23;89504:66;:153;;;;;89656:1;73938:8;89608:17;:26;89626:7;89608:26;;;;;;;;;;;;:44;:49;89504:153;89484:173;;89402:282;;;:::o;106462:431::-;106557:13;106573:16;106581:7;106573;:16::i;:::-;106557:32;;106606:13;:45;;;;;106646:5;106623:28;;:19;:17;:19::i;:::-;:28;;;;106606:45;106602:192;;;106671:44;106688:5;106695:19;:17;:19::i;:::-;106671:16;:44::i;:::-;106666:128;;106743:35;;;;;;;;;;;;;;106666:128;106602:192;106839:2;106806:15;:24;106822:7;106806:24;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;106877:7;106873:2;106857:28;;106866:5;106857:28;;;;;;;;;;;;106546:347;106462:431;;;:::o;79318:178::-;79379:7;73162:13;73300:2;79407:18;:25;79426:5;79407:25;;;;;;;;;;;;;;;;:50;;79406:82;79399:89;;79318:178;;;:::o;133716:101::-;133781:7;133808:1;133801:8;;133716:101;:::o;91670:2825::-;91812:27;91842;91861:7;91842:18;:27::i;:::-;91812:57;;91927:4;91886:45;;91902:19;91886:45;;;91882:86;;91940:28;;;;;;;;;;;;;;91882:86;91982:27;92011:23;92038:35;92065:7;92038:26;:35::i;:::-;91981:92;;;;92173:68;92198:15;92215:4;92221:19;:17;:19::i;:::-;92173:24;:68::i;:::-;92168:180;;92261:43;92278:4;92284:19;:17;:19::i;:::-;92261:16;:43::i;:::-;92256:92;;92313:35;;;;;;;;;;;;;;92256:92;92168:180;92379:1;92365:16;;:2;:16;;;92361:52;;92390:23;;;;;;;;;;;;;;92361:52;92426:43;92448:4;92454:2;92458:7;92467:1;92426:21;:43::i;:::-;92562:15;92559:160;;;92702:1;92681:19;92674:30;92559:160;93099:18;:24;93118:4;93099:24;;;;;;;;;;;;;;;;93097:26;;;;;;;;;;;;93168:18;:22;93187:2;93168:22;;;;;;;;;;;;;;;;93166:24;;;;;;;;;;;93490:146;93527:2;93576:45;93591:4;93597:2;93601:19;93576:14;:45::i;:::-;74218:8;93548:73;93490:18;:146::i;:::-;93461:17;:26;93479:7;93461:26;;;;;;;;;;;:175;;;;93807:1;74218:8;93756:19;:47;:52;93752:627;;93829:19;93861:1;93851:7;:11;93829:33;;94018:1;93984:17;:30;94002:11;93984:30;;;;;;;;;;;;:35;93980:384;;94122:13;;94107:11;:28;94103:242;;94302:19;94269:17;:30;94287:11;94269:30;;;;;;;;;;;:52;;;;94103:242;93980:384;93810:569;93752:627;94426:7;94422:2;94407:27;;94416:4;94407:27;;;;;;;;;;;;94445:42;94466:4;94472:2;94476:7;94485:1;94445:20;:42::i;:::-;91801:2694;;;91670:2825;;;:::o;70126:97::-;70184:6;70210:5;70203:12;;70126:97;:::o;94591:193::-;94737:39;94754:4;94760:2;94764:7;94737:39;;;;;;;;;;;;:16;:39::i;:::-;94591:193;;;:::o;107478:3081::-;107558:27;107588;107607:7;107588:18;:27::i;:::-;107558:57;;107628:12;107659:19;107628:52;;107694:27;107723:23;107750:35;107777:7;107750:26;:35::i;:::-;107693:92;;;;107802:13;107798:316;;;107923:68;107948:15;107965:4;107971:19;:17;:19::i;:::-;107923:24;:68::i;:::-;107918:184;;108015:43;108032:4;108038:19;:17;:19::i;:::-;108015:16;:43::i;:::-;108010:92;;108067:35;;;;;;;;;;;;;;108010:92;107918:184;107798:316;108126:51;108148:4;108162:1;108166:7;108175:1;108126:21;:51::i;:::-;108270:15;108267:160;;;108410:1;108389:19;108382:30;108267:160;109088:1;73427:3;109058:1;:26;;109057:32;109029:18;:24;109048:4;109029:24;;;;;;;;;;;;;;;;:60;;;;;;;;;;;109356:176;109393:4;109464:53;109479:4;109493:1;109497:19;109464:14;:53::i;:::-;74218:8;73938;109417:43;109416:101;109356:18;:176::i;:::-;109327:17;:26;109345:7;109327:26;;;;;;;;;;;:205;;;;109703:1;74218:8;109652:19;:47;:52;109648:627;;109725:19;109757:1;109747:7;:11;109725:33;;109914:1;109880:17;:30;109898:11;109880:30;;;;;;;;;;;;:35;109876:384;;110018:13;;110003:11;:28;109999:242;;110198:19;110165:17;:30;110183:11;110165:30;;;;;;;;;;;:52;;;;109999:242;109876:384;109706:569;109648:627;110330:7;110326:1;110303:35;;110312:4;110303:35;;;;;;;;;;;;110349:50;110370:4;110384:1;110388:7;110397:1;110349:20;:50::i;:::-;110526:12;;:14;;;;;;;;;;;;;107547:3012;;;;107478:3081;;:::o;71277:390::-;71445:17;:15;:17::i;:::-;71429:33;;:12;:33;;;;71421:88;;;;;;;;;;;;:::i;:::-;;;;;;;;;71548:1;71528:22;;:8;:22;;;71520:62;;;;;;;;;;;;:::i;:::-;;;;;;;;;71624:35;;;;;;;;71636:8;71624:35;;;;;;71646:12;71624:35;;;;;71595:17;:26;71613:7;71595:26;;;;;;;;;;;:64;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;71277:390;;;:::o;84616:1275::-;84683:7;84703:12;84718:7;84703:22;;84786:4;84767:15;:13;:15::i;:::-;:23;84763:1061;;84820:13;;84813:4;:20;84809:1015;;;84858:14;84875:17;:23;84893:4;84875:23;;;;;;;;;;;;84858:40;;84992:1;73938:8;84964:6;:24;:29;84960:845;;85629:113;85646:1;85636:6;:11;85629:113;;85689:17;:25;85707:6;;;;;;;85689:25;;;;;;;;;;;;85680:34;;85629:113;;;85775:6;85768:13;;;;;;84960:845;84835:989;84809:1015;84763:1061;85852:31;;;;;;;;;;;;;;84616:1275;;;;:::o;31067:191::-;31141:16;31160:6;;;;;;;;;;;31141:25;;31186:8;31177:6;;:17;;;;;;;;;;;;;;;;;;31241:8;31210:40;;31231:8;31210:40;;;;;;;;;;;;31130:128;31067:191;:::o;84064:161::-;84132:21;;:::i;:::-;84173:44;84192:17;:24;84210:5;84192:24;;;;;;;;;;;;84173:18;:44::i;:::-;84166:51;;84064:161;;;:::o;71778:114::-;71858:17;:26;71876:7;71858:26;;;;;;;;;;;;71851:33;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;71778:114;:::o;77506:103::-;77561:7;77588:13;;77581:20;;77506:103;:::o;105542:112::-;105619:27;105629:2;105633:8;105619:27;;;;;;;;;;;;:9;:27::i;:::-;105542:112;;:::o;112949:105::-;113009:7;113036:10;113029:17;;112949:105;:::o;70902:95::-;70970:19;;70963:26;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70902:95::o;95382:407::-;95557:31;95570:4;95576:2;95580:7;95557:12;:31::i;:::-;95621:1;95603:2;:14;;;:19;95599:183;;95642:56;95673:4;95679:2;95683:7;95692:5;95642:30;:56::i;:::-;95637:145;;95726:40;;;;;;;;;;;;;;95637:145;95599:183;95382:407;;;;:::o;83802:166::-;83872:21;;:::i;:::-;83913:47;83932:27;83951:7;83932:18;:27::i;:::-;83913:18;:47::i;:::-;83906:54;;83802:166;;;:::o;134211:108::-;134271:13;134304:7;134297:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;134211:108;:::o;23207:716::-;23263:13;23314:14;23351:1;23331:17;23342:5;23331:10;:17::i;:::-;:21;23314:38;;23367:20;23401:6;23390:18;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;23367:41;;23423:11;23552:6;23548:2;23544:15;23536:6;23532:28;23525:35;;23589:288;23596:4;23589:288;;;23621:5;;;;;;;;23763:8;23758:2;23751:5;23747:14;23742:30;23737:3;23729:44;23819:2;23810:11;;;;;;:::i;:::-;;;;;23853:1;23844:5;:10;23589:288;23840:21;23589:288;23898:6;23891:13;;;;;23207:716;;;:::o;31963:157::-;32048:4;32087:25;32072:40;;;:11;:40;;;;32065:47;;31963:157;;;:::o;28509:98::-;28562:7;28589:10;28582:17;;28509:98;:::o;90565:485::-;90667:27;90696:23;90737:38;90778:15;:24;90794:7;90778:24;;;;;;;;;;;90737:65;;90955:18;90932:41;;91012:19;91006:26;90987:45;;90917:126;90565:485;;;:::o;89793:659::-;89942:11;90107:16;90100:5;90096:28;90087:37;;90267:16;90256:9;90252:32;90239:45;;90417:15;90406:9;90403:30;90395:5;90384:9;90381:20;90378:56;90368:66;;89793:659;;;;;:::o;96451:159::-;;;;;:::o;112258:311::-;112393:7;112413:16;74342:3;112439:19;:41;;112413:68;;74342:3;112507:31;112518:4;112524:2;112528:9;112507:10;:31::i;:::-;112499:40;;:62;;112492:69;;;112258:311;;;;;:::o;86439:450::-;86519:14;86687:16;86680:5;86676:28;86667:37;;86864:5;86850:11;86825:23;86821:41;86818:52;86811:5;86808:63;86798:73;;86439:450;;;;:::o;97275:158::-;;;;;:::o;85990:366::-;86056:31;;:::i;:::-;86133:6;86100:9;:14;;:41;;;;;;;;;;;73821:3;86186:6;:33;;86152:9;:24;;:68;;;;;;;;;;;86278:1;73938:8;86250:6;:24;:29;;86231:9;:16;;:48;;;;;;;;;;;74342:3;86319:6;:28;;86290:9;:19;;:58;;;;;;;;;;;85990:366;;;:::o;104769:689::-;104900:19;104906:2;104910:8;104900:5;:19::i;:::-;104979:1;104961:2;:14;;;:19;104957:483;;105001:11;105015:13;;105001:27;;105047:13;105069:8;105063:3;:14;105047:30;;105096:233;105127:62;105166:1;105170:2;105174:7;;;;;;105183:5;105127:30;:62::i;:::-;105122:167;;105225:40;;;;;;;;;;;;;;105122:167;105324:3;105316:5;:11;105096:233;;105411:3;105394:13;;:20;105390:34;;105416:8;;;105390:34;104982:458;;104957:483;104769:689;;;:::o;97873:716::-;98036:4;98082:2;98057:45;;;98103:19;:17;:19::i;:::-;98124:4;98130:7;98139:5;98057:88;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;98053:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;98357:1;98340:6;:13;:18;98336:235;;98386:40;;;;;;;;;;;;;;98336:235;98529:6;98523:13;98514:6;98510:2;98506:15;98499:38;98053:529;98226:54;;;98216:64;;;:6;:64;;;;98209:71;;;97873:716;;;;;;:::o;10083:922::-;10136:7;10156:14;10173:1;10156:18;;10223:6;10214:5;:15;10210:102;;10259:6;10250:15;;;;;;:::i;:::-;;;;;10294:2;10284:12;;;;10210:102;10339:6;10330:5;:15;10326:102;;10375:6;10366:15;;;;;;:::i;:::-;;;;;10410:2;10400:12;;;;10326:102;10455:6;10446:5;:15;10442:102;;10491:6;10482:15;;;;;;:::i;:::-;;;;;10526:2;10516:12;;;;10442:102;10571:5;10562;:14;10558:99;;10606:5;10597:14;;;;;;:::i;:::-;;;;;10640:1;10630:11;;;;10558:99;10684:5;10675;:14;10671:99;;10719:5;10710:14;;;;;;:::i;:::-;;;;;10753:1;10743:11;;;;10671:99;10797:5;10788;:14;10784:99;;10832:5;10823:14;;;;;;:::i;:::-;;;;;10866:1;10856:11;;;;10784:99;10910:5;10901;:14;10897:66;;10946:1;10936:11;;;;10897:66;10991:6;10984:13;;;10083:922;;;:::o;111959:147::-;112096:6;111959:147;;;;;:::o;99051:2966::-;99124:20;99147:13;;99124:36;;99187:1;99175:8;:13;99171:44;;99197:18;;;;;;;;;;;;;;99171:44;99228:61;99258:1;99262:2;99266:12;99280:8;99228:21;:61::i;:::-;99772:1;73300:2;99742:1;:26;;99741:32;99729:8;:45;99703:18;:22;99722:2;99703:22;;;;;;;;;;;;;;;;:71;;;;;;;;;;;100051:139;100088:2;100142:33;100165:1;100169:2;100173:1;100142:14;:33::i;:::-;100109:30;100130:8;100109:20;:30::i;:::-;:66;100051:18;:139::i;:::-;100017:17;:31;100035:12;100017:31;;;;;;;;;;;:173;;;;100207:16;100238:11;100267:8;100252:12;:23;100238:37;;100788:16;100784:2;100780:25;100768:37;;101160:12;101120:8;101079:1;101017:25;100958:1;100897;100870:335;101531:1;101517:12;101513:20;101471:346;101572:3;101563:7;101560:16;101471:346;;101790:7;101780:8;101777:1;101750:25;101747:1;101744;101739:59;101625:1;101616:7;101612:15;101601:26;;101471:346;;;101475:77;101862:1;101850:8;:13;101846:45;;101872:19;;;;;;;;;;;;;;101846:45;101924:3;101908:13;:19;;;;99477:2462;;101949:60;101978:1;101982:2;101986:12;102000:8;101949:20;:60::i;:::-;99113:2904;99051:2966;;:::o;86991:324::-;87061:14;87294:1;87284:8;87281:15;87255:24;87251:46;87241:56;;86991:324;;;:::o;-1:-1:-1:-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::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:126::-;1555:7;1595:42;1588:5;1584:54;1573:65;;1518:126;;;:::o;1650:96::-;1687:7;1716:24;1734:5;1716:24;:::i;:::-;1705:35;;1650:96;;;:::o;1752:122::-;1825:24;1843:5;1825:24;:::i;:::-;1818:5;1815:35;1805:63;;1864:1;1861;1854:12;1805:63;1752:122;:::o;1880:139::-;1926:5;1964:6;1951:20;1942:29;;1980:33;2007:5;1980:33;:::i;:::-;1880:139;;;;:::o;2025:109::-;2061:7;2101:26;2094:5;2090:38;2079:49;;2025:109;;;:::o;2140:120::-;2212:23;2229:5;2212:23;:::i;:::-;2205:5;2202:34;2192:62;;2250:1;2247;2240:12;2192:62;2140:120;:::o;2266:137::-;2311:5;2349:6;2336:20;2327:29;;2365:32;2391:5;2365:32;:::i;:::-;2266:137;;;;:::o;2409:472::-;2476:6;2484;2533:2;2521:9;2512:7;2508:23;2504:32;2501:119;;;2539:79;;:::i;:::-;2501:119;2659:1;2684:53;2729:7;2720:6;2709:9;2705:22;2684:53;:::i;:::-;2674:63;;2630:117;2786:2;2812:52;2856:7;2847:6;2836:9;2832:22;2812:52;:::i;:::-;2802:62;;2757:117;2409:472;;;;;:::o;2887:99::-;2939:6;2973:5;2967:12;2957:22;;2887:99;;;:::o;2992:169::-;3076:11;3110:6;3105:3;3098:19;3150:4;3145:3;3141:14;3126:29;;2992:169;;;;:::o;3167:246::-;3248:1;3258:113;3272:6;3269:1;3266:13;3258:113;;;3357:1;3352:3;3348:11;3342:18;3338:1;3333:3;3329:11;3322:39;3294:2;3291:1;3287:10;3282:15;;3258:113;;;3405:1;3396:6;3391:3;3387:16;3380:27;3229:184;3167:246;;;:::o;3419:102::-;3460:6;3511:2;3507:7;3502:2;3495:5;3491:14;3487:28;3477:38;;3419:102;;;:::o;3527:377::-;3615:3;3643:39;3676:5;3643:39;:::i;:::-;3698:71;3762:6;3757:3;3698:71;:::i;:::-;3691:78;;3778:65;3836:6;3831:3;3824:4;3817:5;3813:16;3778:65;:::i;:::-;3868:29;3890:6;3868:29;:::i;:::-;3863:3;3859:39;3852:46;;3619:285;3527:377;;;;:::o;3910:313::-;4023:4;4061:2;4050:9;4046:18;4038:26;;4110:9;4104:4;4100:20;4096:1;4085:9;4081:17;4074:47;4138:78;4211:4;4202:6;4138:78;:::i;:::-;4130:86;;3910:313;;;;:::o;4229:77::-;4266:7;4295:5;4284:16;;4229:77;;;:::o;4312:122::-;4385:24;4403:5;4385:24;:::i;:::-;4378:5;4375:35;4365:63;;4424:1;4421;4414:12;4365:63;4312:122;:::o;4440:139::-;4486:5;4524:6;4511:20;4502:29;;4540:33;4567:5;4540:33;:::i;:::-;4440:139;;;;:::o;4585:329::-;4644:6;4693:2;4681:9;4672:7;4668:23;4664:32;4661:119;;;4699:79;;:::i;:::-;4661:119;4819:1;4844:53;4889:7;4880:6;4869:9;4865:22;4844:53;:::i;:::-;4834:63;;4790:117;4585:329;;;;:::o;4920:118::-;5007:24;5025:5;5007:24;:::i;:::-;5002:3;4995:37;4920:118;;:::o;5044:222::-;5137:4;5175:2;5164:9;5160:18;5152:26;;5188:71;5256:1;5245:9;5241:17;5232:6;5188:71;:::i;:::-;5044:222;;;;:::o;5272:474::-;5340:6;5348;5397:2;5385:9;5376:7;5372:23;5368:32;5365:119;;;5403:79;;:::i;:::-;5365:119;5523:1;5548:53;5593:7;5584:6;5573:9;5569:22;5548:53;:::i;:::-;5538:63;;5494:117;5650:2;5676:53;5721:7;5712:6;5701:9;5697:22;5676:53;:::i;:::-;5666:63;;5621:118;5272:474;;;;;:::o;5752:329::-;5811:6;5860:2;5848:9;5839:7;5835:23;5831:32;5828:119;;;5866:79;;:::i;:::-;5828:119;5986:1;6011:53;6056:7;6047:6;6036:9;6032:22;6011:53;:::i;:::-;6001:63;;5957:117;5752:329;;;;:::o;6087:118::-;6174:24;6192:5;6174:24;:::i;:::-;6169:3;6162:37;6087:118;;:::o;6211:222::-;6304:4;6342:2;6331:9;6327:18;6319:26;;6355:71;6423:1;6412:9;6408:17;6399:6;6355:71;:::i;:::-;6211:222;;;;:::o;6439:117::-;6548:1;6545;6538:12;6562:180;6610:77;6607:1;6600:88;6707:4;6704:1;6697:15;6731:4;6728:1;6721:15;6748:281;6831:27;6853:4;6831:27;:::i;:::-;6823:6;6819:40;6961:6;6949:10;6946:22;6925:18;6913:10;6910:34;6907:62;6904:88;;;6972:18;;:::i;:::-;6904:88;7012:10;7008:2;7001:22;6791:238;6748:281;;:::o;7035:129::-;7069:6;7096:20;;:::i;:::-;7086:30;;7125:33;7153:4;7145:6;7125:33;:::i;:::-;7035:129;;;:::o;7170:311::-;7247:4;7337:18;7329:6;7326:30;7323:56;;;7359:18;;:::i;:::-;7323:56;7409:4;7401:6;7397:17;7389:25;;7469:4;7463;7459:15;7451:23;;7170:311;;;:::o;7487:117::-;7596:1;7593;7586:12;7627:710;7723:5;7748:81;7764:64;7821:6;7764:64;:::i;:::-;7748:81;:::i;:::-;7739:90;;7849:5;7878:6;7871:5;7864:21;7912:4;7905:5;7901:16;7894:23;;7965:4;7957:6;7953:17;7945:6;7941:30;7994:3;7986:6;7983:15;7980:122;;;8013:79;;:::i;:::-;7980:122;8128:6;8111:220;8145:6;8140:3;8137:15;8111:220;;;8220:3;8249:37;8282:3;8270:10;8249:37;:::i;:::-;8244:3;8237:50;8316:4;8311:3;8307:14;8300:21;;8187:144;8171:4;8166:3;8162:14;8155:21;;8111:220;;;8115:21;7729:608;;7627:710;;;;;:::o;8360:370::-;8431:5;8480:3;8473:4;8465:6;8461:17;8457:27;8447:122;;8488:79;;:::i;:::-;8447:122;8605:6;8592:20;8630:94;8720:3;8712:6;8705:4;8697:6;8693:17;8630:94;:::i;:::-;8621:103;;8437:293;8360:370;;;;:::o;8736:311::-;8813:4;8903:18;8895:6;8892:30;8889:56;;;8925:18;;:::i;:::-;8889:56;8975:4;8967:6;8963:17;8955:25;;9035:4;9029;9025:15;9017:23;;8736:311;;;:::o;9070:710::-;9166:5;9191:81;9207:64;9264:6;9207:64;:::i;:::-;9191:81;:::i;:::-;9182:90;;9292:5;9321:6;9314:5;9307:21;9355:4;9348:5;9344:16;9337:23;;9408:4;9400:6;9396:17;9388:6;9384:30;9437:3;9429:6;9426:15;9423:122;;;9456:79;;:::i;:::-;9423:122;9571:6;9554:220;9588:6;9583:3;9580:15;9554:220;;;9663:3;9692:37;9725:3;9713:10;9692:37;:::i;:::-;9687:3;9680:50;9759:4;9754:3;9750:14;9743:21;;9630:144;9614:4;9609:3;9605:14;9598:21;;9554:220;;;9558:21;9172:608;;9070:710;;;;;:::o;9803:370::-;9874:5;9923:3;9916:4;9908:6;9904:17;9900:27;9890:122;;9931:79;;:::i;:::-;9890:122;10048:6;10035:20;10073:94;10163:3;10155:6;10148:4;10140:6;10136:17;10073:94;:::i;:::-;10064:103;;9880:293;9803:370;;;;:::o;10179:894::-;10297:6;10305;10354:2;10342:9;10333:7;10329:23;10325:32;10322:119;;;10360:79;;:::i;:::-;10322:119;10508:1;10497:9;10493:17;10480:31;10538:18;10530:6;10527:30;10524:117;;;10560:79;;:::i;:::-;10524:117;10665:78;10735:7;10726:6;10715:9;10711:22;10665:78;:::i;:::-;10655:88;;10451:302;10820:2;10809:9;10805:18;10792:32;10851:18;10843:6;10840:30;10837:117;;;10873:79;;:::i;:::-;10837:117;10978:78;11048:7;11039:6;11028:9;11024:22;10978:78;:::i;:::-;10968:88;;10763:303;10179:894;;;;;:::o;11079:619::-;11156:6;11164;11172;11221:2;11209:9;11200:7;11196:23;11192:32;11189:119;;;11227:79;;:::i;:::-;11189:119;11347:1;11372:53;11417:7;11408:6;11397:9;11393:22;11372:53;:::i;:::-;11362:63;;11318:117;11474:2;11500:53;11545:7;11536:6;11525:9;11521:22;11500:53;:::i;:::-;11490:63;;11445:118;11602:2;11628:53;11673:7;11664:6;11653:9;11649:22;11628:53;:::i;:::-;11618:63;;11573:118;11079:619;;;;;:::o;11704:474::-;11772:6;11780;11829:2;11817:9;11808:7;11804:23;11800:32;11797:119;;;11835:79;;:::i;:::-;11797:119;11955:1;11980:53;12025:7;12016:6;12005:9;12001:22;11980:53;:::i;:::-;11970:63;;11926:117;12082:2;12108:53;12153:7;12144:6;12133:9;12129:22;12108:53;:::i;:::-;12098:63;;12053:118;11704:474;;;;;:::o;12184:332::-;12305:4;12343:2;12332:9;12328:18;12320:26;;12356:71;12424:1;12413:9;12409:17;12400:6;12356:71;:::i;:::-;12437:72;12505:2;12494:9;12490:18;12481:6;12437:72;:::i;:::-;12184:332;;;;;:::o;12522:117::-;12631:1;12628;12621:12;12645:308;12707:4;12797:18;12789:6;12786:30;12783:56;;;12819:18;;:::i;:::-;12783:56;12857:29;12879:6;12857:29;:::i;:::-;12849:37;;12941:4;12935;12931:15;12923:23;;12645:308;;;:::o;12959:146::-;13056:6;13051:3;13046;13033:30;13097:1;13088:6;13083:3;13079:16;13072:27;12959:146;;;:::o;13111:425::-;13189:5;13214:66;13230:49;13272:6;13230:49;:::i;:::-;13214:66;:::i;:::-;13205:75;;13303:6;13296:5;13289:21;13341:4;13334:5;13330:16;13379:3;13370:6;13365:3;13361:16;13358:25;13355:112;;;13386:79;;:::i;:::-;13355:112;13476:54;13523:6;13518:3;13513;13476:54;:::i;:::-;13195:341;13111:425;;;;;:::o;13556:340::-;13612:5;13661:3;13654:4;13646:6;13642:17;13638:27;13628:122;;13669:79;;:::i;:::-;13628:122;13786:6;13773:20;13811:79;13886:3;13878:6;13871:4;13863:6;13859:17;13811:79;:::i;:::-;13802:88;;13618:278;13556:340;;;;:::o;13902:509::-;13971:6;14020:2;14008:9;13999:7;13995:23;13991:32;13988:119;;;14026:79;;:::i;:::-;13988:119;14174:1;14163:9;14159:17;14146:31;14204:18;14196:6;14193:30;14190:117;;;14226:79;;:::i;:::-;14190:117;14331:63;14386:7;14377:6;14366:9;14362:22;14331:63;:::i;:::-;14321:73;;14117:287;13902:509;;;;:::o;14417:617::-;14493:6;14501;14509;14558:2;14546:9;14537:7;14533:23;14529:32;14526:119;;;14564:79;;:::i;:::-;14526:119;14684:1;14709:53;14754:7;14745:6;14734:9;14730:22;14709:53;:::i;:::-;14699:63;;14655:117;14811:2;14837:53;14882:7;14873:6;14862:9;14858:22;14837:53;:::i;:::-;14827:63;;14782:118;14939:2;14965:52;15009:7;15000:6;14989:9;14985:22;14965:52;:::i;:::-;14955:62;;14910:117;14417:617;;;;;:::o;15040:117::-;15149:1;15146;15139:12;15180:568;15253:8;15263:6;15313:3;15306:4;15298:6;15294:17;15290:27;15280:122;;15321:79;;:::i;:::-;15280:122;15434:6;15421:20;15411:30;;15464:18;15456:6;15453:30;15450:117;;;15486:79;;:::i;:::-;15450:117;15600:4;15592:6;15588:17;15576:29;;15654:3;15646:4;15638:6;15634:17;15624:8;15620:32;15617:41;15614:128;;;15661:79;;:::i;:::-;15614:128;15180:568;;;;;:::o;15754:559::-;15840:6;15848;15897:2;15885:9;15876:7;15872:23;15868:32;15865:119;;;15903:79;;:::i;:::-;15865:119;16051:1;16040:9;16036:17;16023:31;16081:18;16073:6;16070:30;16067:117;;;16103:79;;:::i;:::-;16067:117;16216:80;16288:7;16279:6;16268:9;16264:22;16216:80;:::i;:::-;16198:98;;;;15994:312;15754:559;;;;;:::o;16319:146::-;16418:6;16452:5;16446:12;16436:22;;16319:146;;;:::o;16471:216::-;16602:11;16636:6;16631:3;16624:19;16676:4;16671:3;16667:14;16652:29;;16471:216;;;;:::o;16693:164::-;16792:4;16815:3;16807:11;;16845:4;16840:3;16836:14;16828:22;;16693:164;;;:::o;16863:108::-;16940:24;16958:5;16940:24;:::i;:::-;16935:3;16928:37;16863:108;;:::o;16977:101::-;17013:7;17053:18;17046:5;17042:30;17031:41;;16977:101;;;:::o;17084:105::-;17159:23;17176:5;17159:23;:::i;:::-;17154:3;17147:36;17084:105;;:::o;17195:99::-;17266:21;17281:5;17266:21;:::i;:::-;17261:3;17254:34;17195:99;;:::o;17300:91::-;17336:7;17376:8;17369:5;17365:20;17354:31;;17300:91;;;:::o;17397:105::-;17472:23;17489:5;17472:23;:::i;:::-;17467:3;17460:36;17397:105;;:::o;17580:866::-;17731:4;17726:3;17722:14;17818:4;17811:5;17807:16;17801:23;17837:63;17894:4;17889:3;17885:14;17871:12;17837:63;:::i;:::-;17746:164;18002:4;17995:5;17991:16;17985:23;18021:61;18076:4;18071:3;18067:14;18053:12;18021:61;:::i;:::-;17920:172;18176:4;18169:5;18165:16;18159:23;18195:57;18246:4;18241:3;18237:14;18223:12;18195:57;:::i;:::-;18102:160;18349:4;18342:5;18338:16;18332:23;18368:61;18423:4;18418:3;18414:14;18400:12;18368:61;:::i;:::-;18272:167;17700:746;17580:866;;:::o;18452:307::-;18585:10;18606:110;18712:3;18704:6;18606:110;:::i;:::-;18748:4;18743:3;18739:14;18725:28;;18452:307;;;;:::o;18765:145::-;18867:4;18899;18894:3;18890:14;18882:22;;18765:145;;;:::o;18992:988::-;19175:3;19204:86;19284:5;19204:86;:::i;:::-;19306:118;19417:6;19412:3;19306:118;:::i;:::-;19299:125;;19448:88;19530:5;19448:88;:::i;:::-;19559:7;19590:1;19575:380;19600:6;19597:1;19594:13;19575:380;;;19676:6;19670:13;19703:127;19826:3;19811:13;19703:127;:::i;:::-;19696:134;;19853:92;19938:6;19853:92;:::i;:::-;19843:102;;19635:320;19622:1;19619;19615:9;19610:14;;19575:380;;;19579:14;19971:3;19964:10;;19180:800;;;18992:988;;;;:::o;19986:501::-;20193:4;20231:2;20220:9;20216:18;20208:26;;20280:9;20274:4;20270:20;20266:1;20255:9;20251:17;20244:47;20308:172;20475:4;20466:6;20308:172;:::i;:::-;20300:180;;19986:501;;;;:::o;20493:114::-;20560:6;20594:5;20588:12;20578:22;;20493:114;;;:::o;20613:184::-;20712:11;20746:6;20741:3;20734:19;20786:4;20781:3;20777:14;20762:29;;20613:184;;;;:::o;20803:132::-;20870:4;20893:3;20885:11;;20923:4;20918:3;20914:14;20906:22;;20803:132;;;:::o;20941:108::-;21018:24;21036:5;21018:24;:::i;:::-;21013:3;21006:37;20941:108;;:::o;21055:179::-;21124:10;21145:46;21187:3;21179:6;21145:46;:::i;:::-;21223:4;21218:3;21214:14;21200:28;;21055:179;;;;:::o;21240:113::-;21310:4;21342;21337:3;21333:14;21325:22;;21240:113;;;:::o;21389:732::-;21508:3;21537:54;21585:5;21537:54;:::i;:::-;21607:86;21686:6;21681:3;21607:86;:::i;:::-;21600:93;;21717:56;21767:5;21717:56;:::i;:::-;21796:7;21827:1;21812:284;21837:6;21834:1;21831:13;21812:284;;;21913:6;21907:13;21940:63;21999:3;21984:13;21940:63;:::i;:::-;21933:70;;22026:60;22079:6;22026:60;:::i;:::-;22016:70;;21872:224;21859:1;21856;21852:9;21847:14;;21812:284;;;21816:14;22112:3;22105:10;;21513:608;;;21389:732;;;;:::o;22127:373::-;22270:4;22308:2;22297:9;22293:18;22285:26;;22357:9;22351:4;22347:20;22343:1;22332:9;22328:17;22321:47;22385:108;22488:4;22479:6;22385:108;:::i;:::-;22377:116;;22127:373;;;;:::o;22506:116::-;22576:21;22591:5;22576:21;:::i;:::-;22569:5;22566:32;22556:60;;22612:1;22609;22602:12;22556:60;22506:116;:::o;22628:133::-;22671:5;22709:6;22696:20;22687:29;;22725:30;22749:5;22725:30;:::i;:::-;22628:133;;;;:::o;22767:323::-;22823:6;22872:2;22860:9;22851:7;22847:23;22843:32;22840:119;;;22878:79;;:::i;:::-;22840:119;22998:1;23023:50;23065:7;23056:6;23045:9;23041:22;23023:50;:::i;:::-;23013:60;;22969:114;22767:323;;;;:::o;23096:619::-;23173:6;23181;23189;23238:2;23226:9;23217:7;23213:23;23209:32;23206:119;;;23244:79;;:::i;:::-;23206:119;23364:1;23389:53;23434:7;23425:6;23414:9;23410:22;23389:53;:::i;:::-;23379:63;;23335:117;23491:2;23517:53;23562:7;23553:6;23542:9;23538:22;23517:53;:::i;:::-;23507:63;;23462:118;23619:2;23645:53;23690:7;23681:6;23670:9;23666:22;23645:53;:::i;:::-;23635:63;;23590:118;23096:619;;;;;:::o;23721:468::-;23786:6;23794;23843:2;23831:9;23822:7;23818:23;23814:32;23811:119;;;23849:79;;:::i;:::-;23811:119;23969:1;23994:53;24039:7;24030:6;24019:9;24015:22;23994:53;:::i;:::-;23984:63;;23940:117;24096:2;24122:50;24164:7;24155:6;24144:9;24140:22;24122:50;:::i;:::-;24112:60;;24067:115;23721:468;;;;;:::o;24195:307::-;24256:4;24346:18;24338:6;24335:30;24332:56;;;24368:18;;:::i;:::-;24332:56;24406:29;24428:6;24406:29;:::i;:::-;24398:37;;24490:4;24484;24480:15;24472:23;;24195:307;;;:::o;24508:423::-;24585:5;24610:65;24626:48;24667:6;24626:48;:::i;:::-;24610:65;:::i;:::-;24601:74;;24698:6;24691:5;24684:21;24736:4;24729:5;24725:16;24774:3;24765:6;24760:3;24756:16;24753:25;24750:112;;;24781:79;;:::i;:::-;24750:112;24871:54;24918:6;24913:3;24908;24871:54;:::i;:::-;24591:340;24508:423;;;;;:::o;24950:338::-;25005:5;25054:3;25047:4;25039:6;25035:17;25031:27;25021:122;;25062:79;;:::i;:::-;25021:122;25179:6;25166:20;25204:78;25278:3;25270:6;25263:4;25255:6;25251:17;25204:78;:::i;:::-;25195:87;;25011:277;24950:338;;;;:::o;25294:943::-;25389:6;25397;25405;25413;25462:3;25450:9;25441:7;25437:23;25433:33;25430:120;;;25469:79;;:::i;:::-;25430:120;25589:1;25614:53;25659:7;25650:6;25639:9;25635:22;25614:53;:::i;:::-;25604:63;;25560:117;25716:2;25742:53;25787:7;25778:6;25767:9;25763:22;25742:53;:::i;:::-;25732:63;;25687:118;25844:2;25870:53;25915:7;25906:6;25895:9;25891:22;25870:53;:::i;:::-;25860:63;;25815:118;26000:2;25989:9;25985:18;25972:32;26031:18;26023:6;26020:30;26017:117;;;26053:79;;:::i;:::-;26017:117;26158:62;26212:7;26203:6;26192:9;26188:22;26158:62;:::i;:::-;26148:72;;25943:287;25294:943;;;;;;;:::o;26315:876::-;26476:4;26471:3;26467:14;26563:4;26556:5;26552:16;26546:23;26582:63;26639:4;26634:3;26630:14;26616:12;26582:63;:::i;:::-;26491:164;26747:4;26740:5;26736:16;26730:23;26766:61;26821:4;26816:3;26812:14;26798:12;26766:61;:::i;:::-;26665:172;26921:4;26914:5;26910:16;26904:23;26940:57;26991:4;26986:3;26982:14;26968:12;26940:57;:::i;:::-;26847:160;27094:4;27087:5;27083:16;27077:23;27113:61;27168:4;27163:3;27159:14;27145:12;27113:61;:::i;:::-;27017:167;26445:746;26315:876;;:::o;27197:351::-;27354:4;27392:3;27381:9;27377:19;27369:27;;27406:135;27538:1;27527:9;27523:17;27514:6;27406:135;:::i;:::-;27197:351;;;;:::o;27554:474::-;27622:6;27630;27679:2;27667:9;27658:7;27654:23;27650:32;27647:119;;;27685:79;;:::i;:::-;27647:119;27805:1;27830:53;27875:7;27866:6;27855:9;27851:22;27830:53;:::i;:::-;27820:63;;27776:117;27932:2;27958:53;28003:7;27994:6;27983:9;27979:22;27958:53;:::i;:::-;27948:63;;27903:118;27554:474;;;;;:::o;28034:180::-;28082:77;28079:1;28072:88;28179:4;28176:1;28169:15;28203:4;28200:1;28193:15;28220:320;28264:6;28301:1;28295:4;28291:12;28281:22;;28348:1;28342:4;28338:12;28369:18;28359:81;;28425:4;28417:6;28413:17;28403:27;;28359:81;28487:2;28479:6;28476:14;28456:18;28453:38;28450:84;;28506:18;;:::i;:::-;28450:84;28271:269;28220:320;;;:::o;28546:180::-;28594:77;28591:1;28584:88;28691:4;28688:1;28681:15;28715:4;28712:1;28705:15;28732:180;28780:77;28777:1;28770:88;28877:4;28874:1;28867:15;28901:4;28898:1;28891:15;28918:233;28957:3;28980:24;28998:5;28980:24;:::i;:::-;28971:33;;29026:66;29019:5;29016:77;29013:103;;29096:18;;:::i;:::-;29013:103;29143:1;29136:5;29132:13;29125:20;;28918:233;;;:::o;29157:332::-;29278:4;29316:2;29305:9;29301:18;29293:26;;29329:71;29397:1;29386:9;29382:17;29373:6;29329:71;:::i;:::-;29410:72;29478:2;29467:9;29463:18;29454:6;29410:72;:::i;:::-;29157:332;;;;;:::o;29495:137::-;29549:5;29580:6;29574:13;29565:22;;29596:30;29620:5;29596:30;:::i;:::-;29495:137;;;;:::o;29638:345::-;29705:6;29754:2;29742:9;29733:7;29729:23;29725:32;29722:119;;;29760:79;;:::i;:::-;29722:119;29880:1;29905:61;29958:7;29949:6;29938:9;29934:22;29905:61;:::i;:::-;29895:71;;29851:125;29638:345;;;;:::o;29989:410::-;30029:7;30052:20;30070:1;30052:20;:::i;:::-;30047:25;;30086:20;30104:1;30086:20;:::i;:::-;30081:25;;30141:1;30138;30134:9;30163:30;30181:11;30163:30;:::i;:::-;30152:41;;30342:1;30333:7;30329:15;30326:1;30323:22;30303:1;30296:9;30276:83;30253:139;;30372:18;;:::i;:::-;30253:139;30037:362;29989:410;;;;:::o;30405:180::-;30453:77;30450:1;30443:88;30550:4;30547:1;30540:15;30574:4;30571:1;30564:15;30591:185;30631:1;30648:20;30666:1;30648:20;:::i;:::-;30643:25;;30682:20;30700:1;30682:20;:::i;:::-;30677:25;;30721:1;30711:35;;30726:18;;:::i;:::-;30711:35;30768:1;30765;30761:9;30756:14;;30591:185;;;;:::o;30782:141::-;30831:4;30854:3;30846:11;;30877:3;30874:1;30867:14;30911:4;30908:1;30898:18;30890:26;;30782:141;;;:::o;30929:93::-;30966:6;31013:2;31008;31001:5;30997:14;30993:23;30983:33;;30929:93;;;:::o;31028:107::-;31072:8;31122:5;31116:4;31112:16;31091:37;;31028:107;;;;:::o;31141:393::-;31210:6;31260:1;31248:10;31244:18;31283:97;31313:66;31302:9;31283:97;:::i;:::-;31401:39;31431:8;31420:9;31401:39;:::i;:::-;31389:51;;31473:4;31469:9;31462:5;31458:21;31449:30;;31522:4;31512:8;31508:19;31501:5;31498:30;31488:40;;31217:317;;31141:393;;;;;:::o;31540:60::-;31568:3;31589:5;31582:12;;31540:60;;;:::o;31606:142::-;31656:9;31689:53;31707:34;31716:24;31734:5;31716:24;:::i;:::-;31707:34;:::i;:::-;31689:53;:::i;:::-;31676:66;;31606:142;;;:::o;31754:75::-;31797:3;31818:5;31811:12;;31754:75;;;:::o;31835:269::-;31945:39;31976:7;31945:39;:::i;:::-;32006:91;32055:41;32079:16;32055:41;:::i;:::-;32047:6;32040:4;32034:11;32006:91;:::i;:::-;32000:4;31993:105;31911:193;31835:269;;;:::o;32110:73::-;32155:3;32110:73;:::o;32189:189::-;32266:32;;:::i;:::-;32307:65;32365:6;32357;32351:4;32307:65;:::i;:::-;32242:136;32189:189;;:::o;32384:186::-;32444:120;32461:3;32454:5;32451:14;32444:120;;;32515:39;32552:1;32545:5;32515:39;:::i;:::-;32488:1;32481:5;32477:13;32468:22;;32444:120;;;32384:186;;:::o;32576:543::-;32677:2;32672:3;32669:11;32666:446;;;32711:38;32743:5;32711:38;:::i;:::-;32795:29;32813:10;32795:29;:::i;:::-;32785:8;32781:44;32978:2;32966:10;32963:18;32960:49;;;32999:8;32984:23;;32960:49;33022:80;33078:22;33096:3;33078:22;:::i;:::-;33068:8;33064:37;33051:11;33022:80;:::i;:::-;32681:431;;32666:446;32576:543;;;:::o;33125:117::-;33179:8;33229:5;33223:4;33219:16;33198:37;;33125:117;;;;:::o;33248:169::-;33292:6;33325:51;33373:1;33369:6;33361:5;33358:1;33354:13;33325:51;:::i;:::-;33321:56;33406:4;33400;33396:15;33386:25;;33299:118;33248:169;;;;:::o;33422:295::-;33498:4;33644:29;33669:3;33663:4;33644:29;:::i;:::-;33636:37;;33706:3;33703:1;33699:11;33693:4;33690:21;33682:29;;33422:295;;;;:::o;33722:1395::-;33839:37;33872:3;33839:37;:::i;:::-;33941:18;33933:6;33930:30;33927:56;;;33963:18;;:::i;:::-;33927:56;34007:38;34039:4;34033:11;34007:38;:::i;:::-;34092:67;34152:6;34144;34138:4;34092:67;:::i;:::-;34186:1;34210:4;34197:17;;34242:2;34234:6;34231:14;34259:1;34254:618;;;;34916:1;34933:6;34930:77;;;34982:9;34977:3;34973:19;34967:26;34958:35;;34930:77;35033:67;35093:6;35086:5;35033:67;:::i;:::-;35027:4;35020:81;34889:222;34224:887;;34254:618;34306:4;34302:9;34294:6;34290:22;34340:37;34372:4;34340:37;:::i;:::-;34399:1;34413:208;34427:7;34424:1;34421:14;34413:208;;;34506:9;34501:3;34497:19;34491:26;34483:6;34476:42;34557:1;34549:6;34545:14;34535:24;;34604:2;34593:9;34589:18;34576:31;;34450:4;34447:1;34443:12;34438:17;;34413:208;;;34649:6;34640:7;34637:19;34634:179;;;34707:9;34702:3;34698:19;34692:26;34750:48;34792:4;34784:6;34780:17;34769:9;34750:48;:::i;:::-;34742:6;34735:64;34657:156;34634:179;34859:1;34855;34847:6;34843:14;34839:22;34833:4;34826:36;34261:611;;;34224:887;;33814:1303;;;33722:1395;;:::o;35123:175::-;35263:27;35259:1;35251:6;35247:14;35240:51;35123:175;:::o;35304:366::-;35446:3;35467:67;35531:2;35526:3;35467:67;:::i;:::-;35460:74;;35543:93;35632:3;35543:93;:::i;:::-;35661:2;35656:3;35652:12;35645:19;;35304:366;;;:::o;35676:419::-;35842:4;35880:2;35869:9;35865:18;35857:26;;35929:9;35923:4;35919:20;35915:1;35904:9;35900:17;35893:47;35957:131;36083:4;35957:131;:::i;:::-;35949:139;;35676:419;;;:::o;36101:169::-;36241:21;36237:1;36229:6;36225:14;36218:45;36101:169;:::o;36276:366::-;36418:3;36439:67;36503:2;36498:3;36439:67;:::i;:::-;36432:74;;36515:93;36604:3;36515:93;:::i;:::-;36633:2;36628:3;36624:12;36617:19;;36276:366;;;:::o;36648:419::-;36814:4;36852:2;36841:9;36837:18;36829:26;;36901:9;36895:4;36891:20;36887:1;36876:9;36872:17;36865:47;36929:131;37055:4;36929:131;:::i;:::-;36921:139;;36648:419;;;:::o;37073:229::-;37213:34;37209:1;37201:6;37197:14;37190:58;37282:12;37277:2;37269:6;37265:15;37258:37;37073:229;:::o;37308:366::-;37450:3;37471:67;37535:2;37530:3;37471:67;:::i;:::-;37464:74;;37547:93;37636:3;37547:93;:::i;:::-;37665:2;37660:3;37656:12;37649:19;;37308:366;;;:::o;37680:419::-;37846:4;37884:2;37873:9;37869:18;37861:26;;37933:9;37927:4;37923:20;37919:1;37908:9;37904:17;37897:47;37961:131;38087:4;37961:131;:::i;:::-;37953:139;;37680:419;;;:::o;38105:191::-;38145:3;38164:20;38182:1;38164:20;:::i;:::-;38159:25;;38198:20;38216:1;38198:20;:::i;:::-;38193:25;;38241:1;38238;38234:9;38227:16;;38262:3;38259:1;38256:10;38253:36;;;38269:18;;:::i;:::-;38253:36;38105:191;;;;:::o;38302:180::-;38442:32;38438:1;38430:6;38426:14;38419:56;38302:180;:::o;38488:366::-;38630:3;38651:67;38715:2;38710:3;38651:67;:::i;:::-;38644:74;;38727:93;38816:3;38727:93;:::i;:::-;38845:2;38840:3;38836:12;38829:19;;38488:366;;;:::o;38860:419::-;39026:4;39064:2;39053:9;39049:18;39041:26;;39113:9;39107:4;39103:20;39099:1;39088:9;39084:17;39077:47;39141:131;39267:4;39141:131;:::i;:::-;39133:139;;38860:419;;;:::o;39285:230::-;39425:34;39421:1;39413:6;39409:14;39402:58;39494:13;39489:2;39481:6;39477:15;39470:38;39285:230;:::o;39521:366::-;39663:3;39684:67;39748:2;39743:3;39684:67;:::i;:::-;39677:74;;39760:93;39849:3;39760:93;:::i;:::-;39878:2;39873:3;39869:12;39862:19;;39521:366;;;:::o;39893:419::-;40059:4;40097:2;40086:9;40082:18;40074:26;;40146:9;40140:4;40136:20;40132:1;40121:9;40117:17;40110:47;40174:131;40300:4;40174:131;:::i;:::-;40166:139;;39893:419;;;:::o;40318:226::-;40458:34;40454:1;40446:6;40442:14;40435:58;40527:9;40522:2;40514:6;40510:15;40503:34;40318:226;:::o;40550:366::-;40692:3;40713:67;40777:2;40772:3;40713:67;:::i;:::-;40706:74;;40789:93;40878:3;40789:93;:::i;:::-;40907:2;40902:3;40898:12;40891:19;;40550:366;;;:::o;40922:419::-;41088:4;41126:2;41115:9;41111:18;41103:26;;41175:9;41169:4;41165:20;41161:1;41150:9;41146:17;41139:47;41203:131;41329:4;41203:131;:::i;:::-;41195:139;;40922:419;;;:::o;41347:147::-;41448:11;41485:3;41470:18;;41347:147;;;;:::o;41500:114::-;;:::o;41620:398::-;41779:3;41800:83;41881:1;41876:3;41800:83;:::i;:::-;41793:90;;41892:93;41981:3;41892:93;:::i;:::-;42010:1;42005:3;42001:11;41994:18;;41620:398;;;:::o;42024:379::-;42208:3;42230:147;42373:3;42230:147;:::i;:::-;42223:154;;42394:3;42387:10;;42024:379;;;:::o;42409:166::-;42549:18;42545:1;42537:6;42533:14;42526:42;42409:166;:::o;42581:366::-;42723:3;42744:67;42808:2;42803:3;42744:67;:::i;:::-;42737:74;;42820:93;42909:3;42820:93;:::i;:::-;42938:2;42933:3;42929:12;42922:19;;42581:366;;;:::o;42953:419::-;43119:4;43157:2;43146:9;43142:18;43134:26;;43206:9;43200:4;43196:20;43192:1;43181:9;43177:17;43170:47;43234:131;43360:4;43234:131;:::i;:::-;43226:139;;42953:419;;;:::o;43378:234::-;43518:34;43514:1;43506:6;43502:14;43495:58;43587:17;43582:2;43574:6;43570:15;43563:42;43378:234;:::o;43618:366::-;43760:3;43781:67;43845:2;43840:3;43781:67;:::i;:::-;43774:74;;43857:93;43946:3;43857:93;:::i;:::-;43975:2;43970:3;43966:12;43959:19;;43618:366;;;:::o;43990:419::-;44156:4;44194:2;44183:9;44179:18;44171:26;;44243:9;44237:4;44233:20;44229:1;44218:9;44214:17;44207:47;44271:131;44397:4;44271:131;:::i;:::-;44263:139;;43990:419;;;:::o;44415:148::-;44517:11;44554:3;44539:18;;44415:148;;;;:::o;44569:390::-;44675:3;44703:39;44736:5;44703:39;:::i;:::-;44758:89;44840:6;44835:3;44758:89;:::i;:::-;44751:96;;44856:65;44914:6;44909:3;44902:4;44895:5;44891:16;44856:65;:::i;:::-;44946:6;44941:3;44937:16;44930:23;;44679:280;44569:390;;;;:::o;44989:874::-;45092:3;45129:5;45123:12;45158:36;45184:9;45158:36;:::i;:::-;45210:89;45292:6;45287:3;45210:89;:::i;:::-;45203:96;;45330:1;45319:9;45315:17;45346:1;45341:166;;;;45521:1;45516:341;;;;45308:549;;45341:166;45425:4;45421:9;45410;45406:25;45401:3;45394:38;45487:6;45480:14;45473:22;45465:6;45461:35;45456:3;45452:45;45445:52;;45341:166;;45516:341;45583:38;45615:5;45583:38;:::i;:::-;45643:1;45657:154;45671:6;45668:1;45665:13;45657:154;;;45745:7;45739:14;45735:1;45730:3;45726:11;45719:35;45795:1;45786:7;45782:15;45771:26;;45693:4;45690:1;45686:12;45681:17;;45657:154;;;45840:6;45835:3;45831:16;45824:23;;45523:334;;45308:549;;45096:767;;44989:874;;;;:::o;45869:589::-;46094:3;46116:95;46207:3;46198:6;46116:95;:::i;:::-;46109:102;;46228:95;46319:3;46310:6;46228:95;:::i;:::-;46221:102;;46340:92;46428:3;46419:6;46340:92;:::i;:::-;46333:99;;46449:3;46442:10;;45869:589;;;;;;:::o;46464:225::-;46604:34;46600:1;46592:6;46588:14;46581:58;46673:8;46668:2;46660:6;46656:15;46649:33;46464:225;:::o;46695:366::-;46837:3;46858:67;46922:2;46917:3;46858:67;:::i;:::-;46851:74;;46934:93;47023:3;46934:93;:::i;:::-;47052:2;47047:3;47043:12;47036:19;;46695:366;;;:::o;47067:419::-;47233:4;47271:2;47260:9;47256:18;47248:26;;47320:9;47314:4;47310:20;47306:1;47295:9;47291:17;47284:47;47348:131;47474:4;47348:131;:::i;:::-;47340:139;;47067:419;;;:::o;47492:182::-;47632:34;47628:1;47620:6;47616:14;47609:58;47492:182;:::o;47680:366::-;47822:3;47843:67;47907:2;47902:3;47843:67;:::i;:::-;47836:74;;47919:93;48008:3;47919:93;:::i;:::-;48037:2;48032:3;48028:12;48021:19;;47680:366;;;:::o;48052:419::-;48218:4;48256:2;48245:9;48241:18;48233:26;;48305:9;48299:4;48295:20;48291:1;48280:9;48276:17;48269:47;48333:131;48459:4;48333:131;:::i;:::-;48325:139;;48052:419;;;:::o;48477:229::-;48617:34;48613:1;48605:6;48601:14;48594:58;48686:12;48681:2;48673:6;48669:15;48662:37;48477:229;:::o;48712:366::-;48854:3;48875:67;48939:2;48934:3;48875:67;:::i;:::-;48868:74;;48951:93;49040:3;48951:93;:::i;:::-;49069:2;49064:3;49060:12;49053:19;;48712:366;;;:::o;49084:419::-;49250:4;49288:2;49277:9;49273:18;49265:26;;49337:9;49331:4;49327:20;49323:1;49312:9;49308:17;49301:47;49365:131;49491:4;49365:131;:::i;:::-;49357:139;;49084:419;;;:::o;49509:175::-;49649:27;49645:1;49637:6;49633:14;49626:51;49509:175;:::o;49690:366::-;49832:3;49853:67;49917:2;49912:3;49853:67;:::i;:::-;49846:74;;49929:93;50018:3;49929:93;:::i;:::-;50047:2;50042:3;50038:12;50031:19;;49690:366;;;:::o;50062:419::-;50228:4;50266:2;50255:9;50251:18;50243:26;;50315:9;50309:4;50305:20;50301:1;50290:9;50286:17;50279:47;50343:131;50469:4;50343:131;:::i;:::-;50335:139;;50062:419;;;:::o;50487:177::-;50627:29;50623:1;50615:6;50611:14;50604:53;50487:177;:::o;50670:366::-;50812:3;50833:67;50897:2;50892:3;50833:67;:::i;:::-;50826:74;;50909:93;50998:3;50909:93;:::i;:::-;51027:2;51022:3;51018:12;51011:19;;50670:366;;;:::o;51042:419::-;51208:4;51246:2;51235:9;51231:18;51223:26;;51295:9;51289:4;51285:20;51281:1;51270:9;51266:17;51259:47;51323:131;51449:4;51323:131;:::i;:::-;51315:139;;51042:419;;;:::o;51467:98::-;51518:6;51552:5;51546:12;51536:22;;51467:98;;;:::o;51571:168::-;51654:11;51688:6;51683:3;51676:19;51728:4;51723:3;51719:14;51704:29;;51571:168;;;;:::o;51745:373::-;51831:3;51859:38;51891:5;51859:38;:::i;:::-;51913:70;51976:6;51971:3;51913:70;:::i;:::-;51906:77;;51992:65;52050:6;52045:3;52038:4;52031:5;52027:16;51992:65;:::i;:::-;52082:29;52104:6;52082:29;:::i;:::-;52077:3;52073:39;52066:46;;51835:283;51745:373;;;;:::o;52124:640::-;52319:4;52357:3;52346:9;52342:19;52334:27;;52371:71;52439:1;52428:9;52424:17;52415:6;52371:71;:::i;:::-;52452:72;52520:2;52509:9;52505:18;52496:6;52452:72;:::i;:::-;52534;52602:2;52591:9;52587:18;52578:6;52534:72;:::i;:::-;52653:9;52647:4;52643:20;52638:2;52627:9;52623:18;52616:48;52681:76;52752:4;52743:6;52681:76;:::i;:::-;52673:84;;52124:640;;;;;;;:::o;52770:141::-;52826:5;52857:6;52851:13;52842:22;;52873:32;52899:5;52873:32;:::i;:::-;52770:141;;;;:::o;52917:349::-;52986:6;53035:2;53023:9;53014:7;53010:23;53006:32;53003:119;;;53041:79;;:::i;:::-;53003:119;53161:1;53186:63;53241:7;53232:6;53221:9;53217:22;53186:63;:::i;:::-;53176:73;;53132:127;52917:349;;;;:::o
Swarm Source
ipfs://b67668e595bed29b874105bc717d5732ec5e2b63043552ccb0f269e84c381111
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.