From b3517b062cfe00d032ff787983a89f3fa790d1ec Mon Sep 17 00:00:00 2001 From: Vectorized Date: Thu, 22 Feb 2024 22:59:11 +0800 Subject: [PATCH] Non-sequential minting (a.k.a spot-minting) support (#479) --- contracts/ERC721A.sol | 189 ++++++++++++- contracts/IERC721A.sol | 25 ++ contracts/extensions/ERC721AQueryable.sol | 55 ++-- contracts/mocks/ERC721ASpotMock.sol | 72 +++++ contracts/mocks/SequentialUpToHelper.sol | 33 +++ test/extensions/ERC721ASpot.test.js | 319 ++++++++++++++++++++++ 6 files changed, 656 insertions(+), 37 deletions(-) create mode 100644 contracts/mocks/ERC721ASpotMock.sol create mode 100644 contracts/mocks/SequentialUpToHelper.sol create mode 100644 test/extensions/ERC721ASpot.test.js diff --git a/contracts/ERC721A.sol b/contracts/ERC721A.sol index c0230ef45..dc0cdfcc8 100644 --- a/contracts/ERC721A.sol +++ b/contracts/ERC721A.sol @@ -28,6 +28,9 @@ interface ERC721A__IERC721Receiver { * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...) * starting from `_startTokenId()`. * + * The `_sequentialUpTo()` function can be overriden to enable spot mints + * (i.e. non-consecutive mints) for `tokenId`s greater than `_sequentialUpTo()`. + * * Assumptions: * * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply. @@ -133,6 +136,10 @@ contract ERC721A is IERC721A { // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; + // The amount of tokens minted above `_sequentialUpTo()`. + // We call these spot mints (i.e. non-sequential mints). + uint256 private _spotMinted; + // ============================================================= // CONSTRUCTOR // ============================================================= @@ -141,6 +148,8 @@ contract ERC721A is IERC721A { _name = name_; _symbol = symbol_; _currentIndex = _startTokenId(); + + if (_sequentialUpTo() < _startTokenId()) _revert(SequentialUpToTooSmall.selector); } // ============================================================= @@ -148,13 +157,28 @@ contract ERC721A is IERC721A { // ============================================================= /** - * @dev Returns the starting token ID. - * To change the starting token ID, please override this function. + * @dev Returns the starting token ID for sequential mints. + * + * Override this function to change the starting token ID for sequential mints. + * + * Note: The value returned must never change after any tokens have been minted. */ function _startTokenId() internal view virtual returns (uint256) { return 0; } + /** + * @dev Returns the maximum token ID (inclusive) for sequential mints. + * + * Override this function to return a value less than 2**256 - 1, + * but greater than `_startTokenId()`, to enable spot (non-sequential) mints. + * + * Note: The value returned must never change after any tokens have been minted. + */ + function _sequentialUpTo() internal view virtual returns (uint256) { + return type(uint256).max; + } + /** * @dev Returns the next token ID to be minted. */ @@ -167,22 +191,26 @@ contract ERC721A is IERC721A { * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ - function totalSupply() public view virtual override returns (uint256) { - // Counter underflow is impossible as _burnCounter cannot be incremented - // more than `_currentIndex - _startTokenId()` times. + function totalSupply() public view virtual override returns (uint256 result) { + // Counter underflow is impossible as `_burnCounter` cannot be incremented + // more than `_currentIndex + _spotMinted - _startTokenId()` times. unchecked { - return _currentIndex - _burnCounter - _startTokenId(); + // With spot minting, the intermediate `result` can be temporarily negative, + // and the computation must be unchecked. + result = _currentIndex - _burnCounter - _startTokenId(); + if (_sequentialUpTo() != type(uint256).max) result += _spotMinted; } } /** * @dev Returns the total amount of tokens minted in the contract. */ - function _totalMinted() internal view virtual returns (uint256) { + function _totalMinted() internal view virtual returns (uint256 result) { // Counter underflow is impossible as `_currentIndex` does not decrement, // and it is initialized to `_startTokenId()`. unchecked { - return _currentIndex - _startTokenId(); + result = _currentIndex - _startTokenId(); + if (_sequentialUpTo() != type(uint256).max) result += _spotMinted; } } @@ -193,6 +221,13 @@ contract ERC721A is IERC721A { return _burnCounter; } + /** + * @dev Returns the total number of tokens that are spot-minted. + */ + function _totalSpotMinted() internal view virtual returns (uint256) { + return _spotMinted; + } + // ============================================================= // ADDRESS DATA OPERATIONS // ============================================================= @@ -349,11 +384,17 @@ contract ERC721A is IERC721A { } /** - * Returns the packed ownership data of `tokenId`. + * @dev Returns the packed ownership data of `tokenId`. */ function _packedOwnershipOf(uint256 tokenId) private view returns (uint256 packed) { if (_startTokenId() <= tokenId) { packed = _packedOwnerships[tokenId]; + + if (tokenId > _sequentialUpTo()) { + if (_packedOwnershipExists(packed)) return packed; + _revert(OwnerQueryForNonexistentToken.selector); + } + // If the data at the starting slot does not exist, start the scan. if (packed == 0) { if (tokenId >= _currentIndex) _revert(OwnerQueryForNonexistentToken.selector); @@ -482,6 +523,8 @@ contract ERC721A is IERC721A { */ function _exists(uint256 tokenId) internal view virtual returns (bool result) { if (_startTokenId() <= tokenId) { + if (tokenId > _sequentialUpTo()) return _packedOwnershipExists(_packedOwnerships[tokenId]); + if (tokenId < _currentIndex) { uint256 packed; while ((packed = _packedOwnerships[tokenId]) == 0) --tokenId; @@ -490,6 +533,17 @@ contract ERC721A is IERC721A { } } + /** + * @dev Returns whether `packed` represents a token that exists. + */ + function _packedOwnershipExists(uint256 packed) private pure returns (bool result) { + assembly { + // The following is equivalent to `owner != address(0) && burned == false`. + // Symbolically tested. + result := gt(and(packed, _BITMASK_ADDRESS), and(packed, _BITMASK_BURNED)) + } + } + /** * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`. */ @@ -783,6 +837,8 @@ contract ERC721A is IERC721A { uint256 end = startTokenId + quantity; uint256 tokenId = startTokenId; + if (end - 1 > _sequentialUpTo()) _revert(SequentialMintExceedsLimit.selector); + do { assembly { // Emit the `Transfer` event. @@ -852,6 +908,8 @@ contract ERC721A is IERC721A { _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); + if (startTokenId + quantity - 1 > _sequentialUpTo()) _revert(SequentialMintExceedsLimit.selector); + emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to); _currentIndex = startTokenId + quantity; @@ -888,8 +946,9 @@ contract ERC721A is IERC721A { _revert(TransferToNonERC721ReceiverImplementer.selector); } } while (index < end); - // Reentrancy protection. - if (_currentIndex != end) _revert(bytes4(0)); + // This prevents reentrancy to `_safeMint`. + // It does not prevent reentrancy to `_safeMintSpot`. + if (_currentIndex != end) revert(); } } } @@ -901,6 +960,112 @@ contract ERC721A is IERC721A { _safeMint(to, quantity, ''); } + /** + * @dev Mints a single token at `tokenId`. + * + * Note: A spot-minted `tokenId` that has been burned can be re-minted again. + * + * Requirements: + * + * - `to` cannot be the zero address. + * - `tokenId` must be greater than `_sequentialUpTo()`. + * - `tokenId` must not exist. + * + * Emits a {Transfer} event for each mint. + */ + function _mintSpot(address to, uint256 tokenId) internal virtual { + if (tokenId <= _sequentialUpTo()) _revert(SpotMintTokenIdTooSmall.selector); + uint256 prevOwnershipPacked = _packedOwnerships[tokenId]; + if (_packedOwnershipExists(prevOwnershipPacked)) _revert(TokenAlreadyExists.selector); + + _beforeTokenTransfers(address(0), to, tokenId, 1); + + // Overflows are incredibly unrealistic. + // The `numberMinted` for `to` is incremented by 1, and has a max limit of 2**64 - 1. + // `_spotMinted` is incremented by 1, and has a max limit of 2**256 - 1. + unchecked { + // Updates: + // - `address` to the owner. + // - `startTimestamp` to the timestamp of minting. + // - `burned` to `false`. + // - `nextInitialized` to `true` (as `quantity == 1`). + _packedOwnerships[tokenId] = _packOwnershipData( + to, + _nextInitializedFlag(1) | _nextExtraData(address(0), to, prevOwnershipPacked) + ); + + // Updates: + // - `balance += 1`. + // - `numberMinted += 1`. + // + // We can directly add to the `balance` and `numberMinted`. + _packedAddressData[to] += (1 << _BITPOS_NUMBER_MINTED) | 1; + + // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. + uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; + + if (toMasked == 0) _revert(MintToZeroAddress.selector); + + assembly { + // Emit the `Transfer` event. + log4( + 0, // Start of data (0, since no data). + 0, // End of data (0, since no data). + _TRANSFER_EVENT_SIGNATURE, // Signature. + 0, // `address(0)`. + toMasked, // `to`. + tokenId // `tokenId`. + ) + } + + ++_spotMinted; + } + + _afterTokenTransfers(address(0), to, tokenId, 1); + } + + /** + * @dev Safely mints a single token at `tokenId`. + * + * Note: A spot-minted `tokenId` that has been burned can be re-minted again. + * + * Requirements: + * + * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}. + * - `tokenId` must be greater than `_sequentialUpTo()`. + * - `tokenId` must not exist. + * + * See {_mintSpot}. + * + * Emits a {Transfer} event. + */ + function _safeMintSpot( + address to, + uint256 tokenId, + bytes memory _data + ) internal virtual { + _mintSpot(to, tokenId); + + unchecked { + if (to.code.length != 0) { + uint256 currentSpotMinted = _spotMinted; + if (!_checkContractOnERC721Received(address(0), to, tokenId, _data)) { + _revert(TransferToNonERC721ReceiverImplementer.selector); + } + // This prevents reentrancy to `_safeMintSpot`. + // It does not prevent reentrancy to `_safeMint`. + if (_spotMinted != currentSpotMinted) revert(); + } + } + } + + /** + * @dev Equivalent to `_safeMintSpot(to, tokenId, '')`. + */ + function _safeMintSpot(address to, uint256 tokenId) internal virtual { + _safeMintSpot(to, tokenId, ''); + } + // ============================================================= // APPROVAL OPERATIONS // ============================================================= @@ -1024,7 +1189,7 @@ contract ERC721A is IERC721A { emit Transfer(from, address(0), tokenId); _afterTokenTransfers(from, address(0), tokenId, 1); - // Overflow not possible, as _burnCounter cannot be exceed _currentIndex times. + // Overflow not possible, as `_burnCounter` cannot be exceed `_currentIndex + _spotMinted` times. unchecked { _burnCounter++; } diff --git a/contracts/IERC721A.sol b/contracts/IERC721A.sol index 4bd87db45..29e32f2e0 100644 --- a/contracts/IERC721A.sol +++ b/contracts/IERC721A.sol @@ -74,6 +74,31 @@ interface IERC721A { */ error OwnershipNotInitializedForExtraData(); + /** + * `_sequentialUpTo()` must be greater than `_startTokenId()`. + */ + error SequentialUpToTooSmall(); + + /** + * The `tokenId` of a sequential mint exceeds `_sequentialUpTo()`. + */ + error SequentialMintExceedsLimit(); + + /** + * Spot minting requires a `tokenId` greater than `_sequentialUpTo()`. + */ + error SpotMintTokenIdTooSmall(); + + /** + * Cannot mint over a token that already exists. + */ + error TokenAlreadyExists(); + + /** + * The feature is not compatible with spot mints. + */ + error NotCompatibleWithSpotMints(); + // ============================================================= // STRUCTS // ============================================================= diff --git a/contracts/extensions/ERC721AQueryable.sol b/contracts/extensions/ERC721AQueryable.sol index 081815bd3..475c15119 100644 --- a/contracts/extensions/ERC721AQueryable.sol +++ b/contracts/extensions/ERC721AQueryable.sol @@ -46,6 +46,8 @@ abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { { unchecked { if (tokenId >= _startTokenId()) { + if (tokenId > _sequentialUpTo()) return _ownershipAt(tokenId); + if (tokenId < _nextTokenId()) { // If the `tokenId` is within bounds, // scan backwards for the initialized ownership slot. @@ -125,6 +127,8 @@ abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { * an out-of-gas error (10K collections should be fine). */ function tokensOfOwner(address owner) external view virtual override returns (uint256[] memory) { + // If spot mints are enabled, full-range scan is disabled. + if (_sequentialUpTo() != type(uint256).max) _revert(NotCompatibleWithSpotMints.selector); uint256 start = _startTokenId(); uint256 stop = _nextTokenId(); uint256[] memory tokenIds; @@ -142,37 +146,32 @@ abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { address owner, uint256 start, uint256 stop - ) private view returns (uint256[] memory) { + ) private view returns (uint256[] memory tokenIds) { unchecked { if (start >= stop) _revert(InvalidQueryRange.selector); // Set `start = max(start, _startTokenId())`. - if (start < _startTokenId()) { - start = _startTokenId(); - } - uint256 stopLimit = _nextTokenId(); + if (start < _startTokenId()) start = _startTokenId(); + uint256 nextTokenId = _nextTokenId(); + // If spot mints are enabled, scan all the way until the specified `stop`. + uint256 stopLimit = _sequentialUpTo() != type(uint256).max ? stop : nextTokenId; // Set `stop = min(stop, stopLimit)`. - if (stop >= stopLimit) { - stop = stopLimit; - } - uint256[] memory tokenIds; + if (stop >= stopLimit) stop = stopLimit; + // Number of tokens to scan. uint256 tokenIdsMaxLength = balanceOf(owner); - bool startLtStop = start < stop; - assembly { - // Set `tokenIdsMaxLength` to zero if `start` is less than `stop`. - tokenIdsMaxLength := mul(tokenIdsMaxLength, startLtStop) - } + // Set `tokenIdsMaxLength` to zero if the range contains no tokens. + if (start >= stop) tokenIdsMaxLength = 0; + // If there are one or more tokens to scan. if (tokenIdsMaxLength != 0) { - // Set `tokenIdsMaxLength = min(balanceOf(owner), stop - start)`, - // to cater for cases where `balanceOf(owner)` is too big. - if (stop - start <= tokenIdsMaxLength) { - tokenIdsMaxLength = stop - start; - } + // Set `tokenIdsMaxLength = min(balanceOf(owner), tokenIdsMaxLength)`. + if (stop - start <= tokenIdsMaxLength) tokenIdsMaxLength = stop - start; + uint256 m; // Start of available memory. assembly { // Grab the free memory pointer. tokenIds := mload(0x40) // Allocate one word for the length, and `tokenIdsMaxLength` words // for the data. `shl(5, x)` is equivalent to `mul(32, x)`. - mstore(0x40, add(tokenIds, shl(5, add(tokenIdsMaxLength, 1)))) + m := add(tokenIds, shl(5, add(tokenIdsMaxLength, 1))) + mstore(0x40, m) } // We need to call `explicitOwnershipOf(start)`, // because the slot at `start` may not be initialized. @@ -182,14 +181,18 @@ abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { // initialize `currOwnershipAddr`. // `ownership.address` will not be zero, // as `start` is clamped to the valid token ID range. - if (!ownership.burned) { - currOwnershipAddr = ownership.addr; - } + if (!ownership.burned) currOwnershipAddr = ownership.addr; uint256 tokenIdsIdx; // Use a do-while, which is slightly more efficient for this case, // as the array will at least contain one element. do { - ownership = _ownershipAt(start); + if (_sequentialUpTo() != type(uint256).max) { + // Skip the remaining unused sequential slots. + if (start == nextTokenId) start = _sequentialUpTo() + 1; + // Reset `currOwnershipAddr`, as each spot-minted token is a batch of one. + if (start > _sequentialUpTo()) currOwnershipAddr = address(0); + } + ownership = _ownershipAt(start); // This implicitly allocates memory. assembly { switch mload(add(ownership, 0x40)) // if `ownership.burned == false`. @@ -215,6 +218,9 @@ abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { currOwnershipAddr := 0 } start := add(start, 1) + // Free temporary memory implicitly allocated for ownership + // to avoid quadratic memory expansion costs. + mstore(0x40, m) } } while (!(start == stop || tokenIdsIdx == tokenIdsMaxLength)); // Store the length of the array. @@ -222,7 +228,6 @@ abstract contract ERC721AQueryable is ERC721A, IERC721AQueryable { mstore(tokenIds, tokenIdsIdx) } } - return tokenIds; } } } diff --git a/contracts/mocks/ERC721ASpotMock.sol b/contracts/mocks/ERC721ASpotMock.sol new file mode 100644 index 000000000..3a764a865 --- /dev/null +++ b/contracts/mocks/ERC721ASpotMock.sol @@ -0,0 +1,72 @@ +// SPDX-License-Identifier: MIT +// ERC721A Contracts v4.2.3 +// Creators: Chiru Labs + +pragma solidity ^0.8.4; + +import './ERC721AQueryableMock.sol'; +import './StartTokenIdHelper.sol'; +import './SequentialUpToHelper.sol'; + +contract ERC721ASpotMock is StartTokenIdHelper, SequentialUpToHelper, ERC721AQueryableMock { + constructor( + string memory name_, + string memory symbol_, + uint256 startTokenId_, + uint256 sequentialUpTo_, + uint256 quantity, + bool mintInConstructor + ) StartTokenIdHelper(startTokenId_) SequentialUpToHelper(sequentialUpTo_) ERC721AQueryableMock(name_, symbol_) { + if (mintInConstructor) { + _mintERC2309(msg.sender, quantity); + } + } + + function _startTokenId() internal view override returns (uint256) { + return startTokenId(); + } + + function _sequentialUpTo() internal view override returns (uint256) { + return sequentialUpTo(); + } + + function exists(uint256 tokenId) public view returns (bool) { + return _exists(tokenId); + } + + function getOwnershipOf(uint256 index) public view returns (TokenOwnership memory) { + return _ownershipOf(index); + } + + function safeMintSpot(address to, uint256 tokenId) public { + _safeMintSpot(to, tokenId); + } + + function totalSpotMinted() public view returns (uint256) { + return _totalSpotMinted(); + } + + function totalMinted() public view returns (uint256) { + return _totalMinted(); + } + + function totalBurned() public view returns (uint256) { + return _totalBurned(); + } + + function numberBurned(address owner) public view returns (uint256) { + return _numberBurned(owner); + } + + function setExtraDataAt(uint256 tokenId, uint24 value) public { + _setExtraDataAt(tokenId, value); + } + + function _extraData( + address, + address, + uint24 previousExtraData + ) internal view virtual override returns (uint24) { + return previousExtraData; + } +} diff --git a/contracts/mocks/SequentialUpToHelper.sol b/contracts/mocks/SequentialUpToHelper.sol new file mode 100644 index 000000000..8f5989b25 --- /dev/null +++ b/contracts/mocks/SequentialUpToHelper.sol @@ -0,0 +1,33 @@ +// SPDX-License-Identifier: MIT +// ERC721A Contracts v4.2.3 +// Creators: Chiru Labs + +pragma solidity ^0.8.4; + +/** + * This Helper is used to return a dynamic value in the overridden _sequentialUpTo() function. + * Extending this Helper before the ERC721A contract give us access to the herein set `sequentialUpTo` + * to be returned by the overridden `_sequentialUpTo()` function of ERC721A in the ERC721ASpot mocks. + */ +contract SequentialUpToHelper { + // `bytes4(keccak256('sequentialUpTo'))`. + uint256 private constant SEQUENTIAL_UP_TO_STORAGE_SLOT = 0x9638c59e; + + constructor(uint256 sequentialUpTo_) { + _initializeSequentialUpTo(sequentialUpTo_); + } + + function sequentialUpTo() public view returns (uint256 result) { + assembly { + result := sload(SEQUENTIAL_UP_TO_STORAGE_SLOT) + } + } + + function _initializeSequentialUpTo(uint256 value) private { + // We use assembly to directly set the `sequentialUpTo` in storage so that + // inheriting this class won't affect the layout of other storage slots. + assembly { + sstore(SEQUENTIAL_UP_TO_STORAGE_SLOT, value) + } + } +} diff --git a/test/extensions/ERC721ASpot.test.js b/test/extensions/ERC721ASpot.test.js new file mode 100644 index 000000000..f9be7ca2b --- /dev/null +++ b/test/extensions/ERC721ASpot.test.js @@ -0,0 +1,319 @@ +const { deployContract } = require('../helpers.js'); +const { expect } = require('chai'); +const { BigNumber } = require('ethers'); +const { constants } = require('@openzeppelin/test-helpers'); +const { ZERO_ADDRESS } = constants; + +describe('ERC721ASpot', function () { + + context('constructor', function () { + const testConstructor = async (args, expectedError) => { + const deployment = deployContract('ERC721ASpotMock', args); + if (expectedError) await expect(deployment).to.be.revertedWith(expectedError); + else await deployment; + }; + + it('reverts if _sequentialUpTo is not greater than _startTokenId', async function () { + const t = async (startTokenId, sequentialUpTo, expectSuccess) => { + await testConstructor( + ['Azuki', 'AZUKI', startTokenId, sequentialUpTo, 0, false], + expectSuccess ? false : 'SequentialUpToTooSmall' + ); + }; + await t(0, 0, true); + await t(1, 0, false); + await t(0, 1, true); + await t(100, 99, false); + await t(100, 100, true); + await t(100, 101, true); + await t(100, 999, true); + }); + + it('reverts if ERC2309 mint exceeds limit', async function () { + const t = async (startTokenId, sequentialUpTo, quantity, expectSuccess) => { + await testConstructor( + ['Azuki', 'AZUKI', startTokenId, sequentialUpTo, quantity, true], + expectSuccess ? false : 'SequentialMintExceedsLimit' + ); + }; + await t(0, 1, 1, true); + await t(0, 1, 2, true); + await t(0, 1, 3, false); + await t(100, 101, 1, true); + await t(100, 101, 2, true); + await t(100, 101, 3, false); + await t(100, 109, 2, true); + await t(100, 109, 9, true); + await t(100, 109, 10, true); + await t(100, 109, 11, false); + }); + }); + + context('mint sequential and spot', function () { + beforeEach(async function () { + const [owner, addr1] = await ethers.getSigners(); + this.owner = owner; + this.addr1 = addr1; + this.startTokenId = BigNumber.from(10); + this.sequentialUpTo = BigNumber.from(19); + const args = ['Azuki', 'AZUKI', this.startTokenId, this.sequentialUpTo, 0, false]; + this.erc721aSpot = await deployContract('ERC721ASpotMock', args); + }); + + it('_mintSpot emits a Transfer event', async function () { + await expect(this.erc721aSpot.safeMintSpot(this.addr1.address, 20)) + .to.emit(this.erc721aSpot, 'Transfer') + .withArgs(ZERO_ADDRESS, this.addr1.address, 20); + }); + + it('increases _totalSpotMinted, totalSupply', async function () { + await this.erc721aSpot.safeMint(this.addr1.address, 5); + expect(await this.erc721aSpot.totalSpotMinted()).to.eq(0); + expect(await this.erc721aSpot.totalSupply()).to.eq(5); + + await this.erc721aSpot.safeMintSpot(this.addr1.address, 20); + expect(await this.erc721aSpot.totalSpotMinted()).to.eq(1); + expect(await this.erc721aSpot.totalSupply()).to.eq(6); + + await this.erc721aSpot.safeMintSpot(this.addr1.address, 30); + expect(await this.erc721aSpot.totalSpotMinted()).to.eq(2); + expect(await this.erc721aSpot.totalSupply()).to.eq(7); + }); + + it('tokensOfOwnerIn', async function () { + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 4294967295)).to.eql([]); + + await this.erc721aSpot.safeMint(this.addr1.address, 5); + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 4294967295)) + .to.eql([10, 11, 12, 13, 14].map(BigNumber.from)); + + await this.erc721aSpot.safeMintSpot(this.addr1.address, 21); + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 4294967295)) + .to.eql([10, 11, 12, 13, 14, 21].map(BigNumber.from)); + + await this.erc721aSpot.safeMintSpot(this.addr1.address, 31); + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 4294967295)) + .to.eql([10, 11, 12, 13, 14, 21, 31].map(BigNumber.from)); + + await this.erc721aSpot.safeMintSpot(this.addr1.address, 22); + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 4294967295)) + .to.eql([10, 11, 12, 13, 14, 21, 22, 31].map(BigNumber.from)); + + await this.erc721aSpot.safeMint(this.addr1.address, 5); + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 4294967295)) + .to.eql([10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 21, 22, 31].map(BigNumber.from)); + + await this.erc721aSpot.safeMintSpot(this.addr1.address, 20); + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 4294967295)) + .to.eql([10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 31].map(BigNumber.from)); + + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 32)) + .to.eql([10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 31].map(BigNumber.from)); + + expect(await this.erc721aSpot.tokensOfOwnerIn(this.addr1.address, 0, 31)) + .to.eql([10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22].map(BigNumber.from)); + }); + + it('explicitOwnershipOf', async function () { + let explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(10); + expect(explicitOwnership.addr).to.eq(ZERO_ADDRESS); + expect(explicitOwnership.burned).to.eq(false); + + await this.erc721aSpot.safeMint(this.addr1.address, 1); + explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(10); + expect(explicitOwnership.addr).to.eq(this.addr1.address); + expect(explicitOwnership.burned).to.eq(false); + + explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(11); + expect(explicitOwnership.addr).to.eq(ZERO_ADDRESS); + expect(explicitOwnership.burned).to.eq(false); + + explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(20); + expect(explicitOwnership.addr).to.eq(ZERO_ADDRESS); + expect(explicitOwnership.burned).to.eq(false); + + await this.erc721aSpot.safeMintSpot(this.addr1.address, 20); + explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(20); + expect(explicitOwnership.addr).to.eq(this.addr1.address); + expect(explicitOwnership.burned).to.eq(false); + }); + + it('tokensOfOwner reverts', async function () { + await expect(this.erc721aSpot.tokensOfOwner(this.addr1.address)).to.be.revertedWith( + 'NotCompatibleWithSpotMints' + ); + }); + + it('spot minting to an existing token reverts', async function () { + await this.erc721aSpot.safeMintSpot(this.addr1.address, 20); + await expect(this.erc721aSpot.safeMintSpot(this.addr1.address, 20)).to.be.revertedWith( + 'TokenAlreadyExists' + ); + }); + + it('reverts if sequential mint exceeds limit', async function () { + await expect(this.erc721aSpot.safeMint(this.addr1.address, 11)).to.be.revertedWith( + 'SequentialMintExceedsLimit' + ); + await this.erc721aSpot.safeMint(this.addr1.address, 10); + }); + + it('reverts if _mintSpot tokenId is too small', async function () { + await expect(this.erc721aSpot.safeMintSpot(this.addr1.address, 19)).to.be.revertedWith( + 'SpotMintTokenIdTooSmall' + ); + }); + + context('with transfers', function () { + it('reverts if token is not minted', async function () { + await this.erc721aSpot.safeMint(this.addr1.address, 10); + await expect(this.erc721aSpot + .connect(this.addr1) + .transferFrom(this.addr1.address, this.owner.address, 21)).to.be.revertedWith( + 'OwnerQueryForNonexistentToken' + ); + await this.erc721aSpot.safeMintSpot(this.addr1.address, 21); + await this.erc721aSpot + .connect(this.addr1) + .transferFrom(this.addr1.address, this.owner.address, 21); + }); + + it('edge case 1', async function () { + await this.erc721aSpot.safeMintSpot(this.addr1.address, 20); + await this.erc721aSpot.safeMint(this.addr1.address, 10); + await this.erc721aSpot.connect(this.addr1).transferFrom(this.addr1.address, this.owner.address, 20); + expect(await this.erc721aSpot.ownerOf(20)).to.eq(this.owner.address); + expect(await this.erc721aSpot.ownerOf(19)).to.eq(this.addr1.address); + expect(await this.erc721aSpot.ownerOf(18)).to.eq(this.addr1.address); + await this.erc721aSpot.connect(this.addr1).transferFrom(this.addr1.address, this.owner.address, 19); + expect(await this.erc721aSpot.ownerOf(20)).to.eq(this.owner.address); + expect(await this.erc721aSpot.ownerOf(19)).to.eq(this.owner.address); + expect(await this.erc721aSpot.ownerOf(18)).to.eq(this.addr1.address); + }); + + it('edge case 2', async function () { + await this.erc721aSpot.safeMintSpot(this.addr1.address, 20); + await this.erc721aSpot.safeMint(this.addr1.address, 10); + await this.erc721aSpot.connect(this.addr1).transferFrom(this.addr1.address, this.owner.address, 19); + expect(await this.erc721aSpot.ownerOf(20)).to.eq(this.addr1.address); + expect(await this.erc721aSpot.ownerOf(19)).to.eq(this.owner.address); + expect(await this.erc721aSpot.ownerOf(18)).to.eq(this.addr1.address); + await this.erc721aSpot.connect(this.addr1).transferFrom(this.addr1.address, this.owner.address, 20); + expect(await this.erc721aSpot.ownerOf(20)).to.eq(this.owner.address); + expect(await this.erc721aSpot.ownerOf(19)).to.eq(this.owner.address); + expect(await this.erc721aSpot.ownerOf(18)).to.eq(this.addr1.address); + }); + }); + + context('with burns', function () { + beforeEach(async function () { + await this.erc721aSpot.safeMint(this.addr1.address, 5); + await this.erc721aSpot.safeMintSpot(this.addr1.address, 20); + await this.erc721aSpot.safeMintSpot(this.addr1.address, 30); + }); + + it('sets ownership correctly', async function () { + const t = async (tokenIds) => { + for (let i = 0; i < 35; ++i) { + const tx = this.erc721aSpot.getOwnershipOf(i); + if (tokenIds.includes(i)) await tx; + else await expect(tx).to.be.revertedWith('OwnerQueryForNonexistentToken'); + } + }; + await t([10, 11, 12, 13, 14, 20, 30]); + await this.erc721aSpot.connect(this.addr1).burn(20); + await t([10, 11, 12, 13, 14, 30]); + }); + + it('reduces balanceOf, totalSupply', async function () { + expect(await this.erc721aSpot.balanceOf(this.addr1.address)).to.eq(7); + await this.erc721aSpot.connect(this.addr1).burn(10); + expect(await this.erc721aSpot.balanceOf(this.addr1.address)).to.eq(6); + expect(await this.erc721aSpot.totalSupply()).to.eq(6); + + await this.erc721aSpot.connect(this.addr1).burn(20); + expect(await this.erc721aSpot.balanceOf(this.addr1.address)).to.eq(5); + expect(await this.erc721aSpot.totalSupply()).to.eq(5); + + await this.erc721aSpot.connect(this.addr1).burn(30); + expect(await this.erc721aSpot.balanceOf(this.addr1.address)).to.eq(4); + expect(await this.erc721aSpot.totalSupply()).to.eq(4); + + await this.erc721aSpot.connect(this.addr1).burn(11); + await this.erc721aSpot.connect(this.addr1).burn(12); + await this.erc721aSpot.connect(this.addr1).burn(13); + await this.erc721aSpot.connect(this.addr1).burn(14); + expect(await this.erc721aSpot.balanceOf(this.addr1.address)).to.eq(0); + expect(await this.erc721aSpot.totalSupply()).to.eq(0); + }); + + it('does not reduce totalMinted', async function () { + expect(await this.erc721aSpot.balanceOf(this.addr1.address)).to.eq(7); + await this.erc721aSpot.connect(this.addr1).burn(10); + expect(await this.erc721aSpot.totalMinted()).to.eq(7); + + await this.erc721aSpot.connect(this.addr1).burn(20); + expect(await this.erc721aSpot.totalMinted()).to.eq(7); + + await this.erc721aSpot.connect(this.addr1).burn(30); + expect(await this.erc721aSpot.totalMinted()).to.eq(7); + }); + + it('increases _numberBurned, totalBurned', async function () { + expect(await this.erc721aSpot.balanceOf(this.addr1.address)).to.eq(7); + await this.erc721aSpot.connect(this.addr1).burn(10); + expect(await this.erc721aSpot.numberBurned(this.addr1.address)).to.eq(1); + expect(await this.erc721aSpot.totalBurned()).to.eq(1); + + await this.erc721aSpot.connect(this.addr1).burn(20); + expect(await this.erc721aSpot.numberBurned(this.addr1.address)).to.eq(2); + expect(await this.erc721aSpot.totalBurned()).to.eq(2); + + await this.erc721aSpot.connect(this.addr1).burn(30); + expect(await this.erc721aSpot.numberBurned(this.addr1.address)).to.eq(3); + expect(await this.erc721aSpot.totalBurned()).to.eq(3); + + await this.erc721aSpot.connect(this.addr1).burn(11); + await this.erc721aSpot.connect(this.addr1).burn(12); + await this.erc721aSpot.connect(this.addr1).burn(13); + await this.erc721aSpot.connect(this.addr1).burn(14); + expect(await this.erc721aSpot.numberBurned(this.addr1.address)).to.eq(7); + expect(await this.erc721aSpot.totalBurned()).to.eq(7); + }); + + it('affects _exists', async function () { + expect(await this.erc721aSpot.exists(0)).to.eq(false); + expect(await this.erc721aSpot.exists(9)).to.eq(false); + expect(await this.erc721aSpot.exists(10)).to.eq(true); + + expect(await this.erc721aSpot.exists(20)).to.eq(true); + + await this.erc721aSpot.connect(this.addr1).burn(20); + expect(await this.erc721aSpot.exists(20)).to.eq(false); + + this.erc721aSpot.safeMintSpot(this.owner.address, 20); + expect(await this.erc721aSpot.exists(20)).to.eq(true); + }); + + it('forwards extraData after burn and re-mint', async function () { + await this.erc721aSpot.setExtraDataAt(20, 123); + let explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(20); + expect(explicitOwnership.addr).to.eq(this.addr1.address); + expect(explicitOwnership.burned).to.eq(false); + expect(explicitOwnership.extraData).to.eq(123); + + await this.erc721aSpot.connect(this.addr1).burn(20); + explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(20); + expect(explicitOwnership.addr).to.eq(this.addr1.address); + expect(explicitOwnership.burned).to.eq(true); + expect(explicitOwnership.extraData).to.eq(123); + + this.erc721aSpot.safeMintSpot(this.owner.address, 20); + explicitOwnership = await this.erc721aSpot.explicitOwnershipOf(20); + expect(explicitOwnership.addr).to.eq(this.owner.address); + expect(explicitOwnership.burned).to.eq(false); + expect(explicitOwnership.extraData).to.eq(123); + }); + }); + }); +});