@metamask-previews/assets-controller 2.2.0-preview-384cfdfef → 2.2.0-preview-15dd7d63f

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (44) hide show
  1. package/CHANGELOG.md +1 -2
  2. package/dist/AssetsController-method-action-types.cjs.map +1 -1
  3. package/dist/AssetsController-method-action-types.d.cts +4 -0
  4. package/dist/AssetsController-method-action-types.d.cts.map +1 -1
  5. package/dist/AssetsController-method-action-types.d.mts +4 -0
  6. package/dist/AssetsController-method-action-types.d.mts.map +1 -1
  7. package/dist/AssetsController-method-action-types.mjs.map +1 -1
  8. package/dist/AssetsController.cjs +8 -2
  9. package/dist/AssetsController.cjs.map +1 -1
  10. package/dist/AssetsController.d.cts +10 -2
  11. package/dist/AssetsController.d.cts.map +1 -1
  12. package/dist/AssetsController.d.mts +10 -2
  13. package/dist/AssetsController.d.mts.map +1 -1
  14. package/dist/AssetsController.mjs +8 -2
  15. package/dist/AssetsController.mjs.map +1 -1
  16. package/dist/data-sources/PriceDataSource.cjs +29 -48
  17. package/dist/data-sources/PriceDataSource.cjs.map +1 -1
  18. package/dist/data-sources/PriceDataSource.d.cts.map +1 -1
  19. package/dist/data-sources/PriceDataSource.d.mts.map +1 -1
  20. package/dist/data-sources/PriceDataSource.mjs +29 -48
  21. package/dist/data-sources/PriceDataSource.mjs.map +1 -1
  22. package/dist/types.cjs.map +1 -1
  23. package/dist/types.d.cts +11 -9
  24. package/dist/types.d.cts.map +1 -1
  25. package/dist/types.d.mts +11 -9
  26. package/dist/types.d.mts.map +1 -1
  27. package/dist/types.mjs.map +1 -1
  28. package/dist/utils/formatExchangeRatesForBridge.cjs +60 -51
  29. package/dist/utils/formatExchangeRatesForBridge.cjs.map +1 -1
  30. package/dist/utils/formatExchangeRatesForBridge.d.cts +3 -1
  31. package/dist/utils/formatExchangeRatesForBridge.d.cts.map +1 -1
  32. package/dist/utils/formatExchangeRatesForBridge.d.mts +3 -1
  33. package/dist/utils/formatExchangeRatesForBridge.d.mts.map +1 -1
  34. package/dist/utils/formatExchangeRatesForBridge.mjs +61 -52
  35. package/dist/utils/formatExchangeRatesForBridge.mjs.map +1 -1
  36. package/dist/utils/formatStateForTransactionPay.cjs +12 -4
  37. package/dist/utils/formatStateForTransactionPay.cjs.map +1 -1
  38. package/dist/utils/formatStateForTransactionPay.d.cts +2 -0
  39. package/dist/utils/formatStateForTransactionPay.d.cts.map +1 -1
  40. package/dist/utils/formatStateForTransactionPay.d.mts +2 -0
  41. package/dist/utils/formatStateForTransactionPay.d.mts.map +1 -1
  42. package/dist/utils/formatStateForTransactionPay.mjs +12 -4
  43. package/dist/utils/formatStateForTransactionPay.mjs.map +1 -1
  44. package/package.json +1 -1
