From 0552b7d1f7f266bb70e24cb4de596111316e5659 Mon Sep 17 00:00:00 2001 From: Nuo Xu Date: Tue, 12 Dec 2023 13:28:12 -0500 Subject: [PATCH 1/3] fetch Junk NFTs per session when auto-discovery is enabled. only fetch missed balance and metadata. no loading for hide/unhide network icon --- Sources/BraveWallet/Crypto/NFT/NFTView.swift | 10 +-- .../Crypto/Stores/CryptoStore.swift | 12 ++- .../BraveWallet/Crypto/Stores/NFTStore.swift | 75 ++++++++++++------- Tests/BraveWalletTests/NFTStoreTests.swift | 3 +- 4 files changed, 65 insertions(+), 35 deletions(-) diff --git a/Sources/BraveWallet/Crypto/NFT/NFTView.swift b/Sources/BraveWallet/Crypto/NFT/NFTView.swift index a37c19246a0..f3626754f77 100644 --- a/Sources/BraveWallet/Crypto/NFT/NFTView.swift +++ b/Sources/BraveWallet/Crypto/NFT/NFTView.swift @@ -114,12 +114,12 @@ struct NFTView: View { } } .pickerStyle(.inline) - .disabled(nftStore.isShowingNFTLoadingState) + .disabled(nftStore.isLoadingJunkNFTs) } label: { HStack(spacing: 12) { Text(nftStore.displayType.dropdownTitle) .font(.subheadline.weight(.semibold)) - if !nftStore.isShowingNFTLoadingState { + if !nftStore.isLoadingJunkNFTs { Text("\(nftStore.totalDisplayedNFTCount)") .padding(.horizontal, 8) .padding(.vertical, 4) @@ -141,9 +141,9 @@ struct NFTView: View { Spacer() addCustomAssetButton .padding(.trailing, 10) - .disabled(nftStore.isShowingNFTLoadingState) + .disabled(nftStore.isLoadingJunkNFTs) filtersButton - .disabled(nftStore.isShowingNFTLoadingState) + .disabled(nftStore.isLoadingJunkNFTs) } .padding(.horizontal) .frame(maxWidth: .infinity, alignment: .leading) @@ -274,7 +274,7 @@ struct NFTView: View { var body: some View { LazyVStack(spacing: 16) { nftHeaderView - if nftStore.isShowingNFTLoadingState { + if nftStore.isLoadingJunkNFTs { SkeletonLoadingNFTView() } else if nftStore.isShowingNFTEmptyState { emptyView diff --git a/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift b/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift index a70fe3d6e33..71ba696a00e 100644 --- a/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift +++ b/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift @@ -259,11 +259,17 @@ public class CryptoStore: ObservableObject, WalletObserverStore { // CoreData are added after. guard let self else { return } if !self.isUpdatingUserAssets { + let dispatchGroup = DispatchGroup() for asset in discoveredAssets { - self.userAssetManager.addUserAsset(asset, completion: nil) + dispatchGroup.enter() + self.userAssetManager.addUserAsset(asset) { + dispatchGroup.leave() + } } - if !discoveredAssets.isEmpty { - self.updateAssets() + dispatchGroup.notify(queue: .main) { + if !discoveredAssets.isEmpty { + self.updateAssets() + } } } else { self.autoDiscoveredAssets.append(contentsOf: discoveredAssets) diff --git a/Sources/BraveWallet/Crypto/Stores/NFTStore.swift b/Sources/BraveWallet/Crypto/Stores/NFTStore.swift index 19cdce2feed..8bfb69e83d9 100644 --- a/Sources/BraveWallet/Crypto/Stores/NFTStore.swift +++ b/Sources/BraveWallet/Crypto/Stores/NFTStore.swift @@ -133,8 +133,8 @@ public class NFTStore: ObservableObject, WalletObserverStore { @Published var displayType: NFTDisplayType = .visible /// View model for all NFT include visible, hidden and spams @Published private(set) var userNFTGroups: [NFTGroupViewModel] = [] - /// showing shimmering loading state when the view finishes loading NFT information - @Published var isShowingNFTLoadingState: Bool = false + /// showing shimmering loading state when the view is fetching non-fungible tokens without fetching its metadata or balance + @Published var isLoadingJunkNFTs: Bool = false private let keyringService: BraveWalletKeyringService private let rpcService: BraveWalletJsonRpcService @@ -152,7 +152,11 @@ public class NFTStore: ObservableObject, WalletObserverStore { /// Cache of metadata for NFTs. The key is the token's `id`. private var metadataCache: [String: NFTMetadata] = [:] /// Spam from SimpleHash in form of `NetworkAssets` - private var simpleHashSpamNFTs: [NetworkAssets] = [] + private var simpleHashSpamNFTs: [NetworkAssets] = [] { + didSet { + update() + } + } var isObserving: Bool { rpcServiceObserver != nil && keyringServiceObserver != nil && walletServiveObserber != nil @@ -188,6 +192,12 @@ public class NFTStore: ObservableObject, WalletObserverStore { self.setupObservers() + walletService.nftDiscoveryEnabled { [self] enabled in + if enabled { + fetchJunkNFTs() + } + } + keyringService.isLocked { [self] isLocked in if !isLocked { update() @@ -249,7 +259,6 @@ public class NFTStore: ObservableObject, WalletObserverStore { private var nftBalancesCache: [String: [String: Int]] = [:] func update() { - self.isShowingNFTLoadingState = true self.updateTask?.cancel() self.updateTask = Task { @MainActor in self.allAccounts = await keyringService.allAccounts().accounts @@ -261,14 +270,13 @@ public class NFTStore: ObservableObject, WalletObserverStore { let selectedAccounts = filters.accounts.filter(\.isSelected).map(\.model) let selectedNetworks = filters.networks.filter(\.isSelected).map(\.model) - // all spam NFTs marked by SimpleHash - simpleHashSpamNFTs = await walletService.simpleHashSpamNFTs(for: selectedAccounts, on: selectedNetworks) + // First display grids with placeholder since we haven't fetched balance + // or metadata let unionedSpamNFTs = computeSpamNFTs( selectedNetworks: selectedNetworks, selectedAccounts: selectedAccounts, simpleHashSpamNFTs: simpleHashSpamNFTs ) - let (userNFTGroups, allUserNFTs) = buildNFTGroupModels( groupBy: filters.groupBy, spams: unionedSpamNFTs, @@ -277,13 +285,14 @@ public class NFTStore: ObservableObject, WalletObserverStore { ) self.userNFTGroups = userNFTGroups + // Then, we fetch balance and update the UI // if we're not hiding unowned or grouping by account, balance isn't needed if filters.isHidingUnownedNFTs || filters.groupBy == .accounts { let allAccounts = filters.accounts.map(\.model) nftBalancesCache = await withTaskGroup( of: [String: [String: Int]].self, body: { @MainActor [nftBalancesCache, rpcService] group in - for nft in allUserNFTs { // for each NFT + for nft in allUserNFTs { // for all NFTs that have not yet been fetched its balance guard let networkForNFT = allNetworks.first(where: { $0.chainId == nft.chainId }) else { continue } @@ -292,13 +301,17 @@ public class NFTStore: ObservableObject, WalletObserverStore { of: [String: Int].self, body: { @MainActor group in for account in allAccounts where account.coin == nft.coin { - group.addTask { @MainActor in - let balanceForToken = await rpcService.balance( - for: nft, - in: account, - network: networkForNFT - ) - return [account.address: Int(balanceForToken ?? 0)] + if let cachedBalance = nftBalancesCache[nft.id]?[account.address] { // cached balance + return [account.address: cachedBalance] + } else { // no balance for this account + group.addTask { @MainActor in + let balanceForToken = await rpcService.balance( + for: nft, + in: account, + network: networkForNFT + ) + return [account.address: Int(balanceForToken ?? 0)] + } } } return await group.reduce(into: [String: Int](), { partialResult, new in @@ -315,7 +328,6 @@ public class NFTStore: ObservableObject, WalletObserverStore { }) }) } - guard !Task.isCancelled else { return } let (userNFTGroupsWithBalance, _) = buildNFTGroupModels( groupBy: filters.groupBy, @@ -325,11 +337,12 @@ public class NFTStore: ObservableObject, WalletObserverStore { ) self.userNFTGroups = userNFTGroupsWithBalance - // fetch nft metadata for all NFTs - let allMetadata = await rpcService.fetchNFTMetadata(tokens: allUserNFTs, ipfsApi: ipfsApi) - for (key, value) in allMetadata { // update cached values - metadataCache[key] = value - } + // Last, we fet fetch nft metadata for NFTs that do not have metadata loaded + // and update the UI + let nftsMissingMetadata = allUserNFTs.filter { metadataCache[$0.id] == nil } + let fetchedMetadata = await rpcService.fetchNFTMetadata(tokens: nftsMissingMetadata, ipfsApi: ipfsApi) + metadataCache.merge(with: fetchedMetadata) + guard !Task.isCancelled else { return } let (userNFTGroupsWithMetadata, _) = buildNFTGroupModels( groupBy: filters.groupBy, @@ -338,8 +351,6 @@ public class NFTStore: ObservableObject, WalletObserverStore { selectedNetworks: selectedNetworks ) self.userNFTGroups = userNFTGroupsWithMetadata - - isShowingNFTLoadingState = false } } @@ -569,12 +580,27 @@ public class NFTStore: ObservableObject, WalletObserverStore { return (groups, allUserNFTs) } + private func fetchJunkNFTs() { + Task { @MainActor in + // all spam NFTs marked by SimpleHash (for all accounts on all networks) + self.isLoadingJunkNFTs = true + let allAccounts = await keyringService.allAccounts().accounts + .filter { account in + WalletConstants.supportedCoinTypes().contains(account.coin) + } + let allNetworks = await rpcService.allNetworksForSupportedCoins() + self.simpleHashSpamNFTs = await walletService.simpleHashSpamNFTs(for: allAccounts, on: allNetworks) + self.isLoadingJunkNFTs = false + } + } + @MainActor func isNFTDiscoveryEnabled() async -> Bool { await walletService.nftDiscoveryEnabled() } func enableNFTDiscovery() { walletService.setNftDiscoveryEnabled(true) + fetchJunkNFTs() // junk NFTs is only returned when auto-discovery is enabled } func updateNFTStatus( @@ -583,7 +609,6 @@ public class NFTStore: ObservableObject, WalletObserverStore { isSpam: Bool, isDeletedByUser: Bool ) { - isShowingNFTLoadingState = true assetManager.updateUserAsset( for: token, visible: visible, @@ -606,8 +631,6 @@ public class NFTStore: ObservableObject, WalletObserverStore { selectedAccounts: selectedAccounts, selectedNetworks: selectedNetworks ) - - isShowingNFTLoadingState = false } } diff --git a/Tests/BraveWalletTests/NFTStoreTests.swift b/Tests/BraveWalletTests/NFTStoreTests.swift index 0a042ecaf8d..c4dd157dd0f 100644 --- a/Tests/BraveWalletTests/NFTStoreTests.swift +++ b/Tests/BraveWalletTests/NFTStoreTests.swift @@ -121,13 +121,14 @@ class NFTStoreTests: XCTestCase { } let walletService = BraveWallet.TestBraveWalletService() walletService._addObserver = { _ in } - walletService._simpleHashSpamNfTs = {walletAddress, chainIds, coin, _, completion in + walletService._simpleHashSpamNfTs = { walletAddress, chainIds, coin, _, completion in if walletAddress == self.ethAccount1.address, chainIds.contains(BraveWallet.MainnetChainId), coin == .eth { completion([self.spamEthNFT], nil) } else { completion([], nil) } } + walletService._nftDiscoveryEnabled = { $0(true) } let assetRatioService = BraveWallet.TestAssetRatioService() let mockAssetManager = TestableWalletUserAssetManager() From 46bf9399b5133cfe1b6b5e2c500a5af38deb27fb Mon Sep 17 00:00:00 2001 From: Nuo Xu Date: Tue, 12 Dec 2023 16:09:05 -0500 Subject: [PATCH 2/3] fetch junk NFTs after auto-discover assets is completed. fix unit tests. --- Sources/BraveWallet/Crypto/Stores/CryptoStore.swift | 10 +++++++++- Sources/BraveWallet/Crypto/Stores/NFTStore.swift | 9 +-------- Tests/BraveWalletTests/NFTStoreTests.swift | 8 ++++---- 3 files changed, 14 insertions(+), 13 deletions(-) diff --git a/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift b/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift index 71ba696a00e..258e361960e 100644 --- a/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift +++ b/Sources/BraveWallet/Crypto/Stores/CryptoStore.swift @@ -268,7 +268,7 @@ public class CryptoStore: ObservableObject, WalletObserverStore { } dispatchGroup.notify(queue: .main) { if !discoveredAssets.isEmpty { - self.updateAssets() + self.updateAutoDiscoveredAssets() } } } else { @@ -558,6 +558,14 @@ public class CryptoStore: ObservableObject, WalletObserverStore { nftStore.update() } + func updateAutoDiscoveredAssets() { + // at this point, all auto-discovered assets have been added to CD + // update `Portfolio/Assets` + portfolioStore.update() + // fetch junk NFTs from SimpleHash which will also update `Portfolio/NFTs` + nftStore.fetchJunkNFTs() + } + func prepare(isInitialOpen: Bool = false) { Task { @MainActor in if isInitialOpen { diff --git a/Sources/BraveWallet/Crypto/Stores/NFTStore.swift b/Sources/BraveWallet/Crypto/Stores/NFTStore.swift index 8bfb69e83d9..eb99963adbf 100644 --- a/Sources/BraveWallet/Crypto/Stores/NFTStore.swift +++ b/Sources/BraveWallet/Crypto/Stores/NFTStore.swift @@ -192,12 +192,6 @@ public class NFTStore: ObservableObject, WalletObserverStore { self.setupObservers() - walletService.nftDiscoveryEnabled { [self] enabled in - if enabled { - fetchJunkNFTs() - } - } - keyringService.isLocked { [self] isLocked in if !isLocked { update() @@ -580,7 +574,7 @@ public class NFTStore: ObservableObject, WalletObserverStore { return (groups, allUserNFTs) } - private func fetchJunkNFTs() { + func fetchJunkNFTs() { Task { @MainActor in // all spam NFTs marked by SimpleHash (for all accounts on all networks) self.isLoadingJunkNFTs = true @@ -600,7 +594,6 @@ public class NFTStore: ObservableObject, WalletObserverStore { func enableNFTDiscovery() { walletService.setNftDiscoveryEnabled(true) - fetchJunkNFTs() // junk NFTs is only returned when auto-discovery is enabled } func updateNFTStatus( diff --git a/Tests/BraveWalletTests/NFTStoreTests.swift b/Tests/BraveWalletTests/NFTStoreTests.swift index c4dd157dd0f..322a6e269f8 100644 --- a/Tests/BraveWalletTests/NFTStoreTests.swift +++ b/Tests/BraveWalletTests/NFTStoreTests.swift @@ -128,7 +128,6 @@ class NFTStoreTests: XCTestCase { completion([], nil) } } - walletService._nftDiscoveryEnabled = { $0(true) } let assetRatioService = BraveWallet.TestAssetRatioService() let mockAssetManager = TestableWalletUserAssetManager() @@ -477,7 +476,7 @@ class NFTStoreTests: XCTestCase { XCTAssertEqual(spamNFTs[safe: 0]?.nftMetadata?.name, self.mockERC721Metadata.name) }.store(in: &cancellables) - store.update() + store.fetchJunkNFTs() await fulfillment(of: [userSpamNFTsException], timeout: 1) cancellables.removeAll() } @@ -547,7 +546,7 @@ class NFTStoreTests: XCTestCase { XCTAssertEqual(spamNFTs[safe: 1]?.nftMetadata?.name, self.mockERC721Metadata.name) }.store(in: &cancellables) - store.update() + store.fetchJunkNFTs() await fulfillment(of: [userSpamNFTsException], timeout: 1) cancellables.removeAll() } @@ -615,7 +614,7 @@ class NFTStoreTests: XCTestCase { XCTAssertEqual(spamNFTs[safe: 0]?.nftMetadata?.name, self.mockERC721Metadata.name) }.store(in: &cancellables) - store.update() + store.fetchJunkNFTs() await fulfillment(of: [userSpamNFTsException], timeout: 1) cancellables.removeAll() } @@ -689,6 +688,7 @@ class NFTStoreTests: XCTestCase { await fulfillment(of: [groupByAccountVisibleExpectation], timeout: 1) cancellables.removeAll() } + // MARK: Group by `Accounts` with `displayType`: `hidden` func testUpdateGroupByAccountsHiddenNFTs() async { let mockEthUserAssets: [BraveWallet.BlockchainToken] = [ From f086e5c1f820c55f45aa553a5f1d066174bf7c10 Mon Sep 17 00:00:00 2001 From: Nuo Xu Date: Tue, 12 Dec 2023 16:44:22 -0500 Subject: [PATCH 3/3] update unit tests name --- Tests/BraveWalletTests/NFTStoreTests.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Tests/BraveWalletTests/NFTStoreTests.swift b/Tests/BraveWalletTests/NFTStoreTests.swift index 322a6e269f8..69791e615fa 100644 --- a/Tests/BraveWalletTests/NFTStoreTests.swift +++ b/Tests/BraveWalletTests/NFTStoreTests.swift @@ -414,7 +414,7 @@ class NFTStoreTests: XCTestCase { } // MARK: Group By `None` - func testUpdateSpamGroupOnlyFromSimpleHash() async { + func testSpamOnlyFromSimpleHash() async { let mockEthUserAssets: [BraveWallet.BlockchainToken] = [ .previewToken.copy(asVisibleAsset: true), .mockUSDCToken.copy(asVisibleAsset: false), // Verify non-visible assets not displayed #6386 @@ -482,7 +482,7 @@ class NFTStoreTests: XCTestCase { } // MARK: Group By `None` - func testUpdateSpamGroupFromSimpleHashAndUserMarked() async { + func testSpamFromSimpleHashAndUserMarked() async { let mockEthUserAssets: [BraveWallet.BlockchainToken] = [ .previewToken.copy(asVisibleAsset: true), .mockUSDCToken.copy(asVisibleAsset: false), // Verify non-visible assets not displayed #6386 @@ -552,7 +552,7 @@ class NFTStoreTests: XCTestCase { } // MARK: Group By `None` - func testUpdateSpamGroupDuplicationFromSimpleHashAndUserMarked() async { + func testSpamDuplicationFromSimpleHashAndUserMarked() async { let mockEthUserAssets: [BraveWallet.BlockchainToken] = [ .previewToken.copy(asVisibleAsset: true), .mockUSDCToken.copy(asVisibleAsset: false), // Verify non-visible assets not displayed #6386