Transaction Hash:
Block:
16687459 at Feb-23-2023 12:12:59 AM +UTC
Transaction Fee:
0.004837093536564706 ETH
$11.96
Gas Used:
149,494 Gas / 32.356439299 Gwei
Emitted Events:
223 |
AirdroppableAdventureKey.QuestUpdated( tokenId=996, tokenOwner=[Sender] 0x822cf5e208ebae029e970389fab7980403228a78, adventure=[Receiver] AdventureHub, questId=1, active=True, booted=False )
|
224 |
Achievements.TransferSingle( operator=[Receiver] AdventureHub, from=0x00000000...000000000, to=[Sender] 0x822cf5e208ebae029e970389fab7980403228a78, id=2, value=1 )
|
Account State Difference:
Address | Before | After | State Difference | ||
---|---|---|---|---|---|
0x266F5E58...42BB6fcA9 | |||||
0x3b7fAEc3...436441FfF
Miner
| (MEV Builder: 0x3b...FfF) | 0.65492976720455192 Eth | 0.65500451420455192 Eth | 0.000074747 | |
0x4301b745...3607aA1b3 | |||||
0x822CF5e2...403228A78 |
0.007527873889280782 Eth
Nonce: 25
|
0.002690780352716076 Eth
Nonce: 26
| 0.004837093536564706 |
Execution Trace
AdventureHub.enterQuestWithAdventureKey( adventureKeyAddress=0x266F5E58Bc4D24E0B7df7fbD457675842BB6fcA9, tokenId=996 )
AirdroppableAdventureKey.enterQuest( tokenId=996, questId=1 )
AirdroppableAdventureKey.enterQuest( tokenId=996, questId=1 )
-
AdventureHub.STATICCALL( )
AdventureHub.onQuestEntered( adventurer=0x822CF5e208EBae029E970389faB7980403228A78, 996, 1 )
enterQuestWithAdventureKey[AdventureHub (ln:114)]
_requireCallerOwnsToken[AdventureHub (ln:115)]
ownerOf[AdventureHub (ln:226)]
_msgSender[AdventureHub (ln:226)]
CallerNotTokenOwner[AdventureHub (ln:227)]
AdventureKeyIsInactive[AdventureHub (ln:118)]
enterQuest[AdventureHub (ln:120)]
File 1 of 4: AdventureHub
File 2 of 4: AirdroppableAdventureKey
File 3 of 4: Achievements
File 4 of 4: AirdroppableAdventureKey
// SPDX-License-Identifier: MIT pragma solidity 0.8.9; import "@limit-break/achievements/contracts/IAchievements.sol"; import "limit-break-contracts/contracts/adventures/IAdventure.sol"; import "limit-break-contracts/contracts/adventures/IAdventurousERC721.sol"; import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/token/ERC721/IERC721.sol"; import "@openzeppelin/contracts/utils/introspection/ERC165.sol"; error AdventureKeyAlreadyActivated(); error AdventureKeyIsInactive(); error CallerNotTokenOwner(); error CannotSpecifyZeroAddressForAchievementsToken(); error NoMoreAdventureKeysCanBeActivated(); error NotAnAdventurousContract(); error NotAnERC721Contract(); error OnQuestEnteredCallbackTriggeredByAddressThatIsNotAnActiveAdventureKey(); error UnknownAdventureKey(); /** * @title AdventureHub * @author Limit Break, Inc. * @notice An Adventure that is compatible with all adventure keys, unlocking crossover events with any partner game. */ contract AdventureHub is Ownable, ERC165, IAdventure { struct KeyState { bool isKeyActive; uint32 questId; uint256 achievementId; } /// @dev Largest unsigned int 32 bit value uint256 private constant MAX_UINT32 = type(uint32).max; /// @dev Points to the soulbound achievements token. /// Players earn soulbound achievement badges for entering games using their adventure keys. IAchievements public immutable achievementsToken; /// @dev The quest id of the adventure key that was most recently activated for the first time. uint32 public lastQuestId; /// @dev Maps an adventure key contract address to its key state (activation status and associated quest id) mapping (address => KeyState) public adventureKeyStates; /// @dev Maps a quest id to the adventure key contract address it is bound to /// If needed, off-chain applications can enumerate over this mapping using `lastQuestId` as the upper bound mapping (uint32 => address) public adventureKeyQuestIds; /// @dev Emitted whenever an adventure key is activated or deactivated event AdventureKeyActivationChanged(address indexed adventureKeyAddress, uint256 questId, uint256 achievementId, bool isActive); constructor(address achievementsTokenAddress) { if(achievementsTokenAddress == address(0)) { revert CannotSpecifyZeroAddressForAchievementsToken(); } achievementsToken = IAchievements(achievementsTokenAddress); } /// @notice Activates an adventure key for use with the Adventure Hub. /// Throws when the caller is not the owner of this contract. /// Throws when the number of previously activated adventure keys is equal to the maximum uint32 value. /// Throws when the specified adventure key contract does not implement the IAdventurous interface. /// Throws when the specified adventure key contract is already activated for use in the Adventure Hub. /// Throws if AdventureHub MINTER_ROLE access is revoked on the achievements contract /// /// Postconditions: /// The specified adventure key contract has been activated. New entries into quests with adventure key are permitted. /// `adventureKeyStates` mapping has been updated. /// `lastQuestId` value has been incremented. /// An achievement id has been reserved for users that enter the quest with the activated key. /// /// @dev The metadataURI parameter has no effect for re-activations of adventure keys. /// If a key has already been activated, it is recommended to leave metadataURI blank. function activateAdventureKey(address adventureKeyAddress, string calldata metadataURI) external onlyOwner { if(!IERC165(adventureKeyAddress).supportsInterface(type(IAdventurous).interfaceId)) { revert NotAnAdventurousContract(); } if(!IERC165(adventureKeyAddress).supportsInterface(type(IERC721).interfaceId)) { revert NotAnERC721Contract(); } KeyState memory keyState = adventureKeyStates[adventureKeyAddress]; if(keyState.questId == 0) { if(lastQuestId == MAX_UINT32) { revert NoMoreAdventureKeysCanBeActivated(); } unchecked { uint32 questId = ++lastQuestId; adventureKeyStates[adventureKeyAddress].questId = questId; adventureKeyQuestIds[questId] = adventureKeyAddress; } adventureKeyStates[adventureKeyAddress].achievementId = achievementsToken.reserveAchievementId(metadataURI); } else if(keyState.isKeyActive) { revert AdventureKeyAlreadyActivated(); } adventureKeyStates[adventureKeyAddress].isKeyActive = true; emit AdventureKeyActivationChanged(adventureKeyAddress, adventureKeyStates[adventureKeyAddress].questId, adventureKeyStates[adventureKeyAddress].achievementId, true); } /// @notice Deactivates an adventure key, preventing new entries into quests using the deactivated key. /// Players may still exit after the key is deactivated. /// Throws when the caller is not the owner of this contract. /// Throws when the adventure key address has never been activated, or is currently de-activated. /// /// Postconditions: /// The specified adventure key contract has been de-activated. New entries into quests with the de-activated /// key will be disabled unless the key is re-activated. function deactivateAdventureKey(address adventureKeyAddress) external onlyOwner { KeyState memory keyState = adventureKeyStates[adventureKeyAddress]; if(!keyState.isKeyActive) { revert AdventureKeyIsInactive(); } adventureKeyStates[adventureKeyAddress].isKeyActive = false; emit AdventureKeyActivationChanged(adventureKeyAddress, keyState.questId, keyState.achievementId, false); } /// @notice Enters the quest associated with the specified adventure key contract for the specified token id. /// Throws when the owner of the adventure key is not the caller. /// Throws when the adventure key address has never been activated, or is currently de-activated. /// Throws when the AdventureHub is not currently whitelisted on the specified adventure key. /// Throws when the AdventureHub has not been approved by user for adventures on the specified adventure key. /// Throws when the specified token id on the specified adventure key is already in the quest. /// /// Postconditions: /// The specified token id has entered the quest associated with the specified adventure key. function enterQuestWithAdventureKey(address adventureKeyAddress, uint256 tokenId) external { _requireCallerOwnsToken(adventureKeyAddress, tokenId); KeyState storage keyState = adventureKeyStates[adventureKeyAddress]; if(!keyState.isKeyActive) { revert AdventureKeyIsInactive(); } IAdventurous(adventureKeyAddress).enterQuest(tokenId, keyState.questId); } /// @notice Exits the quest associated with the specified adventure key contract for the specified token id. /// Throws when the owner of the adventure key is not the caller. /// Throws when the adventure key address has never been activated. /// Throws when the AdventureHub is not currently whitelisted on the specified adventure key. /// Throws when the AdventureHub has not been approved by user for adventures on the specified adventure key. /// Throws when the specified token id on the specified adventure key is no longer in the quest. /// - This condition should be rare, and can only happen if the Adventure Hub is removed from the whitelist /// and re-whitelisted, presenting a limited window of opportunity to backdoor userExitQuest. /// /// Postconditions: /// The specified token id has exited from the quest associated with the specified adventure key. function exitQuestWithAdventureKey(address adventureKeyAddress, uint256 tokenId) external { _requireCallerOwnsToken(adventureKeyAddress, tokenId); KeyState storage keyState = adventureKeyStates[adventureKeyAddress]; if(keyState.questId == 0) { revert UnknownAdventureKey(); } IAdventurous(adventureKeyAddress).exitQuest(tokenId, keyState.questId); } /// @dev Callback that mints a soulbound achievement to the adventurer that entered the quest if they haven't received the achievement previously. function onQuestEntered(address adventurer, uint256 /*tokenId*/, uint256 /*questId*/) external override { KeyState storage senderKeyState = adventureKeyStates[_msgSender()]; if(!senderKeyState.isKeyActive) { revert OnQuestEnteredCallbackTriggeredByAddressThatIsNotAnActiveAdventureKey(); } if(achievementsToken.balanceOf(adventurer, senderKeyState.achievementId) == 0) { achievementsToken.mint(adventurer, senderKeyState.achievementId, 1); } } /// @dev onQuestExited callback does nothing for this contract, because there is no state to synchronize function onQuestExited(address /*adventurer*/, uint256 /*tokenId*/, uint256 /*questId*/, uint256 /*questStartTimestamp*/) external override view {} /// @dev Enumerates all tokens that the specified player currently has entered into the quest. /// Never use this function in a transaction context - it is fine for a read-only query for /// external applications, but will consume a lot of gas when used in a transaction. /// Throws if specified adventure key address has never been activated. function findQuestingTokensByAdventureKeyAndPlayer(address adventureKeyAddress, address player, uint256 tokenIdPageStart, uint256 tokenIdPageEnd) external view returns (uint256[] memory tokenIdsInQuest) { uint256 questId = adventureKeyStates[adventureKeyAddress].questId; if(questId == 0) { revert UnknownAdventureKey(); } IAdventurousERC721 adventureKey = IAdventurousERC721(adventureKeyAddress); unchecked { // First, find all the token ids owned by the player uint256 ownerBalance = adventureKey.balanceOf(player); uint256[] memory ownedTokenIds = new uint256[](ownerBalance); uint256 tokenIndex = 0; for(uint256 tokenId = tokenIdPageStart; tokenId <= tokenIdPageEnd; ++tokenId) { try adventureKey.ownerOf(tokenId) returns (address ownerOfToken) { if(ownerOfToken == player) { ownedTokenIds[tokenIndex++] = tokenId; } } catch {} if(tokenIndex == ownerBalance || tokenId == type(uint256).max) { break; } } // For each owned token id, check the quest count // When 1 or greater, the spirit is engaged in a quest on this adventure. address thisAddress = address(this); uint256 numberOfTokenIdsOnQuest = 0; for(uint256 i = 0; i < ownerBalance; ++i) { uint256 ownedTokenId = ownedTokenIds[i]; if(ownedTokenId > 0) { (bool isPartipatingInQuest,,) = adventureKey.isParticipatingInQuest(ownedTokenId, thisAddress, questId); if(isPartipatingInQuest) { ++numberOfTokenIdsOnQuest; } } } // Finally, make one more pass and populate the player quests return array uint256 questIndex = 0; tokenIdsInQuest = new uint256[](numberOfTokenIdsOnQuest); for(uint256 i = 0; i < ownerBalance; ++i) { uint256 ownedTokenId = ownedTokenIds[i]; if(ownedTokenId > 0) { (bool isPartipatingInQuest,,) = adventureKey.isParticipatingInQuest(ownedTokenId, thisAddress, questId); if(isPartipatingInQuest) { tokenIdsInQuest[questIndex] = ownedTokenId; ++questIndex; } } if(questIndex == numberOfTokenIdsOnQuest) { break; } } } return tokenIdsInQuest; } /// @dev Adventure Keys are always locked/non-transferrable while they are participating in Adventure Hub quests function questsLockTokens() external override pure returns (bool) { return false; } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override (ERC165, IERC165) returns (bool) { return interfaceId == type(IAdventure).interfaceId || super.supportsInterface(interfaceId); } /// @dev Validates that the caller owns the specified token for the specified token contract address /// Throws when the caller does not own the specified token. function _requireCallerOwnsToken(address adventureKeyAddress, uint256 tokenId) internal view { if(IERC721(adventureKeyAddress).ownerOf(tokenId) != _msgSender()) { revert CallerNotTokenOwner(); } } }// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventure * @author Limit Break, Inc. * @notice The base interface that all `Adventure` contracts must conform to. * @dev All contracts that implement the adventure/quest system and interact with an {IAdventurous} token are required to implement this interface. */ interface IAdventure is IERC165 { /** * @dev Returns whether or not quests on this adventure lock tokens. * Developers of adventure contract should ensure that this is immutable * after deployment of the adventure contract. Failure to do so * can lead to error that deadlock token transfers. */ function questsLockTokens() external view returns (bool); /** * @dev A callback function that AdventureERC721 must invoke when a quest has been successfully entered. * Throws if the caller is not an expected AdventureERC721 contract designed to work with the Adventure. * Not permitted to throw in any other case, as this could lead to tokens being locked in quests. */ function onQuestEntered(address adventurer, uint256 tokenId, uint256 questId) external; /** * @dev A callback function that AdventureERC721 must invoke when a quest has been successfully exited. * Throws if the caller is not an expected AdventureERC721 contract designed to work with the Adventure. * Not permitted to throw in any other case, as this could lead to tokens being locked in quests. */ function onQuestExited(address adventurer, uint256 tokenId, uint256 questId, uint256 questStartTimestamp) external; } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/token/ERC1155/extensions/IERC1155MetadataURI.sol"; /** * @title IAchievements * @author Limit Break, Inc. * @notice Interface for the Achievements token contract */ interface IAchievements is IERC1155MetadataURI { /// @dev Reserves an achievement id and associates the achievement id with a single allowed minter. function reserveAchievementId(string calldata metadataURI) external returns (uint256); /// @dev Mints an achievement of type `id` to the `to` address. function mint(address to, uint256 id, uint256 amount) external; /// @dev Batch mints achievements to the `to` address. function mintBatch(address to, uint256[] calldata ids, uint256[] calldata amounts) external; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; import "./IAdventurous.sol"; import "@openzeppelin/contracts/token/ERC721/IERC721.sol"; /** * @title IAdventurousERC721 * @author Limit Break, Inc. * @notice Combines all {IAdventurous} and all {IERC721} functionality into a single, unified interface. * @dev This interface may be used as a convenience to interact with tokens that support both interface standards. */ interface IAdventurousERC721 is IERC721, IAdventurous { }// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (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 Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.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 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 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 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 v4.4.1 (token/ERC1155/extensions/IERC1155MetadataURI.sol) pragma solidity ^0.8.0; import "../IERC1155.sol"; /** * @dev Interface of the optional ERC1155MetadataExtension interface, as defined * in the https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[EIP]. * * _Available since v3.1._ */ interface IERC1155MetadataURI is IERC1155 { /** * @dev Returns the URI for token type `id`. * * If the `\\{id\\}` substring is present in the URI, it must be replaced by * clients with the actual token type ID. */ function uri(uint256 id) external view returns (string memory); } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC1155/IERC1155.sol) pragma solidity ^0.8.0; import "../../utils/introspection/IERC165.sol"; /** * @dev Required interface of an ERC1155 compliant contract, as defined in the * https://eips.ethereum.org/EIPS/eip-1155[EIP]. * * _Available since v3.1._ */ interface IERC1155 is IERC165 { /** * @dev Emitted when `value` tokens of token type `id` are transferred from `from` to `to` by `operator`. */ event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value); /** * @dev Equivalent to multiple {TransferSingle} events, where `operator`, `from` and `to` are the same for all * transfers. */ event TransferBatch( address indexed operator, address indexed from, address indexed to, uint256[] ids, uint256[] values ); /** * @dev Emitted when `account` grants or revokes permission to `operator` to transfer their tokens, according to * `approved`. */ event ApprovalForAll(address indexed account, address indexed operator, bool approved); /** * @dev Emitted when the URI for token type `id` changes to `value`, if it is a non-programmatic URI. * * If an {URI} event was emitted for `id`, the standard * https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[guarantees] that `value` will equal the value * returned by {IERC1155MetadataURI-uri}. */ event URI(string value, uint256 indexed id); /** * @dev Returns the amount of tokens of token type `id` owned by `account`. * * Requirements: * * - `account` cannot be the zero address. */ function balanceOf(address account, uint256 id) external view returns (uint256); /** * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {balanceOf}. * * Requirements: * * - `accounts` and `ids` must have the same length. */ function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids) external view returns (uint256[] memory); /** * @dev Grants or revokes permission to `operator` to transfer the caller's tokens, according to `approved`, * * Emits an {ApprovalForAll} event. * * Requirements: * * - `operator` cannot be the caller. */ function setApprovalForAll(address operator, bool approved) external; /** * @dev Returns true if `operator` is approved to transfer ``account``'s tokens. * * See {setApprovalForAll}. */ function isApprovedForAll(address account, address operator) external view returns (bool); /** * @dev Transfers `amount` tokens of token type `id` from `from` to `to`. * * Emits a {TransferSingle} event. * * Requirements: * * - `to` cannot be the zero address. * - If the caller is not `from`, it must have been approved to spend ``from``'s tokens via {setApprovalForAll}. * - `from` must have a balance of tokens of type `id` of at least `amount`. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the * acceptance magic value. */ function safeTransferFrom( address from, address to, uint256 id, uint256 amount, bytes calldata data ) external; /** * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {safeTransferFrom}. * * Emits a {TransferBatch} event. * * Requirements: * * - `ids` and `amounts` must have the same length. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the * acceptance magic value. */ function safeBatchTransferFrom( address from, address to, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data ) external; } // SPDX-License-Identifier: MIT pragma solidity ^0.8.4; import "./Quest.sol"; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventurous * @author Limit Break, Inc. * @notice The base interface that all `Adventurous` token contracts must conform to in order to support adventures and quests. * @dev All contracts that support adventures and quests are required to implement this interface. */ interface IAdventurous is IERC165 { /** * @dev Emitted when a token enters or exits a quest */ event QuestUpdated(uint256 indexed tokenId, address indexed tokenOwner, address indexed adventure, uint256 questId, bool active, bool booted); /** * @notice Transfers a player's token if they have opted into an authorized, whitelisted adventure. */ function adventureTransferFrom(address from, address to, uint256 tokenId) external; /** * @notice Safe transfers a player's token if they have opted into an authorized, whitelisted adventure. */ function adventureSafeTransferFrom(address from, address to, uint256 tokenId) external; /** * @notice Burns a player's token if they have opted into an authorized, whitelisted adventure. */ function adventureBurn(uint256 tokenId) external; /** * @notice Enters a player's token into a quest if they have opted into an authorized, whitelisted adventure. */ function enterQuest(uint256 tokenId, uint256 questId) external; /** * @notice Exits a player's token from a quest if they have opted into an authorized, whitelisted adventure. */ function exitQuest(uint256 tokenId, uint256 questId) external; /** * @notice Returns the number of quests a token is actively participating in for a specified adventure */ function getQuestCount(uint256 tokenId, address adventure) external view returns (uint256); /** * @notice Returns the amount of time a token has been participating in the specified quest */ function getTimeOnQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (uint256); /** * @notice Returns whether or not a token is currently participating in the specified quest as well as the time it was started and the quest index */ function isParticipatingInQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (bool participatingInQuest, uint256 startTimestamp, uint256 index); /** * @notice Returns a list of all active quests for the specified token id and adventure */ function getActiveQuests(uint256 tokenId, address adventure) external view returns (Quest[] memory activeQuests); } // SPDX-License-Identifier: MIT pragma solidity ^0.8.4; /** * @title Quest * @author Limit Break, Inc. * @notice Quest data structure for {IAdventurous} contracts. */ struct Quest { bool isActive; uint32 questId; uint64 startTimestamp; uint32 arrayIndex; }// 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; } }
File 2 of 4: AirdroppableAdventureKey
// SPDX-License-Identifier: MIT pragma solidity 0.8.9; import "limit-break-contracts/contracts/adventures/AdventureNFT.sol"; import "limit-break-contracts/contracts/initializable/IMaxSupplyInitializer.sol"; error MaxSupplyAlreadyInitialized(); error MaxSupplyCannotBeSetToZero(); error MaxSupplyExceeded(uint256 supplyAfterMint, uint256 maxSupply); /** * @title AirdroppableAdventureKey * @author Limit Break, Inc. * @notice An adventure key reference contract that can be airdropped and cloned using EIP-1167. * See https://eips.ethereum.org/EIPS/eip-1167 for details. */ contract AirdroppableAdventureKey is AdventureNFT, IMaxSupplyInitializer { uint256 private nextTokenId; /// @dev The maximum token supply uint256 public maxSupply; constructor() ERC721("", "") {} /// @dev Initializes parameters of tokens with maximum supplies. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeMaxSupply(uint256 maxSupply_) public override onlyOwner { if(maxSupply > 0) { revert MaxSupplyAlreadyInitialized(); } if(maxSupply_ == 0) { revert MaxSupplyCannotBeSetToZero(); } maxSupply = maxSupply_; } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override(AdventureNFT, IERC165) returns (bool) { return interfaceId == type(IMaxSupplyInitializer).interfaceId || super.supportsInterface(interfaceId); } /// @notice Owner bulk mint to airdrop function airdropMint(address[] calldata to) external onlyOwner { if(nextTokenId == 0) { nextTokenId = 1; } uint256 batchSize = to.length; uint256 tokenIdToMint = nextTokenId; uint256 supplyAfterMint = tokenIdToMint + batchSize - 1; uint256 maxSupply_ = maxSupply; if(supplyAfterMint > maxSupply_) { revert MaxSupplyExceeded(supplyAfterMint, maxSupply_); } nextTokenId = nextTokenId + batchSize; unchecked { for(uint256 i = 0; i < batchSize; ++i) { _mint(to[i], tokenIdToMint + i); } } } }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./AdventureERC721.sol"; import "../initializable/IRoyaltiesInitializer.sol"; import "../initializable/IURIInitializer.sol"; import "@openzeppelin/contracts/token/common/ERC2981.sol"; error AlreadyInitializedRoyalties(); error AlreadyInitializedURI(); error ExceedsMaxRoyaltyFee(); error NonexistentToken(); /** * @title AdventureNFT * @author Limit Break, Inc. * @notice Standardizes commonly shared boilerplate code that adds base/suffix URI and EIP-2981 royalties to {AdventureERC721} contracts. */ abstract contract AdventureNFT is AdventureERC721, ERC2981, IRoyaltiesInitializer, IURIInitializer { using Strings for uint256; /// @dev The maximum allowable royalty fee is 10% uint96 public constant MAX_ROYALTY_FEE_NUMERATOR = 1000; /// @notice Specifies whether or not the contract is initialized bool public initializedRoyalties; /// @notice Specifies whether or not the contract is initialized bool public initializedURI; /// @dev Base token uri string public baseTokenURI; /// @dev Token uri suffix/extension string public suffixURI = ".json"; /// @dev Emitted when base URI is set. event BaseURISet(string baseTokenURI); /// @dev Emitted when suffix URI is set. event SuffixURISet(string suffixURI); /// @dev Emitted when royalty is set. event RoyaltySet(address receiver, uint96 feeNumerator); /// @dev Initializes parameters of tokens with royalties. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeRoyalties(address receiver, uint96 feeNumerator) public override onlyOwner { if(initializedRoyalties) { revert AlreadyInitializedRoyalties(); } setRoyaltyInfo(receiver, feeNumerator); initializedRoyalties = true; } /// @dev Initializes parameters of tokens with uri values. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeURI(string memory baseURI_, string memory suffixURI_) public override onlyOwner { if(initializedURI) { revert AlreadyInitializedURI(); } setBaseURI(baseURI_); setSuffixURI(suffixURI_); initializedURI = true; } /// @dev Required to return baseTokenURI for tokenURI function _baseURI() internal view virtual override returns (string memory) { return baseTokenURI; } /// @notice Sets base URI function setBaseURI(string memory baseTokenURI_) public onlyOwner { baseTokenURI = baseTokenURI_; emit BaseURISet(baseTokenURI_); } /// @notice Sets suffix URI function setSuffixURI(string memory suffixURI_) public onlyOwner { suffixURI = suffixURI_; emit SuffixURISet(suffixURI_); } /// @notice Sets royalty information function setRoyaltyInfo(address receiver, uint96 feeNumerator) public onlyOwner { if(feeNumerator > MAX_ROYALTY_FEE_NUMERATOR) { revert ExceedsMaxRoyaltyFee(); } _setDefaultRoyalty(receiver, feeNumerator); emit RoyaltySet(receiver, feeNumerator); } /// @notice Returns tokenURI if baseURI is set function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if(!_exists(tokenId)) { revert NonexistentToken(); } string memory baseURI = _baseURI(); return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString(), suffixURI)) : ""; } function supportsInterface(bytes4 interfaceId) public view virtual override (AdventureERC721, ERC2981, IERC165) returns (bool) { return interfaceId == type(IRoyaltiesInitializer).interfaceId || interfaceId == type(IURIInitializer).interfaceId || super.supportsInterface(interfaceId); } }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IMaxSupplyInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include a maximum supply. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IMaxSupplyInitializer is IERC165 { /** * @notice Initializes max supply parameters */ function initializeMaxSupply(uint256 maxSupply_) external; }// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.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: * * - `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 pragma solidity ^0.8.0; import "./IAdventurous.sol"; import "./AdventurePermissions.sol"; import "../initializable/IAdventureERC721Initializer.sol"; import "../utils/tokens/InitializableERC721.sol"; error AlreadyInitializedAdventureERC721(); error AlreadyOnQuest(); error AnActiveQuestIsPreventingTransfers(); error CallerNotTokenOwner(); error MaxSimultaneousQuestsCannotBeZero(); error MaxSimultaneousQuestsExceeded(); error NotOnQuest(); error QuestIdOutOfRange(); error TooManyActiveQuests(); /** * @title AdventureERC721 * @author Limit Break, Inc. * @notice Implements the {IAdventurous} token standard for ERC721-compliant tokens. * @dev Inherits {InitializableERC721} to provide the option to support EIP-1167. */ abstract contract AdventureERC721 is InitializableERC721, AdventurePermissions, IAdventurous, IAdventureERC721Initializer { uint256 public constant MAX_UINT32 = type(uint32).max; /// @notice Specifies an upper bound for the maximum number of simultaneous quests. uint256 public constant MAX_CONCURRENT_QUESTS = 100; /// @notice Specifies whether or not the contract is initialized bool public initializedAdventureERC721; /// @dev The most simultaneous quests the token may participate in at a time uint256 public maxSimultaneousQuests; /// @dev Maps each token id to a mapping that can enumerate all active quests within an adventure mapping (uint256 => mapping (address => uint32[])) public activeQuestList; /// @dev Maps each token id to a mapping from adventure address to a mapping of quest ids to quest details mapping (uint256 => mapping (address => mapping (uint32 => Quest))) public activeQuestLookup; /// @dev Maps each token id to the number of blocking quests it is currently entered into mapping (uint256 => uint256) public blockingQuestCounts; /// @dev Initializes parameters of AdventureERC721 tokens. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeAdventureERC721(uint256 maxSimultaneousQuests_) public override onlyOwner { if(initializedAdventureERC721) { revert AlreadyInitializedAdventureERC721(); } _validateMaxSimultaneousQuests(maxSimultaneousQuests_); maxSimultaneousQuests = maxSimultaneousQuests_; initializedAdventureERC721 = true; } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override (InitializableERC721, IERC165) returns (bool) { return interfaceId == type(IAdventurous).interfaceId || interfaceId == type(IAdventureERC721Initializer).interfaceId || super.supportsInterface(interfaceId); } /// @notice Allows an authorized game contract to transfer a player's token if they have opted in function adventureTransferFrom(address from, address to, uint256 tokenId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _transfer(from, to, tokenId); } /// @notice Allows an authorized game contract to transfer a player's token if they have opted in function adventureSafeTransferFrom(address from, address to, uint256 tokenId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _safeTransfer(from, to, tokenId, ""); } /// @notice Allows an authorized game contract to burn a player's token if they have opted in function adventureBurn(uint256 tokenId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _burn(tokenId); } /// @notice Allows an authorized game contract to enter a player's token into a quest if they have opted in function enterQuest(uint256 tokenId, uint256 questId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _enterQuest(tokenId, _msgSender(), questId); } /// @notice Allows an authorized game contract to exit a player's token from a quest if they have opted in /// For developers of adventure contracts that perform adventure burns, be aware that the adventure must exitQuest /// before the adventure burn occurs, as _exitQuest emits the owner of the token, which would revert after burning. function exitQuest(uint256 tokenId, uint256 questId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _exitQuest(tokenId, _msgSender(), questId); } /// @notice Admin-only ability to boot a token from all quests on an adventure. /// This ability is only unlocked in the event that an adventure has been unwhitelisted, as early exiting /// from quests can cause out of sync state between the ERC721 token contract and the adventure/quest. function bootFromAllQuests(uint256 tokenId, address adventure) external onlyOwner { _requireAdventureRemovedFromWhitelist(adventure); _exitAllQuests(tokenId, adventure, true); } /// @notice Gives the player the ability to exit a quest without interacting directly with the approved, whitelisted adventure /// This ability is only unlocked in the event that an adventure has been unwhitelisted, as early exiting /// from quests can cause out of sync state between the ERC721 token contract and the adventure/quest. function userExitQuest(uint256 tokenId, address adventure, uint256 questId) external { _requireAdventureRemovedFromWhitelist(adventure); _requireCallerOwnsToken(tokenId); _exitQuest(tokenId, adventure, questId); } /// @notice Gives the player the ability to exit all quests on an adventure without interacting directly with the approved, whitelisted adventure /// This ability is only unlocked in the event that an adventure has been unwhitelisted, as early exiting /// from quests can cause out of sync state between the ERC721 token contract and the adventure/quest. function userExitAllQuests(uint256 tokenId, address adventure) external { _requireAdventureRemovedFromWhitelist(adventure); _requireCallerOwnsToken(tokenId); _exitAllQuests(tokenId, adventure, false); } /// @notice Returns the number of quests a token is actively participating in for a specified adventure function getQuestCount(uint256 tokenId, address adventure) public override view returns (uint256) { return activeQuestList[tokenId][adventure].length; } /// @notice Returns the amount of time a token has been participating in the specified quest function getTimeOnQuest(uint256 tokenId, address adventure, uint256 questId) public override view returns (uint256) { (bool participatingInQuest, uint256 startTimestamp,) = isParticipatingInQuest(tokenId, adventure, questId); return participatingInQuest ? (block.timestamp - startTimestamp) : 0; } /// @notice Returns whether or not a token is currently participating in the specified quest as well as the time it was started and the quest index function isParticipatingInQuest(uint256 tokenId, address adventure, uint256 questId) public override view returns (bool participatingInQuest, uint256 startTimestamp, uint256 index) { Quest memory quest = activeQuestLookup[tokenId][adventure][uint32(questId)]; participatingInQuest = quest.isActive; startTimestamp = quest.startTimestamp; index = quest.arrayIndex; return (participatingInQuest, startTimestamp, index); } /// @notice Returns a list of all active quests for the specified token id and adventure function getActiveQuests(uint256 tokenId, address adventure) public override view returns (Quest[] memory activeQuests) { uint256 questCount = getQuestCount(tokenId, adventure); activeQuests = new Quest[](questCount); uint32[] memory activeQuestIdList = activeQuestList[tokenId][adventure]; for(uint256 i = 0; i < questCount; ++i) { activeQuests[i] = activeQuestLookup[tokenId][adventure][activeQuestIdList[i]]; } return activeQuests; } /// @dev Enters the specified quest for a token id. /// Throws if the token is already participating in the specified quest. /// Throws if the number of active quests exceeds the max allowable for the given adventure. /// Emits a QuestUpdated event for off-chain processing. function _enterQuest(uint256 tokenId, address adventure, uint256 questId) internal { _requireValidQuestId(questId); (bool participatingInQuest,,) = isParticipatingInQuest(tokenId, adventure, questId); if(participatingInQuest) { revert AlreadyOnQuest(); } uint256 currentQuestCount = getQuestCount(tokenId, adventure); if(currentQuestCount == maxSimultaneousQuests) { revert TooManyActiveQuests(); } uint32 castedQuestId = uint32(questId); activeQuestList[tokenId][adventure].push(castedQuestId); activeQuestLookup[tokenId][adventure][castedQuestId] = Quest({ isActive: true, startTimestamp: uint64(block.timestamp), questId: castedQuestId, arrayIndex: uint32(currentQuestCount) }); address ownerOfToken = ownerOf(tokenId); emit QuestUpdated(tokenId, ownerOfToken, adventure, questId, true, false); if(IAdventure(adventure).questsLockTokens()) { unchecked { ++blockingQuestCounts[tokenId]; } } // Invoke callback to the adventure to facilitate state synchronization as needed IAdventure(adventure).onQuestEntered(ownerOfToken, tokenId, questId); } /// @dev Exits the specified quest for a token id. /// Throws if the token is not currently participating on the specified quest. /// Emits a QuestUpdated event for off-chain processing. function _exitQuest(uint256 tokenId, address adventure, uint256 questId) internal { _requireValidQuestId(questId); (bool participatingInQuest, uint256 startTimestamp, uint256 index) = isParticipatingInQuest(tokenId, adventure, questId); if(!participatingInQuest) { revert NotOnQuest(); } uint32 castedQuestId = uint32(questId); uint256 lastArrayIndex = getQuestCount(tokenId, adventure) - 1; activeQuestList[tokenId][adventure][index] = activeQuestList[tokenId][adventure][lastArrayIndex]; activeQuestLookup[tokenId][adventure][activeQuestList[tokenId][adventure][lastArrayIndex]].arrayIndex = uint32(index); activeQuestList[tokenId][adventure].pop(); delete activeQuestLookup[tokenId][adventure][castedQuestId]; address ownerOfToken = ownerOf(tokenId); emit QuestUpdated(tokenId, ownerOfToken, adventure, questId, false, false); if(IAdventure(adventure).questsLockTokens()) { --blockingQuestCounts[tokenId]; } // Invoke callback to the adventure to facilitate state synchronization as needed IAdventure(adventure).onQuestExited(ownerOfToken, tokenId, questId, startTimestamp); } /// @dev Removes the specified token id from all quests on the specified adventure function _exitAllQuests(uint256 tokenId, address adventure, bool booted) internal { address tokenOwner = ownerOf(tokenId); uint256 questCount = getQuestCount(tokenId, adventure); if(IAdventure(adventure).questsLockTokens()) { blockingQuestCounts[tokenId] -= questCount; } for(uint256 i = 0; i < questCount; ++i) { uint256 questId = activeQuestList[tokenId][adventure][i]; Quest memory quest = activeQuestLookup[tokenId][adventure][uint32(questId)]; uint256 startTimestamp = quest.startTimestamp; emit QuestUpdated(tokenId, tokenOwner, adventure, questId, false, booted); delete activeQuestLookup[tokenId][adventure][uint32(questId)]; // Invoke callback to the adventure to facilitate state synchronization as needed IAdventure(adventure).onQuestExited(tokenOwner, tokenId, questId, startTimestamp); } delete activeQuestList[tokenId][adventure]; } /// @dev By default, tokens that are participating in quests are transferrable. However, if a token is participating /// in a quest on an adventure that was designated as a token locker, the transfer will revert and keep the token /// locked. function _beforeTokenTransfer(address /*from*/, address /*to*/, uint256 tokenId) internal virtual override { if(blockingQuestCounts[tokenId] > 0) { revert AnActiveQuestIsPreventingTransfers(); } } /// @dev Validates that the caller owns the specified token /// Throws when the caller does not own the specified token. function _requireCallerOwnsToken(uint256 tokenId) internal view { if(ownerOf(tokenId) != _msgSender()) { revert CallerNotTokenOwner(); } } /// @dev Validates that the specified quest id does not overflow a uint32 /// Throws when questId exceeds the largest uint32 value. function _requireValidQuestId(uint256 questId) internal pure { if(questId > MAX_UINT32) { revert QuestIdOutOfRange(); } } /// @dev Validates that the specified value of max simultaneous quests is in range [1-MAX_CONCURRENT_QUESTS] /// Throws when `maxSimultaneousQuests_` is zero. /// Throws when `maxSimultaneousQuests_` is more than MAX_CONCURRENT_QUESTS. function _validateMaxSimultaneousQuests(uint256 maxSimultaneousQuests_) internal pure { if(maxSimultaneousQuests_ == 0) { revert MaxSimultaneousQuestsCannotBeZero(); } if(maxSimultaneousQuests_ > MAX_CONCURRENT_QUESTS) { revert MaxSimultaneousQuestsExceeded(); } } } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IURIInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include a base uri and suffix uri. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IURIInitializer is IERC165 { /** * @notice Initializes uri parameters */ function initializeURI(string memory baseURI_, string memory suffixURI_) external; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IRoyaltiesInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include OpenZeppelin ERC2981 functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IRoyaltiesInitializer is IERC165 { /** * @notice Initializes royalty parameters */ function initializeRoyalties(address receiver, uint96 feeNumerator) external; }// 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/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 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 pragma solidity ^0.8.0; import "./Quest.sol"; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventurous * @author Limit Break, Inc. * @notice The base interface that all `Adventurous` token contracts must conform to in order to support adventures and quests. * @dev All contracts that support adventures and quests are required to implement this interface. */ interface IAdventurous is IERC165 { /** * @dev Emitted when a token enters or exits a quest */ event QuestUpdated(uint256 indexed tokenId, address indexed tokenOwner, address indexed adventure, uint256 questId, bool active, bool booted); /** * @notice Allows an authorized game contract to transfer a player's token if they have opted in */ function adventureTransferFrom(address from, address to, uint256 tokenId) external; /** * @notice Allows an authorized game contract to safe transfer a player's token if they have opted in */ function adventureSafeTransferFrom(address from, address to, uint256 tokenId) external; /** * @notice Allows an authorized game contract to burn a player's token if they have opted in */ function adventureBurn(uint256 tokenId) external; /** * @notice Allows an authorized game contract to enter a player's token into a quest if they have opted in */ function enterQuest(uint256 tokenId, uint256 questId) external; /** * @notice Allows an authorized game contract to exit a player's token from a quest if they have opted in */ function exitQuest(uint256 tokenId, uint256 questId) external; /** * @notice Returns the number of quests a token is actively participating in for a specified adventure */ function getQuestCount(uint256 tokenId, address adventure) external view returns (uint256); /** * @notice Returns the amount of time a token has been participating in the specified quest */ function getTimeOnQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (uint256); /** * @notice Returns whether or not a token is currently participating in the specified quest as well as the time it was started and the quest index */ function isParticipatingInQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (bool participatingInQuest, uint256 startTimestamp, uint256 index); /** * @notice Returns a list of all active quests for the specified token id and adventure */ function getActiveQuests(uint256 tokenId, address adventure) external view returns (Quest[] memory activeQuests); } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./IAdventure.sol"; import "../utils/access/Ownable.sol"; import "@openzeppelin/contracts/token/ERC721/IERC721.sol"; error AdventureApprovalToCaller(); error AdventureIsStillWhitelisted(); error AlreadyWhitelisted(); error CallerNotApprovedForAdventure(); error CallerNotAWhitelistedAdventure(); error InvalidAdventureContract(); error NotWhitelisted(); /** * @title AdventureERC721Permissions * @author Limit Break, Inc. * @notice Implements the basic security features of the {IAdventurous} token standard for ERC721-compliant tokens. * This includes a whitelist for trusted Adventure contracts designed to interoperate with this token and a user * approval mechanism specific to {IAdventurous} functionality. */ abstract contract AdventurePermissions is Ownable { struct AdventureDetails { bool isWhitelisted; uint128 arrayIndex; } /// @dev Emitted when the adventure whitelist is updated event AdventureWhitelistUpdated(address indexed adventure, bool whitelisted); /// @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets, for special in-game adventures. event AdventureApprovalForAll(address indexed tokenOwner, address indexed operator, bool approved); /// @dev Whitelist array for iteration address[] public whitelistedAdventureList; /// @dev Whitelist mapping mapping (address => AdventureDetails) public whitelistedAdventures; /// @dev Mapping from owner to operator approvals for special gameplay behavior mapping (address => mapping (address => bool)) private _operatorAdventureApprovals; /// @notice Returns whether the specified account is a whitelisted adventure function isAdventureWhitelisted(address account) public view returns (bool) { return whitelistedAdventures[account].isWhitelisted; } /// @notice Whitelists an adventure and specifies whether or not the quests in that adventure lock token transfers /// Throws when the adventure is already in the whitelist. /// Throws when the specified address does not implement the IAdventure interface. /// /// Postconditions: /// The specified adventure contract is in the whitelist. /// An `AdventureWhitelistUpdate` event has been emitted. function whitelistAdventure(address adventure) external onlyOwner { if(isAdventureWhitelisted(adventure)) { revert AlreadyWhitelisted(); } if(!IERC165(adventure).supportsInterface(type(IAdventure).interfaceId)) { revert InvalidAdventureContract(); } whitelistedAdventures[adventure].isWhitelisted = true; whitelistedAdventures[adventure].arrayIndex = uint128(whitelistedAdventureList.length); whitelistedAdventureList.push(adventure); emit AdventureWhitelistUpdated(adventure, true); } /// @notice Removes an adventure from the whitelist /// Throws when the adventure is not in the whitelist. /// /// Postconditions: /// The specified adventure contract is no longer in the whitelist. /// An `AdventureWhitelistUpdate` event has been emitted. function unwhitelistAdventure(address adventure) external onlyOwner { if(!isAdventureWhitelisted(adventure)) { revert NotWhitelisted(); } uint128 itemPositionToDelete = whitelistedAdventures[adventure].arrayIndex; whitelistedAdventureList[itemPositionToDelete] = whitelistedAdventureList[whitelistedAdventureList.length - 1]; whitelistedAdventures[whitelistedAdventureList[itemPositionToDelete]].arrayIndex = itemPositionToDelete; whitelistedAdventureList.pop(); delete whitelistedAdventures[adventure]; emit AdventureWhitelistUpdated(adventure, false); } /// @notice Similar to {IERC721-setApprovalForAll}, but for special in-game adventures only function setAdventuresApprovedForAll(address operator, bool approved) public { _setAdventuresApprovedForAll(_msgSender(), operator, approved); } /// @notice Similar to {IERC721-isApprovedForAll}, but for special in-game adventures only function areAdventuresApprovedForAll(address owner, address operator) public view returns (bool) { return _operatorAdventureApprovals[owner][operator]; } /// @dev Approve `operator` to operate on all of `owner` tokens for special in-game adventures only function _setAdventuresApprovedForAll(address tokenOwner, address operator, bool approved) internal { if(tokenOwner == operator) { revert AdventureApprovalToCaller(); } _operatorAdventureApprovals[tokenOwner][operator] = approved; emit AdventureApprovalForAll(tokenOwner, operator, approved); } /// Modify to remove individual approval check /// @dev Returns whether `spender` is allowed to manage `tokenId`, for special in-game adventures only. function _isApprovedForAdventure(address spender, uint256 tokenId) internal view virtual returns (bool) { address tokenOwner = IERC721(address(this)).ownerOf(tokenId); return (areAdventuresApprovedForAll(tokenOwner, spender)); } /// @dev Validates that the caller is approved for adventure on the specified token id /// Throws when the caller has not been approved by the user. function _requireCallerApprovedForAdventure(uint256 tokenId) internal view { if(!_isApprovedForAdventure(_msgSender(), tokenId)) { revert CallerNotApprovedForAdventure(); } } /// @dev Validates that the caller is a whitelisted adventure /// Throws when the caller is not in the adventure whitelist. function _requireCallerIsWhitelistedAdventure() internal view { if(!isAdventureWhitelisted(_msgSender())) { revert CallerNotAWhitelistedAdventure(); } } /// @dev Validates that the specified adventure has been removed from the whitelist /// to prevent early backdoor exiting from adventures. /// Throws when specified adventure is still whitelisted. function _requireAdventureRemovedFromWhitelist(address adventure) internal view { if(isAdventureWhitelisted(adventure)) { revert AdventureIsStillWhitelisted(); } } } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventureERC721Initializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include Adventure ERC721 functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IAdventureERC721Initializer is IERC165 { /** * @notice Initializes parameters of {AdventureERC721} contracts */ function initializeAdventureERC721(uint256 maxSimultaneousQuests_) external; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../access/Ownable.sol"; import "../../initializable/IERC721Initializer.sol"; import "@openzeppelin/contracts/token/ERC721/ERC721.sol"; error AlreadyInitializedERC721(); /** * @title InitializableERC721 * @author Limit Break, Inc. * @notice Wraps OpenZeppelin ERC721 implementation and makes it compatible with EIP-1167. * @dev Because OpenZeppelin's `_name` and `_symbol` storage variables are private and inaccessible, * this contract defines two new storage variables `_contractName` and `_contractSymbol` and returns them * from the `name()` and `symbol()` functions instead. */ abstract contract InitializableERC721 is Ownable, ERC721, IERC721Initializer { /// @notice Specifies whether or not the contract is initialized bool public initializedERC721; // Token name string internal _contractName; // Token symbol string internal _contractSymbol; /// @dev Initializes parameters of ERC721 tokens. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeERC721(string memory name_, string memory symbol_) public override onlyOwner { if(initializedERC721) { revert AlreadyInitializedERC721(); } _contractName = name_; _contractSymbol = symbol_; initializedERC721 = true; } function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721, IERC165) returns (bool) { return interfaceId == type(IERC721Initializer).interfaceId || super.supportsInterface(interfaceId); } function name() public view virtual override returns (string memory) { return _contractName; } function symbol() public view virtual override returns (string memory) { return _contractSymbol; } } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title Quest * @author Limit Break, Inc. * @notice Quest data structure for {IAdventurous} contracts. */ struct Quest { bool isActive; uint32 questId; uint64 startTimestamp; uint32 arrayIndex; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventure * @author Limit Break, Inc. * @notice The base interface that all `Adventure` contracts must conform to. * @dev All contracts that implement the adventure/quest system and interact with an {IAdventurous} token are required to implement this interface. */ interface IAdventure is IERC165 { /** * @dev Returns whether or not quests on this adventure lock tokens. * Developers of adventure contract should ensure that this is immutable * after deployment of the adventure contract. Failure to do so * can lead to error that deadlock token transfers. */ function questsLockTokens() external view returns (bool); /** * @dev A callback function that AdventureERC721 must invoke when a quest has been successfully entered. * Throws if the caller is not an expected AdventureERC721 contract designed to work with the Adventure. * Not permitted to throw in any other case, as this could lead to tokens being locked in quests. */ function onQuestEntered(address adventurer, uint256 tokenId, uint256 questId) external; /** * @dev A callback function that AdventureERC721 must invoke when a quest has been successfully exited. * Throws if the caller is not an expected AdventureERC721 contract designed to work with the Adventure. * Not permitted to throw in any other case, as this could lead to tokens being locked in quests. */ function onQuestExited(address adventurer, uint256 tokenId, uint256 questId, uint256 questStartTimestamp) external; } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.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 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 pragma solidity ^0.8.0; import "../../initializable/IOwnableInitializer.sol"; import "@openzeppelin/contracts/utils/Context.sol"; error CallerIsNotTheContractOwner(); error NewOwnerIsTheZeroAddress(); error OwnerAlreadyInitialized(); /** * @title Ownable * @author Limit Break, Inc. and OpenZeppelin * @notice A tailored version of the {Ownable} permissions component from OpenZeppelin that is compatible with EIP-1167. * @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. * * This version adds an `initializeOwner` call for use with EIP-1167, * as the constructor will not be called during an EIP-1167 operation. * Because initializeOwner should only be called once and requires that * the owner is not assigned, the `renounceOwnership` function has been removed to avoid * a scenario where a contract takeover could occur. */ abstract contract Ownable is Context, IOwnableInitializer { address private _owner; /// @dev Emitted when contract ownership has been transferred. event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev When EIP-1167 is used to clone a contract that inherits Ownable permissions, * this is required to assign the initial contract owner, as the constructor is * not called during the cloning process. */ function initializeOwner(address owner_) public override { if(_owner != address(0)) { revert OwnerAlreadyInitialized(); } _transferOwnership(owner_); } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { if(owner() != _msgSender()) { revert CallerIsNotTheContractOwner(); } } /** * @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 override onlyOwner { if(newOwner == address(0)) { revert NewOwnerIsTheZeroAddress(); } _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 pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IOwnableInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include OpenZeppelin Ownable functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IOwnableInitializer is IERC165 { /** * @notice Initializes the contract owner to the specified address */ function initializeOwner(address owner_) external; /** * @notice Transfers ownership of the contract to the specified owner */ function transferOwnership(address newOwner) external; }// 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 pragma solidity ^0.8.0; import "@openzeppelin/contracts/token/ERC721/IERC721.sol"; /** * @title IERC721Initializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include OpenZeppelin ERC721 functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IERC721Initializer is IERC721 { /** * @notice Initializes parameters of {ERC721} contracts */ function initializeERC721(string memory name_, string memory symbol_) external; }// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.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: address zero is not a valid owner"); return _balances[owner]; } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { address owner = _owners[tokenId]; require(owner != address(0), "ERC721: invalid token ID"); return owner; } /** * @dev See {IERC721Metadata-name}. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev See {IERC721Metadata-symbol}. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { _requireMinted(tokenId); string memory baseURI = _baseURI(); return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : ""; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ""; } /** * @dev See {IERC721-approve}. */ function approve(address to, uint256 tokenId) public virtual override { address owner = ERC721.ownerOf(tokenId); require(to != owner, "ERC721: approval to current owner"); require( _msgSender() == owner || isApprovedForAll(owner, _msgSender()), "ERC721: approve caller is not token owner nor approved for all" ); _approve(to, tokenId); } /** * @dev See {IERC721-getApproved}. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { _requireMinted(tokenId); return _tokenApprovals[tokenId]; } /** * @dev See {IERC721-setApprovalForAll}. */ function setApprovalForAll(address operator, bool approved) public virtual override { _setApprovalForAll(_msgSender(), operator, approved); } /** * @dev See {IERC721-isApprovedForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev See {IERC721-transferFrom}. */ function transferFrom( address from, address to, uint256 tokenId ) public virtual override { //solhint-disable-next-line max-line-length require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner 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: caller is not token 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) { address owner = ERC721.ownerOf(tokenId); return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender); } /** * @dev Safely mints `tokenId` and transfers it to `to`. * * Requirements: * * - `tokenId` must not exist. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeMint(address to, uint256 tokenId) internal virtual { _safeMint(to, tokenId, ""); } /** * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is * forwarded in {IERC721Receiver-onERC721Received} to contract recipients. */ function _safeMint( address to, uint256 tokenId, bytes memory data ) internal virtual { _mint(to, tokenId); require( _checkOnERC721Received(address(0), to, tokenId, data), "ERC721: transfer to non ERC721Receiver implementer" ); } /** * @dev Mints `tokenId` and transfers it to `to`. * * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible * * Requirements: * * - `tokenId` must not exist. * - `to` cannot be the zero address. * * Emits a {Transfer} event. */ function _mint(address to, uint256 tokenId) internal virtual { require(to != address(0), "ERC721: mint to the zero address"); require(!_exists(tokenId), "ERC721: token already minted"); _beforeTokenTransfer(address(0), to, tokenId); _balances[to] += 1; _owners[tokenId] = to; emit Transfer(address(0), to, tokenId); _afterTokenTransfer(address(0), to, tokenId); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId) internal virtual { address owner = ERC721.ownerOf(tokenId); _beforeTokenTransfer(owner, address(0), tokenId); // Clear approvals _approve(address(0), tokenId); _balances[owner] -= 1; delete _owners[tokenId]; emit Transfer(owner, address(0), tokenId); _afterTokenTransfer(owner, address(0), tokenId); } /** * @dev Transfers `tokenId` from `from` to `to`. * As opposed to {transferFrom}, this imposes no restrictions on msg.sender. * * Requirements: * * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * * Emits a {Transfer} event. */ function _transfer( address from, address to, uint256 tokenId ) internal virtual { require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner"); require(to != address(0), "ERC721: transfer to the zero address"); _beforeTokenTransfer(from, to, tokenId); // Clear approvals from the previous owner _approve(address(0), tokenId); _balances[from] -= 1; _balances[to] += 1; _owners[tokenId] = to; emit Transfer(from, to, tokenId); _afterTokenTransfer(from, to, tokenId); } /** * @dev Approve `to` to operate on `tokenId` * * Emits an {Approval} event. */ function _approve(address to, uint256 tokenId) internal virtual { _tokenApprovals[tokenId] = to; emit Approval(ERC721.ownerOf(tokenId), to, tokenId); } /** * @dev Approve `operator` to operate on all of `owner` tokens * * Emits an {ApprovalForAll} event. */ function _setApprovalForAll( address owner, address operator, bool approved ) internal virtual { require(owner != operator, "ERC721: approve to caller"); _operatorApprovals[owner][operator] = approved; emit ApprovalForAll(owner, operator, approved); } /** * @dev Reverts if the `tokenId` has not been minted yet. */ function _requireMinted(uint256 tokenId) internal view virtual { require(_exists(tokenId), "ERC721: invalid token ID"); } /** * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address. * The call is not executed if the target address is not a contract. * * @param from address representing the previous owner of the given token ID * @param to target address that will receive the tokens * @param tokenId uint256 ID of the token to be transferred * @param data bytes optional data to send along with the call * @return bool whether the call correctly returned the expected magic value */ function _checkOnERC721Received( address from, address to, uint256 tokenId, bytes memory data ) private returns (bool) { if (to.isContract()) { try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) { return retval == IERC721Receiver.onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert("ERC721: transfer to non ERC721Receiver implementer"); } else { /// @solidity memory-safe-assembly assembly { revert(add(32, reason), mload(reason)) } } } } else { return true; } } /** * @dev Hook that is called before any token transfer. This includes minting * and burning. * * 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/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.7.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }
File 3 of 4: Achievements
// SPDX-License-Identifier: MIT pragma solidity 0.8.9; import "./IAchievements.sol"; import "limit-break-contracts/contracts/token/ERC1155/SoulboundERC1155.sol"; import "limit-break-contracts/contracts/token/ERC1155/formatters/InitializableReadOnlyURIFormatter.sol"; import "@openzeppelin/contracts/access/AccessControlEnumerable.sol"; import "@openzeppelin/contracts/proxy/Clones.sol"; error CallerDidNotReserveTheAchievementId(address caller, uint256 achievementId); error CallerDoesNotHaveAdminRole(address caller); error CallerDoesNotHaveMinterRole(address caller); error CallerDidNotReserveTheAchievementIdAndIsNotAnAdmin(address caller, uint256 achievementId); error CannotSetMetadataURIFormatterForUnreservedAchievementId(); error CannotTransferAdminRoleToSelf(); error CannotTransferAdminRoleToZeroAddress(); error IdHasNotBeenReserved(); error InputArraySizeCannotBeZero(); error MintingIdZeroIsNotAllowed(); error RevokingMinterRoleIsNotPermitted(); /** * @title Achievements * @author Limit Break, Inc. * @notice Soulbound ERC-1155 Multi-Token To Track Player Achievements */ contract Achievements is SoulboundERC1155, AccessControlEnumerable, IAchievements { /// @dev Value defining the `Minter Role`. bytes32 public constant MINTER_ROLE = keccak256("MINTER_ROLE"); /// @dev The reference implementation for default URI formatters created when reserving achievement ids address immutable public defaultURIFormatterReference; /// @dev The achievement id that was most recently reserved uint256 public lastAchievementId; /// @dev Maps an achievement id to the address of the minter that created the reservation mapping (uint256 => address) public achievementIdReservations; /// @dev Mapping from token ID to a URI formatter contract mapping (uint256 => IERC1155MetadataURIFormatter) public uriFormatters; /// @dev Emitted when a new achievement id is reserved. event ReserveAchievementId(address indexed minter, uint256 indexed achievementId); constructor() { _setupRole(DEFAULT_ADMIN_ROLE, _msgSender()); // Creates a reference implementation for URI formatters created when reserving achievement ids defaultURIFormatterReference = address(new InitializableReadOnlyURIFormatter()); } /// @notice Allows the current contract admin to transfer the `Admin Role` to a new address. /// Throws if newAdmin is the zero-address /// Throws if the caller is not the current admin. /// Throws if the caller is an admin and tries to transfer admin to itself. /// /// Postconditions: /// The new admin has been granted the `Admin Role`. /// The caller/former admin has had `Admin Role` revoked. function transferAdminRole(address newAdmin) external { if(newAdmin == address(0)) { revert CannotTransferAdminRoleToZeroAddress(); } if(newAdmin == _msgSender()) { revert CannotTransferAdminRoleToSelf(); } if(!hasRole(DEFAULT_ADMIN_ROLE, _msgSender())) { revert CallerDoesNotHaveAdminRole(_msgSender()); } _revokeRole(DEFAULT_ADMIN_ROLE, _msgSender()); _grantRole(DEFAULT_ADMIN_ROLE, newAdmin); } /// @notice Sets the metadata URI formatter for the specified token type id /// Throws if id equals 0. /// Throws if id has not yet been reserved. /// Throws when called by account that is not the contract owner or the account that reserved the id. /// Throws when the specified URI formatter does not implement the {IERC1155MetadataURIFormatter} interface. /// Throws if the specified URI formatter throws when generating a URI. /// /// Postconditions: /// The URI formatter has been linked to the token type id. /// A URI event has been emitted with the new URI of the token type id. function setMetadataURIFormatter(uint256 id, address uriFormatterAddress) external { if(id == 0 || id > lastAchievementId) { revert CannotSetMetadataURIFormatterForUnreservedAchievementId(); } if(achievementIdReservations[id] != _msgSender() && !hasRole(DEFAULT_ADMIN_ROLE, _msgSender())) { revert CallerDidNotReserveTheAchievementIdAndIsNotAnAdmin(_msgSender(), id); } if(!IERC165(uriFormatterAddress).supportsInterface(type(IERC1155MetadataURIFormatter).interfaceId)) { revert InvalidMetadataFormatterContract(); } IERC1155MetadataURIFormatter uriFormatter = IERC1155MetadataURIFormatter(uriFormatterAddress); uriFormatters[id] = uriFormatter; emit URI(uriFormatter.uri(), id); } /// @notice Reserves an achievement id and associates the achievement id with a single allowed minter. /// Initializes the URI formatter for the reserved token id using a simple, inexpensive InitializableReadOnlyURIFormatter created via cloning. /// Throws if caller has not been granted MINTER_ROLE permissions. function reserveAchievementId(string calldata metadataURI) external returns (uint256) { _requireCallerHasMinterRole(); uint256 reservedAchievementId = ++lastAchievementId; achievementIdReservations[reservedAchievementId] = _msgSender(); emit ReserveAchievementId(_msgSender(), reservedAchievementId); InitializableReadOnlyURIFormatter formatter = InitializableReadOnlyURIFormatter(Clones.clone(defaultURIFormatterReference)); formatter.initializeURI(metadataURI); uriFormatters[reservedAchievementId] = formatter; emit URI(formatter.uri(), reservedAchievementId); return reservedAchievementId; } /// @notice Mints an achievement of type `id` to the `to` address. /// Throws if the caller did not reserve the specified achievement `id` /// Throws if the specified achievement `id` has not been reserved /// Throws if attempting to mint to the zero address. function mint(address to, uint256 id, uint256 amount) external { _requireCallerReservedAchievementId(id); _mint(to, id, amount); } /// @notice Batch mints achievements to the `to` address. /// Throws if the caller did not reserve the specified achievement `id` /// Throws if the specified achievement `id` has not been reserved /// Throws if attempting to mint to the zero address. /// Throws if the ids and amounts arrays don't have the same size. function mintBatch(address to, uint256[] calldata ids, uint256[] calldata amounts) external { if(ids.length == 0 || amounts.length == 0) { revert InputArraySizeCannotBeZero(); } unchecked { for (uint256 i = 0; i < ids.length; ++i) { _requireCallerReservedAchievementId(ids[i]); } } _mintBatch(to, ids, amounts); } /// @notice Returns a distinct URI for a given token type id, generated by an external URI formatter contract /// Throws when no URI Formatter has been specified function uri(uint256 id) public view virtual override(SoulboundERC1155, IERC1155MetadataURI) returns (string memory) { IERC1155MetadataURIFormatter uriFormatter = uriFormatters[id]; if(address(uriFormatter) == address(0)) { revert NoMetadataFormatterFoundForSpecifiedId(); } return uriFormatter.uri(); } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override(SoulboundERC1155, AccessControlEnumerable, IERC165) returns (bool) { return interfaceId == type(IAchievements).interfaceId || super.supportsInterface(interfaceId); } /// @dev Validates the the caller has MINTER_ROLE function _requireCallerHasMinterRole() internal view { if(!hasRole(MINTER_ROLE, _msgSender())) { revert CallerDoesNotHaveMinterRole(_msgSender()); } } /// @dev Validates the the caller was the same account that reserved the achievement id function _requireCallerReservedAchievementId(uint256 id) internal view { if(achievementIdReservations[id] != _msgSender()) { revert CallerDidNotReserveTheAchievementId(_msgSender(), id); } } }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/token/ERC1155/extensions/IERC1155MetadataURI.sol"; /** * @title IAchievements * @author Limit Break, Inc. * @notice Interface for the Achievements token contract */ interface IAchievements is IERC1155MetadataURI { /// @dev Reserves an achievement id and associates the achievement id with a single allowed minter. function reserveAchievementId(string calldata metadataURI) external returns (uint256); /// @dev Mints an achievement of type `id` to the `to` address. function mint(address to, uint256 id, uint256 amount) external; /// @dev Batch mints achievements to the `to` address. function mintBatch(address to, uint256[] calldata ids, uint256[] calldata amounts) external; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.4; import "../IERC1155MetadataURIFormatter.sol"; import "@openzeppelin/contracts/utils/introspection/ERC165.sol"; error AlreadyInitializedURI(); /** * @title InitializableReadOnlyURIFormatter * @author Limit Break, Inc. * @notice Cloneable version of the most basic ERC-1155 Metadata URI formatter that points to an off-chain URI */ contract InitializableReadOnlyURIFormatter is ERC165, IERC1155MetadataURIFormatter { /// @notice Specifies whether or not the contract is initialized bool public initializedURI; /// @dev The read-only off-chain URI string private readOnlyURI; constructor() {} /// @dev Initializes the URI. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. /// Throws if already initialized function initializeURI(string calldata readOnlyURI_) public { if(initializedURI) { revert AlreadyInitializedURI(); } readOnlyURI = readOnlyURI_; initializedURI = true; } /// @notice Returns the off-chain URI function uri() external override view returns (string memory) { return readOnlyURI; } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) { return interfaceId == type(IERC1155MetadataURIFormatter).interfaceId || super.supportsInterface(interfaceId); } }// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (proxy/Clones.sol) pragma solidity ^0.8.0; /** * @dev https://eips.ethereum.org/EIPS/eip-1167[EIP 1167] is a standard for * deploying minimal proxy contracts, also known as "clones". * * > To simply and cheaply clone contract functionality in an immutable way, this standard specifies * > a minimal bytecode implementation that delegates all calls to a known, fixed address. * * The library includes functions to deploy a proxy using either `create` (traditional deployment) or `create2` * (salted deterministic deployment). It also includes functions to predict the addresses of clones deployed using the * deterministic method. * * _Available since v3.4._ */ library Clones { /** * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`. * * This function uses the create opcode, which should never revert. */ function clone(address implementation) internal returns (address instance) { /// @solidity memory-safe-assembly assembly { let ptr := mload(0x40) mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000) mstore(add(ptr, 0x14), shl(0x60, implementation)) mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000) instance := create(0, ptr, 0x37) } require(instance != address(0), "ERC1167: create failed"); } /** * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`. * * This function uses the create2 opcode and a `salt` to deterministically deploy * the clone. Using the same `implementation` and `salt` multiple time will revert, since * the clones cannot be deployed twice at the same address. */ function cloneDeterministic(address implementation, bytes32 salt) internal returns (address instance) { /// @solidity memory-safe-assembly assembly { let ptr := mload(0x40) mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000) mstore(add(ptr, 0x14), shl(0x60, implementation)) mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000) instance := create2(0, ptr, 0x37, salt) } require(instance != address(0), "ERC1167: create2 failed"); } /** * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}. */ function predictDeterministicAddress( address implementation, bytes32 salt, address deployer ) internal pure returns (address predicted) { /// @solidity memory-safe-assembly assembly { let ptr := mload(0x40) mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000) mstore(add(ptr, 0x14), shl(0x60, implementation)) mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf3ff00000000000000000000000000000000) mstore(add(ptr, 0x38), shl(0x60, deployer)) mstore(add(ptr, 0x4c), salt) mstore(add(ptr, 0x6c), keccak256(ptr, 0x37)) predicted := keccak256(add(ptr, 0x37), 0x55) } } /** * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}. */ function predictDeterministicAddress(address implementation, bytes32 salt) internal view returns (address predicted) { return predictDeterministicAddress(implementation, salt, address(this)); } } // SPDX-License-Identifier: MIT pragma solidity ^0.8.4; import "@openzeppelin/contracts/token/ERC1155/IERC1155.sol"; import "@openzeppelin/contracts/token/ERC1155/extensions/IERC1155MetadataURI.sol"; import "@openzeppelin/contracts/utils/Context.sol"; import "@openzeppelin/contracts/utils/introspection/ERC165.sol"; error AddressZeroIsNotAValidOwner(); error BurnAmountExceedsBalance(); error CannotBurnFromTheZeroAddress(); error CannotMintToTheZeroAddress(); error CannotSetApprovalStatusForSelf(); error IdDoesNotExist(); error InputArrayLengthMismatch(); error InvalidMetadataFormatterContract(); error NoMetadataFormatterFoundForSpecifiedId(); error SoulboundTokensAreLockedAndMayNotBeTransferred(); /** * @title SoulboundERC1155 * @author Limit Break, Inc. * @notice Base contract for fungible ERC-1155 multi-tokens that are soulbound, yet burnable. */ abstract contract SoulboundERC1155 is Context, ERC165, IERC1155, IERC1155MetadataURI { /// @dev Mapping from token ID to account balances mapping (uint256 => mapping (address => uint256)) private _balances; /// @dev Mapping from account to operator approvals mapping (address => mapping(address => bool)) private _operatorApprovals; /// @notice Approves an operator to burn, but not transfer tokens, as these are soulbound and non-transferrable function setApprovalForAll(address operator, bool approved) external virtual override { if(_msgSender() == operator) { revert CannotSetApprovalStatusForSelf(); } _operatorApprovals[_msgSender()][operator] = approved; emit ApprovalForAll(_msgSender(), operator, approved); } /// @notice Always throws - these tokens are soulbound and non-transferrable but this function is required for ERC-1155 spec compliance function safeTransferFrom(address /*from*/, address /*to*/, uint256 /*id*/, uint256 /*amount*/, bytes memory /*data*/) external override pure { revert SoulboundTokensAreLockedAndMayNotBeTransferred(); } /// @notice Always throws - these tokens are soulbound and non-transferrable but this function is required for ERC-1155 spec compliance function safeBatchTransferFrom(address /*from*/, address /*to*/, uint256[] memory /*ids*/, uint256[] memory /*amounts*/, bytes memory /*data*/) external override pure { revert SoulboundTokensAreLockedAndMayNotBeTransferred(); } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) { return interfaceId == type(IERC1155).interfaceId || interfaceId == type(IERC1155MetadataURI).interfaceId || super.supportsInterface(interfaceId); } /// @notice Returns the balance of the specified account for the specified token ID function balanceOf(address account, uint256 id) public view virtual override returns (uint256) { if(account == address(0)) { revert AddressZeroIsNotAValidOwner(); } return _balances[id][account]; } /// @notice Returns the balance of the specified accounts for the specified token IDs function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids) public view virtual override returns (uint256[] memory) { _requireInputArrayLengthsMatch(accounts.length, ids.length); uint256[] memory batchBalances = new uint256[](accounts.length); unchecked { for (uint256 i = 0; i < accounts.length; ++i) { batchBalances[i] = balanceOf(accounts[i], ids[i]); } } return batchBalances; } /// @notice Returns true if the specified operator has been approved by the specified account to burn tokens function isApprovedForAll(address account, address operator) public view virtual override returns (bool) { return _operatorApprovals[account][operator]; } /// @dev Not implemented - inheriting contracts must implement the uri scheme that is appropriate for their usage. function uri(uint256 id) public virtual view override returns (string memory); /// @dev Mints `amount` tokens of the specified `id` to the `to` address. /// Throws if attempting to mint to the zero address. function _mint(address to, uint256 id, uint256 amount) internal virtual { if(to == address(0)) { revert CannotMintToTheZeroAddress(); } _balances[id][to] += amount; emit TransferSingle(_msgSender(), address(0), to, id, amount); } /// @dev Mints a batch of `amount` tokens of the specified `ids` to the `to` address. /// Throws if attempting to mint to the zero address. /// Throws if the ids and amounts arrays don't have the same size. function _mintBatch(address to, uint256[] memory ids, uint256[] memory amounts) internal virtual { if(to == address(0)) { revert CannotMintToTheZeroAddress(); } _requireInputArrayLengthsMatch(ids.length, amounts.length); for (uint256 i = 0; i < ids.length;) { _balances[ids[i]][to] += amounts[i]; unchecked { ++i; } } emit TransferBatch(_msgSender(), address(0), to, ids, amounts); } /// @dev Burns `amount` tokens of the specified `id` from the `from` address. /// Throws if attempting to burn from the zero address. /// Throws if the amount exceeds the available balance. function _burn(address from, uint256 id, uint256 amount) internal virtual { if(from == address(0)) { revert CannotBurnFromTheZeroAddress(); } address operator = _msgSender(); uint256 fromBalance = _balances[id][from]; if(amount > fromBalance) { revert BurnAmountExceedsBalance(); } unchecked { _balances[id][from] = fromBalance - amount; } emit TransferSingle(operator, from, address(0), id, amount); } /// @dev Burns a batch of `amount` tokens of the specified `ids` from the `from` address. /// Throws if attempting to burn from the zero address. /// Throws if the ids and amounts arrays don't have the same size. /// Throws if any amount exceeds the available balance. function _burnBatch(address from, uint256[] memory ids, uint256[] memory amounts) internal virtual { if(from == address(0)) { revert CannotBurnFromTheZeroAddress(); } _requireInputArrayLengthsMatch(ids.length, amounts.length); address operator = _msgSender(); uint256 id; uint256 amount; for (uint256 i = 0; i < ids.length;) { id = ids[i]; amount = amounts[i]; uint256 fromBalance = _balances[id][from]; if(amount > fromBalance) { revert BurnAmountExceedsBalance(); } unchecked { _balances[id][from] = fromBalance - amount; ++i; } } emit TransferBatch(operator, from, address(0), ids, amounts); } /// @dev Validates that the length of two input arrays matched. /// Throws if the array lengths are mismatched. function _requireInputArrayLengthsMatch(uint256 inputArray1Length, uint256 inputArray2Length) internal pure { if(inputArray1Length != inputArray2Length) { revert InputArrayLengthMismatch(); } } } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (access/AccessControlEnumerable.sol) pragma solidity ^0.8.0; import "./IAccessControlEnumerable.sol"; import "./AccessControl.sol"; import "../utils/structs/EnumerableSet.sol"; /** * @dev Extension of {AccessControl} that allows enumerating the members of each role. */ abstract contract AccessControlEnumerable is IAccessControlEnumerable, AccessControl { using EnumerableSet for EnumerableSet.AddressSet; mapping(bytes32 => EnumerableSet.AddressSet) private _roleMembers; /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IAccessControlEnumerable).interfaceId || super.supportsInterface(interfaceId); } /** * @dev Returns one of the accounts that have `role`. `index` must be a * value between 0 and {getRoleMemberCount}, non-inclusive. * * Role bearers are not sorted in any particular way, and their ordering may * change at any point. * * WARNING: When using {getRoleMember} and {getRoleMemberCount}, make sure * you perform all queries on the same block. See the following * https://forum.openzeppelin.com/t/iterating-over-elements-on-enumerableset-in-openzeppelin-contracts/2296[forum post] * for more information. */ function getRoleMember(bytes32 role, uint256 index) public view virtual override returns (address) { return _roleMembers[role].at(index); } /** * @dev Returns the number of accounts that have `role`. Can be used * together with {getRoleMember} to enumerate all bearers of a role. */ function getRoleMemberCount(bytes32 role) public view virtual override returns (uint256) { return _roleMembers[role].length(); } /** * @dev Overload {_grantRole} to track enumerable memberships */ function _grantRole(bytes32 role, address account) internal virtual override { super._grantRole(role, account); _roleMembers[role].add(account); } /** * @dev Overload {_revokeRole} to track enumerable memberships */ function _revokeRole(bytes32 role, address account) internal virtual override { super._revokeRole(role, account); _roleMembers[role].remove(account); } } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC1155/extensions/IERC1155MetadataURI.sol) pragma solidity ^0.8.0; import "../IERC1155.sol"; /** * @dev Interface of the optional ERC1155MetadataExtension interface, as defined * in the https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[EIP]. * * _Available since v3.1._ */ interface IERC1155MetadataURI is IERC1155 { /** * @dev Returns the URI for token type `id`. * * If the `\\{id\\}` substring is present in the URI, it must be replaced by * clients with the actual token type ID. */ function uri(uint256 id) external view returns (string memory); } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC1155/IERC1155.sol) pragma solidity ^0.8.0; import "../../utils/introspection/IERC165.sol"; /** * @dev Required interface of an ERC1155 compliant contract, as defined in the * https://eips.ethereum.org/EIPS/eip-1155[EIP]. * * _Available since v3.1._ */ interface IERC1155 is IERC165 { /** * @dev Emitted when `value` tokens of token type `id` are transferred from `from` to `to` by `operator`. */ event TransferSingle(address indexed operator, address indexed from, address indexed to, uint256 id, uint256 value); /** * @dev Equivalent to multiple {TransferSingle} events, where `operator`, `from` and `to` are the same for all * transfers. */ event TransferBatch( address indexed operator, address indexed from, address indexed to, uint256[] ids, uint256[] values ); /** * @dev Emitted when `account` grants or revokes permission to `operator` to transfer their tokens, according to * `approved`. */ event ApprovalForAll(address indexed account, address indexed operator, bool approved); /** * @dev Emitted when the URI for token type `id` changes to `value`, if it is a non-programmatic URI. * * If an {URI} event was emitted for `id`, the standard * https://eips.ethereum.org/EIPS/eip-1155#metadata-extensions[guarantees] that `value` will equal the value * returned by {IERC1155MetadataURI-uri}. */ event URI(string value, uint256 indexed id); /** * @dev Returns the amount of tokens of token type `id` owned by `account`. * * Requirements: * * - `account` cannot be the zero address. */ function balanceOf(address account, uint256 id) external view returns (uint256); /** * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {balanceOf}. * * Requirements: * * - `accounts` and `ids` must have the same length. */ function balanceOfBatch(address[] calldata accounts, uint256[] calldata ids) external view returns (uint256[] memory); /** * @dev Grants or revokes permission to `operator` to transfer the caller's tokens, according to `approved`, * * Emits an {ApprovalForAll} event. * * Requirements: * * - `operator` cannot be the caller. */ function setApprovalForAll(address operator, bool approved) external; /** * @dev Returns true if `operator` is approved to transfer ``account``'s tokens. * * See {setApprovalForAll}. */ function isApprovedForAll(address account, address operator) external view returns (bool); /** * @dev Transfers `amount` tokens of token type `id` from `from` to `to`. * * Emits a {TransferSingle} event. * * Requirements: * * - `to` cannot be the zero address. * - If the caller is not `from`, it must have been approved to spend ``from``'s tokens via {setApprovalForAll}. * - `from` must have a balance of tokens of type `id` of at least `amount`. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the * acceptance magic value. */ function safeTransferFrom( address from, address to, uint256 id, uint256 amount, bytes calldata data ) external; /** * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {safeTransferFrom}. * * Emits a {TransferBatch} event. * * Requirements: * * - `ids` and `amounts` must have the same length. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the * acceptance magic value. */ function safeBatchTransferFrom( address from, address to, uint256[] calldata ids, uint256[] calldata amounts, bytes calldata data ) external; } // 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 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 pragma solidity ^0.8.4; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IERC1155MetadataURIFormatter * @author Limit Break, Inc. * @notice Interface for contracts that output ERC-1155 Metadata URIs * @dev May be used to return URIs that point off-chain, or may be used to generate URIs/metadata/image on-chain. */ interface IERC1155MetadataURIFormatter is IERC165 { /// @dev Either returns an off-chain URI or generate metadata and images on-chain. function uri() external view returns (string memory); }// 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 v4.4.1 (access/IAccessControlEnumerable.sol) pragma solidity ^0.8.0; import "./IAccessControl.sol"; /** * @dev External interface of AccessControlEnumerable declared to support ERC165 detection. */ interface IAccessControlEnumerable is IAccessControl { /** * @dev Returns one of the accounts that have `role`. `index` must be a * value between 0 and {getRoleMemberCount}, non-inclusive. * * Role bearers are not sorted in any particular way, and their ordering may * change at any point. * * WARNING: When using {getRoleMember} and {getRoleMemberCount}, make sure * you perform all queries on the same block. See the following * https://forum.openzeppelin.com/t/iterating-over-elements-on-enumerableset-in-openzeppelin-contracts/2296[forum post] * for more information. */ function getRoleMember(bytes32 role, uint256 index) external view returns (address); /** * @dev Returns the number of accounts that have `role`. Can be used * together with {getRoleMember} to enumerate all bearers of a role. */ function getRoleMemberCount(bytes32 role) external view returns (uint256); } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (access/AccessControl.sol) pragma solidity ^0.8.0; import "./IAccessControl.sol"; import "../utils/Context.sol"; import "../utils/Strings.sol"; import "../utils/introspection/ERC165.sol"; /** * @dev Contract module that allows children to implement role-based access * control mechanisms. This is a lightweight version that doesn't allow enumerating role * members except through off-chain means by accessing the contract event logs. Some * applications may benefit from on-chain enumerability, for those cases see * {AccessControlEnumerable}. * * Roles are referred to by their `bytes32` identifier. These should be exposed * in the external API and be unique. The best way to achieve this is by * using `public constant` hash digests: * * ``` * bytes32 public constant MY_ROLE = keccak256("MY_ROLE"); * ``` * * Roles can be used to represent a set of permissions. To restrict access to a * function call, use {hasRole}: * * ``` * function foo() public { * require(hasRole(MY_ROLE, msg.sender)); * ... * } * ``` * * Roles can be granted and revoked dynamically via the {grantRole} and * {revokeRole} functions. Each role has an associated admin role, and only * accounts that have a role's admin role can call {grantRole} and {revokeRole}. * * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means * that only accounts with this role will be able to grant or revoke other * roles. More complex role relationships can be created by using * {_setRoleAdmin}. * * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to * grant and revoke this role. Extra precautions should be taken to secure * accounts that have been granted it. */ abstract contract AccessControl is Context, IAccessControl, ERC165 { struct RoleData { mapping(address => bool) members; bytes32 adminRole; } mapping(bytes32 => RoleData) private _roles; bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00; /** * @dev Modifier that checks that an account has a specific role. Reverts * with a standardized message including the required role. * * The format of the revert reason is given by the following regular expression: * * /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/ * * _Available since v4.1._ */ modifier onlyRole(bytes32 role) { _checkRole(role); _; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IAccessControl).interfaceId || super.supportsInterface(interfaceId); } /** * @dev Returns `true` if `account` has been granted `role`. */ function hasRole(bytes32 role, address account) public view virtual override returns (bool) { return _roles[role].members[account]; } /** * @dev Revert with a standard message if `_msgSender()` is missing `role`. * Overriding this function changes the behavior of the {onlyRole} modifier. * * Format of the revert message is described in {_checkRole}. * * _Available since v4.6._ */ function _checkRole(bytes32 role) internal view virtual { _checkRole(role, _msgSender()); } /** * @dev Revert with a standard message if `account` is missing `role`. * * The format of the revert reason is given by the following regular expression: * * /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/ */ function _checkRole(bytes32 role, address account) internal view virtual { if (!hasRole(role, account)) { revert( string( abi.encodePacked( "AccessControl: account ", Strings.toHexString(uint160(account), 20), " is missing role ", Strings.toHexString(uint256(role), 32) ) ) ); } } /** * @dev Returns the admin role that controls `role`. See {grantRole} and * {revokeRole}. * * To change a role's admin, use {_setRoleAdmin}. */ function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) { return _roles[role].adminRole; } /** * @dev Grants `role` to `account`. * * If `account` had not been already granted `role`, emits a {RoleGranted} * event. * * Requirements: * * - the caller must have ``role``'s admin role. * * May emit a {RoleGranted} event. */ function grantRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) { _grantRole(role, account); } /** * @dev Revokes `role` from `account`. * * If `account` had been granted `role`, emits a {RoleRevoked} event. * * Requirements: * * - the caller must have ``role``'s admin role. * * May emit a {RoleRevoked} event. */ function revokeRole(bytes32 role, address account) public virtual override onlyRole(getRoleAdmin(role)) { _revokeRole(role, account); } /** * @dev Revokes `role` from the calling account. * * Roles are often managed via {grantRole} and {revokeRole}: this function's * purpose is to provide a mechanism for accounts to lose their privileges * if they are compromised (such as when a trusted device is misplaced). * * If the calling account had been revoked `role`, emits a {RoleRevoked} * event. * * Requirements: * * - the caller must be `account`. * * May emit a {RoleRevoked} event. */ function renounceRole(bytes32 role, address account) public virtual override { require(account == _msgSender(), "AccessControl: can only renounce roles for self"); _revokeRole(role, account); } /** * @dev Grants `role` to `account`. * * If `account` had not been already granted `role`, emits a {RoleGranted} * event. Note that unlike {grantRole}, this function doesn't perform any * checks on the calling account. * * May emit a {RoleGranted} event. * * [WARNING] * ==== * This function should only be called from the constructor when setting * up the initial roles for the system. * * Using this function in any other way is effectively circumventing the admin * system imposed by {AccessControl}. * ==== * * NOTE: This function is deprecated in favor of {_grantRole}. */ function _setupRole(bytes32 role, address account) internal virtual { _grantRole(role, account); } /** * @dev Sets `adminRole` as ``role``'s admin role. * * Emits a {RoleAdminChanged} event. */ function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual { bytes32 previousAdminRole = getRoleAdmin(role); _roles[role].adminRole = adminRole; emit RoleAdminChanged(role, previousAdminRole, adminRole); } /** * @dev Grants `role` to `account`. * * Internal function without access restriction. * * May emit a {RoleGranted} event. */ function _grantRole(bytes32 role, address account) internal virtual { if (!hasRole(role, account)) { _roles[role].members[account] = true; emit RoleGranted(role, account, _msgSender()); } } /** * @dev Revokes `role` from `account`. * * Internal function without access restriction. * * May emit a {RoleRevoked} event. */ function _revokeRole(bytes32 role, address account) internal virtual { if (hasRole(role, account)) { _roles[role].members[account] = false; emit RoleRevoked(role, account, _msgSender()); } } } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/structs/EnumerableSet.sol) pragma solidity ^0.8.0; /** * @dev Library for managing * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive * types. * * Sets have the following properties: * * - Elements are added, removed, and checked for existence in constant time * (O(1)). * - Elements are enumerated in O(n). No guarantees are made on the ordering. * * ``` * contract Example { * // Add the library methods * using EnumerableSet for EnumerableSet.AddressSet; * * // Declare a set state variable * EnumerableSet.AddressSet private mySet; * } * ``` * * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`) * and `uint256` (`UintSet`) are supported. * * [WARNING] * ==== * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure unusable. * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info. * * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an array of EnumerableSet. * ==== */ library EnumerableSet { // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Set type with // bytes32 values. // The Set implementation uses private functions, and user-facing // implementations (such as AddressSet) are just wrappers around the // underlying Set. // This means that we can only create new EnumerableSets for types that fit // in bytes32. struct Set { // Storage of set values bytes32[] _values; // Position of the value in the `values` array, plus 1 because index 0 // means a value is not in the set. mapping(bytes32 => uint256) _indexes; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); // The value is stored at length-1, but we add 1 to all indexes // and use 0 as a sentinel value set._indexes[value] = set._values.length; return true; } else { return false; } } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function _remove(Set storage set, bytes32 value) private returns (bool) { // We read and store the value's index to prevent multiple reads from the same storage slot uint256 valueIndex = set._indexes[value]; if (valueIndex != 0) { // Equivalent to contains(set, value) // To delete an element from the _values array in O(1), we swap the element to delete with the last one in // the array, and then remove the last element (sometimes called as 'swap and pop'). // This modifies the order of the array, as noted in {at}. uint256 toDeleteIndex = valueIndex - 1; uint256 lastIndex = set._values.length - 1; if (lastIndex != toDeleteIndex) { bytes32 lastValue = set._values[lastIndex]; // Move the last value to the index where the value to delete is set._values[toDeleteIndex] = lastValue; // Update the index for the moved value set._indexes[lastValue] = valueIndex; // Replace lastValue's index to valueIndex } // Delete the slot where the moved value was stored set._values.pop(); // Delete the index for the deleted slot delete set._indexes[value]; return true; } else { return false; } } /** * @dev Returns true if the value is in the set. O(1). */ function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._indexes[value] != 0; } /** * @dev Returns the number of values on the set. O(1). */ function _length(Set storage set) private view returns (uint256) { return set._values.length; } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // Bytes32Set struct Bytes32Set { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _add(set._inner, value); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _remove(set._inner, value); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) { return _contains(set._inner, value); } /** * @dev Returns the number of values in the set. O(1). */ function length(Bytes32Set storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) { return _at(set._inner, index); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(Bytes32Set storage set) internal view returns (bytes32[] memory) { return _values(set._inner); } // AddressSet struct AddressSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // UintSet struct UintSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(UintSet storage set, uint256 value) internal returns (bool) { return _add(set._inner, bytes32(value)); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(UintSet storage set, uint256 value) internal returns (bool) { return _remove(set._inner, bytes32(value)); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(UintSet storage set, uint256 value) internal view returns (bool) { return _contains(set._inner, bytes32(value)); } /** * @dev Returns the number of values on the set. O(1). */ function length(UintSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintSet storage set, uint256 index) internal view returns (uint256) { return uint256(_at(set._inner, index)); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(UintSet storage set) internal view returns (uint256[] memory) { bytes32[] memory store = _values(set._inner); uint256[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol) pragma solidity ^0.8.0; /** * @dev External interface of AccessControl declared to support ERC165 detection. */ interface IAccessControl { /** * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole` * * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite * {RoleAdminChanged} not being emitted signaling this. * * _Available since v3.1._ */ event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole); /** * @dev Emitted when `account` is granted `role`. * * `sender` is the account that originated the contract call, an admin role * bearer except when using {AccessControl-_setupRole}. */ event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender); /** * @dev Emitted when `account` is revoked `role`. * * `sender` is the account that originated the contract call: * - if using `revokeRole`, it is the admin role bearer * - if using `renounceRole`, it is the role bearer (i.e. `account`) */ event RoleRevoked(bytes32 indexed role, address indexed account, address indexed sender); /** * @dev Returns `true` if `account` has been granted `role`. */ function hasRole(bytes32 role, address account) external view returns (bool); /** * @dev Returns the admin role that controls `role`. See {grantRole} and * {revokeRole}. * * To change a role's admin, use {AccessControl-_setRoleAdmin}. */ function getRoleAdmin(bytes32 role) external view returns (bytes32); /** * @dev Grants `role` to `account`. * * If `account` had not been already granted `role`, emits a {RoleGranted} * event. * * Requirements: * * - the caller must have ``role``'s admin role. */ function grantRole(bytes32 role, address account) external; /** * @dev Revokes `role` from `account`. * * If `account` had been granted `role`, emits a {RoleRevoked} event. * * Requirements: * * - the caller must have ``role``'s admin role. */ function revokeRole(bytes32 role, address account) external; /** * @dev Revokes `role` from the calling account. * * Roles are often managed via {grantRole} and {revokeRole}: this function's * purpose is to provide a mechanism for accounts to lose their privileges * if they are compromised (such as when a trusted device is misplaced). * * If the calling account had been granted `role`, emits a {RoleRevoked} * event. * * Requirements: * * - the caller must be `account`. */ function renounceRole(bytes32 role, address account) external; } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }
File 4 of 4: AirdroppableAdventureKey
// SPDX-License-Identifier: MIT pragma solidity 0.8.9; import "limit-break-contracts/contracts/adventures/AdventureNFT.sol"; import "limit-break-contracts/contracts/initializable/IMaxSupplyInitializer.sol"; error MaxSupplyAlreadyInitialized(); error MaxSupplyCannotBeSetToZero(); error MaxSupplyExceeded(uint256 supplyAfterMint, uint256 maxSupply); /** * @title AirdroppableAdventureKey * @author Limit Break, Inc. * @notice An adventure key reference contract that can be airdropped and cloned using EIP-1167. * See https://eips.ethereum.org/EIPS/eip-1167 for details. */ contract AirdroppableAdventureKey is AdventureNFT, IMaxSupplyInitializer { uint256 private nextTokenId; /// @dev The maximum token supply uint256 public maxSupply; constructor() ERC721("", "") {} /// @dev Initializes parameters of tokens with maximum supplies. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeMaxSupply(uint256 maxSupply_) public override onlyOwner { if(maxSupply > 0) { revert MaxSupplyAlreadyInitialized(); } if(maxSupply_ == 0) { revert MaxSupplyCannotBeSetToZero(); } maxSupply = maxSupply_; } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override(AdventureNFT, IERC165) returns (bool) { return interfaceId == type(IMaxSupplyInitializer).interfaceId || super.supportsInterface(interfaceId); } /// @notice Owner bulk mint to airdrop function airdropMint(address[] calldata to) external onlyOwner { if(nextTokenId == 0) { nextTokenId = 1; } uint256 batchSize = to.length; uint256 tokenIdToMint = nextTokenId; uint256 supplyAfterMint = tokenIdToMint + batchSize - 1; uint256 maxSupply_ = maxSupply; if(supplyAfterMint > maxSupply_) { revert MaxSupplyExceeded(supplyAfterMint, maxSupply_); } nextTokenId = nextTokenId + batchSize; unchecked { for(uint256 i = 0; i < batchSize; ++i) { _mint(to[i], tokenIdToMint + i); } } } }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./AdventureERC721.sol"; import "../initializable/IRoyaltiesInitializer.sol"; import "../initializable/IURIInitializer.sol"; import "@openzeppelin/contracts/token/common/ERC2981.sol"; error AlreadyInitializedRoyalties(); error AlreadyInitializedURI(); error ExceedsMaxRoyaltyFee(); error NonexistentToken(); /** * @title AdventureNFT * @author Limit Break, Inc. * @notice Standardizes commonly shared boilerplate code that adds base/suffix URI and EIP-2981 royalties to {AdventureERC721} contracts. */ abstract contract AdventureNFT is AdventureERC721, ERC2981, IRoyaltiesInitializer, IURIInitializer { using Strings for uint256; /// @dev The maximum allowable royalty fee is 10% uint96 public constant MAX_ROYALTY_FEE_NUMERATOR = 1000; /// @notice Specifies whether or not the contract is initialized bool public initializedRoyalties; /// @notice Specifies whether or not the contract is initialized bool public initializedURI; /// @dev Base token uri string public baseTokenURI; /// @dev Token uri suffix/extension string public suffixURI = ".json"; /// @dev Emitted when base URI is set. event BaseURISet(string baseTokenURI); /// @dev Emitted when suffix URI is set. event SuffixURISet(string suffixURI); /// @dev Emitted when royalty is set. event RoyaltySet(address receiver, uint96 feeNumerator); /// @dev Initializes parameters of tokens with royalties. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeRoyalties(address receiver, uint96 feeNumerator) public override onlyOwner { if(initializedRoyalties) { revert AlreadyInitializedRoyalties(); } setRoyaltyInfo(receiver, feeNumerator); initializedRoyalties = true; } /// @dev Initializes parameters of tokens with uri values. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeURI(string memory baseURI_, string memory suffixURI_) public override onlyOwner { if(initializedURI) { revert AlreadyInitializedURI(); } setBaseURI(baseURI_); setSuffixURI(suffixURI_); initializedURI = true; } /// @dev Required to return baseTokenURI for tokenURI function _baseURI() internal view virtual override returns (string memory) { return baseTokenURI; } /// @notice Sets base URI function setBaseURI(string memory baseTokenURI_) public onlyOwner { baseTokenURI = baseTokenURI_; emit BaseURISet(baseTokenURI_); } /// @notice Sets suffix URI function setSuffixURI(string memory suffixURI_) public onlyOwner { suffixURI = suffixURI_; emit SuffixURISet(suffixURI_); } /// @notice Sets royalty information function setRoyaltyInfo(address receiver, uint96 feeNumerator) public onlyOwner { if(feeNumerator > MAX_ROYALTY_FEE_NUMERATOR) { revert ExceedsMaxRoyaltyFee(); } _setDefaultRoyalty(receiver, feeNumerator); emit RoyaltySet(receiver, feeNumerator); } /// @notice Returns tokenURI if baseURI is set function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if(!_exists(tokenId)) { revert NonexistentToken(); } string memory baseURI = _baseURI(); return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString(), suffixURI)) : ""; } function supportsInterface(bytes4 interfaceId) public view virtual override (AdventureERC721, ERC2981, IERC165) returns (bool) { return interfaceId == type(IRoyaltiesInitializer).interfaceId || interfaceId == type(IURIInitializer).interfaceId || super.supportsInterface(interfaceId); } }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IMaxSupplyInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include a maximum supply. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IMaxSupplyInitializer is IERC165 { /** * @notice Initializes max supply parameters */ function initializeMaxSupply(uint256 maxSupply_) external; }// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.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: * * - `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 pragma solidity ^0.8.0; import "./IAdventurous.sol"; import "./AdventurePermissions.sol"; import "../initializable/IAdventureERC721Initializer.sol"; import "../utils/tokens/InitializableERC721.sol"; error AlreadyInitializedAdventureERC721(); error AlreadyOnQuest(); error AnActiveQuestIsPreventingTransfers(); error CallerNotTokenOwner(); error MaxSimultaneousQuestsCannotBeZero(); error MaxSimultaneousQuestsExceeded(); error NotOnQuest(); error QuestIdOutOfRange(); error TooManyActiveQuests(); /** * @title AdventureERC721 * @author Limit Break, Inc. * @notice Implements the {IAdventurous} token standard for ERC721-compliant tokens. * @dev Inherits {InitializableERC721} to provide the option to support EIP-1167. */ abstract contract AdventureERC721 is InitializableERC721, AdventurePermissions, IAdventurous, IAdventureERC721Initializer { uint256 public constant MAX_UINT32 = type(uint32).max; /// @notice Specifies an upper bound for the maximum number of simultaneous quests. uint256 public constant MAX_CONCURRENT_QUESTS = 100; /// @notice Specifies whether or not the contract is initialized bool public initializedAdventureERC721; /// @dev The most simultaneous quests the token may participate in at a time uint256 public maxSimultaneousQuests; /// @dev Maps each token id to a mapping that can enumerate all active quests within an adventure mapping (uint256 => mapping (address => uint32[])) public activeQuestList; /// @dev Maps each token id to a mapping from adventure address to a mapping of quest ids to quest details mapping (uint256 => mapping (address => mapping (uint32 => Quest))) public activeQuestLookup; /// @dev Maps each token id to the number of blocking quests it is currently entered into mapping (uint256 => uint256) public blockingQuestCounts; /// @dev Initializes parameters of AdventureERC721 tokens. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeAdventureERC721(uint256 maxSimultaneousQuests_) public override onlyOwner { if(initializedAdventureERC721) { revert AlreadyInitializedAdventureERC721(); } _validateMaxSimultaneousQuests(maxSimultaneousQuests_); maxSimultaneousQuests = maxSimultaneousQuests_; initializedAdventureERC721 = true; } /// @dev ERC-165 interface support function supportsInterface(bytes4 interfaceId) public view virtual override (InitializableERC721, IERC165) returns (bool) { return interfaceId == type(IAdventurous).interfaceId || interfaceId == type(IAdventureERC721Initializer).interfaceId || super.supportsInterface(interfaceId); } /// @notice Allows an authorized game contract to transfer a player's token if they have opted in function adventureTransferFrom(address from, address to, uint256 tokenId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _transfer(from, to, tokenId); } /// @notice Allows an authorized game contract to transfer a player's token if they have opted in function adventureSafeTransferFrom(address from, address to, uint256 tokenId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _safeTransfer(from, to, tokenId, ""); } /// @notice Allows an authorized game contract to burn a player's token if they have opted in function adventureBurn(uint256 tokenId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _burn(tokenId); } /// @notice Allows an authorized game contract to enter a player's token into a quest if they have opted in function enterQuest(uint256 tokenId, uint256 questId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _enterQuest(tokenId, _msgSender(), questId); } /// @notice Allows an authorized game contract to exit a player's token from a quest if they have opted in /// For developers of adventure contracts that perform adventure burns, be aware that the adventure must exitQuest /// before the adventure burn occurs, as _exitQuest emits the owner of the token, which would revert after burning. function exitQuest(uint256 tokenId, uint256 questId) external override { _requireCallerIsWhitelistedAdventure(); _requireCallerApprovedForAdventure(tokenId); _exitQuest(tokenId, _msgSender(), questId); } /// @notice Admin-only ability to boot a token from all quests on an adventure. /// This ability is only unlocked in the event that an adventure has been unwhitelisted, as early exiting /// from quests can cause out of sync state between the ERC721 token contract and the adventure/quest. function bootFromAllQuests(uint256 tokenId, address adventure) external onlyOwner { _requireAdventureRemovedFromWhitelist(adventure); _exitAllQuests(tokenId, adventure, true); } /// @notice Gives the player the ability to exit a quest without interacting directly with the approved, whitelisted adventure /// This ability is only unlocked in the event that an adventure has been unwhitelisted, as early exiting /// from quests can cause out of sync state between the ERC721 token contract and the adventure/quest. function userExitQuest(uint256 tokenId, address adventure, uint256 questId) external { _requireAdventureRemovedFromWhitelist(adventure); _requireCallerOwnsToken(tokenId); _exitQuest(tokenId, adventure, questId); } /// @notice Gives the player the ability to exit all quests on an adventure without interacting directly with the approved, whitelisted adventure /// This ability is only unlocked in the event that an adventure has been unwhitelisted, as early exiting /// from quests can cause out of sync state between the ERC721 token contract and the adventure/quest. function userExitAllQuests(uint256 tokenId, address adventure) external { _requireAdventureRemovedFromWhitelist(adventure); _requireCallerOwnsToken(tokenId); _exitAllQuests(tokenId, adventure, false); } /// @notice Returns the number of quests a token is actively participating in for a specified adventure function getQuestCount(uint256 tokenId, address adventure) public override view returns (uint256) { return activeQuestList[tokenId][adventure].length; } /// @notice Returns the amount of time a token has been participating in the specified quest function getTimeOnQuest(uint256 tokenId, address adventure, uint256 questId) public override view returns (uint256) { (bool participatingInQuest, uint256 startTimestamp,) = isParticipatingInQuest(tokenId, adventure, questId); return participatingInQuest ? (block.timestamp - startTimestamp) : 0; } /// @notice Returns whether or not a token is currently participating in the specified quest as well as the time it was started and the quest index function isParticipatingInQuest(uint256 tokenId, address adventure, uint256 questId) public override view returns (bool participatingInQuest, uint256 startTimestamp, uint256 index) { Quest memory quest = activeQuestLookup[tokenId][adventure][uint32(questId)]; participatingInQuest = quest.isActive; startTimestamp = quest.startTimestamp; index = quest.arrayIndex; return (participatingInQuest, startTimestamp, index); } /// @notice Returns a list of all active quests for the specified token id and adventure function getActiveQuests(uint256 tokenId, address adventure) public override view returns (Quest[] memory activeQuests) { uint256 questCount = getQuestCount(tokenId, adventure); activeQuests = new Quest[](questCount); uint32[] memory activeQuestIdList = activeQuestList[tokenId][adventure]; for(uint256 i = 0; i < questCount; ++i) { activeQuests[i] = activeQuestLookup[tokenId][adventure][activeQuestIdList[i]]; } return activeQuests; } /// @dev Enters the specified quest for a token id. /// Throws if the token is already participating in the specified quest. /// Throws if the number of active quests exceeds the max allowable for the given adventure. /// Emits a QuestUpdated event for off-chain processing. function _enterQuest(uint256 tokenId, address adventure, uint256 questId) internal { _requireValidQuestId(questId); (bool participatingInQuest,,) = isParticipatingInQuest(tokenId, adventure, questId); if(participatingInQuest) { revert AlreadyOnQuest(); } uint256 currentQuestCount = getQuestCount(tokenId, adventure); if(currentQuestCount == maxSimultaneousQuests) { revert TooManyActiveQuests(); } uint32 castedQuestId = uint32(questId); activeQuestList[tokenId][adventure].push(castedQuestId); activeQuestLookup[tokenId][adventure][castedQuestId] = Quest({ isActive: true, startTimestamp: uint64(block.timestamp), questId: castedQuestId, arrayIndex: uint32(currentQuestCount) }); address ownerOfToken = ownerOf(tokenId); emit QuestUpdated(tokenId, ownerOfToken, adventure, questId, true, false); if(IAdventure(adventure).questsLockTokens()) { unchecked { ++blockingQuestCounts[tokenId]; } } // Invoke callback to the adventure to facilitate state synchronization as needed IAdventure(adventure).onQuestEntered(ownerOfToken, tokenId, questId); } /// @dev Exits the specified quest for a token id. /// Throws if the token is not currently participating on the specified quest. /// Emits a QuestUpdated event for off-chain processing. function _exitQuest(uint256 tokenId, address adventure, uint256 questId) internal { _requireValidQuestId(questId); (bool participatingInQuest, uint256 startTimestamp, uint256 index) = isParticipatingInQuest(tokenId, adventure, questId); if(!participatingInQuest) { revert NotOnQuest(); } uint32 castedQuestId = uint32(questId); uint256 lastArrayIndex = getQuestCount(tokenId, adventure) - 1; activeQuestList[tokenId][adventure][index] = activeQuestList[tokenId][adventure][lastArrayIndex]; activeQuestLookup[tokenId][adventure][activeQuestList[tokenId][adventure][lastArrayIndex]].arrayIndex = uint32(index); activeQuestList[tokenId][adventure].pop(); delete activeQuestLookup[tokenId][adventure][castedQuestId]; address ownerOfToken = ownerOf(tokenId); emit QuestUpdated(tokenId, ownerOfToken, adventure, questId, false, false); if(IAdventure(adventure).questsLockTokens()) { --blockingQuestCounts[tokenId]; } // Invoke callback to the adventure to facilitate state synchronization as needed IAdventure(adventure).onQuestExited(ownerOfToken, tokenId, questId, startTimestamp); } /// @dev Removes the specified token id from all quests on the specified adventure function _exitAllQuests(uint256 tokenId, address adventure, bool booted) internal { address tokenOwner = ownerOf(tokenId); uint256 questCount = getQuestCount(tokenId, adventure); if(IAdventure(adventure).questsLockTokens()) { blockingQuestCounts[tokenId] -= questCount; } for(uint256 i = 0; i < questCount; ++i) { uint256 questId = activeQuestList[tokenId][adventure][i]; Quest memory quest = activeQuestLookup[tokenId][adventure][uint32(questId)]; uint256 startTimestamp = quest.startTimestamp; emit QuestUpdated(tokenId, tokenOwner, adventure, questId, false, booted); delete activeQuestLookup[tokenId][adventure][uint32(questId)]; // Invoke callback to the adventure to facilitate state synchronization as needed IAdventure(adventure).onQuestExited(tokenOwner, tokenId, questId, startTimestamp); } delete activeQuestList[tokenId][adventure]; } /// @dev By default, tokens that are participating in quests are transferrable. However, if a token is participating /// in a quest on an adventure that was designated as a token locker, the transfer will revert and keep the token /// locked. function _beforeTokenTransfer(address /*from*/, address /*to*/, uint256 tokenId) internal virtual override { if(blockingQuestCounts[tokenId] > 0) { revert AnActiveQuestIsPreventingTransfers(); } } /// @dev Validates that the caller owns the specified token /// Throws when the caller does not own the specified token. function _requireCallerOwnsToken(uint256 tokenId) internal view { if(ownerOf(tokenId) != _msgSender()) { revert CallerNotTokenOwner(); } } /// @dev Validates that the specified quest id does not overflow a uint32 /// Throws when questId exceeds the largest uint32 value. function _requireValidQuestId(uint256 questId) internal pure { if(questId > MAX_UINT32) { revert QuestIdOutOfRange(); } } /// @dev Validates that the specified value of max simultaneous quests is in range [1-MAX_CONCURRENT_QUESTS] /// Throws when `maxSimultaneousQuests_` is zero. /// Throws when `maxSimultaneousQuests_` is more than MAX_CONCURRENT_QUESTS. function _validateMaxSimultaneousQuests(uint256 maxSimultaneousQuests_) internal pure { if(maxSimultaneousQuests_ == 0) { revert MaxSimultaneousQuestsCannotBeZero(); } if(maxSimultaneousQuests_ > MAX_CONCURRENT_QUESTS) { revert MaxSimultaneousQuestsExceeded(); } } } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IURIInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include a base uri and suffix uri. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IURIInitializer is IERC165 { /** * @notice Initializes uri parameters */ function initializeURI(string memory baseURI_, string memory suffixURI_) external; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IRoyaltiesInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include OpenZeppelin ERC2981 functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IRoyaltiesInitializer is IERC165 { /** * @notice Initializes royalty parameters */ function initializeRoyalties(address receiver, uint96 feeNumerator) external; }// 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/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 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 pragma solidity ^0.8.0; import "./Quest.sol"; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventurous * @author Limit Break, Inc. * @notice The base interface that all `Adventurous` token contracts must conform to in order to support adventures and quests. * @dev All contracts that support adventures and quests are required to implement this interface. */ interface IAdventurous is IERC165 { /** * @dev Emitted when a token enters or exits a quest */ event QuestUpdated(uint256 indexed tokenId, address indexed tokenOwner, address indexed adventure, uint256 questId, bool active, bool booted); /** * @notice Allows an authorized game contract to transfer a player's token if they have opted in */ function adventureTransferFrom(address from, address to, uint256 tokenId) external; /** * @notice Allows an authorized game contract to safe transfer a player's token if they have opted in */ function adventureSafeTransferFrom(address from, address to, uint256 tokenId) external; /** * @notice Allows an authorized game contract to burn a player's token if they have opted in */ function adventureBurn(uint256 tokenId) external; /** * @notice Allows an authorized game contract to enter a player's token into a quest if they have opted in */ function enterQuest(uint256 tokenId, uint256 questId) external; /** * @notice Allows an authorized game contract to exit a player's token from a quest if they have opted in */ function exitQuest(uint256 tokenId, uint256 questId) external; /** * @notice Returns the number of quests a token is actively participating in for a specified adventure */ function getQuestCount(uint256 tokenId, address adventure) external view returns (uint256); /** * @notice Returns the amount of time a token has been participating in the specified quest */ function getTimeOnQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (uint256); /** * @notice Returns whether or not a token is currently participating in the specified quest as well as the time it was started and the quest index */ function isParticipatingInQuest(uint256 tokenId, address adventure, uint256 questId) external view returns (bool participatingInQuest, uint256 startTimestamp, uint256 index); /** * @notice Returns a list of all active quests for the specified token id and adventure */ function getActiveQuests(uint256 tokenId, address adventure) external view returns (Quest[] memory activeQuests); } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "./IAdventure.sol"; import "../utils/access/Ownable.sol"; import "@openzeppelin/contracts/token/ERC721/IERC721.sol"; error AdventureApprovalToCaller(); error AdventureIsStillWhitelisted(); error AlreadyWhitelisted(); error CallerNotApprovedForAdventure(); error CallerNotAWhitelistedAdventure(); error InvalidAdventureContract(); error NotWhitelisted(); /** * @title AdventureERC721Permissions * @author Limit Break, Inc. * @notice Implements the basic security features of the {IAdventurous} token standard for ERC721-compliant tokens. * This includes a whitelist for trusted Adventure contracts designed to interoperate with this token and a user * approval mechanism specific to {IAdventurous} functionality. */ abstract contract AdventurePermissions is Ownable { struct AdventureDetails { bool isWhitelisted; uint128 arrayIndex; } /// @dev Emitted when the adventure whitelist is updated event AdventureWhitelistUpdated(address indexed adventure, bool whitelisted); /// @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets, for special in-game adventures. event AdventureApprovalForAll(address indexed tokenOwner, address indexed operator, bool approved); /// @dev Whitelist array for iteration address[] public whitelistedAdventureList; /// @dev Whitelist mapping mapping (address => AdventureDetails) public whitelistedAdventures; /// @dev Mapping from owner to operator approvals for special gameplay behavior mapping (address => mapping (address => bool)) private _operatorAdventureApprovals; /// @notice Returns whether the specified account is a whitelisted adventure function isAdventureWhitelisted(address account) public view returns (bool) { return whitelistedAdventures[account].isWhitelisted; } /// @notice Whitelists an adventure and specifies whether or not the quests in that adventure lock token transfers /// Throws when the adventure is already in the whitelist. /// Throws when the specified address does not implement the IAdventure interface. /// /// Postconditions: /// The specified adventure contract is in the whitelist. /// An `AdventureWhitelistUpdate` event has been emitted. function whitelistAdventure(address adventure) external onlyOwner { if(isAdventureWhitelisted(adventure)) { revert AlreadyWhitelisted(); } if(!IERC165(adventure).supportsInterface(type(IAdventure).interfaceId)) { revert InvalidAdventureContract(); } whitelistedAdventures[adventure].isWhitelisted = true; whitelistedAdventures[adventure].arrayIndex = uint128(whitelistedAdventureList.length); whitelistedAdventureList.push(adventure); emit AdventureWhitelistUpdated(adventure, true); } /// @notice Removes an adventure from the whitelist /// Throws when the adventure is not in the whitelist. /// /// Postconditions: /// The specified adventure contract is no longer in the whitelist. /// An `AdventureWhitelistUpdate` event has been emitted. function unwhitelistAdventure(address adventure) external onlyOwner { if(!isAdventureWhitelisted(adventure)) { revert NotWhitelisted(); } uint128 itemPositionToDelete = whitelistedAdventures[adventure].arrayIndex; whitelistedAdventureList[itemPositionToDelete] = whitelistedAdventureList[whitelistedAdventureList.length - 1]; whitelistedAdventures[whitelistedAdventureList[itemPositionToDelete]].arrayIndex = itemPositionToDelete; whitelistedAdventureList.pop(); delete whitelistedAdventures[adventure]; emit AdventureWhitelistUpdated(adventure, false); } /// @notice Similar to {IERC721-setApprovalForAll}, but for special in-game adventures only function setAdventuresApprovedForAll(address operator, bool approved) public { _setAdventuresApprovedForAll(_msgSender(), operator, approved); } /// @notice Similar to {IERC721-isApprovedForAll}, but for special in-game adventures only function areAdventuresApprovedForAll(address owner, address operator) public view returns (bool) { return _operatorAdventureApprovals[owner][operator]; } /// @dev Approve `operator` to operate on all of `owner` tokens for special in-game adventures only function _setAdventuresApprovedForAll(address tokenOwner, address operator, bool approved) internal { if(tokenOwner == operator) { revert AdventureApprovalToCaller(); } _operatorAdventureApprovals[tokenOwner][operator] = approved; emit AdventureApprovalForAll(tokenOwner, operator, approved); } /// Modify to remove individual approval check /// @dev Returns whether `spender` is allowed to manage `tokenId`, for special in-game adventures only. function _isApprovedForAdventure(address spender, uint256 tokenId) internal view virtual returns (bool) { address tokenOwner = IERC721(address(this)).ownerOf(tokenId); return (areAdventuresApprovedForAll(tokenOwner, spender)); } /// @dev Validates that the caller is approved for adventure on the specified token id /// Throws when the caller has not been approved by the user. function _requireCallerApprovedForAdventure(uint256 tokenId) internal view { if(!_isApprovedForAdventure(_msgSender(), tokenId)) { revert CallerNotApprovedForAdventure(); } } /// @dev Validates that the caller is a whitelisted adventure /// Throws when the caller is not in the adventure whitelist. function _requireCallerIsWhitelistedAdventure() internal view { if(!isAdventureWhitelisted(_msgSender())) { revert CallerNotAWhitelistedAdventure(); } } /// @dev Validates that the specified adventure has been removed from the whitelist /// to prevent early backdoor exiting from adventures. /// Throws when specified adventure is still whitelisted. function _requireAdventureRemovedFromWhitelist(address adventure) internal view { if(isAdventureWhitelisted(adventure)) { revert AdventureIsStillWhitelisted(); } } } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventureERC721Initializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include Adventure ERC721 functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IAdventureERC721Initializer is IERC165 { /** * @notice Initializes parameters of {AdventureERC721} contracts */ function initializeAdventureERC721(uint256 maxSimultaneousQuests_) external; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../access/Ownable.sol"; import "../../initializable/IERC721Initializer.sol"; import "@openzeppelin/contracts/token/ERC721/ERC721.sol"; error AlreadyInitializedERC721(); /** * @title InitializableERC721 * @author Limit Break, Inc. * @notice Wraps OpenZeppelin ERC721 implementation and makes it compatible with EIP-1167. * @dev Because OpenZeppelin's `_name` and `_symbol` storage variables are private and inaccessible, * this contract defines two new storage variables `_contractName` and `_contractSymbol` and returns them * from the `name()` and `symbol()` functions instead. */ abstract contract InitializableERC721 is Ownable, ERC721, IERC721Initializer { /// @notice Specifies whether or not the contract is initialized bool public initializedERC721; // Token name string internal _contractName; // Token symbol string internal _contractSymbol; /// @dev Initializes parameters of ERC721 tokens. /// These cannot be set in the constructor because this contract is optionally compatible with EIP-1167. function initializeERC721(string memory name_, string memory symbol_) public override onlyOwner { if(initializedERC721) { revert AlreadyInitializedERC721(); } _contractName = name_; _contractSymbol = symbol_; initializedERC721 = true; } function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721, IERC165) returns (bool) { return interfaceId == type(IERC721Initializer).interfaceId || super.supportsInterface(interfaceId); } function name() public view virtual override returns (string memory) { return _contractName; } function symbol() public view virtual override returns (string memory) { return _contractSymbol; } } // SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @title Quest * @author Limit Break, Inc. * @notice Quest data structure for {IAdventurous} contracts. */ struct Quest { bool isActive; uint32 questId; uint64 startTimestamp; uint32 arrayIndex; }// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IAdventure * @author Limit Break, Inc. * @notice The base interface that all `Adventure` contracts must conform to. * @dev All contracts that implement the adventure/quest system and interact with an {IAdventurous} token are required to implement this interface. */ interface IAdventure is IERC165 { /** * @dev Returns whether or not quests on this adventure lock tokens. * Developers of adventure contract should ensure that this is immutable * after deployment of the adventure contract. Failure to do so * can lead to error that deadlock token transfers. */ function questsLockTokens() external view returns (bool); /** * @dev A callback function that AdventureERC721 must invoke when a quest has been successfully entered. * Throws if the caller is not an expected AdventureERC721 contract designed to work with the Adventure. * Not permitted to throw in any other case, as this could lead to tokens being locked in quests. */ function onQuestEntered(address adventurer, uint256 tokenId, uint256 questId) external; /** * @dev A callback function that AdventureERC721 must invoke when a quest has been successfully exited. * Throws if the caller is not an expected AdventureERC721 contract designed to work with the Adventure. * Not permitted to throw in any other case, as this could lead to tokens being locked in quests. */ function onQuestExited(address adventurer, uint256 tokenId, uint256 questId, uint256 questStartTimestamp) external; } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.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 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 pragma solidity ^0.8.0; import "../../initializable/IOwnableInitializer.sol"; import "@openzeppelin/contracts/utils/Context.sol"; error CallerIsNotTheContractOwner(); error NewOwnerIsTheZeroAddress(); error OwnerAlreadyInitialized(); /** * @title Ownable * @author Limit Break, Inc. and OpenZeppelin * @notice A tailored version of the {Ownable} permissions component from OpenZeppelin that is compatible with EIP-1167. * @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. * * This version adds an `initializeOwner` call for use with EIP-1167, * as the constructor will not be called during an EIP-1167 operation. * Because initializeOwner should only be called once and requires that * the owner is not assigned, the `renounceOwnership` function has been removed to avoid * a scenario where a contract takeover could occur. */ abstract contract Ownable is Context, IOwnableInitializer { address private _owner; /// @dev Emitted when contract ownership has been transferred. event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev When EIP-1167 is used to clone a contract that inherits Ownable permissions, * this is required to assign the initial contract owner, as the constructor is * not called during the cloning process. */ function initializeOwner(address owner_) public override { if(_owner != address(0)) { revert OwnerAlreadyInitialized(); } _transferOwnership(owner_); } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { if(owner() != _msgSender()) { revert CallerIsNotTheContractOwner(); } } /** * @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 override onlyOwner { if(newOwner == address(0)) { revert NewOwnerIsTheZeroAddress(); } _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 pragma solidity ^0.8.0; import "@openzeppelin/contracts/utils/introspection/IERC165.sol"; /** * @title IOwnableInitializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include OpenZeppelin Ownable functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IOwnableInitializer is IERC165 { /** * @notice Initializes the contract owner to the specified address */ function initializeOwner(address owner_) external; /** * @notice Transfers ownership of the contract to the specified owner */ function transferOwnership(address newOwner) external; }// 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 pragma solidity ^0.8.0; import "@openzeppelin/contracts/token/ERC721/IERC721.sol"; /** * @title IERC721Initializer * @author Limit Break, Inc. * @notice Allows cloneable contracts to include OpenZeppelin ERC721 functionality. * @dev See https://eips.ethereum.org/EIPS/eip-1167 for details. */ interface IERC721Initializer is IERC721 { /** * @notice Initializes parameters of {ERC721} contracts */ function initializeERC721(string memory name_, string memory symbol_) external; }// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.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: address zero is not a valid owner"); return _balances[owner]; } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { address owner = _owners[tokenId]; require(owner != address(0), "ERC721: invalid token ID"); return owner; } /** * @dev See {IERC721Metadata-name}. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev See {IERC721Metadata-symbol}. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { _requireMinted(tokenId); string memory baseURI = _baseURI(); return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : ""; } /** * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each * token will be the concatenation of the `baseURI` and the `tokenId`. Empty * by default, can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ""; } /** * @dev See {IERC721-approve}. */ function approve(address to, uint256 tokenId) public virtual override { address owner = ERC721.ownerOf(tokenId); require(to != owner, "ERC721: approval to current owner"); require( _msgSender() == owner || isApprovedForAll(owner, _msgSender()), "ERC721: approve caller is not token owner nor approved for all" ); _approve(to, tokenId); } /** * @dev See {IERC721-getApproved}. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { _requireMinted(tokenId); return _tokenApprovals[tokenId]; } /** * @dev See {IERC721-setApprovalForAll}. */ function setApprovalForAll(address operator, bool approved) public virtual override { _setApprovalForAll(_msgSender(), operator, approved); } /** * @dev See {IERC721-isApprovedForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @dev See {IERC721-transferFrom}. */ function transferFrom( address from, address to, uint256 tokenId ) public virtual override { //solhint-disable-next-line max-line-length require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner 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: caller is not token 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) { address owner = ERC721.ownerOf(tokenId); return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender); } /** * @dev Safely mints `tokenId` and transfers it to `to`. * * Requirements: * * - `tokenId` must not exist. * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer. * * Emits a {Transfer} event. */ function _safeMint(address to, uint256 tokenId) internal virtual { _safeMint(to, tokenId, ""); } /** * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is * forwarded in {IERC721Receiver-onERC721Received} to contract recipients. */ function _safeMint( address to, uint256 tokenId, bytes memory data ) internal virtual { _mint(to, tokenId); require( _checkOnERC721Received(address(0), to, tokenId, data), "ERC721: transfer to non ERC721Receiver implementer" ); } /** * @dev Mints `tokenId` and transfers it to `to`. * * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible * * Requirements: * * - `tokenId` must not exist. * - `to` cannot be the zero address. * * Emits a {Transfer} event. */ function _mint(address to, uint256 tokenId) internal virtual { require(to != address(0), "ERC721: mint to the zero address"); require(!_exists(tokenId), "ERC721: token already minted"); _beforeTokenTransfer(address(0), to, tokenId); _balances[to] += 1; _owners[tokenId] = to; emit Transfer(address(0), to, tokenId); _afterTokenTransfer(address(0), to, tokenId); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId) internal virtual { address owner = ERC721.ownerOf(tokenId); _beforeTokenTransfer(owner, address(0), tokenId); // Clear approvals _approve(address(0), tokenId); _balances[owner] -= 1; delete _owners[tokenId]; emit Transfer(owner, address(0), tokenId); _afterTokenTransfer(owner, address(0), tokenId); } /** * @dev Transfers `tokenId` from `from` to `to`. * As opposed to {transferFrom}, this imposes no restrictions on msg.sender. * * Requirements: * * - `to` cannot be the zero address. * - `tokenId` token must be owned by `from`. * * Emits a {Transfer} event. */ function _transfer( address from, address to, uint256 tokenId ) internal virtual { require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner"); require(to != address(0), "ERC721: transfer to the zero address"); _beforeTokenTransfer(from, to, tokenId); // Clear approvals from the previous owner _approve(address(0), tokenId); _balances[from] -= 1; _balances[to] += 1; _owners[tokenId] = to; emit Transfer(from, to, tokenId); _afterTokenTransfer(from, to, tokenId); } /** * @dev Approve `to` to operate on `tokenId` * * Emits an {Approval} event. */ function _approve(address to, uint256 tokenId) internal virtual { _tokenApprovals[tokenId] = to; emit Approval(ERC721.ownerOf(tokenId), to, tokenId); } /** * @dev Approve `operator` to operate on all of `owner` tokens * * Emits an {ApprovalForAll} event. */ function _setApprovalForAll( address owner, address operator, bool approved ) internal virtual { require(owner != operator, "ERC721: approve to caller"); _operatorApprovals[owner][operator] = approved; emit ApprovalForAll(owner, operator, approved); } /** * @dev Reverts if the `tokenId` has not been minted yet. */ function _requireMinted(uint256 tokenId) internal view virtual { require(_exists(tokenId), "ERC721: invalid token ID"); } /** * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address. * The call is not executed if the target address is not a contract. * * @param from address representing the previous owner of the given token ID * @param to target address that will receive the tokens * @param tokenId uint256 ID of the token to be transferred * @param data bytes optional data to send along with the call * @return bool whether the call correctly returned the expected magic value */ function _checkOnERC721Received( address from, address to, uint256 tokenId, bytes memory data ) private returns (bool) { if (to.isContract()) { try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) { return retval == IERC721Receiver.onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert("ERC721: transfer to non ERC721Receiver implementer"); } else { /// @solidity memory-safe-assembly assembly { revert(add(32, reason), mload(reason)) } } } } else { return true; } } /** * @dev Hook that is called before any token transfer. This includes minting * and burning. * * 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/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.7.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev Returns true if `account` is a contract. * * [IMPORTANT] * ==== * It is unsafe to assume that an address for which this function returns * false is an externally-owned account (EOA) and not a contract. * * Among others, `isContract` will return false for the following * types of addresses: * * - an externally-owned account * - a contract in construction * - an address where a contract will be created * - an address where a contract lived, but was destroyed * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 0; } /** * @dev Replacement for Solidity's `transfer`: sends `amount` wei to * `recipient`, forwarding all available gas and reverting on errors. * * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost * of certain opcodes, possibly making contracts go over the 2300 gas limit * imposed by `transfer`, making them unable to receive funds via * `transfer`. {sendValue} removes this limitation. * * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more]. * * IMPORTANT: because control is transferred to `recipient`, care must be * taken to not create reentrancy vulnerabilities. Consider using * {ReentrancyGuard} or the * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { require(address(this).balance >= amount, "Address: insufficient balance"); (bool success, ) = recipient.call{value: amount}(""); require(success, "Address: unable to send value, recipient may have reverted"); } /** * @dev Performs a Solidity function call using a low level `call`. A * plain `call` is an unsafe replacement for a function call: use this * function instead. * * If `target` reverts with a revert reason, it is bubbled up by this * function (like regular Solidity function calls). * * Returns the raw returned data. To convert to the expected return value, * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`]. * * Requirements: * * - `target` must be a contract. * - calling `target` with `data` must not revert. * * _Available since v3.1._ */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCall(target, data, "Address: low-level call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with * `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { return functionCallWithValue(target, data, 0, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but also transferring `value` wei to `target`. * * Requirements: * * - the calling contract must have an ETH balance of at least `value`. * - the called Solidity function must be `payable`. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value ) internal returns (bytes memory) { return functionCallWithValue(target, data, value, "Address: low-level call with value failed"); } /** * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but * with `errorMessage` as a fallback revert reason when `target` reverts. * * _Available since v3.1._ */ function functionCallWithValue( address target, bytes memory data, uint256 value, string memory errorMessage ) internal returns (bytes memory) { require(address(this).balance >= value, "Address: insufficient balance for call"); require(isContract(target), "Address: call to non-contract"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { return functionStaticCall(target, data, "Address: low-level static call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a static call. * * _Available since v3.3._ */ function functionStaticCall( address target, bytes memory data, string memory errorMessage ) internal view returns (bytes memory) { require(isContract(target), "Address: static call to non-contract"); (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { return functionDelegateCall(target, data, "Address: low-level delegate call failed"); } /** * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`], * but performing a delegate call. * * _Available since v3.4._ */ function functionDelegateCall( address target, bytes memory data, string memory errorMessage ) internal returns (bytes memory) { require(isContract(target), "Address: delegate call to non-contract"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal pure returns (bytes memory) { if (success) { return returndata; } else { // Look for revert reason and bubble it up if present if (returndata.length > 0) { // The easiest way to bubble the revert reason is using memory via assembly /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } } // SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol) pragma solidity ^0.8.0; /** * @dev String operations. */ library Strings { bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef"; uint8 private constant _ADDRESS_LENGTH = 20; /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { // Inspired by OraclizeAPI's implementation - MIT licence // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol if (value == 0) { return "0"; } uint256 temp = value; uint256 digits; while (temp != 0) { digits++; temp /= 10; } bytes memory buffer = new bytes(digits); while (value != 0) { digits -= 1; buffer[digits] = bytes1(uint8(48 + uint256(value % 10))); value /= 10; } return string(buffer); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { if (value == 0) { return "0x00"; } uint256 temp = value; uint256 length = 0; while (temp != 0) { length++; temp >>= 8; } return toHexString(value, length); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length. */ function toHexString(uint256 value, uint256 length) internal pure returns (string memory) { bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = _HEX_SYMBOLS[value & 0xf]; value >>= 4; } require(value == 0, "Strings: hex length insufficient"); return string(buffer); } /** * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation. */ function toHexString(address addr) internal pure returns (string memory) { return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH); } }