@@ -1 +1 @@
1
- {"version":3,"file":"AssetsController.cjs","sourceRoot":"","sources":["../src/AssetsController.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n AccountTreeControllerGetAccountsFromSelectedAccountGroupAction,\n AccountTreeControllerSelectedAccountGroupChangeEvent,\n} from '@metamask/account-tree-controller';\nimport type { GetTokenListState } from '@metamask/assets-controllers';\nimport { BaseController } from '@metamask/base-controller';\nimport type {\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n StateMetadata,\n} from '@metamask/base-controller';\nimport type { ClientControllerStateChangeEvent } from '@metamask/client-controller';\nimport { clientControllerSelectors } from '@metamask/client-controller';\nimport type {\n ApiPlatformClient,\n BackendWebSocketServiceActions,\n BackendWebSocketServiceEvents,\n SupportedCurrency,\n} from '@metamask/core-backend';\nimport type {\n KeyringControllerLockEvent,\n KeyringControllerUnlockEvent,\n} from '@metamask/keyring-controller';\nimport type { InternalAccount } from '@metamask/keyring-internal-api';\nimport type { Messenger } from '@metamask/messenger';\nimport type {\n NetworkControllerGetNetworkClientByIdAction,\n NetworkControllerGetStateAction,\n NetworkControllerStateChangeEvent,\n} from '@metamask/network-controller';\nimport type {\n NetworkEnablementControllerGetStateAction,\n NetworkEnablementControllerEvents,\n NetworkEnablementControllerState,\n} from '@metamask/network-enablement-controller';\nimport type {\n GetPermissions,\n PermissionControllerStateChange,\n} from '@metamask/permission-controller';\nimport type { PreferencesControllerStateChangeEvent } from '@metamask/preferences-controller';\nimport type {\n GetRunnableSnaps,\n HandleSnapRequest,\n} from '@metamask/snaps-controllers';\nimport type {\n TransactionControllerIncomingTransactionsReceivedEvent,\n TransactionControllerTransactionConfirmedEvent,\n} from '@metamask/transaction-controller';\nimport {\n isCaipChainId,\n isStrictHexString,\n parseCaipAssetType,\n parseCaipChainId,\n} from '@metamask/utils';\nimport { Mutex } from 'async-mutex';\nimport BigNumberJS from 'bignumber.js';\nimport { isEqual } from 'lodash';\n\nimport type { AssetsControllerMethodActions } from './AssetsController-method-action-types';\nimport type {\n AbstractDataSource,\n DataSourceState,\n SubscriptionRequest,\n} from './data-sources/AbstractDataSource';\nimport type { AccountsApiDataSourceConfig } from './data-sources/AccountsApiDataSource';\nimport { AccountsApiDataSource } from './data-sources/AccountsApiDataSource';\nimport { BackendWebsocketDataSource } from './data-sources/BackendWebsocketDataSource';\nimport type { PriceDataSourceConfig } from './data-sources/PriceDataSource';\nimport { PriceDataSource } from './data-sources/PriceDataSource';\nimport type { RpcDataSourceConfig } from './data-sources/RpcDataSource';\nimport { RpcDataSource } from './data-sources/RpcDataSource';\nimport type { AccountsControllerAccountBalancesUpdatedEvent } from './data-sources/SnapDataSource';\nimport { SnapDataSource } from './data-sources/SnapDataSource';\nimport type { StakedBalanceDataSourceConfig } from './data-sources/StakedBalanceDataSource';\nimport { StakedBalanceDataSource } from './data-sources/StakedBalanceDataSource';\nimport { TokenDataSource } from './data-sources/TokenDataSource';\nimport { projectLogger, createModuleLogger } from './logger';\nimport { DetectionMiddleware } from './middlewares/DetectionMiddleware';\nimport {\n createParallelBalanceMiddleware,\n createParallelMiddleware,\n} from './middlewares/ParallelMiddleware';\nimport type {\n AccountId,\n AssetPreferences,\n AssetsUpdateMode,\n ChainId,\n Caip19AssetId,\n AssetMetadata,\n FungibleAssetMetadata,\n AssetPrice,\n AssetBalance,\n AccountWithSupportedChains,\n AssetType,\n DataType,\n DataRequest,\n DataResponse,\n FetchContext,\n FetchNextFunction,\n NextFunction,\n Middleware,\n SubscriptionResponse,\n Asset,\n AssetsControllerStateInternal,\n} from './types';\nimport {\n normalizeAssetId,\n formatExchangeRatesForBridge,\n formatStateForTransactionPay,\n} from './utils';\nimport type {\n BridgeExchangeRatesFormat,\n TransactionPayLegacyFormat,\n} from './utils';\n\n// ============================================================================\n// PENDING TOKEN METADATA (UI input format for addCustomAsset)\n// ============================================================================\n\n/**\n * Metadata format passed from the UI when adding a custom token.\n * Mirrors the \"pendingTokens\" shape used by the extension.\n */\nexport type PendingTokenMetadata = {\n address: string;\n symbol: string;\n name: string;\n decimals: number;\n iconUrl?: string;\n aggregators?: string[];\n occurrences?: number;\n chainId: string;\n unlisted?: boolean;\n};\n\n// ============================================================================\n// CONTROLLER CONSTANTS\n// ============================================================================\n\nconst CONTROLLER_NAME = 'AssetsController' as const;\n\n/** Method names exposed as messenger actions (AssetsController:getAssets, etc.) */\nconst MESSENGER_EXPOSED_METHODS = [\n 'getAssets',\n 'getAssetsBalance',\n 'getAssetMetadata',\n 'getAssetsPrice',\n 'getExchangeRatesForBridge',\n 'getStateForTransactionPay',\n 'addCustomAsset',\n 'removeCustomAsset',\n 'getCustomAssets',\n 'hideAsset',\n 'unhideAsset',\n] as const;\n\n/** Default polling interval hint for data sources (30 seconds) */\nconst DEFAULT_POLLING_INTERVAL_MS = 30_000;\n\nconst log = createModuleLogger(projectLogger, CONTROLLER_NAME);\n\n// ============================================================================\n// STATE TYPES\n// ============================================================================\n\n/**\n * State structure for AssetsController.\n *\n * All values are JSON-serializable. UI preferences (e.g. hidden) are in\n * assetPreferences, not in metadata.\n *\n * @see AssetsControllerStateInternal for the semantic type structure\n */\nexport type AssetsControllerState = {\n /** Shared metadata for all assets (stored once per asset) */\n assetsInfo: { [assetId: string]: AssetMetadata };\n /** Per-account balance data */\n assetsBalance: { [accountId: string]: { [assetId: string]: AssetBalance } };\n /** Price data for assets */\n assetsPrice: { [assetId: string]: AssetPrice };\n /** Custom assets added by users per account (CAIP-19 asset IDs) */\n customAssets: { [accountId: string]: Caip19AssetId[] };\n /** UI preferences per asset (e.g. hidden) */\n assetPreferences: { [assetId: string]: AssetPreferences };\n /** Currently-active ISO 4217 currency code */\n selectedCurrency: SupportedCurrency;\n};\n\n/**\n * Returns the default state for AssetsController.\n *\n * @returns The default AssetsController state with empty maps.\n */\nexport function getDefaultAssetsControllerState(): AssetsControllerState {\n return {\n assetsInfo: {},\n assetsBalance: {},\n assetsPrice: {},\n customAssets: {},\n assetPreferences: {},\n selectedCurrency: 'usd',\n };\n}\n\n// ============================================================================\n// MESSENGER TYPES\n// ============================================================================\n\nexport type AssetsControllerGetStateAction = ControllerGetStateAction<\n typeof CONTROLLER_NAME,\n AssetsControllerState\n>;\n\nexport type AssetsControllerActions =\n | AssetsControllerGetStateAction\n | AssetsControllerMethodActions;\n\nexport type AssetsControllerStateChangeEvent = ControllerStateChangeEvent<\n typeof CONTROLLER_NAME,\n AssetsControllerState\n>;\n\nexport type AssetsControllerBalanceChangedEvent = {\n type: `${typeof CONTROLLER_NAME}:balanceChanged`;\n payload: [\n {\n accountId: AccountId;\n assetId: Caip19AssetId;\n previousAmount: string;\n newAmount: string;\n },\n ];\n};\n\nexport type AssetsControllerPriceChangedEvent = {\n type: `${typeof CONTROLLER_NAME}:priceChanged`;\n payload: [{ prices: Record<Caip19AssetId, AssetPrice> }];\n};\n\nexport type AssetsControllerAssetsDetectedEvent = {\n type: `${typeof CONTROLLER_NAME}:assetsDetected`;\n payload: [{ accountId: AccountId; assetIds: Caip19AssetId[] }];\n};\n\nexport type AssetsControllerEvents =\n | AssetsControllerStateChangeEvent\n | AssetsControllerBalanceChangedEvent\n | AssetsControllerPriceChangedEvent\n | AssetsControllerAssetsDetectedEvent;\n\ntype AllowedActions =\n // AssetsController\n | AccountTreeControllerGetAccountsFromSelectedAccountGroupAction\n // RpcDataSource\n | GetTokenListState\n | NetworkControllerGetStateAction\n | NetworkControllerGetNetworkClientByIdAction\n // RpcDataSource, StakedBalanceDataSource\n | NetworkEnablementControllerGetStateAction\n // SnapDataSource\n | GetRunnableSnaps\n | HandleSnapRequest\n | GetPermissions\n // BackendWebsocketDataSource\n | BackendWebSocketServiceActions;\n\ntype AllowedEvents =\n // AssetsController\n | AccountTreeControllerSelectedAccountGroupChangeEvent\n | ClientControllerStateChangeEvent\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent\n | PreferencesControllerStateChangeEvent\n // RpcDataSource, StakedBalanceDataSource\n | NetworkControllerStateChangeEvent\n | TransactionControllerTransactionConfirmedEvent\n | TransactionControllerIncomingTransactionsReceivedEvent\n // StakedBalanceDataSource\n | NetworkEnablementControllerEvents\n // SnapDataSource\n | AccountsControllerAccountBalancesUpdatedEvent\n | PermissionControllerStateChange\n // BackendWebsocketDataSource\n | BackendWebSocketServiceEvents;\n\nexport type AssetsControllerMessenger = Messenger<\n typeof CONTROLLER_NAME,\n AssetsControllerActions | AllowedActions,\n AssetsControllerEvents | AllowedEvents\n>;\n\n// ============================================================================\n// CONTROLLER OPTIONS\n// ============================================================================\n\n/**\n * Payload for the first init/fetch MetaMetrics event.\n * Passed to the optional trackMetaMetricsEvent callback when the initial\n * asset fetch completes after unlock or app open.\n */\nexport type AssetsControllerFirstInitFetchMetaMetricsPayload = {\n /** Duration of the first init fetch in milliseconds (wall-clock). */\n durationMs: number;\n /** Chain IDs requested in the fetch (e.g. ['eip155:1', 'eip155:137']). */\n chainIds: string[];\n /**\n * Exclusive latency in ms per data source (time spent in that source only).\n * Sum of values approximates durationMs. Order: same as middleware chain.\n */\n durationByDataSource: Record<string, number>;\n};\n\nexport type AssetsControllerOptions = {\n messenger: AssetsControllerMessenger;\n state?: Partial<AssetsControllerState>;\n /** Default polling interval hint passed to data sources (ms) */\n defaultUpdateInterval?: number;\n /** Function to determine if the controller is enabled. Defaults to true. */\n isEnabled?: () => boolean;\n /**\n * Getter for basic functionality (matches the \"Basic functionality\" setting in the UI).\n * When it returns true, internet services are on: token/price APIs are used for metadata, price,\n * and price subscription. When false, only RPC is used (no token/price APIs).\n * No value is stored; the getter is invoked when needed.\n * Defaults to () => true when not provided (APIs enabled).\n */\n isBasicFunctionality?: () => boolean;\n /**\n * Called by the controller with an onChange callback. The consumer subscribes to its own\n * basic-functionality source (e.g. PreferencesController:stateChange in extension, or a\n * different mechanism in mobile) and invokes onChange(isBasic) when the value changes.\n * The controller will then refresh its subscriptions. May return an unsubscribe function\n * called on controller destroy. Optional; when omitted, basic-functionality changes are not\n * subscribed to (e.g. host can notify via root messenger or another path).\n */\n subscribeToBasicFunctionalityChange?: (\n onChange: (isBasic: boolean) => void,\n ) => void | (() => void);\n /**\n * API client for balance/price/metadata. The controller instantiates data sources\n * and uses them directly when this is provided.\n */\n queryApiClient: ApiPlatformClient;\n /** Optional configuration for RpcDataSource. */\n rpcDataSourceConfig?: RpcDataSourceConfig;\n /**\n * Optional callback invoked when the first init/fetch completes (e.g. after unlock).\n * Use this to track first init fetch duration in MetaMetrics.\n */\n trackMetaMetricsEvent?: (\n payload: AssetsControllerFirstInitFetchMetaMetricsPayload,\n ) => void;\n /** Optional configuration for AccountsApiDataSource. */\n accountsApiDataSourceConfig?: AccountsApiDataSourceConfig;\n /** Optional configuration for PriceDataSource. */\n priceDataSourceConfig?: PriceDataSourceConfig;\n /** Optional configuration for StakedBalanceDataSource. */\n stakedBalanceDataSourceConfig?: StakedBalanceDataSourceConfig;\n};\n\n// ============================================================================\n// STATE METADATA\n// ============================================================================\n\nconst stateMetadata: StateMetadata<AssetsControllerState> = {\n assetsInfo: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n assetsBalance: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n assetsPrice: {\n persist: false,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n customAssets: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n assetPreferences: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n selectedCurrency: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n};\n\n// ============================================================================\n// HELPER FUNCTIONS\n// ============================================================================\n\nfunction extractChainId(assetId: Caip19AssetId): ChainId {\n const parsed = parseCaipAssetType(assetId);\n return parsed.chainId;\n}\n\n/**\n * Normalizes all asset IDs in a DataResponse.\n * This is applied at the controller level to ensure consistent state\n * regardless of how data sources format their asset IDs.\n *\n * @param response - The DataResponse to normalize.\n * @returns The normalized DataResponse with checksummed EVM addresses.\n */\nfunction normalizeResponse(response: DataResponse): DataResponse {\n const normalized: DataResponse = {};\n\n if (response.assetsInfo) {\n normalized.assetsInfo = {};\n for (const [assetId, metadata] of Object.entries(response.assetsInfo)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsInfo[normalizedId] = metadata;\n }\n }\n\n if (response.assetsPrice) {\n normalized.assetsPrice = {};\n for (const [assetId, price] of Object.entries(response.assetsPrice)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsPrice[normalizedId] = price;\n }\n }\n\n if (response.assetsBalance) {\n normalized.assetsBalance = {};\n for (const [accountId, balances] of Object.entries(\n response.assetsBalance,\n )) {\n normalized.assetsBalance[accountId] = {};\n for (const [assetId, balance] of Object.entries(balances)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsBalance[accountId][normalizedId] = balance;\n }\n }\n }\n\n // Preserve detectedAssets with normalized asset IDs\n if (response.detectedAssets) {\n normalized.detectedAssets = {};\n for (const [accountId, assetIds] of Object.entries(\n response.detectedAssets,\n )) {\n normalized.detectedAssets[accountId] = assetIds.map((assetId) =>\n normalizeAssetId(assetId),\n );\n }\n }\n\n // Preserve errors (chain IDs don't need normalization)\n if (response.errors) {\n normalized.errors = { ...response.errors };\n }\n\n if (response.updateMode) {\n normalized.updateMode = response.updateMode;\n }\n\n return normalized;\n}\n\n// ============================================================================\n// CONTROLLER IMPLEMENTATION\n// ============================================================================\n\n/**\n * AssetsController provides a unified interface for managing asset balances\n * across all blockchain networks (EVM and non-EVM) and all asset types.\n *\n * ## Core Responsibilities\n *\n * 1. **One-Time Fetch (Sync)**: For initial load, force refresh, or on-demand queries.\n * Uses `getAssets()`, `getAssetsBalance()`, etc. with `forceUpdate: true`.\n *\n * 2. **Async Subscriptions**: Subscribes to data sources for ongoing updates.\n * Data sources push updates via callbacks; the controller updates state.\n *\n * 3. **Dynamic Source Selection**: Routes requests to appropriate data sources\n * based on which chains they support. When active chains change, the controller\n * dynamically adjusts subscriptions.\n *\n * 4. **Client + Keyring Lifecycle**: Starts subscriptions only when both the UI is\n * open (ClientController) and the wallet is unlocked (KeyringController).\n * Stops when either the UI closes or the keyring locks. See client-controller\n * README for the combined pattern.\n *\n * ## Architecture\n *\n * - Data sources declare their supported chains (async, can change over time)\n * - Data sources are responsible for their own update mechanisms (WebSocket, polling, events)\n * - The controller does NOT manage polling - it simply receives pushed updates\n */\nexport class AssetsController extends BaseController<\n typeof CONTROLLER_NAME,\n AssetsControllerState,\n AssetsControllerMessenger\n> {\n /** Whether the controller is enabled */\n readonly #isEnabled: boolean;\n\n /** Getter for basic functionality (only balance fetch/subscribe use RPC; token/price API not used). No attribute stored. */\n readonly #isBasicFunctionality: () => boolean;\n\n /** Default update interval hint passed to data sources */\n readonly #defaultUpdateInterval: number;\n\n /** Optional callback for first init/fetch MetaMetrics (duration). */\n readonly #trackMetaMetricsEvent?: (\n payload: AssetsControllerFirstInitFetchMetaMetricsPayload,\n ) => void;\n\n /** Whether we have already reported first init fetch for this session (reset on #stop). */\n #firstInitFetchReported = false;\n\n /** Whether the client (UI) is open. Combined with #keyringUnlocked for #updateActive. */\n #uiOpen = false;\n\n /** Whether the keyring is unlocked. Combined with #uiOpen for #updateActive. */\n #keyringUnlocked = false;\n\n readonly #controllerMutex = new Mutex();\n\n /**\n * Active balance subscriptions keyed by account ID.\n * Each account has one logical subscription that may span multiple data sources.\n * For example, if WebSocket covers chains A,B and RPC covers chain C,\n * the account subscribes to both data sources for its chains.\n */\n readonly #activeSubscriptions: Map<string, SubscriptionResponse> = new Map();\n\n /** Currently enabled chains from NetworkEnablementController */\n #enabledChains: Set<ChainId> = new Set();\n\n /**\n * Get the currently selected accounts from AccountTreeController.\n * This includes all accounts in the same group as the selected account\n * (EVM, Bitcoin, Solana, Tron, etc. that belong to the same logical account group).\n *\n * @returns Array of InternalAccount objects from the selected account group.\n */\n get #selectedAccounts(): InternalAccount[] {\n return this.messenger.call(\n 'AccountTreeController:getAccountsFromSelectedAccountGroup',\n );\n }\n\n readonly #backendWebsocketDataSource: BackendWebsocketDataSource;\n\n readonly #accountsApiDataSource: AccountsApiDataSource;\n\n readonly #snapDataSource: SnapDataSource;\n\n readonly #rpcDataSource: RpcDataSource;\n\n readonly #stakedBalanceDataSource: StakedBalanceDataSource;\n\n /**\n * All balance data sources (used for unsubscription in #stop so we can clean up\n * regardless of current isBasicFunctionality mode).\n * Note: StakedBalanceDataSource is excluded because it provides supplementary\n * data and should not participate in chain-claiming.\n *\n * @returns The four balance data source instances in priority order.\n */\n get #allBalanceDataSources(): [\n BackendWebsocketDataSource,\n AccountsApiDataSource,\n SnapDataSource,\n RpcDataSource,\n ] {\n return [\n this.#backendWebsocketDataSource,\n this.#accountsApiDataSource,\n this.#snapDataSource,\n this.#rpcDataSource,\n ];\n }\n\n readonly #priceDataSource: PriceDataSource;\n\n readonly #detectionMiddleware: DetectionMiddleware;\n\n readonly #tokenDataSource: TokenDataSource;\n\n #unsubscribeBasicFunctionality: (() => void) | null = null;\n\n readonly #onActiveChainsUpdated: (\n dataSourceId: string,\n activeChains: ChainId[],\n previousChains: ChainId[],\n ) => void;\n\n constructor({\n messenger,\n state = {},\n defaultUpdateInterval = DEFAULT_POLLING_INTERVAL_MS,\n isEnabled = (): boolean => true,\n isBasicFunctionality,\n subscribeToBasicFunctionalityChange,\n queryApiClient,\n rpcDataSourceConfig,\n trackMetaMetricsEvent,\n accountsApiDataSourceConfig,\n priceDataSourceConfig,\n stakedBalanceDataSourceConfig,\n }: AssetsControllerOptions) {\n super({\n name: CONTROLLER_NAME,\n messenger,\n metadata: stateMetadata,\n state: {\n ...getDefaultAssetsControllerState(),\n ...state,\n },\n });\n\n this.#isEnabled = isEnabled();\n this.#isBasicFunctionality = isBasicFunctionality ?? ((): boolean => true);\n this.#defaultUpdateInterval = defaultUpdateInterval;\n this.#trackMetaMetricsEvent = trackMetaMetricsEvent;\n const rpcConfig = rpcDataSourceConfig ?? {};\n\n this.#onActiveChainsUpdated = (\n dataSourceName: string,\n chains: ChainId[],\n previousChains: ChainId[],\n ): void => {\n this.#handleActiveChainsUpdate(dataSourceName, chains, previousChains);\n };\n\n this.#backendWebsocketDataSource = new BackendWebsocketDataSource({\n messenger: this.messenger,\n queryApiClient,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n });\n this.#accountsApiDataSource = new AccountsApiDataSource({\n queryApiClient,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...accountsApiDataSourceConfig,\n });\n this.#snapDataSource = new SnapDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n });\n this.#rpcDataSource = new RpcDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...rpcConfig,\n });\n this.#stakedBalanceDataSource = new StakedBalanceDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...stakedBalanceDataSourceConfig,\n });\n this.#tokenDataSource = new TokenDataSource({\n queryApiClient,\n });\n this.#priceDataSource = new PriceDataSource({\n queryApiClient,\n getSelectedCurrency: (): SupportedCurrency => this.state.selectedCurrency,\n ...priceDataSourceConfig,\n });\n this.#detectionMiddleware = new DetectionMiddleware();\n\n if (!this.#isEnabled) {\n log('AssetsController is disabled, skipping initialization');\n return;\n }\n\n log('Initializing AssetsController', {\n defaultUpdateInterval: this.#defaultUpdateInterval,\n });\n\n this.#initializeState();\n this.#subscribeToEvents();\n this.#registerActionHandlers();\n // Subscriptions start only when both UI is open and keyring unlocked -> #updateActive().\n\n // Subscribe to basic-functionality changes after construction so a synchronous\n // onChange during subscribe cannot run before data sources are initialized.\n if (subscribeToBasicFunctionalityChange) {\n const unsubscribe = subscribeToBasicFunctionalityChange((isBasic) =>\n this.handleBasicFunctionalityChange(isBasic),\n );\n if (typeof unsubscribe === 'function') {\n this.#unsubscribeBasicFunctionality = unsubscribe;\n }\n }\n }\n\n // ============================================================================\n // INITIALIZATION\n // ============================================================================\n\n #initializeState(): void {\n const { enabledNetworkMap } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n this.#enabledChains = this.#extractEnabledChains(enabledNetworkMap);\n\n log('Initialized state', {\n enabledNetworkMap,\n enabledChains: this.#enabledChains,\n });\n }\n\n /**\n * Extract enabled chains from enabledNetworkMap.\n * Returns CAIP-2 chain IDs for all enabled networks across all namespaces.\n *\n * Note: For EIP155 (EVM) chains, the reference is normalized to decimal format\n * to ensure consistency with CAIP-2 standard and API responses.\n *\n * @param enabledNetworkMap - The enabled network map from NetworkEnablementController.\n * @returns Set of CAIP-2 chain IDs for all enabled networks.\n */\n #extractEnabledChains(\n enabledNetworkMap: NetworkEnablementControllerState['enabledNetworkMap'],\n ): Set<ChainId> {\n const chains = new Set<ChainId>();\n\n for (const [namespace, networks] of Object.entries(enabledNetworkMap)) {\n for (const [reference, isEnabled] of Object.entries(networks)) {\n if (isEnabled) {\n // Check if reference is already a full CAIP-2 chain ID (contains colon)\n if (reference.includes(':')) {\n // Already a full chain ID, use as-is\n chains.add(reference as ChainId);\n } else {\n // Normalize EIP155 chain references from hex to decimal (CAIP-2 standard)\n const normalizedReference = this.#normalizeChainReference(\n namespace,\n reference,\n );\n chains.add(`${namespace}:${normalizedReference}`);\n }\n }\n }\n }\n return chains;\n }\n\n /**\n * Normalize chain reference to CAIP-2 standard format.\n * For EIP155, converts hex chain IDs to decimal.\n *\n * @param namespace - The chain namespace (e.g., \"eip155\").\n * @param reference - The chain reference (e.g., \"0x1\" or \"1\").\n * @returns The normalized chain reference in decimal format.\n */\n #normalizeChainReference(namespace: string, reference: string): string {\n if (namespace === 'eip155' && isStrictHexString(reference)) {\n // Convert hex to decimal for EIP155 chains\n return parseInt(reference, 16).toString();\n }\n return reference;\n }\n\n #subscribeToEvents(): void {\n // Subscribe to account group changes (when user switches between account groups like Account 1 -> Account 2)\n this.messenger.subscribe(\n 'AccountTreeController:selectedAccountGroupChange',\n () => {\n this.#handleAccountGroupChanged().catch(console.error);\n },\n );\n\n // Subscribe to network enablement changes (only enabledNetworkMap)\n this.messenger.subscribe(\n 'NetworkEnablementController:stateChange',\n ({ enabledNetworkMap }) => {\n this.#handleEnabledNetworksChanged(enabledNetworkMap).catch(\n console.error,\n );\n },\n );\n\n // Client + Keyring lifecycle: only run when UI is open AND keyring is unlocked\n this.messenger.subscribe(\n 'ClientController:stateChange',\n (isUiOpen: boolean) => {\n this.#uiOpen = isUiOpen;\n this.#updateActive();\n },\n clientControllerSelectors.selectIsUiOpen,\n );\n this.messenger.subscribe('KeyringController:unlock', () => {\n this.#keyringUnlocked = true;\n this.#updateActive();\n });\n this.messenger.subscribe('KeyringController:lock', () => {\n this.#keyringUnlocked = false;\n this.#updateActive();\n });\n }\n\n /**\n * Start or stop asset tracking based on client (UI) open state and keyring\n * unlock state. Only runs when both UI is open and keyring is unlocked.\n */\n #updateActive(): void {\n const shouldRun = this.#uiOpen && this.#keyringUnlocked;\n if (shouldRun) {\n this.#start();\n } else {\n this.#stop();\n }\n }\n\n #registerActionHandlers(): void {\n this.messenger.registerMethodActionHandlers(\n this,\n MESSENGER_EXPOSED_METHODS,\n );\n }\n\n // ============================================================================\n // DATA SOURCE CHAIN MANAGEMENT\n // ============================================================================\n\n /**\n * Handle when a data source's supported chains change.\n * Used to refresh balance subscriptions and run a one-time fetch when a new chain is supported.\n *\n * - On any add/remove: re-subscribes to data sources so chain assignment stays correct.\n * - When chains are added: fetches balances for the new chains (for selected accounts on enabled networks).\n *\n * Controller does not store chains; sources report via this callback. previousChains is required for diff.\n *\n * @param dataSourceId - The identifier of the data source reporting the change.\n * @param activeChains - Currently active (supported and available) chain IDs for this source.\n * @param previousChains - Previous chains; used to compute added/removed.\n */\n #handleActiveChainsUpdate(\n dataSourceId: string,\n activeChains: ChainId[],\n previousChains: ChainId[],\n ): void {\n if (!this.#isEnabled) {\n return;\n }\n log('Data source active chains changed', {\n dataSourceId,\n chainCount: activeChains.length,\n chains: activeChains,\n });\n\n const previous: ChainId[] = previousChains;\n\n const previousSet = new Set(previous);\n const addedChains = activeChains.filter((ch) => !previousSet.has(ch));\n const removedChains = previous.filter((ch) => !activeChains.includes(ch));\n\n if (addedChains.length > 0 || removedChains.length > 0) {\n // Refresh subscriptions to use updated data source availability\n this.#subscribeAssets();\n }\n\n // If chains were added and we have selected accounts, do one-time fetch\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n const addedEnabledChains = addedChains.filter((chain) =>\n this.#enabledChains.has(chain),\n );\n if (addedEnabledChains.length > 0) {\n log('Fetching balances for newly added chains', { addedEnabledChains });\n this.getAssets(this.#selectedAccounts, {\n chainIds: addedEnabledChains,\n forceUpdate: true,\n updateMode: 'merge',\n }).catch((error) => {\n log('Failed to fetch balance for added chains', { error });\n });\n }\n }\n }\n\n /**\n * Returns the callback passed to data sources for reporting active chain updates.\n * Used by tests to simulate a data source reporting chain changes.\n *\n * @returns The onActiveChainsUpdated callback.\n */\n getOnActiveChainsUpdated(): (\n dataSourceId: string,\n activeChains: ChainId[],\n previousChains: ChainId[],\n ) => void {\n return this.#onActiveChainsUpdated;\n }\n\n // ============================================================================\n // MIDDLEWARE EXECUTION\n // ============================================================================\n\n /**\n * Execute middlewares with request/response context.\n * Returns response and exclusive duration per source (sum ≈ wall time).\n *\n * @param sources - Data sources or middlewares with getName() and assetsMiddleware (executed in order).\n * @param request - The data request.\n * @param initialResponse - Optional initial response (for enriching existing data).\n * @returns Response and durationByDataSource (exclusive ms per source name).\n */\n async #executeMiddlewares(\n sources: { getName(): string; assetsMiddleware: Middleware }[],\n request: DataRequest,\n initialResponse: DataResponse = {},\n ): Promise<{\n response: DataResponse;\n durationByDataSource: Record<string, number>;\n }> {\n const names = sources.map((source) => source.getName());\n const middlewares = sources.map((source) => source.assetsMiddleware);\n const inclusive: number[] = [];\n const wrapped = middlewares.map(\n (middleware, i) =>\n (async (\n ctx: FetchContext,\n next: FetchNextFunction,\n ): Promise<{\n request: DataRequest;\n response: DataResponse;\n getAssetsState: () => AssetsControllerStateInternal;\n }> => {\n const start = Date.now();\n try {\n return await middleware(ctx, next);\n } finally {\n inclusive[i] = Date.now() - start;\n }\n }) as Middleware,\n );\n\n const chain = wrapped.reduceRight<NextFunction>(\n (next, middleware) =>\n async (\n ctx,\n ): Promise<{\n request: DataRequest;\n response: DataResponse;\n getAssetsState: () => AssetsControllerStateInternal;\n }> => {\n try {\n return await middleware(ctx, next);\n } catch (error) {\n console.error('[AssetsController] Middleware failed:', error);\n return next(ctx);\n }\n },\n async (ctx) => ctx,\n );\n\n const result = await chain({\n request,\n response: initialResponse,\n getAssetsState: () => this.state as AssetsControllerStateInternal,\n });\n\n const durationByDataSource: Record<string, number> = {};\n for (let i = 0; i < inclusive.length; i++) {\n const nextInc = i + 1 < inclusive.length ? (inclusive[i + 1] ?? 0) : 0;\n const exclusive = Math.max(0, (inclusive[i] ?? 0) - nextInc);\n const name = names[i];\n if (name !== undefined) {\n durationByDataSource[name] = exclusive;\n }\n }\n return { response: result.response, durationByDataSource };\n }\n\n // ============================================================================\n // PUBLIC API: QUERY METHODS\n // ============================================================================\n\n async getAssets(\n accounts: InternalAccount[],\n options?: {\n chainIds?: ChainId[];\n assetTypes?: AssetType[];\n forceUpdate?: boolean;\n dataTypes?: DataType[];\n assetsForPriceUpdate?: Caip19AssetId[];\n /** When set to 'merge', fetch result is merged with existing state instead of replacing. Use for partial fetches (e.g. newly added chains). */\n updateMode?: AssetsUpdateMode;\n },\n ): Promise<Record<AccountId, Record<Caip19AssetId, Asset>>> {\n const chainIds = options?.chainIds ?? [...this.#enabledChains];\n const assetTypes = options?.assetTypes ?? ['fungible'];\n const dataTypes = options?.dataTypes ?? ['balance', 'metadata', 'price'];\n\n if (accounts.length === 0 || chainIds.length === 0) {\n return this.#getAssetsFromState(accounts, chainIds, assetTypes);\n }\n\n // Collect custom assets for all requested accounts\n const customAssets: Caip19AssetId[] = [];\n for (const account of accounts) {\n const accountCustomAssets = this.getCustomAssets(account.id);\n customAssets.push(...accountCustomAssets);\n }\n\n if (options?.forceUpdate) {\n const startTime = Date.now();\n const request = this.#buildDataRequest(accounts, chainIds, {\n assetTypes,\n dataTypes,\n customAssets: customAssets.length > 0 ? customAssets : undefined,\n forceUpdate: true,\n assetsForPriceUpdate: options?.assetsForPriceUpdate,\n });\n const sources = this.#isBasicFunctionality()\n ? [\n createParallelBalanceMiddleware([\n this.#accountsApiDataSource,\n this.#snapDataSource,\n this.#rpcDataSource,\n this.#stakedBalanceDataSource,\n ]),\n this.#detectionMiddleware,\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ]\n : [\n this.#rpcDataSource,\n this.#stakedBalanceDataSource,\n this.#detectionMiddleware,\n ];\n const { response, durationByDataSource } = await this.#executeMiddlewares(\n sources,\n request,\n );\n // Default to 'merge' when fetching a subset of chains so we don't wipe\n // balances from chains that weren't included in this fetch.\n const isPartialChainFetch =\n options?.chainIds !== undefined &&\n options.chainIds.length < this.#enabledChains.size;\n const updateMode =\n options?.updateMode ?? (isPartialChainFetch ? 'merge' : 'full');\n await this.#updateState({ ...response, updateMode });\n if (this.#trackMetaMetricsEvent && !this.#firstInitFetchReported) {\n this.#firstInitFetchReported = true;\n const durationMs = Date.now() - startTime;\n this.#trackMetaMetricsEvent({\n durationMs,\n chainIds,\n durationByDataSource,\n });\n }\n }\n\n const result = this.#getAssetsFromState(accounts, chainIds, assetTypes);\n return result;\n }\n\n async getAssetsBalance(\n accounts: InternalAccount[],\n options?: {\n chainIds?: ChainId[];\n assetTypes?: AssetType[];\n forceUpdate?: boolean;\n },\n ): Promise<Record<AccountId, Record<Caip19AssetId, AssetBalance>>> {\n // Reuse getAssets with dataTypes: ['balance'] only\n const assets = await this.getAssets(accounts, {\n chainIds: options?.chainIds,\n assetTypes: options?.assetTypes,\n forceUpdate: options?.forceUpdate,\n dataTypes: ['balance', 'metadata'],\n });\n\n // Extract just the balance from each asset\n const result: Record<AccountId, Record<Caip19AssetId, AssetBalance>> = {};\n for (const [accountId, accountAssets] of Object.entries(assets)) {\n result[accountId] = {};\n for (const [assetId, asset] of Object.entries(accountAssets)) {\n if (asset.balance) {\n result[accountId][assetId as Caip19AssetId] = asset.balance;\n }\n }\n }\n\n return result;\n }\n\n getAssetMetadata(assetId: Caip19AssetId): AssetMetadata | undefined {\n return this.state.assetsInfo[assetId] as AssetMetadata | undefined;\n }\n\n async getAssetsPrice(\n accounts: InternalAccount[],\n options?: {\n chainIds?: ChainId[];\n assetTypes?: AssetType[];\n forceUpdate?: boolean;\n },\n ): Promise<Record<Caip19AssetId, AssetPrice>> {\n const assets = await this.getAssets(accounts, {\n chainIds: options?.chainIds,\n assetTypes: options?.assetTypes,\n forceUpdate: options?.forceUpdate,\n dataTypes: ['price'],\n });\n\n // Extract just the price from each asset (flattened across accounts)\n const result: Record<Caip19AssetId, AssetPrice> = {};\n for (const accountAssets of Object.values(assets)) {\n for (const [assetId, asset] of Object.entries(accountAssets)) {\n if (asset.price) {\n result[assetId as Caip19AssetId] = asset.price;\n }\n }\n }\n\n return result;\n }\n\n /**\n * Returns exchange rates in the format expected by the bridge controller\n * (conversionRates, currencyRates, marketData, currentCurrency) so that\n * when useAssetsControllerForRates is true the bridge can use a single\n * action instead of MultichainAssetsRatesController, TokenRatesController,\n * and CurrencyRateController.\n *\n * @returns Bridge-compatible exchange rate state derived from assetsPrice and selectedCurrency.\n */\n getExchangeRatesForBridge(): BridgeExchangeRatesFormat {\n const { nativeAssetIdentifiers } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n return formatExchangeRatesForBridge({\n assetsPrice: this.state.assetsPrice,\n selectedCurrency: this.state.selectedCurrency,\n nativeAssetIdentifiers,\n networkConfigurationsByChainId,\n });\n }\n\n /**\n * Returns state in the legacy format expected by transaction-pay-controller\n * (TokenBalancesController, AccountTrackerController, TokensController,\n * TokenRatesController, CurrencyRateController shapes) so that when\n * useAssetsController is true the transaction-pay-controller can use a\n * single action instead of five separate getState calls.\n *\n * @returns Legacy-compatible state for transaction-pay-controller.\n */\n getStateForTransactionPay(): TransactionPayLegacyFormat {\n const accounts = this.#selectedAccounts;\n const { nativeAssetIdentifiers } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n return formatStateForTransactionPay({\n assetsBalance: this.state.assetsBalance,\n assetsInfo: this.state.assetsInfo,\n assetsPrice: this.state.assetsPrice,\n selectedCurrency: this.state.selectedCurrency,\n accounts: accounts.map((a) => ({ id: a.id, address: a.address })),\n nativeAssetIdentifiers,\n networkConfigurationsByChainId,\n });\n }\n\n // ============================================================================\n // CUSTOM ASSETS MANAGEMENT\n // ============================================================================\n\n /**\n * Add a custom asset for an account.\n * Custom assets are included in subscription and fetch operations.\n * Adding a custom asset also unhides it if it was previously hidden.\n *\n * When `pendingMetadata` is provided (e.g. from the extension's pending-tokens\n * flow), the token metadata is persisted immediately into `assetsInfo` so the\n * UI can render it without waiting for the next pipeline fetch.\n *\n * @param accountId - The account ID to add the custom asset for.\n * @param assetId - The CAIP-19 asset ID to add.\n * @param pendingMetadata - Optional token metadata from the UI (pendingTokens format).\n */\n async addCustomAsset(\n accountId: AccountId,\n assetId: Caip19AssetId,\n pendingMetadata?: PendingTokenMetadata,\n ): Promise<void> {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Adding custom asset', { accountId, assetId: normalizedAssetId });\n\n this.update((state) => {\n const customAssets = state.customAssets as Record<string, string[]>;\n if (!customAssets[accountId]) {\n customAssets[accountId] = [];\n }\n\n // Only add if not already present\n if (!customAssets[accountId].includes(normalizedAssetId)) {\n customAssets[accountId].push(normalizedAssetId);\n }\n\n // Unhide the asset if it was hidden (via assetPreferences)\n const prefs = state.assetPreferences[normalizedAssetId];\n if (prefs?.hidden) {\n delete prefs.hidden;\n if (Object.keys(prefs).length === 0) {\n delete state.assetPreferences[normalizedAssetId];\n }\n }\n\n // Persist metadata from the UI so the token is immediately renderable\n if (pendingMetadata) {\n const parsed = parseCaipAssetType(normalizedAssetId);\n let tokenType: FungibleAssetMetadata['type'] = 'erc20';\n if (parsed.assetNamespace === 'slip44') {\n tokenType = 'native';\n } else if (parsed.assetNamespace === 'spl') {\n tokenType = 'spl';\n }\n\n const assetMetadata: FungibleAssetMetadata = {\n type: tokenType,\n symbol: pendingMetadata.symbol,\n name: pendingMetadata.name,\n decimals: pendingMetadata.decimals,\n image: pendingMetadata.iconUrl,\n aggregators: pendingMetadata.aggregators,\n occurrences: pendingMetadata.occurrences,\n };\n\n (state.assetsInfo as Record<string, AssetMetadata>)[normalizedAssetId] =\n assetMetadata;\n }\n });\n\n // Fetch data for the newly added custom asset (merge to preserve other chains)\n const account = this.#selectedAccounts.find((a) => a.id === accountId);\n if (account) {\n const chainId = extractChainId(normalizedAssetId);\n await this.getAssets([account], {\n chainIds: [chainId],\n forceUpdate: true,\n updateMode: 'merge',\n });\n }\n }\n\n /**\n * Remove a custom asset from an account.\n *\n * @param accountId - The account ID to remove the custom asset from.\n * @param assetId - The CAIP-19 asset ID to remove.\n */\n removeCustomAsset(accountId: AccountId, assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Removing custom asset', { accountId, assetId: normalizedAssetId });\n\n this.update((state) => {\n if (state.customAssets[accountId]) {\n state.customAssets[accountId] = state.customAssets[accountId].filter(\n (id) => id !== normalizedAssetId,\n );\n\n // Clean up empty arrays\n if (state.customAssets[accountId].length === 0) {\n delete state.customAssets[accountId];\n }\n }\n });\n }\n\n /**\n * Get all custom assets for an account.\n *\n * @param accountId - The account ID to get custom assets for.\n * @returns Array of CAIP-19 asset IDs for the account's custom assets.\n */\n getCustomAssets(accountId: AccountId): Caip19AssetId[] {\n return this.state.customAssets[accountId] ?? [];\n }\n\n // ============================================================================\n // HIDDEN ASSETS MANAGEMENT\n // ============================================================================\n\n /**\n * Hide an asset globally.\n * Hidden assets are excluded from the asset list returned by getAssets.\n * The hidden state is stored in assetPreferences.\n *\n * @param assetId - The CAIP-19 asset ID to hide.\n */\n hideAsset(assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Hiding asset', { assetId: normalizedAssetId });\n\n this.update((state) => {\n if (!state.assetPreferences[normalizedAssetId]) {\n state.assetPreferences[normalizedAssetId] = {};\n }\n state.assetPreferences[normalizedAssetId].hidden = true;\n });\n }\n\n /**\n * Unhide an asset globally.\n *\n * @param assetId - The CAIP-19 asset ID to unhide.\n */\n unhideAsset(assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Unhiding asset', { assetId: normalizedAssetId });\n\n this.update((state) => {\n const prefs = state.assetPreferences[normalizedAssetId];\n if (prefs) {\n delete prefs.hidden;\n if (Object.keys(prefs).length === 0) {\n delete state.assetPreferences[normalizedAssetId];\n }\n }\n });\n }\n\n // ============================================================================\n // CURRENT CURRENCY MANAGEMENT\n // ============================================================================\n\n /**\n * Set the current currency.\n *\n * @param selectedCurrency - The ISO 4217 currency code to set.\n */\n setSelectedCurrency(selectedCurrency: SupportedCurrency): void {\n const previousCurrency = this.state.selectedCurrency;\n\n if (previousCurrency === selectedCurrency) {\n return;\n }\n\n this.update((state) => {\n state.selectedCurrency = selectedCurrency;\n });\n\n log('Current currency changed', {\n previousCurrency,\n selectedCurrency,\n });\n\n this.getAssets(this.#selectedAccounts, {\n forceUpdate: true,\n dataTypes: ['price'],\n assetsForPriceUpdate: Object.values(this.state.assetsBalance).flatMap(\n (balances) => Object.keys(balances) as Caip19AssetId[],\n ),\n }).catch((error) => {\n log('Failed to fetch asset prices after current currency change', error);\n });\n }\n\n // ============================================================================\n // SUBSCRIPTIONS\n // ============================================================================\n\n /**\n * Subscribe to price updates for all assets held by the given accounts.\n * Polls PriceDataSource which fetches prices from balance state.\n *\n * @param accounts - Accounts to subscribe price updates for.\n * @param chainIds - Chain IDs to filter prices for.\n */\n subscribeAssetsPrice(accounts: InternalAccount[], chainIds: ChainId[]): void {\n if (!this.#isBasicFunctionality()) {\n return;\n }\n const subscriptionKey = 'ds:PriceDataSource';\n\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n const subscribeReq: SubscriptionRequest = {\n request: this.#buildDataRequest(accounts, chainIds, {\n dataTypes: ['price'],\n }),\n subscriptionId: subscriptionKey,\n isUpdate,\n onAssetsUpdate: (response) =>\n this.handleAssetsUpdate(response, 'PriceDataSource'),\n getAssetsState: () => this.state,\n };\n\n this.#priceDataSource.subscribe(subscribeReq).catch(console.error);\n\n // Track subscription\n const subscription: SubscriptionResponse = {\n chains: chainIds,\n accountId: subscriptionKey,\n assetTypes: ['fungible'],\n dataTypes: ['price'],\n unsubscribe: () => {\n this.#activeSubscriptions.delete(subscriptionKey);\n },\n };\n\n this.#activeSubscriptions.set(subscriptionKey, subscription);\n }\n\n /**\n * Unsubscribe from price updates.\n */\n unsubscribeAssetsPrice(): void {\n const subscriptionKey = 'ds:PriceDataSource';\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n\n if (!existingSubscription) {\n return;\n }\n this.#priceDataSource.unsubscribe(subscriptionKey).catch(console.error);\n existingSubscription.unsubscribe();\n }\n\n // ============================================================================\n // STATE MANAGEMENT\n // ============================================================================\n\n /**\n * Resolves native asset IDs (CAIP-19) for the given chains by looking them up\n * in NetworkEnablementController.nativeAssetIdentifiers.\n * Chains without a registered native identifier are skipped.\n *\n * @param chains - The chain IDs to resolve native assets for.\n * @returns Array of native asset IDs for the chains that have a registered identifier.\n */\n #resolveNativeAssetIds(chains: Iterable<ChainId>): Caip19AssetId[] {\n const { nativeAssetIdentifiers } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n const ids: Caip19AssetId[] = [];\n for (const chainId of chains) {\n const nativeId = nativeAssetIdentifiers?.[chainId];\n if (nativeId) {\n ids.push(nativeId as Caip19AssetId);\n }\n }\n return ids;\n }\n\n /**\n * Returns native asset IDs for all enabled chains.\n *\n * @returns Array of native asset IDs, one per enabled chain that has a registered identifier.\n */\n #getNativeAssetIdsForEnabledChains(): Caip19AssetId[] {\n return this.#resolveNativeAssetIds(this.#enabledChains);\n }\n\n /**\n * Returns native asset IDs for the chains that this account supports\n * (account scopes ∩ enabled chains).\n *\n * @param account - The account (scopes determine which chains apply).\n * @returns Array of native asset IDs, one per supported chain that has a registered identifier.\n */\n #getNativeAssetIdsForAccount(account: InternalAccount): Caip19AssetId[] {\n return this.#resolveNativeAssetIds(\n this.#getEnabledChainsForAccount(account),\n );\n }\n\n /**\n * Ensures assetsBalance has a 0 balance for each native token (from\n * NetworkEnablementController.nativeAssetIdentifiers) for each selected account.\n * Only adds natives for chains that the account supports (correct accountId ↔ chain mapping).\n */\n #ensureNativeBalancesDefaultZero(): void {\n const accounts = this.#selectedAccounts;\n if (accounts.length === 0) {\n return;\n }\n this.update((state) => {\n const balances = state.assetsBalance as Record<\n string,\n Record<string, AssetBalance>\n >;\n for (const account of accounts) {\n const accountId = account.id;\n const nativeAssetIds = this.#getNativeAssetIdsForAccount(account);\n if (nativeAssetIds.length === 0) {\n continue;\n }\n if (!balances[accountId]) {\n balances[accountId] = {};\n }\n for (const nativeAssetId of nativeAssetIds) {\n if (!(nativeAssetId in balances[accountId])) {\n balances[accountId][nativeAssetId] = { amount: '0' };\n }\n }\n }\n });\n }\n\n async #updateState(response: DataResponse): Promise<void> {\n const normalizedResponse = normalizeResponse(response);\n const mode: AssetsUpdateMode = normalizedResponse.updateMode ?? 'merge';\n\n const releaseLock = await this.#controllerMutex.acquire();\n\n try {\n const previousState = this.state;\n const previousPrices = { ...this.state.assetsPrice };\n // Use detectedAssets from response (assets without metadata)\n const detectedAssets: Record<AccountId, Caip19AssetId[]> =\n normalizedResponse.detectedAssets ?? {};\n\n // Track actual changes for logging\n const changedBalances: {\n accountId: string;\n assetId: string;\n oldAmount: string | undefined;\n newAmount: string;\n }[] = [];\n const changedMetadata: string[] = [];\n\n this.update((state) => {\n // Use type assertions to avoid deep type instantiation issues with Immer Draft types\n const metadata = state.assetsInfo as Record<string, AssetMetadata>;\n const balances = state.assetsBalance as Record<\n string,\n Record<string, AssetBalance>\n >;\n const prices = state.assetsPrice as Record<string, AssetPrice>;\n\n if (normalizedResponse.assetsInfo) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsInfo,\n )) {\n if (\n !isEqual(previousState.assetsInfo[key as Caip19AssetId], value)\n ) {\n changedMetadata.push(key);\n }\n metadata[key] = value;\n }\n }\n\n if (normalizedResponse.assetsBalance) {\n for (const [accountId, accountBalances] of Object.entries(\n normalizedResponse.assetsBalance,\n )) {\n const previousBalances =\n previousState.assetsBalance[accountId] ?? {};\n const customAssetIds =\n (state.customAssets as Record<string, Caip19AssetId[]>)[\n accountId\n ] ?? [];\n\n // Full: response is authoritative; preserve custom assets not in response.\n // Merge: response overlays previous balances.\n // Callers that fetch partial data (e.g. newly added chains) must set updateMode: 'merge'.\n const effective: Record<string, AssetBalance> =\n mode === 'merge'\n ? { ...previousBalances, ...accountBalances }\n : ((): Record<string, AssetBalance> => {\n const next: Record<string, AssetBalance> = {\n ...accountBalances,\n };\n for (const customId of customAssetIds) {\n if (!(customId in next)) {\n const prev = previousBalances[customId];\n next[customId] =\n prev ?? ({ amount: '0' } as AssetBalance);\n }\n }\n return next;\n })();\n\n // Ensure native tokens have an entry (0 if missing) for chains this account supports\n const account = this.#selectedAccounts.find(\n (a) => a.id === accountId,\n );\n const nativeAssetIdsForAccount = account\n ? this.#getNativeAssetIdsForAccount(account)\n : this.#getNativeAssetIdsForEnabledChains();\n for (const nativeAssetId of nativeAssetIdsForAccount) {\n if (!(nativeAssetId in effective)) {\n effective[nativeAssetId] = { amount: '0' } as AssetBalance;\n }\n }\n\n for (const [assetId, balance] of Object.entries(effective)) {\n const previousBalance = previousBalances[\n assetId as Caip19AssetId\n ] as { amount: string } | undefined;\n const newAmount = (balance as { amount: string }).amount;\n const oldAmount = previousBalance?.amount;\n const isNewDefaultNativeZero =\n oldAmount === undefined &&\n newAmount === '0' &&\n nativeAssetIdsForAccount.includes(assetId as Caip19AssetId);\n if (oldAmount !== newAmount && !isNewDefaultNativeZero) {\n changedBalances.push({\n accountId,\n assetId,\n oldAmount,\n newAmount,\n });\n }\n }\n balances[accountId] = effective;\n }\n }\n\n // Update prices in state\n if (normalizedResponse.assetsPrice) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsPrice,\n )) {\n prices[key] = value;\n }\n }\n });\n\n // Calculate changed prices\n const changedPriceAssets: string[] = normalizedResponse.assetsPrice\n ? Object.keys(normalizedResponse.assetsPrice).filter(\n (assetId) =>\n !isEqual(\n previousPrices[assetId as Caip19AssetId],\n normalizedResponse.assetsPrice?.[assetId as Caip19AssetId],\n ),\n )\n : [];\n\n // Log only actual changes\n if (\n changedBalances.length > 0 ||\n changedMetadata.length > 0 ||\n changedPriceAssets.length > 0\n ) {\n log('State updated', {\n changedBalances:\n changedBalances.length > 0 ? changedBalances : undefined,\n changedMetadataCount:\n changedMetadata.length > 0 ? changedMetadata.length : undefined,\n changedPricesCount:\n changedPriceAssets.length > 0\n ? changedPriceAssets.length\n : undefined,\n newAssets:\n Object.keys(detectedAssets).length > 0\n ? Object.entries(detectedAssets).map(([accountId, assets]) => ({\n accountId,\n assets,\n }))\n : undefined,\n });\n }\n\n // Publish balance changed events\n for (const change of changedBalances) {\n this.messenger.publish('AssetsController:balanceChanged', {\n accountId: change.accountId,\n assetId: change.assetId as Caip19AssetId,\n previousAmount: change.oldAmount ?? '0',\n newAmount: change.newAmount,\n });\n }\n\n // Publish price changed event with full price data\n if (changedPriceAssets.length > 0 && normalizedResponse.assetsPrice) {\n // Build prices object with only changed prices\n const changedPrices: Record<Caip19AssetId, AssetPrice> = {};\n for (const assetId of changedPriceAssets) {\n const price =\n normalizedResponse.assetsPrice[assetId as Caip19AssetId];\n if (price) {\n changedPrices[assetId as Caip19AssetId] = price;\n }\n }\n this.messenger.publish('AssetsController:priceChanged', {\n prices: changedPrices,\n });\n }\n\n // Publish assets detected events\n for (const [accountId, assetIds] of Object.entries(detectedAssets)) {\n if (assetIds.length > 0) {\n this.messenger.publish('AssetsController:assetsDetected', {\n accountId,\n assetIds,\n });\n }\n }\n } finally {\n releaseLock();\n }\n }\n\n #getAssetsFromState(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n assetTypes: AssetType[],\n ): Record<AccountId, Record<Caip19AssetId, Asset>> {\n const result: Record<AccountId, Record<Caip19AssetId, Asset>> = {};\n // Convert to Sets for O(1) lookups\n const chainIdSet = new Set(chainIds);\n const assetTypeSet = new Set(assetTypes);\n\n for (const account of accounts) {\n result[account.id] = {};\n\n const accountBalances = this.state.assetsBalance[account.id] ?? {};\n\n for (const [assetId, balance] of Object.entries(accountBalances)) {\n const typedAssetId = assetId as Caip19AssetId;\n\n const metadataRaw = this.state.assetsInfo[typedAssetId];\n\n // Skip assets without metadata\n if (!metadataRaw) {\n continue;\n }\n\n const metadata = metadataRaw;\n\n // Skip hidden assets (assetPreferences)\n const prefs = this.state.assetPreferences[typedAssetId];\n if (prefs?.hidden) {\n continue;\n }\n\n const assetChainId = extractChainId(typedAssetId);\n\n if (!chainIdSet.has(assetChainId)) {\n continue;\n }\n\n // Filter by asset type\n const tokenAssetType = this.#tokenStandardToAssetType(metadata.type);\n if (!assetTypeSet.has(tokenAssetType)) {\n continue;\n }\n\n const typedBalance = balance;\n const priceRaw = this.state.assetsPrice[typedAssetId];\n const price: AssetPrice = priceRaw ?? {\n price: 0,\n lastUpdated: 0,\n };\n\n // Compute fiat value using BigNumber for precision\n // Note: typedBalance.amount is already in human-readable format (e.g., \"1\" for 1 ETH)\n // so we do NOT divide by 10^decimals here\n const balanceAmount = new BigNumberJS(typedBalance.amount || '0');\n const fiatValue = balanceAmount\n .multipliedBy(price.price || 0)\n .toNumber();\n\n const asset: Asset = {\n id: typedAssetId,\n chainId: assetChainId,\n balance: typedBalance,\n metadata,\n price,\n fiatValue,\n };\n\n result[account.id][typedAssetId] = asset;\n }\n }\n\n return result;\n }\n\n /**\n * Maps a token standard to its corresponding asset type.\n *\n * @param tokenStandard - The token standard from metadata.\n * @returns The corresponding asset type.\n */\n #tokenStandardToAssetType(tokenStandard: string): AssetType {\n switch (tokenStandard) {\n case 'native':\n case 'erc20':\n case 'spl':\n return 'fungible';\n case 'erc721':\n return 'nft';\n case 'erc1155':\n // ERC1155 can be either fungible or non-fungible, treat as collectible\n return 'collectible';\n default:\n // Unknown standards default to fungible\n return 'fungible';\n }\n }\n\n // ============================================================================\n // START / STOP\n // ============================================================================\n\n /**\n * Start asset tracking: subscribe to updates and fetch current balances.\n * Called when app opens, account changes, or keyring unlocks.\n */\n #start(): void {\n log('Starting asset tracking', {\n selectedAccountCount: this.#selectedAccounts.length,\n enabledChainCount: this.#enabledChains.size,\n });\n\n this.#subscribeAssets();\n this.#ensureNativeBalancesDefaultZero();\n this.getAssets(this.#selectedAccounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch assets', error);\n });\n }\n\n /**\n * Stop asset tracking: unsubscribe from all updates.\n * Called when app closes or keyring locks.\n */\n #stop(): void {\n log('Stopping asset tracking', {\n activeSubscriptionCount: this.#activeSubscriptions.size,\n hasPriceSubscription: this.#activeSubscriptions.has('ds:PriceDataSource'),\n });\n\n this.#firstInitFetchReported = false;\n\n // Stop price subscription first (uses direct messenger call)\n this.unsubscribeAssetsPrice();\n\n // Stop balance subscriptions by properly notifying data sources via messenger\n // This ensures data sources stop their polling timers.\n // Use #allBalanceDataSources + staked balance source so we unsubscribe from\n // every source that may have been subscribed.\n const allSources = [\n ...this.#allBalanceDataSources,\n this.#stakedBalanceDataSource,\n ];\n const subscriptionKeys = [...this.#activeSubscriptions.keys()];\n for (const subscriptionKey of subscriptionKeys) {\n if (subscriptionKey.startsWith('ds:')) {\n const sourceId = subscriptionKey.slice(3);\n const source = allSources.find((ds) => ds.getName() === sourceId);\n if (source) {\n this.#unsubscribeDataSource(source);\n }\n }\n }\n this.#activeSubscriptions.clear();\n }\n\n /**\n * Handle basic functionality toggle change. Call this from the consumer (extension or mobile)\n * when the user changes the \"Basic functionality\" setting. Refreshes subscriptions so the\n * current {@link AssetsControllerOptions.isBasicFunctionality} getter is used (true = APIs on,\n * false = RPC only).\n *\n * @param _isBasic - The new value (for call-site clarity; the getter is the source of truth).\n */\n handleBasicFunctionalityChange(_isBasic: boolean): void {\n this.#stop();\n this.#subscribeAssets();\n }\n\n /**\n * Subscribe to asset updates for all selected accounts.\n */\n #subscribeAssets(): void {\n if (this.#selectedAccounts.length === 0 || this.#enabledChains.size === 0) {\n return;\n }\n\n // Subscribe to balance updates (batched by data source)\n this.#subscribeAssetsBalance(this.#selectedAccounts, [\n ...this.#enabledChains,\n ]);\n\n // Subscribe to staked balance updates (separate from regular balance chain-claiming)\n this.#subscribeStakedBalance(this.#selectedAccounts, [\n ...this.#enabledChains,\n ]);\n\n // Subscribe to price updates for all assets held by selected accounts\n this.subscribeAssetsPrice(this.#selectedAccounts, [...this.#enabledChains]);\n }\n\n /**\n * Subscribe to balance updates for the given accounts and chains.\n *\n * Strategy to minimize data source calls:\n * 1. Collect all chains to subscribe based on enabled networks\n * 2. Map chains to accounts based on their scopes\n * 3. Split by data source (ordered by priority) - each data source gets ONE subscription\n *\n * This ensures we make minimal subscriptions to each data source while covering\n * all accounts and chains.\n *\n * @param accounts - Accounts to subscribe balance updates for.\n * @param chainIds - Chain IDs to subscribe for.\n */\n #subscribeAssetsBalance(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n ): void {\n const chainToAccounts = this.#buildChainToAccountsMap(\n accounts,\n new Set(chainIds),\n );\n const remainingChains = new Set(chainToAccounts.keys());\n\n // When basic functionality is on (getter true), use all balance data sources; when off (getter false), RPC only.\n const balanceDataSources = this.#isBasicFunctionality()\n ? this.#allBalanceDataSources\n : [this.#rpcDataSource];\n\n for (const source of balanceDataSources) {\n const availableChains = new Set(source.getActiveChainsSync());\n const assignedChains: ChainId[] = [];\n\n for (const chainId of remainingChains) {\n if (availableChains.has(chainId)) {\n assignedChains.push(chainId);\n remainingChains.delete(chainId);\n }\n }\n\n if (assignedChains.length === 0) {\n this.#unsubscribeDataSource(source);\n continue;\n }\n\n const seenIds = new Set<string>();\n const accountsForSource = assignedChains\n .flatMap((chainId) => chainToAccounts.get(chainId) ?? [])\n .filter(\n (account) =>\n !seenIds.has(account.id) && (seenIds.add(account.id), true),\n );\n if (accountsForSource.length > 0) {\n this.#subscribeDataSource(source, accountsForSource, assignedChains);\n }\n }\n }\n\n /**\n * Subscribe to staked balance updates.\n * Unlike regular balance data sources, the staked balance data source provides\n * supplementary data and does not participate in chain-claiming.\n *\n * @param accounts - Accounts to subscribe staked balance updates for.\n * @param chainIds - Chain IDs to subscribe for.\n */\n #subscribeStakedBalance(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n ): void {\n const source = this.#stakedBalanceDataSource;\n if (!source) {\n return;\n }\n const availableChains = new Set(source.getActiveChainsSync());\n const assignedChains = chainIds.filter((chainId) =>\n availableChains.has(chainId),\n );\n\n if (assignedChains.length === 0) {\n this.#unsubscribeDataSource(source);\n return;\n }\n\n this.#subscribeDataSource(source, accounts, assignedChains);\n }\n\n /**\n * Build a mapping of chainId -> accounts that support that chain.\n * Only includes chains that are in the chainsToSubscribe set.\n *\n * @param accounts - Array of accounts to build mapping for.\n * @param chainsToSubscribe - Set of chain IDs to include in the mapping.\n * @returns Map of chainId to array of accounts that support that chain.\n */\n #buildChainToAccountsMap(\n accounts: InternalAccount[],\n chainsToSubscribe: Set<ChainId>,\n ): Map<ChainId, InternalAccount[]> {\n const chainToAccounts = new Map<ChainId, InternalAccount[]>();\n for (const account of accounts) {\n for (const chainId of this.#getEnabledChainsForAccount(account)) {\n if (!chainsToSubscribe.has(chainId)) {\n continue;\n }\n let list = chainToAccounts.get(chainId);\n if (!list) {\n list = [];\n chainToAccounts.set(chainId, list);\n }\n list.push(account);\n }\n }\n return chainToAccounts;\n }\n\n /**\n * Subscribe to a specific data source with accounts and chains.\n * Uses the data source name as the subscription key for batching.\n *\n * @param source - The balance data source instance.\n * @param accounts - Array of accounts to subscribe for.\n * @param chains - Array of chain IDs to subscribe for.\n */\n #subscribeDataSource(\n source: AbstractDataSource<string, DataSourceState>,\n accounts: InternalAccount[],\n chains: ChainId[],\n ): void {\n const sourceId = source.getName();\n const subscriptionKey = `ds:${sourceId}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n log('Subscribe to data source', {\n sourceId,\n subscriptionKey,\n isUpdate,\n accountCount: accounts.length,\n chainCount: chains.length,\n });\n\n const subscribeReq: SubscriptionRequest = {\n request: this.#buildDataRequest(accounts, chains, {\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n updateInterval: this.#defaultUpdateInterval,\n }),\n subscriptionId: subscriptionKey,\n isUpdate,\n onAssetsUpdate: (response, request) =>\n this.handleAssetsUpdate(response, sourceId, request),\n getAssetsState: () => this.state,\n };\n\n source.subscribe(subscribeReq).catch((error) => {\n console.error(\n `[AssetsController] Failed to subscribe to '${sourceId}':`,\n error,\n );\n });\n\n // Track subscription\n const subscription: SubscriptionResponse = {\n chains,\n accountId: subscriptionKey,\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n unsubscribe: () => {\n this.#activeSubscriptions.delete(subscriptionKey);\n },\n };\n\n this.#activeSubscriptions.set(subscriptionKey, subscription);\n }\n\n /**\n * Unsubscribe from a data source if we have an active subscription.\n *\n * @param source - The balance data source instance to unsubscribe from.\n */\n #unsubscribeDataSource(\n source: AbstractDataSource<string, DataSourceState>,\n ): void {\n const subscriptionKey = `ds:${source.getName()}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n\n if (existingSubscription) {\n source.unsubscribe(subscriptionKey).catch(() => undefined);\n existingSubscription.unsubscribe();\n }\n }\n\n // ============================================================================\n // HELPERS\n // ============================================================================\n\n /**\n * Build a DataRequest with accountsWithSupportedChains (enabled chains ∩ account scope ∩ requested chainIds).\n *\n * @param accounts - Accounts to include.\n * @param chainIds - Requested chain IDs (e.g. enabled chains or subset).\n * @param partial - Rest of the request (dataTypes, assetTypes, etc.).\n * @returns DataRequest with accountsWithSupportedChains and chainIds.\n */\n #buildDataRequest(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n partial: Partial<\n Omit<DataRequest, 'accountsWithSupportedChains' | 'chainIds'>\n > = {},\n ): DataRequest {\n const chainIdSet = new Set(chainIds);\n const accountsWithSupportedChains: AccountWithSupportedChains[] =\n accounts.map((account) => ({\n account,\n supportedChains: this.#getEnabledChainsForAccount(account).filter(\n (chain) => chainIdSet.has(chain),\n ),\n }));\n return {\n accountsWithSupportedChains,\n chainIds,\n ...partial,\n } as DataRequest;\n }\n\n /**\n * Get the chains that an account supports based on its scopes.\n * Returns the intersection of the account's scopes and the enabled chains.\n *\n * @param account - The account to get supported chains for.\n * @returns Array of ChainIds that the account supports and are enabled.\n */\n #getEnabledChainsForAccount(account: InternalAccount): ChainId[] {\n // Account scopes are CAIP-2 chain IDs like \"eip155:1\", \"solana:mainnet\", \"bip122:...\"\n const scopes = account.scopes ?? [];\n const result: ChainId[] = [];\n\n for (const scope of scopes) {\n const scopeStr = scope as string;\n if (!isCaipChainId(scopeStr)) {\n result.push(scope);\n continue;\n }\n const { namespace, reference } = parseCaipChainId(scopeStr);\n\n // Wildcard scope (e.g., \"eip155:0\" means all enabled chains in that namespace)\n if (reference === '0') {\n for (const chain of this.#enabledChains) {\n if (isCaipChainId(chain)) {\n const chainParsed = parseCaipChainId(chain);\n if (chainParsed.namespace === namespace) {\n result.push(chain);\n }\n }\n }\n } else if (namespace === 'eip155' && isStrictHexString(reference)) {\n // Normalize hex to decimal for EIP155\n result.push(`eip155:${parseInt(reference, 16)}` as ChainId);\n } else {\n result.push(scope);\n }\n }\n\n return result;\n }\n\n // ============================================================================\n // EVENT HANDLERS\n // ============================================================================\n\n async #handleAccountGroupChanged(): Promise<void> {\n const accounts = this.#selectedAccounts;\n\n log('Account group changed', {\n accountCount: accounts.length,\n accountIds: accounts.map((a) => a.id),\n });\n\n // Subscribe and fetch for the new account group\n this.#subscribeAssets();\n if (accounts.length > 0) {\n await this.getAssets(accounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n });\n }\n\n this.#ensureNativeBalancesDefaultZero();\n }\n\n async #handleEnabledNetworksChanged(\n enabledNetworkMap: NetworkEnablementControllerState['enabledNetworkMap'],\n ): Promise<void> {\n const previousChains = this.#enabledChains;\n this.#enabledChains = this.#extractEnabledChains(enabledNetworkMap);\n\n // Find newly enabled chains (in new set but not in previous)\n const addedChains: ChainId[] = [];\n for (const chain of this.#enabledChains) {\n if (!previousChains.has(chain)) {\n addedChains.push(chain);\n }\n }\n\n // Find disabled chains to clean up (in previous but not in new)\n const removedChains: ChainId[] = [];\n for (const chain of previousChains) {\n if (!this.#enabledChains.has(chain)) {\n removedChains.push(chain);\n }\n }\n\n log('Enabled networks changed', {\n previousCount: previousChains.size,\n newCount: this.#enabledChains.size,\n addedChains,\n removedChains,\n });\n\n // Note: We intentionally do NOT delete balance data for disabled chains.\n // Users may want to see historical balances even if the network is currently disabled.\n // The data will simply not be updated until the network is re-enabled.\n\n // Refresh subscriptions for new chain set\n this.#subscribeAssets();\n\n // Do one-time fetch for newly enabled chains; merge so we keep existing chain balances\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n await this.getAssets(this.#selectedAccounts, {\n chainIds: addedChains,\n forceUpdate: true,\n updateMode: 'merge',\n });\n }\n\n this.#ensureNativeBalancesDefaultZero();\n }\n\n /**\n * Handle assets updated from a data source.\n * Called via the onAssetsUpdate callback passed in SubscriptionRequest when the controller subscribes to a data source.\n * Enriches the response with token metadata (via middlewares) before updating state.\n *\n * @param response - The data response with updated assets\n * @param sourceId - The data source ID reporting the update\n * @param request - Optional original request for context when enriching\n */\n async handleAssetsUpdate(\n response: DataResponse,\n sourceId: string,\n request?: DataRequest,\n ): Promise<void> {\n log('Assets updated from data source', {\n sourceId,\n hasBalance: Boolean(response.assetsBalance),\n hasPrice: Boolean(response.assetsPrice),\n });\n\n // Run through enrichment middlewares (Detection, then Token + Price in parallel)\n // Include 'metadata' in dataTypes so TokenDataSource runs to enrich detected assets\n const { response: enrichedResponse } = await this.#executeMiddlewares(\n [\n this.#detectionMiddleware,\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ],\n request ?? {\n accountsWithSupportedChains: [],\n chainIds: [],\n dataTypes: ['balance', 'metadata', 'price'],\n },\n response,\n );\n\n await this.#updateState(enrichedResponse);\n }\n\n // ============================================================================\n // CLEANUP\n // ============================================================================\n\n destroy(): void {\n log('Destroying AssetsController', {\n dataSourceCount: this.#allBalanceDataSources.length,\n subscriptionCount: this.#activeSubscriptions.size,\n });\n\n // Destroy instantiated data sources\n this.#backendWebsocketDataSource?.destroy?.();\n this.#accountsApiDataSource?.destroy?.();\n this.#snapDataSource?.destroy?.();\n this.#rpcDataSource?.destroy?.();\n this.#stakedBalanceDataSource?.destroy?.();\n\n // Stop all active subscriptions\n this.#stop();\n\n if (this.#unsubscribeBasicFunctionality) {\n this.#unsubscribeBasicFunctionality();\n this.#unsubscribeBasicFunctionality = null;\n }\n\n // Unregister action handlers\n this.messenger.unregisterActionHandler('AssetsController:getAssets');\n this.messenger.unregisterActionHandler('AssetsController:getAssetsBalance');\n this.messenger.unregisterActionHandler('AssetsController:getAssetMetadata');\n this.messenger.unregisterActionHandler('AssetsController:getAssetsPrice');\n this.messenger.unregisterActionHandler(\n 'AssetsController:getExchangeRatesForBridge',\n );\n this.messenger.unregisterActionHandler(\n 'AssetsController:getStateForTransactionPay',\n );\n this.messenger.unregisterActionHandler('AssetsController:addCustomAsset');\n this.messenger.unregisterActionHandler(\n 'AssetsController:removeCustomAsset',\n );\n this.messenger.unregisterActionHandler('AssetsController:getCustomAssets');\n this.messenger.unregisterActionHandler('AssetsController:hideAsset');\n this.messenger.unregisterActionHandler('AssetsController:unhideAsset');\n }\n}\n"]}
1
+ {"version":3,"file":"AssetsController.cjs","sourceRoot":"","sources":["../src/AssetsController.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n AccountTreeControllerGetAccountsFromSelectedAccountGroupAction,\n AccountTreeControllerSelectedAccountGroupChangeEvent,\n} from '@metamask/account-tree-controller';\nimport type { GetTokenListState } from '@metamask/assets-controllers';\nimport { BaseController } from '@metamask/base-controller';\nimport type {\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n StateMetadata,\n} from '@metamask/base-controller';\nimport type { ClientControllerStateChangeEvent } from '@metamask/client-controller';\nimport { clientControllerSelectors } from '@metamask/client-controller';\nimport type {\n ApiPlatformClient,\n BackendWebSocketServiceActions,\n BackendWebSocketServiceEvents,\n SupportedCurrency,\n} from '@metamask/core-backend';\nimport type {\n KeyringControllerLockEvent,\n KeyringControllerUnlockEvent,\n} from '@metamask/keyring-controller';\nimport type { InternalAccount } from '@metamask/keyring-internal-api';\nimport type { Messenger } from '@metamask/messenger';\nimport type {\n NetworkControllerGetNetworkClientByIdAction,\n NetworkControllerGetStateAction,\n NetworkControllerStateChangeEvent,\n} from '@metamask/network-controller';\nimport type {\n NetworkEnablementControllerGetStateAction,\n NetworkEnablementControllerEvents,\n NetworkEnablementControllerState,\n} from '@metamask/network-enablement-controller';\nimport type {\n GetPermissions,\n PermissionControllerStateChange,\n} from '@metamask/permission-controller';\nimport type { PreferencesControllerStateChangeEvent } from '@metamask/preferences-controller';\nimport type {\n GetRunnableSnaps,\n HandleSnapRequest,\n} from '@metamask/snaps-controllers';\nimport type {\n TransactionControllerIncomingTransactionsReceivedEvent,\n TransactionControllerTransactionConfirmedEvent,\n} from '@metamask/transaction-controller';\nimport {\n isCaipChainId,\n isStrictHexString,\n parseCaipAssetType,\n parseCaipChainId,\n} from '@metamask/utils';\nimport { Mutex } from 'async-mutex';\nimport BigNumberJS from 'bignumber.js';\nimport { isEqual } from 'lodash';\n\nimport type { AssetsControllerMethodActions } from './AssetsController-method-action-types';\nimport type {\n AbstractDataSource,\n DataSourceState,\n SubscriptionRequest,\n} from './data-sources/AbstractDataSource';\nimport type { AccountsApiDataSourceConfig } from './data-sources/AccountsApiDataSource';\nimport { AccountsApiDataSource } from './data-sources/AccountsApiDataSource';\nimport { BackendWebsocketDataSource } from './data-sources/BackendWebsocketDataSource';\nimport type { PriceDataSourceConfig } from './data-sources/PriceDataSource';\nimport { PriceDataSource } from './data-sources/PriceDataSource';\nimport type { RpcDataSourceConfig } from './data-sources/RpcDataSource';\nimport { RpcDataSource } from './data-sources/RpcDataSource';\nimport type { AccountsControllerAccountBalancesUpdatedEvent } from './data-sources/SnapDataSource';\nimport { SnapDataSource } from './data-sources/SnapDataSource';\nimport type { StakedBalanceDataSourceConfig } from './data-sources/StakedBalanceDataSource';\nimport { StakedBalanceDataSource } from './data-sources/StakedBalanceDataSource';\nimport { TokenDataSource } from './data-sources/TokenDataSource';\nimport { projectLogger, createModuleLogger } from './logger';\nimport { DetectionMiddleware } from './middlewares/DetectionMiddleware';\nimport {\n createParallelBalanceMiddleware,\n createParallelMiddleware,\n} from './middlewares/ParallelMiddleware';\nimport type {\n AccountId,\n AssetPreferences,\n AssetsUpdateMode,\n ChainId,\n Caip19AssetId,\n AssetMetadata,\n FungibleAssetMetadata,\n AssetPrice,\n AssetBalance,\n AccountWithSupportedChains,\n AssetType,\n DataType,\n DataRequest,\n DataResponse,\n FetchContext,\n FetchNextFunction,\n NextFunction,\n Middleware,\n SubscriptionResponse,\n Asset,\n AssetsControllerStateInternal,\n} from './types';\nimport {\n normalizeAssetId,\n formatExchangeRatesForBridge,\n formatStateForTransactionPay,\n} from './utils';\nimport type {\n BridgeExchangeRatesFormat,\n TransactionPayLegacyFormat,\n} from './utils';\n\n// ============================================================================\n// PENDING TOKEN METADATA (UI input format for addCustomAsset)\n// ============================================================================\n\n/**\n * Metadata format passed from the UI when adding a custom token.\n * Mirrors the \"pendingTokens\" shape used by the extension.\n */\nexport type PendingTokenMetadata = {\n address: string;\n symbol: string;\n name: string;\n decimals: number;\n iconUrl?: string;\n aggregators?: string[];\n occurrences?: number;\n chainId: string;\n unlisted?: boolean;\n};\n\n// ============================================================================\n// CONTROLLER CONSTANTS\n// ============================================================================\n\nconst CONTROLLER_NAME = 'AssetsController' as const;\n\n/** Method names exposed as messenger actions (AssetsController:getAssets, etc.) */\nconst MESSENGER_EXPOSED_METHODS = [\n 'getAssets',\n 'getAssetsBalance',\n 'getAssetMetadata',\n 'getAssetsPrice',\n 'getExchangeRatesForBridge',\n 'getStateForTransactionPay',\n 'addCustomAsset',\n 'removeCustomAsset',\n 'getCustomAssets',\n 'hideAsset',\n 'unhideAsset',\n] as const;\n\n/** Default polling interval hint for data sources (30 seconds) */\nconst DEFAULT_POLLING_INTERVAL_MS = 30_000;\n\nconst log = createModuleLogger(projectLogger, CONTROLLER_NAME);\n\n// ============================================================================\n// STATE TYPES\n// ============================================================================\n\n/**\n * State structure for AssetsController.\n *\n * All values are JSON-serializable. UI preferences (e.g. hidden) are in\n * assetPreferences, not in metadata.\n *\n * @see AssetsControllerStateInternal for the semantic type structure\n */\nexport type AssetsControllerState = {\n /** Shared metadata for all assets (stored once per asset) */\n assetsInfo: { [assetId: string]: AssetMetadata };\n /** Per-account balance data */\n assetsBalance: { [accountId: string]: { [assetId: string]: AssetBalance } };\n /** Price data for assets */\n assetsPrice: { [assetId: string]: AssetPrice };\n /** Custom assets added by users per account (CAIP-19 asset IDs) */\n customAssets: { [accountId: string]: Caip19AssetId[] };\n /** UI preferences per asset (e.g. hidden) */\n assetPreferences: { [assetId: string]: AssetPreferences };\n /** Currently-active ISO 4217 currency code */\n selectedCurrency: SupportedCurrency;\n};\n\n/**\n * Returns the default state for AssetsController.\n *\n * @returns The default AssetsController state with empty maps.\n */\nexport function getDefaultAssetsControllerState(): AssetsControllerState {\n return {\n assetsInfo: {},\n assetsBalance: {},\n assetsPrice: {},\n customAssets: {},\n assetPreferences: {},\n selectedCurrency: 'usd',\n };\n}\n\n// ============================================================================\n// MESSENGER TYPES\n// ============================================================================\n\nexport type AssetsControllerGetStateAction = ControllerGetStateAction<\n typeof CONTROLLER_NAME,\n AssetsControllerState\n>;\n\nexport type AssetsControllerActions =\n | AssetsControllerGetStateAction\n | AssetsControllerMethodActions;\n\nexport type AssetsControllerStateChangeEvent = ControllerStateChangeEvent<\n typeof CONTROLLER_NAME,\n AssetsControllerState\n>;\n\nexport type AssetsControllerBalanceChangedEvent = {\n type: `${typeof CONTROLLER_NAME}:balanceChanged`;\n payload: [\n {\n accountId: AccountId;\n assetId: Caip19AssetId;\n previousAmount: string;\n newAmount: string;\n },\n ];\n};\n\nexport type AssetsControllerPriceChangedEvent = {\n type: `${typeof CONTROLLER_NAME}:priceChanged`;\n payload: [{ prices: Record<Caip19AssetId, AssetPrice> }];\n};\n\nexport type AssetsControllerAssetsDetectedEvent = {\n type: `${typeof CONTROLLER_NAME}:assetsDetected`;\n payload: [{ accountId: AccountId; assetIds: Caip19AssetId[] }];\n};\n\nexport type AssetsControllerEvents =\n | AssetsControllerStateChangeEvent\n | AssetsControllerBalanceChangedEvent\n | AssetsControllerPriceChangedEvent\n | AssetsControllerAssetsDetectedEvent;\n\ntype AllowedActions =\n // AssetsController\n | AccountTreeControllerGetAccountsFromSelectedAccountGroupAction\n // RpcDataSource\n | GetTokenListState\n | NetworkControllerGetStateAction\n | NetworkControllerGetNetworkClientByIdAction\n // RpcDataSource, StakedBalanceDataSource\n | NetworkEnablementControllerGetStateAction\n // SnapDataSource\n | GetRunnableSnaps\n | HandleSnapRequest\n | GetPermissions\n // BackendWebsocketDataSource\n | BackendWebSocketServiceActions;\n\ntype AllowedEvents =\n // AssetsController\n | AccountTreeControllerSelectedAccountGroupChangeEvent\n | ClientControllerStateChangeEvent\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent\n | PreferencesControllerStateChangeEvent\n // RpcDataSource, StakedBalanceDataSource\n | NetworkControllerStateChangeEvent\n | TransactionControllerTransactionConfirmedEvent\n | TransactionControllerIncomingTransactionsReceivedEvent\n // StakedBalanceDataSource\n | NetworkEnablementControllerEvents\n // SnapDataSource\n | AccountsControllerAccountBalancesUpdatedEvent\n | PermissionControllerStateChange\n // BackendWebsocketDataSource\n | BackendWebSocketServiceEvents;\n\nexport type AssetsControllerMessenger = Messenger<\n typeof CONTROLLER_NAME,\n AssetsControllerActions | AllowedActions,\n AssetsControllerEvents | AllowedEvents\n>;\n\n// ============================================================================\n// CONTROLLER OPTIONS\n// ============================================================================\n\n/**\n * Payload for the first init/fetch MetaMetrics event.\n * Passed to the optional trackMetaMetricsEvent callback when the initial\n * asset fetch completes after unlock or app open.\n */\nexport type AssetsControllerFirstInitFetchMetaMetricsPayload = {\n /** Duration of the first init fetch in milliseconds (wall-clock). */\n durationMs: number;\n /** Chain IDs requested in the fetch (e.g. ['eip155:1', 'eip155:137']). */\n chainIds: string[];\n /**\n * Exclusive latency in ms per data source (time spent in that source only).\n * Sum of values approximates durationMs. Order: same as middleware chain.\n */\n durationByDataSource: Record<string, number>;\n};\n\nexport type AssetsControllerOptions = {\n messenger: AssetsControllerMessenger;\n state?: Partial<AssetsControllerState>;\n /** Default polling interval hint passed to data sources (ms) */\n defaultUpdateInterval?: number;\n /** Function to determine if the controller is enabled. Defaults to true. */\n isEnabled?: () => boolean;\n /**\n * Getter for basic functionality (matches the \"Basic functionality\" setting in the UI).\n * When it returns true, internet services are on: token/price APIs are used for metadata, price,\n * and price subscription. When false, only RPC is used (no token/price APIs).\n * No value is stored; the getter is invoked when needed.\n * Defaults to () => true when not provided (APIs enabled).\n */\n isBasicFunctionality?: () => boolean;\n /**\n * Called by the controller with an onChange callback. The consumer subscribes to its own\n * basic-functionality source (e.g. PreferencesController:stateChange in extension, or a\n * different mechanism in mobile) and invokes onChange(isBasic) when the value changes.\n * The controller will then refresh its subscriptions. May return an unsubscribe function\n * called on controller destroy. Optional; when omitted, basic-functionality changes are not\n * subscribed to (e.g. host can notify via root messenger or another path).\n */\n subscribeToBasicFunctionalityChange?: (\n onChange: (isBasic: boolean) => void,\n ) => void | (() => void);\n /**\n * API client for balance/price/metadata. The controller instantiates data sources\n * and uses them directly when this is provided.\n */\n queryApiClient: ApiPlatformClient;\n /** Optional configuration for RpcDataSource. */\n rpcDataSourceConfig?: RpcDataSourceConfig;\n /**\n * Optional callback invoked when the first init/fetch completes (e.g. after unlock).\n * Use this to track first init fetch duration in MetaMetrics.\n */\n trackMetaMetricsEvent?: (\n payload: AssetsControllerFirstInitFetchMetaMetricsPayload,\n ) => void;\n /** Optional configuration for AccountsApiDataSource. */\n accountsApiDataSourceConfig?: AccountsApiDataSourceConfig;\n /** Optional configuration for PriceDataSource. */\n priceDataSourceConfig?: PriceDataSourceConfig;\n /** Optional configuration for StakedBalanceDataSource. */\n stakedBalanceDataSourceConfig?: StakedBalanceDataSourceConfig;\n};\n\n// ============================================================================\n// STATE METADATA\n// ============================================================================\n\nconst stateMetadata: StateMetadata<AssetsControllerState> = {\n assetsInfo: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n assetsBalance: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n assetsPrice: {\n persist: false,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n customAssets: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n assetPreferences: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n selectedCurrency: {\n persist: true,\n includeInStateLogs: false,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n};\n\n// ============================================================================\n// HELPER FUNCTIONS\n// ============================================================================\n\nfunction extractChainId(assetId: Caip19AssetId): ChainId {\n const parsed = parseCaipAssetType(assetId);\n return parsed.chainId;\n}\n\n/**\n * Normalizes all asset IDs in a DataResponse.\n * This is applied at the controller level to ensure consistent state\n * regardless of how data sources format their asset IDs.\n *\n * @param response - The DataResponse to normalize.\n * @returns The normalized DataResponse with checksummed EVM addresses.\n */\nfunction normalizeResponse(response: DataResponse): DataResponse {\n const normalized: DataResponse = {};\n\n if (response.assetsInfo) {\n normalized.assetsInfo = {};\n for (const [assetId, metadata] of Object.entries(response.assetsInfo)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsInfo[normalizedId] = metadata;\n }\n }\n\n if (response.assetsPrice) {\n normalized.assetsPrice = {};\n for (const [assetId, price] of Object.entries(response.assetsPrice)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsPrice[normalizedId] = price;\n }\n }\n\n if (response.assetsBalance) {\n normalized.assetsBalance = {};\n for (const [accountId, balances] of Object.entries(\n response.assetsBalance,\n )) {\n normalized.assetsBalance[accountId] = {};\n for (const [assetId, balance] of Object.entries(balances)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsBalance[accountId][normalizedId] = balance;\n }\n }\n }\n\n // Preserve detectedAssets with normalized asset IDs\n if (response.detectedAssets) {\n normalized.detectedAssets = {};\n for (const [accountId, assetIds] of Object.entries(\n response.detectedAssets,\n )) {\n normalized.detectedAssets[accountId] = assetIds.map((assetId) =>\n normalizeAssetId(assetId),\n );\n }\n }\n\n // Preserve errors (chain IDs don't need normalization)\n if (response.errors) {\n normalized.errors = { ...response.errors };\n }\n\n if (response.updateMode) {\n normalized.updateMode = response.updateMode;\n }\n\n return normalized;\n}\n\n// ============================================================================\n// CONTROLLER IMPLEMENTATION\n// ============================================================================\n\n/**\n * AssetsController provides a unified interface for managing asset balances\n * across all blockchain networks (EVM and non-EVM) and all asset types.\n *\n * ## Core Responsibilities\n *\n * 1. **One-Time Fetch (Sync)**: For initial load, force refresh, or on-demand queries.\n * Uses `getAssets()`, `getAssetsBalance()`, etc. with `forceUpdate: true`.\n *\n * 2. **Async Subscriptions**: Subscribes to data sources for ongoing updates.\n * Data sources push updates via callbacks; the controller updates state.\n *\n * 3. **Dynamic Source Selection**: Routes requests to appropriate data sources\n * based on which chains they support. When active chains change, the controller\n * dynamically adjusts subscriptions.\n *\n * 4. **Client + Keyring Lifecycle**: Starts subscriptions only when both the UI is\n * open (ClientController) and the wallet is unlocked (KeyringController).\n * Stops when either the UI closes or the keyring locks. See client-controller\n * README for the combined pattern.\n *\n * ## Architecture\n *\n * - Data sources declare their supported chains (async, can change over time)\n * - Data sources are responsible for their own update mechanisms (WebSocket, polling, events)\n * - The controller does NOT manage polling - it simply receives pushed updates\n */\nexport class AssetsController extends BaseController<\n typeof CONTROLLER_NAME,\n AssetsControllerState,\n AssetsControllerMessenger\n> {\n /** Whether the controller is enabled */\n readonly #isEnabled: boolean;\n\n /** Getter for basic functionality (only balance fetch/subscribe use RPC; token/price API not used). No attribute stored. */\n readonly #isBasicFunctionality: () => boolean;\n\n /** Default update interval hint passed to data sources */\n readonly #defaultUpdateInterval: number;\n\n /** Optional callback for first init/fetch MetaMetrics (duration). */\n readonly #trackMetaMetricsEvent?: (\n payload: AssetsControllerFirstInitFetchMetaMetricsPayload,\n ) => void;\n\n /** Whether we have already reported first init fetch for this session (reset on #stop). */\n #firstInitFetchReported = false;\n\n /** Whether the client (UI) is open. Combined with #keyringUnlocked for #updateActive. */\n #uiOpen = false;\n\n /** Whether the keyring is unlocked. Combined with #uiOpen for #updateActive. */\n #keyringUnlocked = false;\n\n readonly #controllerMutex = new Mutex();\n\n /**\n * Active balance subscriptions keyed by account ID.\n * Each account has one logical subscription that may span multiple data sources.\n * For example, if WebSocket covers chains A,B and RPC covers chain C,\n * the account subscribes to both data sources for its chains.\n */\n readonly #activeSubscriptions: Map<string, SubscriptionResponse> = new Map();\n\n /** Currently enabled chains from NetworkEnablementController */\n #enabledChains: Set<ChainId> = new Set();\n\n /**\n * Get the currently selected accounts from AccountTreeController.\n * This includes all accounts in the same group as the selected account\n * (EVM, Bitcoin, Solana, Tron, etc. that belong to the same logical account group).\n *\n * @returns Array of InternalAccount objects from the selected account group.\n */\n get #selectedAccounts(): InternalAccount[] {\n return this.messenger.call(\n 'AccountTreeController:getAccountsFromSelectedAccountGroup',\n );\n }\n\n readonly #backendWebsocketDataSource: BackendWebsocketDataSource;\n\n readonly #accountsApiDataSource: AccountsApiDataSource;\n\n readonly #snapDataSource: SnapDataSource;\n\n readonly #rpcDataSource: RpcDataSource;\n\n readonly #stakedBalanceDataSource: StakedBalanceDataSource;\n\n /**\n * All balance data sources (used for unsubscription in #stop so we can clean up\n * regardless of current isBasicFunctionality mode).\n * Note: StakedBalanceDataSource is excluded because it provides supplementary\n * data and should not participate in chain-claiming.\n *\n * @returns The four balance data source instances in priority order.\n */\n get #allBalanceDataSources(): [\n BackendWebsocketDataSource,\n AccountsApiDataSource,\n SnapDataSource,\n RpcDataSource,\n ] {\n return [\n this.#backendWebsocketDataSource,\n this.#accountsApiDataSource,\n this.#snapDataSource,\n this.#rpcDataSource,\n ];\n }\n\n readonly #priceDataSource: PriceDataSource;\n\n readonly #detectionMiddleware: DetectionMiddleware;\n\n readonly #tokenDataSource: TokenDataSource;\n\n #unsubscribeBasicFunctionality: (() => void) | null = null;\n\n readonly #onActiveChainsUpdated: (\n dataSourceId: string,\n activeChains: ChainId[],\n previousChains: ChainId[],\n ) => void;\n\n constructor({\n messenger,\n state = {},\n defaultUpdateInterval = DEFAULT_POLLING_INTERVAL_MS,\n isEnabled = (): boolean => true,\n isBasicFunctionality,\n subscribeToBasicFunctionalityChange,\n queryApiClient,\n rpcDataSourceConfig,\n trackMetaMetricsEvent,\n accountsApiDataSourceConfig,\n priceDataSourceConfig,\n stakedBalanceDataSourceConfig,\n }: AssetsControllerOptions) {\n super({\n name: CONTROLLER_NAME,\n messenger,\n metadata: stateMetadata,\n state: {\n ...getDefaultAssetsControllerState(),\n ...state,\n },\n });\n\n this.#isEnabled = isEnabled();\n this.#isBasicFunctionality = isBasicFunctionality ?? ((): boolean => true);\n this.#defaultUpdateInterval = defaultUpdateInterval;\n this.#trackMetaMetricsEvent = trackMetaMetricsEvent;\n const rpcConfig = rpcDataSourceConfig ?? {};\n\n this.#onActiveChainsUpdated = (\n dataSourceName: string,\n chains: ChainId[],\n previousChains: ChainId[],\n ): void => {\n this.#handleActiveChainsUpdate(dataSourceName, chains, previousChains);\n };\n\n this.#backendWebsocketDataSource = new BackendWebsocketDataSource({\n messenger: this.messenger,\n queryApiClient,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n });\n this.#accountsApiDataSource = new AccountsApiDataSource({\n queryApiClient,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...accountsApiDataSourceConfig,\n });\n this.#snapDataSource = new SnapDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n });\n this.#rpcDataSource = new RpcDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...rpcConfig,\n });\n this.#stakedBalanceDataSource = new StakedBalanceDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...stakedBalanceDataSourceConfig,\n });\n this.#tokenDataSource = new TokenDataSource({\n queryApiClient,\n });\n this.#priceDataSource = new PriceDataSource({\n queryApiClient,\n getSelectedCurrency: (): SupportedCurrency => this.state.selectedCurrency,\n ...priceDataSourceConfig,\n });\n this.#detectionMiddleware = new DetectionMiddleware();\n\n if (!this.#isEnabled) {\n log('AssetsController is disabled, skipping initialization');\n return;\n }\n\n log('Initializing AssetsController', {\n defaultUpdateInterval: this.#defaultUpdateInterval,\n });\n\n this.#initializeState();\n this.#subscribeToEvents();\n this.#registerActionHandlers();\n // Subscriptions start only when both UI is open and keyring unlocked -> #updateActive().\n\n // Subscribe to basic-functionality changes after construction so a synchronous\n // onChange during subscribe cannot run before data sources are initialized.\n if (subscribeToBasicFunctionalityChange) {\n const unsubscribe = subscribeToBasicFunctionalityChange((isBasic) =>\n this.handleBasicFunctionalityChange(isBasic),\n );\n if (typeof unsubscribe === 'function') {\n this.#unsubscribeBasicFunctionality = unsubscribe;\n }\n }\n }\n\n // ============================================================================\n // INITIALIZATION\n // ============================================================================\n\n #initializeState(): void {\n const { enabledNetworkMap } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n this.#enabledChains = this.#extractEnabledChains(enabledNetworkMap);\n\n log('Initialized state', {\n enabledNetworkMap,\n enabledChains: this.#enabledChains,\n });\n }\n\n /**\n * Extract enabled chains from enabledNetworkMap.\n * Returns CAIP-2 chain IDs for all enabled networks across all namespaces.\n *\n * Note: For EIP155 (EVM) chains, the reference is normalized to decimal format\n * to ensure consistency with CAIP-2 standard and API responses.\n *\n * @param enabledNetworkMap - The enabled network map from NetworkEnablementController.\n * @returns Set of CAIP-2 chain IDs for all enabled networks.\n */\n #extractEnabledChains(\n enabledNetworkMap: NetworkEnablementControllerState['enabledNetworkMap'],\n ): Set<ChainId> {\n const chains = new Set<ChainId>();\n\n for (const [namespace, networks] of Object.entries(enabledNetworkMap)) {\n for (const [reference, isEnabled] of Object.entries(networks)) {\n if (isEnabled) {\n // Check if reference is already a full CAIP-2 chain ID (contains colon)\n if (reference.includes(':')) {\n // Already a full chain ID, use as-is\n chains.add(reference as ChainId);\n } else {\n // Normalize EIP155 chain references from hex to decimal (CAIP-2 standard)\n const normalizedReference = this.#normalizeChainReference(\n namespace,\n reference,\n );\n chains.add(`${namespace}:${normalizedReference}`);\n }\n }\n }\n }\n return chains;\n }\n\n /**\n * Normalize chain reference to CAIP-2 standard format.\n * For EIP155, converts hex chain IDs to decimal.\n *\n * @param namespace - The chain namespace (e.g., \"eip155\").\n * @param reference - The chain reference (e.g., \"0x1\" or \"1\").\n * @returns The normalized chain reference in decimal format.\n */\n #normalizeChainReference(namespace: string, reference: string): string {\n if (namespace === 'eip155' && isStrictHexString(reference)) {\n // Convert hex to decimal for EIP155 chains\n return parseInt(reference, 16).toString();\n }\n return reference;\n }\n\n #subscribeToEvents(): void {\n // Subscribe to account group changes (when user switches between account groups like Account 1 -> Account 2)\n this.messenger.subscribe(\n 'AccountTreeController:selectedAccountGroupChange',\n () => {\n this.#handleAccountGroupChanged().catch(console.error);\n },\n );\n\n // Subscribe to network enablement changes (only enabledNetworkMap)\n this.messenger.subscribe(\n 'NetworkEnablementController:stateChange',\n ({ enabledNetworkMap }) => {\n this.#handleEnabledNetworksChanged(enabledNetworkMap).catch(\n console.error,\n );\n },\n );\n\n // Client + Keyring lifecycle: only run when UI is open AND keyring is unlocked\n this.messenger.subscribe(\n 'ClientController:stateChange',\n (isUiOpen: boolean) => {\n this.#uiOpen = isUiOpen;\n this.#updateActive();\n },\n clientControllerSelectors.selectIsUiOpen,\n );\n this.messenger.subscribe('KeyringController:unlock', () => {\n this.#keyringUnlocked = true;\n this.#updateActive();\n });\n this.messenger.subscribe('KeyringController:lock', () => {\n this.#keyringUnlocked = false;\n this.#updateActive();\n });\n }\n\n /**\n * Start or stop asset tracking based on client (UI) open state and keyring\n * unlock state. Only runs when both UI is open and keyring is unlocked.\n */\n #updateActive(): void {\n const shouldRun = this.#uiOpen && this.#keyringUnlocked;\n if (shouldRun) {\n this.#start();\n } else {\n this.#stop();\n }\n }\n\n #registerActionHandlers(): void {\n this.messenger.registerMethodActionHandlers(\n this,\n MESSENGER_EXPOSED_METHODS,\n );\n }\n\n // ============================================================================\n // DATA SOURCE CHAIN MANAGEMENT\n // ============================================================================\n\n /**\n * Handle when a data source's supported chains change.\n * Used to refresh balance subscriptions and run a one-time fetch when a new chain is supported.\n *\n * - On any add/remove: re-subscribes to data sources so chain assignment stays correct.\n * - When chains are added: fetches balances for the new chains (for selected accounts on enabled networks).\n *\n * Controller does not store chains; sources report via this callback. previousChains is required for diff.\n *\n * @param dataSourceId - The identifier of the data source reporting the change.\n * @param activeChains - Currently active (supported and available) chain IDs for this source.\n * @param previousChains - Previous chains; used to compute added/removed.\n */\n #handleActiveChainsUpdate(\n dataSourceId: string,\n activeChains: ChainId[],\n previousChains: ChainId[],\n ): void {\n if (!this.#isEnabled) {\n return;\n }\n log('Data source active chains changed', {\n dataSourceId,\n chainCount: activeChains.length,\n chains: activeChains,\n });\n\n const previous: ChainId[] = previousChains;\n\n const previousSet = new Set(previous);\n const addedChains = activeChains.filter((ch) => !previousSet.has(ch));\n const removedChains = previous.filter((ch) => !activeChains.includes(ch));\n\n if (addedChains.length > 0 || removedChains.length > 0) {\n // Refresh subscriptions to use updated data source availability\n this.#subscribeAssets();\n }\n\n // If chains were added and we have selected accounts, do one-time fetch\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n const addedEnabledChains = addedChains.filter((chain) =>\n this.#enabledChains.has(chain),\n );\n if (addedEnabledChains.length > 0) {\n log('Fetching balances for newly added chains', { addedEnabledChains });\n this.getAssets(this.#selectedAccounts, {\n chainIds: addedEnabledChains,\n forceUpdate: true,\n updateMode: 'merge',\n }).catch((error) => {\n log('Failed to fetch balance for added chains', { error });\n });\n }\n }\n }\n\n /**\n * Returns the callback passed to data sources for reporting active chain updates.\n * Used by tests to simulate a data source reporting chain changes.\n *\n * @returns The onActiveChainsUpdated callback.\n */\n getOnActiveChainsUpdated(): (\n dataSourceId: string,\n activeChains: ChainId[],\n previousChains: ChainId[],\n ) => void {\n return this.#onActiveChainsUpdated;\n }\n\n // ============================================================================\n // MIDDLEWARE EXECUTION\n // ============================================================================\n\n /**\n * Execute middlewares with request/response context.\n * Returns response and exclusive duration per source (sum ≈ wall time).\n *\n * @param sources - Data sources or middlewares with getName() and assetsMiddleware (executed in order).\n * @param request - The data request.\n * @param initialResponse - Optional initial response (for enriching existing data).\n * @returns Response and durationByDataSource (exclusive ms per source name).\n */\n async #executeMiddlewares(\n sources: { getName(): string; assetsMiddleware: Middleware }[],\n request: DataRequest,\n initialResponse: DataResponse = {},\n ): Promise<{\n response: DataResponse;\n durationByDataSource: Record<string, number>;\n }> {\n const names = sources.map((source) => source.getName());\n const middlewares = sources.map((source) => source.assetsMiddleware);\n const inclusive: number[] = [];\n const wrapped = middlewares.map(\n (middleware, i) =>\n (async (\n ctx: FetchContext,\n next: FetchNextFunction,\n ): Promise<{\n request: DataRequest;\n response: DataResponse;\n getAssetsState: () => AssetsControllerStateInternal;\n }> => {\n const start = Date.now();\n try {\n return await middleware(ctx, next);\n } finally {\n inclusive[i] = Date.now() - start;\n }\n }) as Middleware,\n );\n\n const chain = wrapped.reduceRight<NextFunction>(\n (next, middleware) =>\n async (\n ctx,\n ): Promise<{\n request: DataRequest;\n response: DataResponse;\n getAssetsState: () => AssetsControllerStateInternal;\n }> => {\n try {\n return await middleware(ctx, next);\n } catch (error) {\n console.error('[AssetsController] Middleware failed:', error);\n return next(ctx);\n }\n },\n async (ctx) => ctx,\n );\n\n const result = await chain({\n request,\n response: initialResponse,\n getAssetsState: () => this.state as AssetsControllerStateInternal,\n });\n\n const durationByDataSource: Record<string, number> = {};\n for (let i = 0; i < inclusive.length; i++) {\n const nextInc = i + 1 < inclusive.length ? (inclusive[i + 1] ?? 0) : 0;\n const exclusive = Math.max(0, (inclusive[i] ?? 0) - nextInc);\n const name = names[i];\n if (name !== undefined) {\n durationByDataSource[name] = exclusive;\n }\n }\n return { response: result.response, durationByDataSource };\n }\n\n // ============================================================================\n // PUBLIC API: QUERY METHODS\n // ============================================================================\n\n async getAssets(\n accounts: InternalAccount[],\n options?: {\n chainIds?: ChainId[];\n assetTypes?: AssetType[];\n forceUpdate?: boolean;\n dataTypes?: DataType[];\n assetsForPriceUpdate?: Caip19AssetId[];\n /** When set to 'merge', fetch result is merged with existing state instead of replacing. Use for partial fetches (e.g. newly added chains). */\n updateMode?: AssetsUpdateMode;\n },\n ): Promise<Record<AccountId, Record<Caip19AssetId, Asset>>> {\n const chainIds = options?.chainIds ?? [...this.#enabledChains];\n const assetTypes = options?.assetTypes ?? ['fungible'];\n const dataTypes = options?.dataTypes ?? ['balance', 'metadata', 'price'];\n\n if (accounts.length === 0 || chainIds.length === 0) {\n return this.#getAssetsFromState(accounts, chainIds, assetTypes);\n }\n\n // Collect custom assets for all requested accounts\n const customAssets: Caip19AssetId[] = [];\n for (const account of accounts) {\n const accountCustomAssets = this.getCustomAssets(account.id);\n customAssets.push(...accountCustomAssets);\n }\n\n if (options?.forceUpdate) {\n const startTime = Date.now();\n const request = this.#buildDataRequest(accounts, chainIds, {\n assetTypes,\n dataTypes,\n customAssets: customAssets.length > 0 ? customAssets : undefined,\n forceUpdate: true,\n assetsForPriceUpdate: options?.assetsForPriceUpdate,\n });\n const sources = this.#isBasicFunctionality()\n ? [\n createParallelBalanceMiddleware([\n this.#accountsApiDataSource,\n this.#snapDataSource,\n this.#rpcDataSource,\n this.#stakedBalanceDataSource,\n ]),\n this.#detectionMiddleware,\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ]\n : [\n this.#rpcDataSource,\n this.#stakedBalanceDataSource,\n this.#detectionMiddleware,\n ];\n const { response, durationByDataSource } = await this.#executeMiddlewares(\n sources,\n request,\n );\n // Default to 'merge' when fetching a subset of chains so we don't wipe\n // balances from chains that weren't included in this fetch.\n const isPartialChainFetch =\n options?.chainIds !== undefined &&\n options.chainIds.length < this.#enabledChains.size;\n const updateMode =\n options?.updateMode ?? (isPartialChainFetch ? 'merge' : 'full');\n await this.#updateState({ ...response, updateMode });\n if (this.#trackMetaMetricsEvent && !this.#firstInitFetchReported) {\n this.#firstInitFetchReported = true;\n const durationMs = Date.now() - startTime;\n this.#trackMetaMetricsEvent({\n durationMs,\n chainIds,\n durationByDataSource,\n });\n }\n }\n\n const result = this.#getAssetsFromState(accounts, chainIds, assetTypes);\n return result;\n }\n\n async getAssetsBalance(\n accounts: InternalAccount[],\n options?: {\n chainIds?: ChainId[];\n assetTypes?: AssetType[];\n forceUpdate?: boolean;\n },\n ): Promise<Record<AccountId, Record<Caip19AssetId, AssetBalance>>> {\n // Reuse getAssets with dataTypes: ['balance'] only\n const assets = await this.getAssets(accounts, {\n chainIds: options?.chainIds,\n assetTypes: options?.assetTypes,\n forceUpdate: options?.forceUpdate,\n dataTypes: ['balance', 'metadata'],\n });\n\n // Extract just the balance from each asset\n const result: Record<AccountId, Record<Caip19AssetId, AssetBalance>> = {};\n for (const [accountId, accountAssets] of Object.entries(assets)) {\n result[accountId] = {};\n for (const [assetId, asset] of Object.entries(accountAssets)) {\n if (asset.balance) {\n result[accountId][assetId as Caip19AssetId] = asset.balance;\n }\n }\n }\n\n return result;\n }\n\n getAssetMetadata(assetId: Caip19AssetId): AssetMetadata | undefined {\n return this.state.assetsInfo[assetId] as AssetMetadata | undefined;\n }\n\n async getAssetsPrice(\n accounts: InternalAccount[],\n options?: {\n chainIds?: ChainId[];\n assetTypes?: AssetType[];\n forceUpdate?: boolean;\n },\n ): Promise<Record<Caip19AssetId, AssetPrice>> {\n const assets = await this.getAssets(accounts, {\n chainIds: options?.chainIds,\n assetTypes: options?.assetTypes,\n forceUpdate: options?.forceUpdate,\n dataTypes: ['price'],\n });\n\n // Extract just the price from each asset (flattened across accounts)\n const result: Record<Caip19AssetId, AssetPrice> = {};\n for (const accountAssets of Object.values(assets)) {\n for (const [assetId, asset] of Object.entries(accountAssets)) {\n if (asset.price) {\n result[assetId as Caip19AssetId] = asset.price;\n }\n }\n }\n\n return result;\n }\n\n /**\n * Returns exchange rates in the format expected by the bridge controller\n * (conversionRates, currencyRates, marketData, currentCurrency) so that\n * when useAssetsControllerForRates is true the bridge can use a single\n * action instead of MultichainAssetsRatesController, TokenRatesController,\n * and CurrencyRateController.\n *\n * @param options - Optional options for bridge rate conversion.\n * @param options.usdToSelectedCurrencyRate - When selectedCurrency is not 'usd', pass 1 USD = this many units of selected currency so that currencyRates and conversionRates use correct user-currency vs USD values; otherwise the bridge USD conversion will be wrong for non-USD currencies.\n * @returns Bridge-compatible exchange rate state derived from assetsPrice and selectedCurrency.\n */\n getExchangeRatesForBridge(options?: {\n usdToSelectedCurrencyRate?: number;\n }): BridgeExchangeRatesFormat {\n const { nativeAssetIdentifiers } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n return formatExchangeRatesForBridge({\n assetsPrice: this.state.assetsPrice,\n selectedCurrency: this.state.selectedCurrency,\n nativeAssetIdentifiers,\n networkConfigurationsByChainId,\n usdToSelectedCurrencyRate: options?.usdToSelectedCurrencyRate,\n });\n }\n\n /**\n * Returns state in the legacy format expected by transaction-pay-controller\n * (TokenBalancesController, AccountTrackerController, TokensController,\n * TokenRatesController, CurrencyRateController shapes) so that when\n * useAssetsController is true the transaction-pay-controller can use a\n * single action instead of five separate getState calls.\n *\n * @param options - Optional options for legacy rate conversion.\n * @param options.usdToSelectedCurrencyRate - When selectedCurrency is not 'usd', pass 1 USD = this many units of selected currency so that currencyRates and marketData use correct user-currency vs USD values; otherwise conversion rates will be wrong for non-USD currencies.\n * @returns Legacy-compatible state for transaction-pay-controller.\n */\n getStateForTransactionPay(options?: {\n usdToSelectedCurrencyRate?: number;\n }): TransactionPayLegacyFormat {\n const accounts = this.#selectedAccounts;\n const { nativeAssetIdentifiers } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n return formatStateForTransactionPay({\n assetsBalance: this.state.assetsBalance,\n assetsInfo: this.state.assetsInfo,\n assetsPrice: this.state.assetsPrice,\n selectedCurrency: this.state.selectedCurrency,\n accounts: accounts.map((a) => ({ id: a.id, address: a.address })),\n nativeAssetIdentifiers,\n networkConfigurationsByChainId,\n usdToSelectedCurrencyRate: options?.usdToSelectedCurrencyRate,\n });\n }\n\n // ============================================================================\n // CUSTOM ASSETS MANAGEMENT\n // ============================================================================\n\n /**\n * Add a custom asset for an account.\n * Custom assets are included in subscription and fetch operations.\n * Adding a custom asset also unhides it if it was previously hidden.\n *\n * When `pendingMetadata` is provided (e.g. from the extension's pending-tokens\n * flow), the token metadata is persisted immediately into `assetsInfo` so the\n * UI can render it without waiting for the next pipeline fetch.\n *\n * @param accountId - The account ID to add the custom asset for.\n * @param assetId - The CAIP-19 asset ID to add.\n * @param pendingMetadata - Optional token metadata from the UI (pendingTokens format).\n */\n async addCustomAsset(\n accountId: AccountId,\n assetId: Caip19AssetId,\n pendingMetadata?: PendingTokenMetadata,\n ): Promise<void> {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Adding custom asset', { accountId, assetId: normalizedAssetId });\n\n this.update((state) => {\n const customAssets = state.customAssets as Record<string, string[]>;\n if (!customAssets[accountId]) {\n customAssets[accountId] = [];\n }\n\n // Only add if not already present\n if (!customAssets[accountId].includes(normalizedAssetId)) {\n customAssets[accountId].push(normalizedAssetId);\n }\n\n // Unhide the asset if it was hidden (via assetPreferences)\n const prefs = state.assetPreferences[normalizedAssetId];\n if (prefs?.hidden) {\n delete prefs.hidden;\n if (Object.keys(prefs).length === 0) {\n delete state.assetPreferences[normalizedAssetId];\n }\n }\n\n // Persist metadata from the UI so the token is immediately renderable\n if (pendingMetadata) {\n const parsed = parseCaipAssetType(normalizedAssetId);\n let tokenType: FungibleAssetMetadata['type'] = 'erc20';\n if (parsed.assetNamespace === 'slip44') {\n tokenType = 'native';\n } else if (parsed.assetNamespace === 'spl') {\n tokenType = 'spl';\n }\n\n const assetMetadata: FungibleAssetMetadata = {\n type: tokenType,\n symbol: pendingMetadata.symbol,\n name: pendingMetadata.name,\n decimals: pendingMetadata.decimals,\n image: pendingMetadata.iconUrl,\n aggregators: pendingMetadata.aggregators,\n occurrences: pendingMetadata.occurrences,\n };\n\n (state.assetsInfo as Record<string, AssetMetadata>)[normalizedAssetId] =\n assetMetadata;\n }\n });\n\n // Fetch data for the newly added custom asset (merge to preserve other chains)\n const account = this.#selectedAccounts.find((a) => a.id === accountId);\n if (account) {\n const chainId = extractChainId(normalizedAssetId);\n await this.getAssets([account], {\n chainIds: [chainId],\n forceUpdate: true,\n updateMode: 'merge',\n });\n }\n }\n\n /**\n * Remove a custom asset from an account.\n *\n * @param accountId - The account ID to remove the custom asset from.\n * @param assetId - The CAIP-19 asset ID to remove.\n */\n removeCustomAsset(accountId: AccountId, assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Removing custom asset', { accountId, assetId: normalizedAssetId });\n\n this.update((state) => {\n if (state.customAssets[accountId]) {\n state.customAssets[accountId] = state.customAssets[accountId].filter(\n (id) => id !== normalizedAssetId,\n );\n\n // Clean up empty arrays\n if (state.customAssets[accountId].length === 0) {\n delete state.customAssets[accountId];\n }\n }\n });\n }\n\n /**\n * Get all custom assets for an account.\n *\n * @param accountId - The account ID to get custom assets for.\n * @returns Array of CAIP-19 asset IDs for the account's custom assets.\n */\n getCustomAssets(accountId: AccountId): Caip19AssetId[] {\n return this.state.customAssets[accountId] ?? [];\n }\n\n // ============================================================================\n // HIDDEN ASSETS MANAGEMENT\n // ============================================================================\n\n /**\n * Hide an asset globally.\n * Hidden assets are excluded from the asset list returned by getAssets.\n * The hidden state is stored in assetPreferences.\n *\n * @param assetId - The CAIP-19 asset ID to hide.\n */\n hideAsset(assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Hiding asset', { assetId: normalizedAssetId });\n\n this.update((state) => {\n if (!state.assetPreferences[normalizedAssetId]) {\n state.assetPreferences[normalizedAssetId] = {};\n }\n state.assetPreferences[normalizedAssetId].hidden = true;\n });\n }\n\n /**\n * Unhide an asset globally.\n *\n * @param assetId - The CAIP-19 asset ID to unhide.\n */\n unhideAsset(assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Unhiding asset', { assetId: normalizedAssetId });\n\n this.update((state) => {\n const prefs = state.assetPreferences[normalizedAssetId];\n if (prefs) {\n delete prefs.hidden;\n if (Object.keys(prefs).length === 0) {\n delete state.assetPreferences[normalizedAssetId];\n }\n }\n });\n }\n\n // ============================================================================\n // CURRENT CURRENCY MANAGEMENT\n // ============================================================================\n\n /**\n * Set the current currency.\n *\n * @param selectedCurrency - The ISO 4217 currency code to set.\n */\n setSelectedCurrency(selectedCurrency: SupportedCurrency): void {\n const previousCurrency = this.state.selectedCurrency;\n\n if (previousCurrency === selectedCurrency) {\n return;\n }\n\n this.update((state) => {\n state.selectedCurrency = selectedCurrency;\n });\n\n log('Current currency changed', {\n previousCurrency,\n selectedCurrency,\n });\n\n this.getAssets(this.#selectedAccounts, {\n forceUpdate: true,\n dataTypes: ['price'],\n assetsForPriceUpdate: Object.values(this.state.assetsBalance).flatMap(\n (balances) => Object.keys(balances) as Caip19AssetId[],\n ),\n }).catch((error) => {\n log('Failed to fetch asset prices after current currency change', error);\n });\n }\n\n // ============================================================================\n // SUBSCRIPTIONS\n // ============================================================================\n\n /**\n * Subscribe to price updates for all assets held by the given accounts.\n * Polls PriceDataSource which fetches prices from balance state.\n *\n * @param accounts - Accounts to subscribe price updates for.\n * @param chainIds - Chain IDs to filter prices for.\n */\n subscribeAssetsPrice(accounts: InternalAccount[], chainIds: ChainId[]): void {\n if (!this.#isBasicFunctionality()) {\n return;\n }\n const subscriptionKey = 'ds:PriceDataSource';\n\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n const subscribeReq: SubscriptionRequest = {\n request: this.#buildDataRequest(accounts, chainIds, {\n dataTypes: ['price'],\n }),\n subscriptionId: subscriptionKey,\n isUpdate,\n onAssetsUpdate: (response) =>\n this.handleAssetsUpdate(response, 'PriceDataSource'),\n getAssetsState: () => this.state,\n };\n\n this.#priceDataSource.subscribe(subscribeReq).catch(console.error);\n\n // Track subscription\n const subscription: SubscriptionResponse = {\n chains: chainIds,\n accountId: subscriptionKey,\n assetTypes: ['fungible'],\n dataTypes: ['price'],\n unsubscribe: () => {\n this.#activeSubscriptions.delete(subscriptionKey);\n },\n };\n\n this.#activeSubscriptions.set(subscriptionKey, subscription);\n }\n\n /**\n * Unsubscribe from price updates.\n */\n unsubscribeAssetsPrice(): void {\n const subscriptionKey = 'ds:PriceDataSource';\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n\n if (!existingSubscription) {\n return;\n }\n this.#priceDataSource.unsubscribe(subscriptionKey).catch(console.error);\n existingSubscription.unsubscribe();\n }\n\n // ============================================================================\n // STATE MANAGEMENT\n // ============================================================================\n\n /**\n * Resolves native asset IDs (CAIP-19) for the given chains by looking them up\n * in NetworkEnablementController.nativeAssetIdentifiers.\n * Chains without a registered native identifier are skipped.\n *\n * @param chains - The chain IDs to resolve native assets for.\n * @returns Array of native asset IDs for the chains that have a registered identifier.\n */\n #resolveNativeAssetIds(chains: Iterable<ChainId>): Caip19AssetId[] {\n const { nativeAssetIdentifiers } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n const ids: Caip19AssetId[] = [];\n for (const chainId of chains) {\n const nativeId = nativeAssetIdentifiers?.[chainId];\n if (nativeId) {\n ids.push(nativeId as Caip19AssetId);\n }\n }\n return ids;\n }\n\n /**\n * Returns native asset IDs for all enabled chains.\n *\n * @returns Array of native asset IDs, one per enabled chain that has a registered identifier.\n */\n #getNativeAssetIdsForEnabledChains(): Caip19AssetId[] {\n return this.#resolveNativeAssetIds(this.#enabledChains);\n }\n\n /**\n * Returns native asset IDs for the chains that this account supports\n * (account scopes ∩ enabled chains).\n *\n * @param account - The account (scopes determine which chains apply).\n * @returns Array of native asset IDs, one per supported chain that has a registered identifier.\n */\n #getNativeAssetIdsForAccount(account: InternalAccount): Caip19AssetId[] {\n return this.#resolveNativeAssetIds(\n this.#getEnabledChainsForAccount(account),\n );\n }\n\n /**\n * Ensures assetsBalance has a 0 balance for each native token (from\n * NetworkEnablementController.nativeAssetIdentifiers) for each selected account.\n * Only adds natives for chains that the account supports (correct accountId ↔ chain mapping).\n */\n #ensureNativeBalancesDefaultZero(): void {\n const accounts = this.#selectedAccounts;\n if (accounts.length === 0) {\n return;\n }\n this.update((state) => {\n const balances = state.assetsBalance as Record<\n string,\n Record<string, AssetBalance>\n >;\n for (const account of accounts) {\n const accountId = account.id;\n const nativeAssetIds = this.#getNativeAssetIdsForAccount(account);\n if (nativeAssetIds.length === 0) {\n continue;\n }\n if (!balances[accountId]) {\n balances[accountId] = {};\n }\n for (const nativeAssetId of nativeAssetIds) {\n if (!(nativeAssetId in balances[accountId])) {\n balances[accountId][nativeAssetId] = { amount: '0' };\n }\n }\n }\n });\n }\n\n async #updateState(response: DataResponse): Promise<void> {\n const normalizedResponse = normalizeResponse(response);\n const mode: AssetsUpdateMode = normalizedResponse.updateMode ?? 'merge';\n\n const releaseLock = await this.#controllerMutex.acquire();\n\n try {\n const previousState = this.state;\n const previousPrices = { ...this.state.assetsPrice };\n // Use detectedAssets from response (assets without metadata)\n const detectedAssets: Record<AccountId, Caip19AssetId[]> =\n normalizedResponse.detectedAssets ?? {};\n\n // Track actual changes for logging\n const changedBalances: {\n accountId: string;\n assetId: string;\n oldAmount: string | undefined;\n newAmount: string;\n }[] = [];\n const changedMetadata: string[] = [];\n\n this.update((state) => {\n // Use type assertions to avoid deep type instantiation issues with Immer Draft types\n const metadata = state.assetsInfo as Record<string, AssetMetadata>;\n const balances = state.assetsBalance as Record<\n string,\n Record<string, AssetBalance>\n >;\n const prices = state.assetsPrice as Record<string, AssetPrice>;\n\n if (normalizedResponse.assetsInfo) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsInfo,\n )) {\n if (\n !isEqual(previousState.assetsInfo[key as Caip19AssetId], value)\n ) {\n changedMetadata.push(key);\n }\n metadata[key] = value;\n }\n }\n\n if (normalizedResponse.assetsBalance) {\n for (const [accountId, accountBalances] of Object.entries(\n normalizedResponse.assetsBalance,\n )) {\n const previousBalances =\n previousState.assetsBalance[accountId] ?? {};\n const customAssetIds =\n (state.customAssets as Record<string, Caip19AssetId[]>)[\n accountId\n ] ?? [];\n\n // Full: response is authoritative; preserve custom assets not in response.\n // Merge: response overlays previous balances.\n // Callers that fetch partial data (e.g. newly added chains) must set updateMode: 'merge'.\n const effective: Record<string, AssetBalance> =\n mode === 'merge'\n ? { ...previousBalances, ...accountBalances }\n : ((): Record<string, AssetBalance> => {\n const next: Record<string, AssetBalance> = {\n ...accountBalances,\n };\n for (const customId of customAssetIds) {\n if (!(customId in next)) {\n const prev = previousBalances[customId];\n next[customId] =\n prev ?? ({ amount: '0' } as AssetBalance);\n }\n }\n return next;\n })();\n\n // Ensure native tokens have an entry (0 if missing) for chains this account supports\n const account = this.#selectedAccounts.find(\n (a) => a.id === accountId,\n );\n const nativeAssetIdsForAccount = account\n ? this.#getNativeAssetIdsForAccount(account)\n : this.#getNativeAssetIdsForEnabledChains();\n for (const nativeAssetId of nativeAssetIdsForAccount) {\n if (!(nativeAssetId in effective)) {\n effective[nativeAssetId] = { amount: '0' } as AssetBalance;\n }\n }\n\n for (const [assetId, balance] of Object.entries(effective)) {\n const previousBalance = previousBalances[\n assetId as Caip19AssetId\n ] as { amount: string } | undefined;\n const newAmount = (balance as { amount: string }).amount;\n const oldAmount = previousBalance?.amount;\n const isNewDefaultNativeZero =\n oldAmount === undefined &&\n newAmount === '0' &&\n nativeAssetIdsForAccount.includes(assetId as Caip19AssetId);\n if (oldAmount !== newAmount && !isNewDefaultNativeZero) {\n changedBalances.push({\n accountId,\n assetId,\n oldAmount,\n newAmount,\n });\n }\n }\n balances[accountId] = effective;\n }\n }\n\n // Update prices in state\n if (normalizedResponse.assetsPrice) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsPrice,\n )) {\n prices[key] = value;\n }\n }\n });\n\n // Calculate changed prices\n const changedPriceAssets: string[] = normalizedResponse.assetsPrice\n ? Object.keys(normalizedResponse.assetsPrice).filter(\n (assetId) =>\n !isEqual(\n previousPrices[assetId as Caip19AssetId],\n normalizedResponse.assetsPrice?.[assetId as Caip19AssetId],\n ),\n )\n : [];\n\n // Log only actual changes\n if (\n changedBalances.length > 0 ||\n changedMetadata.length > 0 ||\n changedPriceAssets.length > 0\n ) {\n log('State updated', {\n changedBalances:\n changedBalances.length > 0 ? changedBalances : undefined,\n changedMetadataCount:\n changedMetadata.length > 0 ? changedMetadata.length : undefined,\n changedPricesCount:\n changedPriceAssets.length > 0\n ? changedPriceAssets.length\n : undefined,\n newAssets:\n Object.keys(detectedAssets).length > 0\n ? Object.entries(detectedAssets).map(([accountId, assets]) => ({\n accountId,\n assets,\n }))\n : undefined,\n });\n }\n\n // Publish balance changed events\n for (const change of changedBalances) {\n this.messenger.publish('AssetsController:balanceChanged', {\n accountId: change.accountId,\n assetId: change.assetId as Caip19AssetId,\n previousAmount: change.oldAmount ?? '0',\n newAmount: change.newAmount,\n });\n }\n\n // Publish price changed event with full price data\n if (changedPriceAssets.length > 0 && normalizedResponse.assetsPrice) {\n // Build prices object with only changed prices\n const changedPrices: Record<Caip19AssetId, AssetPrice> = {};\n for (const assetId of changedPriceAssets) {\n const price =\n normalizedResponse.assetsPrice[assetId as Caip19AssetId];\n if (price) {\n changedPrices[assetId as Caip19AssetId] = price;\n }\n }\n this.messenger.publish('AssetsController:priceChanged', {\n prices: changedPrices,\n });\n }\n\n // Publish assets detected events\n for (const [accountId, assetIds] of Object.entries(detectedAssets)) {\n if (assetIds.length > 0) {\n this.messenger.publish('AssetsController:assetsDetected', {\n accountId,\n assetIds,\n });\n }\n }\n } finally {\n releaseLock();\n }\n }\n\n #getAssetsFromState(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n assetTypes: AssetType[],\n ): Record<AccountId, Record<Caip19AssetId, Asset>> {\n const result: Record<AccountId, Record<Caip19AssetId, Asset>> = {};\n // Convert to Sets for O(1) lookups\n const chainIdSet = new Set(chainIds);\n const assetTypeSet = new Set(assetTypes);\n\n for (const account of accounts) {\n result[account.id] = {};\n\n const accountBalances = this.state.assetsBalance[account.id] ?? {};\n\n for (const [assetId, balance] of Object.entries(accountBalances)) {\n const typedAssetId = assetId as Caip19AssetId;\n\n const metadataRaw = this.state.assetsInfo[typedAssetId];\n\n // Skip assets without metadata\n if (!metadataRaw) {\n continue;\n }\n\n const metadata = metadataRaw;\n\n // Skip hidden assets (assetPreferences)\n const prefs = this.state.assetPreferences[typedAssetId];\n if (prefs?.hidden) {\n continue;\n }\n\n const assetChainId = extractChainId(typedAssetId);\n\n if (!chainIdSet.has(assetChainId)) {\n continue;\n }\n\n // Filter by asset type\n const tokenAssetType = this.#tokenStandardToAssetType(metadata.type);\n if (!assetTypeSet.has(tokenAssetType)) {\n continue;\n }\n\n const typedBalance = balance;\n const priceRaw = this.state.assetsPrice[typedAssetId];\n const price: AssetPrice = priceRaw ?? {\n price: 0,\n lastUpdated: 0,\n };\n\n // Compute fiat value using BigNumber for precision\n // Note: typedBalance.amount is already in human-readable format (e.g., \"1\" for 1 ETH)\n // so we do NOT divide by 10^decimals here\n const balanceAmount = new BigNumberJS(typedBalance.amount || '0');\n const fiatValue = balanceAmount\n .multipliedBy(price.price || 0)\n .toNumber();\n\n const asset: Asset = {\n id: typedAssetId,\n chainId: assetChainId,\n balance: typedBalance,\n metadata,\n price,\n fiatValue,\n };\n\n result[account.id][typedAssetId] = asset;\n }\n }\n\n return result;\n }\n\n /**\n * Maps a token standard to its corresponding asset type.\n *\n * @param tokenStandard - The token standard from metadata.\n * @returns The corresponding asset type.\n */\n #tokenStandardToAssetType(tokenStandard: string): AssetType {\n switch (tokenStandard) {\n case 'native':\n case 'erc20':\n case 'spl':\n return 'fungible';\n case 'erc721':\n return 'nft';\n case 'erc1155':\n // ERC1155 can be either fungible or non-fungible, treat as collectible\n return 'collectible';\n default:\n // Unknown standards default to fungible\n return 'fungible';\n }\n }\n\n // ============================================================================\n // START / STOP\n // ============================================================================\n\n /**\n * Start asset tracking: subscribe to updates and fetch current balances.\n * Called when app opens, account changes, or keyring unlocks.\n */\n #start(): void {\n log('Starting asset tracking', {\n selectedAccountCount: this.#selectedAccounts.length,\n enabledChainCount: this.#enabledChains.size,\n });\n\n this.#subscribeAssets();\n this.#ensureNativeBalancesDefaultZero();\n this.getAssets(this.#selectedAccounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch assets', error);\n });\n }\n\n /**\n * Stop asset tracking: unsubscribe from all updates.\n * Called when app closes or keyring locks.\n */\n #stop(): void {\n log('Stopping asset tracking', {\n activeSubscriptionCount: this.#activeSubscriptions.size,\n hasPriceSubscription: this.#activeSubscriptions.has('ds:PriceDataSource'),\n });\n\n this.#firstInitFetchReported = false;\n\n // Stop price subscription first (uses direct messenger call)\n this.unsubscribeAssetsPrice();\n\n // Stop balance subscriptions by properly notifying data sources via messenger\n // This ensures data sources stop their polling timers.\n // Use #allBalanceDataSources + staked balance source so we unsubscribe from\n // every source that may have been subscribed.\n const allSources = [\n ...this.#allBalanceDataSources,\n this.#stakedBalanceDataSource,\n ];\n const subscriptionKeys = [...this.#activeSubscriptions.keys()];\n for (const subscriptionKey of subscriptionKeys) {\n if (subscriptionKey.startsWith('ds:')) {\n const sourceId = subscriptionKey.slice(3);\n const source = allSources.find((ds) => ds.getName() === sourceId);\n if (source) {\n this.#unsubscribeDataSource(source);\n }\n }\n }\n this.#activeSubscriptions.clear();\n }\n\n /**\n * Handle basic functionality toggle change. Call this from the consumer (extension or mobile)\n * when the user changes the \"Basic functionality\" setting. Refreshes subscriptions so the\n * current {@link AssetsControllerOptions.isBasicFunctionality} getter is used (true = APIs on,\n * false = RPC only).\n *\n * @param _isBasic - The new value (for call-site clarity; the getter is the source of truth).\n */\n handleBasicFunctionalityChange(_isBasic: boolean): void {\n this.#stop();\n this.#subscribeAssets();\n }\n\n /**\n * Subscribe to asset updates for all selected accounts.\n */\n #subscribeAssets(): void {\n if (this.#selectedAccounts.length === 0 || this.#enabledChains.size === 0) {\n return;\n }\n\n // Subscribe to balance updates (batched by data source)\n this.#subscribeAssetsBalance(this.#selectedAccounts, [\n ...this.#enabledChains,\n ]);\n\n // Subscribe to staked balance updates (separate from regular balance chain-claiming)\n this.#subscribeStakedBalance(this.#selectedAccounts, [\n ...this.#enabledChains,\n ]);\n\n // Subscribe to price updates for all assets held by selected accounts\n this.subscribeAssetsPrice(this.#selectedAccounts, [...this.#enabledChains]);\n }\n\n /**\n * Subscribe to balance updates for the given accounts and chains.\n *\n * Strategy to minimize data source calls:\n * 1. Collect all chains to subscribe based on enabled networks\n * 2. Map chains to accounts based on their scopes\n * 3. Split by data source (ordered by priority) - each data source gets ONE subscription\n *\n * This ensures we make minimal subscriptions to each data source while covering\n * all accounts and chains.\n *\n * @param accounts - Accounts to subscribe balance updates for.\n * @param chainIds - Chain IDs to subscribe for.\n */\n #subscribeAssetsBalance(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n ): void {\n const chainToAccounts = this.#buildChainToAccountsMap(\n accounts,\n new Set(chainIds),\n );\n const remainingChains = new Set(chainToAccounts.keys());\n\n // When basic functionality is on (getter true), use all balance data sources; when off (getter false), RPC only.\n const balanceDataSources = this.#isBasicFunctionality()\n ? this.#allBalanceDataSources\n : [this.#rpcDataSource];\n\n for (const source of balanceDataSources) {\n const availableChains = new Set(source.getActiveChainsSync());\n const assignedChains: ChainId[] = [];\n\n for (const chainId of remainingChains) {\n if (availableChains.has(chainId)) {\n assignedChains.push(chainId);\n remainingChains.delete(chainId);\n }\n }\n\n if (assignedChains.length === 0) {\n this.#unsubscribeDataSource(source);\n continue;\n }\n\n const seenIds = new Set<string>();\n const accountsForSource = assignedChains\n .flatMap((chainId) => chainToAccounts.get(chainId) ?? [])\n .filter(\n (account) =>\n !seenIds.has(account.id) && (seenIds.add(account.id), true),\n );\n if (accountsForSource.length > 0) {\n this.#subscribeDataSource(source, accountsForSource, assignedChains);\n }\n }\n }\n\n /**\n * Subscribe to staked balance updates.\n * Unlike regular balance data sources, the staked balance data source provides\n * supplementary data and does not participate in chain-claiming.\n *\n * @param accounts - Accounts to subscribe staked balance updates for.\n * @param chainIds - Chain IDs to subscribe for.\n */\n #subscribeStakedBalance(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n ): void {\n const source = this.#stakedBalanceDataSource;\n if (!source) {\n return;\n }\n const availableChains = new Set(source.getActiveChainsSync());\n const assignedChains = chainIds.filter((chainId) =>\n availableChains.has(chainId),\n );\n\n if (assignedChains.length === 0) {\n this.#unsubscribeDataSource(source);\n return;\n }\n\n this.#subscribeDataSource(source, accounts, assignedChains);\n }\n\n /**\n * Build a mapping of chainId -> accounts that support that chain.\n * Only includes chains that are in the chainsToSubscribe set.\n *\n * @param accounts - Array of accounts to build mapping for.\n * @param chainsToSubscribe - Set of chain IDs to include in the mapping.\n * @returns Map of chainId to array of accounts that support that chain.\n */\n #buildChainToAccountsMap(\n accounts: InternalAccount[],\n chainsToSubscribe: Set<ChainId>,\n ): Map<ChainId, InternalAccount[]> {\n const chainToAccounts = new Map<ChainId, InternalAccount[]>();\n for (const account of accounts) {\n for (const chainId of this.#getEnabledChainsForAccount(account)) {\n if (!chainsToSubscribe.has(chainId)) {\n continue;\n }\n let list = chainToAccounts.get(chainId);\n if (!list) {\n list = [];\n chainToAccounts.set(chainId, list);\n }\n list.push(account);\n }\n }\n return chainToAccounts;\n }\n\n /**\n * Subscribe to a specific data source with accounts and chains.\n * Uses the data source name as the subscription key for batching.\n *\n * @param source - The balance data source instance.\n * @param accounts - Array of accounts to subscribe for.\n * @param chains - Array of chain IDs to subscribe for.\n */\n #subscribeDataSource(\n source: AbstractDataSource<string, DataSourceState>,\n accounts: InternalAccount[],\n chains: ChainId[],\n ): void {\n const sourceId = source.getName();\n const subscriptionKey = `ds:${sourceId}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n log('Subscribe to data source', {\n sourceId,\n subscriptionKey,\n isUpdate,\n accountCount: accounts.length,\n chainCount: chains.length,\n });\n\n const subscribeReq: SubscriptionRequest = {\n request: this.#buildDataRequest(accounts, chains, {\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n updateInterval: this.#defaultUpdateInterval,\n }),\n subscriptionId: subscriptionKey,\n isUpdate,\n onAssetsUpdate: (response, request) =>\n this.handleAssetsUpdate(response, sourceId, request),\n getAssetsState: () => this.state,\n };\n\n source.subscribe(subscribeReq).catch((error) => {\n console.error(\n `[AssetsController] Failed to subscribe to '${sourceId}':`,\n error,\n );\n });\n\n // Track subscription\n const subscription: SubscriptionResponse = {\n chains,\n accountId: subscriptionKey,\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n unsubscribe: () => {\n this.#activeSubscriptions.delete(subscriptionKey);\n },\n };\n\n this.#activeSubscriptions.set(subscriptionKey, subscription);\n }\n\n /**\n * Unsubscribe from a data source if we have an active subscription.\n *\n * @param source - The balance data source instance to unsubscribe from.\n */\n #unsubscribeDataSource(\n source: AbstractDataSource<string, DataSourceState>,\n ): void {\n const subscriptionKey = `ds:${source.getName()}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n\n if (existingSubscription) {\n source.unsubscribe(subscriptionKey).catch(() => undefined);\n existingSubscription.unsubscribe();\n }\n }\n\n // ============================================================================\n // HELPERS\n // ============================================================================\n\n /**\n * Build a DataRequest with accountsWithSupportedChains (enabled chains ∩ account scope ∩ requested chainIds).\n *\n * @param accounts - Accounts to include.\n * @param chainIds - Requested chain IDs (e.g. enabled chains or subset).\n * @param partial - Rest of the request (dataTypes, assetTypes, etc.).\n * @returns DataRequest with accountsWithSupportedChains and chainIds.\n */\n #buildDataRequest(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n partial: Partial<\n Omit<DataRequest, 'accountsWithSupportedChains' | 'chainIds'>\n > = {},\n ): DataRequest {\n const chainIdSet = new Set(chainIds);\n const accountsWithSupportedChains: AccountWithSupportedChains[] =\n accounts.map((account) => ({\n account,\n supportedChains: this.#getEnabledChainsForAccount(account).filter(\n (chain) => chainIdSet.has(chain),\n ),\n }));\n return {\n accountsWithSupportedChains,\n chainIds,\n ...partial,\n } as DataRequest;\n }\n\n /**\n * Get the chains that an account supports based on its scopes.\n * Returns the intersection of the account's scopes and the enabled chains.\n *\n * @param account - The account to get supported chains for.\n * @returns Array of ChainIds that the account supports and are enabled.\n */\n #getEnabledChainsForAccount(account: InternalAccount): ChainId[] {\n // Account scopes are CAIP-2 chain IDs like \"eip155:1\", \"solana:mainnet\", \"bip122:...\"\n const scopes = account.scopes ?? [];\n const result: ChainId[] = [];\n\n for (const scope of scopes) {\n const scopeStr = scope as string;\n if (!isCaipChainId(scopeStr)) {\n result.push(scope);\n continue;\n }\n const { namespace, reference } = parseCaipChainId(scopeStr);\n\n // Wildcard scope (e.g., \"eip155:0\" means all enabled chains in that namespace)\n if (reference === '0') {\n for (const chain of this.#enabledChains) {\n if (isCaipChainId(chain)) {\n const chainParsed = parseCaipChainId(chain);\n if (chainParsed.namespace === namespace) {\n result.push(chain);\n }\n }\n }\n } else if (namespace === 'eip155' && isStrictHexString(reference)) {\n // Normalize hex to decimal for EIP155\n result.push(`eip155:${parseInt(reference, 16)}` as ChainId);\n } else {\n result.push(scope);\n }\n }\n\n return result;\n }\n\n // ============================================================================\n // EVENT HANDLERS\n // ============================================================================\n\n async #handleAccountGroupChanged(): Promise<void> {\n const accounts = this.#selectedAccounts;\n\n log('Account group changed', {\n accountCount: accounts.length,\n accountIds: accounts.map((a) => a.id),\n });\n\n // Subscribe and fetch for the new account group\n this.#subscribeAssets();\n if (accounts.length > 0) {\n await this.getAssets(accounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n });\n }\n\n this.#ensureNativeBalancesDefaultZero();\n }\n\n async #handleEnabledNetworksChanged(\n enabledNetworkMap: NetworkEnablementControllerState['enabledNetworkMap'],\n ): Promise<void> {\n const previousChains = this.#enabledChains;\n this.#enabledChains = this.#extractEnabledChains(enabledNetworkMap);\n\n // Find newly enabled chains (in new set but not in previous)\n const addedChains: ChainId[] = [];\n for (const chain of this.#enabledChains) {\n if (!previousChains.has(chain)) {\n addedChains.push(chain);\n }\n }\n\n // Find disabled chains to clean up (in previous but not in new)\n const removedChains: ChainId[] = [];\n for (const chain of previousChains) {\n if (!this.#enabledChains.has(chain)) {\n removedChains.push(chain);\n }\n }\n\n log('Enabled networks changed', {\n previousCount: previousChains.size,\n newCount: this.#enabledChains.size,\n addedChains,\n removedChains,\n });\n\n // Note: We intentionally do NOT delete balance data for disabled chains.\n // Users may want to see historical balances even if the network is currently disabled.\n // The data will simply not be updated until the network is re-enabled.\n\n // Refresh subscriptions for new chain set\n this.#subscribeAssets();\n\n // Do one-time fetch for newly enabled chains; merge so we keep existing chain balances\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n await this.getAssets(this.#selectedAccounts, {\n chainIds: addedChains,\n forceUpdate: true,\n updateMode: 'merge',\n });\n }\n\n this.#ensureNativeBalancesDefaultZero();\n }\n\n /**\n * Handle assets updated from a data source.\n * Called via the onAssetsUpdate callback passed in SubscriptionRequest when the controller subscribes to a data source.\n * Enriches the response with token metadata (via middlewares) before updating state.\n *\n * @param response - The data response with updated assets\n * @param sourceId - The data source ID reporting the update\n * @param request - Optional original request for context when enriching\n */\n async handleAssetsUpdate(\n response: DataResponse,\n sourceId: string,\n request?: DataRequest,\n ): Promise<void> {\n log('Assets updated from data source', {\n sourceId,\n hasBalance: Boolean(response.assetsBalance),\n hasPrice: Boolean(response.assetsPrice),\n });\n\n // Run through enrichment middlewares (Detection, then Token + Price in parallel)\n // Include 'metadata' in dataTypes so TokenDataSource runs to enrich detected assets\n const { response: enrichedResponse } = await this.#executeMiddlewares(\n [\n this.#detectionMiddleware,\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ],\n request ?? {\n accountsWithSupportedChains: [],\n chainIds: [],\n dataTypes: ['balance', 'metadata', 'price'],\n },\n response,\n );\n\n await this.#updateState(enrichedResponse);\n }\n\n // ============================================================================\n // CLEANUP\n // ============================================================================\n\n destroy(): void {\n log('Destroying AssetsController', {\n dataSourceCount: this.#allBalanceDataSources.length,\n subscriptionCount: this.#activeSubscriptions.size,\n });\n\n // Destroy instantiated data sources\n this.#backendWebsocketDataSource?.destroy?.();\n this.#accountsApiDataSource?.destroy?.();\n this.#snapDataSource?.destroy?.();\n this.#rpcDataSource?.destroy?.();\n this.#stakedBalanceDataSource?.destroy?.();\n\n // Stop all active subscriptions\n this.#stop();\n\n if (this.#unsubscribeBasicFunctionality) {\n this.#unsubscribeBasicFunctionality();\n this.#unsubscribeBasicFunctionality = null;\n }\n\n // Unregister action handlers\n this.messenger.unregisterActionHandler('AssetsController:getAssets');\n this.messenger.unregisterActionHandler('AssetsController:getAssetsBalance');\n this.messenger.unregisterActionHandler('AssetsController:getAssetMetadata');\n this.messenger.unregisterActionHandler('AssetsController:getAssetsPrice');\n this.messenger.unregisterActionHandler(\n 'AssetsController:getExchangeRatesForBridge',\n );\n this.messenger.unregisterActionHandler(\n 'AssetsController:getStateForTransactionPay',\n );\n this.messenger.unregisterActionHandler('AssetsController:addCustomAsset');\n this.messenger.unregisterActionHandler(\n 'AssetsController:removeCustomAsset',\n );\n this.messenger.unregisterActionHandler('AssetsController:getCustomAssets');\n this.messenger.unregisterActionHandler('AssetsController:hideAsset');\n this.messenger.unregisterActionHandler('AssetsController:unhideAsset');\n }\n}\n"]}