From 22fe8749819db65447a7b3f4aeca3bb8117a672a Mon Sep 17 00:00:00 2001 From: sahar-fehri Date: Thu, 30 Jan 2025 16:22:31 +0100 Subject: [PATCH] fix: fix metadata response type and tests --- .../MultichainAssetsController.test.ts | 412 ++++++++++++++++-- .../MultichainAssetsController.ts | 54 ++- 2 files changed, 420 insertions(+), 46 deletions(-) diff --git a/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.test.ts b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.test.ts index d0d8854c9e4..3a357f64c02 100644 --- a/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.test.ts +++ b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.test.ts @@ -13,6 +13,7 @@ import { useFakeTimers } from 'sinon'; import { v4 as uuidv4 } from 'uuid'; import type { + AssetMetadataResponse, MultichainAssetsControllerMessenger, MultichainAssetsControllerState, } from './MultichainAssetsController'; @@ -721,26 +722,26 @@ const mockGetPermissionsReturnValue = [ }, ]; -const mockGetMetadataReturnValue = { - 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/slip44:501': { - name: 'Solana', - symbol: 'SOL', - native: true, - fungible: true, - iconBase64: - '', - units: [{ name: 'Solana', symbol: 'SOL', decimals: 9 }], - }, - 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/token:Gh9ZwEmdLJ8DscKNTkTqPbNwLNNBjuSzaG9Vp2KGtKJr': - { - name: 'USDC', - symbol: 'USDC', +const mockGetMetadataReturnValue: AssetMetadataResponse | undefined = { + assets: { + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/slip44:501': { + name: 'Solana', + symbol: 'SOL', native: true, fungible: true, - iconBase64: - '', - units: [{ name: 'USDC', symbol: 'SUSDCOL', decimals: 18 }], + iconUrl: 'url1', + units: [{ name: 'Solana', symbol: 'SOL', decimals: 9 }], }, + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/token:Gh9ZwEmdLJ8DscKNTkTqPbNwLNNBjuSzaG9Vp2KGtKJr': + { + name: 'USDC', + symbol: 'USDC', + native: true, + fungible: true, + iconUrl: 'url2', + units: [{ name: 'USDC', symbol: 'SUSDCOL', decimals: 18 }], + }, + }, }; /** @@ -904,7 +905,364 @@ describe('MultichainAssetsController', () => { allNonEvmTokens: { [mockSolanaAccount.id]: mockGetAssetsResult, }, - metadata: mockGetMetadataReturnValue, + metadata: mockGetMetadataReturnValue.assets, + }); + }); + + it('updates metadata in state successfully all calls succeed to fetch metadata', async () => { + const { controller, messenger, mockSnapHandleRequest, mockGetPermissions } = + setupController(); + + const mockAssetsResponse = [ + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/slip44:501', + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/token:Gh9ZwEmdLJ8DscKNTkTqPbNwLNNBjuSzaG9Vp2KGtKJr', + 'solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp/slip44:501', + ]; + const mockSnapPermissionReturnVal = { + 'endowment:assets': { + caveats: [ + { + type: 'chainIds', + value: [ + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1', + 'solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp', + ], + }, + ], + }, + 'endowment:cronjob': { + caveats: [ + { + type: 'snapCronjob', + value: { + jobs: [ + { + expression: '* * * * *', + request: { + method: 'refreshTokenPrices', + params: {}, + }, + }, + { + expression: '* * * * *', + request: { + method: 'refreshTransactions', + params: {}, + }, + }, + ], + }, + }, + ], + date: 1736869806349, + id: 'OQ3d1RXEPbcvi3sNzP6GV', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:cronjob', + }, + 'endowment:keyring': { + caveats: [ + { + type: 'keyringOrigin', + value: { + allowedOrigins: [ + 'http://localhost:3000', + 'https://portfolio.metamask.io', + 'https://portfolio-builds.metafi-dev.codefi.network', + 'https://dev.portfolio.metamask.io', + 'https://ramps-dev.portfolio.metamask.io', + ], + }, + }, + ], + date: 1736869806348, + id: 'WkNypQHrfAaP8VyDZZmIG', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:keyring', + }, + 'endowment:network-access': { + caveats: null, + date: 1736869806348, + id: 'wMPlAtMQCzt6TeQe4j3f-', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:network-access', + }, + 'endowment:rpc': { + caveats: [ + { + type: 'rpcOrigin', + value: { + dapps: true, + snaps: false, + }, + }, + ], + date: 1736869806348, + id: '7j1Elw4kcL4KOnlQ6xt7A', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:rpc', + }, + snap_dialog: { + caveats: null, + date: 1736869806349, + id: '2OtPNZTdpK1FadSM0s0rv', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_dialog', + }, + snap_getBip32Entropy: { + caveats: [ + { + type: 'permittedDerivationPaths', + value: [ + { + curve: 'ed25519', + path: ['m', "44'", "501'"], + }, + ], + }, + ], + date: 1736869806348, + id: 'WiOkUgu4Y89p2youlm7ro', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_getBip32Entropy', + }, + snap_getPreferences: { + caveats: null, + date: 1736869806349, + id: 'LmYGhkzdyDWmhAbM1UJfx', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_getPreferences', + }, + snap_manageAccounts: { + caveats: null, + date: 1736869806349, + id: 'BaiVanA9U-BIfbCgYpeo6', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_manageAccounts', + }, + snap_manageState: { + caveats: null, + date: 1736869806349, + id: 'oIJeVj2SsWh6uFam1RMi4', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_manageState', + }, + }; + const mockGetMetadataResponse: AssetMetadataResponse | undefined = { + assets: { + 'solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp/slip44:501': { + name: 'Solana2', + symbol: 'SOL', + native: true, + fungible: true, + iconUrl: 'url1', + units: [{ name: 'Solana2', symbol: 'SOL', decimals: 9 }], + }, + }, + }; + + mockSnapHandleRequest + .mockReturnValueOnce(mockAssetsResponse) + .mockReturnValueOnce(mockGetMetadataReturnValue) + .mockReturnValueOnce(mockGetMetadataResponse); + + mockGetPermissions + .mockReturnValueOnce(mockSnapPermissionReturnVal) + .mockReturnValueOnce(mockGetPermissionsReturnValue[1]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[2]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[3]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[4]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[5]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[6]); + + messenger.publish( + 'AccountsController:accountAdded', + mockSolanaAccount as unknown as InternalAccount, + ); + + await advanceTime({ clock, duration: 1 }); + + expect(mockSnapHandleRequest).toHaveBeenCalledTimes(3); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: { + [mockSolanaAccount.id]: mockAssetsResponse, + }, + metadata: { + ...mockGetMetadataResponse.assets, + ...mockGetMetadataReturnValue.assets, + }, + }); + }); + + it('updates metadata in state successfully one call to fetch metadata fails', async () => { + const { controller, messenger, mockSnapHandleRequest, mockGetPermissions } = + setupController(); + + const mockAssetsResponse = [ + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/slip44:501', + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/token:Gh9ZwEmdLJ8DscKNTkTqPbNwLNNBjuSzaG9Vp2KGtKJr', + 'solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp/slip44:501', + ]; + const mockSnapPermissionReturnVal = { + 'endowment:assets': { + caveats: [ + { + type: 'chainIds', + value: [ + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1', + 'solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp', + ], + }, + ], + }, + 'endowment:cronjob': { + caveats: [ + { + type: 'snapCronjob', + value: { + jobs: [ + { + expression: '* * * * *', + request: { + method: 'refreshTokenPrices', + params: {}, + }, + }, + { + expression: '* * * * *', + request: { + method: 'refreshTransactions', + params: {}, + }, + }, + ], + }, + }, + ], + date: 1736869806349, + id: 'OQ3d1RXEPbcvi3sNzP6GV', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:cronjob', + }, + 'endowment:keyring': { + caveats: [ + { + type: 'keyringOrigin', + value: { + allowedOrigins: [ + 'http://localhost:3000', + 'https://portfolio.metamask.io', + 'https://portfolio-builds.metafi-dev.codefi.network', + 'https://dev.portfolio.metamask.io', + 'https://ramps-dev.portfolio.metamask.io', + ], + }, + }, + ], + date: 1736869806348, + id: 'WkNypQHrfAaP8VyDZZmIG', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:keyring', + }, + 'endowment:network-access': { + caveats: null, + date: 1736869806348, + id: 'wMPlAtMQCzt6TeQe4j3f-', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:network-access', + }, + 'endowment:rpc': { + caveats: [ + { + type: 'rpcOrigin', + value: { + dapps: true, + snaps: false, + }, + }, + ], + date: 1736869806348, + id: '7j1Elw4kcL4KOnlQ6xt7A', + invoker: 'local:http://localhost:8080', + parentCapability: 'endowment:rpc', + }, + snap_dialog: { + caveats: null, + date: 1736869806349, + id: '2OtPNZTdpK1FadSM0s0rv', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_dialog', + }, + snap_getBip32Entropy: { + caveats: [ + { + type: 'permittedDerivationPaths', + value: [ + { + curve: 'ed25519', + path: ['m', "44'", "501'"], + }, + ], + }, + ], + date: 1736869806348, + id: 'WiOkUgu4Y89p2youlm7ro', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_getBip32Entropy', + }, + snap_getPreferences: { + caveats: null, + date: 1736869806349, + id: 'LmYGhkzdyDWmhAbM1UJfx', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_getPreferences', + }, + snap_manageAccounts: { + caveats: null, + date: 1736869806349, + id: 'BaiVanA9U-BIfbCgYpeo6', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_manageAccounts', + }, + snap_manageState: { + caveats: null, + date: 1736869806349, + id: 'oIJeVj2SsWh6uFam1RMi4', + invoker: 'local:http://localhost:8080', + parentCapability: 'snap_manageState', + }, + }; + + mockSnapHandleRequest + .mockReturnValueOnce(mockAssetsResponse) + .mockReturnValueOnce(mockGetMetadataReturnValue) + .mockRejectedValueOnce('Error'); + + mockGetPermissions + .mockReturnValueOnce(mockSnapPermissionReturnVal) + .mockReturnValueOnce(mockGetPermissionsReturnValue[1]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[2]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[3]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[4]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[5]) + .mockReturnValueOnce(mockGetPermissionsReturnValue[6]); + + messenger.publish( + 'AccountsController:accountAdded', + mockSolanaAccount as unknown as InternalAccount, + ); + + await advanceTime({ clock, duration: 1 }); + + expect(mockSnapHandleRequest).toHaveBeenCalledTimes(3); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: { + [mockSolanaAccount.id]: mockAssetsResponse, + }, + metadata: { + ...mockGetMetadataReturnValue.assets, + }, }); }); @@ -938,7 +1296,7 @@ describe('MultichainAssetsController', () => { [mockSolanaAccount.id]: mockGetAssetsResult, }, - metadata: mockGetMetadataReturnValue, + metadata: mockGetMetadataReturnValue.assets, }); // Remove an EVM account messenger.publish('AccountsController:accountRemoved', mockEthAccount.id); @@ -950,7 +1308,7 @@ describe('MultichainAssetsController', () => { [mockSolanaAccount.id]: mockGetAssetsResult, }, - metadata: mockGetMetadataReturnValue, + metadata: mockGetMetadataReturnValue.assets, }); }); @@ -984,7 +1342,7 @@ describe('MultichainAssetsController', () => { [mockSolanaAccount.id]: mockGetAssetsResult, }, - metadata: mockGetMetadataReturnValue, + metadata: mockGetMetadataReturnValue.assets, }); // Remove the added solana account messenger.publish( @@ -997,7 +1355,7 @@ describe('MultichainAssetsController', () => { expect(controller.state).toStrictEqual({ allNonEvmTokens: {}, - metadata: mockGetMetadataReturnValue, + metadata: mockGetMetadataReturnValue.assets, }); }); @@ -1011,7 +1369,7 @@ describe('MultichainAssetsController', () => { allNonEvmTokens: { [mockSolanaAccountId1]: mockGetAssetsResult, }, - metadata: mockGetMetadataReturnValue, + metadata: mockGetMetadataReturnValue.assets, } as MultichainAssetsControllerState, }); @@ -1030,8 +1388,10 @@ describe('MultichainAssetsController', () => { }, }; mockSnapHandleRequest.mockReturnValue({ - ...mockGetMetadataReturnValue1, - ...mockGetMetadataReturnValue2, + assets: { + ...mockGetMetadataReturnValue1, + ...mockGetMetadataReturnValue2, + }, }); mockGetPermissions @@ -1077,7 +1437,7 @@ describe('MultichainAssetsController', () => { expect(mockSnapHandleRequest).toHaveBeenCalledTimes(1); expect(controller.state.metadata).toStrictEqual({ - ...mockGetMetadataReturnValue, + ...mockGetMetadataReturnValue.assets, 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/token:newToken': { name: 'newToken', symbol: 'newToken', diff --git a/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.ts b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.ts index a32b0a5f032..a93dc32444e 100644 --- a/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.ts +++ b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.ts @@ -62,8 +62,8 @@ type FungibleAssetMetadata = { // Represents a fungible asset fungible: true; - // Base64 representation of the asset icon. - iconBase64: string; + // icon url + iconUrl: string; // List of asset units. units: FungibleAssetUnit[]; @@ -79,6 +79,13 @@ export type MultichainAssetsControllerState = { allNonEvmTokens: { [account: string]: CaipAssetType[] }; }; +// Represents the response of the asset snap's onAssetLookup handler +export type AssetMetadataResponse = { + assets: { + [asset: CaipAssetType]: AssetMetadata; + }; +}; + /** * Constructs the default {@link MultichainAssetsController} state. This allows * consumers to provide a partial state object when initializing the controller @@ -333,8 +340,7 @@ export class MultichainAssetsController extends BaseController< } assetsByScope[chainId].push(asset); } - - let newMetadata: Record; + let newMetadata: Record = {}; for (const chainId in assetsByScope) { if (hasProperty(assetsByScope, chainId)) { const assetsForChain = assetsByScope[chainId as CaipChainId]; @@ -343,14 +349,17 @@ export class MultichainAssetsController extends BaseController< if (snap) { const metadata = await this.#getMetadata(assetsForChain, snap.id); newMetadata = { - ...this.state.metadata, - ...metadata, + ...newMetadata, + ...(metadata ? metadata.assets : {}), }; } } } this.update((state) => { - state.metadata = newMetadata; + state.metadata = { + ...this.state.metadata, + ...newMetadata, + }; }); } @@ -435,20 +444,25 @@ export class MultichainAssetsController extends BaseController< async #getMetadata( assets: CaipAssetType[], snapId: string, - ): Promise> { - return (await this.messagingSystem.call('SnapController:handleRequest', { - snapId: snapId as SnapId, - origin: 'metamask', - handler: HandlerType.OnAssetsLookup, - request: { - id: '4dbf133d-9ce3-4d3f-96ac-bfc88d351046', - jsonrpc: '2.0', - method: 'onAssetLookup', - params: { - assets, + ): Promise { + try { + return (await this.messagingSystem.call('SnapController:handleRequest', { + snapId: snapId as SnapId, + origin: 'metamask', + handler: HandlerType.OnAssetsLookup, + request: { + id: '4dbf133d-9ce3-4d3f-96ac-bfc88d351046', + jsonrpc: '2.0', + method: 'onAssetLookup', + params: { + assets, + }, }, - }, - })) as Record; + })) as Promise; + } catch { + // ignore + return undefined; + } } /**