From 02c9ce6c5008c8c9d8200acd3cd079220a3241bf Mon Sep 17 00:00:00 2001 From: immrsd <103599616+immrsd@users.noreply.github.com> Date: Wed, 6 Nov 2024 20:37:43 +0100 Subject: [PATCH] ERC721Enumerable helper function (#1196) * Implement helper function for ERC721Enumerable * Change function name * Run linter * Add changelog entry * Add documentation * Remove unnecessary Enumerable mocks * Add ERC721EnumerableExtended impl to ERC721Enumerable mock * Remove IERC721EnumerableExtended interface and make all_tokens_of_owner an internal function * Update doc * Update changelog * Add more test cases * Try to fix coverage issue * Revert test coverage fix changes --- CHANGELOG.md | 1 + docs/modules/ROOT/pages/api/erc721.adoc | 13 +++ packages/test_common/src/mocks/erc721.cairo | 10 +++ .../erc721_enumerable/erc721_enumerable.cairo | 22 ++++- .../erc721_enumerable/interface.cairo | 8 ++ .../tests/erc721/test_erc721_enumerable.cairo | 81 ++++++++++++++++--- 6 files changed, 120 insertions(+), 15 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 018474dfb..f0c630e04 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -13,6 +13,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - `is_valid_p256_signature` utility function to `openzeppelin_account::utils::signature` (#1189) - `Secp256r1KeyPair` type and helpers to `openzeppelin_testing::signing` (#1189) +- `all_tokens_of_owner` function to `ERC721EnumerableComponent` fetching all owner's tokens in a single call (#1196) - Embeddable impls for ERC2981 component (#1173) - `ERC2981Info` with read functions for discovering the component's state - `ERC2981AdminOwnable` providing admin functions for a token that implements Ownable component diff --git a/docs/modules/ROOT/pages/api/erc721.adoc b/docs/modules/ROOT/pages/api/erc721.adoc index 05491f088..6a4fb8bf8 100644 --- a/docs/modules/ROOT/pages/api/erc721.adoc +++ b/docs/modules/ROOT/pages/api/erc721.adoc @@ -886,6 +886,7 @@ mod ERC721EnumerableContract { .InternalImpl * xref:#ERC721EnumerableComponent-initializer[`++initializer(self)++`] * xref:#ERC721EnumerableComponent-before_update[`++before_update(self, to, token_id)++`] +* xref:#ERC721EnumerableComponent-all_tokens_of_owner[`++all_tokens_of_owner(self, owner)++`] * xref:#ERC721EnumerableComponent-_add_token_to_owner_enumeration[`++_add_token_to_owner_enumeration(self, to, token_id)++`] * xref:#ERC721EnumerableComponent-_add_token_to_all_tokens_enumeration[`++_add_token_to_all_tokens_enumeration(self, token_id)++`] * xref:#ERC721EnumerableComponent-_remove_token_from_owner_enumeration[`++_remove_token_from_owner_enumeration(self, from, token_id)++`] @@ -943,6 +944,18 @@ When a token is transferred, minted, or burned, the ownership-tracking data stru This must be added to the implementing contract's xref:ERC721Component-before_update[ERC721HooksTrait::before_update] hook. +[.contract-item] +[[ERC721EnumerableComponent-all_tokens_of_owner]] +==== `[.contract-item-name]#++all_tokens_of_owner++#++(self: @ContractState, owner: ContractAddress) → Span++` [.item-kind]#internal# + +Returns a list of all token ids owned by the specified `owner`. +This function provides a more efficient alternative to calling `ERC721::balance_of` +and iterating through tokens with `ERC721Enumerable::token_of_owner_by_index`. + +Requirements: + +- `owner` is not the zero address. + [.contract-item] [[ERC721EnumerableComponent-_add_token_to_owner_enumeration]] ==== `[.contract-item-name]#++_add_token_to_owner_enumeration++#++(ref self: ContractState, to: ContractAddress, token_id: u256)++` [.item-kind]#internal# diff --git a/packages/test_common/src/mocks/erc721.cairo b/packages/test_common/src/mocks/erc721.cairo index c14e16538..3d9be74a2 100644 --- a/packages/test_common/src/mocks/erc721.cairo +++ b/packages/test_common/src/mocks/erc721.cairo @@ -181,6 +181,7 @@ pub mod DualCaseERC721ReceiverMock { pub mod ERC721EnumerableMock { use openzeppelin_introspection::src5::SRC5Component; use openzeppelin_token::erc721::ERC721Component; + use openzeppelin_token::erc721::extensions::ERC721EnumerableComponent::InternalTrait; use openzeppelin_token::erc721::extensions::ERC721EnumerableComponent; use starknet::ContractAddress; @@ -238,6 +239,15 @@ pub mod ERC721EnumerableMock { } } + #[generate_trait] + #[abi(per_item)] + impl ExternalImpl of ExternalTrait { + #[external(v0)] + fn all_tokens_of_owner(self: @ContractState, owner: ContractAddress) -> Span { + self.erc721_enumerable.all_tokens_of_owner(owner) + } + } + #[constructor] fn constructor( ref self: ContractState, diff --git a/packages/token/src/erc721/extensions/erc721_enumerable/erc721_enumerable.cairo b/packages/token/src/erc721/extensions/erc721_enumerable/erc721_enumerable.cairo index 405991053..580ddb63c 100644 --- a/packages/token/src/erc721/extensions/erc721_enumerable/erc721_enumerable.cairo +++ b/packages/token/src/erc721/extensions/erc721_enumerable/erc721_enumerable.cairo @@ -20,7 +20,6 @@ pub mod ERC721EnumerableComponent { use crate::erc721::ERC721Component::ERC721Impl; use crate::erc721::ERC721Component::InternalImpl as ERC721InternalImpl; use crate::erc721::ERC721Component; - use crate::erc721::extensions::erc721_enumerable::interface::IERC721Enumerable; use crate::erc721::extensions::erc721_enumerable::interface; use openzeppelin_introspection::src5::SRC5Component::InternalTrait as SRC5InternalTrait; use openzeppelin_introspection::src5::SRC5Component; @@ -51,7 +50,7 @@ pub mod ERC721EnumerableComponent { +ERC721Component::ERC721HooksTrait, +SRC5Component::HasComponent, +Drop - > of IERC721Enumerable> { + > of interface::IERC721Enumerable> { /// Returns the total amount of tokens stored by the contract. fn total_supply(self: @ComponentState) -> u256 { self.ERC721Enumerable_all_tokens_len.read() @@ -133,6 +132,25 @@ pub mod ERC721EnumerableComponent { } } + /// Returns a list of all token ids owned by the specified `owner`. + /// This function provides a more efficient alternative to calling `ERC721::balance_of` + /// and iterating through tokens with `ERC721Enumerable::token_of_owner_by_index`. + /// + /// Requirements: + /// + /// - `owner` is not the zero address. + fn all_tokens_of_owner( + self: @ComponentState, owner: ContractAddress + ) -> Span { + let mut result = array![]; + let balance = get_dep_component!(self, ERC721).balance_of(owner); + for index in 0 + ..balance { + result.append(self.ERC721Enumerable_owned_tokens.read((owner, index))); + }; + result.span() + } + /// Adds token to this extension's ownership-tracking data structures. fn _add_token_to_owner_enumeration( ref self: ComponentState, to: ContractAddress, token_id: u256 diff --git a/packages/token/src/erc721/extensions/erc721_enumerable/interface.cairo b/packages/token/src/erc721/extensions/erc721_enumerable/interface.cairo index 54ce6a817..ab8ca0c96 100644 --- a/packages/token/src/erc721/extensions/erc721_enumerable/interface.cairo +++ b/packages/token/src/erc721/extensions/erc721_enumerable/interface.cairo @@ -13,3 +13,11 @@ pub trait IERC721Enumerable { fn token_by_index(self: @TState, index: u256) -> u256; fn token_of_owner_by_index(self: @TState, owner: ContractAddress, index: u256) -> u256; } + +#[starknet::interface] +pub trait ERC721EnumerableABI { + fn total_supply(self: @TState) -> u256; + fn token_by_index(self: @TState, index: u256) -> u256; + fn token_of_owner_by_index(self: @TState, owner: ContractAddress, index: u256) -> u256; + fn all_tokens_of_owner(self: @TState, owner: ContractAddress) -> Span; +} diff --git a/packages/token/src/tests/erc721/test_erc721_enumerable.cairo b/packages/token/src/tests/erc721/test_erc721_enumerable.cairo index 1c60b5695..ad0a2f893 100644 --- a/packages/token/src/tests/erc721/test_erc721_enumerable.cairo +++ b/packages/token/src/tests/erc721/test_erc721_enumerable.cairo @@ -3,7 +3,7 @@ use crate::erc721::extensions::erc721_enumerable::ERC721EnumerableComponent::{ ERC721EnumerableImpl, InternalImpl }; use crate::erc721::extensions::erc721_enumerable::ERC721EnumerableComponent; -use crate::erc721::extensions::erc721_enumerable::interface; +use crate::erc721::extensions::erc721_enumerable::interface::IERC721ENUMERABLE_ID; use openzeppelin_introspection::interface::ISRC5_ID; use openzeppelin_introspection::src5::SRC5Component::SRC5Impl; use openzeppelin_test_common::mocks::erc721::ERC721EnumerableMock; @@ -57,7 +57,7 @@ fn test_initializer() { state.initializer(); - let supports_ierc721_enum = mock_state.supports_interface(interface::IERC721ENUMERABLE_ID); + let supports_ierc721_enum = mock_state.supports_interface(IERC721ENUMERABLE_ID); assert!(supports_ierc721_enum); let supports_isrc5 = mock_state.supports_interface(ISRC5_ID); @@ -102,7 +102,7 @@ fn test_token_by_index() { } #[test] -#[should_panic(expected: ('ERC721Enum: out of bounds index',))] +#[should_panic(expected: 'ERC721Enum: out of bounds index')] fn test_token_by_index_equal_to_supply() { let (state, token_list) = setup(); let supply = token_list.len().into(); @@ -111,7 +111,7 @@ fn test_token_by_index_equal_to_supply() { } #[test] -#[should_panic(expected: ('ERC721Enum: out of bounds index',))] +#[should_panic(expected: 'ERC721Enum: out of bounds index')] fn test_token_by_index_greater_than_supply() { let (state, token_list) = setup(); let supply_plus_one = token_list.len().into() + 1; @@ -121,7 +121,7 @@ fn test_token_by_index_greater_than_supply() { #[test] fn test_token_by_index_burn_last_token() { - let (_, _) = setup(); + let _ = setup(); let mut contract_state = CONTRACT_STATE(); let last_token = TOKEN_3; @@ -133,7 +133,7 @@ fn test_token_by_index_burn_last_token() { #[test] fn test_token_by_index_burn_first_token() { - let (_, _) = setup(); + let _ = setup(); let mut contract_state = CONTRACT_STATE(); let first_token = TOKEN_1; @@ -177,7 +177,7 @@ fn test_token_of_owner_by_index() { } #[test] -#[should_panic(expected: ('ERC721Enum: out of bounds index',))] +#[should_panic(expected: 'ERC721Enum: out of bounds index')] fn test_token_of_owner_by_index_when_index_equals_owned_tokens() { let (state, tokens_list) = setup(); let owned_token_len = tokens_list.len().into(); @@ -186,7 +186,7 @@ fn test_token_of_owner_by_index_when_index_equals_owned_tokens() { } #[test] -#[should_panic(expected: ('ERC721Enum: out of bounds index',))] +#[should_panic(expected: 'ERC721Enum: out of bounds index')] fn test_token_of_owner_by_index_when_index_exceeds_owned_tokens() { let (state, tokens_list) = setup(); let owned_tokens_len_plus_one = tokens_list.len().into() + 1; @@ -195,7 +195,7 @@ fn test_token_of_owner_by_index_when_index_exceeds_owned_tokens() { } #[test] -#[should_panic(expected: ('ERC721Enum: out of bounds index',))] +#[should_panic(expected: 'ERC721Enum: out of bounds index')] fn test_token_of_owner_by_index_when_target_has_no_tokens() { let (state, _) = setup(); @@ -203,7 +203,7 @@ fn test_token_of_owner_by_index_when_target_has_no_tokens() { } #[test] -#[should_panic(expected: ('ERC721: invalid account',))] +#[should_panic(expected: 'ERC721: invalid account')] fn test_token_of_owner_by_index_when_owner_is_zero() { let (state, _) = setup(); @@ -503,6 +503,58 @@ fn test__remove_token_from_all_tokens_enumeration_with_first_token() { assert_eq!(initial_supply - 1, new_supply); } +// +// all_tokens_of_owner +// + +#[test] +fn test_all_tokens_of_owner() { + let (_, tokens_list) = setup(); + assert_all_tokens_of_owner(OWNER(), tokens_list); +} + +#[test] +fn test_all_tokens_of_owner_after_transfer_first_token() { + let _ = setup(); + let mut contract_state = CONTRACT_STATE(); + + contract_state.erc721.transfer(OWNER(), RECIPIENT(), TOKEN_1); + + assert_all_tokens_of_owner(OWNER(), array![TOKEN_3, TOKEN_2].span()); + assert_all_tokens_of_owner(RECIPIENT(), array![TOKEN_1].span()); +} + +#[test] +fn test_all_tokens_of_owner_after_transfer_last_token() { + let _ = setup(); + let mut contract_state = CONTRACT_STATE(); + + contract_state.erc721.transfer(OWNER(), RECIPIENT(), TOKEN_3); + + assert_all_tokens_of_owner(OWNER(), array![TOKEN_1, TOKEN_2].span()); + assert_all_tokens_of_owner(RECIPIENT(), array![TOKEN_3].span()); +} + +#[test] +fn test_all_tokens_of_owner_after_burn_first_token() { + let _ = setup(); + let mut contract_state = CONTRACT_STATE(); + + contract_state.erc721.burn(TOKEN_1); + + assert_all_tokens_of_owner(OWNER(), array![TOKEN_3, TOKEN_2].span()); +} + +#[test] +fn test_all_tokens_of_owner_after_burn_last_token() { + let _ = setup(); + let mut contract_state = CONTRACT_STATE(); + + contract_state.erc721.burn(TOKEN_3); + + assert_all_tokens_of_owner(OWNER(), array![TOKEN_1, TOKEN_2].span()); +} + // // Helpers // @@ -568,21 +620,24 @@ fn assert_all_tokens_index_to_id(index: u256, exp_token_id: u256) { fn assert_all_tokens_id_to_index(token_id: u256, exp_index: u256) { let state = @COMPONENT_STATE(); - let id_to_index = state.ERC721Enumerable_all_tokens_index.read(token_id); assert_eq!(id_to_index, exp_index); } fn assert_owner_tokens_index_to_id(owner: ContractAddress, index: u256, exp_token_id: u256) { let state = @COMPONENT_STATE(); - let index_to_id = state.ERC721Enumerable_owned_tokens.read((owner, index)); assert_eq!(index_to_id, exp_token_id); } fn assert_owner_tokens_id_to_index(token_id: u256, exp_index: u256) { let state = @COMPONENT_STATE(); - let id_to_index = state.ERC721Enumerable_owned_tokens_index.read(token_id); assert_eq!(id_to_index, exp_index); } + +fn assert_all_tokens_of_owner(owner: ContractAddress, exp_tokens: Span) { + let state = @COMPONENT_STATE(); + let tokens = state.all_tokens_of_owner(owner); + assert_eq!(tokens, exp_tokens); +}