ERC-721
Overview
Max Total Supply
500 OMWC
Holders
21
Market
Volume (24H)
N/A
Min Price (24H)
N/A
Max Price (24H)
N/A
Other Info
Token Contract
Balance
1 OMWCLoading...
Loading
Loading...
Loading
Loading...
Loading
# | Exchange | Pair | Price | 24H Volume | % Volume |
---|
Contract Name:
TicketNFT
Compiler Version
v0.8.4+commit.c7e474f2
Optimization Enabled:
Yes with 1000 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity >=0.8.4 <0.8.6; /** * @title NFTs for the TicketCompetition to the World Cup for Metaframes * @author MetaFrames * @notice This is the NFT contract used as basis to determine the winner of the World Cup Ticket. This is also * related to the AccessPassNFT * @dev The flow of the contract is as follows: * Deployment: Contract is deployed and configured * ----------------------------------------------- * Airdrop: Minting TicketNFTs to 500 Random Users * -airdrop() * ----------------------------------------------- * Ticket Competition: Selection of the ticket winner * -setRegistered() means registration was done off-chain * -requestRandomWord() requests the random number from VRF * -ticketWinners() then returns the winner * ----------------------------------------------- * Winners Frozen: When winning tokens are barred from trading their tokens * ----------------------------------------------- * Trading Enabled: When all trading has been enabled again */ import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/token/ERC721/extensions/ERC721Royalty.sol"; import "@chainlink/contracts/src/v0.8/interfaces/VRFCoordinatorV2Interface.sol"; import "@chainlink/contracts/src/v0.8/VRFConsumerBaseV2.sol"; import "./AccessPassNFT.sol"; import "./interfaces/IAccessPassNFT.sol"; import "./interfaces/ITicketNFT.sol"; contract TicketNFT is Ownable, ERC721Royalty, VRFConsumerBaseV2, ITicketNFT { using Strings for uint256; uint16 public constant NUMBER_OF_WINNERS = 2; /** * @dev maxTotalSupply is the amount of NFTs that can be minted */ uint16 public immutable maxTotalSupply; /** * @dev contractURI is an OpenSea standard. This should point to a metadata that tells who will receive revenues * from OpensSea. See https://docs.opensea.io/docs/contract-level-metadata */ string public contractURI; /** * @dev frozenPeriod is a timestamp for when the ticket winners can start trading again */ uint256 public frozenPeriod; /** * @dev if set to true, that particular nft is a winner */ mapping(uint256 => bool) public isWinner; /** * @dev array of winning ids */ uint256[] public ticketWinners; /** * @dev baseURI is the base of the token metadata used in conjunction with the token id */ string public baseURI; /** * @dev flag that tells if the tickets have been airdropped */ bool public ticketsAirdropped; /** * @dev flag that tells if the registration has been set */ bool public hasSetRegistration; /** * @dev Mapping of token id to if the owner of that token id has not registered off-chain. * For example: * 1. owner of token id 0 has registered, so 0 => false * 2. owner of token id 2 has NOT registered, so 1 => true * This was purposely made as hasNotRegistered so that we only write for values that have not registered. * This is to save gas since there should be more people who have registered than those who have not. * The registration comes from an off-chain database. */ mapping(uint16 => bool) private _hasNotRegistered; /** * @dev the related AccessPassNFT to this contract */ AccessPassNFT public accessPassNFT; /** * @dev The following variables are needed to request a random value from Chainlink. * See https://docs.chain.link/docs/vrf-contracts/ */ address public chainlinkCoordinator; // Chainlink Coordinator address uint256 public requestId; // Chainlink request id for the selection of the ticket winner uint256 public randomWord; // Random value received from Chainlink VRF /** * @notice initializes the contract * @param baseURI_ is the metadata's uri * @param contractURI_ is for OpenSeas compatability * @param royaltyAddress receives royalties fee from selling this token * @param royaltyFee is the fees taken from second-hand selling. This is expressed in _royaltyFee/1000. * So to do 5% means supplying 50 since 50/1000 is 5% (see ERC2981 function _setDefaultRoyalty(address receiver, uint96 feeNumerator)) * @param accessPassNFT_ is the address of the AccessPassNFT related to this token * @param vrfCoordinator_ is the address of the VRF used for getting a random number * @param nftHolder is the temporary holder of the NFTs before the airdrop * @param frozenPeriod_ is a timestamp for when the ticket winners can start trading again */ constructor( string memory baseURI_, string memory contractURI_, address royaltyAddress, uint96 royaltyFee, AccessPassNFT accessPassNFT_, address vrfCoordinator_, address nftHolder, uint256 frozenPeriod_ ) ERC721("Maradona Official World Cup Ticket", "OMWC") VRFConsumerBaseV2(vrfCoordinator_){ // crucial to check if there is a '/' in the end since this can no longer be changed once set // must have a '/' in the end since the token id follows the '/' bytes memory bytesBaseURI = bytes(baseURI_); if (bytesBaseURI[bytesBaseURI.length - 1] != bytes("/")[0]) revert IncorrectValue("baseURI"); baseURI = baseURI_; if (bytes(contractURI_).length == 0) revert EmptyString("contractURI"); contractURI = contractURI_; if(royaltyAddress == address(0)) revert ZeroAddress("royaltyAddress"); // not checking royaltyFee on purpose here _setDefaultRoyalty(royaltyAddress, royaltyFee); uint16 maxTotalSupply_ = accessPassNFT_.ticketSupply(); maxTotalSupply = maxTotalSupply_; bytes4 accessPassNFTInterfaceId = type(IAccessPassNFT).interfaceId; if(!accessPassNFT_.supportsInterface(accessPassNFTInterfaceId)) revert IncorrectValue("accessPassNFT"); accessPassNFT = accessPassNFT_; if(address(vrfCoordinator_) == address(0)) revert ZeroAddress("vrfCoordinator"); chainlinkCoordinator = vrfCoordinator_; if(nftHolder == address(0)) revert ZeroAddress("nftHolder"); // sending nfts to nftHolder which will be the eventual owner of the contract who will do the airdrop for (uint256 i = 0; i < maxTotalSupply_; i++) { _safeMint(nftHolder, i); } // not checking frozenPeriod_ on purpose here because there's a way to change it later frozenPeriod = frozenPeriod_; } /********************** EXTERNAL ********************************/ /** * @inheritdoc ITicketNFT */ function airdrop( uint16[] calldata winners ) external override onlyOwner onlyDuring(ContractStatus.TICKETS_REVEALED) { if (winners.length != maxTotalSupply) revert IncorrectValue("winners"); for (uint256 i = 0; i < winners.length; i++) { safeTransferFrom(msg.sender, accessPassNFT.ownerOf(winners[i]), i); } ticketsAirdropped = true; emit TicketsAirdropped(winners); } /** * @inheritdoc ITicketNFT */ function requestRandomWord( uint64 subscriptionId, bytes32 gasLane, uint32 callbackGasLimit ) external override onlyOwner onlyDuring(ContractStatus.SET_REGISTRATION) { // making sure that the request has enough callbackGasLimit to execute if (callbackGasLimit < 150_000) revert IncorrectValue("callbackGasLimit"); /// Call Chainlink to receive a random word /// Will revert if subscription is not funded. VRFCoordinatorV2Interface coordinator = VRFCoordinatorV2Interface(chainlinkCoordinator); requestId = coordinator.requestRandomWords( gasLane, subscriptionId, 3, /// Request confirmations callbackGasLimit, 1 /// request 1 random number ); /// Now Chainlink will call us back in a future transaction, see function fulfillRandomWords emit RandomWordRequested(requestId); } /** * @inheritdoc ITicketNFT */ function setContractURI(string memory uri) external override onlyOwner() { if (bytes(uri).length == 0) revert EmptyString("contractURI"); emit ContractURISet(contractURI, uri); contractURI = uri; } /** * @inheritdoc ITicketNFT */ function setDefaultRoyalty(address royaltyAddress, uint96 royaltyFee) external override onlyOwner(){ if (address(0) == royaltyAddress) revert ZeroAddress("royaltyAddress"); _setDefaultRoyalty(royaltyAddress, royaltyFee); emit RoyaltiesSet(royaltyAddress, royaltyFee); } /** * @inheritdoc ITicketNFT */ function setRegistered( bool[] calldata hasRegistered_ ) external override onlyOwner onlyDuring(ContractStatus.AIRDROPPED_TICKETS) { // sending an empty array means all accounts have registered if (hasRegistered_.length == 0) { hasSetRegistration = true; } else { if (hasRegistered_.length != maxTotalSupply) revert IncorrectValue("hasRegistered"); uint16 notRegisteredCounter = 0; for (uint16 i = 0; i < hasRegistered_.length; i++) { if (!hasRegistered_[i]) { // only writing for those who have not registered _hasNotRegistered[i] = true; // counting how many accounts have not registred notRegisteredCounter++; } } // ensuring that there are enough registered to have enough winners if (maxTotalSupply - notRegisteredCounter < NUMBER_OF_WINNERS) revert IncorrectValue("notRegisteredCounter"); hasSetRegistration = true; } emit RegistrationSet(hasRegistered_); } /** * @inheritdoc ITicketNFT */ function setFrozenPeriod(uint256 frozenPeriod_) external override onlyOwner() onlyBefore(ContractStatus.WINNERS_FROZEN) { if (frozenPeriod_ < block.timestamp && frozenPeriod_ != 0) revert IncorrectValue("frozenPeriod_"); emit FrozenPeriodSet(frozenPeriod, frozenPeriod_); frozenPeriod = frozenPeriod_; } /********************** PUBLIC VIEW ********************************/ /** * @notice returns a token metadata's uri * @param tokenId is the id of the token being queried */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if(!_exists(tokenId)) revert NonExistentToken(); return string(abi.encodePacked(baseURI, tokenId.toString(), ".json")); } /** * @inheritdoc ITicketNFT */ function hasRegistered( uint16 tokenId ) public view override onlyOnOrAfter(ContractStatus.SET_REGISTRATION) returns (bool) { if(tokenId >= maxTotalSupply) revert NonExistentToken(); return !_hasNotRegistered[tokenId]; } /** * @inheritdoc ITicketNFT */ function isAccountWinner(address account) public view override returns (bool){ for (uint16 i = 0; i < ticketWinners.length; i++) { if (ownerOf(ticketWinners[i]) == account) return true; } return false; } /** * @inheritdoc ITicketNFT */ function contractStatus() public view override returns (ContractStatus) { if(randomWord != 0) { if(block.timestamp < frozenPeriod) return ContractStatus.WINNERS_FROZEN; else return ContractStatus.TRADING_ENABLED; } if(hasSetRegistration) return ContractStatus.SET_REGISTRATION; if(ticketsAirdropped) return ContractStatus.AIRDROPPED_TICKETS; if(accessPassNFT.ticketsRevealed()) return ContractStatus.TICKETS_REVEALED; return ContractStatus.PRE_AIRDROP; } /** * @inheritdoc ITicketNFT */ function totalSupply() public view override returns (uint256) { return maxTotalSupply; } /** * @inheritdoc IERC165 */ function supportsInterface(bytes4 interfaceId) public view override(ERC721Royalty, IERC165) returns (bool) { return interfaceId == type(ITicketNFT).interfaceId || super.supportsInterface(interfaceId); } /********************** INTERNAL ********************************/ /** * @notice check if token is frozen before transferring * @inheritdoc ERC721 * @param from is the address that will give the token * @param to is the address that will receive the token * @param tokenId is the id being transferred */ function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal override { // not allowing winningIds to be transferred when in frozenPeriod if ( block.timestamp < frozenPeriod && isWinner[tokenId] ) revert TransferringFrozenToken(tokenId, block.timestamp, frozenPeriod); } /** * @notice sets the winners */ function setWinners() internal { // Setup a pool with random values uint256 randomPoolSize = 16; uint256 batch = 0; uint16[] memory randomPool = randArray(randomPoolSize, batch++); uint256 counter = 0; uint16 randomId; for (uint16 i = 0; i < NUMBER_OF_WINNERS; i++) { randomId = randomPool[counter++]; if (counter == randomPoolSize) { randomPool = randArray(randomPoolSize, batch++); counter = 0; } // only stays in the loop when the current id has not registered or if the current id already won while(_hasNotRegistered[randomId] || isWinner[randomId]) { randomId = randomPool[counter++]; if (counter == randomPoolSize) { randomPool = randArray(randomPoolSize, batch++); counter = 0; } } ticketWinners.push(randomId); isWinner[randomId] = true; // Using mapping to keep track for if the id was already chosen as a winner } emit TicketWinnersFrozen(frozenPeriod); } /** * @notice Chainlink calls us with a random value. (See VRFConsumerBaseV2's fulfillRandomWords function) * @dev Note that this happens in a later transaction than the request. This approximately costs 139_000 in gas * @param requestId_ is the id of the request from VRF's side * @param randomWords is an array of random numbers generated by VRF */ function fulfillRandomWords( uint256 requestId_, uint256[] memory randomWords ) internal override { if(requestId != requestId_) revert IncorrectValue("requestId"); if(randomWord != 0) revert CallingMoreThanOnce(); randomWord = randomWords[0]; emit TicketWinnersSelected(randomWords[0]); setWinners(); } /** * @notice Returns a list of x random numbers, in increments of 16 numbers. * So you may receive x random numbers or up to 15 more. The random numbers are between 0 and 499 * Each batch will be different, you can call multiple times with different batch numbers * This routine is deterministic and will always return the same result if randomWord is the same * @param max is the max numbers needed in a batch * @param batch represents the batch number */ function randArray( uint256 max, uint256 batch ) internal view returns (uint16[] memory) { uint256 mask = 0xFFFF; // 0xFFFF == [1111111111111111], masking the last 16 bits uint256 mainCounterMax = max / 16; if (max % 16 > 0) { mainCounterMax +=1; } uint256 batchOffset = (batch * mainCounterMax * 16); uint16[] memory randomValues = new uint16[](mainCounterMax * 16); for (uint256 mainCounter = 0; mainCounter < mainCounterMax; mainCounter++) { uint256 randomValue = uint256(keccak256(abi.encode(randomWord, mainCounter + batchOffset))); for (uint256 subCounter = 0; subCounter < 16; subCounter++) { // Mask 16 bits, value between 0 .. maxTotalSupply-1 randomValues[mainCounter * 16 + subCounter] = uint16(randomValue & mask) % maxTotalSupply; // Right shift 16 bits into oblivion randomValue = randomValue / 2 ** 16; } } return randomValues; } /********************** MODIFIER ********************************/ /** * @notice functions like a less than to the supplied status * @param status is a ContractStatus in which the function must happen before in. For example: * setting the frozenPeriod should only happen before the ticketWinners have been selected to ensure that no one * messes with the trading period during ContractStatus.WINNERS_FROZEN. To do that add this modifier * with the parameter: ContractStatus.WINNERS_FROZEN */ modifier onlyBefore(ContractStatus status) { // asserting here because there should be no state before PRE_AIRDROP assert(status != ContractStatus.PRE_AIRDROP); ContractStatus lastStatus = ContractStatus(uint(status) - 1); if (contractStatus() >= status) revert IncorrectContractStatus(contractStatus(), lastStatus); _; } /** * @notice the current status must be equal to the status in the parameter * @param status is the ContractStatus it must be in */ modifier onlyDuring(ContractStatus status) { if (status != contractStatus()) revert IncorrectContractStatus(contractStatus(), status); _; } /** * @notice the current status must be greater than or equal to the status in the parameter * @param status that the contract must at least be in. For example: * getting the nftTiers should only happen when TIERS_RANDOMIZED has already happened. so the parameter will be * TIERS_RANDOMIZED, because the function can only work once the status is TIERS_RANDOMIZED or has passed that */ modifier onlyOnOrAfter(ContractStatus status) { if (contractStatus() < status) revert IncorrectContractStatus(contractStatus(), status); _; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(owner() == _msgSender(), "Ownable: caller is not the owner"); _; } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/extensions/ERC721Royalty.sol) pragma solidity ^0.8.0; import "../ERC721.sol"; import "../../common/ERC2981.sol"; import "../../../utils/introspection/ERC165.sol"; /** * @dev Extension of ERC721 with the ERC2981 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. * * 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 ERC721Royalty is ERC2981, ERC721 { /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721, ERC2981) returns (bool) { return super.supportsInterface(interfaceId); } /** * @dev See {ERC721-_burn}. This override additionally clears the royalty information for the token. */ function _burn(uint256 tokenId) internal virtual override { super._burn(tokenId); _resetTokenRoyalty(tokenId); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; interface VRFCoordinatorV2Interface { /** * @notice Get configuration relevant for making requests * @return minimumRequestConfirmations global min for request confirmations * @return maxGasLimit global max for request gas limit * @return s_provingKeyHashes list of registered key hashes */ function getRequestConfig() external view returns ( uint16, uint32, bytes32[] memory ); /** * @notice Request a set of random words. * @param keyHash - Corresponds to a particular oracle job which uses * that key for generating the VRF proof. Different keyHash's have different gas price * ceilings, so you can select a specific one to bound your maximum per request cost. * @param subId - The ID of the VRF subscription. Must be funded * with the minimum subscription balance required for the selected keyHash. * @param minimumRequestConfirmations - How many blocks you'd like the * oracle to wait before responding to the request. See SECURITY CONSIDERATIONS * for why you may want to request more. The acceptable range is * [minimumRequestBlockConfirmations, 200]. * @param callbackGasLimit - How much gas you'd like to receive in your * fulfillRandomWords callback. Note that gasleft() inside fulfillRandomWords * may be slightly less than this amount because of gas used calling the function * (argument decoding etc.), so you may need to request slightly more than you expect * to have inside fulfillRandomWords. The acceptable range is * [0, maxGasLimit] * @param numWords - The number of uint256 random values you'd like to receive * in your fulfillRandomWords callback. Note these numbers are expanded in a * secure way by the VRFCoordinator from a single random value supplied by the oracle. * @return requestId - A unique identifier of the request. Can be used to match * a request to a response in fulfillRandomWords. */ function requestRandomWords( bytes32 keyHash, uint64 subId, uint16 minimumRequestConfirmations, uint32 callbackGasLimit, uint32 numWords ) external returns (uint256 requestId); /** * @notice Create a VRF subscription. * @return subId - A unique subscription id. * @dev You can manage the consumer set dynamically with addConsumer/removeConsumer. * @dev Note to fund the subscription, use transferAndCall. For example * @dev LINKTOKEN.transferAndCall( * @dev address(COORDINATOR), * @dev amount, * @dev abi.encode(subId)); */ function createSubscription() external returns (uint64 subId); /** * @notice Get a VRF subscription. * @param subId - ID of the subscription * @return balance - LINK balance of the subscription in juels. * @return reqCount - number of requests for this subscription, determines fee tier. * @return owner - owner of the subscription. * @return consumers - list of consumer address which are able to use this subscription. */ function getSubscription(uint64 subId) external view returns ( uint96 balance, uint64 reqCount, address owner, address[] memory consumers ); /** * @notice Request subscription owner transfer. * @param subId - ID of the subscription * @param newOwner - proposed new owner of the subscription */ function requestSubscriptionOwnerTransfer(uint64 subId, address newOwner) external; /** * @notice Request subscription owner transfer. * @param subId - ID of the subscription * @dev will revert if original owner of subId has * not requested that msg.sender become the new owner. */ function acceptSubscriptionOwnerTransfer(uint64 subId) external; /** * @notice Add a consumer to a VRF subscription. * @param subId - ID of the subscription * @param consumer - New consumer which can use the subscription */ function addConsumer(uint64 subId, address consumer) external; /** * @notice Remove a consumer from a VRF subscription. * @param subId - ID of the subscription * @param consumer - Consumer to remove from the subscription */ function removeConsumer(uint64 subId, address consumer) external; /** * @notice Cancel a subscription * @param subId - ID of the subscription * @param to - Where to send the remaining LINK to */ function cancelSubscription(uint64 subId, address to) external; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; /** **************************************************************************** * @notice Interface for contracts using VRF randomness * ***************************************************************************** * @dev PURPOSE * * @dev Reggie the Random Oracle (not his real job) wants to provide randomness * @dev to Vera the verifier in such a way that Vera can be sure he's not * @dev making his output up to suit himself. Reggie provides Vera a public key * @dev to which he knows the secret key. Each time Vera provides a seed to * @dev Reggie, he gives back a value which is computed completely * @dev deterministically from the seed and the secret key. * * @dev Reggie provides a proof by which Vera can verify that the output was * @dev correctly computed once Reggie tells it to her, but without that proof, * @dev the output is indistinguishable to her from a uniform random sample * @dev from the output space. * * @dev The purpose of this contract is to make it easy for unrelated contracts * @dev to talk to Vera the verifier about the work Reggie is doing, to provide * @dev simple access to a verifiable source of randomness. It ensures 2 things: * @dev 1. The fulfillment came from the VRFCoordinator * @dev 2. The consumer contract implements fulfillRandomWords. * ***************************************************************************** * @dev USAGE * * @dev Calling contracts must inherit from VRFConsumerBase, and can * @dev initialize VRFConsumerBase's attributes in their constructor as * @dev shown: * * @dev contract VRFConsumer { * @dev constructor(<other arguments>, address _vrfCoordinator, address _link) * @dev VRFConsumerBase(_vrfCoordinator) public { * @dev <initialization with other arguments goes here> * @dev } * @dev } * * @dev The oracle will have given you an ID for the VRF keypair they have * @dev committed to (let's call it keyHash). Create subscription, fund it * @dev and your consumer contract as a consumer of it (see VRFCoordinatorInterface * @dev subscription management functions). * @dev Call requestRandomWords(keyHash, subId, minimumRequestConfirmations, * @dev callbackGasLimit, numWords), * @dev see (VRFCoordinatorInterface for a description of the arguments). * * @dev Once the VRFCoordinator has received and validated the oracle's response * @dev to your request, it will call your contract's fulfillRandomWords method. * * @dev The randomness argument to fulfillRandomWords is a set of random words * @dev generated from your requestId and the blockHash of the request. * * @dev If your contract could have concurrent requests open, you can use the * @dev requestId returned from requestRandomWords to track which response is associated * @dev with which randomness request. * @dev See "SECURITY CONSIDERATIONS" for principles to keep in mind, * @dev if your contract could have multiple requests in flight simultaneously. * * @dev Colliding `requestId`s are cryptographically impossible as long as seeds * @dev differ. * * ***************************************************************************** * @dev SECURITY CONSIDERATIONS * * @dev A method with the ability to call your fulfillRandomness method directly * @dev could spoof a VRF response with any random value, so it's critical that * @dev it cannot be directly called by anything other than this base contract * @dev (specifically, by the VRFConsumerBase.rawFulfillRandomness method). * * @dev For your users to trust that your contract's random behavior is free * @dev from malicious interference, it's best if you can write it so that all * @dev behaviors implied by a VRF response are executed *during* your * @dev fulfillRandomness method. If your contract must store the response (or * @dev anything derived from it) and use it later, you must ensure that any * @dev user-significant behavior which depends on that stored value cannot be * @dev manipulated by a subsequent VRF request. * * @dev Similarly, both miners and the VRF oracle itself have some influence * @dev over the order in which VRF responses appear on the blockchain, so if * @dev your contract could have multiple VRF requests in flight simultaneously, * @dev you must ensure that the order in which the VRF responses arrive cannot * @dev be used to manipulate your contract's user-significant behavior. * * @dev Since the block hash of the block which contains the requestRandomness * @dev call is mixed into the input to the VRF *last*, a sufficiently powerful * @dev miner could, in principle, fork the blockchain to evict the block * @dev containing the request, forcing the request to be included in a * @dev different block with a different hash, and therefore a different input * @dev to the VRF. However, such an attack would incur a substantial economic * @dev cost. This cost scales with the number of blocks the VRF oracle waits * @dev until it calls responds to a request. It is for this reason that * @dev that you can signal to an oracle you'd like them to wait longer before * @dev responding to the request (however this is not enforced in the contract * @dev and so remains effective only in the case of unmodified oracle software). */ abstract contract VRFConsumerBaseV2 { error OnlyCoordinatorCanFulfill(address have, address want); address private immutable vrfCoordinator; /** * @param _vrfCoordinator address of VRFCoordinator contract */ constructor(address _vrfCoordinator) { vrfCoordinator = _vrfCoordinator; } /** * @notice fulfillRandomness handles the VRF response. Your contract must * @notice implement it. See "SECURITY CONSIDERATIONS" above for important * @notice principles to keep in mind when implementing your fulfillRandomness * @notice method. * * @dev VRFConsumerBaseV2 expects its subcontracts to have a method with this * @dev signature, and will call it once it has verified the proof * @dev associated with the randomness. (It is triggered via a call to * @dev rawFulfillRandomness, below.) * * @param requestId The Id initially returned by requestRandomness * @param randomWords the VRF output expanded to the requested number of words */ function fulfillRandomWords(uint256 requestId, uint256[] memory randomWords) internal virtual; // rawFulfillRandomness is called by VRFCoordinator when it receives a valid VRF // proof. rawFulfillRandomness then calls fulfillRandomness, after validating // the origin of the call function rawFulfillRandomWords(uint256 requestId, uint256[] memory randomWords) external { if (msg.sender != vrfCoordinator) { revert OnlyCoordinatorCanFulfill(msg.sender, vrfCoordinator); } fulfillRandomWords(requestId, randomWords); } }
// SPDX-License-Identifier: MIT pragma solidity >=0.8.4 <0.8.6; /** * @title NFTs for the AccessPass to Metaframes Maradona Club * @author MetaFrames * @notice This NFT Contract follows the ERC721 Standards and sets one of its properties (tier), on chain. * The three tiers are as follows: GOLD, SILVER, BRONZE. This contract is also connected to Metaframes' TicketNFT * which are used to join Metaframes' Ticket Competition for the World Cup Ticket. * @dev The flow of this contract is as follows: * Deployment: Contract is deployed and configured * ----------------------------------------------- * Additional Configuration * -setWhitelistSigner() * -setPrivateMintingTimestamp() if privateMintingTimestamp is 0 * ----------------------------------------------- * Private Minting: Allows accounts in the mint whitelist to mint * ----------------------------------------------- * Public Minting: Allows all accounts to mint * -setPublicMintingTimestamp() if publicMintingTimestamp is 0 * ----------------------------------------------- * Reveal: Revealing the tiers * -randomizeTiers() * -nftTiers() then builds the final token metadata * ----------------------------------------------- * Airdrop: Minting TicketNFTs to 500 Random Users * -randomizeTickets() * -winners() then builds the 500 random users * NOTE: the actual minting will happen in the TicketNFT contract */ import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/utils/Counters.sol"; import "./WhitelistVerifier.sol"; import "@openzeppelin/contracts/utils/Strings.sol"; import "@openzeppelin/contracts/token/ERC721/extensions/ERC721Royalty.sol"; import "@chainlink/contracts/src/v0.8/interfaces/VRFCoordinatorV2Interface.sol"; import "@chainlink/contracts/src/v0.8/VRFConsumerBaseV2.sol"; import "./TicketNFT.sol"; import "./interfaces/IAccessPassNFT.sol"; import "./interfaces/ITicketNFT.sol"; contract AccessPassNFT is Ownable, WhitelistVerifier, ERC721Royalty, VRFConsumerBaseV2, IAccessPassNFT { using Counters for Counters.Counter; using Strings for uint256; /** * @dev maxTotalSupply is the amount of NFTs that can be minted */ uint16 public immutable maxTotalSupply; /** * @dev goldenTierSupply is the amount of GOLD NFTs */ uint16 public immutable goldenTierSupply; /** * @dev silverTierSupply is the amount of SILVER NFTs */ uint16 public immutable silverTierSupply; /** * @dev ticketSupply is the amount of TicketNFTs to be airdropped */ uint16 public immutable ticketSupply; /** * @dev Mapping minter address to amount minted */ struct Minted { uint256 publicMinted; uint256 privateMinted; } mapping(address => Minted) private minted; /** * @dev Keeps track of how many NFTs have been minted */ Counters.Counter public tokenIdCounter; /** * @dev privateMintingTimestamp sets when privateMinting is enabled. When this is 0, * it means all minting is disabled */ uint256 public privateMintingTimestamp; /** * @dev publicMintingTimestamp sets when publicMinting is enabled. When this is 0, it means * public minting is disabled. This value must be greater than the privateMintingTimestamp if this is not 0 */ uint256 public publicMintingTimestamp; /** * @dev price specifies how much eth an account pays for a mint. This is in wei */ uint256 public price; /** * @dev maxPublicMintable is the maximum an account can publicMint */ uint16 public maxPublicMintable; /** * @dev flag that tells if the final uri has been set */ bool public tiersRevealed; /** * @dev unrevealedURI is the placeholder token metadata when the reveal has not happened yet */ string unrevealedURI; /** * @dev baseURI is the base of the real token metadata after the reveal */ string baseURI; /** * @dev contractURI is an OpenSea standard. This should point to a metadata that tells who will * receive revenues from OpensSea. See https://docs.opensea.io/docs/contract-level-metadata */ string public contractURI; /** * @dev receives the eth from accounts private and public minting and the royalties from selling the token. * All revenues should be sent to this address */ address payable public treasury; /** * @dev ticketNFT decides the trade freeze when the winners have been selected */ TicketNFT public ticketNFT; /** * @dev The following variables are needed to request a random value from Chainlink * see https://docs.chain.link/docs/vrf-contracts/ */ address public chainlinkCoordinator; // Chainlink coordinator address uint256 public tiersRequestId; // Chainlink request id for tier randomization uint256 public tiersRandomWord; // Random value received from Chainlink VRF uint256 public ticketsRequestId; // Chainlink request id for ticket randomization uint256 public ticketsRandomWord; // Random value received from Chainlink VRF /** * @notice initializes the contract * @param treasury_ is the recipient of eth from private and public minting as well as the recipient for token selling fees * @param vrfCoordinator_ is the address of the VRF Contract for generating random number * @param maxTotalSupply_ is the max number of tokens that can be minted * @param goldenTierSupply_ is the max number of golden tiered tokens * @param silverTierSupply_ is the max number of silver tiered tokens * @param ticketSupply_ is the max number of tickets that will be airdropped * @param privateMintingTimestamp_ is when the private minting will be enabled. NOTE: this could also be set later. 0 is an acceptable value * @param royaltyFee is the fees taken from second-hand selling. This is expressed in _royaltyFee/10_000. * So to do 5% means supplying 500 since 500/10_000 is 5% (see ERC2981 function _setDefaultRoyalty(address receiver, uint96 feeNumerator)) * @param price_ is the price of a public or private mint in wei * @param contractURI_ is an OpenSeas standard and is necessary for getting revenues from OpenSeas * @param unrevealedURI_ is the token metadata placeholder while the reveal has not happened yet. */ constructor( address payable treasury_, address vrfCoordinator_, uint16 maxTotalSupply_, uint16 goldenTierSupply_, uint16 silverTierSupply_, uint16 ticketSupply_, uint256 privateMintingTimestamp_, uint96 royaltyFee, uint256 price_, string memory contractURI_, string memory unrevealedURI_ ) ERC721("Maradona Official Access Pass", "OMFC") WhitelistVerifier() VRFConsumerBaseV2(vrfCoordinator_) { if (treasury_ == address(0)) revert ZeroAddress("treasury"); treasury = treasury_; if (vrfCoordinator_ == address(0)) revert ZeroAddress("vrfCoordinator"); chainlinkCoordinator = vrfCoordinator_; if (maxTotalSupply_ == 0) revert IsZero("maxTotalSupply"); maxTotalSupply = maxTotalSupply_; // The following is to ensure that there will be bronzeTierSupply require( goldenTierSupply_ + silverTierSupply_ < maxTotalSupply_, "Tier Supplies must be less than maxTotalSupply" ); if (goldenTierSupply_ == 0) revert IsZero("goldenTierSupply"); goldenTierSupply = goldenTierSupply_; if (silverTierSupply_ == 0) revert IsZero("silverTierSupply"); silverTierSupply = silverTierSupply_; if (ticketSupply_ > maxTotalSupply_) revert IncorrectValue("ticketSupply"); ticketSupply = ticketSupply_; // not checking for zero on purpose here privateMintingTimestamp = privateMintingTimestamp_; if (royaltyFee == 0) revert IsZero("royaltyFee"); _setDefaultRoyalty(treasury_, royaltyFee); if (price_ == 0) revert IsZero("price"); price = price_; bytes memory bytesUnrevealedURI = bytes(unrevealedURI_); if (bytesUnrevealedURI[bytesUnrevealedURI.length - 1] != bytes("/")[0]) revert IncorrectValue("unrevealedURI"); unrevealedURI = unrevealedURI_; if (bytes(contractURI_).length == 0) revert EmptyString("contractURI"); contractURI = contractURI_; maxPublicMintable = 10; // mint one to deployer so the OpenSeas store front can be edited before private minting starts uint256 tokenId = tokenIdCounter.current(); _safeMint(msg.sender, tokenId); tokenIdCounter.increment(); // classifying this mint as a private mint minted[msg.sender].privateMinted += 1; } /********************** EXTERNAL ********************************/ /** * @inheritdoc IAccessPassNFT */ function privateMint( VerifiedSlot calldata verifiedSlot ) external override payable onlyDuring(ContractStatus.PRIVATE_MINTING) { validateVerifiedSlot(msg.sender, minted[msg.sender].privateMinted, verifiedSlot); internalMint(msg.sender, msg.value, MintingType.PRIVATE_MINT); } /** * @inheritdoc IAccessPassNFT */ function publicMint() external override payable onlyDuring(ContractStatus.PUBLIC_MINTING) { if (minted[msg.sender].publicMinted >= maxPublicMintable) revert ExceedMintingCapacity(minted[msg.sender].publicMinted); internalMint(msg.sender, msg.value, MintingType.PUBLIC_MINT); } /** * @inheritdoc IAccessPassNFT */ function randomizeTiers( uint64 subscriptionId, bytes32 gasLane, uint32 callbackGasLimit ) external override onlyOwner onlyOnOrAfter(ContractStatus.END_MINTING) { /// Only allow randomize if random word has not been set if (tiersRandomWord != 0) revert CanNoLongerCall(); // making sure that the request has enough callbackGasLimit to execute if (callbackGasLimit < 40_000) revert IncorrectValue("callbackGasLimit"); /// Call Chainlink to receive a random word /// Will revert if subscription is not funded. VRFCoordinatorV2Interface coordinator = VRFCoordinatorV2Interface(chainlinkCoordinator); /// Now Chainlink will call us back in a future transaction, see function fulfillRandomWords tiersRequestId = coordinator.requestRandomWords( gasLane, subscriptionId, 3, /// Request confirmations callbackGasLimit, 1 /// request 1 random number ); emit TiersRandomWordRequested(tiersRequestId); } /** * @inheritdoc IAccessPassNFT */ function revealTiers( string memory revealedURI ) external override onlyOwner onlyOnOrAfter(ContractStatus.TIERS_RANDOMIZED) { if (tiersRevealed) revert CallingMoreThanOnce(); bytes memory bytesRevealedURI = bytes(revealedURI); if (bytesRevealedURI[bytesRevealedURI.length - 1] != bytes("/")[0]) revert IncorrectValue("revealedURI"); baseURI = revealedURI; tiersRevealed = true; emit TiersRevealed(); } /** * @inheritdoc IAccessPassNFT */ function randomizeTickets( uint64 subscriptionId, bytes32 gasLane, uint32 callbackGasLimit ) external override onlyOwner onlyOnOrAfter(ContractStatus.END_MINTING) { // Only allow randomize if random word has not been set if (ticketsRandomWord != 0) revert CanNoLongerCall(); // making sure that the request has enough callbackGasLimit to execute if (callbackGasLimit < 40_000) revert IncorrectValue("callbackGasLimit"); /// Call Chainlink to receive a random word /// Will revert if subscription is not funded. VRFCoordinatorV2Interface coordinator = VRFCoordinatorV2Interface(chainlinkCoordinator); ticketsRequestId = coordinator.requestRandomWords( gasLane, subscriptionId, 3, /// Request confirmations callbackGasLimit, 1 /// request 1 random number ); emit TicketsRandomWordRequested(ticketsRequestId); /// Now Chainlink will call us back in a future transaction, see function fulfillRandomWords } /** * @inheritdoc IAccessPassNFT */ function setWhitelistSigner(address whiteListSigner_) external override onlyOwner { _setWhiteListSigner(whiteListSigner_); } /** * @inheritdoc IAccessPassNFT */ function setTicketNFT(TicketNFT ticketNFT_) external override onlyOwner() { bytes4 ticketNFTInterfaceId = type(ITicketNFT).interfaceId; if (!ticketNFT_.supportsInterface(ticketNFTInterfaceId)) revert IncorrectValue("ticketNFT_"); // should not be able to setTicketNFT if ticketNFTs have been airdropped if (address(ticketNFT) != address(0)) { // contractStatus 2 means that the tickets have been airdropped so any status before that should be good if (uint(ticketNFT.contractStatus()) > 1) revert CanNoLongerCall(); } emit TicketNFTSet(ticketNFT, ticketNFT_); ticketNFT = ticketNFT_; } /** * @inheritdoc IAccessPassNFT */ function setTreasury(address payable treasury_) external override onlyOwner() { if (treasury_ == address(0)) revert ZeroAddress("treasury"); emit TreasurySet(treasury, treasury_); treasury = treasury_; } /** * @inheritdoc IAccessPassNFT */ function setRoyaltyFee(uint96 royaltyFee) external override onlyOwner() { _setDefaultRoyalty(treasury, royaltyFee); emit RoyaltyFeesSet(royaltyFee); } /** * @inheritdoc IAccessPassNFT */ function setPrice(uint256 price_) external override onlyOwner { if (price_ == 0) revert IsZero("price"); emit PriceSet(price, price_); price = price_; } /** * @inheritdoc IAccessPassNFT */ function setContractURI(string memory contractURI_) external override onlyOwner() { if (bytes(contractURI_).length == 0) revert EmptyString("contractURI"); emit ContractURISet(contractURI, contractURI_); contractURI = contractURI_; } /** * @inheritdoc IAccessPassNFT */ function setUnrevealedURI(string memory unrevealedURI_) external override onlyOwner { bytes memory bytesUnrevealedURI = bytes(unrevealedURI_); if (bytesUnrevealedURI[bytesUnrevealedURI.length - 1] != bytes("/")[0]) revert IncorrectValue("unrevealedURI"); emit UnrevealedURISet(unrevealedURI, unrevealedURI_); unrevealedURI = unrevealedURI_; } /** * @inheritdoc IAccessPassNFT */ function setPrivateMintingTimestamp( uint256 privateMintingTimestamp_ ) external override onlyOwner onlyBefore(ContractStatus.PRIVATE_MINTING) { if ( privateMintingTimestamp_ >= publicMintingTimestamp && privateMintingTimestamp_ != 0 && publicMintingTimestamp != 0 ) revert IncorrectValue("privateMintingTimestamp"); emit PrivateMintingTimestampSet(privateMintingTimestamp, privateMintingTimestamp_); privateMintingTimestamp = privateMintingTimestamp_; } /** * @inheritdoc IAccessPassNFT */ function setPublicMintingTimestamp( uint256 publicMintingTimestamp_ ) external override onlyOwner onlyBefore(ContractStatus.PUBLIC_MINTING) { if ( publicMintingTimestamp_ < privateMintingTimestamp && publicMintingTimestamp_ != 0 ) revert IncorrectValue("publicMintingTimestamp"); emit PublicMintingTimestampSet(publicMintingTimestamp, publicMintingTimestamp_); publicMintingTimestamp = publicMintingTimestamp_; } /** * @inheritdoc IAccessPassNFT */ function setMaxPublicMintable(uint16 maxPublicMintable_) external override onlyOwner { if (maxPublicMintable_ == 0) revert IsZero("maxPublicMintable"); emit MaxPublicMintableSet(maxPublicMintable, maxPublicMintable_); maxPublicMintable = maxPublicMintable_; } /********************** EXTERNAL VIEW ********************************/ /** * @inheritdoc IAccessPassNFT */ function mintedBy(address minter) external view override returns (uint256) { if(minter == address(0)) revert ZeroAddressQuery(); return minted[minter].privateMinted + minted[minter].publicMinted; } /** * @inheritdoc IAccessPassNFT */ function mintedBy(address minter, MintingType mintingType) external view override returns (uint256) { if(minter == address(0)) revert ZeroAddressQuery(); if (mintingType == MintingType.PRIVATE_MINT) return minted[minter].privateMinted; else return minted[minter].publicMinted; } /** * @inheritdoc IAccessPassNFT */ function nftTier( uint256 tokenId ) external view override onlyOnOrAfter(ContractStatus.TIERS_RANDOMIZED) returns (uint16 tier) { if (!_exists(tokenId)) revert NonExistentToken(); return nftTiers()[tokenId]; } /** * @inheritdoc IAccessPassNFT */ function ticketsRevealed() external view override returns(bool) { return ticketsRandomWord != 0; } /** * @inheritdoc IAccessPassNFT */ function winners() external view override onlyOnOrAfter(ContractStatus.TICKETS_RANDOMIZED) returns (uint16[] memory) { // Setup a pool with random values uint256 randomPoolSize = 100; uint256 batch = 0; uint16[] memory randomPool = randArray(ticketsRandomWord, randomPoolSize, batch++); /// Setup an array with nfts that will be returned uint16[] memory nfts = new uint16[](maxTotalSupply); uint256 counter; uint256 randomId; // Assign 500 winners for(uint256 i = 0; i < ticketSupply; i++) { randomId = randomPool[counter++]; if (counter == randomPoolSize) { randomPool = randArray(ticketsRandomWord, randomPoolSize, batch++); counter = 0; } while(nfts[randomId] != 0) { randomId = randomPool[counter++]; if (counter == randomPoolSize) { randomPool = randArray(ticketsRandomWord, randomPoolSize, batch++); counter = 0; } } nfts[randomId] = 1; // Winner } return nfts; } /********************** PUBLIC ********************************/ /** * @inheritdoc IAccessPassNFT */ function totalSupply() public view override returns (uint256) { return tokenIdCounter.current(); } /** * @inheritdoc IAccessPassNFT */ function contractStatus() public view override returns (ContractStatus) { if (ticketsRandomWord != 0) return ContractStatus.TICKETS_RANDOMIZED; if (tiersRevealed) return ContractStatus.TIERS_REVEALED; if (tiersRandomWord != 0) return ContractStatus.TIERS_RANDOMIZED; if (maxTotalSupply == tokenIdCounter.current()) return ContractStatus.END_MINTING; if ( block.timestamp >= privateMintingTimestamp && privateMintingTimestamp != 0 && ( block.timestamp < publicMintingTimestamp || publicMintingTimestamp == 0 ) ) return ContractStatus.PRIVATE_MINTING; if ( block.timestamp >= publicMintingTimestamp && publicMintingTimestamp != 0 && privateMintingTimestamp != 0 ) return ContractStatus.PUBLIC_MINTING; return ContractStatus.NO_MINTING; } /** * @notice returns the unrevealed uri when the reveal hasn't happened yet and when it has, returns the real uri * @param tokenId should be a minted tokenId owned by an account */ function tokenURI(uint256 tokenId) public view override returns (string memory) { if (!_exists(tokenId)) revert NonExistentToken(); if (!tiersRevealed) return string(abi.encodePacked(unrevealedURI, tokenId.toString(), ".json")); return string(abi.encodePacked(baseURI, tokenId.toString(), ".json")); } /** * @inheritdoc IAccessPassNFT */ function nftTiers() public view override onlyOnOrAfter(ContractStatus.TIERS_RANDOMIZED) returns (uint16[] memory) { /// Setup a pool with random values uint256 randomPoolSize = 500; uint256 batch = 0; uint16[] memory randomPool = randArray(tiersRandomWord, randomPoolSize, batch++); /// Setup an array with nfts that will be returned uint16[] memory nfts = new uint16[](maxTotalSupply); uint256 counter; /// Loop counter to check when we exhaust our random pool and need to fill it again uint256 randomId; /// Random NFT id /// Assign goldenTierSupply golden tier nfts for(uint256 i = 0; i < goldenTierSupply; i++) { randomId = randomPool[counter++]; if (counter == randomPoolSize) { /// If we exhaust the random pool, fill it again randomPool = randArray(tiersRandomWord, randomPoolSize, batch++); counter = 0; } while(nfts[randomId] != 0) { /// Loop while the NFT id already has a tier assigned randomId = randomPool[counter++]; /// If we exhaust the random pool, fill it again if (counter == randomPoolSize) { randomPool = randArray(tiersRandomWord, randomPoolSize, batch++); counter = 0; } } nfts[randomId] = uint16(Tier.GOLD); } // Assign silverTierSupply silver tier nfts for(uint256 i = 0; i < silverTierSupply; i++) { randomId = randomPool[counter++]; if (counter == randomPoolSize) { /// If we exhaust the random pool, fill it again randomPool = randArray(tiersRandomWord, randomPoolSize, batch++); counter = 0; } while(nfts[randomId] != 0) { /// Loop while the NFT id already has a tier assigned randomId = randomPool[counter++]; if (counter == randomPoolSize) { /// If we exhaust the random pool, fill it again randomPool = randArray(tiersRandomWord, randomPoolSize, batch++); counter = 0; } } nfts[randomId] = uint16(Tier.SILVER); } // All remaining nfts are automatically bronze because they are already set to 0 return nfts; } /** * @inheritdoc IERC165 */ function supportsInterface(bytes4 interfaceId) public view override(ERC721Royalty, IERC165) returns (bool) { return interfaceId == type(IAccessPassNFT).interfaceId || super.supportsInterface(interfaceId); } /********************** INTERNAL ********************************/ /** * @notice check if the owner has a winning ticket * @inheritdoc ERC721 */ function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal override { if (address(ticketNFT) != address(0)) { uint256 frozenPeriod = ticketNFT.frozenPeriod(); // not allowing winners to transfer if they only have one AccessPassNFT if ( block.timestamp < frozenPeriod && ticketNFT.isAccountWinner(from) && balanceOf(from) == 1 ) revert TransferringFrozenAccount(from, block.timestamp, frozenPeriod); } } /** * @notice pays treasury the amount * @param account is the account that paid * @param amount is how much the account has paid */ function payTreasury(address account, uint256 amount) internal { (bool success, ) = treasury.call{value: amount}(""); require (success, "Could not pay treasury"); emit TreasuryPaid(account, amount); } /** * @notice internal mint function * @param to is the account receiving the NFT * @param amountPaid is the amount that the account has paid for the mint * @param mintingType could be PRIVATE_MINT or PUBLIC_MINT */ function internalMint( address to, uint256 amountPaid, MintingType mintingType ) internal onlyBefore(ContractStatus.END_MINTING) { if (amountPaid != price) revert IncorrectValue("amountPaid"); uint256 tokenId = tokenIdCounter.current(); payTreasury(to, amountPaid); tokenIdCounter.increment(); if (MintingType.PRIVATE_MINT == mintingType) { minted[to].privateMinted += 1; } else { minted[to].publicMinted += 1; } _safeMint(to, tokenId); } /** * @notice Chainlink calls us with a random value. (See VRFConsumerBaseV2's fulfillRandomWords function) * @dev Note that this happens in a later transaction than the request. * @param requestId is the id of the request from VRF's side * @param randomWords is an array of random numbers generated by VRF */ function fulfillRandomWords( uint256 requestId, uint256[] memory randomWords ) internal override { if (requestId == 0) revert IsZero("requestId"); if (requestId == tiersRequestId) { if (tiersRandomWord != 0) revert CallingMoreThanOnce(); tiersRandomWord = randomWords[0]; /// Set the random value received from Chainlink emit TiersRandomized(tiersRandomWord); } else if (requestId == ticketsRequestId) { if (ticketsRandomWord != 0) revert CallingMoreThanOnce(); ticketsRandomWord = randomWords[0]; /// Set the random value received from Chainlink emit TicketsRandomized(ticketsRandomWord); } } /** * @notice Returns a list of x random numbers, in increments of 16 numbers. * So you may receive x random numbers or up to 15 more. The random numbers are between 0 and 499 * Each batch will be different, you can call multiple times with different batch numbers * This routine is deterministic and will always return the same result if randomWord is the same * @param randomWord can only be tiersRandomWord and ticketsRandomWord * @param max is the max numbers needed in a batch * @param batch represents the batch number */ function randArray(uint256 randomWord, uint256 max, uint256 batch) internal view returns (uint16[] memory) { // First make sure the random chainlinkVRF value is initialized if (randomWord == 0) revert IsZero("randomWord"); uint256 mask = 0xFFFF; // 0xFFFF == [1111111111111111], masking the last 16 bits uint256 mainCounterMax = max / 16; if (max % 16 > 0) { mainCounterMax +=1; } uint256 batchOffset = (batch * mainCounterMax * 16); uint16[] memory randomValues = new uint16[](mainCounterMax * 16); for (uint256 mainCounter = 0; mainCounter < mainCounterMax; mainCounter++) { uint256 randomValue = uint256(keccak256(abi.encode(randomWord, mainCounter + batchOffset))); for (uint256 subCounter = 0; subCounter < 16; subCounter++) { randomValues[mainCounter * 16 + subCounter] = uint16(randomValue & mask) % maxTotalSupply; // Mask 16 bits, value between 0 .. MAX_TOTAL_SUPPLY-1 randomValue = randomValue / 2 ** 16; // Right shift 16 bits into oblivion } } return randomValues; } /********************** MODIFIERS ********************************/ /** * @notice functions like a less than to the supplied status * @param status is a ContractStatus in which the function must happen before in. For example: * setting the privateMintTimestamp should only happen before private minting starts to ensure that no one * messes with the privateMint settings during ContractStatus.PrivateMinting. To do that add this modifier * with the parameter: ContractStatus.PrivateMinting */ modifier onlyBefore(ContractStatus status) { // asserting here because there should be no state before NO_MINTING assert(status != ContractStatus.NO_MINTING); ContractStatus lastStatus = ContractStatus(uint(status) - 1); if (contractStatus() >= status) revert IncorrectContractStatus(contractStatus(), lastStatus); _; } /** * @notice functions like a an equal to the supplied status * @param status is the ContractStatus it must be in */ modifier onlyDuring(ContractStatus status) { if (contractStatus() != status) revert IncorrectContractStatus(contractStatus(), status); _; } /** * @notice functions like a greater than or equal to. The current status must be the same as or happened after the parameter. * @param status that the contract must at least be in. For example: * getting the nftTiers should only happen when TIERS_RANDOMIZED has already happened. so the parameter will be * TIERS_RANDOMIZED, because the function can only work once the status is TIERS_RANDOMIZED or has passed that */ modifier onlyOnOrAfter(ContractStatus status) { if (contractStatus() < status) revert IncorrectContractStatus(contractStatus(), status); _; } }
// SPDX-License-Identifier: MIT pragma solidity >=0.8.4 <0.8.6; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; import "../TicketNFT.sol"; import "./IVerifiedSlot.sol"; /** * @title Required interface for an AccessPassNFT compliant contract * @author Oost & Voort, Inc */ interface IAccessPassNFT is IERC165, IVerifiedSlot { /** * @dev The following are the stages of the contract in order: * NO_MINTING: Minting is not yet allowed * PRIVATE_MINTING: Only people in the mint whitelist can mint * PUBLIC_MINTING: Everyone can mint * END_MINTING: When everything's been minted already * TIERS_RANDOMIZED: When a random number has been set for the tiers * TIERS_REVEALED: When the final token metadata has been uploaded to IPFS * TICKETS_RANDOMIZED: When a random number has been set for the tickets airdrop */ enum ContractStatus { NO_MINTING, PRIVATE_MINTING, PUBLIC_MINTING, END_MINTING, TIERS_RANDOMIZED, TIERS_REVEALED, TICKETS_RANDOMIZED } /** * @dev Minting types are explained below: * PRIVATE_MINT: minted using the private mint function * PUBLIC_MINT: minted using the public mint function */ enum MintingType {PRIVATE_MINT, PUBLIC_MINT} /** * @dev The on-chain property of the nft that is determined by a random number */ enum Tier {BRONZE, SILVER, GOLD} /** * @dev emitted when the owner has set the private minting timestamp * @param oldTimestamp is for what the timestamp used to be * @param newTimestamp is the new value */ event PrivateMintingTimestampSet(uint256 oldTimestamp, uint256 newTimestamp); /** * @dev emitted when the owner has set the public minting timestamp * @param oldTimestamp is for what the timestamp used to be * @param newTimestamp is the new value */ event PublicMintingTimestampSet(uint256 oldTimestamp, uint256 newTimestamp); /** * @dev emitted when the owner has changed the max number of nfts a public user can mint * @param oldMaxPublicMintable is the old value for the maximum a public account can mint * @param newMaxPublicMintable is the new value for the maximum a public account can mint */ event MaxPublicMintableSet(uint16 oldMaxPublicMintable, uint16 newMaxPublicMintable); /** * @dev emitted when the owner changes the treasury * @param oldTreasury is the old value for the treasury * @param newTreasury is the new value for the treasury */ event TreasurySet(address oldTreasury, address newTreasury); /** * @dev emitted when the owner changes the minting price * @param oldPrice is the price the minting was set as * @param newPrice is the new price minting will cost as */ event PriceSet(uint256 oldPrice, uint256 newPrice); /** * @dev emitted when the owner changes the royalties * @param newRoyalties is the new royalties set by the owner */ event RoyaltyFeesSet(uint96 newRoyalties); /** * @dev emitted when the owner has changed the contract uri * @param oldURI is the uri it was set as before * @param newURI is the uri it is now set in */ event ContractURISet(string oldURI, string newURI); /** * @dev emitted when the owner has changed the unrevealed uri * @param oldURI is the uri it was set as before * @param newURI is the uri it is now set in */ event UnrevealedURISet(string oldURI, string newURI); /** * @dev emitted when the TicketNFT has been set * @param oldTicketNFT is the old TicketNFT it was pointing to * @param newTicketNFT is the TicketNFT it is now pointing to */ event TicketNFTSet(TicketNFT oldTicketNFT, TicketNFT newTicketNFT); /** * @dev emitted when the treasury has been paid in ETH * @param account is the account that paid the treasury * @param amount is how much ETH the account sent to the treasury */ event TreasuryPaid(address indexed account, uint256 amount); /** * @dev the following events must be done in order */ /** * @dev emitted when the owner has requested a random word from VRF to set the tiers of each NFT * @param requestId is the id set by VRF */ event TiersRandomWordRequested(uint256 requestId); /** * @dev emitted when VRF has used fulfillRandomness to set the random number * @param randomWord is the randomWord given back in a callback by VRF */ event TiersRandomized(uint256 randomWord); /** * @dev emitted when the owner has put the final token metadata uri for the nfts */ event TiersRevealed(); /** * @dev emitted when the owner has requested a random word from VRF to set who will be airdropped TicketNFTs * @param requestId is the id set by VRF */ event TicketsRandomWordRequested(uint256 requestId); /** * @dev emitted when VRF has used fulfillRandomness to set the random number * @param randomWord is the randomWord given back in a callback by VRF */ event TicketsRandomized(uint256 randomWord); /** * @dev reverted with this error when the address being supplied is Zero Address * @param addressName is for whom the Zero Address is being set for */ error ZeroAddress(string addressName); /** * @dev reverted with this error when a view function is asking for a Zero Address' information */ error ZeroAddressQuery(); /** * @dev reverted with this error when a view function is being used to look for a nonExistent Token */ error NonExistentToken(); /** * @dev reverted with this error when a function is being called more than once */ error CallingMoreThanOnce(); /** * @dev reverted with this error when a function should no longer be called */ error CanNoLongerCall(); /** * @dev reverted with this error when a variable being supplied is valued 0 * @param variableName is the name of the variable being supplied with 0 */ error IsZero(string variableName); /** * @dev reverted with this error when a variable has an incorrect value * @param variableName is the name of the variable with an incorrect value */ error IncorrectValue(string variableName); /** * @dev reverted with this error when a string being supplied should not be empty * @param stringName is the name of the string being supplied with an empty value */ error EmptyString(string stringName); /** * @dev reverted with this error when a function being called should not be called with the current Contract Status * @param currentStatus is the contract's current status * @param requiredStatus is the status the current must be in for the function to not revert */ error IncorrectContractStatus(ContractStatus currentStatus, ContractStatus requiredStatus); /** * @dev reverted with this error when an account that has won is trying to transfer his or her last AccessPassNFT * during WINNERS_FROZEN in TicketNFT * @param account is the address trying to transfer * @param currentTimestamp is the current block's timestamp * @param requiredTimestamp is the timestamp the block must at least be in */ error TransferringFrozenAccount(address account, uint256 currentTimestamp, uint256 requiredTimestamp); /********************** EXTERNAL ********************************/ /** * @notice private mints for people in the whitelist * @param verifiedSlot is a signed message by the whitelist signer that presents how many the minter can mint */ function privateMint(VerifiedSlot calldata verifiedSlot) external payable; /* * @notice public mints for anyone */ function publicMint() external payable; /** * @notice Randomize the NFT. This requests a random Chainlink value, which causes the tier of each nft id to be known. * @dev See https://docs.chain.link/docs/vrf-contracts/#configurations for Chainlink VRF documentation * @param subscriptionId The chainlink subscription id that pays for the call to Chainlink, needs to be setup with ChainLink beforehand * @param gasLane The maximum gas price you are willing to pay for a Chainlink VRF request in wei * @param callbackGasLimit How much gas to use for the callback request. Approximately 29_000 is used up solely by * fulfillRandomWords */ function randomizeTiers( uint64 subscriptionId, bytes32 gasLane, uint32 callbackGasLimit ) external; /** * @notice sets the base URI for the token metadata * @dev This can only happen once after the generation of the token metadata in unison with the winners function. * @param revealedURI must end in a '/' (slash), because the tokenURI expects it to end in a slash. */ function revealTiers(string memory revealedURI) external; /** * @notice Randomize the tickets. This requests a random Chainlink value, which causes the winners to be known. * @dev See https://docs.chain.link/docs/vrf-contracts/#configurations for Chainlink VRF documentation * @param subscriptionId The chainlink subscription id that pays for the call to Chainlink, needs to be setup with ChainLink beforehand * @param gasLane The maximum gas price you are willing to pay for a Chainlink VRF request in wei * @param callbackGasLimit How much gas to use for the callback request. Approximately 31_000 gas is used up * solely by fulfillRandomWords. */ function randomizeTickets( uint64 subscriptionId, bytes32 gasLane, uint32 callbackGasLimit ) external; /** * @notice sets the whitelist signer * @dev immediately do this after deploying the contract * @param whiteListSigner_ is the signer address for verifying the minting slots */ function setWhitelistSigner(address whiteListSigner_) external; /** * @notice sets the ticketNFT * @dev set this before selecting the TicketWinners in TicketNFT * @param ticketNFT_ is the TicketNFT that selects the ticketWinners */ function setTicketNFT(TicketNFT ticketNFT_) external; /** * @notice sets the recipient of the eth from public and private minting and the royalty fees * @dev setRoyaltyFee right after setting the treasury * @param treasury_ could be an EOA or a gnosis contract that receives eth and royalty fees */ function setTreasury(address payable treasury_) external; /** * @notice sets the royalty fee for the second hand market selling * @param royaltyFee is the fees taken from second-hand selling. This is expressed in a _royaltyFee/10_000. * So to do 5% means supplying 500 since 500/10_000 is 5% (see ERC2981 function _setDefaultRoyalty(address receiver, uint96 feeNumerator)) */ function setRoyaltyFee(uint96 royaltyFee) external; /** * @notice sets the price of minting. the amount is sent to the treasury right after the minting * @param price_ is expressed in wei */ function setPrice(uint256 price_) external; /** * @notice sets the contract uri * @param contractURI_ points to a json file that follows OpenSeas standard (see https://docs.opensea.io/docs/contract-level-metadata) */ function setContractURI(string memory contractURI_) external; /** * @notice sets the unrevealedURI * @param unrevealedURI_ points to a json file with the placeholder image inside */ function setUnrevealedURI(string memory unrevealedURI_) external; /** * @notice sets the private minting timestamp * @param privateMintingTimestamp_ is when private minting is enabled. Setting this to zero disables all minting */ function setPrivateMintingTimestamp(uint256 privateMintingTimestamp_) external; /** * @notice sets the public minting timestamp * @param publicMintingTimestamp_ is when public minting will be enabled. * Setting this to zero disables public minting. * If set, public minting must happen after private minting */ function setPublicMintingTimestamp(uint256 publicMintingTimestamp_) external; /** /* @notice sets how many a minter can public mint /* @param maxPublicMintable_ is how many a public account can mint */ function setMaxPublicMintable(uint16 maxPublicMintable_) external; /********************** EXTERNAL VIEW ********************************/ /** * @notice returns the count an account has minted * @param minter is for the account being queried */ function mintedBy(address minter) external view returns (uint256); /** * @notice returns the count an account has minted per type * @param minter is for the account being queried * @param mintingType is the type of minting expected */ function mintedBy(address minter, MintingType mintingType) external view returns (uint256); /** * @notice Returns the tier for an nft id * @param tokenId is the id of the token being queried */ function nftTier(uint256 tokenId) external view returns (uint16 tier); /** * @notice Returns true if the ticketsRandomWord has been set in the VRF Callback * @dev this is used by TicketNFT as a prerequisite for the airdrop. See TicketNFT for more info. */ function ticketsRevealed() external view returns(bool); /** * @notice Returns an array of all NFT id's, with 500 winners, indicated by 1. The others are indicated by 0. */ function winners() external view returns (uint16[] memory); /** * @notice returns the current supply of the NFT */ function totalSupply() external view returns (uint256); /** * @notice returns the current contract status of the NFT */ function contractStatus() external view returns (ContractStatus); /** * @notice Returns an array with all nft id's and their tier * @dev This function works by filling a pool with random values. When we exhaust the pool, * we refill the pool again with different values. We do it like this because we don't * know in advance how many random values we need. */ function nftTiers() external view returns (uint16[] memory); }
// SPDX-License-Identifier: MIT pragma solidity >=0.8.4 <0.8.6; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title Required interface for a TicketNFT compliant contract * @author Oost & Voort, Inc */ interface ITicketNFT is IERC165 { /** * @dev The following are the stages of the contract in order: * PRE_AIRDROP: Before the airdrop has happened * TICKETS_REVEALED: when accessPassNFT has already set who the TicketWinners will be * AIRDROPPED_TICKETS: when the nfts have been airdropped * SET_REGISTRATION: when the _hasNotRegistered have been filled up * WINNERS_FROZEN: When the winners have been frozen from doing transfers * TRADING_ENABLED: When all trading has been enabled again */ enum ContractStatus { PRE_AIRDROP, TICKETS_REVEALED, AIRDROPPED_TICKETS, SET_REGISTRATION, WINNERS_FROZEN, TRADING_ENABLED } /** * @dev emitted when the owner has changed the contract uri * @param oldURI is the uri it was set as before * @param newURI is the uri it is now set in */ event ContractURISet(string oldURI, string newURI); /** * @dev emitted when the owner changes the royalties * @param newRoyaltyAddress is the new royalty address that will receive the royalties. * @param newRoyalties is the new royalties set by the owner */ event RoyaltiesSet(address newRoyaltyAddress, uint96 newRoyalties); /** * @dev emitted when the frozenPeriod has been set * @param oldTimestamp is the old timestamp for when the frozenPeriod was set * @param newTimestamp is the timestamp for when the frozenPeriod will now correspond as */ event FrozenPeriodSet(uint256 oldTimestamp, uint256 newTimestamp); /** * @dev the following events must be done in order */ /** * @dev emitted when the airdrop happens * @param winners is the ids of winners from AccessPassNFT. See AccessPassNFT's winners function for more information. */ event TicketsAirdropped(uint16[] winners); /** * @dev emitted when the registration has been set * @param hasRegistered is an array boolean that represents if the onwer of that index has registered off-chain */ event RegistrationSet(bool[] hasRegistered); /** * @dev emitted when a random number has been requested from VRF * @param requestId is the id sent back by VRF to keep track of the request */ event RandomWordRequested(uint256 requestId); /** * @dev emitted when a ticket winner has been selected * @param randomWord is used to determine the TicketWinner */ event TicketWinnersSelected(uint256 randomWord); /** * @dev emitted when the trading for winners have been frozen * @param frozenTimestamp is until when trading for winning nfts have been frozen for */ event TicketWinnersFrozen(uint256 frozenTimestamp); /** * @dev reverted with this error when the address being supplied is Zero Address * @param addressName is for whom the Zero Address is being set for */ error ZeroAddress(string addressName); /** * @dev reverted with this error when a view function is being used to look for a nonExistent Token */ error NonExistentToken(); /** * @dev reverted with this error when a function is being called more than once */ error CallingMoreThanOnce(); /** * @dev reverted with this error when a variable has an incorrect value * @param variableName is the name of the variable with an incorrect value */ error IncorrectValue(string variableName); /** * @dev reverted with this error when a string being supplied should not be empty * @param stringName is the name of the string being supplied with an empty value */ error EmptyString(string stringName); /** * @dev reverted with this error when a function being called should not be called with the current Contract Status * @param currentStatus is the contract's current status * @param requiredStatus is the status the current must be in for the function to not revert */ error IncorrectContractStatus(ContractStatus currentStatus, ContractStatus requiredStatus); /** * @dev reverted with this error when transferring a winningId during frozenPeriod * @param tokenId is the id being transferred * @param currentTimestamp is the current block's timestamp * @param requiredTimestamp is the timestamp the block must at least be in */ error TransferringFrozenToken(uint256 tokenId, uint256 currentTimestamp, uint256 requiredTimestamp); /** * @notice airdrops to accessPassNFT winners * @param winners are accessPassNFT winners taken off-chain */ function airdrop( uint16[] calldata winners ) external; /** * @notice requests a random word from VRF to be used for selecting a ticket winner * @dev See https://docs.chain.link/docs/vrf-contracts/#configurations for Chainlink VRF documentation * @param subscriptionId The chainlink subscription id that pays for the call to Chainlink, needs to be setup with ChainLink beforehand * @param gasLane The maximum gas price you are willing to pay for a Chainlink VRF request in wei * @param callbackGasLimit How much gas to use for the callback request. Approximately 139_000 gas is used up solely * by fulfillRandomWords. */ function requestRandomWord( uint64 subscriptionId, bytes32 gasLane, uint32 callbackGasLimit ) external; /** * @notice sets the contract uri * @param uri points to a json file that follows OpenSeas standard (see https://docs.opensea.io/docs/contract-level-metadata) */ function setContractURI(string memory uri) external; /** * @notice sets the royalty fee for the second hand market selling * @param royaltyAddress is the recepient of royalty fees from second hand market. * @param royaltyFee is the fees taken from second-hand selling. This is expressed in a _royaltyFee/1000. * So to do 5% means supplying 50 since 50/1000 is 5% (see ERC2981 function _setDefaultRoyalty(address receiver, uint96 feeNumerator)) */ function setDefaultRoyalty(address royaltyAddress, uint96 royaltyFee) external; /** * @notice sets the ids of the people who have not registered * @dev It is important to do this before requesting a random word. To make it cheaper gas-wise, sending an empty * array signifies that all token owners registered off-chain. An explanation of what the array of hasRegistered looks * like will follow: * if the owner of token id 0 has registered in the array it will show as true, * so [true, ...] * if the owner of token id 1 has not registered in the array it will show as false * so [true, false, ...] * and so on.. * @param hasRegistered_ is an array of boolean that tells if the owner of the id has registered off-chain */ function setRegistered(bool[] calldata hasRegistered_) external; /** * @notice sets the frozenPeriod for when trading winning token ids is disabled * @param frozenPeriod_ is a timestamp for when the ticket winners can start trading again */ function setFrozenPeriod(uint256 frozenPeriod_) external; /** * @notice returns if the token id has registered or not * @param tokenId is the id of the token being queried */ function hasRegistered( uint16 tokenId ) external view returns (bool); /** * @notice Returns if the address owns a winning nft * @param account is the queried address */ function isAccountWinner(address account) external view returns (bool); /** * @notice returns the current contract status of the NFT */ function contractStatus() external view returns (ContractStatus); /** * @notice returns the current supply of the NFT */ function totalSupply() external view returns (uint256); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) pragma solidity ^0.8.0; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/ERC721.sol) pragma solidity ^0.8.0; import "./IERC721.sol"; import "./IERC721Receiver.sol"; import "./extensions/IERC721Metadata.sol"; import "../../utils/Address.sol"; import "../../utils/Context.sol"; import "../../utils/Strings.sol"; import "../../utils/introspection/ERC165.sol"; /** * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including * the Metadata extension, but not including the Enumerable extension, which is available separately as * {ERC721Enumerable}. */ contract ERC721 is Context, ERC165, IERC721, IERC721Metadata { using Address for address; using Strings for uint256; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to owner address mapping(uint256 => address) private _owners; // Mapping owner address to token count mapping(address => uint256) private _balances; // Mapping from token ID to approved address mapping(uint256 => address) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; /** * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection. */ constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) { return interfaceId == type(IERC721).interfaceId || interfaceId == type(IERC721Metadata).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721-balanceOf}. */ function balanceOf(address owner) public view virtual override returns (uint256) { require(owner != address(0), "ERC721: balance query for the zero address"); return _balances[owner]; } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { address owner = _owners[tokenId]; require(owner != address(0), "ERC721: owner query for nonexistent token"); return owner; } /** * @dev See {IERC721Metadata-name}. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev See {IERC721Metadata-symbol}. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token"); string memory baseURI = _baseURI(); return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : ""; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ""; } /** * @dev See {IERC721-approve}. */ function approve(address to, uint256 tokenId) public virtual override { address owner = ERC721.ownerOf(tokenId); require(to != owner, "ERC721: approval to current owner"); require( _msgSender() == owner || isApprovedForAll(owner, _msgSender()), "ERC721: approve caller is not owner nor approved for all" ); _approve(to, tokenId); } /** * @dev See {IERC721-getApproved}. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { require(_exists(tokenId), "ERC721: approved query for nonexistent token"); return _tokenApprovals[tokenId]; } /** * @dev See {IERC721-setApprovalForAll}. */ function setApprovalForAll(address operator, bool approved) public virtual override { _setApprovalForAll(_msgSender(), operator, approved); } /** * @dev See {IERC721-isApprovedForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev See {IERC721-transferFrom}. */ function transferFrom( address from, address to, uint256 tokenId ) public virtual override { //solhint-disable-next-line max-line-length require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved"); _transfer(from, to, tokenId); } /** * @dev See {IERC721-safeTransferFrom}. */ function safeTransferFrom( address from, address to, uint256 tokenId ) public virtual override { safeTransferFrom(from, to, tokenId, ""); } /** * @dev See {IERC721-safeTransferFrom}. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes memory _data ) public virtual override { require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved"); _safeTransfer(from, to, tokenId, _data); } /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * `_data` is additional data, it has no specified format and it is sent in call to `to`. * * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g. * implement alternative mechanisms to perform token transfer, such as signature-based. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeTransfer( address from, address to, uint256 tokenId, bytes memory _data ) internal virtual { _transfer(from, to, tokenId); require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer"); } /** * @dev Returns whether `tokenId` exists. * * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}. * * Tokens start existing when they are minted (`_mint`), * and stop existing when they are burned (`_burn`). */ function _exists(uint256 tokenId) internal view virtual returns (bool) { return _owners[tokenId] != address(0); } /** * @dev Returns whether `spender` is allowed to manage `tokenId`. * * Requirements: * * - `tokenId` must exist. */ function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) { require(_exists(tokenId), "ERC721: operator query for nonexistent token"); address owner = ERC721.ownerOf(tokenId); return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender); } /** * @dev Safely mints `tokenId` and transfers it to `to`. * * Requirements: * * - `tokenId` must not exist. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeMint(address to, uint256 tokenId) internal virtual { _safeMint(to, tokenId, ""); } /** * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is * forwarded in {IERC721Receiver-onERC721Received} to contract recipients. */ function _safeMint( address to, uint256 tokenId, bytes memory _data ) internal virtual { _mint(to, tokenId); require( _checkOnERC721Received(address(0), to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer" ); } /** * @dev Mints `tokenId` and transfers it to `to`. * * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible * * Requirements: * * - `tokenId` must not exist. * - `to` cannot be the zero address. * * Emits a {Transfer} event. */ function _mint(address to, uint256 tokenId) internal virtual { require(to != address(0), "ERC721: mint to the zero address"); require(!_exists(tokenId), "ERC721: token already minted"); _beforeTokenTransfer(address(0), to, tokenId); _balances[to] += 1; _owners[tokenId] = to; emit Transfer(address(0), to, tokenId); _afterTokenTransfer(address(0), to, tokenId); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId) internal virtual { address owner = ERC721.ownerOf(tokenId); _beforeTokenTransfer(owner, address(0), tokenId); // Clear approvals _approve(address(0), tokenId); _balances[owner] -= 1; delete _owners[tokenId]; emit Transfer(owner, address(0), tokenId); _afterTokenTransfer(owner, address(0), tokenId); } /** * @dev Transfers `tokenId` from `from` to `to`. * As opposed to {transferFrom}, this imposes no restrictions on msg.sender. * * Requirements: * * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * * Emits a {Transfer} event. */ function _transfer( address from, address to, uint256 tokenId ) internal virtual { require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner"); require(to != address(0), "ERC721: transfer to the zero address"); _beforeTokenTransfer(from, to, tokenId); // Clear approvals from the previous owner _approve(address(0), tokenId); _balances[from] -= 1; _balances[to] += 1; _owners[tokenId] = to; emit Transfer(from, to, tokenId); _afterTokenTransfer(from, to, tokenId); } /** * @dev Approve `to` to operate on `tokenId` * * Emits a {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 a {ApprovalForAll} event. */ function _setApprovalForAll( address owner, address operator, bool approved ) internal virtual { require(owner != operator, "ERC721: approve to caller"); _operatorApprovals[owner][operator] = approved; emit ApprovalForAll(owner, operator, approved); } /** * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address. * The call is not executed if the target address is not a contract. * * @param from address representing the previous owner of the given token ID * @param to target address that will receive the tokens * @param tokenId uint256 ID of the token to be transferred * @param _data bytes optional data to send along with the call * @return bool whether the call correctly returned the expected magic value */ function _checkOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { if (to.isContract()) { try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) { return retval == IERC721Receiver.onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert("ERC721: transfer to non ERC721Receiver implementer"); } else { assembly { revert(add(32, reason), mload(reason)) } } } } else { return true; } } /** * @dev Hook that is called before any token transfer. This includes minting * and burning. * * Calling conditions: * * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, ``from``'s `tokenId` will be burned. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _beforeTokenTransfer( address from, address to, uint256 tokenId ) internal virtual {} /** * @dev Hook that is called after any transfer of tokens. This includes * minting and burning. * * Calling conditions: * * - when `from` and `to` are both non-zero. * - `from` and `to` are never both zero. * * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks]. */ function _afterTokenTransfer( address from, address to, uint256 tokenId ) internal virtual {} }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (token/common/ERC2981.sol) pragma solidity ^0.8.0; import "../../interfaces/IERC2981.sol"; import "../../utils/introspection/ERC165.sol"; /** * @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: * * - `tokenId` must be already minted. * - `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]; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol) pragma solidity ^0.8.0; import "./IERC165.sol"; /** * @dev Implementation of the {IERC165} interface. * * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check * for the additional interface id that will be supported. For example: * * ```solidity * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { * return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId); * } * ``` * * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation. */ abstract contract ERC165 is IERC165 { /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IERC165).interfaceId; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721.sol) pragma solidity ^0.8.0; import "../../utils/introspection/IERC165.sol"; /** * @dev Required interface of an ERC721 compliant contract. */ interface IERC721 is IERC165 { /** * @dev Emitted when `tokenId` token is transferred from `from` to `to`. */ event Transfer(address indexed from, address indexed to, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token. */ event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId); /** * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets. */ event ApprovalForAll(address indexed owner, address indexed operator, bool approved); /** * @dev Returns the number of tokens in ``owner``'s account. */ function balanceOf(address owner) external view returns (uint256 balance); /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) external view returns (address owner); /** * @dev Safely transfers `tokenId` token from `from` to `to`. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId, bytes calldata data ) external; /** * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients * are aware of the ERC721 protocol to prevent tokens from being forever locked. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must exist and be owned by `from`. * - If the caller is not `from`, it must be have been allowed to move this token by either {approve} or {setApprovalForAll}. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Transfers `tokenId` token from `from` to `to`. * * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible. * * Requirements: * * - `from` cannot be the zero address. * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}. * * Emits a {Transfer} event. */ function transferFrom( address from, address to, uint256 tokenId ) external; /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. * The approval is cleared when the token is transferred. * * Only a single account can be approved at a time, so approving the zero address clears previous approvals. * * Requirements: * * - The caller must own the token or be an approved operator. * - `tokenId` must exist. * * Emits an {Approval} event. */ function approve(address to, uint256 tokenId) external; /** * @dev Approve or remove `operator` as an operator for the caller. * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller. * * Requirements: * * - The `operator` cannot be the caller. * * Emits an {ApprovalForAll} event. */ function setApprovalForAll(address operator, bool _approved) external; /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) external view returns (address operator); /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll} */ function isApprovedForAll(address owner, address operator) external view returns (bool); }
// SPDX-License-Identifier: MIT // 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); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol) pragma solidity ^0.8.0; import "../IERC721.sol"; /** * @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); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity ^0.8.0; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol) pragma solidity ^0.8.0; import "../utils/introspection/IERC165.sol"; /** * @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); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Counters.sol) pragma solidity ^0.8.0; /** * @title Counters * @author Matt Condon (@shrugs) * @dev Provides counters that can only be incremented, decremented or reset. This can be used e.g. to track the number * of elements in a mapping, issuing ERC721 ids, or counting request ids. * * Include with `using Counters for Counters.Counter;` */ library Counters { struct Counter { // This variable should never be directly accessed by users of the library: interactions must be restricted to // the library's function. As of Solidity v0.5.2, this cannot be enforced, though there is a proposal to add // this feature: see https://github.com/ethereum/solidity/issues/4637 uint256 _value; // default: 0 } function current(Counter storage counter) internal view returns (uint256) { return counter._value; } function increment(Counter storage counter) internal { unchecked { counter._value += 1; } } function decrement(Counter storage counter) internal { uint256 value = counter._value; require(value > 0, "Counter: decrement overflow"); unchecked { counter._value = value - 1; } } function reset(Counter storage counter) internal { counter._value = 0; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title An abstract contract that checks if the verified slot is valid * @author Oost & Voort, Inc * @notice This contract is to be used in conjunction with the AccessPassNFT contract */ import "@openzeppelin/contracts/utils/cryptography/ECDSA.sol"; import "./interfaces/IVerifiedSlot.sol"; abstract contract WhitelistVerifier is IVerifiedSlot { using ECDSA for bytes32; /** * @dev The following struct follows the EIP712 Standard */ struct EIP712Domain { string name; string version; uint256 chainId; address verifyingContract; } /** * @dev The typehash for EIP712Domain */ bytes32 constant EIP712DOMAIN_TYPEHASH = keccak256( "EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)" ); /** * @dev The typehash for the message being sent to the contract */ bytes32 constant VERIFIED_SLOT_TYPEHASH = keccak256("VerifiedSlot(address minter,uint256 mintingCapacity)"); /** * @dev The hashed Domain Message */ bytes32 DOMAIN_SEPARATOR; /** * @dev the address of the whiteListSigner which is an EOA that signs a message that confirms who can mint how much */ address public whiteListSigner; /** * @dev emitted when the whitelistSigner has been set * @param oldSigner represents the old signer for the Contract * @param newSigner represents the newly set signer for the Contract */ event WhitelistSignerSet(address oldSigner, address newSigner); /** * @dev reverts with this message when the Zero Address is being used to set the Whitelist Signer */ error WhitelistSignerIsZeroAddress(); /** * @dev reverts with this message when the Caller of the mint is not the same as the one in the VerifiedSLot * @param caller is the account that called for the mint * @param minter is the address specified in the VerifiedSlot */ error CallerIsNotMinter(address caller, address minter); /** * @dev reverts with this message when the message is not correct or if it is not signed by the WhitelistSigner * @param unknownSigner is the signer that signed the message * @param whitelistSigner is the signer who should have signed the message */ error UnknownSigner(address unknownSigner, address whitelistSigner); /** * @dev reverts with this message when the caller is trying to mint more than allowed * @param minted is the amount of tokens the caller has minted already */ error ExceedMintingCapacity(uint256 minted); /** * @notice initializes the contract */ constructor () { DOMAIN_SEPARATOR = hash(EIP712Domain({ name: "AccessPassNFT", version: '1', chainId: block.chainid, verifyingContract: address(this) })); } /** * @notice sets the whitelistSigner * @param whitelistSigner_ is an EOA that signs verified slots */ function _setWhiteListSigner(address whitelistSigner_) internal virtual { if (whitelistSigner_ == address(0)) revert WhitelistSignerIsZeroAddress(); emit WhitelistSignerSet(whiteListSigner, whitelistSigner_); whiteListSigner = whitelistSigner_; } /** * @notice validates verified slot * @param minter is msg.sender * @param minted is the amount the minter has minted * @param verifiedSlot is an object with the following: * minter: address of the minter, * mintingCapacity: amount Metaframes has decided to grant to the minter, * r and s --- The x co-ordinate of r and the s value of the signature * v: The parity of the y co-ordinate of r */ function validateVerifiedSlot( address minter, uint256 minted, VerifiedSlot memory verifiedSlot ) internal view { if (whiteListSigner == address(0)) revert WhitelistSignerIsZeroAddress(); if (verifiedSlot.minter != minter) revert CallerIsNotMinter(minter, verifiedSlot.minter); if(verifiedSlot.mintingCapacity <= minted) revert ExceedMintingCapacity(minted); address wouldBeSigner = getSigner(verifiedSlot); if (wouldBeSigner != whiteListSigner) revert UnknownSigner(wouldBeSigner, whiteListSigner); } /** * @notice hashes the DOMAIN object using keccak256 * @param eip712Domain represents the EIP712 object to be hashed */ function hash(EIP712Domain memory eip712Domain) internal pure returns (bytes32) { return keccak256(abi.encode( EIP712DOMAIN_TYPEHASH, keccak256(bytes(eip712Domain.name)), keccak256(bytes(eip712Domain.version)), eip712Domain.chainId, eip712Domain.verifyingContract )); } /** * @notice hashes the verifiedslot object using keccak256 * @param verifiedSlot is an object with the following: * minter: address of the minter, * mintingCapacity: amount Metaframes has decided to grant to the minter, * r and s --- The x co-ordinate of r and the s value of the signature * v: The parity of the y co-ordinate of r */ function hash(VerifiedSlot memory verifiedSlot) internal pure returns (bytes32) { return keccak256(abi.encode( VERIFIED_SLOT_TYPEHASH, verifiedSlot.minter, verifiedSlot.mintingCapacity )); } /** * @notice returns the signer of a given verifiedSlot to be used to check who signed the message * @param verifiedSlot is an object with the following: * minter: address of the minter, * mintingCapacity: amount Metaframes has decided to grant to the minter, * r and s --- The x co-ordinate of r and the s value of the signature * v: The parity of the y co-ordinate of r */ function getSigner(VerifiedSlot memory verifiedSlot) internal view returns (address) { // Note: we need to use `encodePacked` here instead of `encode`. bytes32 digest = keccak256(abi.encodePacked( "\x19\x01", DOMAIN_SEPARATOR, hash(verifiedSlot) )); return ecrecover(digest, verifiedSlot.v, verifiedSlot.r, verifiedSlot.s); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (utils/cryptography/ECDSA.sol) pragma solidity ^0.8.0; import "../Strings.sol"; /** * @dev Elliptic Curve Digital Signature Algorithm (ECDSA) operations. * * These functions can be used to verify that a message was signed by the holder * of the private keys of a given address. */ library ECDSA { enum RecoverError { NoError, InvalidSignature, InvalidSignatureLength, InvalidSignatureS, InvalidSignatureV } function _throwError(RecoverError error) private pure { if (error == RecoverError.NoError) { return; // no error: do nothing } else if (error == RecoverError.InvalidSignature) { revert("ECDSA: invalid signature"); } else if (error == RecoverError.InvalidSignatureLength) { revert("ECDSA: invalid signature length"); } else if (error == RecoverError.InvalidSignatureS) { revert("ECDSA: invalid signature 's' value"); } else if (error == RecoverError.InvalidSignatureV) { revert("ECDSA: invalid signature 'v' value"); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature` or error string. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. * * Documentation for signature generation: * - with https://web3js.readthedocs.io/en/v1.3.4/web3-eth-accounts.html#sign[Web3.js] * - with https://docs.ethers.io/v5/api/signer/#Signer-signMessage[ethers] * * _Available since v4.3._ */ function tryRecover(bytes32 hash, bytes memory signature) internal pure returns (address, RecoverError) { // Check the signature length // - case 65: r,s,v signature (standard) // - case 64: r,vs signature (cf https://eips.ethereum.org/EIPS/eip-2098) _Available since v4.1._ if (signature.length == 65) { bytes32 r; bytes32 s; uint8 v; // ecrecover takes the signature parameters, and the only way to get them // currently is to use assembly. assembly { r := mload(add(signature, 0x20)) s := mload(add(signature, 0x40)) v := byte(0, mload(add(signature, 0x60))) } return tryRecover(hash, v, r, s); } else if (signature.length == 64) { bytes32 r; bytes32 vs; // ecrecover takes the signature parameters, and the only way to get them // currently is to use assembly. assembly { r := mload(add(signature, 0x20)) vs := mload(add(signature, 0x40)) } return tryRecover(hash, r, vs); } else { return (address(0), RecoverError.InvalidSignatureLength); } } /** * @dev Returns the address that signed a hashed message (`hash`) with * `signature`. This address can then be used for verification purposes. * * The `ecrecover` EVM opcode allows for malleable (non-unique) signatures: * this function rejects them by requiring the `s` value to be in the lower * half order, and the `v` value to be either 27 or 28. * * IMPORTANT: `hash` _must_ be the result of a hash operation for the * verification to be secure: it is possible to craft signatures that * recover to arbitrary addresses for non-hashed data. A safe way to ensure * this is by receiving a hash of the original message (which may otherwise * be too long), and then calling {toEthSignedMessageHash} on it. */ function recover(bytes32 hash, bytes memory signature) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, signature); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `r` and `vs` short-signature fields separately. * * See https://eips.ethereum.org/EIPS/eip-2098[EIP-2098 short signatures] * * _Available since v4.3._ */ function tryRecover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address, RecoverError) { bytes32 s = vs & bytes32(0x7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff); uint8 v = uint8((uint256(vs) >> 255) + 27); return tryRecover(hash, v, r, s); } /** * @dev Overload of {ECDSA-recover} that receives the `r and `vs` short-signature fields separately. * * _Available since v4.2._ */ function recover( bytes32 hash, bytes32 r, bytes32 vs ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, r, vs); _throwError(error); return recovered; } /** * @dev Overload of {ECDSA-tryRecover} that receives the `v`, * `r` and `s` signature fields separately. * * _Available since v4.3._ */ function tryRecover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address, RecoverError) { // EIP-2 still allows signature malleability for ecrecover(). Remove this possibility and make the signature // unique. Appendix F in the Ethereum Yellow paper (https://ethereum.github.io/yellowpaper/paper.pdf), defines // the valid range for s in (301): 0 < s < secp256k1n ÷ 2 + 1, and for v in (302): v ∈ {27, 28}. Most // signatures from current libraries generate a unique signature with an s-value in the lower half order. // // If your library generates malleable signatures, such as s-values in the upper range, calculate a new s-value // with 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141 - s1 and flip v from 27 to 28 or // vice versa. If your library also generates signatures with 0/1 for v instead 27/28, add 27 to v to accept // these malleable signatures as well. if (uint256(s) > 0x7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF5D576E7357A4501DDFE92F46681B20A0) { return (address(0), RecoverError.InvalidSignatureS); } if (v != 27 && v != 28) { return (address(0), RecoverError.InvalidSignatureV); } // If the signature is valid (and not malleable), return the signer address address signer = ecrecover(hash, v, r, s); if (signer == address(0)) { return (address(0), RecoverError.InvalidSignature); } return (signer, RecoverError.NoError); } /** * @dev Overload of {ECDSA-recover} that receives the `v`, * `r` and `s` signature fields separately. */ function recover( bytes32 hash, uint8 v, bytes32 r, bytes32 s ) internal pure returns (address) { (address recovered, RecoverError error) = tryRecover(hash, v, r, s); _throwError(error); return recovered; } /** * @dev Returns an Ethereum Signed Message, created from a `hash`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes32 hash) internal pure returns (bytes32) { // 32 is the length in bytes of hash, // enforced by the type signature above return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n32", hash)); } /** * @dev Returns an Ethereum Signed Message, created from `s`. This * produces hash corresponding to the one signed with the * https://eth.wiki/json-rpc/API#eth_sign[`eth_sign`] * JSON-RPC method as part of EIP-191. * * See {recover}. */ function toEthSignedMessageHash(bytes memory s) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19Ethereum Signed Message:\n", Strings.toString(s.length), s)); } /** * @dev Returns an Ethereum Signed Typed Data, created from a * `domainSeparator` and a `structHash`. This produces hash corresponding * to the one signed with the * https://eips.ethereum.org/EIPS/eip-712[`eth_signTypedData`] * JSON-RPC method as part of EIP-712. * * See {recover}. */ function toTypedDataHash(bytes32 domainSeparator, bytes32 structHash) internal pure returns (bytes32) { return keccak256(abi.encodePacked("\x19\x01", domainSeparator, structHash)); } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; interface IVerifiedSlot { struct VerifiedSlot { address minter; uint16 mintingCapacity; uint8 v; bytes32 r; bytes32 s; } }
{ "optimizer": { "enabled": true, "runs": 1000 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"string","name":"baseURI_","type":"string"},{"internalType":"string","name":"contractURI_","type":"string"},{"internalType":"address","name":"royaltyAddress","type":"address"},{"internalType":"uint96","name":"royaltyFee","type":"uint96"},{"internalType":"contract AccessPassNFT","name":"accessPassNFT_","type":"address"},{"internalType":"address","name":"vrfCoordinator_","type":"address"},{"internalType":"address","name":"nftHolder","type":"address"},{"internalType":"uint256","name":"frozenPeriod_","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"CallingMoreThanOnce","type":"error"},{"inputs":[{"internalType":"string","name":"stringName","type":"string"}],"name":"EmptyString","type":"error"},{"inputs":[{"internalType":"enum ITicketNFT.ContractStatus","name":"currentStatus","type":"uint8"},{"internalType":"enum ITicketNFT.ContractStatus","name":"requiredStatus","type":"uint8"}],"name":"IncorrectContractStatus","type":"error"},{"inputs":[{"internalType":"string","name":"variableName","type":"string"}],"name":"IncorrectValue","type":"error"},{"inputs":[],"name":"NonExistentToken","type":"error"},{"inputs":[{"internalType":"address","name":"have","type":"address"},{"internalType":"address","name":"want","type":"address"}],"name":"OnlyCoordinatorCanFulfill","type":"error"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"currentTimestamp","type":"uint256"},{"internalType":"uint256","name":"requiredTimestamp","type":"uint256"}],"name":"TransferringFrozenToken","type":"error"},{"inputs":[{"internalType":"string","name":"addressName","type":"string"}],"name":"ZeroAddress","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":false,"internalType":"string","name":"oldURI","type":"string"},{"indexed":false,"internalType":"string","name":"newURI","type":"string"}],"name":"ContractURISet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"oldTimestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newTimestamp","type":"uint256"}],"name":"FrozenPeriodSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"requestId","type":"uint256"}],"name":"RandomWordRequested","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bool[]","name":"hasRegistered","type":"bool[]"}],"name":"RegistrationSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"newRoyaltyAddress","type":"address"},{"indexed":false,"internalType":"uint96","name":"newRoyalties","type":"uint96"}],"name":"RoyaltiesSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"frozenTimestamp","type":"uint256"}],"name":"TicketWinnersFrozen","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"randomWord","type":"uint256"}],"name":"TicketWinnersSelected","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint16[]","name":"winners","type":"uint16[]"}],"name":"TicketsAirdropped","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":"NUMBER_OF_WINNERS","outputs":[{"internalType":"uint16","name":"","type":"uint16"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"accessPassNFT","outputs":[{"internalType":"contract AccessPassNFT","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16[]","name":"winners","type":"uint16[]"}],"name":"airdrop","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"chainlinkCoordinator","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"contractStatus","outputs":[{"internalType":"enum ITicketNFT.ContractStatus","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"frozenPeriod","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"tokenId","type":"uint16"}],"name":"hasRegistered","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"hasSetRegistration","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"isAccountWinner","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":"uint256","name":"","type":"uint256"}],"name":"isWinner","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxTotalSupply","outputs":[{"internalType":"uint16","name":"","type":"uint16"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"randomWord","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"requestId","type":"uint256"},{"internalType":"uint256[]","name":"randomWords","type":"uint256[]"}],"name":"rawFulfillRandomWords","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"requestId","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint64","name":"subscriptionId","type":"uint64"},{"internalType":"bytes32","name":"gasLane","type":"bytes32"},{"internalType":"uint32","name":"callbackGasLimit","type":"uint32"}],"name":"requestRandomWord","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":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"_data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"uri","type":"string"}],"name":"setContractURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"royaltyAddress","type":"address"},{"internalType":"uint96","name":"royaltyFee","type":"uint96"}],"name":"setDefaultRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"frozenPeriod_","type":"uint256"}],"name":"setFrozenPeriod","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool[]","name":"hasRegistered_","type":"bool[]"}],"name":"setRegistered","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":"","type":"uint256"}],"name":"ticketWinners","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ticketsAirdropped","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
60c06040523480156200001157600080fd5b506040516200434938038062004349833981016040819052620000349162000b29565b8260405180606001604052806022815260200162004327602291396040805180820190915260048152634f4d574360e01b602082015262000075336200047e565b81516200008a9060039060208501906200098f565b508051620000a09060049060208401906200098f565b50505060601b6001600160601b031916608052604080518082019091526001808252602f60f81b602090920182905289518a92918391620000e2919062000c9b565b815181106200010157634e487b7160e01b600052603260045260246000fd5b01602001517fff0000000000000000000000000000000000000000000000000000000000000016146200016657604051636b6afd1760e11b81526020600482015260076024820152666261736555524960c81b60448201526064015b60405180910390fd5b88516200017b90600d9060208c01906200098f565b508751620001bb576040516318a996bb60e21b815260206004820152600b60248201526a636f6e747261637455524960a81b60448201526064016200015d565b8751620001d09060099060208b01906200098f565b506001600160a01b0387166200021b5760405163eac0d38960e01b815260206004820152600e60248201526d726f79616c74794164647265737360901b60448201526064016200015d565b620002278787620004ce565b6000856001600160a01b0316635d83c2b86040518163ffffffff1660e01b815260040160206040518083038186803b1580156200026357600080fd5b505afa15801562000278573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906200029e919062000c06565b60f081901b6001600160f01b03191660a0526040516301ffc9a760e01b815263b75e1f9f60e01b600482018190529192506001600160a01b038816906301ffc9a79060240160206040518083038186803b158015620002fc57600080fd5b505afa15801562000311573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019062000337919062000ad6565b6200037657604051636b6afd1760e11b815260206004820152600d60248201526c1858d8d95cdcd4185cdcd39195609a1b60448201526064016200015d565b601080546001600160a01b0319166001600160a01b03898116919091179091558616620003d85760405163eac0d38960e01b815260206004820152600e60248201526d3b393321b7b7b93234b730ba37b960911b60448201526064016200015d565b601180546001600160a01b0319166001600160a01b03888116919091179091558516620004355760405163eac0d38960e01b815260206004820152600960248201526837333a2437b63232b960b91b60448201526064016200015d565b60005b8261ffff168110156200046657620004518682620005cf565b806200045d8162000d25565b91505062000438565b505050600a919091555062000d889650505050505050565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b6127106001600160601b03821611156200053e5760405162461bcd60e51b815260206004820152602a60248201527f455243323938313a20726f79616c7479206665652077696c6c206578636565646044820152692073616c65507269636560b01b60648201526084016200015d565b6001600160a01b038216620005965760405162461bcd60e51b815260206004820152601960248201527f455243323938313a20696e76616c69642072656365697665720000000000000060448201526064016200015d565b604080518082019091526001600160a01b039092168083526001600160601b039091166020909201829052600160a01b90910217600155565b620005f1828260405180602001604052806000815250620005f560201b60201c565b5050565b6200060183836200066d565b620006106000848484620007c3565b620006685760405162461bcd60e51b815260206004820152603260248201526000805160206200430783398151915260448201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b60648201526084016200015d565b505050565b6001600160a01b038216620006c55760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f206164647265737360448201526064016200015d565b6000818152600560205260409020546001600160a01b0316156200072c5760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e7465640000000060448201526064016200015d565b6200073a600083836200092c565b6001600160a01b03821660009081526006602052604081208054600192906200076590849062000c80565b909155505060008181526005602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b6000620007e4846001600160a01b03166200098060201b62001c311760201c565b156200092057604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906200081e90339089908890889060040162000c2a565b602060405180830381600087803b1580156200083957600080fd5b505af19250505080156200086c575060408051601f3d908101601f19168201909252620008699181019062000aff565b60015b62000905573d8080156200089d576040519150601f19603f3d011682016040523d82523d6000602084013e620008a2565b606091505b508051620008fd5760405162461bcd60e51b815260206004820152603260248201526000805160206200430783398151915260448201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b60648201526084016200015d565b805181602001fd5b6001600160e01b031916630a85bd0160e11b14905062000924565b5060015b949350505050565b600a54421080156200094c57506000818152600b602052604090205460ff165b156200066857600a54604051630f69d62960e41b81526004810183905242602482015260448101919091526064016200015d565b6001600160a01b03163b151590565b8280546200099d9062000ce8565b90600052602060002090601f016020900481019282620009c1576000855562000a0c565b82601f10620009dc57805160ff191683800117855562000a0c565b8280016001018555821562000a0c579182015b8281111562000a0c578251825591602001919060010190620009ef565b5062000a1a92915062000a1e565b5090565b5b8082111562000a1a576000815560010162000a1f565b805162000a428162000d6f565b919050565b600082601f83011262000a58578081fd5b81516001600160401b038082111562000a755762000a7562000d59565b604051601f8301601f19908116603f0116810190828211818310171562000aa05762000aa062000d59565b8160405283815286602085880101111562000ab9578485fd5b62000acc84602083016020890162000cb5565b9695505050505050565b60006020828403121562000ae8578081fd5b8151801515811462000af8578182fd5b9392505050565b60006020828403121562000b11578081fd5b81516001600160e01b03198116811462000af8578182fd5b600080600080600080600080610100898b03121562000b46578384fd5b88516001600160401b038082111562000b5d578586fd5b62000b6b8c838d0162000a47565b995060208b015191508082111562000b81578586fd5b5062000b908b828c0162000a47565b975050604089015162000ba38162000d6f565b60608a01519096506001600160601b038116811462000bc0578485fd5b945062000bd060808a0162000a35565b935062000be060a08a0162000a35565b925062000bf060c08a0162000a35565b915060e089015190509295985092959890939650565b60006020828403121562000c18578081fd5b815161ffff8116811462000af8578182fd5b600060018060a01b03808716835280861660208401525083604083015260806060830152825180608084015262000c698160a085016020870162000cb5565b601f01601f19169190910160a00195945050505050565b6000821982111562000c965762000c9662000d43565b500190565b60008282101562000cb05762000cb062000d43565b500390565b60005b8381101562000cd257818101518382015260200162000cb8565b8381111562000ce2576000848401525b50505050565b600181811c9082168062000cfd57607f821691505b6020821081141562000d1f57634e487b7160e01b600052602260045260246000fd5b50919050565b600060001982141562000d3c5762000d3c62000d43565b5060010190565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160a01b038116811462000d8557600080fd5b50565b60805160601c60a05160f01c61352262000de5600039600081816103c90152818161044e015281816107630152818161086601528181611140015281816118ce0152612a3e015260008181610e960152610ef101526135226000f3fe608060405234801561001057600080fd5b50600436106102d25760003560e01c80636926726e11610186578063a22cb465116100e3578063d7068bb011610097578063e985e9c511610071578063e985e9c514610603578063f2fde38b1461063f578063f56841831461065257600080fd5b8063d7068bb0146105d5578063e59af732146105e8578063e8a3d485146105fb57600080fd5b8063c6ee20d2116100c8578063c6ee20d21461059a578063c87b56dd146105af578063d01b84ec146105c257600080fd5b8063a22cb46514610574578063b88d4fde1461058757600080fd5b80638da5cb5b1161013a578063938e3d7b1161011f578063938e3d7b1461054c57806395d89b411461055f5780639d7142301461056757600080fd5b80638da5cb5b14610528578063927e603d1461053957600080fd5b806370084da51161016b57806370084da5146104fa57806370a082311461050d578063715018a61461052057600080fd5b80636926726e146104ea5780636c0360eb146104f257600080fd5b80631fe543e31161023457806342842e0e116101e85780635d20df71116101cd5780635d20df71146104b25780635f24c872146104c55780636352211e146104d757600080fd5b806342842e0e1461048c5780635cd6d3111461049f57600080fd5b80632a55205a116102195780632a55205a146104175780632ab4d052146104495780632c77ddfd1461048357600080fd5b80631fe543e3146103f157806323b872dd1461040457600080fd5b8063081812fc1161028b578063095ea7b311610270578063095ea7b3146103915780630e392288146103a457806318160ddd146103c757600080fd5b8063081812fc1461035357806308e0ae991461037e57600080fd5b806301ffc9a7116102bc57806301ffc9a71461030857806304634d8d1461032b57806306fdde031461033e57600080fd5b80626d6cae146102d757806301e362bf146102f3575b600080fd5b6102e060125481565b6040519081526020015b60405180910390f35b610306610301366004612e02565b61065b565b005b61031b610316366004612e7a565b61092c565b60405190151581526020016102ea565b610306610339366004612dc5565b610970565b610346610a97565b6040516102ea919061323f565b610366610361366004612f12565b610b29565b6040516001600160a01b0390911681526020016102ea565b61030661038c366004612f12565b610bbe565b61030661039f366004612d9a565b610d3b565b61031b6103b2366004612f12565b600b6020526000908152604090205460ff1681565b7f000000000000000000000000000000000000000000000000000000000000000061ffff166102e0565b6103066103ff366004612f42565b610e8b565b610306610412366004612cb0565b610f2c565b61042a610425366004612ff6565b610fb3565b604080516001600160a01b0390931683526020830191909152016102ea565b6104707f000000000000000000000000000000000000000000000000000000000000000081565b60405161ffff90911681526020016102ea565b6102e0600a5481565b61030661049a366004612cb0565b611070565b6103066104ad366004612e02565b61108b565b6102e06104c0366004612f12565b6112d0565b600e5461031b90610100900460ff1681565b6103666104e5366004612f12565b6112f1565b610470600281565b61034661137c565b601054610366906001600160a01b031681565b6102e061051b366004612c39565b61140a565b6103066114a4565b6000546001600160a01b0316610366565b61031b610547366004612c39565b61150a565b61030661055a366004612eb2565b611591565b6103466116a0565b600e5461031b9060ff1681565b610306610582366004612d6d565b6116af565b610306610595366004612cf0565b6116ba565b6105a2611748565b6040516102ea9190613216565b6103466105bd366004612f12565b611828565b61031b6105d0366004612ef8565b611892565b6103066105e3366004613017565b611935565b601154610366906001600160a01b031681565b610346611b42565b61031b610611366004612c78565b6001600160a01b03918216600090815260086020908152604080832093909416825291909152205460ff1690565b61030661064d366004612c39565b611b4f565b6102e060135481565b6000546001600160a01b031633146106ba5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064015b60405180910390fd5b60026106c4611748565b60058111156106e357634e487b7160e01b600052602160045260246000fd5b81600581111561070357634e487b7160e01b600052602160045260246000fd5b1461074557610710611748565b816040517ff975326c0000000000000000000000000000000000000000000000000000000081526004016106b1929190613224565b8161075e57600e805461ff0019166101001790556108ee565b61ffff7f00000000000000000000000000000000000000000000000000000000000000001682146107d257604051636b6afd1760e11b815260206004820152600d60248201527f686173526567697374657265640000000000000000000000000000000000000060448201526064016106b1565b6000805b61ffff811684111561085d5784848261ffff1681811061080657634e487b7160e01b600052603260045260246000fd5b905060200201602081019061081b9190612e42565b61084b5761ffff81166000908152600f60205260409020805460ff1916600117905581610847816133ff565b9250505b80610855816133ff565b9150506107d6565b50600261088a827f0000000000000000000000000000000000000000000000000000000000000000613364565b61ffff1610156108dd57604051636b6afd1760e11b815260206004820152601460248201527f6e6f7452656769737465726564436f756e74657200000000000000000000000060448201526064016106b1565b50600e805461ff0019166101001790555b7fd6d49fea2b82ccb15931df3db5db7b18cf21a3ce0d589e7f416924cdcf5d8786838360405161091f92919061319e565b60405180910390a1505050565b60006001600160e01b031982167f89a32daf00000000000000000000000000000000000000000000000000000000148061096a575061096a82611c40565b92915050565b6000546001600160a01b031633146109ca5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b6001600160a01b038216610a3a576040517feac0d38900000000000000000000000000000000000000000000000000000000815260206004820152600e60248201527f726f79616c74794164647265737300000000000000000000000000000000000060448201526064016106b1565b610a448282611c4b565b604080516001600160a01b03841681526bffffffffffffffffffffffff831660208201527f8a0eef20ed65c9cc65bee4dba772db9344af8922e098f35e280b99494f859163910160405180910390a15050565b606060038054610aa6906133ca565b80601f0160208091040260200160405190810160405280929190818152602001828054610ad2906133ca565b8015610b1f5780601f10610af457610100808354040283529160200191610b1f565b820191906000526020600020905b815481529060010190602001808311610b0257829003601f168201915b5050505050905090565b6000818152600560205260408120546001600160a01b0316610ba25760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084016106b1565b506000908152600760205260409020546001600160a01b031690565b6000546001600160a01b03163314610c185760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b60046000610c27600183613387565b6005811115610c4657634e487b7160e01b600052602160045260246000fd5b9050816005811115610c6857634e487b7160e01b600052602160045260246000fd5b610c70611748565b6005811115610c8f57634e487b7160e01b600052602160045260246000fd5b10610c9c57610710611748565b4283108015610caa57508215155b15610cf857604051636b6afd1760e11b815260206004820152600d60248201527f66726f7a656e506572696f645f0000000000000000000000000000000000000060448201526064016106b1565b600a5460408051918252602082018590527fca3a479dbfc39910d316dad70bca2decb3bdf89328b9549ce05650a850f7f5f6910160405180910390a15050600a55565b6000610d46826112f1565b9050806001600160a01b0316836001600160a01b03161415610dd05760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e6560448201527f720000000000000000000000000000000000000000000000000000000000000060648201526084016106b1565b336001600160a01b0382161480610e0a57506001600160a01b038116600090815260086020908152604080832033845290915290205460ff165b610e7c5760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c000000000000000060648201526084016106b1565b610e868383611d65565b505050565b336001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001614610f1e576040517f1cf993f40000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b037f00000000000000000000000000000000000000000000000000000000000000001660248201526044016106b1565b610f288282611de0565b5050565b610f363382611f04565b610fa85760405162461bcd60e51b815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016106b1565b610e86838383611ffb565b60008281526002602090815260408083208151808301909252546001600160a01b038116808352600160a01b9091046bffffffffffffffffffffffff169282019290925282916110325750604080518082019091526001546001600160a01b0381168252600160a01b90046bffffffffffffffffffffffff1660208201525b602081015160009061271090611056906bffffffffffffffffffffffff1687613345565b6110609190613331565b91519350909150505b9250929050565b610e86838383604051806020016040528060008152506116ba565b6000546001600160a01b031633146110e55760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b60016110ef611748565b600581111561110e57634e487b7160e01b600052602160045260246000fd5b81600581111561112e57634e487b7160e01b600052602160045260246000fd5b1461113b57610710611748565b61ffff7f00000000000000000000000000000000000000000000000000000000000000001682146111af57604051636b6afd1760e11b815260206004820152600760248201527f77696e6e6572730000000000000000000000000000000000000000000000000060448201526064016106b1565b60005b828110156112905760105461127e9033906001600160a01b0316636352211e8787868181106111f157634e487b7160e01b600052603260045260246000fd5b90506020020160208101906112069190612ef8565b6040516001600160e01b031960e084901b16815261ffff909116600482015260240160206040518083038186803b15801561124057600080fd5b505afa158015611254573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112789190612c5c565b83611070565b8061128881613421565b9150506111b2565b50600e805460ff191660011790556040517f4537f390251a569ded7118789a782a771e102b70a50817f5d58fb1fe5bf126899061091f90859085906131da565b600c81815481106112e057600080fd5b600091825260209091200154905081565b6000818152600560205260408120546001600160a01b03168061096a5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201527f656e7420746f6b656e000000000000000000000000000000000000000000000060648201526084016106b1565b600d8054611389906133ca565b80601f01602080910402602001604051908101604052809291908181526020018280546113b5906133ca565b80156114025780601f106113d757610100808354040283529160200191611402565b820191906000526020600020905b8154815290600101906020018083116113e557829003601f168201915b505050505081565b60006001600160a01b0382166114885760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a6560448201527f726f20616464726573730000000000000000000000000000000000000000000060648201526084016106b1565b506001600160a01b031660009081526006602052604090205490565b6000546001600160a01b031633146114fe5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b61150860006121e0565b565b6000805b600c5461ffff8216101561158857826001600160a01b031661155e600c8361ffff168154811061154e57634e487b7160e01b600052603260045260246000fd5b90600052602060002001546112f1565b6001600160a01b031614156115765750600192915050565b80611580816133ff565b91505061150e565b50600092915050565b6000546001600160a01b031633146115eb5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b8051611653576040517f62a65aec00000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f636f6e747261637455524900000000000000000000000000000000000000000060448201526064016106b1565b7fd144c5914915e649557dfce6d3199a18bae1ec7c249c1766603a2b76ade11c06600982604051611685929190613252565b60405180910390a18051610f28906009906020840190612aee565b606060048054610aa6906133ca565b610f2833838361223d565b6116c43383611f04565b6117365760405162461bcd60e51b815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016106b1565b6117428484848461230c565b50505050565b600060135460001461176a57600a544210156117645750600490565b50600590565b600e54610100900460ff16156117805750600390565b600e5460ff16156117915750600290565b601060009054906101000a90046001600160a01b03166001600160a01b0316633f3ac92c6040518163ffffffff1660e01b815260040160206040518083038186803b1580156117df57600080fd5b505afa1580156117f3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118179190612e5e565b156118225750600190565b50600090565b6000818152600560205260409020546060906001600160a01b031661186057604051634a1850bf60e11b815260040160405180910390fd5b600d61186b83612395565b60405160200161187c9291906130ba565b6040516020818303038152906040529050919050565b600060038061189f611748565b60058111156118be57634e487b7160e01b600052602160045260246000fd5b10156118cc57610710611748565b7f000000000000000000000000000000000000000000000000000000000000000061ffff168361ffff161061191457604051634a1850bf60e11b815260040160405180910390fd5b61ffff83166000908152600f602052604090205460ff161591505b50919050565b6000546001600160a01b0316331461198f5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b6003611999611748565b60058111156119b857634e487b7160e01b600052602160045260246000fd5b8160058111156119d857634e487b7160e01b600052602160045260246000fd5b146119e557610710611748565b620249f08263ffffffff161015611a3f57604051636b6afd1760e11b815260206004820152601060248201527f63616c6c6261636b4761734c696d69740000000000000000000000000000000060448201526064016106b1565b6011546040517f5d3b1d300000000000000000000000000000000000000000000000000000000081526004810185905267ffffffffffffffff861660248201526003604482015263ffffffff84166064820152600160848201526001600160a01b03909116908190635d3b1d309060a401602060405180830381600087803b158015611aca57600080fd5b505af1158015611ade573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b029190612f2a565b60128190556040519081527f34035302829687f51841b2858b7a87f37db58335690aec14aced0df98ad7f590906020015b60405180910390a15050505050565b60098054611389906133ca565b6000546001600160a01b03163314611ba95760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b6001600160a01b038116611c255760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016106b1565b611c2e816121e0565b50565b6001600160a01b03163b151590565b600061096a826124e3565b6127106bffffffffffffffffffffffff82161115611cd15760405162461bcd60e51b815260206004820152602a60248201527f455243323938313a20726f79616c7479206665652077696c6c2065786365656460448201527f2073616c6550726963650000000000000000000000000000000000000000000060648201526084016106b1565b6001600160a01b038216611d275760405162461bcd60e51b815260206004820152601960248201527f455243323938313a20696e76616c69642072656365697665720000000000000060448201526064016106b1565b604080518082019091526001600160a01b039092168083526bffffffffffffffffffffffff9091166020909201829052600160a01b90910217600155565b6000818152600760205260409020805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0384169081179091558190611da7826112f1565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b8160125414611e3257604051636b6afd1760e11b815260206004820152600960248201527f726571756573744964000000000000000000000000000000000000000000000060448201526064016106b1565b60135415611e6c576040517fba4ace8d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b80600081518110611e8d57634e487b7160e01b600052603260045260246000fd5b60200260200101516013819055507f3d870ce8fb4499f2050897f38d3d9d5a938b7710a6a5e1d712db97db55dfbb9d81600081518110611edd57634e487b7160e01b600052603260045260246000fd5b6020026020010151604051611ef491815260200190565b60405180910390a1610f28612555565b6000818152600560205260408120546001600160a01b0316611f7d5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084016106b1565b6000611f88836112f1565b9050806001600160a01b0316846001600160a01b03161480611fcf57506001600160a01b0380821660009081526008602090815260408083209388168352929052205460ff165b80611ff35750836001600160a01b0316611fe884610b29565b6001600160a01b0316145b949350505050565b826001600160a01b031661200e826112f1565b6001600160a01b03161461208a5760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201527f6f776e657200000000000000000000000000000000000000000000000000000060648201526084016106b1565b6001600160a01b0382166121055760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f2061646460448201527f726573730000000000000000000000000000000000000000000000000000000060648201526084016106b1565b612110838383612704565b61211b600082611d65565b6001600160a01b0383166000908152600660205260408120805460019290612144908490613387565b90915550506001600160a01b0382166000908152600660205260408120805460019290612172908490613319565b9091555050600081815260056020526040808220805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600080546001600160a01b0383811673ffffffffffffffffffffffffffffffffffffffff19831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b816001600160a01b0316836001600160a01b0316141561229f5760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c65720000000000000060448201526064016106b1565b6001600160a01b03838116600081815260086020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b612317848484611ffb565b6123238484848461276e565b6117425760405162461bcd60e51b815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016106b1565b6060816123d557505060408051808201909152600181527f3000000000000000000000000000000000000000000000000000000000000000602082015290565b8160005b81156123ff57806123e981613421565b91506123f89050600a83613331565b91506123d9565b60008167ffffffffffffffff81111561242857634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f191660200182016040528015612452576020820181803683370190505b5090505b8415611ff357612467600183613387565b9150612474600a8661345d565b61247f906030613319565b60f81b8183815181106124a257634e487b7160e01b600052603260045260246000fd5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a9053506124dc600a86613331565b9450612456565b60006001600160e01b031982167f80ac58cd00000000000000000000000000000000000000000000000000000000148061254657506001600160e01b031982167f5b5e139f00000000000000000000000000000000000000000000000000000000145b8061096a575061096a826128d1565b601060008061256f838261256881613421565b9450612938565b9050600080805b600261ffff821610156126d057838361258e81613421565b9450815181106125ae57634e487b7160e01b600052603260045260246000fd5b60200260200101519150858314156125dc576125d586866125ce81613421565b9750612938565b9350600092505b61ffff82166000908152600f602052604090205460ff1680612611575061ffff82166000908152600b602052604090205460ff165b1561266d57838361262181613421565b94508151811061264157634e487b7160e01b600052603260045260246000fd5b60200260200101519150858314156126685761266186866125ce81613421565b9350600092505b6125dc565b600c8054600181810190925561ffff84167fdf6966c971051c3d54ec59162606531493a51404a002842f56009d7e5cf4a8c79091018190556000908152600b60205260409020805460ff19169091179055806126c8816133ff565b915050612576565b507fecfc5e59150509bdb8e40e673268957cf1c290eddbdf40be1d1973c84bd7cde0600a54604051611b3391815260200190565b600a544210801561272357506000818152600b602052604090205460ff165b15610e8657600a546040517ff69d62900000000000000000000000000000000000000000000000000000000081526004810183905242602482015260448101919091526064016106b1565b60006001600160a01b0384163b156128c657604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906127b2903390899088908890600401613162565b602060405180830381600087803b1580156127cc57600080fd5b505af19250505080156127fc575060408051601f3d908101601f191682019092526127f991810190612e96565b60015b6128ac573d80801561282a576040519150601f19603f3d011682016040523d82523d6000602084013e61282f565b606091505b5080516128a45760405162461bcd60e51b815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016106b1565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611ff3565b506001949350505050565b60006001600160e01b031982167f2a55205a00000000000000000000000000000000000000000000000000000000148061096a57507f01ffc9a7000000000000000000000000000000000000000000000000000000006001600160e01b031983161461096a565b606061ffff600061294a601086613331565b9050600061295960108761345d565b111561296d5761296a600182613319565b90505b60006129798286613345565b612984906010613345565b90506000612993836010613345565b67ffffffffffffffff8111156129b957634e487b7160e01b600052604160045260246000fd5b6040519080825280602002602001820160405280156129e2578160200160208202803683370190505b50905060005b83811015612ae357601354600090612a008584613319565b6040805160208101939093528201526060016040516020818303038152906040528051906020012060001c905060005b6010811015612ace57612a657f000000000000000000000000000000000000000000000000000000000000000083891661343c565b8482612a72866010613345565b612a7c9190613319565b81518110612a9a57634e487b7160e01b600052603260045260246000fd5b61ffff90921660209283029190910190910152612aba6201000083613331565b915080612ac681613421565b915050612a30565b50508080612adb90613421565b9150506129e8565b509695505050505050565b828054612afa906133ca565b90600052602060002090601f016020900481019282612b1c5760008555612b62565b82601f10612b3557805160ff1916838001178555612b62565b82800160010185558215612b62579182015b82811115612b62578251825591602001919060010190612b47565b50612b6e929150612b72565b5090565b5b80821115612b6e5760008155600101612b73565b600067ffffffffffffffff831115612ba157612ba161349d565b612bb4601f8401601f19166020016132e8565b9050828152838383011115612bc857600080fd5b828260208301376000602084830101529392505050565b60008083601f840112612bf0578182fd5b50813567ffffffffffffffff811115612c07578182fd5b6020830191508360208260051b850101111561106957600080fd5b803561ffff81168114612c3457600080fd5b919050565b600060208284031215612c4a578081fd5b8135612c55816134b3565b9392505050565b600060208284031215612c6d578081fd5b8151612c55816134b3565b60008060408385031215612c8a578081fd5b8235612c95816134b3565b91506020830135612ca5816134b3565b809150509250929050565b600080600060608486031215612cc4578081fd5b8335612ccf816134b3565b92506020840135612cdf816134b3565b929592945050506040919091013590565b60008060008060808587031215612d05578081fd5b8435612d10816134b3565b93506020850135612d20816134b3565b925060408501359150606085013567ffffffffffffffff811115612d42578182fd5b8501601f81018713612d52578182fd5b612d6187823560208401612b87565b91505092959194509250565b60008060408385031215612d7f578182fd5b8235612d8a816134b3565b91506020830135612ca5816134c8565b60008060408385031215612dac578182fd5b8235612db7816134b3565b946020939093013593505050565b60008060408385031215612dd7578182fd5b8235612de2816134b3565b915060208301356bffffffffffffffffffffffff81168114612ca5578182fd5b60008060208385031215612e14578182fd5b823567ffffffffffffffff811115612e2a578283fd5b612e3685828601612bdf565b90969095509350505050565b600060208284031215612e53578081fd5b8135612c55816134c8565b600060208284031215612e6f578081fd5b8151612c55816134c8565b600060208284031215612e8b578081fd5b8135612c55816134d6565b600060208284031215612ea7578081fd5b8151612c55816134d6565b600060208284031215612ec3578081fd5b813567ffffffffffffffff811115612ed9578182fd5b8201601f81018413612ee9578182fd5b611ff384823560208401612b87565b600060208284031215612f09578081fd5b612c5582612c22565b600060208284031215612f23578081fd5b5035919050565b600060208284031215612f3b578081fd5b5051919050565b60008060408385031215612f54578182fd5b8235915060208084013567ffffffffffffffff80821115612f73578384fd5b818601915086601f830112612f86578384fd5b813581811115612f9857612f9861349d565b8060051b9150612fa98483016132e8565b8181528481019084860184860187018b1015612fc3578788fd5b8795505b83861015612fe5578035835260019590950194918601918601612fc7565b508096505050505050509250929050565b60008060408385031215613008578182fd5b50508035926020909101359150565b60008060006060848603121561302b578081fd5b833567ffffffffffffffff81168114613042578182fd5b925060208401359150604084013563ffffffff81168114613061578182fd5b809150509250925092565b6000815180845261308481602086016020860161339e565b601f01601f19169290920160200192915050565b600681106130b657634e487b7160e01b600052602160045260246000fd5b9052565b60008084546130c8816133ca565b600182811680156130e057600181146130f15761311d565b60ff1984168752828701945061311d565b8886526020808720875b858110156131145781548a8201529084019082016130fb565b50505082870194505b50505050835161313181836020880161339e565b7f2e6a736f6e0000000000000000000000000000000000000000000000000000009101908152600501949350505050565b60006001600160a01b03808716835280861660208401525083604083015260806060830152613194608083018461306c565b9695505050505050565b60208082528181018390526000908460408401835b86811015612ae35782356131c6816134c8565b1515825291830191908301906001016131b3565b60208082528181018390526000908460408401835b86811015612ae35761ffff61320384612c22565b16825291830191908301906001016131ef565b6020810161096a8284613098565b604081016132328285613098565b612c556020830184613098565b602081526000612c55602083018461306c565b604081526000808454613264816133ca565b8060408601526060600180841660008114613286576001811461329a576132c8565b60ff198516888401526080880195506132c8565b8987526020808820885b868110156132bf5781548b82018701529084019082016132a4565b8a018501975050505b505050505082810360208401526132df818561306c565b95945050505050565b604051601f8201601f1916810167ffffffffffffffff811182821017156133115761331161349d565b604052919050565b6000821982111561332c5761332c613471565b500190565b60008261334057613340613487565b500490565b600081600019048311821515161561335f5761335f613471565b500290565b600061ffff8381169083168181101561337f5761337f613471565b039392505050565b60008282101561339957613399613471565b500390565b60005b838110156133b95781810151838201526020016133a1565b838111156117425750506000910152565b600181811c908216806133de57607f821691505b6020821081141561192f57634e487b7160e01b600052602260045260246000fd5b600061ffff8083168181141561341757613417613471565b6001019392505050565b600060001982141561343557613435613471565b5060010190565b600061ffff8084168061345157613451613487565b92169190910692915050565b60008261346c5761346c613487565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160a01b0381168114611c2e57600080fd5b8015158114611c2e57600080fd5b6001600160e01b031981168114611c2e57600080fdfea2646970667358221220f2034f1e229b88831cb52007ccca0a46d48362bac9aaae242edb2a9ac942c27c64736f6c634300080400334552433732313a207472616e7366657220746f206e6f6e2045524337323152654d617261646f6e61204f6666696369616c20576f726c6420437570205469636b657400000000000000000000000000000000000000000000000000000000000001000000000000000000000000000000000000000000000000000000000000000160000000000000000000000000455ad5835b5f732a8c28d0ad79abf7942f5869060000000000000000000000000000000000000000000000000000000000000352000000000000000000000000ac1dc28708b2ff61a5d89a74fc09feed065692df000000000000000000000000271682deb8c4e0901d1a1550ad2e64d568e69909000000000000000000000000455ad5835b5f732a8c28d0ad79abf7942f5869060000000000000000000000000000000000000000000000000000000063b0cd000000000000000000000000000000000000000000000000000000000000000036697066733a2f2f516d63544d58597a79524145326f4258537634436a4752774742616545764e796f44707173734b7436425155346d2f00000000000000000000000000000000000000000000000000000000000000000000000000000000005068747470733a2f2f73332e65752d776573742d322e616d617a6f6e6177732e636f6d2f6d617261646f6e616f6666696369616c2e696f2f7469636b65744e4654436f6e74726163745552492e6a736f6e00000000000000000000000000000000
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106102d25760003560e01c80636926726e11610186578063a22cb465116100e3578063d7068bb011610097578063e985e9c511610071578063e985e9c514610603578063f2fde38b1461063f578063f56841831461065257600080fd5b8063d7068bb0146105d5578063e59af732146105e8578063e8a3d485146105fb57600080fd5b8063c6ee20d2116100c8578063c6ee20d21461059a578063c87b56dd146105af578063d01b84ec146105c257600080fd5b8063a22cb46514610574578063b88d4fde1461058757600080fd5b80638da5cb5b1161013a578063938e3d7b1161011f578063938e3d7b1461054c57806395d89b411461055f5780639d7142301461056757600080fd5b80638da5cb5b14610528578063927e603d1461053957600080fd5b806370084da51161016b57806370084da5146104fa57806370a082311461050d578063715018a61461052057600080fd5b80636926726e146104ea5780636c0360eb146104f257600080fd5b80631fe543e31161023457806342842e0e116101e85780635d20df71116101cd5780635d20df71146104b25780635f24c872146104c55780636352211e146104d757600080fd5b806342842e0e1461048c5780635cd6d3111461049f57600080fd5b80632a55205a116102195780632a55205a146104175780632ab4d052146104495780632c77ddfd1461048357600080fd5b80631fe543e3146103f157806323b872dd1461040457600080fd5b8063081812fc1161028b578063095ea7b311610270578063095ea7b3146103915780630e392288146103a457806318160ddd146103c757600080fd5b8063081812fc1461035357806308e0ae991461037e57600080fd5b806301ffc9a7116102bc57806301ffc9a71461030857806304634d8d1461032b57806306fdde031461033e57600080fd5b80626d6cae146102d757806301e362bf146102f3575b600080fd5b6102e060125481565b6040519081526020015b60405180910390f35b610306610301366004612e02565b61065b565b005b61031b610316366004612e7a565b61092c565b60405190151581526020016102ea565b610306610339366004612dc5565b610970565b610346610a97565b6040516102ea919061323f565b610366610361366004612f12565b610b29565b6040516001600160a01b0390911681526020016102ea565b61030661038c366004612f12565b610bbe565b61030661039f366004612d9a565b610d3b565b61031b6103b2366004612f12565b600b6020526000908152604090205460ff1681565b7f00000000000000000000000000000000000000000000000000000000000001f461ffff166102e0565b6103066103ff366004612f42565b610e8b565b610306610412366004612cb0565b610f2c565b61042a610425366004612ff6565b610fb3565b604080516001600160a01b0390931683526020830191909152016102ea565b6104707f00000000000000000000000000000000000000000000000000000000000001f481565b60405161ffff90911681526020016102ea565b6102e0600a5481565b61030661049a366004612cb0565b611070565b6103066104ad366004612e02565b61108b565b6102e06104c0366004612f12565b6112d0565b600e5461031b90610100900460ff1681565b6103666104e5366004612f12565b6112f1565b610470600281565b61034661137c565b601054610366906001600160a01b031681565b6102e061051b366004612c39565b61140a565b6103066114a4565b6000546001600160a01b0316610366565b61031b610547366004612c39565b61150a565b61030661055a366004612eb2565b611591565b6103466116a0565b600e5461031b9060ff1681565b610306610582366004612d6d565b6116af565b610306610595366004612cf0565b6116ba565b6105a2611748565b6040516102ea9190613216565b6103466105bd366004612f12565b611828565b61031b6105d0366004612ef8565b611892565b6103066105e3366004613017565b611935565b601154610366906001600160a01b031681565b610346611b42565b61031b610611366004612c78565b6001600160a01b03918216600090815260086020908152604080832093909416825291909152205460ff1690565b61030661064d366004612c39565b611b4f565b6102e060135481565b6000546001600160a01b031633146106ba5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064015b60405180910390fd5b60026106c4611748565b60058111156106e357634e487b7160e01b600052602160045260246000fd5b81600581111561070357634e487b7160e01b600052602160045260246000fd5b1461074557610710611748565b816040517ff975326c0000000000000000000000000000000000000000000000000000000081526004016106b1929190613224565b8161075e57600e805461ff0019166101001790556108ee565b61ffff7f00000000000000000000000000000000000000000000000000000000000001f41682146107d257604051636b6afd1760e11b815260206004820152600d60248201527f686173526567697374657265640000000000000000000000000000000000000060448201526064016106b1565b6000805b61ffff811684111561085d5784848261ffff1681811061080657634e487b7160e01b600052603260045260246000fd5b905060200201602081019061081b9190612e42565b61084b5761ffff81166000908152600f60205260409020805460ff1916600117905581610847816133ff565b9250505b80610855816133ff565b9150506107d6565b50600261088a827f00000000000000000000000000000000000000000000000000000000000001f4613364565b61ffff1610156108dd57604051636b6afd1760e11b815260206004820152601460248201527f6e6f7452656769737465726564436f756e74657200000000000000000000000060448201526064016106b1565b50600e805461ff0019166101001790555b7fd6d49fea2b82ccb15931df3db5db7b18cf21a3ce0d589e7f416924cdcf5d8786838360405161091f92919061319e565b60405180910390a1505050565b60006001600160e01b031982167f89a32daf00000000000000000000000000000000000000000000000000000000148061096a575061096a82611c40565b92915050565b6000546001600160a01b031633146109ca5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b6001600160a01b038216610a3a576040517feac0d38900000000000000000000000000000000000000000000000000000000815260206004820152600e60248201527f726f79616c74794164647265737300000000000000000000000000000000000060448201526064016106b1565b610a448282611c4b565b604080516001600160a01b03841681526bffffffffffffffffffffffff831660208201527f8a0eef20ed65c9cc65bee4dba772db9344af8922e098f35e280b99494f859163910160405180910390a15050565b606060038054610aa6906133ca565b80601f0160208091040260200160405190810160405280929190818152602001828054610ad2906133ca565b8015610b1f5780601f10610af457610100808354040283529160200191610b1f565b820191906000526020600020905b815481529060010190602001808311610b0257829003601f168201915b5050505050905090565b6000818152600560205260408120546001600160a01b0316610ba25760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a20617070726f76656420717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084016106b1565b506000908152600760205260409020546001600160a01b031690565b6000546001600160a01b03163314610c185760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b60046000610c27600183613387565b6005811115610c4657634e487b7160e01b600052602160045260246000fd5b9050816005811115610c6857634e487b7160e01b600052602160045260246000fd5b610c70611748565b6005811115610c8f57634e487b7160e01b600052602160045260246000fd5b10610c9c57610710611748565b4283108015610caa57508215155b15610cf857604051636b6afd1760e11b815260206004820152600d60248201527f66726f7a656e506572696f645f0000000000000000000000000000000000000060448201526064016106b1565b600a5460408051918252602082018590527fca3a479dbfc39910d316dad70bca2decb3bdf89328b9549ce05650a850f7f5f6910160405180910390a15050600a55565b6000610d46826112f1565b9050806001600160a01b0316836001600160a01b03161415610dd05760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e6560448201527f720000000000000000000000000000000000000000000000000000000000000060648201526084016106b1565b336001600160a01b0382161480610e0a57506001600160a01b038116600090815260086020908152604080832033845290915290205460ff165b610e7c5760405162461bcd60e51b815260206004820152603860248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f74206f7760448201527f6e6572206e6f7220617070726f76656420666f7220616c6c000000000000000060648201526084016106b1565b610e868383611d65565b505050565b336001600160a01b037f000000000000000000000000271682deb8c4e0901d1a1550ad2e64d568e699091614610f1e576040517f1cf993f40000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b037f000000000000000000000000271682deb8c4e0901d1a1550ad2e64d568e699091660248201526044016106b1565b610f288282611de0565b5050565b610f363382611f04565b610fa85760405162461bcd60e51b815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016106b1565b610e86838383611ffb565b60008281526002602090815260408083208151808301909252546001600160a01b038116808352600160a01b9091046bffffffffffffffffffffffff169282019290925282916110325750604080518082019091526001546001600160a01b0381168252600160a01b90046bffffffffffffffffffffffff1660208201525b602081015160009061271090611056906bffffffffffffffffffffffff1687613345565b6110609190613331565b91519350909150505b9250929050565b610e86838383604051806020016040528060008152506116ba565b6000546001600160a01b031633146110e55760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b60016110ef611748565b600581111561110e57634e487b7160e01b600052602160045260246000fd5b81600581111561112e57634e487b7160e01b600052602160045260246000fd5b1461113b57610710611748565b61ffff7f00000000000000000000000000000000000000000000000000000000000001f41682146111af57604051636b6afd1760e11b815260206004820152600760248201527f77696e6e6572730000000000000000000000000000000000000000000000000060448201526064016106b1565b60005b828110156112905760105461127e9033906001600160a01b0316636352211e8787868181106111f157634e487b7160e01b600052603260045260246000fd5b90506020020160208101906112069190612ef8565b6040516001600160e01b031960e084901b16815261ffff909116600482015260240160206040518083038186803b15801561124057600080fd5b505afa158015611254573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112789190612c5c565b83611070565b8061128881613421565b9150506111b2565b50600e805460ff191660011790556040517f4537f390251a569ded7118789a782a771e102b70a50817f5d58fb1fe5bf126899061091f90859085906131da565b600c81815481106112e057600080fd5b600091825260209091200154905081565b6000818152600560205260408120546001600160a01b03168061096a5760405162461bcd60e51b815260206004820152602960248201527f4552433732313a206f776e657220717565727920666f72206e6f6e657869737460448201527f656e7420746f6b656e000000000000000000000000000000000000000000000060648201526084016106b1565b600d8054611389906133ca565b80601f01602080910402602001604051908101604052809291908181526020018280546113b5906133ca565b80156114025780601f106113d757610100808354040283529160200191611402565b820191906000526020600020905b8154815290600101906020018083116113e557829003601f168201915b505050505081565b60006001600160a01b0382166114885760405162461bcd60e51b815260206004820152602a60248201527f4552433732313a2062616c616e636520717565727920666f7220746865207a6560448201527f726f20616464726573730000000000000000000000000000000000000000000060648201526084016106b1565b506001600160a01b031660009081526006602052604090205490565b6000546001600160a01b031633146114fe5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b61150860006121e0565b565b6000805b600c5461ffff8216101561158857826001600160a01b031661155e600c8361ffff168154811061154e57634e487b7160e01b600052603260045260246000fd5b90600052602060002001546112f1565b6001600160a01b031614156115765750600192915050565b80611580816133ff565b91505061150e565b50600092915050565b6000546001600160a01b031633146115eb5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b8051611653576040517f62a65aec00000000000000000000000000000000000000000000000000000000815260206004820152600b60248201527f636f6e747261637455524900000000000000000000000000000000000000000060448201526064016106b1565b7fd144c5914915e649557dfce6d3199a18bae1ec7c249c1766603a2b76ade11c06600982604051611685929190613252565b60405180910390a18051610f28906009906020840190612aee565b606060048054610aa6906133ca565b610f2833838361223d565b6116c43383611f04565b6117365760405162461bcd60e51b815260206004820152603160248201527f4552433732313a207472616e736665722063616c6c6572206973206e6f74206f60448201527f776e6572206e6f7220617070726f76656400000000000000000000000000000060648201526084016106b1565b6117428484848461230c565b50505050565b600060135460001461176a57600a544210156117645750600490565b50600590565b600e54610100900460ff16156117805750600390565b600e5460ff16156117915750600290565b601060009054906101000a90046001600160a01b03166001600160a01b0316633f3ac92c6040518163ffffffff1660e01b815260040160206040518083038186803b1580156117df57600080fd5b505afa1580156117f3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906118179190612e5e565b156118225750600190565b50600090565b6000818152600560205260409020546060906001600160a01b031661186057604051634a1850bf60e11b815260040160405180910390fd5b600d61186b83612395565b60405160200161187c9291906130ba565b6040516020818303038152906040529050919050565b600060038061189f611748565b60058111156118be57634e487b7160e01b600052602160045260246000fd5b10156118cc57610710611748565b7f00000000000000000000000000000000000000000000000000000000000001f461ffff168361ffff161061191457604051634a1850bf60e11b815260040160405180910390fd5b61ffff83166000908152600f602052604090205460ff161591505b50919050565b6000546001600160a01b0316331461198f5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b6003611999611748565b60058111156119b857634e487b7160e01b600052602160045260246000fd5b8160058111156119d857634e487b7160e01b600052602160045260246000fd5b146119e557610710611748565b620249f08263ffffffff161015611a3f57604051636b6afd1760e11b815260206004820152601060248201527f63616c6c6261636b4761734c696d69740000000000000000000000000000000060448201526064016106b1565b6011546040517f5d3b1d300000000000000000000000000000000000000000000000000000000081526004810185905267ffffffffffffffff861660248201526003604482015263ffffffff84166064820152600160848201526001600160a01b03909116908190635d3b1d309060a401602060405180830381600087803b158015611aca57600080fd5b505af1158015611ade573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611b029190612f2a565b60128190556040519081527f34035302829687f51841b2858b7a87f37db58335690aec14aced0df98ad7f590906020015b60405180910390a15050505050565b60098054611389906133ca565b6000546001600160a01b03163314611ba95760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657260448201526064016106b1565b6001600160a01b038116611c255760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f646472657373000000000000000000000000000000000000000000000000000060648201526084016106b1565b611c2e816121e0565b50565b6001600160a01b03163b151590565b600061096a826124e3565b6127106bffffffffffffffffffffffff82161115611cd15760405162461bcd60e51b815260206004820152602a60248201527f455243323938313a20726f79616c7479206665652077696c6c2065786365656460448201527f2073616c6550726963650000000000000000000000000000000000000000000060648201526084016106b1565b6001600160a01b038216611d275760405162461bcd60e51b815260206004820152601960248201527f455243323938313a20696e76616c69642072656365697665720000000000000060448201526064016106b1565b604080518082019091526001600160a01b039092168083526bffffffffffffffffffffffff9091166020909201829052600160a01b90910217600155565b6000818152600760205260409020805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0384169081179091558190611da7826112f1565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b8160125414611e3257604051636b6afd1760e11b815260206004820152600960248201527f726571756573744964000000000000000000000000000000000000000000000060448201526064016106b1565b60135415611e6c576040517fba4ace8d00000000000000000000000000000000000000000000000000000000815260040160405180910390fd5b80600081518110611e8d57634e487b7160e01b600052603260045260246000fd5b60200260200101516013819055507f3d870ce8fb4499f2050897f38d3d9d5a938b7710a6a5e1d712db97db55dfbb9d81600081518110611edd57634e487b7160e01b600052603260045260246000fd5b6020026020010151604051611ef491815260200190565b60405180910390a1610f28612555565b6000818152600560205260408120546001600160a01b0316611f7d5760405162461bcd60e51b815260206004820152602c60248201527f4552433732313a206f70657261746f7220717565727920666f72206e6f6e657860448201526b34b9ba32b73a103a37b5b2b760a11b60648201526084016106b1565b6000611f88836112f1565b9050806001600160a01b0316846001600160a01b03161480611fcf57506001600160a01b0380821660009081526008602090815260408083209388168352929052205460ff165b80611ff35750836001600160a01b0316611fe884610b29565b6001600160a01b0316145b949350505050565b826001600160a01b031661200e826112f1565b6001600160a01b03161461208a5760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201527f6f776e657200000000000000000000000000000000000000000000000000000060648201526084016106b1565b6001600160a01b0382166121055760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f2061646460448201527f726573730000000000000000000000000000000000000000000000000000000060648201526084016106b1565b612110838383612704565b61211b600082611d65565b6001600160a01b0383166000908152600660205260408120805460019290612144908490613387565b90915550506001600160a01b0382166000908152600660205260408120805460019290612172908490613319565b9091555050600081815260056020526040808220805473ffffffffffffffffffffffffffffffffffffffff19166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600080546001600160a01b0383811673ffffffffffffffffffffffffffffffffffffffff19831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b816001600160a01b0316836001600160a01b0316141561229f5760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c65720000000000000060448201526064016106b1565b6001600160a01b03838116600081815260086020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b612317848484611ffb565b6123238484848461276e565b6117425760405162461bcd60e51b815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016106b1565b6060816123d557505060408051808201909152600181527f3000000000000000000000000000000000000000000000000000000000000000602082015290565b8160005b81156123ff57806123e981613421565b91506123f89050600a83613331565b91506123d9565b60008167ffffffffffffffff81111561242857634e487b7160e01b600052604160045260246000fd5b6040519080825280601f01601f191660200182016040528015612452576020820181803683370190505b5090505b8415611ff357612467600183613387565b9150612474600a8661345d565b61247f906030613319565b60f81b8183815181106124a257634e487b7160e01b600052603260045260246000fd5b60200101907effffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1916908160001a9053506124dc600a86613331565b9450612456565b60006001600160e01b031982167f80ac58cd00000000000000000000000000000000000000000000000000000000148061254657506001600160e01b031982167f5b5e139f00000000000000000000000000000000000000000000000000000000145b8061096a575061096a826128d1565b601060008061256f838261256881613421565b9450612938565b9050600080805b600261ffff821610156126d057838361258e81613421565b9450815181106125ae57634e487b7160e01b600052603260045260246000fd5b60200260200101519150858314156125dc576125d586866125ce81613421565b9750612938565b9350600092505b61ffff82166000908152600f602052604090205460ff1680612611575061ffff82166000908152600b602052604090205460ff165b1561266d57838361262181613421565b94508151811061264157634e487b7160e01b600052603260045260246000fd5b60200260200101519150858314156126685761266186866125ce81613421565b9350600092505b6125dc565b600c8054600181810190925561ffff84167fdf6966c971051c3d54ec59162606531493a51404a002842f56009d7e5cf4a8c79091018190556000908152600b60205260409020805460ff19169091179055806126c8816133ff565b915050612576565b507fecfc5e59150509bdb8e40e673268957cf1c290eddbdf40be1d1973c84bd7cde0600a54604051611b3391815260200190565b600a544210801561272357506000818152600b602052604090205460ff165b15610e8657600a546040517ff69d62900000000000000000000000000000000000000000000000000000000081526004810183905242602482015260448101919091526064016106b1565b60006001600160a01b0384163b156128c657604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906127b2903390899088908890600401613162565b602060405180830381600087803b1580156127cc57600080fd5b505af19250505080156127fc575060408051601f3d908101601f191682019092526127f991810190612e96565b60015b6128ac573d80801561282a576040519150601f19603f3d011682016040523d82523d6000602084013e61282f565b606091505b5080516128a45760405162461bcd60e51b815260206004820152603260248201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560448201527f63656976657220696d706c656d656e746572000000000000000000000000000060648201526084016106b1565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611ff3565b506001949350505050565b60006001600160e01b031982167f2a55205a00000000000000000000000000000000000000000000000000000000148061096a57507f01ffc9a7000000000000000000000000000000000000000000000000000000006001600160e01b031983161461096a565b606061ffff600061294a601086613331565b9050600061295960108761345d565b111561296d5761296a600182613319565b90505b60006129798286613345565b612984906010613345565b90506000612993836010613345565b67ffffffffffffffff8111156129b957634e487b7160e01b600052604160045260246000fd5b6040519080825280602002602001820160405280156129e2578160200160208202803683370190505b50905060005b83811015612ae357601354600090612a008584613319565b6040805160208101939093528201526060016040516020818303038152906040528051906020012060001c905060005b6010811015612ace57612a657f00000000000000000000000000000000000000000000000000000000000001f483891661343c565b8482612a72866010613345565b612a7c9190613319565b81518110612a9a57634e487b7160e01b600052603260045260246000fd5b61ffff90921660209283029190910190910152612aba6201000083613331565b915080612ac681613421565b915050612a30565b50508080612adb90613421565b9150506129e8565b509695505050505050565b828054612afa906133ca565b90600052602060002090601f016020900481019282612b1c5760008555612b62565b82601f10612b3557805160ff1916838001178555612b62565b82800160010185558215612b62579182015b82811115612b62578251825591602001919060010190612b47565b50612b6e929150612b72565b5090565b5b80821115612b6e5760008155600101612b73565b600067ffffffffffffffff831115612ba157612ba161349d565b612bb4601f8401601f19166020016132e8565b9050828152838383011115612bc857600080fd5b828260208301376000602084830101529392505050565b60008083601f840112612bf0578182fd5b50813567ffffffffffffffff811115612c07578182fd5b6020830191508360208260051b850101111561106957600080fd5b803561ffff81168114612c3457600080fd5b919050565b600060208284031215612c4a578081fd5b8135612c55816134b3565b9392505050565b600060208284031215612c6d578081fd5b8151612c55816134b3565b60008060408385031215612c8a578081fd5b8235612c95816134b3565b91506020830135612ca5816134b3565b809150509250929050565b600080600060608486031215612cc4578081fd5b8335612ccf816134b3565b92506020840135612cdf816134b3565b929592945050506040919091013590565b60008060008060808587031215612d05578081fd5b8435612d10816134b3565b93506020850135612d20816134b3565b925060408501359150606085013567ffffffffffffffff811115612d42578182fd5b8501601f81018713612d52578182fd5b612d6187823560208401612b87565b91505092959194509250565b60008060408385031215612d7f578182fd5b8235612d8a816134b3565b91506020830135612ca5816134c8565b60008060408385031215612dac578182fd5b8235612db7816134b3565b946020939093013593505050565b60008060408385031215612dd7578182fd5b8235612de2816134b3565b915060208301356bffffffffffffffffffffffff81168114612ca5578182fd5b60008060208385031215612e14578182fd5b823567ffffffffffffffff811115612e2a578283fd5b612e3685828601612bdf565b90969095509350505050565b600060208284031215612e53578081fd5b8135612c55816134c8565b600060208284031215612e6f578081fd5b8151612c55816134c8565b600060208284031215612e8b578081fd5b8135612c55816134d6565b600060208284031215612ea7578081fd5b8151612c55816134d6565b600060208284031215612ec3578081fd5b813567ffffffffffffffff811115612ed9578182fd5b8201601f81018413612ee9578182fd5b611ff384823560208401612b87565b600060208284031215612f09578081fd5b612c5582612c22565b600060208284031215612f23578081fd5b5035919050565b600060208284031215612f3b578081fd5b5051919050565b60008060408385031215612f54578182fd5b8235915060208084013567ffffffffffffffff80821115612f73578384fd5b818601915086601f830112612f86578384fd5b813581811115612f9857612f9861349d565b8060051b9150612fa98483016132e8565b8181528481019084860184860187018b1015612fc3578788fd5b8795505b83861015612fe5578035835260019590950194918601918601612fc7565b508096505050505050509250929050565b60008060408385031215613008578182fd5b50508035926020909101359150565b60008060006060848603121561302b578081fd5b833567ffffffffffffffff81168114613042578182fd5b925060208401359150604084013563ffffffff81168114613061578182fd5b809150509250925092565b6000815180845261308481602086016020860161339e565b601f01601f19169290920160200192915050565b600681106130b657634e487b7160e01b600052602160045260246000fd5b9052565b60008084546130c8816133ca565b600182811680156130e057600181146130f15761311d565b60ff1984168752828701945061311d565b8886526020808720875b858110156131145781548a8201529084019082016130fb565b50505082870194505b50505050835161313181836020880161339e565b7f2e6a736f6e0000000000000000000000000000000000000000000000000000009101908152600501949350505050565b60006001600160a01b03808716835280861660208401525083604083015260806060830152613194608083018461306c565b9695505050505050565b60208082528181018390526000908460408401835b86811015612ae35782356131c6816134c8565b1515825291830191908301906001016131b3565b60208082528181018390526000908460408401835b86811015612ae35761ffff61320384612c22565b16825291830191908301906001016131ef565b6020810161096a8284613098565b604081016132328285613098565b612c556020830184613098565b602081526000612c55602083018461306c565b604081526000808454613264816133ca565b8060408601526060600180841660008114613286576001811461329a576132c8565b60ff198516888401526080880195506132c8565b8987526020808820885b868110156132bf5781548b82018701529084019082016132a4565b8a018501975050505b505050505082810360208401526132df818561306c565b95945050505050565b604051601f8201601f1916810167ffffffffffffffff811182821017156133115761331161349d565b604052919050565b6000821982111561332c5761332c613471565b500190565b60008261334057613340613487565b500490565b600081600019048311821515161561335f5761335f613471565b500290565b600061ffff8381169083168181101561337f5761337f613471565b039392505050565b60008282101561339957613399613471565b500390565b60005b838110156133b95781810151838201526020016133a1565b838111156117425750506000910152565b600181811c908216806133de57607f821691505b6020821081141561192f57634e487b7160e01b600052602260045260246000fd5b600061ffff8083168181141561341757613417613471565b6001019392505050565b600060001982141561343557613435613471565b5060010190565b600061ffff8084168061345157613451613487565b92169190910692915050565b60008261346c5761346c613487565b500690565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052601260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b6001600160a01b0381168114611c2e57600080fd5b8015158114611c2e57600080fd5b6001600160e01b031981168114611c2e57600080fdfea2646970667358221220f2034f1e229b88831cb52007ccca0a46d48362bac9aaae242edb2a9ac942c27c64736f6c63430008040033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
00000000000000000000000000000000000000000000000000000000000001000000000000000000000000000000000000000000000000000000000000000160000000000000000000000000455ad5835b5f732a8c28d0ad79abf7942f5869060000000000000000000000000000000000000000000000000000000000000352000000000000000000000000ac1dc28708b2ff61a5d89a74fc09feed065692df000000000000000000000000271682deb8c4e0901d1a1550ad2e64d568e69909000000000000000000000000455ad5835b5f732a8c28d0ad79abf7942f5869060000000000000000000000000000000000000000000000000000000063b0cd000000000000000000000000000000000000000000000000000000000000000036697066733a2f2f516d63544d58597a79524145326f4258537634436a4752774742616545764e796f44707173734b7436425155346d2f00000000000000000000000000000000000000000000000000000000000000000000000000000000005068747470733a2f2f73332e65752d776573742d322e616d617a6f6e6177732e636f6d2f6d617261646f6e616f6666696369616c2e696f2f7469636b65744e4654436f6e74726163745552492e6a736f6e00000000000000000000000000000000
-----Decoded View---------------
Arg [0] : baseURI_ (string): ipfs://QmcTMXYzyRAE2oBXSv4CjGRwGBaeEvNyoDpqssKt6BQU4m/
Arg [1] : contractURI_ (string): https://s3.eu-west-2.amazonaws.com/maradonaofficial.io/ticketNFTContractURI.json
Arg [2] : royaltyAddress (address): 0x455AD5835B5f732A8C28d0ad79abF7942f586906
Arg [3] : royaltyFee (uint96): 850
Arg [4] : accessPassNFT_ (address): 0xAC1dc28708B2ff61a5d89A74fc09FeED065692Df
Arg [5] : vrfCoordinator_ (address): 0x271682DEB8C4E0901D1a1550aD2e64D568E69909
Arg [6] : nftHolder (address): 0x455AD5835B5f732A8C28d0ad79abF7942f586906
Arg [7] : frozenPeriod_ (uint256): 1672531200
-----Encoded View---------------
15 Constructor Arguments found :
Arg [0] : 0000000000000000000000000000000000000000000000000000000000000100
Arg [1] : 0000000000000000000000000000000000000000000000000000000000000160
Arg [2] : 000000000000000000000000455ad5835b5f732a8c28d0ad79abf7942f586906
Arg [3] : 0000000000000000000000000000000000000000000000000000000000000352
Arg [4] : 000000000000000000000000ac1dc28708b2ff61a5d89a74fc09feed065692df
Arg [5] : 000000000000000000000000271682deb8c4e0901d1a1550ad2e64d568e69909
Arg [6] : 000000000000000000000000455ad5835b5f732a8c28d0ad79abf7942f586906
Arg [7] : 0000000000000000000000000000000000000000000000000000000063b0cd00
Arg [8] : 0000000000000000000000000000000000000000000000000000000000000036
Arg [9] : 697066733a2f2f516d63544d58597a79524145326f4258537634436a47527747
Arg [10] : 42616545764e796f44707173734b7436425155346d2f00000000000000000000
Arg [11] : 0000000000000000000000000000000000000000000000000000000000000050
Arg [12] : 68747470733a2f2f73332e65752d776573742d322e616d617a6f6e6177732e63
Arg [13] : 6f6d2f6d617261646f6e616f6666696369616c2e696f2f7469636b65744e4654
Arg [14] : 436f6e74726163745552492e6a736f6e00000000000000000000000000000000
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.