Feature Tip: Add private address tag to any address under My Name Tag !
ERC-721
Overview
Max Total Supply
1,212 MS
Holders
374
Market
Volume (24H)
N/A
Min Price (24H)
N/A
Max Price (24H)
N/A
Other Info
Token Contract
Balance
3 MSLoading...
Loading
Loading...
Loading
Loading...
Loading
# | Exchange | Pair | Price | 24H Volume | % Volume |
---|
Contract Name:
MiladyStationUpgraded
Compiler Version
v0.8.17+commit.8df45f5f
Contract Source Code (Solidity)
/** *Submitted for verification at Etherscan.io on 2022-12-20 */ // File: closedsea/OperatorFilterer.sol pragma solidity ^0.8.4; /// @notice Optimized and flexible operator filterer to abide to OpenSea's /// mandatory on-chain royalty enforcement in order for new collections to /// receive royalties. /// For more information, see: /// See: https://github.com/ProjectOpenSea/operator-filter-registry abstract contract OperatorFilterer { /// @dev The default OpenSea operator blocklist subscription. address internal constant _DEFAULT_SUBSCRIPTION = 0x3cc6CddA760b79bAfa08dF41ECFA224f810dCeB6; /// @dev The OpenSea operator filter registry. address internal constant _OPERATOR_FILTER_REGISTRY = 0x000000000000AAeB6D7670E522A718067333cd4E; /// @dev Registers the current contract to OpenSea's operator filter, /// and subscribe to the default OpenSea operator blocklist. /// Note: Will not revert nor update existing settings for repeated registration. function _registerForOperatorFiltering() internal virtual { _registerForOperatorFiltering(_DEFAULT_SUBSCRIPTION, true); } /// @dev Registers the current contract to OpenSea's operator filter. /// Note: Will not revert nor update existing settings for repeated registration. function _registerForOperatorFiltering(address subscriptionOrRegistrantToCopy, bool subscribe) internal virtual { /// @solidity memory-safe-assembly assembly { let functionSelector := 0x7d3e3dbe // `registerAndSubscribe(address,address)`. // Clean the upper 96 bits of `subscriptionOrRegistrantToCopy` in case they are dirty. subscriptionOrRegistrantToCopy := shr(96, shl(96, subscriptionOrRegistrantToCopy)) for {} iszero(subscribe) {} { if iszero(subscriptionOrRegistrantToCopy) { functionSelector := 0x4420e486 // `register(address)`. break } functionSelector := 0xa0af2903 // `registerAndCopyEntries(address,address)`. break } // Store the function selector. mstore(0x00, shl(224, functionSelector)) // Store the `address(this)`. mstore(0x04, address()) // Store the `subscriptionOrRegistrantToCopy`. mstore(0x24, subscriptionOrRegistrantToCopy) // Register into the registry. if iszero(call(gas(), _OPERATOR_FILTER_REGISTRY, 0, 0x00, 0x44, 0x00, 0x04)) { // If the function selector has not been overwritten, // it is an out-of-gas error. if eq(shr(224, mload(0x00)), functionSelector) { // To prevent gas under-estimation. revert(0, 0) } } // Restore the part of the free memory pointer that was overwritten, // which is guaranteed to be zero, because of Solidity's memory size limits. mstore(0x24, 0) } } /// @dev Modifier to guard a function and revert if the caller is a blocked operator. modifier onlyAllowedOperator(address from) virtual { if (from != msg.sender) { if (!_isPriorityOperator(msg.sender)) { if (_operatorFilteringEnabled()) _revertIfBlocked(msg.sender); } } _; } /// @dev Modifier to guard a function from approving a blocked operator.. modifier onlyAllowedOperatorApproval(address operator) virtual { if (!_isPriorityOperator(operator)) { if (_operatorFilteringEnabled()) _revertIfBlocked(operator); } _; } /// @dev Helper function that reverts if the `operator` is blocked by the registry. function _revertIfBlocked(address operator) private view { /// @solidity memory-safe-assembly assembly { // Store the function selector of `isOperatorAllowed(address,address)`, // shifted left by 6 bytes, which is enough for 8tb of memory. // We waste 6-3 = 3 bytes to save on 6 runtime gas (PUSH1 0x224 SHL). mstore(0x00, 0xc6171134001122334455) // Store the `address(this)`. mstore(0x1a, address()) // Store the `operator`. mstore(0x3a, operator) // `isOperatorAllowed` always returns true if it does not revert. if iszero(staticcall(gas(), _OPERATOR_FILTER_REGISTRY, 0x16, 0x44, 0x00, 0x00)) { // Bubble up the revert if the staticcall reverts. returndatacopy(0x00, 0x00, returndatasize()) revert(0x00, returndatasize()) } // We'll skip checking if `from` is inside the blacklist. // Even though that can block transferring out of wrapper contracts, // we don't want tokens to be stuck. // Restore the part of the free memory pointer that was overwritten, // which is guaranteed to be zero, if less than 8tb of memory is used. mstore(0x3a, 0) } } /// @dev For deriving contracts to override, so that operator filtering /// can be turned on / off. /// Returns true by default. function _operatorFilteringEnabled() internal view virtual returns (bool) { return true; } /// @dev For deriving contracts to override, so that preferred marketplaces can /// skip operator filtering, helping users save gas. /// Returns false for all inputs by default. function _isPriorityOperator(address) internal view virtual returns (bool) { return false; } } // File: solady/auth/Ownable.sol pragma solidity ^0.8.4; /// @notice Simple single owner authorization mixin. /// @author Solady (https://github.com/vectorized/solady/blob/main/src/auth/Ownable.sol) /// @dev While the ownable portion follows [EIP-173](https://eips.ethereum.org/EIPS/eip-173) /// for compatibility, the nomenclature for the 2-step ownership handover /// may be unique to this codebase. abstract contract Ownable { /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* CUSTOM ERRORS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The caller is not authorized to call the function. error Unauthorized(); /// @dev The `newOwner` cannot be the zero address. error NewOwnerIsZeroAddress(); /// @dev The `pendingOwner` does not have a valid handover request. error NoHandoverRequest(); /// @dev `bytes4(keccak256(bytes("Unauthorized()")))`. uint256 private constant _UNAUTHORIZED_ERROR_SELECTOR = 0x82b42900; /// @dev `bytes4(keccak256(bytes("NewOwnerIsZeroAddress()")))`. uint256 private constant _NEW_OWNER_IS_ZERO_ADDRESS_ERROR_SELECTOR = 0x7448fbae; /// @dev `bytes4(keccak256(bytes("NoHandoverRequest()")))`. uint256 private constant _NO_HANDOVER_REQUEST_ERROR_SELECTOR = 0x6f5e8818; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* EVENTS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The ownership is transferred from `oldOwner` to `newOwner`. /// This event is intentionally kept the same as OpenZeppelin's Ownable to be /// compatible with indexers and [EIP-173](https://eips.ethereum.org/EIPS/eip-173), /// despite it not being as lightweight as a single argument event. event OwnershipTransferred(address indexed oldOwner, address indexed newOwner); /// @dev An ownership handover to `pendingOwner` has been requested. event OwnershipHandoverRequested(address indexed pendingOwner); /// @dev The ownership handover to `pendingOwner` has been canceled. event OwnershipHandoverCanceled(address indexed pendingOwner); /// @dev `keccak256(bytes("OwnershipTransferred(address,address)"))`. uint256 private constant _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE = 0x8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0; /// @dev `keccak256(bytes("OwnershipHandoverRequested(address)"))`. uint256 private constant _OWNERSHIP_HANDOVER_REQUESTED_EVENT_SIGNATURE = 0xdbf36a107da19e49527a7176a1babf963b4b0ff8cde35ee35d6cd8f1f9ac7e1d; /// @dev `keccak256(bytes("OwnershipHandoverCanceled(address)"))`. uint256 private constant _OWNERSHIP_HANDOVER_CANCELED_EVENT_SIGNATURE = 0xfa7b8eab7da67f412cc9575ed43464468f9bfbae89d1675917346ca6d8fe3c92; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* STORAGE */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev The owner slot is given by: `not(_OWNER_SLOT_NOT)`. /// It is intentionally choosen to be a high value /// to avoid collision with lower slots. /// The choice of manual storage layout is to enable compatibility /// with both regular and upgradeable contracts. uint256 private constant _OWNER_SLOT_NOT = 0x8b78c6d8; /// The ownership handover slot of `newOwner` is given by: /// ``` /// mstore(0x00, or(shl(96, user), _HANDOVER_SLOT_SEED)) /// let handoverSlot := keccak256(0x00, 0x20) /// ``` /// It stores the expiry timestamp of the two-step ownership handover. uint256 private constant _HANDOVER_SLOT_SEED = 0x389a75e1; /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* INTERNAL FUNCTIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Initializes the owner directly without authorization guard. /// This function must be called upon initialization, /// regardless of whether the contract is upgradeable or not. /// This is to enable generalization to both regular and upgradeable contracts, /// and to save gas in case the initial owner is not the caller. /// For performance reasons, this function will not check if there /// is an existing owner. function _initializeOwner(address newOwner) internal virtual { /// @solidity memory-safe-assembly assembly { // Clean the upper 96 bits. newOwner := shr(96, shl(96, newOwner)) // Store the new value. sstore(not(_OWNER_SLOT_NOT), newOwner) // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, 0, newOwner) } } /// @dev Sets the owner directly without authorization guard. function _setOwner(address newOwner) internal virtual { /// @solidity memory-safe-assembly assembly { let ownerSlot := not(_OWNER_SLOT_NOT) // Clean the upper 96 bits. newOwner := shr(96, shl(96, newOwner)) // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, sload(ownerSlot), newOwner) // Store the new value. sstore(ownerSlot, newOwner) } } /// @dev Throws if the sender is not the owner. function _checkOwner() internal view virtual { /// @solidity memory-safe-assembly assembly { // If the caller is not the stored owner, revert. if iszero(eq(caller(), sload(not(_OWNER_SLOT_NOT)))) { mstore(0x00, _UNAUTHORIZED_ERROR_SELECTOR) revert(0x1c, 0x04) } } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* PUBLIC UPDATE FUNCTIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Allows the owner to transfer the ownership to `newOwner`. function transferOwnership(address newOwner) public payable virtual onlyOwner { if (newOwner == address(0)) revert NewOwnerIsZeroAddress(); _setOwner(newOwner); } /// @dev Allows the owner to renounce their ownership. function renounceOwnership() public payable virtual onlyOwner { _setOwner(address(0)); } /// @dev Request a two-step ownership handover to the caller. /// The request will be automatically expire in 48 hours (172800 seconds) by default. function requestOwnershipHandover() public payable virtual { unchecked { uint256 expires = block.timestamp + ownershipHandoverValidFor(); /// @solidity memory-safe-assembly assembly { // Compute and set the handover slot to 1. mstore(0x0c, _HANDOVER_SLOT_SEED) mstore(0x00, caller()) sstore(keccak256(0x0c, 0x20), expires) // Emit the {OwnershipHandoverRequested} event. log2(0, 0, _OWNERSHIP_HANDOVER_REQUESTED_EVENT_SIGNATURE, caller()) } } } /// @dev Cancels the two-step ownership handover to the caller, if any. function cancelOwnershipHandover() public payable virtual { /// @solidity memory-safe-assembly assembly { // Compute and set the handover slot to 0. mstore(0x0c, _HANDOVER_SLOT_SEED) mstore(0x00, caller()) sstore(keccak256(0x0c, 0x20), 0) // Emit the {OwnershipHandoverCanceled} event. log2(0, 0, _OWNERSHIP_HANDOVER_CANCELED_EVENT_SIGNATURE, caller()) } } /// @dev Allows the owner to complete the two-step ownership handover to `pendingOwner`. /// Reverts if there is no existing ownership handover requested by `pendingOwner`. function completeOwnershipHandover(address pendingOwner) public payable virtual onlyOwner { /// @solidity memory-safe-assembly assembly { // Compute and set the handover slot to 0. mstore(0x0c, _HANDOVER_SLOT_SEED) mstore(0x00, pendingOwner) let handoverSlot := keccak256(0x0c, 0x20) // If the handover does not exist, or has expired. if gt(timestamp(), sload(handoverSlot)) { mstore(0x00, _NO_HANDOVER_REQUEST_ERROR_SELECTOR) revert(0x1c, 0x04) } // Set the handover slot to 0. sstore(handoverSlot, 0) // Clean the upper 96 bits. let newOwner := shr(96, mload(0x0c)) // Emit the {OwnershipTransferred} event. log3(0, 0, _OWNERSHIP_TRANSFERRED_EVENT_SIGNATURE, caller(), newOwner) // Store the new value. sstore(not(_OWNER_SLOT_NOT), newOwner) } } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* PUBLIC READ FUNCTIONS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Returns the owner of the contract. function owner() public view virtual returns (address result) { /// @solidity memory-safe-assembly assembly { result := sload(not(_OWNER_SLOT_NOT)) } } /// @dev Returns the expiry timestamp for the two-step ownership handover to `pendingOwner`. function ownershipHandoverExpiresAt(address pendingOwner) public view virtual returns (uint256 result) { /// @solidity memory-safe-assembly assembly { // Compute the handover slot. mstore(0x0c, _HANDOVER_SLOT_SEED) mstore(0x00, pendingOwner) // Load the handover slot. result := sload(keccak256(0x0c, 0x20)) } } /// @dev Returns how long a two-step ownership handover is valid for in seconds. function ownershipHandoverValidFor() public view virtual returns (uint64) { return 48 * 3600; } /*´:°•.°+.*•´.*:˚.°*.˚•´.°:°•.°•.*•´.*:˚.°*.˚•´.°:°•.°+.*•´.*:*/ /* MODIFIERS */ /*.•°:°.´+˚.*°.˚:*.´•*.+°.•°:´*.´•*.•°.•°:°.´:•˚°.*°.˚:*.´+°.•*/ /// @dev Marks a function as only callable by the owner. modifier onlyOwner() virtual { _checkOwner(); _; } } // File: erc721a/contracts/IERC721A.sol // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @dev Interface of ERC721A. */ interface IERC721A { /** * The caller must own the token or be an approved operator. */ error ApprovalCallerNotOwnerNorApproved(); /** * The token does not exist. */ error ApprovalQueryForNonexistentToken(); /** * 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); } // File: erc721a/contracts/extensions/IERC721AQueryable.sol // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @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); } // File: erc721a/contracts/extensions/IERC721ABurnable.sol // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @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; } // File: erc721a/contracts/ERC721A.sol // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @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 packed) { if (_startTokenId() <= tokenId) { packed = _packedOwnerships[tokenId]; // If not burned. if (packed & _BITMASK_BURNED == 0) { // If the data at the starting slot does not exist, start the scan. if (packed == 0) { if (tokenId >= _currentIndex) revert OwnerQueryForNonexistentToken(); // 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, `tokenId` will not underflow. // // We can directly compare the packed value. // If the address is zero, packed will be zero. for (;;) { unchecked { packed = _packedOwnerships[--tokenId]; } if (packed == 0) continue; return packed; } } // Otherwise, the data exists and is not burned. We can skip the scan. // This is possible because we have already achieved the target condition. // This saves 2143 gas on transfers of initialized tokens. 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) if (_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) } } } // File: erc721a/contracts/extensions/ERC721AQueryable.sol // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @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; } } } // File: erc721a/contracts/extensions/ERC721ABurnable.sol // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @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); } } // File: @openzeppelin/contracts/utils/math/Math.sol // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/Math.sol) pragma solidity ^0.8.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); } } } // File: @openzeppelin/contracts/utils/Strings.sol // OpenZeppelin Contracts (last updated v4.8.0) (utils/Strings.sol) pragma solidity ^0.8.0; /** * @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); } } // File: @openzeppelin/contracts/utils/Context.sol // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } } // File: @openzeppelin/contracts/utils/Address.sol // OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } } // File: @openzeppelin/contracts/token/ERC721/IERC721Receiver.sol // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol) pragma solidity ^0.8.0; /** * @title ERC721 token receiver interface * @dev Interface for any contract that wants to support safeTransfers * from ERC721 asset contracts. */ interface IERC721Receiver { /** * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom} * by `operator` from `from`, this function is called. * * It must return its Solidity selector to confirm the token transfer. * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted. * * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`. */ function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } // File: @openzeppelin/contracts/utils/introspection/IERC165.sol // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); } // File: @openzeppelin/contracts/interfaces/IERC2981.sol // OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol) pragma solidity ^0.8.0; /** * @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); } // File: @openzeppelin/contracts/utils/introspection/ERC165.sol // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol) pragma solidity ^0.8.0; /** * @dev Implementation of the {IERC165} interface. * * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check * for the additional interface id that will be supported. For example: * * ```solidity * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId); * } * ``` * * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation. */ abstract contract ERC165 is IERC165 { /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IERC165).interfaceId; } } // File: @openzeppelin/contracts/token/common/ERC2981.sol // OpenZeppelin Contracts (last updated v4.7.0) (token/common/ERC2981.sol) pragma solidity ^0.8.0; /** * @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]; } } // File: @openzeppelin/contracts/token/ERC721/IERC721.sol // OpenZeppelin Contracts (last updated v4.8.0) (token/ERC721/IERC721.sol) pragma solidity ^0.8.0; /** * @dev Required interface of an ERC721 compliant contract. */ interface IERC721 is IERC165 { /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in ``owner``'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external; /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Transfers `tokenId` token from `from` to `to`. * * WARNING: 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); } // File: @openzeppelin/contracts/token/ERC721/extensions/IERC721Enumerable.sol // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/extensions/IERC721Enumerable.sol) pragma solidity ^0.8.0; /** * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Enumerable is IERC721 { /** * @dev Returns the total amount of tokens stored by the contract. */ function totalSupply() external view returns (uint256); /** * @dev Returns a token ID owned by `owner` at a given `index` of its token list. * Use along with {balanceOf} to enumerate all of ``owner``'s tokens. */ function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256); /** * @dev Returns a token ID at a given `index` of all the tokens stored by the contract. * Use along with {totalSupply} to enumerate all tokens. */ function tokenByIndex(uint256 index) external view returns (uint256); } // File: @openzeppelin/contracts/token/ERC721/extensions/IERC721Metadata.sol // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol) pragma solidity ^0.8.0; /** * @title ERC-721 Non-Fungible Token Standard, optional metadata extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ interface IERC721Metadata is IERC721 { /** * @dev Returns the token collection name. */ function name() external view returns (string memory); /** * @dev Returns the token collection symbol. */ function symbol() external view returns (string memory); /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) external view returns (string memory); } // File: @openzeppelin/contracts/token/ERC721/ERC721.sol // OpenZeppelin Contracts (last updated v4.8.0) (token/ERC721/ERC721.sol) pragma solidity ^0.8.0; /** * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including * the Metadata extension, but not including the Enumerable extension, which is available separately as * {ERC721Enumerable}. */ contract ERC721 is Context, ERC165, IERC721, IERC721Metadata { using Address for address; using Strings for uint256; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to owner address mapping(uint256 => address) private _owners; // Mapping owner address to token count mapping(address => uint256) private _balances; // Mapping from token ID to approved address mapping(uint256 => address) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; /** * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection. */ constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) { return interfaceId == type(IERC721).interfaceId || interfaceId == type(IERC721Metadata).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721-balanceOf}. */ function balanceOf(address owner) public view virtual override returns (uint256) { require(owner != address(0), "ERC721: address zero is not a valid owner"); return _balances[owner]; } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { address owner = _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 {} } // File: @openzeppelin/contracts/token/ERC721/extensions/ERC721Enumerable.sol // OpenZeppelin Contracts (last updated v4.8.0) (token/ERC721/extensions/ERC721Enumerable.sol) pragma solidity ^0.8.0; /** * @dev This implements an optional extension of {ERC721} defined in the EIP that adds * enumerability of all the token ids in the contract as well as all token ids owned by each * account. */ abstract contract ERC721Enumerable is ERC721, IERC721Enumerable { // Mapping from owner to list of owned token IDs mapping(address => mapping(uint256 => uint256)) private _ownedTokens; // Mapping from token ID to index of the owner tokens list mapping(uint256 => uint256) private _ownedTokensIndex; // Array with all token ids, used for enumeration uint256[] private _allTokens; // Mapping from token id to position in the allTokens array mapping(uint256 => uint256) private _allTokensIndex; /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC721) returns (bool) { return interfaceId == type(IERC721Enumerable).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}. */ function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) { require(index < ERC721.balanceOf(owner), "ERC721Enumerable: owner index out of bounds"); return _ownedTokens[owner][index]; } /** * @dev See {IERC721Enumerable-totalSupply}. */ function totalSupply() public view virtual override returns (uint256) { return _allTokens.length; } /** * @dev See {IERC721Enumerable-tokenByIndex}. */ function tokenByIndex(uint256 index) public view virtual override returns (uint256) { require(index < ERC721Enumerable.totalSupply(), "ERC721Enumerable: global index out of bounds"); return _allTokens[index]; } /** * @dev 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(); } } // File: miladystation2/miladystationA.sol pragma solidity ^0.8.17; /*⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣾⣿⣷⣶⣤⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⠿⣿⣿⣿⣿⣿⣿⣶⣤⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣀⣠⣤⣴⣶⣾⣿⣷⣆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠙⠻⢿⣿⣿⣿⣿⣷⣦⣄⡀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⠏⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠻⢿⣿⣿⣿⣿⣦⡀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣾⣿⣿⣿⣿⣿⡿⠿⠟⠛⠉⠉⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠻⣿⣿⣿⣆⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⡿⠟⠋⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠙⣿⣿⣧⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣿⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢻⣿⣇⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⣠⣾⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣀⣴⣶⣿⣿⣿⣿⣿⣿⣿⣶⣤⡀⠀⠙⠋⠀⠀⠀ ⠀⠀⠀⠀⠀⣠⣾⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣤⣾⠟⢋⣥⣤⠀⣶⣶⣶⣦⣤⣌⣉⠛⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⣴⣿⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠋⢁⣴⣿⣿⡿⠀⣿⣿⣿⣿⣿⣿⣿⣷⡄⠀⠀⠀⠀⠀ ⠀⠀⠀⣼⣿⠟⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣠⣤⣤⣶⣶⣾⣿⣿⣿⣿⣷⣶⡆⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣠⣿⣿⣿⠁⠀⠀⢹⣿⣿⣿⣿⣿⣿⢻⣿⡄⠀⠀⠀⠀ ⠀⠀⠀⠛⠋⠀⠀⠀⠀⠀⠀⠀⢀⣤⣾⣿⠿⠛⣛⣉⣉⣀⣀⡀⠀⠀⠀⠀⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣸⣿⣿⣿⣿⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⢸⣿⣿⡄⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣾⡿⢋⣩⣶⣾⣿⣿⣿⣿⣿⣿⣿⣿⣶⣦⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢠⣿⣿⣿⣿⣿⣦⣀⣀⣴⣿⣿⣿⣿⣿⡿⢸⣿⢿⣷⡀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⣡⣄⠀⠋⠁⠀⠈⠹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣷⣄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣸⣿⡟⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡇⢸⡿⠀⠛⠃⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⣧⡀⠀⠀⠀⠀⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣦⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠛⠛⠃⢹⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⠈⠁⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⣿⢿⣿⣿⣿⣷⣦⣤⣾⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⣶⣶⠀⠈⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡟⠀⣿⠇⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⢠⣿⣿⣿⠟⠉⠀⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢿⣿⠀⠀⢹⣿⣿⣿⣿⣿⣿⣿⣿⣿⠁⢸⣿⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⣼⣿⡟⠁⣠⣦⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⠉⣿⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢸⣿⡆⠀⠀⢻⣿⣿⣿⣿⣿⣿⣿⠏⠀⣸⡏⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⣿⡏⠀⠀⣿⣿⡀⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠀⢹⣿⣧⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⣿⣇⠀⠀⠀⠙⢿⣿⣿⡿⠟⠁⠀⣸⡿⠁⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⢸⣿⠁⠀⠀⢸⣿⣇⠀⠀⠘⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⠀⢀⣀⡀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⢿⣦⡀⠀⠀⠀⠈⠉⠀⠀⠀⣼⡿⠁⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠈⠁⠀⠀⠀⠀⢿⣿⡄⠀⠀⠈⢿⣿⣿⣿⣿⣿⣿⣿⣿⣿⡿⠁⠀⠀⣼⣿⡇⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠙⢿⣷⣦⣄⣀⠀⠀⢀⡈⠙⠁⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⢻⣿⣆⠀⠀⠀⠉⠛⠿⢿⣿⣿⠿⠛⠁⠀⠀⠀⣠⣿⣿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⠿⣿⣿⣷⣿⣯⣤⣶⠄⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠹⣿⣷⣀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⣴⣿⣿⠃⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠉⠉⠙⠛⠋⠁⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠙⢿⣷⣤⣀⠀⠀⠀⠀⠀⠀⠀⠺⣿⣿⡿⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠈⠛⢻⣿⣶⣤⣤⣤⣶⣷⣤⠈⠁⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⢿⣿⣿⣿⣿⡿⠿⠛⠋⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠉⠉⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⢀⡄⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠘⠶⢤⣄⣀⣀⣤⠶⠋⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ :::: :::: ::::::::::: ::: ::: ::::::::: ::: ::: :::::::: ::::::::::: ::: ::::::::::: ::::::::::: :::::::: :::: ::: +:+:+: :+:+:+ :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+: :+:+: :+: +:+ +:+:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ +:+ :+:+:+ +:+ +#+ +:+ +#+ +#+ +#+ +#++:++#++: +#+ +:+ +#++: +#++:++#++ +#+ +#++:++#++: +#+ +#+ +#+ +:+ +#+ +:+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+ +#+#+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+# #+#+# ### ### ########### ########## ### ### ######### ### ######## ### ### ### ### ########### ######## ### #### ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ ⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀⠀ */ contract MiladyStationUpgraded is ERC721A, ERC721AQueryable, ERC721ABurnable, OperatorFilterer, Ownable, ERC2981 { uint public constant s_maxMiladyPurchase = 30; uint256 public constant s_MAXMILADYSTATIONS = 1212; uint256 public constant s_miladystationOG = 362; string public MILADYSTATION_PROVENANCE = ''; string public IPFSURI = ''; bool public s_saleIsActive = false; bool public operatorFilteringEnabled; IERC721Enumerable milady = IERC721Enumerable(0x5Af0D9827E0c53E4799BB226655A1de152A425a5); IERC721Enumerable ghiblady = IERC721Enumerable(0x186E74aD45bF81fb3712e9657560f8f6361cbBef); IERC721Enumerable pixelady = IERC721Enumerable(0x8Fc0D90f2C45a5e7f94904075c952e0943CFCCfd); IERC721Enumerable cig = IERC721Enumerable(0xEEd41d06AE195CA8f5CaCACE4cd691EE75F0683f); mapping(address => bool) public whitelistOneMint; mapping(address => bool) public miladyMinted; constructor() ERC721A("MiladyStation", "MS") { _registerForOperatorFiltering(); operatorFilteringEnabled = true; // Set royalty receiver to the contract creator, // at 5% (default denominator is 10000). _setDefaultRoyalty(msg.sender, 500); _initializeOwner(msg.sender); } function setProvenanceHash(string memory provenanceHash) public onlyOwner { MILADYSTATION_PROVENANCE = provenanceHash; } function setBaseURI(string memory baseURI) public onlyOwner { IPFSURI = baseURI; } function _baseURI() internal view virtual override(ERC721A) returns (string memory) { return IPFSURI; } // the following functions are overrides for creator fee opensea default operator stuff function setApprovalForAll(address operator, bool approved) public override (IERC721A, ERC721A) onlyAllowedOperatorApproval(operator) { super.setApprovalForAll(operator, approved); } function approve(address operator, uint256 tokenId) public payable override (IERC721A, ERC721A) onlyAllowedOperatorApproval(operator) { super.approve(operator, tokenId); } function transferFrom(address from, address to, uint256 tokenId) public payable override (IERC721A, ERC721A) onlyAllowedOperator(from) { super.transferFrom(from, to, tokenId); } function safeTransferFrom(address from, address to, uint256 tokenId) public payable override (IERC721A, ERC721A) onlyAllowedOperator(from) { super.safeTransferFrom(from, to, tokenId); } function safeTransferFrom(address from, address to, uint256 tokenId, bytes memory data) public payable override (IERC721A, ERC721A) onlyAllowedOperator(from) { super.safeTransferFrom(from, to, tokenId, data); } function supportsInterface(bytes4 interfaceId) public view virtual override (IERC721A, ERC721A, ERC2981) returns (bool) { // Supports the following `interfaceId`s: // - IERC165: 0x01ffc9a7 // - IERC721: 0x80ac58cd // - IERC721Metadata: 0x5b5e139f // - IERC2981: 0x2a55205a return ERC721A.supportsInterface(interfaceId) || ERC2981.supportsInterface(interfaceId); } function setDefaultRoyalty(address receiver, uint96 feeNumerator) public onlyOwner { _setDefaultRoyalty(receiver, feeNumerator); } function setOperatorFilteringEnabled(bool value) public onlyOwner { operatorFilteringEnabled = value; } function _operatorFilteringEnabled() internal view override returns (bool) { return operatorFilteringEnabled; } function _isPriorityOperator(address operator) internal pure override returns (bool) { // OpenSea Seaport Conduit: // https://etherscan.io/address/0x1E0049783F008A0085193E00003D00cd54003c71 // https://goerli.etherscan.io/address/0x1E0049783F008A0085193E00003D00cd54003c71 return operator == address(0x1E0049783F008A0085193E00003D00cd54003c71); } function withdraw() public onlyOwner { uint balance = address(this).balance; payable(msg.sender).transfer(balance); } // Milady check function miladyHolderCheck(address holder) public view returns (uint256) { uint256 tokenNum = 0; try milady.balanceOf(holder) returns (uint256 miladyHolderIndex) { // First token owned by user tokenNum = miladyHolderIndex; } catch (bytes memory) { // No tokens owned by user } return tokenNum; } //Friend check function miladyFriendCheck(address holder) public view returns (uint256) { //fake token number uint256 tokenNum = 0; try ghiblady.balanceOf(holder) returns (uint256 miladyGhibIndex) { // First token owned by user tokenNum += miladyGhibIndex; } catch (bytes memory) { // No tokens owned by user } try pixelady.balanceOf(holder) returns (uint256 miladyPixIndex) { // First token owned by user tokenNum += miladyPixIndex; } catch (bytes memory) { // No tokens owned by user } try cig.balanceOf(holder) returns (uint256 miladyCigIndex) { // First token owned by user tokenNum += miladyCigIndex; } catch (bytes memory) { // No tokens owned by user } return tokenNum; } // secret free mint function editWhitelistOne(address[] memory array) public onlyOwner { for(uint256 i = 0; i < array.length; i++) { address addressElement = array[i]; whitelistOneMint[addressElement] = true; } } function reserveMintMiladyStations(address target) public onlyOwner { require(totalSupply() == 0, "the time for this has passed."); uint256 numberOfTokens = s_miladystationOG; _mint(target, numberOfTokens); } function reserveMintMiladys() public { uint256 miladys = miladyHolderCheck(msg.sender); if (miladys > 0 && !miladyMinted[msg.sender]){ miladyMinted[msg.sender] = true; _safeMint(msg.sender,1); } else { require(false, "Nice try buster, miladys only"); } } function reserveMintWhitelist() public { if (msg.sender == owner()){ _safeMint(msg.sender, 10); _safeMint(msg.sender, 10); _safeMint(msg.sender, 10); } else if (whitelistOneMint[msg.sender]){ whitelistOneMint[msg.sender] = false; _safeMint(msg.sender, 2); } else { require(false, "Nice try buster, not on the list"); } } function flipSaleState() public onlyOwner { s_saleIsActive = !s_saleIsActive; } function mintMiladys(uint256 numberOfTokens) public payable { require(s_saleIsActive, "Sale must be active to mint Miladys"); require(numberOfTokens <= 30, "Can only mint up to 30 tokens at a time"); require(totalSupply() + numberOfTokens < 1212, "Purchase would exceed max supply of Miladys"); require(miladyHolderCheck(msg.sender) > 0, "wait but you don't have a milady.. "); //leaving in miladyholder check for posterity uint256 miladyPrice; //Prices defined 12/3/22 //Mint for Miladys if (numberOfTokens == 30) { miladyPrice = 2000000000000000; // 0.002 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else if (numberOfTokens >= 15) { miladyPrice = 3000000000000000; // 0.003 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else if (numberOfTokens >= 5) { miladyPrice = 4000000000000000; // 0.004 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else { miladyPrice = 5000000000000000; // 0.005 ETH should be 6 dollars require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } //mint in batches of 8 to save gas on transfers if(numberOfTokens % 8 == 0){ for(uint i = 0; i < numberOfTokens/8; i++) { _mint(msg.sender,8); } } else { for(uint i = 0; i < numberOfTokens/8; i++) { _mint(msg.sender,8); } _mint(msg.sender,numberOfTokens%8); } } function mintFriends(uint256 numberOfTokens) public payable { require(s_saleIsActive, "Sale must be active to mint MiladyStations"); require(numberOfTokens <= 30, "Can only mint up to 30 tokens at a time"); require(totalSupply() + numberOfTokens < 1212, "Purchase would exceed max supply of MiladyStations"); // require(miladyFriendCheck(msg.sender) > 0, "pick up a ghib, pixel or cig for friend price."); //no friend check to save gas. exploitable, but really everyone is friend so is ok uint256 miladyPrice; //Mint for Friend of Milady //Ghiblady, Pixelady, Cigawrette Honorable Mentions: Miaura, Sonora, Milaidy if (numberOfTokens == 30) { miladyPrice = 6000000000000000; // 0.006 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else if (numberOfTokens >= 15) { miladyPrice = 7000000000000000; // 0.007 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else if (numberOfTokens >= 5) { miladyPrice = 8000000000000000; // 0.008 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else { miladyPrice = 9000000000000000; // 0.009 ETH should be 11 dollars require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } //mint in batches of 8 to save gas on transfers if(numberOfTokens % 8 == 0){ for(uint i = 0; i < numberOfTokens/8; i++) { _mint(msg.sender,8); } } else { for(uint i = 0; i < numberOfTokens/8; i++) { _mint(msg.sender,8); } _mint(msg.sender,numberOfTokens%8); } } function mintNew(uint256 numberOfTokens) public payable { require(s_saleIsActive, "Sale must be active to mint MiladyStations"); require(numberOfTokens <= 30, "Can only mint up to 30 tokens at a time"); require(totalSupply() + numberOfTokens < 1212, "Purchase would exceed max supply of MiladyStations"); uint256 miladyPrice; if (numberOfTokens == 30) { miladyPrice = 9000000000000000; /// 0.009 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else if (numberOfTokens >= 15) { miladyPrice = 10000000000000000; // 0.010 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else if (numberOfTokens >= 5) { miladyPrice = 11000000000000000; // 0.011 ETH require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } else { miladyPrice = 12000000000000000; // 0.012 ETH should be 15 dollars require(miladyPrice*(numberOfTokens) <= msg.value, "Ether value sent is not correct"); } //mint in batches of 8 to save gas on erc721A transferFrom if(numberOfTokens % 8 == 0){ for(uint i = 0; i < numberOfTokens/8; i++) { _mint(msg.sender,8); } } else { for(uint i = 0; i < numberOfTokens/8; i++) { _mint(msg.sender,8); } _mint(msg.sender,numberOfTokens%8); } } }
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":[],"name":"NewOwnerIsZeroAddress","type":"error"},{"inputs":[],"name":"NoHandoverRequest","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"},{"inputs":[],"name":"Unauthorized","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":"pendingOwner","type":"address"}],"name":"OwnershipHandoverCanceled","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pendingOwner","type":"address"}],"name":"OwnershipHandoverRequested","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"oldOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"IPFSURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MILADYSTATION_PROVENANCE","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"operator","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":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"cancelOwnershipHandover","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"pendingOwner","type":"address"}],"name":"completeOwnershipHandover","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address[]","name":"array","type":"address[]"}],"name":"editWhitelistOne","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":[],"name":"flipSaleState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"holder","type":"address"}],"name":"miladyFriendCheck","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"holder","type":"address"}],"name":"miladyHolderCheck","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"miladyMinted","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfTokens","type":"uint256"}],"name":"mintFriends","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfTokens","type":"uint256"}],"name":"mintMiladys","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfTokens","type":"uint256"}],"name":"mintNew","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"operatorFilteringEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"result","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":[{"internalType":"address","name":"pendingOwner","type":"address"}],"name":"ownershipHandoverExpiresAt","outputs":[{"internalType":"uint256","name":"result","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ownershipHandoverValidFor","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[],"name":"requestOwnershipHandover","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"target","type":"address"}],"name":"reserveMintMiladyStations","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"reserveMintMiladys","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"reserveMintWhitelist","outputs":[],"stateMutability":"nonpayable","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":[],"name":"s_MAXMILADYSTATIONS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"s_maxMiladyPurchase","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"s_miladystationOG","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"s_saleIsActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"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":"baseURI","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":"bool","name":"value","type":"bool"}],"name":"setOperatorFilteringEnabled","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"provenanceHash","type":"string"}],"name":"setProvenanceHash","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":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"whitelistOneMint","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
608060405260405180602001604052806000815250600a9081620000249190620007cc565b5060405180602001604052806000815250600b9081620000459190620007cc565b506000600c60006101000a81548160ff021916908315150217905550735af0d9827e0c53e4799bb226655a1de152a425a5600c60026101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555073186e74ad45bf81fb3712e9657560f8f6361cbbef600d60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550738fc0d90f2c45a5e7f94904075c952e0943cfccfd600e60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555073eed41d06ae195ca8f5cacace4cd691ee75f0683f600f60006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550348015620001c257600080fd5b506040518060400160405280600d81526020017f4d696c61647953746174696f6e000000000000000000000000000000000000008152506040518060400160405280600281526020017f4d530000000000000000000000000000000000000000000000000000000000008152508160029081620002409190620007cc565b508060039081620002529190620007cc565b5062000263620002c160201b60201c565b60008190555050506200027b620002c660201b60201c565b6001600c60016101000a81548160ff021916908315150217905550620002aa336101f4620002ef60201b60201c565b620002bb336200049260201b60201c565b620009ce565b600090565b620002ed733cc6cdda760b79bafa08df41ecfa224f810dceb66001620004ce60201b60201c565b565b620002ff6200054860201b60201c565b6bffffffffffffffffffffffff16816bffffffffffffffffffffffff16111562000360576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040162000357906200093a565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603620003d2576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401620003c990620009ac565b60405180910390fd5b60405180604001604052808373ffffffffffffffffffffffffffffffffffffffff168152602001826bffffffffffffffffffffffff16815250600860008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a8154816bffffffffffffffffffffffff02191690836bffffffffffffffffffffffff1602179055509050505050565b8060601b60601c905080638b78c6d819558060007f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a350565b637d3e3dbe8260601b60601c925081620004fd5782620004f557634420e4869050620004fd565b63a0af290390505b8060e01b60005230600452826024526004600060446000806daaeb6d7670e522a718067333cd4e5af16200053e578060005160e01c036200053d57600080fd5b5b6000602452505050565b6000612710905090565b600081519050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b60006002820490506001821680620005d457607f821691505b602082108103620005ea57620005e96200058c565b5b50919050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b600060088302620006547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8262000615565b62000660868362000615565b95508019841693508086168417925050509392505050565b6000819050919050565b6000819050919050565b6000620006ad620006a7620006a18462000678565b62000682565b62000678565b9050919050565b6000819050919050565b620006c9836200068c565b620006e1620006d882620006b4565b84845462000622565b825550505050565b600090565b620006f8620006e9565b62000705818484620006be565b505050565b5b818110156200072d5762000721600082620006ee565b6001810190506200070b565b5050565b601f8211156200077c576200074681620005f0565b620007518462000605565b8101602085101562000761578190505b62000779620007708562000605565b8301826200070a565b50505b505050565b600082821c905092915050565b6000620007a16000198460080262000781565b1980831691505092915050565b6000620007bc83836200078e565b9150826002028217905092915050565b620007d78262000552565b67ffffffffffffffff811115620007f357620007f26200055d565b5b620007ff8254620005bb565b6200080c82828562000731565b600060209050601f8311600181146200084457600084156200082f578287015190505b6200083b8582620007ae565b865550620008ab565b601f1984166200085486620005f0565b60005b828110156200087e5784890151825560018201915060208501945060208101905062000857565b868310156200089e57848901516200089a601f8916826200078e565b8355505b6001600288020188555050505b505050505050565b600082825260208201905092915050565b7f455243323938313a20726f79616c7479206665652077696c6c2065786365656460008201527f2073616c65507269636500000000000000000000000000000000000000000000602082015250565b600062000922602a83620008b3565b91506200092f82620008c4565b604082019050919050565b60006020820190508181036000830152620009558162000913565b9050919050565b7f455243323938313a20696e76616c696420726563656976657200000000000000600082015250565b600062000994601983620008b3565b9150620009a1826200095c565b602082019050919050565b60006020820190508181036000830152620009c78162000985565b9050919050565b615a7a80620009de6000396000f3fe6080604052600436106102ff5760003560e01c80635bbb217711610190578063b6d8e86f116100dc578063d7533f0211610095578063f2fde38b1161006f578063f2fde38b14610ae8578063fb796e6c14610b04578063fee81cf414610b2f578063ff35def614610b6c576102ff565b8063d7533f0214610a64578063e985e9c514610a8f578063f04e283e14610acc576102ff565b8063b6d8e86f14610953578063b7c0b8e81461097c578063b88d4fde146109a5578063c23dc68f146109c1578063c45fc170146109fe578063c87b56dd14610a27576102ff565b80638b83df451161014957806399a2557a1161012357806399a2557a146108a6578063a22cb465146108e3578063b1e283de1461090c578063b6ca7ea314610928576102ff565b80638b83df45146108255780638da5cb5b1461085057806395d89b411461087b576102ff565b80635bbb2177146106ea5780636352211e1461072757806370a0823114610764578063715018a6146107a15780637d5fcf9c146107ab5780638462151c146107e8576102ff565b806334918dfd1161024f578063448dbc9c116102085780634b01570a116101e25780634b01570a1461064f57806354d1f13d1461068c57806355f804b3146106965780635854d949146106bf576102ff565b8063448dbc9c146105f1578063451807b71461061c578063487bddf614610633576102ff565b806334918dfd146105045780633494f1bf1461051b5780633ccfd60b146105585780633eab94351461056f57806342842e0e146105ac57806342966c68146105c8576102ff565b80630a5ef3e3116102bc57806323b872dd1161029657806323b872dd1461048957806325692962146104a55780632a55205a146104af578063323ab474146104ed576102ff565b80630a5ef3e31461040a578063109695231461043557806318160ddd1461045e576102ff565b806301ffc9a71461030457806304634d8d1461034157806306a94cf61461036a57806306fdde0314610386578063081812fc146103b1578063095ea7b3146103ee575b600080fd5b34801561031057600080fd5b5061032b600480360381019061032691906141bd565b610b97565b6040516103389190614205565b60405180910390f35b34801561034d57600080fd5b50610368600480360381019061036391906142c2565b610bb9565b005b610384600480360381019061037f9190614338565b610bcf565b005b34801561039257600080fd5b5061039b610ee4565b6040516103a891906143f5565b60405180910390f35b3480156103bd57600080fd5b506103d860048036038101906103d39190614338565b610f76565b6040516103e59190614426565b60405180910390f35b61040860048036038101906104039190614441565b610ff5565b005b34801561041657600080fd5b5061041f61102a565b60405161042c9190614490565b60405180910390f35b34801561044157600080fd5b5061045c600480360381019061045791906145e0565b611030565b005b34801561046a57600080fd5b5061047361104b565b6040516104809190614490565b60405180910390f35b6104a3600480360381019061049e9190614629565b611062565b005b6104ad6110cd565b005b3480156104bb57600080fd5b506104d660048036038101906104d1919061467c565b611121565b6040516104e49291906146bc565b60405180910390f35b3480156104f957600080fd5b5061050261130b565b005b34801561051057600080fd5b50610519611424565b005b34801561052757600080fd5b50610542600480360381019061053d91906146e5565b611458565b60405161054f9190614490565b60405180910390f35b34801561056457600080fd5b5061056d61153e565b005b34801561057b57600080fd5b50610596600480360381019061059191906146e5565b611595565b6040516105a39190614205565b60405180910390f35b6105c660048036038101906105c19190614629565b6115b5565b005b3480156105d457600080fd5b506105ef60048036038101906105ea9190614338565b611620565b005b3480156105fd57600080fd5b5061060661162e565b6040516106139190614490565b60405180910390f35b34801561062857600080fd5b50610631611634565b005b61064d60048036038101906106489190614338565b611793565b005b34801561065b57600080fd5b50610676600480360381019061067191906146e5565b611aa8565b6040516106839190614490565b60405180910390f35b610694611d5b565b005b3480156106a257600080fd5b506106bd60048036038101906106b891906145e0565b611d97565b005b3480156106cb57600080fd5b506106d4611db2565b6040516106e191906143f5565b60405180910390f35b3480156106f657600080fd5b50610711600480360381019061070c9190614772565b611e40565b60405161071e9190614922565b60405180910390f35b34801561073357600080fd5b5061074e60048036038101906107499190614338565b611f03565b60405161075b9190614426565b60405180910390f35b34801561077057600080fd5b5061078b600480360381019061078691906146e5565b611f15565b6040516107989190614490565b60405180910390f35b6107a9611fcd565b005b3480156107b757600080fd5b506107d260048036038101906107cd91906146e5565b611fe1565b6040516107df9190614205565b60405180910390f35b3480156107f457600080fd5b5061080f600480360381019061080a91906146e5565b612001565b60405161081c9190614a02565b60405180910390f35b34801561083157600080fd5b5061083a612144565b60405161084791906143f5565b60405180910390f35b34801561085c57600080fd5b506108656121d2565b6040516108729190614426565b60405180910390f35b34801561088757600080fd5b506108906121e0565b60405161089d91906143f5565b60405180910390f35b3480156108b257600080fd5b506108cd60048036038101906108c89190614a24565b612272565b6040516108da9190614a02565b60405180910390f35b3480156108ef57600080fd5b5061090a60048036038101906109059190614aa3565b61247e565b005b61092660048036038101906109219190614338565b6124b3565b005b34801561093457600080fd5b5061093d612813565b60405161094a9190614490565b60405180910390f35b34801561095f57600080fd5b5061097a60048036038101906109759190614ba6565b612818565b005b34801561098857600080fd5b506109a3600480360381019061099e9190614bef565b6128bb565b005b6109bf60048036038101906109ba9190614cbd565b6128e0565b005b3480156109cd57600080fd5b506109e860048036038101906109e39190614338565b61294d565b6040516109f59190614d95565b60405180910390f35b348015610a0a57600080fd5b50610a256004803603810190610a2091906146e5565b6129b7565b005b348015610a3357600080fd5b50610a4e6004803603810190610a499190614338565b612a1e565b604051610a5b91906143f5565b60405180910390f35b348015610a7057600080fd5b50610a79612abc565b604051610a869190614dbf565b60405180910390f35b348015610a9b57600080fd5b50610ab66004803603810190610ab19190614dda565b612ac7565b604051610ac39190614205565b60405180910390f35b610ae66004803603810190610ae191906146e5565b612b5b565b005b610b026004803603810190610afd91906146e5565b612bc9565b005b348015610b1057600080fd5b50610b19612c43565b604051610b269190614205565b60405180910390f35b348015610b3b57600080fd5b50610b566004803603810190610b5191906146e5565b612c56565b604051610b639190614490565b60405180910390f35b348015610b7857600080fd5b50610b81612c71565b604051610b8e9190614205565b60405180910390f35b6000610ba282612c84565b80610bb25750610bb182612d16565b5b9050919050565b610bc1612d90565b610bcb8282612dad565b5050565b600c60009054906101000a900460ff16610c1e576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c1590614e8c565b60405180910390fd5b601e811115610c62576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c5990614f1e565b60405180910390fd5b6104bc81610c6e61104b565b610c789190614f6d565b10610cb8576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610caf90615013565b60405180910390fd5b6000601e8203610d1f57661550f7dca700009050348282610cd99190615033565b1115610d1a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d11906150c1565b60405180910390fd5b610e44565b600f8210610d84576618de76816d80009050348282610d3e9190615033565b1115610d7f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d76906150c1565b60405180910390fd5b610e43565b60058210610de957661c6bf5263400009050348282610da39190615033565b1115610de4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ddb906150c1565b60405180910390fd5b610e42565b661ff973cafa80009050348282610e009190615033565b1115610e41576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e38906150c1565b60405180910390fd5b5b5b5b6000600883610e539190615110565b03610e935760005b600883610e689190615141565b811015610e8d57610e7a336008612f42565b8080610e8590615172565b915050610e5b565b50610ee0565b60005b600883610ea39190615141565b811015610ec857610eb5336008612f42565b8080610ec090615172565b915050610e96565b50610edf33600884610eda9190615110565b612f42565b5b5050565b606060028054610ef3906151e9565b80601f0160208091040260200160405190810160405280929190818152602001828054610f1f906151e9565b8015610f6c5780601f10610f4157610100808354040283529160200191610f6c565b820191906000526020600020905b815481529060010190602001808311610f4f57829003601f168201915b5050505050905090565b6000610f81826130fd565b610fb7576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b81610fff8161315c565b61101b5761100b6131a8565b1561101a57611019816131bf565b5b5b6110258383613203565b505050565b6104bc81565b611038612d90565b80600a908161104791906153c6565b5050565b6000611055613213565b6001546000540303905090565b823373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146110bc5761109f3361315c565b6110bb576110ab6131a8565b156110ba576110b9336131bf565b5b5b5b6110c7848484613218565b50505050565b60006110d7612abc565b67ffffffffffffffff164201905063389a75e1600c5233600052806020600c2055337fdbf36a107da19e49527a7176a1babf963b4b0ff8cde35ee35d6cd8f1f9ac7e1d600080a250565b6000806000600960008681526020019081526020016000206040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff16815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff16036112b65760086040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff168152505090505b60006112c061353a565b6bffffffffffffffffffffffff1682602001516bffffffffffffffffffffffff16866112ec9190615033565b6112f69190615141565b90508160000151819350935050509250929050565b600061131633611458565b90506000811180156113725750601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16155b156113df576001601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055506113da336001613544565b611421565b6000611420576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611417906154e4565b60405180910390fd5b5b50565b61142c612d90565b600c60009054906101000a900460ff1615600c60006101000a81548160ff021916908315150217905550565b60008060009050600c60029054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b81526004016114ba9190614426565b602060405180830381865afa9250505080156114f457506040513d601f19601f820116820180604052508101906114f19190615519565b60015b611530573d8060008114611524576040519150601f19603f3d011682016040523d82523d6000602084013e611529565b606091505b5050611535565b809150505b80915050919050565b611546612d90565b60004790503373ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f19350505050158015611591573d6000803e3d6000fd5b5050565b60116020528060005260406000206000915054906101000a900460ff1681565b823373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161461160f576115f23361315c565b61160e576115fe6131a8565b1561160d5761160c336131bf565b5b5b5b61161a848484613562565b50505050565b61162b816001613582565b50565b61016a81565b61163c6121d2565b73ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff16036116945761167933600a613544565b61168433600a613544565b61168f33600a613544565b611791565b601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff161561174e576000601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff021916908315150217905550611749336002613544565b611790565b600061178f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161178690615592565b60405180910390fd5b5b5b565b600c60009054906101000a900460ff166117e2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016117d990614e8c565b60405180910390fd5b601e811115611826576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161181d90614f1e565b60405180910390fd5b6104bc8161183261104b565b61183c9190614f6d565b1061187c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161187390615013565b60405180910390fd5b6000601e82036118e357661ff973cafa8000905034828261189d9190615033565b11156118de576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016118d5906150c1565b60405180910390fd5b611a08565b600f821061194857662386f26fc1000090503482826119029190615033565b1115611943576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161193a906150c1565b60405180910390fd5b611a07565b600582106119ad57662714711487800090503482826119679190615033565b11156119a8576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161199f906150c1565b60405180910390fd5b611a06565b662aa1efb94e000090503482826119c49190615033565b1115611a05576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016119fc906150c1565b60405180910390fd5b5b5b5b6000600883611a179190615110565b03611a575760005b600883611a2c9190615141565b811015611a5157611a3e336008612f42565b8080611a4990615172565b915050611a1f565b50611aa4565b60005b600883611a679190615141565b811015611a8c57611a79336008612f42565b8080611a8490615172565b915050611a5a565b50611aa333600884611a9e9190615110565b612f42565b5b5050565b60008060009050600d60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b8152600401611b0a9190614426565b602060405180830381865afa925050508015611b4457506040513d601f19601f82011682018060405250810190611b419190615519565b60015b611b80573d8060008114611b74576040519150601f19603f3d011682016040523d82523d6000602084013e611b79565b606091505b5050611b90565b8082611b8c9190614f6d565b9150505b600e60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b8152600401611beb9190614426565b602060405180830381865afa925050508015611c2557506040513d601f19601f82011682018060405250810190611c229190615519565b60015b611c61573d8060008114611c55576040519150601f19603f3d011682016040523d82523d6000602084013e611c5a565b606091505b5050611c71565b8082611c6d9190614f6d565b9150505b600f60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b8152600401611ccc9190614426565b602060405180830381865afa925050508015611d0657506040513d601f19601f82011682018060405250810190611d039190615519565b60015b611d42573d8060008114611d36576040519150601f19603f3d011682016040523d82523d6000602084013e611d3b565b606091505b5050611d52565b8082611d4e9190614f6d565b9150505b80915050919050565b63389a75e1600c523360005260006020600c2055337ffa7b8eab7da67f412cc9575ed43464468f9bfbae89d1675917346ca6d8fe3c92600080a2565b611d9f612d90565b80600b9081611dae91906153c6565b5050565b600a8054611dbf906151e9565b80601f0160208091040260200160405190810160405280929190818152602001828054611deb906151e9565b8015611e385780601f10611e0d57610100808354040283529160200191611e38565b820191906000526020600020905b815481529060010190602001808311611e1b57829003601f168201915b505050505081565b6060600083839050905060008167ffffffffffffffff811115611e6657611e656144b5565b5b604051908082528060200260200182016040528015611e9f57816020015b611e8c614102565b815260200190600190039081611e845790505b50905060005b828114611ef757611ece868683818110611ec257611ec16155b2565b5b9050602002013561294d565b828281518110611ee157611ee06155b2565b5b6020026020010181905250806001019050611ea5565b50809250505092915050565b6000611f0e826137d4565b9050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611f7c576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b611fd5612d90565b611fdf60006138cc565b565b60106020528060005260406000206000915054906101000a900460ff1681565b6060600080600061201185611f15565b905060008167ffffffffffffffff81111561202f5761202e6144b5565b5b60405190808252806020026020018201604052801561205d5781602001602082028036833780820191505090505b509050612068614102565b6000612072613213565b90505b838614612136576120858161390a565b9150816040015161212b57600073ffffffffffffffffffffffffffffffffffffffff16826000015173ffffffffffffffffffffffffffffffffffffffff16146120d057816000015194505b8773ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff160361212a578083878060010198508151811061211d5761211c6155b2565b5b6020026020010181815250505b5b806001019050612075565b508195505050505050919050565b600b8054612151906151e9565b80601f016020809104026020016040519081016040528092919081815260200182805461217d906151e9565b80156121ca5780601f1061219f576101008083540402835291602001916121ca565b820191906000526020600020905b8154815290600101906020018083116121ad57829003601f168201915b505050505081565b6000638b78c6d81954905090565b6060600380546121ef906151e9565b80601f016020809104026020016040519081016040528092919081815260200182805461221b906151e9565b80156122685780601f1061223d57610100808354040283529160200191612268565b820191906000526020600020905b81548152906001019060200180831161224b57829003601f168201915b5050505050905090565b60608183106122ad576040517f32c1995a00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000806122b8613935565b90506122c2613213565b8510156122d4576122d1613213565b94505b808411156122e0578093505b60006122eb87611f15565b90508486101561230e576000868603905081811015612308578091505b50612313565b600090505b60008167ffffffffffffffff81111561232f5761232e6144b5565b5b60405190808252806020026020018201604052801561235d5781602001602082028036833780820191505090505b509050600082036123745780945050505050612477565b600061237f8861294d565b90506000816040015161239457816000015190505b60008990505b8881141580156123aa5750848714155b15612469576123b88161390a565b9250826040015161245e57600073ffffffffffffffffffffffffffffffffffffffff16836000015173ffffffffffffffffffffffffffffffffffffffff161461240357826000015191505b8a73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff160361245d57808488806001019950815181106124505761244f6155b2565b5b6020026020010181815250505b5b80600101905061239a565b508583528296505050505050505b9392505050565b816124888161315c565b6124a4576124946131a8565b156124a3576124a2816131bf565b5b5b6124ae838361393e565b505050565b600c60009054906101000a900460ff16612502576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016124f990615653565b60405180910390fd5b601e811115612546576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161253d90614f1e565b60405180910390fd5b6104bc8161255261104b565b61255c9190614f6d565b1061259c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612593906156e5565b60405180910390fd5b60006125a733611458565b116125e7576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016125de90615777565b60405180910390fd5b6000601e820361264e5766071afd498d000090503482826126089190615033565b1115612649576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612640906150c1565b60405180910390fd5b612773565b600f82106126b357660aa87bee538000905034828261266d9190615033565b11156126ae576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016126a5906150c1565b60405180910390fd5b612772565b6005821061271857660e35fa931a000090503482826126d29190615033565b1115612713576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161270a906150c1565b60405180910390fd5b612771565b6611c37937e08000905034828261272f9190615033565b1115612770576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612767906150c1565b60405180910390fd5b5b5b5b60006008836127829190615110565b036127c25760005b6008836127979190615141565b8110156127bc576127a9336008612f42565b80806127b490615172565b91505061278a565b5061280f565b60005b6008836127d29190615141565b8110156127f7576127e4336008612f42565b80806127ef90615172565b9150506127c5565b5061280e336008846128099190615110565b612f42565b5b5050565b601e81565b612820612d90565b60005b81518110156128b7576000828281518110612841576128406155b2565b5b602002602001015190506001601060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055505080806128af90615172565b915050612823565b5050565b6128c3612d90565b80600c60016101000a81548160ff02191690831515021790555050565b833373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161461293a5761291d3361315c565b612939576129296131a8565b1561293857612937336131bf565b5b5b5b61294685858585613a49565b5050505050565b612955614102565b61295d614102565b612965613213565b8310806129795750612975613935565b8310155b1561298757809150506129b2565b6129908361390a565b90508060400151156129a557809150506129b2565b6129ae83613abc565b9150505b919050565b6129bf612d90565b60006129c961104b565b14612a09576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612a00906157e3565b60405180910390fd5b600061016a9050612a1a8282612f42565b5050565b6060612a29826130fd565b612a5f576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000612a69613adc565b90506000815103612a895760405180602001604052806000815250612ab4565b80612a9384613b6e565b604051602001612aa492919061583f565b6040516020818303038152906040525b915050919050565b60006202a300905090565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b612b63612d90565b63389a75e1600c52806000526020600c208054421115612b8b57636f5e88186000526004601cfd5b60008155600c5160601c80337f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a380638b78c6d81955505050565b612bd1612d90565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603612c37576040517f7448fbae00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612c40816138cc565b50565b600c60019054906101000a900460ff1681565b600063389a75e1600c52816000526020600c20549050919050565b600c60009054906101000a900460ff1681565b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480612cdf57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80612d0f5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b60007f2a55205a000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480612d895750612d8882613bbe565b5b9050919050565b638b78c6d819543314612dab576382b429006000526004601cfd5b565b612db561353a565b6bffffffffffffffffffffffff16816bffffffffffffffffffffffff161115612e13576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612e0a906158d5565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603612e82576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612e7990615941565b60405180910390fd5b60405180604001604052808373ffffffffffffffffffffffffffffffffffffffff168152602001826bffffffffffffffffffffffff16815250600860008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a8154816bffffffffffffffffffffffff02191690836bffffffffffffffffffffffff1602179055509050505050565b60008054905060008203612f82576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612f8f6000848385613c28565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254019250508190555061300683612ff76000866000613c2e565b61300085613c56565b17613c66565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b8181146130a757808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a460018101905061306c565b50600082036130e2576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060008190555050506130f86000848385613c91565b505050565b600081613108613213565b11158015613117575060005482105b8015613155575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b6000731e0049783f008a0085193e00003d00cd54003c7173ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16149050919050565b6000600c60019054906101000a900460ff16905090565b69c617113400112233445560005230601a5280603a52600080604460166daaeb6d7670e522a718067333cd4e5afa6131fb573d6000803e3d6000fd5b6000603a5250565b61320f82826001613c97565b5050565b600090565b6000613223826137d4565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161461328a576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008061329684613de3565b915091506132ac81876132a7613e0a565b613e12565b6132f8576132c1866132bc613e0a565b612ac7565b6132f7576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff160361335e576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61336b8686866001613c28565b801561337657600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001019190508190555061344485613420888887613c2e565b7c020000000000000000000000000000000000000000000000000000000017613c66565b600460008681526020019081526020016000208190555060007c02000000000000000000000000000000000000000000000000000000008416036134ca57600060018501905060006004600083815260200190815260200160002054036134c85760005481146134c7578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a46135328686866001613c91565b505050505050565b6000612710905090565b61355e828260405180602001604052806000815250613e56565b5050565b61357d838383604051806020016040528060008152506128e0565b505050565b600061358d836137d4565b905060008190506000806135a086613de3565b915091508415613609576135bc81846135b7613e0a565b613e12565b613608576135d1836135cc613e0a565b612ac7565b613607576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b5b613617836000886001613c28565b801561362257600082555b600160806001901b03600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055506136ca8361368785600088613c2e565b7c02000000000000000000000000000000000000000000000000000000007c01000000000000000000000000000000000000000000000000000000001717613c66565b600460008881526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000851603613750576000600187019050600060046000838152602001908152602001600020540361374e57600054811461374d578460046000838152602001908152602001600020819055505b5b505b85600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a46137ba836000886001613c91565b600160008154809291906001019190505550505050505050565b6000816137df613213565b11613895576004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821603613894576000810361388f576000548210613864576040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b6004600083600190039350838152602001908152602001600020549050600081036138c757613865565b6138c7565b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b638b78c6d8198160601b60601c91508181547f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a38181555050565b613912614102565b61392e6004600084815260200190815260200160002054613ef3565b9050919050565b60008054905090565b806007600061394b613e0a565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff166139f8613e0a565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c3183604051613a3d9190614205565b60405180910390a35050565b613a54848484611062565b60008373ffffffffffffffffffffffffffffffffffffffff163b14613ab657613a7f84848484613fa9565b613ab5576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b613ac4614102565b613ad5613ad0836137d4565b613ef3565b9050919050565b6060600b8054613aeb906151e9565b80601f0160208091040260200160405190810160405280929190818152602001828054613b17906151e9565b8015613b645780601f10613b3957610100808354040283529160200191613b64565b820191906000526020600020905b815481529060010190602001808311613b4757829003601f168201915b5050505050905090565b606060a060405101806040526020810391506000825281835b600115613ba957600184039350600a81066030018453600a8104905080613b87575b50828103602084039350808452505050919050565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b50505050565b60008060e883901c905060e8613c458686846140f9565b62ffffff16901b9150509392505050565b60006001821460e11b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b6000613ca283611f03565b90508115613d2d578073ffffffffffffffffffffffffffffffffffffffff16613cc9613e0a565b73ffffffffffffffffffffffffffffffffffffffff1614613d2c57613cf581613cf0613e0a565b612ac7565b613d2b576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b5b836006600085815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550828473ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a450505050565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600033905090565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b613e608383612f42565b60008373ffffffffffffffffffffffffffffffffffffffff163b14613eee57600080549050600083820390505b613ea06000868380600101945086613fa9565b613ed6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b818110613e8d578160005414613eeb57600080fd5b50505b505050565b613efb614102565b81816000019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060a082901c816020019067ffffffffffffffff16908167ffffffffffffffff168152505060007c01000000000000000000000000000000000000000000000000000000008316141581604001901515908115158152505060e882901c816060019062ffffff16908162ffffff1681525050919050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02613fcf613e0a565b8786866040518563ffffffff1660e01b8152600401613ff194939291906159b6565b6020604051808303816000875af192505050801561402d57506040513d601f19601f8201168201806040525081019061402a9190615a17565b60015b6140a6573d806000811461405d576040519150601f19603f3d011682016040523d82523d6000602084013e614062565b606091505b50600081510361409e576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b60009392505050565b6040518060800160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff168152602001600015158152602001600062ffffff1681525090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b61419a81614165565b81146141a557600080fd5b50565b6000813590506141b781614191565b92915050565b6000602082840312156141d3576141d261415b565b5b60006141e1848285016141a8565b91505092915050565b60008115159050919050565b6141ff816141ea565b82525050565b600060208201905061421a60008301846141f6565b92915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b600061424b82614220565b9050919050565b61425b81614240565b811461426657600080fd5b50565b60008135905061427881614252565b92915050565b60006bffffffffffffffffffffffff82169050919050565b61429f8161427e565b81146142aa57600080fd5b50565b6000813590506142bc81614296565b92915050565b600080604083850312156142d9576142d861415b565b5b60006142e785828601614269565b92505060206142f8858286016142ad565b9150509250929050565b6000819050919050565b61431581614302565b811461432057600080fd5b50565b6000813590506143328161430c565b92915050565b60006020828403121561434e5761434d61415b565b5b600061435c84828501614323565b91505092915050565b600081519050919050565b600082825260208201905092915050565b60005b8381101561439f578082015181840152602081019050614384565b60008484015250505050565b6000601f19601f8301169050919050565b60006143c782614365565b6143d18185614370565b93506143e1818560208601614381565b6143ea816143ab565b840191505092915050565b6000602082019050818103600083015261440f81846143bc565b905092915050565b61442081614240565b82525050565b600060208201905061443b6000830184614417565b92915050565b600080604083850312156144585761445761415b565b5b600061446685828601614269565b925050602061447785828601614323565b9150509250929050565b61448a81614302565b82525050565b60006020820190506144a56000830184614481565b92915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6144ed826143ab565b810181811067ffffffffffffffff8211171561450c5761450b6144b5565b5b80604052505050565b600061451f614151565b905061452b82826144e4565b919050565b600067ffffffffffffffff82111561454b5761454a6144b5565b5b614554826143ab565b9050602081019050919050565b82818337600083830152505050565b600061458361457e84614530565b614515565b90508281526020810184848401111561459f5761459e6144b0565b5b6145aa848285614561565b509392505050565b600082601f8301126145c7576145c66144ab565b5b81356145d7848260208601614570565b91505092915050565b6000602082840312156145f6576145f561415b565b5b600082013567ffffffffffffffff81111561461457614613614160565b5b614620848285016145b2565b91505092915050565b6000806000606084860312156146425761464161415b565b5b600061465086828701614269565b935050602061466186828701614269565b925050604061467286828701614323565b9150509250925092565b600080604083850312156146935761469261415b565b5b60006146a185828601614323565b92505060206146b285828601614323565b9150509250929050565b60006040820190506146d16000830185614417565b6146de6020830184614481565b9392505050565b6000602082840312156146fb576146fa61415b565b5b600061470984828501614269565b91505092915050565b600080fd5b600080fd5b60008083601f840112614732576147316144ab565b5b8235905067ffffffffffffffff81111561474f5761474e614712565b5b60208301915083602082028301111561476b5761476a614717565b5b9250929050565b600080602083850312156147895761478861415b565b5b600083013567ffffffffffffffff8111156147a7576147a6614160565b5b6147b38582860161471c565b92509250509250929050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b6147f481614240565b82525050565b600067ffffffffffffffff82169050919050565b614817816147fa565b82525050565b614826816141ea565b82525050565b600062ffffff82169050919050565b6148448161482c565b82525050565b60808201600082015161486060008501826147eb565b506020820151614873602085018261480e565b506040820151614886604085018261481d565b506060820151614899606085018261483b565b50505050565b60006148ab838361484a565b60808301905092915050565b6000602082019050919050565b60006148cf826147bf565b6148d981856147ca565b93506148e4836147db565b8060005b838110156149155781516148fc888261489f565b9750614907836148b7565b9250506001810190506148e8565b5085935050505092915050565b6000602082019050818103600083015261493c81846148c4565b905092915050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b61497981614302565b82525050565b600061498b8383614970565b60208301905092915050565b6000602082019050919050565b60006149af82614944565b6149b9818561494f565b93506149c483614960565b8060005b838110156149f55781516149dc888261497f565b97506149e783614997565b9250506001810190506149c8565b5085935050505092915050565b60006020820190508181036000830152614a1c81846149a4565b905092915050565b600080600060608486031215614a3d57614a3c61415b565b5b6000614a4b86828701614269565b9350506020614a5c86828701614323565b9250506040614a6d86828701614323565b9150509250925092565b614a80816141ea565b8114614a8b57600080fd5b50565b600081359050614a9d81614a77565b92915050565b60008060408385031215614aba57614ab961415b565b5b6000614ac885828601614269565b9250506020614ad985828601614a8e565b9150509250929050565b600067ffffffffffffffff821115614afe57614afd6144b5565b5b602082029050602081019050919050565b6000614b22614b1d84614ae3565b614515565b90508083825260208201905060208402830185811115614b4557614b44614717565b5b835b81811015614b6e5780614b5a8882614269565b845260208401935050602081019050614b47565b5050509392505050565b600082601f830112614b8d57614b8c6144ab565b5b8135614b9d848260208601614b0f565b91505092915050565b600060208284031215614bbc57614bbb61415b565b5b600082013567ffffffffffffffff811115614bda57614bd9614160565b5b614be684828501614b78565b91505092915050565b600060208284031215614c0557614c0461415b565b5b6000614c1384828501614a8e565b91505092915050565b600067ffffffffffffffff821115614c3757614c366144b5565b5b614c40826143ab565b9050602081019050919050565b6000614c60614c5b84614c1c565b614515565b905082815260208101848484011115614c7c57614c7b6144b0565b5b614c87848285614561565b509392505050565b600082601f830112614ca457614ca36144ab565b5b8135614cb4848260208601614c4d565b91505092915050565b60008060008060808587031215614cd757614cd661415b565b5b6000614ce587828801614269565b9450506020614cf687828801614269565b9350506040614d0787828801614323565b925050606085013567ffffffffffffffff811115614d2857614d27614160565b5b614d3487828801614c8f565b91505092959194509250565b608082016000820151614d5660008501826147eb565b506020820151614d69602085018261480e565b506040820151614d7c604085018261481d565b506060820151614d8f606085018261483b565b50505050565b6000608082019050614daa6000830184614d40565b92915050565b614db9816147fa565b82525050565b6000602082019050614dd46000830184614db0565b92915050565b60008060408385031215614df157614df061415b565b5b6000614dff85828601614269565b9250506020614e1085828601614269565b9150509250929050565b7f53616c65206d7573742062652061637469766520746f206d696e74204d696c6160008201527f647953746174696f6e7300000000000000000000000000000000000000000000602082015250565b6000614e76602a83614370565b9150614e8182614e1a565b604082019050919050565b60006020820190508181036000830152614ea581614e69565b9050919050565b7f43616e206f6e6c79206d696e7420757020746f20333020746f6b656e7320617460008201527f20612074696d6500000000000000000000000000000000000000000000000000602082015250565b6000614f08602783614370565b9150614f1382614eac565b604082019050919050565b60006020820190508181036000830152614f3781614efb565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000614f7882614302565b9150614f8383614302565b9250828201905080821115614f9b57614f9a614f3e565b5b92915050565b7f507572636861736520776f756c6420657863656564206d617820737570706c7960008201527f206f66204d696c61647953746174696f6e730000000000000000000000000000602082015250565b6000614ffd603283614370565b915061500882614fa1565b604082019050919050565b6000602082019050818103600083015261502c81614ff0565b9050919050565b600061503e82614302565b915061504983614302565b925082820261505781614302565b9150828204841483151761506e5761506d614f3e565b5b5092915050565b7f45746865722076616c75652073656e74206973206e6f7420636f727265637400600082015250565b60006150ab601f83614370565b91506150b682615075565b602082019050919050565b600060208201905081810360008301526150da8161509e565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b600061511b82614302565b915061512683614302565b925082615136576151356150e1565b5b828206905092915050565b600061514c82614302565b915061515783614302565b925082615167576151666150e1565b5b828204905092915050565b600061517d82614302565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82036151af576151ae614f3e565b5b600182019050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b6000600282049050600182168061520157607f821691505b602082108103615214576152136151ba565b5b50919050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b60006008830261527c7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8261523f565b615286868361523f565b95508019841693508086168417925050509392505050565b6000819050919050565b60006152c36152be6152b984614302565b61529e565b614302565b9050919050565b6000819050919050565b6152dd836152a8565b6152f16152e9826152ca565b84845461524c565b825550505050565b600090565b6153066152f9565b6153118184846152d4565b505050565b5b818110156153355761532a6000826152fe565b600181019050615317565b5050565b601f82111561537a5761534b8161521a565b6153548461522f565b81016020851015615363578190505b61537761536f8561522f565b830182615316565b50505b505050565b600082821c905092915050565b600061539d6000198460080261537f565b1980831691505092915050565b60006153b6838361538c565b9150826002028217905092915050565b6153cf82614365565b67ffffffffffffffff8111156153e8576153e76144b5565b5b6153f282546151e9565b6153fd828285615339565b600060209050601f831160018114615430576000841561541e578287015190505b61542885826153aa565b865550615490565b601f19841661543e8661521a565b60005b8281101561546657848901518255600182019150602085019450602081019050615441565b86831015615483578489015161547f601f89168261538c565b8355505b6001600288020188555050505b505050505050565b7f4e69636520747279206275737465722c206d696c61647973206f6e6c79000000600082015250565b60006154ce601d83614370565b91506154d982615498565b602082019050919050565b600060208201905081810360008301526154fd816154c1565b9050919050565b6000815190506155138161430c565b92915050565b60006020828403121561552f5761552e61415b565b5b600061553d84828501615504565b91505092915050565b7f4e69636520747279206275737465722c206e6f74206f6e20746865206c697374600082015250565b600061557c602083614370565b915061558782615546565b602082019050919050565b600060208201905081810360008301526155ab8161556f565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f53616c65206d7573742062652061637469766520746f206d696e74204d696c6160008201527f6479730000000000000000000000000000000000000000000000000000000000602082015250565b600061563d602383614370565b9150615648826155e1565b604082019050919050565b6000602082019050818103600083015261566c81615630565b9050919050565b7f507572636861736520776f756c6420657863656564206d617820737570706c7960008201527f206f66204d696c61647973000000000000000000000000000000000000000000602082015250565b60006156cf602b83614370565b91506156da82615673565b604082019050919050565b600060208201905081810360008301526156fe816156c2565b9050919050565b7f776169742062757420796f7520646f6e277420686176652061206d696c61647960008201527f2e2e200000000000000000000000000000000000000000000000000000000000602082015250565b6000615761602383614370565b915061576c82615705565b604082019050919050565b6000602082019050818103600083015261579081615754565b9050919050565b7f7468652074696d6520666f72207468697320686173207061737365642e000000600082015250565b60006157cd601d83614370565b91506157d882615797565b602082019050919050565b600060208201905081810360008301526157fc816157c0565b9050919050565b600081905092915050565b600061581982614365565b6158238185615803565b9350615833818560208601614381565b80840191505092915050565b600061584b828561580e565b9150615857828461580e565b91508190509392505050565b7f455243323938313a20726f79616c7479206665652077696c6c2065786365656460008201527f2073616c65507269636500000000000000000000000000000000000000000000602082015250565b60006158bf602a83614370565b91506158ca82615863565b604082019050919050565b600060208201905081810360008301526158ee816158b2565b9050919050565b7f455243323938313a20696e76616c696420726563656976657200000000000000600082015250565b600061592b601983614370565b9150615936826158f5565b602082019050919050565b6000602082019050818103600083015261595a8161591e565b9050919050565b600081519050919050565b600082825260208201905092915050565b600061598882615961565b615992818561596c565b93506159a2818560208601614381565b6159ab816143ab565b840191505092915050565b60006080820190506159cb6000830187614417565b6159d86020830186614417565b6159e56040830185614481565b81810360608301526159f7818461597d565b905095945050505050565b600081519050615a1181614191565b92915050565b600060208284031215615a2d57615a2c61415b565b5b6000615a3b84828501615a02565b9150509291505056fea2646970667358221220ca2dfa77036efa5e87754a265606d8564a127fdd9a104bf47d8918401886ac0464736f6c63430008110033
Deployed Bytecode
0x6080604052600436106102ff5760003560e01c80635bbb217711610190578063b6d8e86f116100dc578063d7533f0211610095578063f2fde38b1161006f578063f2fde38b14610ae8578063fb796e6c14610b04578063fee81cf414610b2f578063ff35def614610b6c576102ff565b8063d7533f0214610a64578063e985e9c514610a8f578063f04e283e14610acc576102ff565b8063b6d8e86f14610953578063b7c0b8e81461097c578063b88d4fde146109a5578063c23dc68f146109c1578063c45fc170146109fe578063c87b56dd14610a27576102ff565b80638b83df451161014957806399a2557a1161012357806399a2557a146108a6578063a22cb465146108e3578063b1e283de1461090c578063b6ca7ea314610928576102ff565b80638b83df45146108255780638da5cb5b1461085057806395d89b411461087b576102ff565b80635bbb2177146106ea5780636352211e1461072757806370a0823114610764578063715018a6146107a15780637d5fcf9c146107ab5780638462151c146107e8576102ff565b806334918dfd1161024f578063448dbc9c116102085780634b01570a116101e25780634b01570a1461064f57806354d1f13d1461068c57806355f804b3146106965780635854d949146106bf576102ff565b8063448dbc9c146105f1578063451807b71461061c578063487bddf614610633576102ff565b806334918dfd146105045780633494f1bf1461051b5780633ccfd60b146105585780633eab94351461056f57806342842e0e146105ac57806342966c68146105c8576102ff565b80630a5ef3e3116102bc57806323b872dd1161029657806323b872dd1461048957806325692962146104a55780632a55205a146104af578063323ab474146104ed576102ff565b80630a5ef3e31461040a578063109695231461043557806318160ddd1461045e576102ff565b806301ffc9a71461030457806304634d8d1461034157806306a94cf61461036a57806306fdde0314610386578063081812fc146103b1578063095ea7b3146103ee575b600080fd5b34801561031057600080fd5b5061032b600480360381019061032691906141bd565b610b97565b6040516103389190614205565b60405180910390f35b34801561034d57600080fd5b50610368600480360381019061036391906142c2565b610bb9565b005b610384600480360381019061037f9190614338565b610bcf565b005b34801561039257600080fd5b5061039b610ee4565b6040516103a891906143f5565b60405180910390f35b3480156103bd57600080fd5b506103d860048036038101906103d39190614338565b610f76565b6040516103e59190614426565b60405180910390f35b61040860048036038101906104039190614441565b610ff5565b005b34801561041657600080fd5b5061041f61102a565b60405161042c9190614490565b60405180910390f35b34801561044157600080fd5b5061045c600480360381019061045791906145e0565b611030565b005b34801561046a57600080fd5b5061047361104b565b6040516104809190614490565b60405180910390f35b6104a3600480360381019061049e9190614629565b611062565b005b6104ad6110cd565b005b3480156104bb57600080fd5b506104d660048036038101906104d1919061467c565b611121565b6040516104e49291906146bc565b60405180910390f35b3480156104f957600080fd5b5061050261130b565b005b34801561051057600080fd5b50610519611424565b005b34801561052757600080fd5b50610542600480360381019061053d91906146e5565b611458565b60405161054f9190614490565b60405180910390f35b34801561056457600080fd5b5061056d61153e565b005b34801561057b57600080fd5b50610596600480360381019061059191906146e5565b611595565b6040516105a39190614205565b60405180910390f35b6105c660048036038101906105c19190614629565b6115b5565b005b3480156105d457600080fd5b506105ef60048036038101906105ea9190614338565b611620565b005b3480156105fd57600080fd5b5061060661162e565b6040516106139190614490565b60405180910390f35b34801561062857600080fd5b50610631611634565b005b61064d60048036038101906106489190614338565b611793565b005b34801561065b57600080fd5b50610676600480360381019061067191906146e5565b611aa8565b6040516106839190614490565b60405180910390f35b610694611d5b565b005b3480156106a257600080fd5b506106bd60048036038101906106b891906145e0565b611d97565b005b3480156106cb57600080fd5b506106d4611db2565b6040516106e191906143f5565b60405180910390f35b3480156106f657600080fd5b50610711600480360381019061070c9190614772565b611e40565b60405161071e9190614922565b60405180910390f35b34801561073357600080fd5b5061074e60048036038101906107499190614338565b611f03565b60405161075b9190614426565b60405180910390f35b34801561077057600080fd5b5061078b600480360381019061078691906146e5565b611f15565b6040516107989190614490565b60405180910390f35b6107a9611fcd565b005b3480156107b757600080fd5b506107d260048036038101906107cd91906146e5565b611fe1565b6040516107df9190614205565b60405180910390f35b3480156107f457600080fd5b5061080f600480360381019061080a91906146e5565b612001565b60405161081c9190614a02565b60405180910390f35b34801561083157600080fd5b5061083a612144565b60405161084791906143f5565b60405180910390f35b34801561085c57600080fd5b506108656121d2565b6040516108729190614426565b60405180910390f35b34801561088757600080fd5b506108906121e0565b60405161089d91906143f5565b60405180910390f35b3480156108b257600080fd5b506108cd60048036038101906108c89190614a24565b612272565b6040516108da9190614a02565b60405180910390f35b3480156108ef57600080fd5b5061090a60048036038101906109059190614aa3565b61247e565b005b61092660048036038101906109219190614338565b6124b3565b005b34801561093457600080fd5b5061093d612813565b60405161094a9190614490565b60405180910390f35b34801561095f57600080fd5b5061097a60048036038101906109759190614ba6565b612818565b005b34801561098857600080fd5b506109a3600480360381019061099e9190614bef565b6128bb565b005b6109bf60048036038101906109ba9190614cbd565b6128e0565b005b3480156109cd57600080fd5b506109e860048036038101906109e39190614338565b61294d565b6040516109f59190614d95565b60405180910390f35b348015610a0a57600080fd5b50610a256004803603810190610a2091906146e5565b6129b7565b005b348015610a3357600080fd5b50610a4e6004803603810190610a499190614338565b612a1e565b604051610a5b91906143f5565b60405180910390f35b348015610a7057600080fd5b50610a79612abc565b604051610a869190614dbf565b60405180910390f35b348015610a9b57600080fd5b50610ab66004803603810190610ab19190614dda565b612ac7565b604051610ac39190614205565b60405180910390f35b610ae66004803603810190610ae191906146e5565b612b5b565b005b610b026004803603810190610afd91906146e5565b612bc9565b005b348015610b1057600080fd5b50610b19612c43565b604051610b269190614205565b60405180910390f35b348015610b3b57600080fd5b50610b566004803603810190610b5191906146e5565b612c56565b604051610b639190614490565b60405180910390f35b348015610b7857600080fd5b50610b81612c71565b604051610b8e9190614205565b60405180910390f35b6000610ba282612c84565b80610bb25750610bb182612d16565b5b9050919050565b610bc1612d90565b610bcb8282612dad565b5050565b600c60009054906101000a900460ff16610c1e576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c1590614e8c565b60405180910390fd5b601e811115610c62576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610c5990614f1e565b60405180910390fd5b6104bc81610c6e61104b565b610c789190614f6d565b10610cb8576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610caf90615013565b60405180910390fd5b6000601e8203610d1f57661550f7dca700009050348282610cd99190615033565b1115610d1a576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d11906150c1565b60405180910390fd5b610e44565b600f8210610d84576618de76816d80009050348282610d3e9190615033565b1115610d7f576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610d76906150c1565b60405180910390fd5b610e43565b60058210610de957661c6bf5263400009050348282610da39190615033565b1115610de4576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610ddb906150c1565b60405180910390fd5b610e42565b661ff973cafa80009050348282610e009190615033565b1115610e41576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610e38906150c1565b60405180910390fd5b5b5b5b6000600883610e539190615110565b03610e935760005b600883610e689190615141565b811015610e8d57610e7a336008612f42565b8080610e8590615172565b915050610e5b565b50610ee0565b60005b600883610ea39190615141565b811015610ec857610eb5336008612f42565b8080610ec090615172565b915050610e96565b50610edf33600884610eda9190615110565b612f42565b5b5050565b606060028054610ef3906151e9565b80601f0160208091040260200160405190810160405280929190818152602001828054610f1f906151e9565b8015610f6c5780601f10610f4157610100808354040283529160200191610f6c565b820191906000526020600020905b815481529060010190602001808311610f4f57829003601f168201915b5050505050905090565b6000610f81826130fd565b610fb7576040517fcf4700e400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6006600083815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b81610fff8161315c565b61101b5761100b6131a8565b1561101a57611019816131bf565b5b5b6110258383613203565b505050565b6104bc81565b611038612d90565b80600a908161104791906153c6565b5050565b6000611055613213565b6001546000540303905090565b823373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff16146110bc5761109f3361315c565b6110bb576110ab6131a8565b156110ba576110b9336131bf565b5b5b5b6110c7848484613218565b50505050565b60006110d7612abc565b67ffffffffffffffff164201905063389a75e1600c5233600052806020600c2055337fdbf36a107da19e49527a7176a1babf963b4b0ff8cde35ee35d6cd8f1f9ac7e1d600080a250565b6000806000600960008681526020019081526020016000206040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff16815250509050600073ffffffffffffffffffffffffffffffffffffffff16816000015173ffffffffffffffffffffffffffffffffffffffff16036112b65760086040518060400160405290816000820160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020016000820160149054906101000a90046bffffffffffffffffffffffff166bffffffffffffffffffffffff166bffffffffffffffffffffffff168152505090505b60006112c061353a565b6bffffffffffffffffffffffff1682602001516bffffffffffffffffffffffff16866112ec9190615033565b6112f69190615141565b90508160000151819350935050509250929050565b600061131633611458565b90506000811180156113725750601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16155b156113df576001601160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055506113da336001613544565b611421565b6000611420576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401611417906154e4565b60405180910390fd5b5b50565b61142c612d90565b600c60009054906101000a900460ff1615600c60006101000a81548160ff021916908315150217905550565b60008060009050600c60029054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b81526004016114ba9190614426565b602060405180830381865afa9250505080156114f457506040513d601f19601f820116820180604052508101906114f19190615519565b60015b611530573d8060008114611524576040519150601f19603f3d011682016040523d82523d6000602084013e611529565b606091505b5050611535565b809150505b80915050919050565b611546612d90565b60004790503373ffffffffffffffffffffffffffffffffffffffff166108fc829081150290604051600060405180830381858888f19350505050158015611591573d6000803e3d6000fd5b5050565b60116020528060005260406000206000915054906101000a900460ff1681565b823373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161461160f576115f23361315c565b61160e576115fe6131a8565b1561160d5761160c336131bf565b5b5b5b61161a848484613562565b50505050565b61162b816001613582565b50565b61016a81565b61163c6121d2565b73ffffffffffffffffffffffffffffffffffffffff163373ffffffffffffffffffffffffffffffffffffffff16036116945761167933600a613544565b61168433600a613544565b61168f33600a613544565b611791565b601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff161561174e576000601060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff021916908315150217905550611749336002613544565b611790565b600061178f576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161178690615592565b60405180910390fd5b5b5b565b600c60009054906101000a900460ff166117e2576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016117d990614e8c565b60405180910390fd5b601e811115611826576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161181d90614f1e565b60405180910390fd5b6104bc8161183261104b565b61183c9190614f6d565b1061187c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161187390615013565b60405180910390fd5b6000601e82036118e357661ff973cafa8000905034828261189d9190615033565b11156118de576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016118d5906150c1565b60405180910390fd5b611a08565b600f821061194857662386f26fc1000090503482826119029190615033565b1115611943576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161193a906150c1565b60405180910390fd5b611a07565b600582106119ad57662714711487800090503482826119679190615033565b11156119a8576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161199f906150c1565b60405180910390fd5b611a06565b662aa1efb94e000090503482826119c49190615033565b1115611a05576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016119fc906150c1565b60405180910390fd5b5b5b5b6000600883611a179190615110565b03611a575760005b600883611a2c9190615141565b811015611a5157611a3e336008612f42565b8080611a4990615172565b915050611a1f565b50611aa4565b60005b600883611a679190615141565b811015611a8c57611a79336008612f42565b8080611a8490615172565b915050611a5a565b50611aa333600884611a9e9190615110565b612f42565b5b5050565b60008060009050600d60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b8152600401611b0a9190614426565b602060405180830381865afa925050508015611b4457506040513d601f19601f82011682018060405250810190611b419190615519565b60015b611b80573d8060008114611b74576040519150601f19603f3d011682016040523d82523d6000602084013e611b79565b606091505b5050611b90565b8082611b8c9190614f6d565b9150505b600e60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b8152600401611beb9190614426565b602060405180830381865afa925050508015611c2557506040513d601f19601f82011682018060405250810190611c229190615519565b60015b611c61573d8060008114611c55576040519150601f19603f3d011682016040523d82523d6000602084013e611c5a565b606091505b5050611c71565b8082611c6d9190614f6d565b9150505b600f60009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166370a08231846040518263ffffffff1660e01b8152600401611ccc9190614426565b602060405180830381865afa925050508015611d0657506040513d601f19601f82011682018060405250810190611d039190615519565b60015b611d42573d8060008114611d36576040519150601f19603f3d011682016040523d82523d6000602084013e611d3b565b606091505b5050611d52565b8082611d4e9190614f6d565b9150505b80915050919050565b63389a75e1600c523360005260006020600c2055337ffa7b8eab7da67f412cc9575ed43464468f9bfbae89d1675917346ca6d8fe3c92600080a2565b611d9f612d90565b80600b9081611dae91906153c6565b5050565b600a8054611dbf906151e9565b80601f0160208091040260200160405190810160405280929190818152602001828054611deb906151e9565b8015611e385780601f10611e0d57610100808354040283529160200191611e38565b820191906000526020600020905b815481529060010190602001808311611e1b57829003601f168201915b505050505081565b6060600083839050905060008167ffffffffffffffff811115611e6657611e656144b5565b5b604051908082528060200260200182016040528015611e9f57816020015b611e8c614102565b815260200190600190039081611e845790505b50905060005b828114611ef757611ece868683818110611ec257611ec16155b2565b5b9050602002013561294d565b828281518110611ee157611ee06155b2565b5b6020026020010181905250806001019050611ea5565b50809250505092915050565b6000611f0e826137d4565b9050919050565b60008073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603611f7c576040517f8f4eb60400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b67ffffffffffffffff600560008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054169050919050565b611fd5612d90565b611fdf60006138cc565b565b60106020528060005260406000206000915054906101000a900460ff1681565b6060600080600061201185611f15565b905060008167ffffffffffffffff81111561202f5761202e6144b5565b5b60405190808252806020026020018201604052801561205d5781602001602082028036833780820191505090505b509050612068614102565b6000612072613213565b90505b838614612136576120858161390a565b9150816040015161212b57600073ffffffffffffffffffffffffffffffffffffffff16826000015173ffffffffffffffffffffffffffffffffffffffff16146120d057816000015194505b8773ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff160361212a578083878060010198508151811061211d5761211c6155b2565b5b6020026020010181815250505b5b806001019050612075565b508195505050505050919050565b600b8054612151906151e9565b80601f016020809104026020016040519081016040528092919081815260200182805461217d906151e9565b80156121ca5780601f1061219f576101008083540402835291602001916121ca565b820191906000526020600020905b8154815290600101906020018083116121ad57829003601f168201915b505050505081565b6000638b78c6d81954905090565b6060600380546121ef906151e9565b80601f016020809104026020016040519081016040528092919081815260200182805461221b906151e9565b80156122685780601f1061223d57610100808354040283529160200191612268565b820191906000526020600020905b81548152906001019060200180831161224b57829003601f168201915b5050505050905090565b60608183106122ad576040517f32c1995a00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000806122b8613935565b90506122c2613213565b8510156122d4576122d1613213565b94505b808411156122e0578093505b60006122eb87611f15565b90508486101561230e576000868603905081811015612308578091505b50612313565b600090505b60008167ffffffffffffffff81111561232f5761232e6144b5565b5b60405190808252806020026020018201604052801561235d5781602001602082028036833780820191505090505b509050600082036123745780945050505050612477565b600061237f8861294d565b90506000816040015161239457816000015190505b60008990505b8881141580156123aa5750848714155b15612469576123b88161390a565b9250826040015161245e57600073ffffffffffffffffffffffffffffffffffffffff16836000015173ffffffffffffffffffffffffffffffffffffffff161461240357826000015191505b8a73ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff160361245d57808488806001019950815181106124505761244f6155b2565b5b6020026020010181815250505b5b80600101905061239a565b508583528296505050505050505b9392505050565b816124888161315c565b6124a4576124946131a8565b156124a3576124a2816131bf565b5b5b6124ae838361393e565b505050565b600c60009054906101000a900460ff16612502576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016124f990615653565b60405180910390fd5b601e811115612546576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161253d90614f1e565b60405180910390fd5b6104bc8161255261104b565b61255c9190614f6d565b1061259c576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612593906156e5565b60405180910390fd5b60006125a733611458565b116125e7576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016125de90615777565b60405180910390fd5b6000601e820361264e5766071afd498d000090503482826126089190615033565b1115612649576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612640906150c1565b60405180910390fd5b612773565b600f82106126b357660aa87bee538000905034828261266d9190615033565b11156126ae576040517f08c379a00000000000000000000000000000000000000000000000000000000081526004016126a5906150c1565b60405180910390fd5b612772565b6005821061271857660e35fa931a000090503482826126d29190615033565b1115612713576040517f08c379a000000000000000000000000000000000000000000000000000000000815260040161270a906150c1565b60405180910390fd5b612771565b6611c37937e08000905034828261272f9190615033565b1115612770576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612767906150c1565b60405180910390fd5b5b5b5b60006008836127829190615110565b036127c25760005b6008836127979190615141565b8110156127bc576127a9336008612f42565b80806127b490615172565b91505061278a565b5061280f565b60005b6008836127d29190615141565b8110156127f7576127e4336008612f42565b80806127ef90615172565b9150506127c5565b5061280e336008846128099190615110565b612f42565b5b5050565b601e81565b612820612d90565b60005b81518110156128b7576000828281518110612841576128406155b2565b5b602002602001015190506001601060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055505080806128af90615172565b915050612823565b5050565b6128c3612d90565b80600c60016101000a81548160ff02191690831515021790555050565b833373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161461293a5761291d3361315c565b612939576129296131a8565b1561293857612937336131bf565b5b5b5b61294685858585613a49565b5050505050565b612955614102565b61295d614102565b612965613213565b8310806129795750612975613935565b8310155b1561298757809150506129b2565b6129908361390a565b90508060400151156129a557809150506129b2565b6129ae83613abc565b9150505b919050565b6129bf612d90565b60006129c961104b565b14612a09576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612a00906157e3565b60405180910390fd5b600061016a9050612a1a8282612f42565b5050565b6060612a29826130fd565b612a5f576040517fa14c4b5000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b6000612a69613adc565b90506000815103612a895760405180602001604052806000815250612ab4565b80612a9384613b6e565b604051602001612aa492919061583f565b6040516020818303038152906040525b915050919050565b60006202a300905090565b6000600760008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16905092915050565b612b63612d90565b63389a75e1600c52806000526020600c208054421115612b8b57636f5e88186000526004601cfd5b60008155600c5160601c80337f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a380638b78c6d81955505050565b612bd1612d90565b600073ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff1603612c37576040517f7448fbae00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612c40816138cc565b50565b600c60019054906101000a900460ff1681565b600063389a75e1600c52816000526020600c20549050919050565b600c60009054906101000a900460ff1681565b60006301ffc9a760e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480612cdf57506380ac58cd60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b80612d0f5750635b5e139f60e01b827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916145b9050919050565b60007f2a55205a000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff19161480612d895750612d8882613bbe565b5b9050919050565b638b78c6d819543314612dab576382b429006000526004601cfd5b565b612db561353a565b6bffffffffffffffffffffffff16816bffffffffffffffffffffffff161115612e13576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612e0a906158d5565b60405180910390fd5b600073ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff1603612e82576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401612e7990615941565b60405180910390fd5b60405180604001604052808373ffffffffffffffffffffffffffffffffffffffff168152602001826bffffffffffffffffffffffff16815250600860008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a8154816bffffffffffffffffffffffff02191690836bffffffffffffffffffffffff1602179055509050505050565b60008054905060008203612f82576040517fb562e8dd00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b612f8f6000848385613c28565b600160406001901b178202600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000828254019250508190555061300683612ff76000866000613c2e565b61300085613c56565b17613c66565b6004600083815260200190815260200160002081905550600080838301905073ffffffffffffffffffffffffffffffffffffffff85169150828260007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a4600183015b8181146130a757808360007fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef600080a460018101905061306c565b50600082036130e2576040517f2e07630000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b8060008190555050506130f86000848385613c91565b505050565b600081613108613213565b11158015613117575060005482105b8015613155575060007c0100000000000000000000000000000000000000000000000000000000600460008581526020019081526020016000205416145b9050919050565b6000731e0049783f008a0085193e00003d00cd54003c7173ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff16149050919050565b6000600c60019054906101000a900460ff16905090565b69c617113400112233445560005230601a5280603a52600080604460166daaeb6d7670e522a718067333cd4e5afa6131fb573d6000803e3d6000fd5b6000603a5250565b61320f82826001613c97565b5050565b600090565b6000613223826137d4565b90508373ffffffffffffffffffffffffffffffffffffffff168173ffffffffffffffffffffffffffffffffffffffff161461328a576040517fa114810000000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b60008061329684613de3565b915091506132ac81876132a7613e0a565b613e12565b6132f8576132c1866132bc613e0a565b612ac7565b6132f7576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b600073ffffffffffffffffffffffffffffffffffffffff168573ffffffffffffffffffffffffffffffffffffffff160361335e576040517fea553b3400000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b61336b8686866001613c28565b801561337657600082555b600560008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001900391905081905550600560008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600081546001019190508190555061344485613420888887613c2e565b7c020000000000000000000000000000000000000000000000000000000017613c66565b600460008681526020019081526020016000208190555060007c02000000000000000000000000000000000000000000000000000000008416036134ca57600060018501905060006004600083815260200190815260200160002054036134c85760005481146134c7578360046000838152602001908152602001600020819055505b5b505b838573ffffffffffffffffffffffffffffffffffffffff168773ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a46135328686866001613c91565b505050505050565b6000612710905090565b61355e828260405180602001604052806000815250613e56565b5050565b61357d838383604051806020016040528060008152506128e0565b505050565b600061358d836137d4565b905060008190506000806135a086613de3565b915091508415613609576135bc81846135b7613e0a565b613e12565b613608576135d1836135cc613e0a565b612ac7565b613607576040517f59c896be00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b5b613617836000886001613c28565b801561362257600082555b600160806001901b03600560008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600082825401925050819055506136ca8361368785600088613c2e565b7c02000000000000000000000000000000000000000000000000000000007c01000000000000000000000000000000000000000000000000000000001717613c66565b600460008881526020019081526020016000208190555060007c0200000000000000000000000000000000000000000000000000000000851603613750576000600187019050600060046000838152602001908152602001600020540361374e57600054811461374d578460046000838152602001908152602001600020819055505b5b505b85600073ffffffffffffffffffffffffffffffffffffffff168473ffffffffffffffffffffffffffffffffffffffff167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef60405160405180910390a46137ba836000886001613c91565b600160008154809291906001019190505550505050505050565b6000816137df613213565b11613895576004600083815260200190815260200160002054905060007c0100000000000000000000000000000000000000000000000000000000821603613894576000810361388f576000548210613864576040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b6004600083600190039350838152602001908152602001600020549050600081036138c757613865565b6138c7565b5b6040517fdf2d9b4200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b919050565b638b78c6d8198160601b60601c91508181547f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a38181555050565b613912614102565b61392e6004600084815260200190815260200160002054613ef3565b9050919050565b60008054905090565b806007600061394b613e0a565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055508173ffffffffffffffffffffffffffffffffffffffff166139f8613e0a565b73ffffffffffffffffffffffffffffffffffffffff167f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c3183604051613a3d9190614205565b60405180910390a35050565b613a54848484611062565b60008373ffffffffffffffffffffffffffffffffffffffff163b14613ab657613a7f84848484613fa9565b613ab5576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b50505050565b613ac4614102565b613ad5613ad0836137d4565b613ef3565b9050919050565b6060600b8054613aeb906151e9565b80601f0160208091040260200160405190810160405280929190818152602001828054613b17906151e9565b8015613b645780601f10613b3957610100808354040283529160200191613b64565b820191906000526020600020905b815481529060010190602001808311613b4757829003601f168201915b5050505050905090565b606060a060405101806040526020810391506000825281835b600115613ba957600184039350600a81066030018453600a8104905080613b87575b50828103602084039350808452505050919050565b60007f01ffc9a7000000000000000000000000000000000000000000000000000000007bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916827bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916149050919050565b50505050565b60008060e883901c905060e8613c458686846140f9565b62ffffff16901b9150509392505050565b60006001821460e11b9050919050565b600073ffffffffffffffffffffffffffffffffffffffff83169250814260a01b178317905092915050565b50505050565b6000613ca283611f03565b90508115613d2d578073ffffffffffffffffffffffffffffffffffffffff16613cc9613e0a565b73ffffffffffffffffffffffffffffffffffffffff1614613d2c57613cf581613cf0613e0a565b612ac7565b613d2b576040517fcfb3b94200000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b5b5b836006600085815260200190815260200160002060000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550828473ffffffffffffffffffffffffffffffffffffffff168273ffffffffffffffffffffffffffffffffffffffff167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a450505050565b60008060006006600085815260200190815260200160002090508092508254915050915091565b600033905090565b600073ffffffffffffffffffffffffffffffffffffffff8316925073ffffffffffffffffffffffffffffffffffffffff821691508382148383141790509392505050565b613e608383612f42565b60008373ffffffffffffffffffffffffffffffffffffffff163b14613eee57600080549050600083820390505b613ea06000868380600101945086613fa9565b613ed6576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b818110613e8d578160005414613eeb57600080fd5b50505b505050565b613efb614102565b81816000019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060a082901c816020019067ffffffffffffffff16908167ffffffffffffffff168152505060007c01000000000000000000000000000000000000000000000000000000008316141581604001901515908115158152505060e882901c816060019062ffffff16908162ffffff1681525050919050565b60008373ffffffffffffffffffffffffffffffffffffffff1663150b7a02613fcf613e0a565b8786866040518563ffffffff1660e01b8152600401613ff194939291906159b6565b6020604051808303816000875af192505050801561402d57506040513d601f19601f8201168201806040525081019061402a9190615a17565b60015b6140a6573d806000811461405d576040519150601f19603f3d011682016040523d82523d6000602084013e614062565b606091505b50600081510361409e576040517fd1a57ed600000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b805181602001fd5b63150b7a0260e01b7bffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916817bffffffffffffffffffffffffffffffffffffffffffffffffffffffff191614915050949350505050565b60009392505050565b6040518060800160405280600073ffffffffffffffffffffffffffffffffffffffff168152602001600067ffffffffffffffff168152602001600015158152602001600062ffffff1681525090565b6000604051905090565b600080fd5b600080fd5b60007fffffffff0000000000000000000000000000000000000000000000000000000082169050919050565b61419a81614165565b81146141a557600080fd5b50565b6000813590506141b781614191565b92915050565b6000602082840312156141d3576141d261415b565b5b60006141e1848285016141a8565b91505092915050565b60008115159050919050565b6141ff816141ea565b82525050565b600060208201905061421a60008301846141f6565b92915050565b600073ffffffffffffffffffffffffffffffffffffffff82169050919050565b600061424b82614220565b9050919050565b61425b81614240565b811461426657600080fd5b50565b60008135905061427881614252565b92915050565b60006bffffffffffffffffffffffff82169050919050565b61429f8161427e565b81146142aa57600080fd5b50565b6000813590506142bc81614296565b92915050565b600080604083850312156142d9576142d861415b565b5b60006142e785828601614269565b92505060206142f8858286016142ad565b9150509250929050565b6000819050919050565b61431581614302565b811461432057600080fd5b50565b6000813590506143328161430c565b92915050565b60006020828403121561434e5761434d61415b565b5b600061435c84828501614323565b91505092915050565b600081519050919050565b600082825260208201905092915050565b60005b8381101561439f578082015181840152602081019050614384565b60008484015250505050565b6000601f19601f8301169050919050565b60006143c782614365565b6143d18185614370565b93506143e1818560208601614381565b6143ea816143ab565b840191505092915050565b6000602082019050818103600083015261440f81846143bc565b905092915050565b61442081614240565b82525050565b600060208201905061443b6000830184614417565b92915050565b600080604083850312156144585761445761415b565b5b600061446685828601614269565b925050602061447785828601614323565b9150509250929050565b61448a81614302565b82525050565b60006020820190506144a56000830184614481565b92915050565b600080fd5b600080fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6144ed826143ab565b810181811067ffffffffffffffff8211171561450c5761450b6144b5565b5b80604052505050565b600061451f614151565b905061452b82826144e4565b919050565b600067ffffffffffffffff82111561454b5761454a6144b5565b5b614554826143ab565b9050602081019050919050565b82818337600083830152505050565b600061458361457e84614530565b614515565b90508281526020810184848401111561459f5761459e6144b0565b5b6145aa848285614561565b509392505050565b600082601f8301126145c7576145c66144ab565b5b81356145d7848260208601614570565b91505092915050565b6000602082840312156145f6576145f561415b565b5b600082013567ffffffffffffffff81111561461457614613614160565b5b614620848285016145b2565b91505092915050565b6000806000606084860312156146425761464161415b565b5b600061465086828701614269565b935050602061466186828701614269565b925050604061467286828701614323565b9150509250925092565b600080604083850312156146935761469261415b565b5b60006146a185828601614323565b92505060206146b285828601614323565b9150509250929050565b60006040820190506146d16000830185614417565b6146de6020830184614481565b9392505050565b6000602082840312156146fb576146fa61415b565b5b600061470984828501614269565b91505092915050565b600080fd5b600080fd5b60008083601f840112614732576147316144ab565b5b8235905067ffffffffffffffff81111561474f5761474e614712565b5b60208301915083602082028301111561476b5761476a614717565b5b9250929050565b600080602083850312156147895761478861415b565b5b600083013567ffffffffffffffff8111156147a7576147a6614160565b5b6147b38582860161471c565b92509250509250929050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b6147f481614240565b82525050565b600067ffffffffffffffff82169050919050565b614817816147fa565b82525050565b614826816141ea565b82525050565b600062ffffff82169050919050565b6148448161482c565b82525050565b60808201600082015161486060008501826147eb565b506020820151614873602085018261480e565b506040820151614886604085018261481d565b506060820151614899606085018261483b565b50505050565b60006148ab838361484a565b60808301905092915050565b6000602082019050919050565b60006148cf826147bf565b6148d981856147ca565b93506148e4836147db565b8060005b838110156149155781516148fc888261489f565b9750614907836148b7565b9250506001810190506148e8565b5085935050505092915050565b6000602082019050818103600083015261493c81846148c4565b905092915050565b600081519050919050565b600082825260208201905092915050565b6000819050602082019050919050565b61497981614302565b82525050565b600061498b8383614970565b60208301905092915050565b6000602082019050919050565b60006149af82614944565b6149b9818561494f565b93506149c483614960565b8060005b838110156149f55781516149dc888261497f565b97506149e783614997565b9250506001810190506149c8565b5085935050505092915050565b60006020820190508181036000830152614a1c81846149a4565b905092915050565b600080600060608486031215614a3d57614a3c61415b565b5b6000614a4b86828701614269565b9350506020614a5c86828701614323565b9250506040614a6d86828701614323565b9150509250925092565b614a80816141ea565b8114614a8b57600080fd5b50565b600081359050614a9d81614a77565b92915050565b60008060408385031215614aba57614ab961415b565b5b6000614ac885828601614269565b9250506020614ad985828601614a8e565b9150509250929050565b600067ffffffffffffffff821115614afe57614afd6144b5565b5b602082029050602081019050919050565b6000614b22614b1d84614ae3565b614515565b90508083825260208201905060208402830185811115614b4557614b44614717565b5b835b81811015614b6e5780614b5a8882614269565b845260208401935050602081019050614b47565b5050509392505050565b600082601f830112614b8d57614b8c6144ab565b5b8135614b9d848260208601614b0f565b91505092915050565b600060208284031215614bbc57614bbb61415b565b5b600082013567ffffffffffffffff811115614bda57614bd9614160565b5b614be684828501614b78565b91505092915050565b600060208284031215614c0557614c0461415b565b5b6000614c1384828501614a8e565b91505092915050565b600067ffffffffffffffff821115614c3757614c366144b5565b5b614c40826143ab565b9050602081019050919050565b6000614c60614c5b84614c1c565b614515565b905082815260208101848484011115614c7c57614c7b6144b0565b5b614c87848285614561565b509392505050565b600082601f830112614ca457614ca36144ab565b5b8135614cb4848260208601614c4d565b91505092915050565b60008060008060808587031215614cd757614cd661415b565b5b6000614ce587828801614269565b9450506020614cf687828801614269565b9350506040614d0787828801614323565b925050606085013567ffffffffffffffff811115614d2857614d27614160565b5b614d3487828801614c8f565b91505092959194509250565b608082016000820151614d5660008501826147eb565b506020820151614d69602085018261480e565b506040820151614d7c604085018261481d565b506060820151614d8f606085018261483b565b50505050565b6000608082019050614daa6000830184614d40565b92915050565b614db9816147fa565b82525050565b6000602082019050614dd46000830184614db0565b92915050565b60008060408385031215614df157614df061415b565b5b6000614dff85828601614269565b9250506020614e1085828601614269565b9150509250929050565b7f53616c65206d7573742062652061637469766520746f206d696e74204d696c6160008201527f647953746174696f6e7300000000000000000000000000000000000000000000602082015250565b6000614e76602a83614370565b9150614e8182614e1a565b604082019050919050565b60006020820190508181036000830152614ea581614e69565b9050919050565b7f43616e206f6e6c79206d696e7420757020746f20333020746f6b656e7320617460008201527f20612074696d6500000000000000000000000000000000000000000000000000602082015250565b6000614f08602783614370565b9150614f1382614eac565b604082019050919050565b60006020820190508181036000830152614f3781614efb565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b6000614f7882614302565b9150614f8383614302565b9250828201905080821115614f9b57614f9a614f3e565b5b92915050565b7f507572636861736520776f756c6420657863656564206d617820737570706c7960008201527f206f66204d696c61647953746174696f6e730000000000000000000000000000602082015250565b6000614ffd603283614370565b915061500882614fa1565b604082019050919050565b6000602082019050818103600083015261502c81614ff0565b9050919050565b600061503e82614302565b915061504983614302565b925082820261505781614302565b9150828204841483151761506e5761506d614f3e565b5b5092915050565b7f45746865722076616c75652073656e74206973206e6f7420636f727265637400600082015250565b60006150ab601f83614370565b91506150b682615075565b602082019050919050565b600060208201905081810360008301526150da8161509e565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b600061511b82614302565b915061512683614302565b925082615136576151356150e1565b5b828206905092915050565b600061514c82614302565b915061515783614302565b925082615167576151666150e1565b5b828204905092915050565b600061517d82614302565b91507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff82036151af576151ae614f3e565b5b600182019050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602260045260246000fd5b6000600282049050600182168061520157607f821691505b602082108103615214576152136151ba565b5b50919050565b60008190508160005260206000209050919050565b60006020601f8301049050919050565b600082821b905092915050565b60006008830261527c7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8261523f565b615286868361523f565b95508019841693508086168417925050509392505050565b6000819050919050565b60006152c36152be6152b984614302565b61529e565b614302565b9050919050565b6000819050919050565b6152dd836152a8565b6152f16152e9826152ca565b84845461524c565b825550505050565b600090565b6153066152f9565b6153118184846152d4565b505050565b5b818110156153355761532a6000826152fe565b600181019050615317565b5050565b601f82111561537a5761534b8161521a565b6153548461522f565b81016020851015615363578190505b61537761536f8561522f565b830182615316565b50505b505050565b600082821c905092915050565b600061539d6000198460080261537f565b1980831691505092915050565b60006153b6838361538c565b9150826002028217905092915050565b6153cf82614365565b67ffffffffffffffff8111156153e8576153e76144b5565b5b6153f282546151e9565b6153fd828285615339565b600060209050601f831160018114615430576000841561541e578287015190505b61542885826153aa565b865550615490565b601f19841661543e8661521a565b60005b8281101561546657848901518255600182019150602085019450602081019050615441565b86831015615483578489015161547f601f89168261538c565b8355505b6001600288020188555050505b505050505050565b7f4e69636520747279206275737465722c206d696c61647973206f6e6c79000000600082015250565b60006154ce601d83614370565b91506154d982615498565b602082019050919050565b600060208201905081810360008301526154fd816154c1565b9050919050565b6000815190506155138161430c565b92915050565b60006020828403121561552f5761552e61415b565b5b600061553d84828501615504565b91505092915050565b7f4e69636520747279206275737465722c206e6f74206f6e20746865206c697374600082015250565b600061557c602083614370565b915061558782615546565b602082019050919050565b600060208201905081810360008301526155ab8161556f565b9050919050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f53616c65206d7573742062652061637469766520746f206d696e74204d696c6160008201527f6479730000000000000000000000000000000000000000000000000000000000602082015250565b600061563d602383614370565b9150615648826155e1565b604082019050919050565b6000602082019050818103600083015261566c81615630565b9050919050565b7f507572636861736520776f756c6420657863656564206d617820737570706c7960008201527f206f66204d696c61647973000000000000000000000000000000000000000000602082015250565b60006156cf602b83614370565b91506156da82615673565b604082019050919050565b600060208201905081810360008301526156fe816156c2565b9050919050565b7f776169742062757420796f7520646f6e277420686176652061206d696c61647960008201527f2e2e200000000000000000000000000000000000000000000000000000000000602082015250565b6000615761602383614370565b915061576c82615705565b604082019050919050565b6000602082019050818103600083015261579081615754565b9050919050565b7f7468652074696d6520666f72207468697320686173207061737365642e000000600082015250565b60006157cd601d83614370565b91506157d882615797565b602082019050919050565b600060208201905081810360008301526157fc816157c0565b9050919050565b600081905092915050565b600061581982614365565b6158238185615803565b9350615833818560208601614381565b80840191505092915050565b600061584b828561580e565b9150615857828461580e565b91508190509392505050565b7f455243323938313a20726f79616c7479206665652077696c6c2065786365656460008201527f2073616c65507269636500000000000000000000000000000000000000000000602082015250565b60006158bf602a83614370565b91506158ca82615863565b604082019050919050565b600060208201905081810360008301526158ee816158b2565b9050919050565b7f455243323938313a20696e76616c696420726563656976657200000000000000600082015250565b600061592b601983614370565b9150615936826158f5565b602082019050919050565b6000602082019050818103600083015261595a8161591e565b9050919050565b600081519050919050565b600082825260208201905092915050565b600061598882615961565b615992818561596c565b93506159a2818560208601614381565b6159ab816143ab565b840191505092915050565b60006080820190506159cb6000830187614417565b6159d86020830186614417565b6159e56040830185614481565b81810360608301526159f7818461597d565b905095945050505050565b600081519050615a1181614191565b92915050565b600060208284031215615a2d57615a2c61415b565b5b6000615a3b84828501615a02565b9150509291505056fea2646970667358221220ca2dfa77036efa5e87754a265606d8564a127fdd9a104bf47d8918401886ac0464736f6c63430008110033
Deployed Bytecode Sourcemap
152798:12462:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;155791:472;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;156271:144;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;161767:1896;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;39387:100;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;45787:218;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;154796:226;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;152972:50;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;154093:134;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;35138:323;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;155030:232;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;12986:621;;;:::i;:::-;;109978:442;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;;:::i;:::-;;;;;;;;159089:334;;;;;;;;;;;;;:::i;:::-;;159882:93;;;;;;;;;;;;;:::i;:::-;;157248:384;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;157079:140;;;;;;;;;;;;;:::i;:::-;;153700:44;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;155270:240;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;79678:94;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;153029:47;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;159431:439;;;;;;;;;;;;;:::i;:::-;;163671:1586;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;157664:893;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;13692:466;;;:::i;:::-;;154235:96;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;153085:43;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;74391:528;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;40780:152;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;36322:233;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;12718:102;;;:::i;:::-;;153645:48;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;78267:900;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;153135:26;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;15701:196;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;39563:104;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;75307:2513;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;154560:228;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;159983:1776;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;152920:45;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;158592:241;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;156423:117;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;155518:265;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;73804:428;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;158841:240;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;39773:318;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;16546:109;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;46736:164;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;14349:1008;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;12465:185;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;153211:36;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;16003:449;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;153170:34;;;;;;;;;;;;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;155791:472;155950:4;156175:38;156201:11;156175:25;:38::i;:::-;:80;;;;156217:38;156243:11;156217:25;:38::i;:::-;156175:80;156168:87;;155791:472;;;:::o;156271:144::-;17052:13;:11;:13::i;:::-;156365:42:::1;156384:8;156394:12;156365:18;:42::i;:::-;156271:144:::0;;:::o;161767:1896::-;161846:14;;;;;;;;;;;161838:69;;;;;;;;;;;;:::i;:::-;;;;;;;;;161944:2;161926:14;:20;;161918:72;;;;;;;;;;;;:::i;:::-;;;;;;;;;162042:4;162025:14;162009:13;:11;:13::i;:::-;:30;;;;:::i;:::-;:37;162001:100;;;;;;;;;;;;:::i;:::-;;;;;;;;;162310:19;162487:2;162469:14;:20;162465:796;;162520:16;162506:30;;162604:9;162585:14;162572:11;:28;;;;:::i;:::-;:41;;162564:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;162465:796;;;162689:2;162671:14;:20;162667:594;;162722:16;162708:30;;162806:9;162787:14;162774:11;:28;;;;:::i;:::-;:41;;162766:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;162667:594;;;162891:1;162873:14;:19;162869:392;;162923:16;162909:30;;163007:9;162988:14;162975:11;:28;;;;:::i;:::-;:41;;162967:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;162869:392;;;163099:16;163085:30;;163204:9;163185:14;163172:11;:28;;;;:::i;:::-;:41;;163164:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;162869:392;162667:594;162465:796;163353:1;163348;163331:14;:18;;;;:::i;:::-;:23;163328:328;;163374:6;163370:97;163405:1;163390:14;:16;;;;:::i;:::-;163386:1;:20;163370:97;;;163432:19;163438:10;163449:1;163432:5;:19::i;:::-;163408:3;;;;;:::i;:::-;;;;163370:97;;;;163328:328;;;163503:6;163499:97;163534:1;163519:14;:16;;;;:::i;:::-;163515:1;:20;163499:97;;;163561:19;163567:10;163578:1;163561:5;:19::i;:::-;163537:3;;;;;:::i;:::-;;;;163499:97;;;;163610:34;163616:10;163642:1;163627:14;:16;;;;:::i;:::-;163610:5;:34::i;:::-;163328:328;161827:1836;161767:1896;:::o;39387:100::-;39441:13;39474:5;39467:12;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;39387:100;:::o;45787:218::-;45863:7;45888:16;45896:7;45888;:16::i;:::-;45883:64;;45913:34;;;;;;;;;;;;;;45883:64;45967:15;:24;45983:7;45967:24;;;;;;;;;;;:30;;;;;;;;;;;;45960:37;;45787:218;;;:::o;154796:226::-;154956:8;3590:29;3610:8;3590:19;:29::i;:::-;3585:122;;3640:27;:25;:27::i;:::-;3636:59;;;3669:26;3686:8;3669:16;:26::i;:::-;3636:59;3585:122;154982:32:::1;154996:8;155006:7;154982:13;:32::i;:::-;154796:226:::0;;;:::o;152972:50::-;153018:4;152972:50;:::o;154093:134::-;17052:13;:11;:13::i;:::-;154205:14:::1;154178:24;:41;;;;;;:::i;:::-;;154093:134:::0;:::o;35138:323::-;35199:7;35427:15;:13;:15::i;:::-;35412:12;;35396:13;;:28;:46;35389:53;;35138:323;:::o;155030:232::-;155195:4;3233:10;3225:18;;:4;:18;;;3221:184;;3265:31;3285:10;3265:19;:31::i;:::-;3260:134;;3321:27;:25;:27::i;:::-;3317:61;;;3350:28;3367:10;3350:16;:28::i;:::-;3317:61;3260:134;3221:184;155217:37:::1;155236:4;155242:2;155246:7;155217:18;:37::i;:::-;155030:232:::0;;;;:::o;12986:621::-;13081:15;13117:27;:25;:27::i;:::-;13099:45;;:15;:45;13081:63;;13308:19;13302:4;13295:33;13359:8;13353:4;13346:22;13416:7;13409:4;13403;13393:21;13386:38;13565:8;13518:45;13515:1;13512;13507:67;13216:373;12986:621::o;109978:442::-;110075:7;110084;110104:26;110133:17;:27;110151:8;110133:27;;;;;;;;;;;110104:56;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;110205:1;110177:30;;:7;:16;;;:30;;;110173:92;;110234:19;110224:29;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;110173:92;110277:21;110342:17;:15;:17::i;:::-;110301:58;;110315:7;:23;;;110302:36;;:10;:36;;;;:::i;:::-;110301:58;;;;:::i;:::-;110277:82;;110380:7;:16;;;110398:13;110372:40;;;;;;109978:442;;;;;:::o;159089:334::-;159137:15;159155:29;159173:10;159155:17;:29::i;:::-;159137:47;;159209:1;159199:7;:11;:40;;;;;159215:12;:24;159228:10;159215:24;;;;;;;;;;;;;;;;;;;;;;;;;159214:25;159199:40;159195:221;;;159282:4;159255:12;:24;159268:10;159255:24;;;;;;;;;;;;;;;;:31;;;;;;;;;;;;;;;;;;159301:23;159311:10;159322:1;159301:9;:23::i;:::-;159195:221;;;159365:5;159357:47;;;;;;;;;;;;:::i;:::-;;;;;;;;;159195:221;159126:297;159089:334::o;159882:93::-;17052:13;:11;:13::i;:::-;159953:14:::1;;;;;;;;;;;159952:15;159935:14;;:32;;;;;;;;;;;;;;;;;;159882:93::o:0;157248:384::-;157312:7;157332:16;157351:1;157332:20;;157367:6;;;;;;;;;;;:16;;;157384:6;157367:24;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;157363:236;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;157526:73;157363:236;;;157496:17;157485:28;;157392:133;157363:236;157616:8;157609:15;;;157248:384;;;:::o;157079:140::-;17052:13;:11;:13::i;:::-;157127:12:::1;157142:21;157127:36;;157182:10;157174:28;;:37;157203:7;157174:37;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;;;;;;157116:103;157079:140::o:0;153700:44::-;;;;;;;;;;;;;;;;;;;;;;:::o;155270:240::-;155439:4;3233:10;3225:18;;:4;:18;;;3221:184;;3265:31;3285:10;3265:19;:31::i;:::-;3260:134;;3321:27;:25;:27::i;:::-;3317:61;;;3350:28;3367:10;3350:16;:28::i;:::-;3317:61;3260:134;3221:184;155461:41:::1;155484:4;155490:2;155494:7;155461:22;:41::i;:::-;155270:240:::0;;;;:::o;79678:94::-;79744:20;79750:7;79759:4;79744:5;:20::i;:::-;79678:94;:::o;153029:47::-;153073:3;153029:47;:::o;159431:439::-;159499:7;:5;:7::i;:::-;159485:21;;:10;:21;;;159481:382;;159522:25;159532:10;159544:2;159522:9;:25::i;:::-;159562;159572:10;159584:2;159562:9;:25::i;:::-;159602;159612:10;159624:2;159602:9;:25::i;:::-;159481:382;;;159649:16;:28;159666:10;159649:28;;;;;;;;;;;;;;;;;;;;;;;;;159645:218;;;159724:5;159693:16;:28;159710:10;159693:28;;;;;;;;;;;;;;;;:36;;;;;;;;;;;;;;;;;;159744:24;159754:10;159766:1;159744:9;:24::i;:::-;159645:218;;;159809:5;159801:50;;;;;;;;;;;;:::i;:::-;;;;;;;;;159645:218;159481:382;159431:439::o;163671:1586::-;163746:14;;;;;;;;;;;163738:69;;;;;;;;;;;;:::i;:::-;;;;;;;;;163844:2;163826:14;:20;;163818:72;;;;;;;;;;;;:::i;:::-;;;;;;;;;163942:4;163925:14;163909:13;:11;:13::i;:::-;:30;;;;:::i;:::-;:37;163901:100;;;;;;;;;;;;:::i;:::-;;;;;;;;;164012:19;164066:2;164048:14;:20;164044:800;;164099:16;164085:30;;164184:9;164165:14;164152:11;:28;;;;:::i;:::-;:41;;164144:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;164044:800;;;164269:2;164251:14;:20;164247:597;;164302:17;164288:31;;164387:9;164368:14;164355:11;:28;;;;:::i;:::-;:41;;164347:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;164247:597;;;164472:1;164454:14;:19;164450:394;;164504:17;164490:31;;164589:9;164570:14;164557:11;:28;;;;:::i;:::-;:41;;164549:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;164450:394;;;164681:17;164667:31;;164787:9;164768:14;164755:11;:28;;;;:::i;:::-;:41;;164747:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;164450:394;164247:597;164044:800;164947:1;164942;164925:14;:18;;;;:::i;:::-;:23;164922:328;;164968:6;164964:97;164999:1;164984:14;:16;;;;:::i;:::-;164980:1;:20;164964:97;;;165026:19;165032:10;165043:1;165026:5;:19::i;:::-;165002:3;;;;;:::i;:::-;;;;164964:97;;;;164922:328;;;165097:6;165093:97;165128:1;165113:14;:16;;;;:::i;:::-;165109:1;:20;165093:97;;;165155:19;165161:10;165172:1;165155:5;:19::i;:::-;165131:3;;;;;:::i;:::-;;;;165093:97;;;;165204:34;165210:10;165236:1;165221:14;:16;;;;:::i;:::-;165204:5;:34::i;:::-;164922:328;163727:1530;163671:1586;:::o;157664:893::-;157728:7;157777:16;157796:1;157777:20;;157812:8;;;;;;;;;;;:18;;;157831:6;157812:26;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;157808:235;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;157970:73;157808:235;;;157942:15;157930:27;;;;;:::i;:::-;;;157839:130;157808:235;158057:8;;;;;;;;;;;:18;;;158076:6;158057:26;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;158053:233;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;158213:73;158053:233;;;158186:14;158174:26;;;;;:::i;:::-;;;158084:128;158053:233;158300:3;;;;;;;;;;;:13;;;158314:6;158300:21;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;158296:228;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;158451:73;158296:228;;;158424:14;158412:26;;;;;:::i;:::-;;;158322:128;158296:228;158541:8;158534:15;;;157664:893;;;:::o;13692:466::-;13898:19;13892:4;13885:33;13945:8;13939:4;13932:22;13998:1;13991:4;13985;13975:21;13968:32;14131:8;14085:44;14082:1;14079;14074:66;13692:466::o;154235:96::-;17052:13;:11;:13::i;:::-;154316:7:::1;154306;:17;;;;;;:::i;:::-;;154235:96:::0;:::o;153085:43::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;74391:528::-;74535:23;74601:22;74626:8;;:15;;74601:40;;74656:34;74714:14;74693:36;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;74656:73;;74749:9;74744:125;74765:14;74760:1;:19;74744:125;;74821:32;74841:8;;74850:1;74841:11;;;;;;;:::i;:::-;;;;;;;;74821:19;:32::i;:::-;74805:10;74816:1;74805:13;;;;;;;;:::i;:::-;;;;;;;:48;;;;74781:3;;;;;74744:125;;;;74890:10;74883:17;;;;74391:528;;;;:::o;40780:152::-;40852:7;40895:27;40914:7;40895:18;:27::i;:::-;40872:52;;40780:152;;;:::o;36322:233::-;36394:7;36435:1;36418:19;;:5;:19;;;36414:60;;36446:28;;;;;;;;;;;;;;36414:60;30481:13;36492:18;:25;36511:5;36492:25;;;;;;;;;;;;;;;;:55;36485:62;;36322:233;;;:::o;12718:102::-;17052:13;:11;:13::i;:::-;12791:21:::1;12809:1;12791:9;:21::i;:::-;12718:102::o:0;153645:48::-;;;;;;;;;;;;;;;;;;;;;;:::o;78267:900::-;78345:16;78399:19;78433:25;78473:22;78498:16;78508:5;78498:9;:16::i;:::-;78473:41;;78529:25;78571:14;78557:29;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;78529:57;;78601:31;;:::i;:::-;78652:9;78664:15;:13;:15::i;:::-;78652:27;;78647:472;78696:14;78681:11;:29;78647:472;;78748:15;78761:1;78748:12;:15::i;:::-;78736:27;;78786:9;:16;;;78827:8;78782:73;78903:1;78877:28;;:9;:14;;;:28;;;78873:111;;78950:9;:14;;;78930:34;;78873:111;79027:5;79006:26;;:17;:26;;;79002:102;;79083:1;79057:8;79066:13;;;;;;79057:23;;;;;;;;:::i;:::-;;;;;;;:27;;;;;79002:102;78647:472;78712:3;;;;;78647:472;;;;79140:8;79133:15;;;;;;;78267:900;;;:::o;153135:26::-;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::o;15701:196::-;15747:14;15862:15;15858:20;15852:27;15842:37;;15701:196;:::o;39563:104::-;39619:13;39652:7;39645:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;39563:104;:::o;75307:2513::-;75450:16;75517:4;75508:5;:13;75504:45;;75530:19;;;;;;;;;;;;;;75504:45;75564:19;75598:17;75618:14;:12;:14::i;:::-;75598:34;;75718:15;:13;:15::i;:::-;75710:5;:23;75706:87;;;75762:15;:13;:15::i;:::-;75754:23;;75706:87;75869:9;75862:4;:16;75858:73;;;75906:9;75899:16;;75858:73;75945:25;75973:16;75983:5;75973:9;:16::i;:::-;75945:44;;76167:4;76159:5;:12;76155:278;;;76192:19;76221:5;76214:4;:12;76192:34;;76263:17;76249:11;:31;76245:111;;;76325:11;76305:31;;76245:111;76173:198;76155:278;;;76416:1;76396:21;;76155:278;76447:25;76489:17;76475:32;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;76447:60;;76547:1;76526:17;:22;76522:78;;76576:8;76569:15;;;;;;;;76522:78;76744:31;76778:26;76798:5;76778:19;:26::i;:::-;76744:60;;76819:25;77064:9;:16;;;77059:92;;77121:9;:14;;;77101:34;;77059:92;77170:9;77182:5;77170:17;;77165:478;77194:4;77189:1;:9;;:45;;;;;77217:17;77202:11;:32;;77189:45;77165:478;;;77272:15;77285:1;77272:12;:15::i;:::-;77260:27;;77310:9;:16;;;77351:8;77306:73;77427:1;77401:28;;:9;:14;;;:28;;;77397:111;;77474:9;:14;;;77454:34;;77397:111;77551:5;77530:26;;:17;:26;;;77526:102;;77607:1;77581:8;77590:13;;;;;;77581:23;;;;;;;;:::i;:::-;;;;;;;:27;;;;;77526:102;77165:478;77236:3;;;;;77165:478;;;;77745:11;77735:8;77728:29;77793:8;77786:15;;;;;;;;75307:2513;;;;;;:::o;154560:228::-;154711:8;3590:29;3610:8;3590:19;:29::i;:::-;3585:122;;3640:27;:25;:27::i;:::-;3636:59;;;3669:26;3686:8;3669:16;:26::i;:::-;3636:59;3585:122;154737:43:::1;154761:8;154771;154737:23;:43::i;:::-;154560:228:::0;;;:::o;159983:1776::-;160062:14;;;;;;;;;;;160054:62;;;;;;;;;;;;:::i;:::-;;;;;;;;;160153:2;160135:14;:20;;160127:72;;;;;;;;;;;;:::i;:::-;;;;;;;;;160251:4;160234:14;160218:13;:11;:13::i;:::-;:30;;;;:::i;:::-;:37;160210:93;;;;;;;;;;;;:::i;:::-;;;;;;;;;160354:1;160322:29;160340:10;160322:17;:29::i;:::-;:33;160314:81;;;;;;;;;;;;:::i;:::-;;;;;;;;;160463:19;160581:2;160563:14;:20;160559:798;;160614:16;160600:30;;160699:9;160680:14;160667:11;:28;;;;:::i;:::-;:41;;160659:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;160559:798;;;160784:2;160766:14;:20;160762:595;;160817:16;160803:30;;160902:9;160883:14;160870:11;:28;;;;:::i;:::-;:41;;160862:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;160762:595;;;160987:1;160969:14;:19;160965:392;;161019:16;161005:30;;161104:9;161085:14;161072:11;:28;;;;:::i;:::-;:41;;161064:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;160965:392;;;161196:16;161182:30;;161300:9;161281:14;161268:11;:28;;;;:::i;:::-;:41;;161260:85;;;;;;;;;;;;:::i;:::-;;;;;;;;;160965:392;160762:595;160559:798;161449:1;161444;161427:14;:18;;;;:::i;:::-;:23;161424:328;;161470:6;161466:97;161501:1;161486:14;:16;;;;:::i;:::-;161482:1;:20;161466:97;;;161528:19;161534:10;161545:1;161528:5;:19::i;:::-;161504:3;;;;;:::i;:::-;;;;161466:97;;;;161424:328;;;161599:6;161595:97;161630:1;161615:14;:16;;;;:::i;:::-;161611:1;:20;161595:97;;;161657:19;161663:10;161674:1;161657:5;:19::i;:::-;161633:3;;;;;:::i;:::-;;;;161595:97;;;;161706:34;161712:10;161738:1;161723:14;:16;;;;:::i;:::-;161706:5;:34::i;:::-;161424:328;160043:1716;159983:1776;:::o;152920:45::-;152963:2;152920:45;:::o;158592:241::-;17052:13;:11;:13::i;:::-;158674:9:::1;158670:156;158693:5;:12;158689:1;:16;158670:156;;;158727:22;158752:5;158758:1;158752:8;;;;;;;;:::i;:::-;;;;;;;;158727:33;;158810:4;158775:16;:32;158792:14;158775:32;;;;;;;;;;;;;;;;:39;;;;;;;;;;;;;;;;;;158712:114;158707:3;;;;;:::i;:::-;;;;158670:156;;;;158592:241:::0;:::o;156423:117::-;17052:13;:11;:13::i;:::-;156527:5:::1;156500:24;;:32;;;;;;;;;;;;;;;;;;156423:117:::0;:::o;155518:265::-;155706:4;3233:10;3225:18;;:4;:18;;;3221:184;;3265:31;3285:10;3265:19;:31::i;:::-;3260:134;;3321:27;:25;:27::i;:::-;3317:61;;;3350:28;3367:10;3350:16;:28::i;:::-;3317:61;3260:134;3221:184;155728:47:::1;155751:4;155757:2;155761:7;155770:4;155728:22;:47::i;:::-;155518:265:::0;;;;;:::o;73804:428::-;73888:21;;:::i;:::-;73922:31;;:::i;:::-;73978:15;:13;:15::i;:::-;73968:7;:25;:54;;;;74008:14;:12;:14::i;:::-;73997:7;:25;;73968:54;73964:103;;;74046:9;74039:16;;;;;73964:103;74089:21;74102:7;74089:12;:21::i;:::-;74077:33;;74125:9;:16;;;74121:65;;;74165:9;74158:16;;;;;74121:65;74203:21;74216:7;74203:12;:21::i;:::-;74196:28;;;73804:428;;;;:::o;158841:240::-;17052:13;:11;:13::i;:::-;158945:1:::1;158928:13;:11;:13::i;:::-;:18;158920:60;;;;;;;;;;;;:::i;:::-;;;;;;;;;158991:22;153073:3;158991:42;;159044:29;159050:6;159058:14;159044:5;:29::i;:::-;158909:172;158841:240:::0;:::o;39773:318::-;39846:13;39877:16;39885:7;39877;:16::i;:::-;39872:59;;39902:29;;;;;;;;;;;;;;39872:59;39944:21;39968:10;:8;:10::i;:::-;39944:34;;40021:1;40002:7;39996:21;:26;:87;;;;;;;;;;;;;;;;;40049:7;40058:18;40068:7;40058:9;:18::i;:::-;40032:45;;;;;;;;;:::i;:::-;;;;;;;;;;;;;39996:87;39989:94;;;39773:318;;;:::o;16546:109::-;16612:6;16638:9;16631:16;;16546:109;:::o;46736:164::-;46833:4;46857:18;:25;46876:5;46857:25;;;;;;;;;;;;;;;:35;46883:8;46857:35;;;;;;;;;;;;;;;;;;;;;;;;;46850:42;;46736:164;;;;:::o;14349:1008::-;17052:13;:11;:13::i;:::-;14587:19:::1;14581:4;14574:33;14634:12;14628:4;14621:26;14697:4;14691;14681:21;14805:12;14799:19;14786:11;14783:36;14780:159;;;14852:35;14846:4;14839:49;14919:4;14913;14906:18;14780:159;15018:1;15004:12;14997:23;15105:4;15099:11;15095:2;15091:20;15241:8;15231;15191:38;15188:1;15185::::0;15180:70:::1;15330:8;15312:15;15308:20;15301:38;14503:847;;14349:1008:::0;:::o;12465:185::-;17052:13;:11;:13::i;:::-;12578:1:::1;12558:22;;:8;:22;;::::0;12554:58:::1;;12589:23;;;;;;;;;;;;;;12554:58;12623:19;12633:8;12623:9;:19::i;:::-;12465:185:::0;:::o;153211:36::-;;;;;;;;;;;;;:::o;16003:449::-;16126:14;16282:19;16276:4;16269:33;16329:12;16323:4;16316:26;16428:4;16422;16412:21;16406:28;16396:38;;16003:449;;;:::o;153170:34::-;;;;;;;;;;;;;:::o;38485:639::-;38570:4;38909:10;38894:25;;:11;:25;;;;:102;;;;38986:10;38971:25;;:11;:25;;;;38894:102;:179;;;;39063:10;39048:25;;:11;:25;;;;38894:179;38874:199;;38485:639;;;:::o;109708:215::-;109810:4;109849:26;109834:41;;;:11;:41;;;;:81;;;;109879:36;109903:11;109879:23;:36::i;:::-;109834:81;109827:88;;109708:215;;;:::o;11728:370::-;11948:15;11944:20;11938:27;11928:8;11925:41;11915:165;;12000:28;11994:4;11987:42;12060:4;12054;12047:18;11915:165;11728:370::o;111070:332::-;111189:17;:15;:17::i;:::-;111173:33;;:12;:33;;;;111165:88;;;;;;;;;;;;:::i;:::-;;;;;;;;;111292:1;111272:22;;:8;:22;;;111264:60;;;;;;;;;;;;:::i;:::-;;;;;;;;;111359:35;;;;;;;;111371:8;111359:35;;;;;;111381:12;111359:35;;;;;111337:19;:57;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;111070:332;;:::o;56807:2966::-;56880:20;56903:13;;56880:36;;56943:1;56931:8;:13;56927:44;;56953:18;;;;;;;;;;;;;;56927:44;56984:61;57014:1;57018:2;57022:12;57036:8;56984:21;:61::i;:::-;57528:1;30619:2;57498:1;:26;;57497:32;57485:8;:45;57459:18;:22;57478:2;57459:22;;;;;;;;;;;;;;;;:71;;;;;;;;;;;57807:139;57844:2;57898:33;57921:1;57925:2;57929:1;57898:14;:33::i;:::-;57865:30;57886:8;57865:20;:30::i;:::-;:66;57807:18;:139::i;:::-;57773:17;:31;57791:12;57773:31;;;;;;;;;;;:173;;;;57963:16;57994:11;58023:8;58008:12;:23;57994:37;;58544:16;58540:2;58536:25;58524:37;;58916:12;58876:8;58835:1;58773:25;58714:1;58653;58626:335;59287:1;59273:12;59269:20;59227:346;59328:3;59319:7;59316:16;59227:346;;59546:7;59536:8;59533:1;59506:25;59503:1;59500;59495:59;59381:1;59372:7;59368:15;59357:26;;59227:346;;;59231:77;59618:1;59606:8;:13;59602:45;;59628:19;;;;;;;;;;;;;;59602:45;59680:3;59664:13;:19;;;;57233:2462;;59705:60;59734:1;59738:2;59742:12;59756:8;59705:20;:60::i;:::-;56869:2904;56807:2966;;:::o;47158:282::-;47223:4;47279:7;47260:15;:13;:15::i;:::-;:26;;:66;;;;;47313:13;;47303:7;:23;47260:66;:153;;;;;47412:1;31257:8;47364:17;:26;47382:7;47364:26;;;;;;;;;;;;:44;:49;47260:153;47240:173;;47158:282;;;:::o;156681:386::-;156760:4;157016:42;156996:63;;:8;:63;;;156989:70;;156681:386;;;:::o;156548:125::-;156617:4;156641:24;;;;;;;;;;;156634:31;;156548:125;:::o;3823:1359::-;4216:22;4210:4;4203:36;4309:9;4303:4;4296:23;4384:8;4378:4;4371:22;4561:4;4555;4549;4543;4516:25;4509:5;4498:68;4488:274;;4682:16;4676:4;4670;4655:44;4730:16;4724:4;4717:30;4488:274;5162:1;5156:4;5149:15;3823:1359;:::o;45504:124::-;45593:27;45602:2;45606:7;45615:4;45593:8;:27::i;:::-;45504:124;;:::o;34654:92::-;34710:7;34654:92;:::o;49426:2825::-;49568:27;49598;49617:7;49598:18;:27::i;:::-;49568:57;;49683:4;49642:45;;49658:19;49642:45;;;49638:86;;49696:28;;;;;;;;;;;;;;49638:86;49738:27;49767:23;49794:35;49821:7;49794:26;:35::i;:::-;49737:92;;;;49929:68;49954:15;49971:4;49977:19;:17;:19::i;:::-;49929:24;:68::i;:::-;49924:180;;50017:43;50034:4;50040:19;:17;:19::i;:::-;50017:16;:43::i;:::-;50012:92;;50069:35;;;;;;;;;;;;;;50012:92;49924:180;50135:1;50121:16;;:2;:16;;;50117:52;;50146:23;;;;;;;;;;;;;;50117:52;50182:43;50204:4;50210:2;50214:7;50223:1;50182:21;:43::i;:::-;50318:15;50315:160;;;50458:1;50437:19;50430:30;50315:160;50855:18;:24;50874:4;50855:24;;;;;;;;;;;;;;;;50853:26;;;;;;;;;;;;50924:18;:22;50943:2;50924:22;;;;;;;;;;;;;;;;50922:24;;;;;;;;;;;51246:146;51283:2;51332:45;51347:4;51353:2;51357:19;51332:14;:45::i;:::-;31537:8;51304:73;51246:18;:146::i;:::-;51217:17;:26;51235:7;51217:26;;;;;;;;;;;:175;;;;51563:1;31537:8;51512:19;:47;:52;51508:627;;51585:19;51617:1;51607:7;:11;51585:33;;51774:1;51740:17;:30;51758:11;51740:30;;;;;;;;;;;;:35;51736:384;;51878:13;;51863:11;:28;51859:242;;52058:19;52025:17;:30;52043:11;52025:30;;;;;;;;;;;:52;;;;51859:242;51736:384;51566:569;51508:627;52182:7;52178:2;52163:27;;52172:4;52163:27;;;;;;;;;;;;52201:42;52222:4;52228:2;52232:7;52241:1;52201:20;:42::i;:::-;49557:2694;;;49426:2825;;;:::o;110702:97::-;110760:6;110786:5;110779:12;;110702:97;:::o;63298:112::-;63375:27;63385:2;63389:8;63375:27;;;;;;;;;;;;:9;:27::i;:::-;63298:112;;:::o;52347:193::-;52493:39;52510:4;52516:2;52520:7;52493:39;;;;;;;;;;;;:16;:39::i;:::-;52347:193;;;:::o;65293:3081::-;65373:27;65403;65422:7;65403:18;:27::i;:::-;65373:57;;65443:12;65474:19;65443:52;;65509:27;65538:23;65565:35;65592:7;65565:26;:35::i;:::-;65508:92;;;;65617:13;65613:316;;;65738:68;65763:15;65780:4;65786:19;:17;:19::i;:::-;65738:24;:68::i;:::-;65733:184;;65830:43;65847:4;65853:19;:17;:19::i;:::-;65830:16;:43::i;:::-;65825:92;;65882:35;;;;;;;;;;;;;;65825:92;65733:184;65613:316;65941:51;65963:4;65977:1;65981:7;65990:1;65941:21;:51::i;:::-;66085:15;66082:160;;;66225:1;66204:19;66197:30;66082:160;66903:1;30746:3;66873:1;:26;;66872:32;66844:18;:24;66863:4;66844:24;;;;;;;;;;;;;;;;:60;;;;;;;;;;;67171:176;67208:4;67279:53;67294:4;67308:1;67312:19;67279:14;:53::i;:::-;31537:8;31257;67232:43;67231:101;67171:18;:176::i;:::-;67142:17;:26;67160:7;67142:26;;;;;;;;;;;:205;;;;67518:1;31537:8;67467:19;:47;:52;67463:627;;67540:19;67572:1;67562:7;:11;67540:33;;67729:1;67695:17;:30;67713:11;67695:30;;;;;;;;;;;;:35;67691:384;;67833:13;;67818:11;:28;67814:242;;68013:19;67980:17;:30;67998:11;67980:30;;;;;;;;;;;:52;;;;67814:242;67691:384;67521:569;67463:627;68145:7;68141:1;68118:35;;68127:4;68118:35;;;;;;;;;;;;68164:50;68185:4;68199:1;68203:7;68212:1;68164:20;:50::i;:::-;68341:12;;:14;;;;;;;;;;;;;65362:3012;;;;65293:3081;;:::o;41935:1712::-;42002:14;42052:7;42033:15;:13;:15::i;:::-;:26;42029:1562;;42085:17;:26;42103:7;42085:26;;;;;;;;;;;;42076:35;;42189:1;31257:8;42161:6;:24;:29;42157:1423;;42310:1;42300:6;:11;42296:981;;42351:13;;42340:7;:24;42336:68;;42373:31;;;;;;;;;;;;;;42336:68;43001:257;43087:17;:28;43105:9;;;;;;;43087:28;;;;;;;;;;;;43078:37;;43183:1;43173:6;:11;43221:13;43169:25;43001:257;;42296:981;43551:13;;42157:1423;42029:1562;43608:31;;;;;;;;;;;;;;41935:1712;;;;:::o;11161:506::-;11315:15;11311:20;11414:8;11410:2;11406:17;11402:2;11398:26;11386:38;;11562:8;11550:9;11544:16;11504:38;11501:1;11498;11493:78;11640:8;11629:9;11622:27;11279:381;11161:506;:::o;41383:161::-;41451:21;;:::i;:::-;41492:44;41511:17;:24;41529:5;41511:24;;;;;;;;;;;;41492:18;:44::i;:::-;41485:51;;41383:161;;;:::o;34825:103::-;34880:7;34907:13;;34900:20;;34825:103;:::o;46345:234::-;46492:8;46440:18;:39;46459:19;:17;:19::i;:::-;46440:39;;;;;;;;;;;;;;;:49;46480:8;46440:49;;;;;;;;;;;;;;;;:60;;;;;;;;;;;;;;;;;;46552:8;46516:55;;46531:19;:17;:19::i;:::-;46516:55;;;46562:8;46516:55;;;;;;:::i;:::-;;;;;;;;46345:234;;:::o;53138:407::-;53313:31;53326:4;53332:2;53336:7;53313:12;:31::i;:::-;53377:1;53359:2;:14;;;:19;53355:183;;53398:56;53429:4;53435:2;53439:7;53448:5;53398:30;:56::i;:::-;53393:145;;53482:40;;;;;;;;;;;;;;53393:145;53355:183;53138:407;;;;:::o;41121:166::-;41191:21;;:::i;:::-;41232:47;41251:27;41270:7;41251:18;:27::i;:::-;41232:18;:47::i;:::-;41225:54;;41121:166;;;:::o;154339:117::-;154408:13;154441:7;154434:14;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;154339:117;:::o;70971:1745::-;71036:17;71470:4;71463;71457:11;71453:22;71562:1;71556:4;71549:15;71637:4;71634:1;71630:12;71623:19;;71719:1;71714:3;71707:14;71823:3;72062:5;72044:428;72070:1;72044:428;;;72110:1;72105:3;72101:11;72094:18;;72281:2;72275:4;72271:13;72267:2;72263:22;72258:3;72250:36;72375:2;72369:4;72365:13;72357:21;;72442:4;72044:428;72432:25;72044:428;72048:21;72511:3;72506;72502:13;72626:4;72621:3;72617:14;72610:21;;72691:6;72686:3;72679:19;71075:1634;;;70971:1745;;;:::o;108158:157::-;108243:4;108282:25;108267:40;;;:11;:40;;;;108260:47;;108158:157;;;:::o;54207:159::-;;;;;:::o;70073:311::-;70208:7;70228:16;31661:3;70254:19;:41;;70228:68;;31661:3;70322:31;70333:4;70339:2;70343:9;70322:10;:31::i;:::-;70314:40;;:62;;70307:69;;;70073:311;;;;;:::o;44747:324::-;44817:14;45050:1;45040:8;45037:15;45011:24;45007:46;44997:56;;44747:324;;;:::o;44195:450::-;44275:14;44443:16;44436:5;44432:28;44423:37;;44620:5;44606:11;44581:23;44577:41;44574:52;44567:5;44564:63;44554:73;;44195:450;;;;:::o;55031:158::-;;;;;:::o;64216:492::-;64345:13;64361:16;64369:7;64361;:16::i;:::-;64345:32;;64394:13;64390:219;;;64449:5;64426:28;;:19;:17;:19::i;:::-;:28;;;64422:187;;64478:44;64495:5;64502:19;:17;:19::i;:::-;64478:16;:44::i;:::-;64473:136;;64554:35;;;;;;;;;;;;;;64473:136;64422:187;64390:219;64654:2;64621:15;:24;64637:7;64621:24;;;;;;;;;;;:30;;;:35;;;;;;;;;;;;;;;;;;64692:7;64688:2;64672:28;;64681:5;64672:28;;;;;;;;;;;;64334:374;64216:492;;;:::o;48321:485::-;48423:27;48452:23;48493:38;48534:15;:24;48550:7;48534:24;;;;;;;;;;;48493:65;;48711:18;48688:41;;48768:19;48762:26;48743:45;;48673:126;48321:485;;;:::o;70764:105::-;70824:7;70851:10;70844:17;;70764:105;:::o;47549:659::-;47698:11;47863:16;47856:5;47852:28;47843:37;;48023:16;48012:9;48008:32;47995:45;;48173:15;48162:9;48159:30;48151:5;48140:9;48137:20;48134:56;48124:66;;47549:659;;;;;:::o;62525:689::-;62656:19;62662:2;62666:8;62656:5;:19::i;:::-;62735:1;62717:2;:14;;;:19;62713:483;;62757:11;62771:13;;62757:27;;62803:13;62825:8;62819:3;:14;62803:30;;62852:233;62883:62;62922:1;62926:2;62930:7;;;;;;62939:5;62883:30;:62::i;:::-;62878:167;;62981:40;;;;;;;;;;;;;;62878:167;63080:3;63072:5;:11;62852:233;;63167:3;63150:13;;:20;63146:34;;63172:8;;;63146:34;62738:458;;62713:483;62525:689;;;:::o;43746:366::-;43812:31;;:::i;:::-;43889:6;43856:9;:14;;:41;;;;;;;;;;;31140:3;43942:6;:33;;43908:9;:24;;:68;;;;;;;;;;;44034:1;31257:8;44006:6;:24;:29;;43987:9;:16;;:48;;;;;;;;;;;31661:3;44075:6;:28;;44046:9;:19;;:58;;;;;;;;;;;43746:366;;;:::o;55629:716::-;55792:4;55838:2;55813:45;;;55859:19;:17;:19::i;:::-;55880:4;55886:7;55895:5;55813:88;;;;;;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;55809:529;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;56113:1;56096:6;:13;:18;56092:235;;56142:40;;;;;;;;;;;;;;56092:235;56285:6;56279:13;56270:6;56266:2;56262:15;56255:38;55809:529;55982:54;;;55972:64;;;:6;:64;;;;55965:71;;;55629:716;;;;;;:::o;69774:147::-;69911:6;69774:147;;;;;:::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:77::-;2924:7;2953:5;2942:16;;2887:77;;;:::o;2970:122::-;3043:24;3061:5;3043:24;:::i;:::-;3036:5;3033:35;3023:63;;3082:1;3079;3072:12;3023:63;2970:122;:::o;3098:139::-;3144:5;3182:6;3169:20;3160:29;;3198:33;3225:5;3198:33;:::i;:::-;3098:139;;;;:::o;3243:329::-;3302:6;3351:2;3339:9;3330:7;3326:23;3322:32;3319:119;;;3357:79;;:::i;:::-;3319:119;3477:1;3502:53;3547:7;3538:6;3527:9;3523:22;3502:53;:::i;:::-;3492:63;;3448:117;3243:329;;;;:::o;3578:99::-;3630:6;3664:5;3658:12;3648:22;;3578:99;;;:::o;3683:169::-;3767:11;3801:6;3796:3;3789:19;3841:4;3836:3;3832:14;3817:29;;3683:169;;;;:::o;3858:246::-;3939:1;3949:113;3963:6;3960:1;3957:13;3949:113;;;4048:1;4043:3;4039:11;4033:18;4029:1;4024:3;4020:11;4013:39;3985:2;3982:1;3978:10;3973:15;;3949:113;;;4096:1;4087:6;4082:3;4078:16;4071:27;3920:184;3858:246;;;:::o;4110:102::-;4151:6;4202:2;4198:7;4193:2;4186:5;4182:14;4178:28;4168:38;;4110:102;;;:::o;4218:377::-;4306:3;4334:39;4367:5;4334:39;:::i;:::-;4389:71;4453:6;4448:3;4389:71;:::i;:::-;4382:78;;4469:65;4527:6;4522:3;4515:4;4508:5;4504:16;4469:65;:::i;:::-;4559:29;4581:6;4559:29;:::i;:::-;4554:3;4550:39;4543:46;;4310:285;4218:377;;;;:::o;4601:313::-;4714:4;4752:2;4741:9;4737:18;4729:26;;4801:9;4795:4;4791:20;4787:1;4776:9;4772:17;4765:47;4829:78;4902:4;4893:6;4829:78;:::i;:::-;4821:86;;4601:313;;;;:::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:118::-;5839:24;5857:5;5839:24;:::i;:::-;5834:3;5827:37;5752:118;;:::o;5876:222::-;5969:4;6007:2;5996:9;5992:18;5984:26;;6020:71;6088:1;6077:9;6073:17;6064:6;6020:71;:::i;:::-;5876:222;;;;:::o;6104:117::-;6213:1;6210;6203:12;6227:117;6336:1;6333;6326:12;6350:180;6398:77;6395:1;6388:88;6495:4;6492:1;6485:15;6519:4;6516:1;6509:15;6536:281;6619:27;6641:4;6619:27;:::i;:::-;6611:6;6607:40;6749:6;6737:10;6734:22;6713:18;6701:10;6698:34;6695:62;6692:88;;;6760:18;;:::i;:::-;6692:88;6800:10;6796:2;6789:22;6579:238;6536:281;;:::o;6823:129::-;6857:6;6884:20;;:::i;:::-;6874:30;;6913:33;6941:4;6933:6;6913:33;:::i;:::-;6823:129;;;:::o;6958:308::-;7020:4;7110:18;7102:6;7099:30;7096:56;;;7132:18;;:::i;:::-;7096:56;7170:29;7192:6;7170:29;:::i;:::-;7162:37;;7254:4;7248;7244:15;7236:23;;6958:308;;;:::o;7272:146::-;7369:6;7364:3;7359;7346:30;7410:1;7401:6;7396:3;7392:16;7385:27;7272:146;;;:::o;7424:425::-;7502:5;7527:66;7543:49;7585:6;7543:49;:::i;:::-;7527:66;:::i;:::-;7518:75;;7616:6;7609:5;7602:21;7654:4;7647:5;7643:16;7692:3;7683:6;7678:3;7674:16;7671:25;7668:112;;;7699:79;;:::i;:::-;7668:112;7789:54;7836:6;7831:3;7826;7789:54;:::i;:::-;7508:341;7424:425;;;;;:::o;7869:340::-;7925:5;7974:3;7967:4;7959:6;7955:17;7951:27;7941:122;;7982:79;;:::i;:::-;7941:122;8099:6;8086:20;8124:79;8199:3;8191:6;8184:4;8176:6;8172:17;8124:79;:::i;:::-;8115:88;;7931:278;7869:340;;;;:::o;8215:509::-;8284:6;8333:2;8321:9;8312:7;8308:23;8304:32;8301:119;;;8339:79;;:::i;:::-;8301:119;8487:1;8476:9;8472:17;8459:31;8517:18;8509:6;8506:30;8503:117;;;8539:79;;:::i;:::-;8503:117;8644:63;8699:7;8690:6;8679:9;8675:22;8644:63;:::i;:::-;8634:73;;8430:287;8215:509;;;;:::o;8730:619::-;8807:6;8815;8823;8872:2;8860:9;8851:7;8847:23;8843:32;8840:119;;;8878:79;;:::i;:::-;8840:119;8998:1;9023:53;9068:7;9059:6;9048:9;9044:22;9023:53;:::i;:::-;9013:63;;8969:117;9125:2;9151:53;9196:7;9187:6;9176:9;9172:22;9151:53;:::i;:::-;9141:63;;9096:118;9253:2;9279:53;9324:7;9315:6;9304:9;9300:22;9279:53;:::i;:::-;9269:63;;9224:118;8730:619;;;;;:::o;9355:474::-;9423:6;9431;9480:2;9468:9;9459:7;9455:23;9451:32;9448:119;;;9486:79;;:::i;:::-;9448:119;9606:1;9631:53;9676:7;9667:6;9656:9;9652:22;9631:53;:::i;:::-;9621:63;;9577:117;9733:2;9759:53;9804:7;9795:6;9784:9;9780:22;9759:53;:::i;:::-;9749:63;;9704:118;9355:474;;;;;:::o;9835:332::-;9956:4;9994:2;9983:9;9979:18;9971:26;;10007:71;10075:1;10064:9;10060:17;10051:6;10007:71;:::i;:::-;10088:72;10156:2;10145:9;10141:18;10132:6;10088:72;:::i;:::-;9835:332;;;;;:::o;10173:329::-;10232:6;10281:2;10269:9;10260:7;10256:23;10252:32;10249:119;;;10287:79;;:::i;:::-;10249:119;10407:1;10432:53;10477:7;10468:6;10457:9;10453:22;10432:53;:::i;:::-;10422:63;;10378:117;10173:329;;;;:::o;10508:117::-;10617:1;10614;10607:12;10631:117;10740:1;10737;10730:12;10771:568;10844:8;10854:6;10904:3;10897:4;10889:6;10885:17;10881:27;10871:122;;10912:79;;:::i;:::-;10871:122;11025:6;11012:20;11002:30;;11055:18;11047:6;11044:30;11041:117;;;11077:79;;:::i;:::-;11041:117;11191:4;11183:6;11179:17;11167:29;;11245:3;11237:4;11229:6;11225:17;11215:8;11211:32;11208:41;11205:128;;;11252:79;;:::i;:::-;11205:128;10771:568;;;;;:::o;11345:559::-;11431:6;11439;11488:2;11476:9;11467:7;11463:23;11459:32;11456:119;;;11494:79;;:::i;:::-;11456:119;11642:1;11631:9;11627:17;11614:31;11672:18;11664:6;11661:30;11658:117;;;11694:79;;:::i;:::-;11658:117;11807:80;11879:7;11870:6;11859:9;11855:22;11807:80;:::i;:::-;11789:98;;;;11585:312;11345:559;;;;;:::o;11910:145::-;12008:6;12042:5;12036:12;12026:22;;11910:145;;;:::o;12061:215::-;12191:11;12225:6;12220:3;12213:19;12265:4;12260:3;12256:14;12241:29;;12061:215;;;;:::o;12282:163::-;12380:4;12403:3;12395:11;;12433:4;12428:3;12424:14;12416:22;;12282:163;;;:::o;12451:108::-;12528:24;12546:5;12528:24;:::i;:::-;12523:3;12516:37;12451:108;;:::o;12565:101::-;12601:7;12641:18;12634:5;12630:30;12619:41;;12565:101;;;:::o;12672:105::-;12747:23;12764:5;12747:23;:::i;:::-;12742:3;12735:36;12672:105;;:::o;12783:99::-;12854:21;12869:5;12854:21;:::i;:::-;12849:3;12842:34;12783:99;;:::o;12888:91::-;12924:7;12964:8;12957:5;12953:20;12942:31;;12888:91;;;:::o;12985:105::-;13060:23;13077:5;13060:23;:::i;:::-;13055:3;13048:36;12985:105;;:::o;13168:864::-;13317:4;13312:3;13308:14;13404:4;13397:5;13393:16;13387:23;13423:63;13480:4;13475:3;13471:14;13457:12;13423:63;:::i;:::-;13332:164;13588:4;13581:5;13577:16;13571:23;13607:61;13662:4;13657:3;13653:14;13639:12;13607:61;:::i;:::-;13506:172;13762:4;13755:5;13751:16;13745:23;13781:57;13832:4;13827:3;13823:14;13809:12;13781:57;:::i;:::-;13688:160;13935:4;13928:5;13924:16;13918:23;13954:61;14009:4;14004:3;14000:14;13986:12;13954:61;:::i;:::-;13858:167;13286:746;13168:864;;:::o;14038:303::-;14169:10;14190:108;14294:3;14286:6;14190:108;:::i;:::-;14330:4;14325:3;14321:14;14307:28;;14038:303;;;;:::o;14347:144::-;14448:4;14480;14475:3;14471:14;14463:22;;14347:144;;;:::o;14573:980::-;14754:3;14783:85;14862:5;14783:85;:::i;:::-;14884:117;14994:6;14989:3;14884:117;:::i;:::-;14877:124;;15025:87;15106:5;15025:87;:::i;:::-;15135:7;15166:1;15151:377;15176:6;15173:1;15170:13;15151:377;;;15252:6;15246:13;15279:125;15400:3;15385:13;15279:125;:::i;:::-;15272:132;;15427:91;15511:6;15427:91;:::i;:::-;15417:101;;15211:317;15198:1;15195;15191:9;15186:14;;15151:377;;;15155:14;15544:3;15537:10;;14759:794;;;14573:980;;;;:::o;15559:497::-;15764:4;15802:2;15791:9;15787:18;15779:26;;15851:9;15845:4;15841:20;15837:1;15826:9;15822:17;15815:47;15879:170;16044:4;16035:6;15879:170;:::i;:::-;15871:178;;15559:497;;;;:::o;16062:114::-;16129:6;16163:5;16157:12;16147:22;;16062:114;;;:::o;16182:184::-;16281:11;16315:6;16310:3;16303:19;16355:4;16350:3;16346:14;16331:29;;16182:184;;;;:::o;16372:132::-;16439:4;16462:3;16454:11;;16492:4;16487:3;16483:14;16475:22;;16372:132;;;:::o;16510:108::-;16587:24;16605:5;16587:24;:::i;:::-;16582:3;16575:37;16510:108;;:::o;16624:179::-;16693:10;16714:46;16756:3;16748:6;16714:46;:::i;:::-;16792:4;16787:3;16783:14;16769:28;;16624:179;;;;:::o;16809:113::-;16879:4;16911;16906:3;16902:14;16894:22;;16809:113;;;:::o;16958:732::-;17077:3;17106:54;17154:5;17106:54;:::i;:::-;17176:86;17255:6;17250:3;17176:86;:::i;:::-;17169:93;;17286:56;17336:5;17286:56;:::i;:::-;17365:7;17396:1;17381:284;17406:6;17403:1;17400:13;17381:284;;;17482:6;17476:13;17509:63;17568:3;17553:13;17509:63;:::i;:::-;17502:70;;17595:60;17648:6;17595:60;:::i;:::-;17585:70;;17441:224;17428:1;17425;17421:9;17416:14;;17381:284;;;17385:14;17681:3;17674:10;;17082:608;;;16958:732;;;;:::o;17696:373::-;17839:4;17877:2;17866:9;17862:18;17854:26;;17926:9;17920:4;17916:20;17912:1;17901:9;17897:17;17890:47;17954:108;18057:4;18048:6;17954:108;:::i;:::-;17946:116;;17696:373;;;;:::o;18075:619::-;18152:6;18160;18168;18217:2;18205:9;18196:7;18192:23;18188:32;18185:119;;;18223:79;;:::i;:::-;18185:119;18343:1;18368:53;18413:7;18404:6;18393:9;18389:22;18368:53;:::i;:::-;18358:63;;18314:117;18470:2;18496:53;18541:7;18532:6;18521:9;18517:22;18496:53;:::i;:::-;18486:63;;18441:118;18598:2;18624:53;18669:7;18660:6;18649:9;18645:22;18624:53;:::i;:::-;18614:63;;18569:118;18075:619;;;;;:::o;18700:116::-;18770:21;18785:5;18770:21;:::i;:::-;18763:5;18760:32;18750:60;;18806:1;18803;18796:12;18750:60;18700:116;:::o;18822:133::-;18865:5;18903:6;18890:20;18881:29;;18919:30;18943:5;18919:30;:::i;:::-;18822:133;;;;:::o;18961:468::-;19026:6;19034;19083:2;19071:9;19062:7;19058:23;19054:32;19051:119;;;19089:79;;:::i;:::-;19051:119;19209:1;19234:53;19279:7;19270:6;19259:9;19255:22;19234:53;:::i;:::-;19224:63;;19180:117;19336:2;19362:50;19404:7;19395:6;19384:9;19380:22;19362:50;:::i;:::-;19352:60;;19307:115;18961:468;;;;;:::o;19435:311::-;19512:4;19602:18;19594:6;19591:30;19588:56;;;19624:18;;:::i;:::-;19588:56;19674:4;19666:6;19662:17;19654:25;;19734:4;19728;19724:15;19716:23;;19435:311;;;:::o;19769:710::-;19865:5;19890:81;19906:64;19963:6;19906:64;:::i;:::-;19890:81;:::i;:::-;19881:90;;19991:5;20020:6;20013:5;20006:21;20054:4;20047:5;20043:16;20036:23;;20107:4;20099:6;20095:17;20087:6;20083:30;20136:3;20128:6;20125:15;20122:122;;;20155:79;;:::i;:::-;20122:122;20270:6;20253:220;20287:6;20282:3;20279:15;20253:220;;;20362:3;20391:37;20424:3;20412:10;20391:37;:::i;:::-;20386:3;20379:50;20458:4;20453:3;20449:14;20442:21;;20329:144;20313:4;20308:3;20304:14;20297:21;;20253:220;;;20257:21;19871:608;;19769:710;;;;;:::o;20502:370::-;20573:5;20622:3;20615:4;20607:6;20603:17;20599:27;20589:122;;20630:79;;:::i;:::-;20589:122;20747:6;20734:20;20772:94;20862:3;20854:6;20847:4;20839:6;20835:17;20772:94;:::i;:::-;20763:103;;20579:293;20502:370;;;;:::o;20878:539::-;20962:6;21011:2;20999:9;20990:7;20986:23;20982:32;20979:119;;;21017:79;;:::i;:::-;20979:119;21165:1;21154:9;21150:17;21137:31;21195:18;21187:6;21184:30;21181:117;;;21217:79;;:::i;:::-;21181:117;21322:78;21392:7;21383:6;21372:9;21368:22;21322:78;:::i;:::-;21312:88;;21108:302;20878:539;;;;:::o;21423:323::-;21479:6;21528:2;21516:9;21507:7;21503:23;21499:32;21496:119;;;21534:79;;:::i;:::-;21496:119;21654:1;21679:50;21721:7;21712:6;21701:9;21697:22;21679:50;:::i;:::-;21669:60;;21625:114;21423:323;;;;:::o;21752:307::-;21813:4;21903:18;21895:6;21892:30;21889:56;;;21925:18;;:::i;:::-;21889:56;21963:29;21985:6;21963:29;:::i;:::-;21955:37;;22047:4;22041;22037:15;22029:23;;21752:307;;;:::o;22065:423::-;22142:5;22167:65;22183:48;22224:6;22183:48;:::i;:::-;22167:65;:::i;:::-;22158:74;;22255:6;22248:5;22241:21;22293:4;22286:5;22282:16;22331:3;22322:6;22317:3;22313:16;22310:25;22307:112;;;22338:79;;:::i;:::-;22307:112;22428:54;22475:6;22470:3;22465;22428:54;:::i;:::-;22148:340;22065:423;;;;;:::o;22507:338::-;22562:5;22611:3;22604:4;22596:6;22592:17;22588:27;22578:122;;22619:79;;:::i;:::-;22578:122;22736:6;22723:20;22761:78;22835:3;22827:6;22820:4;22812:6;22808:17;22761:78;:::i;:::-;22752:87;;22568:277;22507:338;;;;:::o;22851:943::-;22946:6;22954;22962;22970;23019:3;23007:9;22998:7;22994:23;22990:33;22987:120;;;23026:79;;:::i;:::-;22987:120;23146:1;23171:53;23216:7;23207:6;23196:9;23192:22;23171:53;:::i;:::-;23161:63;;23117:117;23273:2;23299:53;23344:7;23335:6;23324:9;23320:22;23299:53;:::i;:::-;23289:63;;23244:118;23401:2;23427:53;23472:7;23463:6;23452:9;23448:22;23427:53;:::i;:::-;23417:63;;23372:118;23557:2;23546:9;23542:18;23529:32;23588:18;23580:6;23577:30;23574:117;;;23610:79;;:::i;:::-;23574:117;23715:62;23769:7;23760:6;23749:9;23745:22;23715:62;:::i;:::-;23705:72;;23500:287;22851:943;;;;;;;:::o;23872:874::-;24031:4;24026:3;24022:14;24118:4;24111:5;24107:16;24101:23;24137:63;24194:4;24189:3;24185:14;24171:12;24137:63;:::i;:::-;24046:164;24302:4;24295:5;24291:16;24285:23;24321:61;24376:4;24371:3;24367:14;24353:12;24321:61;:::i;:::-;24220:172;24476:4;24469:5;24465:16;24459:23;24495:57;24546:4;24541:3;24537:14;24523:12;24495:57;:::i;:::-;24402:160;24649:4;24642:5;24638:16;24632:23;24668:61;24723:4;24718:3;24714:14;24700:12;24668:61;:::i;:::-;24572:167;24000:746;23872:874;;:::o;24752:347::-;24907:4;24945:3;24934:9;24930:19;24922:27;;24959:133;25089:1;25078:9;25074:17;25065:6;24959:133;:::i;:::-;24752:347;;;;:::o;25105:115::-;25190:23;25207:5;25190:23;:::i;:::-;25185:3;25178:36;25105:115;;:::o;25226:218::-;25317:4;25355:2;25344:9;25340:18;25332:26;;25368:69;25434:1;25423:9;25419:17;25410:6;25368:69;:::i;:::-;25226:218;;;;:::o;25450:474::-;25518:6;25526;25575:2;25563:9;25554:7;25550:23;25546:32;25543:119;;;25581:79;;:::i;:::-;25543:119;25701:1;25726:53;25771:7;25762:6;25751:9;25747:22;25726:53;:::i;:::-;25716:63;;25672:117;25828:2;25854:53;25899:7;25890:6;25879:9;25875:22;25854:53;:::i;:::-;25844:63;;25799:118;25450:474;;;;;:::o;25930:229::-;26070:34;26066:1;26058:6;26054:14;26047:58;26139:12;26134:2;26126:6;26122:15;26115:37;25930:229;:::o;26165:366::-;26307:3;26328:67;26392:2;26387:3;26328:67;:::i;:::-;26321:74;;26404:93;26493:3;26404:93;:::i;:::-;26522:2;26517:3;26513:12;26506:19;;26165:366;;;:::o;26537:419::-;26703:4;26741:2;26730:9;26726:18;26718:26;;26790:9;26784:4;26780:20;26776:1;26765:9;26761:17;26754:47;26818:131;26944:4;26818:131;:::i;:::-;26810:139;;26537:419;;;:::o;26962:226::-;27102:34;27098:1;27090:6;27086:14;27079:58;27171:9;27166:2;27158:6;27154:15;27147:34;26962:226;:::o;27194:366::-;27336:3;27357:67;27421:2;27416:3;27357:67;:::i;:::-;27350:74;;27433:93;27522:3;27433:93;:::i;:::-;27551:2;27546:3;27542:12;27535:19;;27194:366;;;:::o;27566:419::-;27732:4;27770:2;27759:9;27755:18;27747:26;;27819:9;27813:4;27809:20;27805:1;27794:9;27790:17;27783:47;27847:131;27973:4;27847:131;:::i;:::-;27839:139;;27566:419;;;:::o;27991:180::-;28039:77;28036:1;28029:88;28136:4;28133:1;28126:15;28160:4;28157:1;28150:15;28177:191;28217:3;28236:20;28254:1;28236:20;:::i;:::-;28231:25;;28270:20;28288:1;28270:20;:::i;:::-;28265:25;;28313:1;28310;28306:9;28299:16;;28334:3;28331:1;28328:10;28325:36;;;28341:18;;:::i;:::-;28325:36;28177:191;;;;:::o;28374:237::-;28514:34;28510:1;28502:6;28498:14;28491:58;28583:20;28578:2;28570:6;28566:15;28559:45;28374:237;:::o;28617:366::-;28759:3;28780:67;28844:2;28839:3;28780:67;:::i;:::-;28773:74;;28856:93;28945:3;28856:93;:::i;:::-;28974:2;28969:3;28965:12;28958:19;;28617:366;;;:::o;28989:419::-;29155:4;29193:2;29182:9;29178:18;29170:26;;29242:9;29236:4;29232:20;29228:1;29217:9;29213:17;29206:47;29270:131;29396:4;29270:131;:::i;:::-;29262:139;;28989:419;;;:::o;29414:410::-;29454:7;29477:20;29495:1;29477:20;:::i;:::-;29472:25;;29511:20;29529:1;29511:20;:::i;:::-;29506:25;;29566:1;29563;29559:9;29588:30;29606:11;29588:30;:::i;:::-;29577:41;;29767:1;29758:7;29754:15;29751:1;29748:22;29728:1;29721:9;29701:83;29678:139;;29797:18;;:::i;:::-;29678:139;29462:362;29414:410;;;;:::o;29830:181::-;29970:33;29966:1;29958:6;29954:14;29947:57;29830:181;:::o;30017:366::-;30159:3;30180:67;30244:2;30239:3;30180:67;:::i;:::-;30173:74;;30256:93;30345:3;30256:93;:::i;:::-;30374:2;30369:3;30365:12;30358:19;;30017:366;;;:::o;30389:419::-;30555:4;30593:2;30582:9;30578:18;30570:26;;30642:9;30636:4;30632:20;30628:1;30617:9;30613:17;30606:47;30670:131;30796:4;30670:131;:::i;:::-;30662:139;;30389:419;;;:::o;30814:180::-;30862:77;30859:1;30852:88;30959:4;30956:1;30949:15;30983:4;30980:1;30973:15;31000:176;31032:1;31049:20;31067:1;31049:20;:::i;:::-;31044:25;;31083:20;31101:1;31083:20;:::i;:::-;31078:25;;31122:1;31112:35;;31127:18;;:::i;:::-;31112:35;31168:1;31165;31161:9;31156:14;;31000:176;;;;:::o;31182:185::-;31222:1;31239:20;31257:1;31239:20;:::i;:::-;31234:25;;31273:20;31291:1;31273:20;:::i;:::-;31268:25;;31312:1;31302:35;;31317:18;;:::i;:::-;31302:35;31359:1;31356;31352:9;31347:14;;31182:185;;;;:::o;31373:233::-;31412:3;31435:24;31453:5;31435:24;:::i;:::-;31426:33;;31481:66;31474:5;31471:77;31468:103;;31551:18;;:::i;:::-;31468:103;31598:1;31591:5;31587:13;31580:20;;31373:233;;;:::o;31612:180::-;31660:77;31657:1;31650:88;31757:4;31754:1;31747:15;31781:4;31778:1;31771:15;31798:320;31842:6;31879:1;31873:4;31869:12;31859:22;;31926:1;31920:4;31916:12;31947:18;31937:81;;32003:4;31995:6;31991:17;31981:27;;31937:81;32065:2;32057:6;32054:14;32034:18;32031:38;32028:84;;32084:18;;:::i;:::-;32028:84;31849:269;31798:320;;;:::o;32124:141::-;32173:4;32196:3;32188:11;;32219:3;32216:1;32209:14;32253:4;32250:1;32240:18;32232:26;;32124:141;;;:::o;32271:93::-;32308:6;32355:2;32350;32343:5;32339:14;32335:23;32325:33;;32271:93;;;:::o;32370:107::-;32414:8;32464:5;32458:4;32454:16;32433:37;;32370:107;;;;:::o;32483:393::-;32552:6;32602:1;32590:10;32586:18;32625:97;32655:66;32644:9;32625:97;:::i;:::-;32743:39;32773:8;32762:9;32743:39;:::i;:::-;32731:51;;32815:4;32811:9;32804:5;32800:21;32791:30;;32864:4;32854:8;32850:19;32843:5;32840:30;32830:40;;32559:317;;32483:393;;;;;:::o;32882:60::-;32910:3;32931:5;32924:12;;32882:60;;;:::o;32948:142::-;32998:9;33031:53;33049:34;33058:24;33076:5;33058:24;:::i;:::-;33049:34;:::i;:::-;33031:53;:::i;:::-;33018:66;;32948:142;;;:::o;33096:75::-;33139:3;33160:5;33153:12;;33096:75;;;:::o;33177:269::-;33287:39;33318:7;33287:39;:::i;:::-;33348:91;33397:41;33421:16;33397:41;:::i;:::-;33389:6;33382:4;33376:11;33348:91;:::i;:::-;33342:4;33335:105;33253:193;33177:269;;;:::o;33452:73::-;33497:3;33452:73;:::o;33531:189::-;33608:32;;:::i;:::-;33649:65;33707:6;33699;33693:4;33649:65;:::i;:::-;33584:136;33531:189;;:::o;33726:186::-;33786:120;33803:3;33796:5;33793:14;33786:120;;;33857:39;33894:1;33887:5;33857:39;:::i;:::-;33830:1;33823:5;33819:13;33810:22;;33786:120;;;33726:186;;:::o;33918:543::-;34019:2;34014:3;34011:11;34008:446;;;34053:38;34085:5;34053:38;:::i;:::-;34137:29;34155:10;34137:29;:::i;:::-;34127:8;34123:44;34320:2;34308:10;34305:18;34302:49;;;34341:8;34326:23;;34302:49;34364:80;34420:22;34438:3;34420:22;:::i;:::-;34410:8;34406:37;34393:11;34364:80;:::i;:::-;34023:431;;34008:446;33918:543;;;:::o;34467:117::-;34521:8;34571:5;34565:4;34561:16;34540:37;;34467:117;;;;:::o;34590:169::-;34634:6;34667:51;34715:1;34711:6;34703:5;34700:1;34696:13;34667:51;:::i;:::-;34663:56;34748:4;34742;34738:15;34728:25;;34641:118;34590:169;;;;:::o;34764:295::-;34840:4;34986:29;35011:3;35005:4;34986:29;:::i;:::-;34978:37;;35048:3;35045:1;35041:11;35035:4;35032:21;35024:29;;34764:295;;;;:::o;35064:1395::-;35181:37;35214:3;35181:37;:::i;:::-;35283:18;35275:6;35272:30;35269:56;;;35305:18;;:::i;:::-;35269:56;35349:38;35381:4;35375:11;35349:38;:::i;:::-;35434:67;35494:6;35486;35480:4;35434:67;:::i;:::-;35528:1;35552:4;35539:17;;35584:2;35576:6;35573:14;35601:1;35596:618;;;;36258:1;36275:6;36272:77;;;36324:9;36319:3;36315:19;36309:26;36300:35;;36272:77;36375:67;36435:6;36428:5;36375:67;:::i;:::-;36369:4;36362:81;36231:222;35566:887;;35596:618;35648:4;35644:9;35636:6;35632:22;35682:37;35714:4;35682:37;:::i;:::-;35741:1;35755:208;35769:7;35766:1;35763:14;35755:208;;;35848:9;35843:3;35839:19;35833:26;35825:6;35818:42;35899:1;35891:6;35887:14;35877:24;;35946:2;35935:9;35931:18;35918:31;;35792:4;35789:1;35785:12;35780:17;;35755:208;;;35991:6;35982:7;35979:19;35976:179;;;36049:9;36044:3;36040:19;36034:26;36092:48;36134:4;36126:6;36122:17;36111:9;36092:48;:::i;:::-;36084:6;36077:64;35999:156;35976:179;36201:1;36197;36189:6;36185:14;36181:22;36175:4;36168:36;35603:611;;;35566:887;;35156:1303;;;35064:1395;;:::o;36465:179::-;36605:31;36601:1;36593:6;36589:14;36582:55;36465:179;:::o;36650:366::-;36792:3;36813:67;36877:2;36872:3;36813:67;:::i;:::-;36806:74;;36889:93;36978:3;36889:93;:::i;:::-;37007:2;37002:3;36998:12;36991:19;;36650:366;;;:::o;37022:419::-;37188:4;37226:2;37215:9;37211:18;37203:26;;37275:9;37269:4;37265:20;37261:1;37250:9;37246:17;37239:47;37303:131;37429:4;37303:131;:::i;:::-;37295:139;;37022:419;;;:::o;37447:143::-;37504:5;37535:6;37529:13;37520:22;;37551:33;37578:5;37551:33;:::i;:::-;37447:143;;;;:::o;37596:351::-;37666:6;37715:2;37703:9;37694:7;37690:23;37686:32;37683:119;;;37721:79;;:::i;:::-;37683:119;37841:1;37866:64;37922:7;37913:6;37902:9;37898:22;37866:64;:::i;:::-;37856:74;;37812:128;37596:351;;;;:::o;37953:182::-;38093:34;38089:1;38081:6;38077:14;38070:58;37953:182;:::o;38141:366::-;38283:3;38304:67;38368:2;38363:3;38304:67;:::i;:::-;38297:74;;38380:93;38469:3;38380:93;:::i;:::-;38498:2;38493:3;38489:12;38482:19;;38141:366;;;:::o;38513:419::-;38679:4;38717:2;38706:9;38702:18;38694:26;;38766:9;38760:4;38756:20;38752:1;38741:9;38737:17;38730:47;38794:131;38920:4;38794:131;:::i;:::-;38786:139;;38513:419;;;:::o;38938:180::-;38986:77;38983:1;38976:88;39083:4;39080:1;39073:15;39107:4;39104:1;39097:15;39124:222;39264:34;39260:1;39252:6;39248:14;39241:58;39333:5;39328:2;39320:6;39316:15;39309:30;39124:222;:::o;39352:366::-;39494:3;39515:67;39579:2;39574:3;39515:67;:::i;:::-;39508:74;;39591:93;39680:3;39591:93;:::i;:::-;39709:2;39704:3;39700:12;39693:19;;39352:366;;;:::o;39724:419::-;39890:4;39928:2;39917:9;39913:18;39905:26;;39977:9;39971:4;39967:20;39963:1;39952:9;39948:17;39941:47;40005:131;40131:4;40005:131;:::i;:::-;39997:139;;39724:419;;;:::o;40149:230::-;40289:34;40285:1;40277:6;40273:14;40266:58;40358:13;40353:2;40345:6;40341:15;40334:38;40149:230;:::o;40385:366::-;40527:3;40548:67;40612:2;40607:3;40548:67;:::i;:::-;40541:74;;40624:93;40713:3;40624:93;:::i;:::-;40742:2;40737:3;40733:12;40726:19;;40385:366;;;:::o;40757:419::-;40923:4;40961:2;40950:9;40946:18;40938:26;;41010:9;41004:4;41000:20;40996:1;40985:9;40981:17;40974:47;41038:131;41164:4;41038:131;:::i;:::-;41030:139;;40757:419;;;:::o;41182:222::-;41322:34;41318:1;41310:6;41306:14;41299:58;41391:5;41386:2;41378:6;41374:15;41367:30;41182:222;:::o;41410:366::-;41552:3;41573:67;41637:2;41632:3;41573:67;:::i;:::-;41566:74;;41649:93;41738:3;41649:93;:::i;:::-;41767:2;41762:3;41758:12;41751:19;;41410:366;;;:::o;41782:419::-;41948:4;41986:2;41975:9;41971:18;41963:26;;42035:9;42029:4;42025:20;42021:1;42010:9;42006:17;41999:47;42063:131;42189:4;42063:131;:::i;:::-;42055:139;;41782:419;;;:::o;42207:179::-;42347:31;42343:1;42335:6;42331:14;42324:55;42207:179;:::o;42392:366::-;42534:3;42555:67;42619:2;42614:3;42555:67;:::i;:::-;42548:74;;42631:93;42720:3;42631:93;:::i;:::-;42749:2;42744:3;42740:12;42733:19;;42392:366;;;:::o;42764:419::-;42930:4;42968:2;42957:9;42953:18;42945:26;;43017:9;43011:4;43007:20;43003:1;42992:9;42988:17;42981:47;43045:131;43171:4;43045:131;:::i;:::-;43037:139;;42764:419;;;:::o;43189:148::-;43291:11;43328:3;43313:18;;43189:148;;;;:::o;43343:390::-;43449:3;43477:39;43510:5;43477:39;:::i;:::-;43532:89;43614:6;43609:3;43532:89;:::i;:::-;43525:96;;43630:65;43688:6;43683:3;43676:4;43669:5;43665:16;43630:65;:::i;:::-;43720:6;43715:3;43711:16;43704:23;;43453:280;43343:390;;;;:::o;43739:435::-;43919:3;43941:95;44032:3;44023:6;43941:95;:::i;:::-;43934:102;;44053:95;44144:3;44135:6;44053:95;:::i;:::-;44046:102;;44165:3;44158:10;;43739:435;;;;;:::o;44180:229::-;44320:34;44316:1;44308:6;44304:14;44297:58;44389:12;44384:2;44376:6;44372:15;44365:37;44180:229;:::o;44415:366::-;44557:3;44578:67;44642:2;44637:3;44578:67;:::i;:::-;44571:74;;44654:93;44743:3;44654:93;:::i;:::-;44772:2;44767:3;44763:12;44756:19;;44415:366;;;:::o;44787:419::-;44953:4;44991:2;44980:9;44976:18;44968:26;;45040:9;45034:4;45030:20;45026:1;45015:9;45011:17;45004:47;45068:131;45194:4;45068:131;:::i;:::-;45060:139;;44787:419;;;:::o;45212:175::-;45352:27;45348:1;45340:6;45336:14;45329:51;45212:175;:::o;45393:366::-;45535:3;45556:67;45620:2;45615:3;45556:67;:::i;:::-;45549:74;;45632:93;45721:3;45632:93;:::i;:::-;45750:2;45745:3;45741:12;45734:19;;45393:366;;;:::o;45765:419::-;45931:4;45969:2;45958:9;45954:18;45946:26;;46018:9;46012:4;46008:20;46004:1;45993:9;45989:17;45982:47;46046:131;46172:4;46046:131;:::i;:::-;46038:139;;45765:419;;;:::o;46190:98::-;46241:6;46275:5;46269:12;46259:22;;46190:98;;;:::o;46294:168::-;46377:11;46411:6;46406:3;46399:19;46451:4;46446:3;46442:14;46427:29;;46294:168;;;;:::o;46468:373::-;46554:3;46582:38;46614:5;46582:38;:::i;:::-;46636:70;46699:6;46694:3;46636:70;:::i;:::-;46629:77;;46715:65;46773:6;46768:3;46761:4;46754:5;46750:16;46715:65;:::i;:::-;46805:29;46827:6;46805:29;:::i;:::-;46800:3;46796:39;46789:46;;46558:283;46468:373;;;;:::o;46847:640::-;47042:4;47080:3;47069:9;47065:19;47057:27;;47094:71;47162:1;47151:9;47147:17;47138:6;47094:71;:::i;:::-;47175:72;47243:2;47232:9;47228:18;47219:6;47175:72;:::i;:::-;47257;47325:2;47314:9;47310:18;47301:6;47257:72;:::i;:::-;47376:9;47370:4;47366:20;47361:2;47350:9;47346:18;47339:48;47404:76;47475:4;47466:6;47404:76;:::i;:::-;47396:84;;46847:640;;;;;;;:::o;47493:141::-;47549:5;47580:6;47574:13;47565:22;;47596:32;47622:5;47596:32;:::i;:::-;47493:141;;;;:::o;47640:349::-;47709:6;47758:2;47746:9;47737:7;47733:23;47729:32;47726:119;;;47764:79;;:::i;:::-;47726:119;47884:1;47909:63;47964:7;47955:6;47944:9;47940:22;47909:63;:::i;:::-;47899:73;;47855:127;47640:349;;;;:::o
Swarm Source
ipfs://ca2dfa77036efa5e87754a265606d8564a127fdd9a104bf47d8918401886ac04
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.