diff --git a/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.test.ts b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.test.ts new file mode 100644 index 0000000000..1967b4ca96 --- /dev/null +++ b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.test.ts @@ -0,0 +1,952 @@ +import { ControllerMessenger } from '@metamask/base-controller'; +import type { Balance, CaipAssetType } from '@metamask/keyring-api'; +import { EthAccountType, EthMethod, EthScopes } from '@metamask/keyring-api'; +import { KeyringTypes } from '@metamask/keyring-controller'; +import type { InternalAccount } from '@metamask/keyring-internal-api'; +import type { PermissionConstraint } from '@metamask/permission-controller'; +import type { SubjectPermissions } from '@metamask/permission-controller'; +import type { Snap } from '@metamask/snaps-utils'; +import { useFakeTimers } from 'sinon'; +import { v4 as uuidv4 } from 'uuid'; + +import type { + MultichainAssetsControllerMessenger, + MultichainAssetsControllerState, +} from './MultichainAssetsController'; +import { + getDefaultMultichainAssetsControllerState, + MultichainAssetsController, +} from './MultichainAssetsController'; +import { advanceTime } from '../../../../tests/helpers'; +import type { + ExtractAvailableAction, + ExtractAvailableEvent, +} from '../../../base-controller/tests/helpers'; + +const mockSolanaAccount = { + type: 'solana:data-account', + id: 'a3fc6831-d229-4cd1-87c1-13b1756213d4', + address: 'EBBYfhQzVzurZiweJ2keeBWpgGLs1cbWYcz28gjGgi5x', + options: { + scope: 'solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp', + }, + methods: ['sendAndConfirmTransaction'], + metadata: { + name: 'Snap Account 1', + importTime: 1737022568097, + keyring: { + type: 'Snap Keyring', + }, + snap: { + id: 'local:http://localhost:8080', + name: 'Solana', + enabled: true, + }, + lastSelected: 0, + }, +}; + +const mockEthAccount = { + address: '0x807dE1cf8f39E83258904b2f7b473E5C506E4aC1', + id: uuidv4(), + metadata: { + name: 'Ethereum Account 1', + importTime: Date.now(), + keyring: { + type: KeyringTypes.snap, + }, + snap: { + id: 'mock-eth-snap', + name: 'mock-eth-snap', + enabled: true, + }, + lastSelected: 0, + }, + scopes: [EthScopes.Namespace], + options: {}, + methods: [EthMethod.SignTypedDataV4, EthMethod.SignTransaction], + type: EthAccountType.Eoa, +}; + +const mockGetAssetsResult = [ + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/slip44:501', + 'solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1/token:Gh9ZwEmdLJ8DscKNTkTqPbNwLNNBjuSzaG9Vp2KGtKJr', +]; + +const mockGetAllSnapsReturnValue = [ + { + blocked: false, + enabled: true, + id: 'local:http://localhost:8080', + initialPermissions: { + 'endowment:cronjob': { + jobs: [ + { + expression: '* * * * *', + request: { + method: 'refreshTokenPrices', + params: {}, + }, + }, + { + expression: '* * * * *', + request: { + method: 'refreshTransactions', + params: {}, + }, + }, + ], + }, + 'endowment:keyring': { + 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', + ], + }, + 'endowment:network-access': {}, + 'endowment:rpc': { + dapps: true, + snaps: false, + }, + snap_dialog: {}, + snap_getBip32Entropy: [ + { + curve: 'ed25519', + path: ['m', "44'", "501'"], + }, + ], + snap_getPreferences: {}, + snap_manageAccounts: {}, + snap_manageState: {}, + }, + version: '1.0.4', + }, + { + blocked: false, + enabled: true, + id: 'npm:@metamask/account-watcher', + initialPermissions: { + 'endowment:ethereum-provider': {}, + 'endowment:keyring': { + allowedOrigins: ['https://snaps.metamask.io'], + }, + 'endowment:page-home': {}, + 'endowment:rpc': { + allowedOrigins: ['https://snaps.metamask.io'], + }, + snap_dialog: {}, + snap_manageAccounts: {}, + snap_manageState: {}, + }, + version: '4.1.0', + }, + { + blocked: false, + enabled: true, + id: 'npm:@metamask/bitcoin-wallet-snap', + initialPermissions: { + 'endowment:keyring': { + allowedOrigins: [ + 'https://portfolio.metamask.io', + 'https://portfolio-builds.metafi-dev.codefi.network', + 'https://dev.portfolio.metamask.io', + 'https://ramps-dev.portfolio.metamask.io', + ], + }, + 'endowment:network-access': {}, + 'endowment:rpc': { + dapps: true, + snaps: false, + }, + snap_dialog: {}, + snap_getBip32Entropy: [ + { + curve: 'secp256k1', + path: ['m', "84'", "0'"], + }, + { + curve: 'secp256k1', + path: ['m', "84'", "1'"], + }, + ], + snap_manageAccounts: {}, + snap_manageState: {}, + }, + version: '0.8.2', + }, + { + blocked: false, + enabled: true, + id: 'npm:@metamask/ens-resolver-snap', + initialPermissions: { + 'endowment:ethereum-provider': {}, + 'endowment:name-lookup': {}, + 'endowment:network-access': {}, + }, + version: '0.1.2', + }, + { + blocked: false, + enabled: true, + id: 'npm:@metamask/message-signing-snap', + initialPermissions: { + 'endowment:rpc': { + dapps: true, + snaps: false, + }, + snap_getEntropy: {}, + }, + version: '0.6.0', + }, + { + blocked: false, + enabled: true, + id: 'npm:@metamask/preinstalled-example-snap', + initialPermissions: { + 'endowment:rpc': { + dapps: true, + }, + snap_dialog: {}, + }, + version: '0.2.0', + }, + { + blocked: false, + enabled: true, + id: 'npm:@metamask/solana-wallet-snap', + initialPermissions: { + 'endowment:cronjob': { + jobs: [ + { + expression: '* * * * *', + request: { + method: 'refreshTokenPrices', + params: {}, + }, + }, + ], + }, + 'endowment:keyring': { + 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', + ], + }, + 'endowment:network-access': {}, + 'endowment:rpc': { + dapps: true, + snaps: false, + }, + snap_dialog: {}, + snap_getBip32Entropy: [ + { + curve: 'ed25519', + path: ['m', "44'", "501'"], + }, + ], + snap_getPreferences: {}, + snap_manageAccounts: {}, + snap_manageState: {}, + }, + version: '1.0.3', + }, +]; + +const mockGetPermissionsReturnValue = [ + { + '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', + }, + }, + { + 'endowment:ethereum-provider': { + caveats: null, + date: 1736868793768, + id: 'CTUx_19iltoLo-xnIjGMc', + invoker: 'npm:@metamask/account-watcher', + parentCapability: 'endowment:ethereum-provider', + }, + 'endowment:keyring': { + caveats: [ + { + type: 'keyringOrigin', + value: { + allowedOrigins: ['https://snaps.metamask.io'], + }, + }, + ], + date: 1736868793768, + id: 'Kov-E2ET5_VFdUXjmYYHP', + invoker: 'npm:@metamask/account-watcher', + parentCapability: 'endowment:keyring', + }, + 'endowment:page-home': { + caveats: null, + date: 1736868793768, + id: 'c-0RxHEdyaH1ykli6XVBU', + invoker: 'npm:@metamask/account-watcher', + parentCapability: 'endowment:page-home', + }, + 'endowment:rpc': { + caveats: [ + { + type: 'rpcOrigin', + value: { + allowedOrigins: ['https://snaps.metamask.io'], + }, + }, + ], + date: 1736868793768, + id: 'zT7i1dwZutCoah2KhYq0A', + invoker: 'npm:@metamask/account-watcher', + parentCapability: 'endowment:rpc', + }, + snap_dialog: { + caveats: null, + date: 1736868793768, + id: 'y2jAkxa2FLNBxxw8p1GIW', + invoker: 'npm:@metamask/account-watcher', + parentCapability: 'snap_dialog', + }, + snap_manageAccounts: { + caveats: null, + date: 1736868793768, + id: 'Cn3WoTJ-Ute4BIoQncG9H', + invoker: 'npm:@metamask/account-watcher', + parentCapability: 'snap_manageAccounts', + }, + snap_manageState: { + caveats: null, + date: 1736868793768, + id: '3VpjOrbSgm1iiSRdKfSK4', + invoker: 'npm:@metamask/account-watcher', + parentCapability: 'snap_manageState', + }, + }, + { + 'endowment:keyring': { + caveats: [ + { + type: 'keyringOrigin', + value: { + allowedOrigins: [ + 'https://portfolio.metamask.io', + 'https://portfolio-builds.metafi-dev.codefi.network', + 'https://dev.portfolio.metamask.io', + 'https://ramps-dev.portfolio.metamask.io', + ], + }, + }, + ], + date: 1736868793769, + id: 'qrSg-gbPPoWUT0QC-cX_E', + invoker: 'npm:@metamask/bitcoin-wallet-snap', + parentCapability: 'endowment:keyring', + }, + 'endowment:network-access': { + caveats: null, + date: 1736868793769, + id: '9NST-8ZIQO7_BVVJP6JyD', + invoker: 'npm:@metamask/bitcoin-wallet-snap', + parentCapability: 'endowment:network-access', + }, + 'endowment:rpc': { + caveats: [ + { + type: 'rpcOrigin', + value: { + dapps: true, + snaps: false, + }, + }, + ], + date: 1736868793769, + id: 'pmxVKfS_aa7atONpOswiG', + invoker: 'npm:@metamask/bitcoin-wallet-snap', + parentCapability: 'endowment:rpc', + }, + snap_dialog: { + caveats: null, + date: 1736868793769, + id: '2GVLgDfehEN6_gOxHjF9y', + invoker: 'npm:@metamask/bitcoin-wallet-snap', + parentCapability: 'snap_dialog', + }, + snap_getBip32Entropy: { + caveats: [ + { + type: 'permittedDerivationPaths', + value: [ + { + curve: 'secp256k1', + path: ['m', "84'", "0'"], + }, + { + curve: 'secp256k1', + path: ['m', "84'", "1'"], + }, + ], + }, + ], + date: 1736868793769, + id: 'R6fsWjHr1dv1njukHgVc9', + invoker: 'npm:@metamask/bitcoin-wallet-snap', + parentCapability: 'snap_getBip32Entropy', + }, + snap_manageAccounts: { + caveats: null, + date: 1736868793769, + id: 'CIPKC1JmlTcg1vx5m25_y', + invoker: 'npm:@metamask/bitcoin-wallet-snap', + parentCapability: 'snap_manageAccounts', + }, + snap_manageState: { + caveats: null, + date: 1736868793769, + id: 'Hy75pNHCkG899mB02Rey1', + invoker: 'npm:@metamask/bitcoin-wallet-snap', + parentCapability: 'snap_manageState', + }, + }, + { + 'endowment:ethereum-provider': { + caveats: null, + date: 1736868793767, + id: '8cUIGf_BjDke2xJSn_kBL', + invoker: 'npm:@metamask/ens-resolver-snap', + parentCapability: 'endowment:ethereum-provider', + }, + 'endowment:name-lookup': { + caveats: null, + date: 1736868793767, + id: 'y0K_nuZVDc3LP4-5tu9aP', + invoker: 'npm:@metamask/ens-resolver-snap', + parentCapability: 'endowment:name-lookup', + }, + 'endowment:network-access': { + caveats: null, + date: 1736868793767, + id: 'JU4JfpT3aoeo61_KN1pRW', + invoker: 'npm:@metamask/ens-resolver-snap', + parentCapability: 'endowment:network-access', + }, + }, + { + 'endowment:rpc': { + caveats: [ + { + type: 'rpcOrigin', + value: { + dapps: true, + snaps: false, + }, + }, + ], + date: 1736868793765, + id: 'j8XfK-fPq13COl7xFQxXn', + invoker: 'npm:@metamask/message-signing-snap', + parentCapability: 'endowment:rpc', + }, + snap_getEntropy: { + caveats: null, + date: 1736868793765, + id: 'igu3INtYezAnQFXtEZfsD', + invoker: 'npm:@metamask/message-signing-snap', + parentCapability: 'snap_getEntropy', + }, + }, + { + 'endowment:rpc': { + caveats: [ + { + type: 'rpcOrigin', + value: { + dapps: true, + }, + }, + ], + date: 1736868793771, + id: 'Yd155j5BoXh3BIndgMkAM', + invoker: 'npm:@metamask/preinstalled-example-snap', + parentCapability: 'endowment:rpc', + }, + snap_dialog: { + caveats: null, + date: 1736868793771, + id: 'Mg3jlxAPZd-z2ktR_d-s3', + invoker: 'npm:@metamask/preinstalled-example-snap', + parentCapability: 'snap_dialog', + }, + }, + { + 'endowment:cronjob': { + caveats: [ + { + type: 'snapCronjob', + value: { + jobs: [ + { + expression: '* * * * *', + request: { + method: 'refreshTokenPrices', + params: {}, + }, + }, + ], + }, + }, + ], + date: 1736868793773, + id: 'HvFji18XmC4Z8X6aZRX0U', + invoker: 'npm:@metamask/solana-wallet-snap', + 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: 1736868793773, + id: 'gE03aDEESBJhHPOohNF_D', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'endowment:keyring', + }, + 'endowment:network-access': { + caveats: null, + date: 1736868793773, + id: 'HbXb8MLHbRrQMexyVpQQ7', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'endowment:network-access', + }, + 'endowment:rpc': { + caveats: [ + { + type: 'rpcOrigin', + value: { + dapps: true, + snaps: false, + }, + }, + ], + date: 1736868793773, + id: 'c673VaxUJ2XiqpxU-NMwk', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'endowment:rpc', + }, + snap_dialog: { + caveats: null, + date: 1736868793773, + id: '4_LLZgFa6BMO00xXjxUic', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'snap_dialog', + }, + snap_getBip32Entropy: { + caveats: [ + { + type: 'permittedDerivationPaths', + value: [ + { + curve: 'ed25519', + path: ['m', "44'", "501'"], + }, + ], + }, + ], + date: 1736868793773, + id: 'uK-0Ig3Kwoz_hni63t3dl', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'snap_getBip32Entropy', + }, + snap_getPreferences: { + caveats: null, + date: 1736868793773, + id: 'yW6iC0gWWEMJ0TWZMIbDb', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'snap_getPreferences', + }, + snap_manageAccounts: { + caveats: null, + date: 1736868793773, + id: 'G0kZa4d7GekeMNxA76Yqz', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'snap_manageAccounts', + }, + snap_manageState: { + caveats: null, + date: 1736868793773, + id: 'zjugDVmAof_4yztwpZveV', + invoker: 'npm:@metamask/solana-wallet-snap', + parentCapability: 'snap_manageState', + }, + }, +]; + +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', + native: true, + fungible: true, + iconBase64: + '', + units: [{ name: 'USDC', symbol: 'SUSDCOL', decimals: 18 }], + }, +}; + +/** + * The union of actions that the root messenger allows. + */ +type RootAction = ExtractAvailableAction; + +/** + * The union of events that the root messenger allows. + */ +type RootEvent = ExtractAvailableEvent; + +/** + * Constructs the unrestricted messenger. This can be used to call actions and + * publish events within the tests for this controller. + * + * @returns The unrestricted messenger suited for PetNamesController. + */ +function getRootControllerMessenger(): ControllerMessenger< + RootAction, + RootEvent +> { + return new ControllerMessenger(); +} + +const setupController = ({ + state = getDefaultMultichainAssetsControllerState(), + mocks, +}: { + state?: MultichainAssetsControllerState; + mocks?: { + listMultichainAccounts?: InternalAccount[]; + handleRequestReturnValue?: Record; + getAllReturnValue?: Snap[]; + getPermissionsReturnValue?: SubjectPermissions; + }; +} = {}) => { + const controllerMessenger = getRootControllerMessenger(); + + const multichainAssetsControllerMessenger: MultichainAssetsControllerMessenger = + controllerMessenger.getRestricted({ + name: 'MultichainAssetsController', + allowedActions: [ + 'SnapController:handleRequest', + 'AccountsController:listMultichainAccounts', + 'SnapController:getAll', + 'PermissionController:getPermissions', + ], + allowedEvents: [ + 'AccountsController:accountAdded', + 'AccountsController:accountRemoved', + ], + }); + + const mockSnapHandleRequest = jest.fn(); + controllerMessenger.registerActionHandler( + 'SnapController:handleRequest', + mockSnapHandleRequest.mockReturnValue( + mocks?.handleRequestReturnValue ?? mockGetAssetsResult, + ), + ); + + const mockListMultichainAccounts = jest.fn(); + controllerMessenger.registerActionHandler( + 'AccountsController:listMultichainAccounts', + mockListMultichainAccounts.mockReturnValue( + mocks?.listMultichainAccounts ?? [mockSolanaAccount, mockEthAccount], + ), + ); + + const mockGetAllSnaps = jest.fn(); + controllerMessenger.registerActionHandler( + 'SnapController:getAll', + mockGetAllSnaps.mockReturnValue( + mocks?.getAllReturnValue ?? mockGetAllSnapsReturnValue, + ), + ); + + const mockGetPermissions = jest.fn(); + controllerMessenger.registerActionHandler( + 'PermissionController:getPermissions', + mockGetPermissions.mockReturnValue( + mocks?.getPermissionsReturnValue ?? mockGetPermissionsReturnValue, + ), + ); + + const controller = new MultichainAssetsController({ + messenger: multichainAssetsControllerMessenger, + state, + }); + + return { + controller, + messenger: controllerMessenger, + mockSnapHandleRequest, + mockListMultichainAccounts, + mockGetAllSnaps, + mockGetPermissions, + }; +}; + +describe('MultichainAssetsController', () => { + let clock: sinon.SinonFakeTimers; + + beforeEach(() => { + clock = useFakeTimers(); + }); + + afterEach(() => { + clock.restore(); + }); + it('initialize with default state', () => { + const { controller } = setupController({}); + expect(controller.state).toStrictEqual({ + allNonEvmTokens: {}, + allNonEvmIgnoredTokens: {}, + metadata: {}, + }); + }); + + it('should not update state when new account added is EVM', async () => { + const { controller, messenger } = setupController(); + + messenger.publish( + 'AccountsController:accountAdded', + mockEthAccount as unknown as InternalAccount, + ); + + await advanceTime({ clock, duration: 1 }); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: {}, + metadata: {}, + allNonEvmIgnoredTokens: {}, + }); + }); + + it('updates allNonEvmTokens when "AccountsController:accountAdded" is fired', async () => { + const { controller, messenger } = setupController(); + + messenger.publish( + 'AccountsController:accountAdded', + mockSolanaAccount as unknown as InternalAccount, + ); + + await advanceTime({ clock, duration: 1 }); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: { + [mockSolanaAccount.id]: mockGetAssetsResult, + }, + allNonEvmIgnoredTokens: {}, + metadata: mockGetMetadataReturnValue, + }); + }); + + it('should not delete account from allNonEvmTokens when "AccountsController:accountRemoved" is fired with EVM account', async () => { + const { controller, messenger } = setupController(); + // Add a solana account first + messenger.publish( + 'AccountsController:accountAdded', + mockSolanaAccount as unknown as InternalAccount, + ); + + await advanceTime({ clock, duration: 1 }); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: { + [mockSolanaAccount.id]: mockGetAssetsResult, + }, + allNonEvmIgnoredTokens: {}, + metadata: mockGetMetadataReturnValue, + }); + // Remove an EVM account + messenger.publish('AccountsController:accountRemoved', mockEthAccount.id); + + await advanceTime({ clock, duration: 1 }); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: { + [mockSolanaAccount.id]: mockGetAssetsResult, + }, + allNonEvmIgnoredTokens: {}, + metadata: mockGetMetadataReturnValue, + }); + }); + + it('updates allNonEvmTokens when "AccountsController:accountRemoved" is fired', async () => { + const { controller, messenger } = setupController(); + + // Add a solana account first + messenger.publish( + 'AccountsController:accountAdded', + mockSolanaAccount as unknown as InternalAccount, + ); + + await advanceTime({ clock, duration: 1 }); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: { + [mockSolanaAccount.id]: mockGetAssetsResult, + }, + allNonEvmIgnoredTokens: {}, + metadata: mockGetMetadataReturnValue, + }); + // Remove the added solana account + messenger.publish( + 'AccountsController:accountRemoved', + mockSolanaAccount.id, + ); + + await advanceTime({ clock, duration: 1 }); + + expect(controller.state).toStrictEqual({ + allNonEvmTokens: {}, + allNonEvmIgnoredTokens: {}, + metadata: mockGetMetadataReturnValue, + }); + }); +}); diff --git a/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.ts b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.ts new file mode 100644 index 0000000000..eed47d6506 --- /dev/null +++ b/packages/assets-controllers/src/MultichainAssetsController/MultichainAssetsController.ts @@ -0,0 +1,398 @@ +import type { + AccountsControllerAccountAddedEvent, + AccountsControllerAccountRemovedEvent, + AccountsControllerListMultichainAccountsAction, +} from '@metamask/accounts-controller'; +import { + BaseController, + type ControllerGetStateAction, + type ControllerStateChangeEvent, + type RestrictedControllerMessenger, +} from '@metamask/base-controller'; +import { isEvmAccountType } from '@metamask/keyring-api'; +import type { CaipAssetType } from '@metamask/keyring-api'; +import type { InternalAccount } from '@metamask/keyring-internal-api'; +import type { + GetPermissions, + PermissionConstraint, + SubjectPermissions, +} from '@metamask/permission-controller'; +import type { + GetAllSnaps, + HandleSnapRequest, +} from '@metamask/snaps-controllers'; +import type { Snap, SnapId } from '@metamask/snaps-sdk'; +import { HandlerType } from '@metamask/snaps-utils'; + +const controllerName = 'MultichainAssetsController'; + +// Represents an asset unit. +type FungibleAssetUnit = { + // Human-friendly name of the asset unit. + name: string; + + // Ticker symbol of the asset unit. + symbol: string; + + // Number of decimals of the asset unit. + decimals: number; +}; + +// Fungible asset metadata. +type FungibleAssetMetadata = { + // Human-friendly name of the asset. + name: string; + + // Ticker symbol of the asset's main unit. + symbol: string; + + // Whether the asset is native to the chain. + native: boolean; + + // Represents a fungible asset + fungible: true; + + // Base64 representation of the asset icon. + iconBase64: string; + + // List of asset units. + units: FungibleAssetUnit[]; +}; + +// Represents the metadata of an asset. +type AssetMetadata = FungibleAssetMetadata; + +export type MultichainAssetsControllerState = { + metadata: { + [asset: CaipAssetType]: AssetMetadata; + }; + allNonEvmTokens: { [account: string]: CaipAssetType[] }; + allNonEvmIgnoredTokens: { [account: string]: CaipAssetType[] }; +}; + +/** + * Constructs the default {@link MultichainAssetsController} state. This allows + * consumers to provide a partial state object when initializing the controller + * and also helps in constructing complete state objects for this controller in + * tests. + * + * @returns The default {@link MultichainAssetsController} state. + */ +export function getDefaultMultichainAssetsControllerState(): MultichainAssetsControllerState { + return { allNonEvmTokens: {}, allNonEvmIgnoredTokens: {}, metadata: {} }; +} + +/** + * Returns the state of the {@link MultichainAssetsController}. + */ +export type MultichainAssetsControllerGetStateAction = ControllerGetStateAction< + typeof controllerName, + MultichainAssetsControllerState +>; + +/** + * Event emitted when the state of the {@link MultichainAssetsController} changes. + */ +export type MultichainAssetsControllerStateChange = ControllerStateChangeEvent< + typeof controllerName, + MultichainAssetsControllerState +>; + +/** + * Actions exposed by the {@link MultichainAssetsController}. + */ +export type MultichainAssetsControllerActions = + MultichainAssetsControllerGetStateAction; + +/** + * Events emitted by {@link MultichainAssetsController}. + */ +export type MultichainAssetsControllerEvents = + MultichainAssetsControllerStateChange; + +/** + * Actions that this controller is allowed to call. + */ +type AllowedActions = + | HandleSnapRequest + | GetAllSnaps + | GetPermissions + | AccountsControllerListMultichainAccountsAction; + +/** + * Events that this controller is allowed to subscribe. + */ +type AllowedEvents = + | AccountsControllerAccountAddedEvent + | AccountsControllerAccountRemovedEvent; + +type AssetLookupResponse = { + assets: Record; +}; + +/** + * Messenger type for the MultichainAssetsController. + */ +export type MultichainAssetsControllerMessenger = RestrictedControllerMessenger< + typeof controllerName, + MultichainAssetsControllerActions | AllowedActions, + MultichainAssetsControllerEvents | AllowedEvents, + AllowedActions['type'], + AllowedEvents['type'] +>; + +/** + * {@link MultichainAssetsController}'s metadata. + * + * This allows us to choose if fields of the state should be persisted or not + * using the `persist` flag; and if they can be sent to Sentry or not, using + * the `anonymous` flag. + */ +const assetsControllerMetadata = { + metadata: { + persist: true, + anonymous: false, + }, + allNonEvmTokens: { + persist: true, + anonymous: false, + }, + allNonEvmIgnoredTokens: { + persist: true, + anonymous: false, + }, +}; + +// Define a temporary interface for the permission structure +type AssetEndowment = { + 'endowment:assets'?: { + scopes: string[]; + }; +}; + +export class MultichainAssetsController extends BaseController< + typeof controllerName, + MultichainAssetsControllerState, + MultichainAssetsControllerMessenger +> { + constructor({ + messenger, + state = {}, + }: { + messenger: MultichainAssetsControllerMessenger; + state?: Partial; + }) { + super({ + messenger, + name: controllerName, + metadata: assetsControllerMetadata, + state: { + ...getDefaultMultichainAssetsControllerState(), + ...state, + }, + }); + + this.messagingSystem.subscribe( + 'AccountsController:accountAdded', + async (account) => await this.#handleOnAccountAdded(account), + ); + this.messagingSystem.subscribe( + 'AccountsController:accountRemoved', + (account) => this.#handleOnAccountRemoved(account), + ); + } + + /** + * Checks for non-EVM accounts. + * + * @param account - The new account to be checked. + * @returns True if the account is a non-EVM account, false otherwise. + */ + #isNonEvmAccount(account: InternalAccount): boolean { + return ( + !isEvmAccountType(account.type) && + // Non-EVM accounts are backed by a Snap for now + account.metadata.snap !== undefined + ); + } + + /** + * Handles changes when a new account has been added. + * + * @param account - The new account being added. + */ + async #handleOnAccountAdded(account: InternalAccount) { + if (!this.#isNonEvmAccount(account)) { + // Nothing to do here for EVM accounts + return; + } + + // Get assets list + if (account.metadata.snap) { + const assets = await this.#getAssets( + account.id, + account.metadata.snap.id, + ); + const assetsWithoutMetadata = assets.filter( + (asset) => !this.state.metadata[asset], + ); + const snaps = this.#getAllSnaps(); + + const permissions = snaps.map((snap) => + this.#getSnapsPermissions(snap.id), + ); + + /* Mock that every permission returned includeds "endowment:assets": { + "scopes": [ + "bip122:000000000019d6689c085ae165831e93" + ] + } */ + // Mock start To be removed once the above is implemented + permissions.forEach((singlePermission) => { + (singlePermission as unknown as AssetEndowment) = { + ...singlePermission, + 'endowment:assets': { + scopes: ['bip122:000000000019d6689c085ae165831e93'], + }, + }; + }); + (permissions[0] as unknown as AssetEndowment) = { + ...permissions[0], + 'endowment:assets': { + scopes: ['solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1'], + }, + }; + // Mock End To be removed once the above is implemented + + // Identify the correct snap that has the right endowment:assets permission + const currentAssetChain = assets[0].split('/')[0]; + const permissionIndex = permissions.findIndex( + (permission: AssetEndowment) => + permission['endowment:assets']?.scopes.includes(currentAssetChain), + ); + const snapId = snaps[permissionIndex].id; + console.log('🚀 ~ #handleOnAccountAdded ~ snapId:', snapId); + + // call the snap to get the metadata + if (assetsWithoutMetadata.length > 0) { + const metadata = await this.#getMetadata(assetsWithoutMetadata); + + const newMetadata = { + ...this.state.metadata, + ...metadata.assets, + }; + this.update((state) => { + state.metadata = newMetadata; + }); + } + this.update((state) => { + state.allNonEvmTokens[account.id] = assets; + }); + } + } + + /** + * Handles changes when a new account has been removed. + * + * @param accountId - The new account id being removed. + */ + async #handleOnAccountRemoved(accountId: string): Promise { + const selectedAccounts = this.messagingSystem.call( + 'AccountsController:listMultichainAccounts', + ); + + const nonEvmAccounts = selectedAccounts.filter((account) => + this.#isNonEvmAccount(account), + ); + const account: InternalAccount | undefined = nonEvmAccounts.find( + (multichainAccount) => multichainAccount.id === accountId, + ); + if (!account) { + return; + } + + this.update((state) => { + delete state.allNonEvmTokens[accountId]; + }); + } + + #getAllSnaps(): Snap[] { + return this.messagingSystem.call('SnapController:getAll') as Snap[]; + } + + #getSnapsPermissions( + origin: string, + ): SubjectPermissions { + return this.messagingSystem.call( + 'PermissionController:getPermissions', + origin, + ) as SubjectPermissions; + } + + async #getAssets( + accountId: string, + snapId: string, + ): Promise { + return await this.#getAssetsList(snapId, accountId); + } + + /** + * Gets a `KeyringClient` for a Snap. + * + * @param snapId - ID of the Snap to get the client for. + * @param accountId - ID of the account to get the assets for. + * @returns A `KeyringClient` for the Snap. + */ + // TODO: update this to use the snap handler + async #getAssetsList( + snapId: string, + accountId: string, + ): Promise { + const result = (await this.messagingSystem.call( + 'SnapController:handleRequest', + { + snapId: snapId as SnapId, + origin: 'metamask', + handler: HandlerType.OnRpcRequest, + request: { + id: '4dbf133d-9ce3-4d3f-96ac-bfc88d351046', + jsonrpc: '2.0', + method: 'listAccountAssets', + params: { + id: accountId, + }, + }, + }, + )) as CaipAssetType[]; + + return result; + } + + // TODO: update this function to get metadata from the snap + async #getMetadata(assets: CaipAssetType[]): Promise { + console.log('🚀 ~ #getMetadata ~ assets:', assets); + return Promise.resolve({ + assets: { + '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', + native: true, + fungible: true, + iconBase64: + '', + units: [{ name: 'USDC', symbol: 'SUSDCOL', decimals: 18 }], + }, + }, + }); + } +} diff --git a/packages/assets-controllers/src/MultichainAssetsController/index.ts b/packages/assets-controllers/src/MultichainAssetsController/index.ts new file mode 100644 index 0000000000..8086284e61 --- /dev/null +++ b/packages/assets-controllers/src/MultichainAssetsController/index.ts @@ -0,0 +1,11 @@ + +export { MultichainAssetsController } from './MultichainAssetsController'; + +export type { + MultichainAssetsControllerState, + MultichainAssetsControllerGetStateAction, + MultichainAssetsControllerStateChange, + MultichainAssetsControllerActions, + MultichainAssetsControllerEvents, + MultichainAssetsControllerMessenger, +} from './MultichainAssetsController'; diff --git a/packages/assets-controllers/src/index.ts b/packages/assets-controllers/src/index.ts index 410054b59e..b5daf83f94 100644 --- a/packages/assets-controllers/src/index.ts +++ b/packages/assets-controllers/src/index.ts @@ -166,3 +166,17 @@ export type { MultichainBalancesControllerEvents, MultichainBalancesControllerMessenger, } from './MultichainBalancesController'; + +export { + MultichainAssetsController, +} from './MultichainAssetsController'; + +export type { + MultichainAssetsControllerState, + MultichainAssetsControllerGetStateAction, + + MultichainAssetsControllerStateChange, + MultichainAssetsControllerActions, + MultichainAssetsControllerEvents, + MultichainAssetsControllerMessenger, +} from './MultichainAssetsController'; diff --git a/packages/assets-controllers/tsconfig.build.json b/packages/assets-controllers/tsconfig.build.json index 5d38b99686..5e74c070fc 100644 --- a/packages/assets-controllers/tsconfig.build.json +++ b/packages/assets-controllers/tsconfig.build.json @@ -13,7 +13,8 @@ { "path": "../keyring-controller/tsconfig.build.json" }, { "path": "../network-controller/tsconfig.build.json" }, { "path": "../preferences-controller/tsconfig.build.json" }, - { "path": "../polling-controller/tsconfig.build.json" } + { "path": "../polling-controller/tsconfig.build.json" }, + { "path": "../permission-controller/tsconfig.build.json" } ], "include": ["../../types", "./src"] }