@metamask-previews/assets-controller 6.2.1-preview-e0eba6dbb → 6.2.1-preview-56dd1249f

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.
@@ -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 AccountTreeControllerStateChangeEvent,\n} from '@metamask/account-tree-controller';\nimport type { AccountsControllerGetSelectedAccountAction } from '@metamask/accounts-controller';\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 { CHAIN_IDS_WITH_NO_NATIVE_TOKEN } from '@metamask/controller-utils';\nimport type { TraceCallback } from '@metamask/controller-utils';\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 { PhishingControllerBulkScanTokensAction } from '@metamask/phishing-controller';\nimport type { PreferencesControllerStateChangeEvent } from '@metamask/preferences-controller';\nimport type {\n SnapControllerGetRunnableSnapsAction,\n SnapControllerHandleRequestAction,\n} from '@metamask/snaps-controllers';\nimport type {\n TransactionControllerIncomingTransactionsReceivedEvent,\n TransactionControllerTransactionConfirmedEvent,\n TransactionControllerUnapprovedTransactionAddedEvent,\n TransactionMeta,\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 { AssetsDataSourceError } from './errors';\nimport { projectLogger, createModuleLogger } from './logger';\nimport { CustomAssetGraduationMiddleware } from './middlewares/CustomAssetGraduationMiddleware';\nimport { DetectionMiddleware } from './middlewares/DetectionMiddleware';\nimport {\n createParallelBalanceMiddleware,\n createParallelMiddleware,\n} from './middlewares/ParallelMiddleware';\nimport { RpcFallbackMiddleware } from './middlewares/RpcFallbackMiddleware';\nimport type {\n AccountId,\n AssetPreferences,\n AssetsControllerStateInternal,\n AssetsDataSource,\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} from './types';\nimport {\n normalizeAssetId,\n formatExchangeRatesForBridge,\n formatStateForTransactionPay,\n buildNativeAssetsFromConstant,\n buildNativeAssetsFromApi,\n} from './utils';\nimport type {\n BridgeExchangeRatesFormat,\n TransactionPayLegacyFormat,\n} from './utils';\nimport { ZERO_ADDRESS } from './utils/constants';\n\nconst NATIVE_ASSETS_QUERY_KEY = ['nativeAssets'];\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\n// ============================================================================\n// TRACE NAMES — used in Sentry spans (search these strings in Discover)\n// ============================================================================\nconst TRACE_FIRST_INIT_FETCH = 'AssetsControllerFirstInitFetch';\nconst TRACE_FULL_FETCH = 'AssetsFullFetch';\nconst TRACE_DATA_SOURCE_TIMING = 'AssetsDataSourceTiming';\nconst TRACE_DATA_SOURCE_ERROR = 'AssetsDataSourceError';\nconst TRACE_UPDATE_PIPELINE = 'AssetsUpdatePipeline';\nconst TRACE_SUBSCRIPTION_ERROR = 'AssetsSubscriptionError';\nconst TRACE_STATE_SIZE = 'AssetsStateSize';\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 | AccountsControllerGetSelectedAccountAction\n | AccountTreeControllerGetAccountsFromSelectedAccountGroupAction\n // RpcDataSource\n | NetworkControllerGetStateAction\n | NetworkControllerGetNetworkClientByIdAction\n // StakedBalanceDataSource\n | NetworkEnablementControllerGetStateAction\n // SnapDataSource\n | SnapControllerGetRunnableSnapsAction\n | SnapControllerHandleRequestAction\n | GetPermissions\n // BackendWebsocketDataSource\n | BackendWebSocketServiceActions\n // PhishingController\n | PhishingControllerBulkScanTokensAction;\n\ntype AllowedEvents =\n // AssetsController\n | AccountTreeControllerSelectedAccountGroupChangeEvent\n | AccountTreeControllerStateChangeEvent\n | ClientControllerStateChangeEvent\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent\n | PreferencesControllerStateChangeEvent\n | TransactionControllerUnapprovedTransactionAddedEvent\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\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 trace callback. When provided, the controller invokes it with a\n * trace request when the first init/fetch completes (e.g. after unlock).\n * Use this to report first init fetch duration to Sentry (e.g. via addBreadcrumb or setMeasurement).\n */\n trace?: TraceCallback;\n /**\n * Optional Sentry (or compatible) reporter for **issue** events. When data source middlewares\n * fail, the controller constructs an `AssetsDataSourceError` and passes it here so you can call\n * `Sentry.captureException`. Span-only trace callbacks do not create Issues;\n * wire this if you need Sentry alerts on middleware failures.\n */\n captureException?: (error: Error) => 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 * Function returning whether onboarding is complete. When false,\n * RPC and staked balance data sources skip fetch and subscribe\n * (no on-chain calls until the user has finished onboarding).\n * Defaults to () => true.\n */\n isOnboarded?: () => boolean;\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 trace callback for first init/fetch measurement (duration). */\n readonly #trace?: TraceCallback;\n\n /** Optional reporter for Issue-style errors (e.g. Sentry.captureException). */\n readonly #captureException?: (error: Error) => void;\n\n /** Whether we have already reported first init fetch for this session (reset on #stop). */\n #firstInitFetchReported = false;\n\n /** Whether we have already reported state size for this session (reset on #stop). */\n #stateSizeReported = false;\n\n /**\n * Fire-and-forget trace helper. Swallows errors so telemetry never breaks the controller.\n *\n * @param name - Trace / span name visible in Sentry.\n * @param data - Key-value pairs attached as span data.\n * @param tags - Key-value pairs used for Sentry filtering.\n */\n #emitTrace(\n name: string,\n data: Record<string, number | string | boolean>,\n tags: Record<string, number | string | boolean> = {\n controller: 'AssetsController',\n },\n ): void {\n if (!this.#trace) {\n return;\n }\n this.#trace({ name, data, tags }, () => undefined)?.catch(() => {\n // Telemetry failure must not break.\n });\n }\n\n /**\n * Emit a state-size trace once on app start (first state update after unlock).\n */\n #emitStateSizeTrace(): void {\n if (!this.#trace || this.#stateSizeReported) {\n return;\n }\n this.#stateSizeReported = true;\n\n const {\n assetsBalance: balances,\n customAssets,\n assetsInfo,\n assetsPrice,\n } = this.state;\n\n // Count balance entries and collect unique asset IDs / chain IDs in one pass.\n let balanceEntries = 0;\n const uniqueAssets = new Set<string>();\n const uniqueNetworks = new Set<string>();\n\n for (const acct of Object.values(balances)) {\n const assetIds = Object.keys(acct);\n balanceEntries += assetIds.length;\n for (const assetId of assetIds) {\n uniqueAssets.add(assetId);\n // CAIP-19 format: \"eip155:1/slip44:60\" — chainId is everything before \"/\"\n const slash = assetId.indexOf('/');\n if (slash > 0) {\n uniqueNetworks.add(assetId.slice(0, slash));\n }\n }\n }\n\n let customAssetEntries = 0;\n for (const ids of Object.values(customAssets)) {\n if (Array.isArray(ids)) {\n customAssetEntries += ids.length;\n }\n }\n\n this.#emitTrace(TRACE_STATE_SIZE, {\n balance_entries: balanceEntries,\n balance_accounts: Object.keys(balances).length,\n unique_asset_count: uniqueAssets.size,\n network_count: uniqueNetworks.size,\n metadata_entries: Object.keys(assetsInfo).length,\n price_entries: Object.keys(assetsPrice).length,\n custom_asset_entries: customAssetEntries,\n });\n }\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 * Snapshot of account IDs that were active when the last subscription/fetch\n * cycle ran. Used by #handleAccountTreeStateChange to skip redundant\n * re-subscriptions when the account set hasn't actually changed.\n */\n #lastKnownAccountIds: ReadonlySet<string> = 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 #getSelectedAccounts(): InternalAccount[] {\n const accounts = this.messenger.call(\n 'AccountTreeController:getAccountsFromSelectedAccountGroup',\n );\n if (accounts.length > 0) {\n return accounts;\n }\n const selectedAccount = this.messenger.call(\n 'AccountsController:getSelectedAccount',\n );\n if (selectedAccount) {\n return [selectedAccount];\n }\n return [];\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 in priority order for chain-claiming and cleanup.\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 #customAssetGraduationMiddleware: CustomAssetGraduationMiddleware;\n\n readonly #rpcFallbackMiddleware: RpcFallbackMiddleware;\n\n readonly #tokenDataSource: TokenDataSource;\n\n #unsubscribeBasicFunctionality: (() => void) | null = null;\n\n readonly #queryApiClient: ApiPlatformClient;\n\n readonly #onActiveChainsUpdated: (\n dataSourceName: string,\n chains: 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 trace,\n captureException,\n accountsApiDataSourceConfig,\n priceDataSourceConfig,\n stakedBalanceDataSourceConfig,\n isOnboarded,\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.#trace = trace;\n this.#captureException = captureException;\n this.#queryApiClient = queryApiClient;\n const rpcConfig = rpcDataSourceConfig ?? {};\n\n this.#initializeNativeAssetsMap(queryApiClient);\n\n this.#onActiveChainsUpdated = (\n dataSourceName: string,\n chains: ChainId[],\n previousChains: ChainId[],\n ): void => {\n try {\n this.#handleActiveChainsUpdate(dataSourceName, chains, previousChains);\n } catch (error) {\n log('Failed to handle active chains update', {\n dataSourceName,\n error,\n });\n }\n };\n\n this.#backendWebsocketDataSource = new BackendWebsocketDataSource({\n messenger: this.messenger,\n queryApiClient,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n isNativeAsset: (assetId: Caip19AssetId): boolean =>\n this.#isNativeAsset(assetId),\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 getNativeAssetForChain: (chainId: ChainId): Caip19AssetId =>\n this.#getNativeAssetMap()[chainId] ??\n `${chainId}/erc20:${ZERO_ADDRESS}`,\n ...rpcConfig,\n isOnboarded: rpcConfig.isOnboarded ?? isOnboarded,\n });\n this.#stakedBalanceDataSource = new StakedBalanceDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...stakedBalanceDataSourceConfig,\n });\n this.#tokenDataSource = new TokenDataSource(this.messenger, {\n queryApiClient,\n getNativeAssetIds: (): Caip19AssetId[] => {\n return this.#getNativeAssetIdsForEnabledChains();\n },\n });\n this.#priceDataSource = new PriceDataSource({\n queryApiClient,\n getSelectedCurrency: (): SupportedCurrency => this.state.selectedCurrency,\n ...priceDataSourceConfig,\n });\n this.#detectionMiddleware = new DetectionMiddleware();\n this.#customAssetGraduationMiddleware = new CustomAssetGraduationMiddleware(\n {\n getSelectedAccountId: (): AccountId | undefined => {\n try {\n return this.#getSelectedAccounts()[0]?.id;\n } catch {\n return undefined;\n }\n },\n removeCustomAsset: (accountId, assetId): void =>\n this.removeCustomAsset(accountId, assetId),\n },\n );\n this.#rpcFallbackMiddleware = new RpcFallbackMiddleware({\n rpcDataSource: this.#rpcDataSource,\n });\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 #initializeNativeAssetsMap(queryApiClient: ApiPlatformClient): void {\n queryApiClient.queryClient\n .fetchQuery({\n queryKey: NATIVE_ASSETS_QUERY_KEY,\n queryFn: buildNativeAssetsFromApi,\n staleTime: Infinity,\n gcTime: Infinity,\n })\n .catch((error) => {\n // Failure to populate native asset cache is non-fatal;\n // #isNativeAsset falls back to the seed data from buildNativeAssetsFromConstant.\n log(\n 'Failed to populate native asset cache, falling back to seed data',\n error,\n );\n });\n\n // Seed the cache synchronously so that synchronous consumers (e.g.\n // #isNativeAsset, #resolveNativeAssetIds) have data available immediately.\n // When fetchQuery resolves (currently instant, future: API), it overwrites\n // this with potentially fresher data.\n const initialNativeAssetMap = buildNativeAssetsFromConstant();\n queryApiClient.setCachedData(\n NATIVE_ASSETS_QUERY_KEY,\n initialNativeAssetMap,\n );\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 // Catch the initial tree build. On returning users,\n // `selectedAccountGroupChange` does NOT fire when the persisted group\n // is unchanged, and `accountTreeChange` doesn't fire either (init()\n // rebuilds from persisted accounts without publishing it).\n // The base-controller `:stateChange` event is guaranteed to fire\n // when init() calls this.update(). #start() is idempotent so\n // repeated fires are safe.\n this.messenger.subscribe('AccountTreeController:stateChange', () => {\n this.#handleAccountTreeStateChange();\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 // Subscribe to unapproved transactions - TXs that need confirmation\n // Ensures that balances for the account making transaction are updated (e.g. for gas estimations)\n this.messenger.subscribe(\n 'TransactionController:unapprovedTransactionAdded',\n (transactionMeta: TransactionMeta) => {\n this.#onUnapprovedTransactionAdded(transactionMeta);\n },\n );\n }\n\n #onUnapprovedTransactionAdded(transactionMeta: TransactionMeta): void {\n const hexChainId = transactionMeta.chainId;\n if (!hexChainId) {\n return;\n }\n\n const caipChainId = `eip155:${parseInt(hexChainId, 16)}` as ChainId;\n const fromAddress = transactionMeta.txParams.from?.toLowerCase();\n if (!fromAddress) {\n return;\n }\n\n const matchedAccount = this.#getSelectedAccounts().find(\n (account) => account.address.toLowerCase() === fromAddress,\n );\n if (!matchedAccount) {\n return;\n }\n\n this.getAssets([matchedAccount], {\n chainIds: [caipChainId],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to refresh assets after unapproved transaction added', {\n error,\n });\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 /**\n * Handle AccountTreeController state changes.\n * If already running, re-subscribe only when the set of selected accounts\n * has actually changed (e.g. a snap account was added after initial startup).\n * This guards against the many tree mutations that don't affect which\n * accounts are selected — without this check every tree update would\n * trigger a redundant full re-subscribe + forceUpdate fetch.\n * If not running yet, delegate to #start() for the normal start flow.\n */\n #handleAccountTreeStateChange(): void {\n const shouldRun = this.#uiOpen && this.#keyringUnlocked;\n if (!shouldRun) {\n return;\n }\n if (this.#activeSubscriptions.size > 0) {\n const accounts = this.#getSelectedAccounts();\n const currentIds = new Set(accounts.map((a) => a.id));\n\n const accountsChanged =\n currentIds.size !== this.#lastKnownAccountIds.size ||\n [...currentIds].some((id) => !this.#lastKnownAccountIds.has(id));\n\n if (!accountsChanged) {\n return;\n }\n\n log('Account tree changed with new accounts, re-subscribing', {\n previousCount: this.#lastKnownAccountIds.size,\n currentCount: currentIds.size,\n });\n\n this.#lastKnownAccountIds = currentIds;\n this.#subscribeAssets();\n this.#ensureNativeBalancesDefaultZero();\n this.getAssets(accounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch assets after tree change', error);\n });\n } else {\n this.#start();\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.#getSelectedAccounts().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.#getSelectedAccounts(), {\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: AssetsDataSource[],\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 = performance.now();\n try {\n return await middleware(ctx, next);\n } finally {\n inclusive[i] = performance.now() - start;\n }\n }) as Middleware,\n );\n\n const middlewareErrors: string[] = [];\n const chain = wrapped.reduceRight<NextFunction>(\n (next, middleware, index) =>\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 const sourceName = names[index] ?? `middleware_${index}`;\n middlewareErrors.push(sourceName);\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 if (result.durationByDataSource) {\n for (const [key, ms] of Object.entries(result.durationByDataSource)) {\n durationByDataSource[key] = ms;\n }\n }\n\n // Emit per-source timing traces for the Assets Health dashboard\n for (const [sourceName, durationMs] of Object.entries(\n durationByDataSource,\n )) {\n this.#emitTrace(TRACE_DATA_SOURCE_TIMING, {\n source: sourceName,\n duration_ms: durationMs,\n chain_count: request.chainIds.length,\n account_count: request.accountsWithSupportedChains.length,\n });\n }\n\n // Failed middlewares: Issues (optional) + perf/Dashboard spans\n if (middlewareErrors.length > 0) {\n const failedSources = middlewareErrors.join(',');\n const assetsError = new AssetsDataSourceError({\n failedSources,\n errorCount: middlewareErrors.length,\n chainCount: request.chainIds.length,\n });\n try {\n this.#captureException?.(assetsError);\n } catch {\n // Never let telemetry throw.\n }\n this.#emitTrace(\n TRACE_DATA_SOURCE_ERROR,\n {\n failed_sources: failedSources,\n error_count: middlewareErrors.length,\n chain_count: request.chainIds.length,\n },\n {\n controller: 'AssetsController',\n severity: 'error',\n error_type: assetsError.name,\n },\n );\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 = performance.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 // Fast pipeline: accountsApi + stakedBalance → detection → token + price.\n // Snap and RPC are excluded here due to their latency (snap triggers account\n // creation, RPC is slow on many chains). Results are committed to state\n // immediately so the UI can display balances without waiting for them.\n //\n // Both the fast and background pipelines use 'merge' mode because neither\n // alone represents the full set of data sources. Using 'full' in either\n // would wipe balances from the sources handled by the other pipeline.\n const fastSources = this.#isBasicFunctionality()\n ? [\n createParallelBalanceMiddleware([\n this.#accountsApiDataSource,\n this.#stakedBalanceDataSource,\n ]),\n // Graduation must run BEFORE the RPC fallback so it only sees\n // AccountsApi/Websocket balances. RPC intentionally carries\n // custom assets and must never trigger graduation.\n this.#customAssetGraduationMiddleware,\n this.#rpcFallbackMiddleware,\n this.#detectionMiddleware,\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ]\n : [this.#stakedBalanceDataSource, this.#detectionMiddleware];\n const { response, durationByDataSource } = await this.#executeMiddlewares(\n fastSources,\n request,\n );\n // The fast pipeline only contains a subset of data sources (AccountsApi +\n // StakedBalance), so it must always merge to avoid wiping Snap/RPC\n // balances that the background pipeline hasn't yet replaced.\n await this.#updateState({ ...response, updateMode: 'merge' });\n\n // Background pipeline: snap and RPC run in parallel after the fast path\n // commits to state. Their balances are merged together before detection.\n // Token + price enrichment matches the pre-split behavior: only when basic\n // functionality is on (RPC-only mode must not call token/price APIs).\n const slowSources = this.#isBasicFunctionality()\n ? [this.#snapDataSource, this.#rpcDataSource]\n : [this.#rpcDataSource];\n\n this.#executeMiddlewares(\n [\n createParallelBalanceMiddleware(slowSources),\n this.#detectionMiddleware,\n ...(this.#isBasicFunctionality()\n ? [\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ]\n : []),\n ],\n request,\n )\n .then(({ response: slowResponse }) =>\n this.#updateState({ ...slowResponse, updateMode: 'merge' }),\n )\n .catch((error) => log('Background pipeline failed', { error }));\n\n const durationMs = performance.now() - startTime;\n\n // Emit trace for every full fetch (Assets Health dashboard)\n this.#emitTrace(TRACE_FULL_FETCH, {\n duration_ms: durationMs,\n chain_count: chainIds.length,\n account_count: accounts.length,\n basic_functionality: this.#isBasicFunctionality(),\n asset_count: response.assetsBalance\n ? Object.values(response.assetsBalance).reduce(\n (sum, acct) => sum + Object.keys(acct).length,\n 0,\n )\n : 0,\n price_count: response.assetsPrice\n ? Object.keys(response.assetsPrice).length\n : 0,\n ...durationByDataSource,\n });\n\n if (!this.#firstInitFetchReported) {\n this.#firstInitFetchReported = true;\n this.#emitTrace(TRACE_FIRST_INIT_FETCH, {\n duration_ms: durationMs,\n chain_ids: JSON.stringify(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 { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n return formatExchangeRatesForBridge({\n assetsInfo: this.state.assetsInfo,\n assetsPrice: this.state.assetsPrice,\n selectedCurrency: this.state.selectedCurrency,\n nativeAssetIdentifiers: this.#getNativeAssetMap(),\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.#getSelectedAccounts();\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: this.#getNativeAssetMap(),\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 (this.#isNativeAsset(normalizedAssetId)) {\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.#getSelectedAccounts().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 // Re-evaluate subscriptions so the supplemental RPC poll picks up the\n // new customAsset on chains another data source already owns.\n this.#subscribeAssets();\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 // Re-evaluate subscriptions so the supplemental RPC poll for that chain\n // is torn down when no more customAssets remain there.\n this.#subscribeAssets();\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 if (!this.#isBasicFunctionality()) {\n return;\n }\n\n this.getAssets(this.#getSelectedAccounts(), {\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 request = this.#buildDataRequest(accounts, chainIds, {\n dataTypes: ['price'],\n });\n const subscribeReq: SubscriptionRequest = {\n request,\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 * Reads the native asset map (CAIP-2 chain ID -> CAIP-19 native asset ID)\n * from the QueryClient cache. Populated at construction via fetchQuery.\n *\n * @returns Cached map, or empty object if not yet populated.\n */\n #getNativeAssetMap(): Record<ChainId, Caip19AssetId> {\n return (\n this.#queryApiClient.getCachedData<Record<ChainId, Caip19AssetId>>(\n NATIVE_ASSETS_QUERY_KEY,\n ) ?? {}\n );\n }\n\n /**\n * Checks whether the given CAIP-19 asset ID represents a native asset\n * according to the cached native asset map, the asset namespace, and the asset reference.\n *\n * @param assetId - The CAIP-19 asset ID to check (case-insensitive).\n * @returns True if the asset ID is a native asset.\n */\n #isNativeAsset(assetId: Caip19AssetId): boolean {\n const parsed = parseCaipAssetType(assetId);\n\n if (parsed.assetNamespace === 'slip44') {\n return true;\n }\n\n const lower = assetId.toLowerCase();\n const isInNativeAssetMap = Object.values(this.#getNativeAssetMap()).some(\n (id) => id.toLowerCase() === lower,\n );\n if (isInNativeAssetMap) {\n return true;\n }\n\n if (\n parsed.assetNamespace === 'erc20' &&\n parsed.assetReference === ZERO_ADDRESS\n ) {\n return true;\n }\n\n return false;\n }\n\n /**\n * Resolves native asset IDs (CAIP-19) for the given chains by looking them up\n * in the cached native asset map.\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 nativeAssetMap = this.#getNativeAssetMap();\n const ids: Caip19AssetId[] = [];\n for (const chainId of chains) {\n const nativeId = nativeAssetMap[chainId];\n if (nativeId) {\n ids.push(nativeId);\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.#getSelectedAccounts();\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\n const existing = metadata[key] as FungibleAssetMetadata | undefined;\n const incoming = value as FungibleAssetMetadata;\n\n // When the API returns no symbol or name for this token, it has no\n // real knowledge of it (e.g. a user-deployed token not indexed by\n // the API). Preserve richer metadata already in state (e.g. from\n // pendingMetadata set by addCustomAsset) so that the correct\n // decimals/symbol/name/image are not overwritten with empty values.\n if (existing && !incoming.symbol && !incoming.name) {\n metadata[key] = {\n ...existing,\n ...incoming,\n symbol: existing.symbol,\n name: existing.name,\n decimals: existing.decimals ?? incoming.decimals,\n image: existing.image ?? incoming.image,\n };\n } else {\n metadata[key] = value;\n }\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.#getSelectedAccounts().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 // Emit state size trace (throttled to avoid JSON.stringify on every update)\n this.#emitStateSizeTrace();\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 // Skip native tokens on Tempo networks\n if (this.#shouldHideNativeToken(assetChainId, metadata)) {\n continue;\n }\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 * Determines if a native token should be hidden on specific networks.\n *\n * @param chainId - The CAIP-2 chain ID (e.g., \"eip155:42431\").\n * @param metadata - The asset metadata.\n * @returns True if the token should be hidden, false otherwise.\n */\n #shouldHideNativeToken(chainId: ChainId, metadata: AssetMetadata): boolean {\n // Check if it's a chain that should skip native tokens\n if (\n !CHAIN_IDS_WITH_NO_NATIVE_TOKEN.includes(\n chainId as (typeof CHAIN_IDS_WITH_NO_NATIVE_TOKEN)[number],\n )\n ) {\n return false;\n }\n\n return metadata.type === 'native';\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 * Idempotent — returns early if accounts/chains are not yet available or\n * subscriptions are already active.\n */\n #start(): void {\n const accounts = this.#getSelectedAccounts();\n const chainIds = [...this.#enabledChains];\n\n if (accounts.length === 0 || chainIds.length === 0) {\n return;\n }\n\n if (this.#activeSubscriptions.size > 0) {\n return;\n }\n\n log('Starting asset tracking', {\n selectedAccountCount: accounts.length,\n enabledChainCount: chainIds.length,\n });\n\n this.#lastKnownAccountIds = new Set(accounts.map((a) => a.id));\n this.#subscribeAssets();\n this.#ensureNativeBalancesDefaultZero();\n this.getAssets(accounts, {\n chainIds,\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 this.#stateSizeReported = false;\n this.#lastKnownAccountIds = new Set();\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 continue;\n }\n // Subscription keys take the form `ds:<SourceName>` for the regular\n // subscription or `ds:<SourceName>:<suffix>` for supplemental\n // subscriptions (e.g. `ds:RpcDataSource:custom`). Split on `:` and\n // pick the source-name segment so both shapes resolve correctly.\n const [, sourceId] = subscriptionKey.split(':');\n const source = allSources.find((ds) => ds.getName() === sourceId);\n if (source) {\n // Unsubscribe by the actual key — `#unsubscribeDataSource` only\n // knows the regular `ds:<SourceName>` shape and would miss\n // supplemental subscriptions, leaking their polling timers.\n this.#unsubscribeBySubscriptionKey(source, subscriptionKey);\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 const accounts = this.#getSelectedAccounts();\n const enabledChains = [...this.#enabledChains];\n if (accounts.length === 0 || enabledChains.length === 0) {\n return;\n }\n\n // Subscribe to balance updates (batched by data source)\n this.#subscribeAssetsBalance(accounts, enabledChains);\n\n // Subscribe to staked balance updates (separate from regular balance chain-claiming)\n this.#subscribeStakedBalance(accounts, enabledChains);\n\n // Subscribe to price updates for all assets held by selected accounts\n this.subscribeAssetsPrice(accounts, 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 // When basic functionality is on, use all balance data sources; when off, RPC only.\n const balanceDataSources = this.#isBasicFunctionality()\n ? this.#allBalanceDataSources\n : [this.#rpcDataSource];\n\n let rpcAssignedChains: Set<ChainId> = new Set();\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 if (source === this.#rpcDataSource) {\n rpcAssignedChains = new Set(assignedChains);\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 // Supplemental RPC subscription for customAssets on chains another data\n // source claimed during regular handoff. RPC is the sole balance fetcher\n // for customAssets, so we must always poll them — even when (e.g.)\n // AccountsApi is already covering the chain for normal balances. The\n // supplemental subscription runs in `customAssetsOnly` mode so it does\n // NOT double-poll the regular tracked balances.\n this.#subscribeRpcCustomAssetsSupplement(\n accounts,\n chainToAccounts,\n rpcAssignedChains,\n );\n }\n\n /**\n * Subscribe RPC to chains where the user has customAssets but another\n * balance data source already owns the chain in regular handoff. Uses a\n * separate subscription key (`customAssetsOnly` mode) so the regular RPC\n * subscription, if any, is unaffected.\n *\n * @param accounts - Accounts to consider for customAssets.\n * @param chainToAccounts - Map of chain → accounts (built by caller).\n * @param rpcAssignedChains - Chains RPC was assigned in the regular handoff.\n */\n #subscribeRpcCustomAssetsSupplement(\n accounts: InternalAccount[],\n chainToAccounts: Map<ChainId, InternalAccount[]>,\n rpcAssignedChains: Set<ChainId>,\n ): void {\n const rpc = this.#rpcDataSource;\n const rpcAvailableChains = new Set(rpc.getActiveChainsSync());\n\n // Collect chains that have customAssets for at least one of the given\n // accounts and are NOT already covered by the regular RPC subscription.\n const supplementalChainSet = new Set<ChainId>();\n const accountsWithCustomAssets = new Set<string>();\n for (const account of accounts) {\n const customForAccount = this.state.customAssets[account.id] ?? [];\n if (customForAccount.length === 0) {\n continue;\n }\n accountsWithCustomAssets.add(account.id);\n for (const assetId of customForAccount) {\n let chainId: ChainId;\n try {\n chainId = extractChainId(assetId);\n } catch {\n continue;\n }\n if (rpcAssignedChains.has(chainId)) {\n continue;\n }\n if (!rpcAvailableChains.has(chainId)) {\n continue;\n }\n if (!chainToAccounts.has(chainId)) {\n continue;\n }\n supplementalChainSet.add(chainId);\n }\n }\n\n const supplementalKey = `ds:${rpc.getName()}:custom`;\n if (supplementalChainSet.size === 0) {\n this.#unsubscribeBySubscriptionKey(rpc, supplementalKey);\n return;\n }\n\n const supplementalChains = [...supplementalChainSet];\n const supplementalAccounts = accounts.filter((account) =>\n accountsWithCustomAssets.has(account.id),\n );\n if (supplementalAccounts.length === 0) {\n this.#unsubscribeBySubscriptionKey(rpc, supplementalKey);\n return;\n }\n\n this.#subscribeDataSource(rpc, supplementalAccounts, supplementalChains, {\n subscriptionKey: supplementalKey,\n customAssetsOnly: true,\n });\n }\n\n /**\n * Unsubscribe a data source by an explicit subscription key.\n *\n * @param source - The data source instance.\n * @param subscriptionKey - The subscription key to unsubscribe.\n */\n #unsubscribeBySubscriptionKey(\n source: AbstractDataSource<string, DataSourceState>,\n subscriptionKey: string,\n ): void {\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n if (existingSubscription) {\n source.unsubscribe(subscriptionKey).catch(() => undefined);\n existingSubscription.unsubscribe();\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 * @param options - Optional subscription overrides.\n * @param options.subscriptionKey - Custom subscription key (default: `ds:<sourceId>`).\n * @param options.customAssetsOnly - When true, only poll customAssets for these chains.\n */\n #subscribeDataSource(\n source: AbstractDataSource<string, DataSourceState>,\n accounts: InternalAccount[],\n chains: ChainId[],\n options: { subscriptionKey?: string; customAssetsOnly?: boolean } = {},\n ): void {\n const sourceId = source.getName();\n const subscriptionKey = options.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 customAssetsOnly: options.customAssetsOnly === true,\n });\n\n const subscribeReq: SubscriptionRequest = {\n request: this.#buildDataRequest(accounts, chains, {\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n updateInterval: this.#defaultUpdateInterval,\n ...(options.customAssetsOnly === true\n ? { customAssetsOnly: true }\n : {}),\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 this.#emitTrace(TRACE_SUBSCRIPTION_ERROR, {\n source: sourceId,\n error_message: String(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 // Force update will pass in additional chains for wildcard account scope\n // Enables updates for chains that are not selected (e.g. for unapproved transactions)\n const forceUpdateChains = partial.forceUpdate ? chainIds : undefined;\n const accountsWithSupportedChains: AccountWithSupportedChains[] =\n accounts.map((account) => ({\n account,\n supportedChains: this.#getEnabledChainsForAccount(\n account,\n forceUpdateChains,\n ).filter((chain) => chainIdSet.has(chain)),\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 * @param additionalChains - Optional extra chains to include for wildcard scopes.\n * @returns Array of ChainIds that the account supports and are enabled.\n */\n #getEnabledChainsForAccount(\n account: InternalAccount,\n additionalChains?: ChainId[],\n ): 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 // Add enabled chains\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 // Add additional chains\n for (const chain of additionalChains ?? []) {\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.#getSelectedAccounts();\n\n log('Account group changed', {\n accountCount: accounts.length,\n accountIds: accounts.map((a) => a.id),\n });\n\n this.#lastKnownAccountIds = new Set(accounts.map((a) => a.id));\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.#getSelectedAccounts().length > 0) {\n await this.getAssets(this.#getSelectedAccounts(), {\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 * Runs detection, then (when basic functionality is enabled) token metadata and price enrichment before updating state.\n * When basic functionality is disabled (RPC-only mode), only detection runs; token and price APIs are not used.\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 const updateStart = performance.now();\n log('Assets updated from data source', {\n sourceId,\n hasBalance: Boolean(response.assetsBalance),\n hasPrice: Boolean(response.assetsPrice),\n });\n\n const resolvedRequest: DataRequest = request ?? {\n accountsWithSupportedChains: [],\n chainIds: [],\n dataTypes: ['balance', 'metadata', 'price'],\n };\n\n // RPC-only mode (basic functionality off): never run token/price APIs. Strip\n // those data types so downstream middleware cannot treat them as requested.\n const pipelineRequest: DataRequest = this.#isBasicFunctionality()\n ? resolvedRequest\n : {\n ...resolvedRequest,\n dataTypes: resolvedRequest.dataTypes.filter(\n (dt) => dt !== 'metadata' && dt !== 'price',\n ),\n };\n\n // Graduate custom assets only when AccountsAPI / Websocket reports them.\n // RPC already fetches custom assets on purpose, and Snap handles non-EVM\n // chains the rule does not apply to, so skip the middleware for those.\n const shouldGraduateCustomAssets =\n sourceId === 'AccountsApiDataSource' ||\n sourceId === 'BackendWebsocketDataSource';\n\n const enrichmentSources: AssetsDataSource[] = [\n ...(shouldGraduateCustomAssets\n ? [this.#customAssetGraduationMiddleware]\n : []),\n this.#detectionMiddleware,\n ];\n if (this.#isBasicFunctionality()) {\n enrichmentSources.push(\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n );\n }\n\n const { response: enrichedResponse } = await this.#executeMiddlewares(\n enrichmentSources,\n pipelineRequest,\n response,\n );\n\n await this.#updateState(enrichedResponse);\n\n this.#emitTrace(TRACE_UPDATE_PIPELINE, {\n source: sourceId,\n duration_ms: performance.now() - updateStart,\n has_balance: Boolean(response.assetsBalance),\n has_price: Boolean(response.assetsPrice),\n has_metadata: Boolean(enrichedResponse.assetsInfo),\n balance_account_count: response.assetsBalance\n ? Object.keys(response.assetsBalance).length\n : 0,\n });\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 AccountTreeControllerStateChangeEvent,\n} from '@metamask/account-tree-controller';\nimport type { AccountsControllerGetSelectedAccountAction } from '@metamask/accounts-controller';\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 { CHAIN_IDS_WITH_NO_NATIVE_TOKEN } from '@metamask/controller-utils';\nimport type { TraceCallback } from '@metamask/controller-utils';\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 NetworkControllerNetworkAddedEvent,\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 { PhishingControllerBulkScanTokensAction } from '@metamask/phishing-controller';\nimport type { PreferencesControllerStateChangeEvent } from '@metamask/preferences-controller';\nimport type {\n SnapControllerGetRunnableSnapsAction,\n SnapControllerHandleRequestAction,\n} from '@metamask/snaps-controllers';\nimport type {\n TransactionControllerIncomingTransactionsReceivedEvent,\n TransactionControllerTransactionConfirmedEvent,\n TransactionControllerUnapprovedTransactionAddedEvent,\n TransactionMeta,\n} from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\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 {\n CHAINS_WITH_DEFAULT_TRACKED_ASSETS,\n DEFAULT_TRACKED_ASSETS_BY_CHAIN,\n buildDefaultAssetsInfo,\n getDefaultAssetMetadata,\n} from './defaults';\nimport { AssetsDataSourceError } from './errors';\nimport { projectLogger, createModuleLogger } from './logger';\nimport { CustomAssetGraduationMiddleware } from './middlewares/CustomAssetGraduationMiddleware';\nimport { DetectionMiddleware } from './middlewares/DetectionMiddleware';\nimport {\n createParallelBalanceMiddleware,\n createParallelMiddleware,\n} from './middlewares/ParallelMiddleware';\nimport { RpcFallbackMiddleware } from './middlewares/RpcFallbackMiddleware';\nimport type {\n AccountId,\n AssetPreferences,\n AssetsControllerStateInternal,\n AssetsDataSource,\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} from './types';\nimport {\n normalizeAssetId,\n formatExchangeRatesForBridge,\n formatStateForTransactionPay,\n buildNativeAssetsFromConstant,\n buildNativeAssetsFromApi,\n} from './utils';\nimport type {\n BridgeExchangeRatesFormat,\n TransactionPayLegacyFormat,\n} from './utils';\nimport { ZERO_ADDRESS } from './utils/constants';\n\nconst NATIVE_ASSETS_QUERY_KEY = ['nativeAssets'];\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\n// ============================================================================\n// TRACE NAMES — used in Sentry spans (search these strings in Discover)\n// ============================================================================\nconst TRACE_FIRST_INIT_FETCH = 'AssetsControllerFirstInitFetch';\nconst TRACE_FULL_FETCH = 'AssetsFullFetch';\nconst TRACE_DATA_SOURCE_TIMING = 'AssetsDataSourceTiming';\nconst TRACE_DATA_SOURCE_ERROR = 'AssetsDataSourceError';\nconst TRACE_UPDATE_PIPELINE = 'AssetsUpdatePipeline';\nconst TRACE_SUBSCRIPTION_ERROR = 'AssetsSubscriptionError';\nconst TRACE_STATE_SIZE = 'AssetsStateSize';\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. Balance, price, and custom\n * asset maps start empty; `assetsInfo` is pre-seeded for default tracked\n * assets (see {@link buildDefaultAssetsInfo}).\n */\nexport function getDefaultAssetsControllerState(): AssetsControllerState {\n return {\n // Pre-populate metadata for controller-managed default tracked assets\n // (e.g. mUSD on Ethereum mainnet, Linea, Monad) so they are\n // immediately renderable in the UI before any RPC poll completes.\n // Per-account zero-balance entries are filled into `assetsBalance`\n // by `#ensureDefaultTrackedAssetsSeeded` once accounts are known.\n assetsInfo: buildDefaultAssetsInfo(),\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 | AccountsControllerGetSelectedAccountAction\n | AccountTreeControllerGetAccountsFromSelectedAccountGroupAction\n // RpcDataSource\n | NetworkControllerGetStateAction\n | NetworkControllerGetNetworkClientByIdAction\n // StakedBalanceDataSource\n | NetworkEnablementControllerGetStateAction\n // SnapDataSource\n | SnapControllerGetRunnableSnapsAction\n | SnapControllerHandleRequestAction\n | GetPermissions\n // BackendWebsocketDataSource\n | BackendWebSocketServiceActions\n // PhishingController\n | PhishingControllerBulkScanTokensAction;\n\ntype AllowedEvents =\n // AssetsController\n | AccountTreeControllerSelectedAccountGroupChangeEvent\n | AccountTreeControllerStateChangeEvent\n | ClientControllerStateChangeEvent\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent\n | PreferencesControllerStateChangeEvent\n | TransactionControllerUnapprovedTransactionAddedEvent\n // RpcDataSource, StakedBalanceDataSource\n | NetworkControllerStateChangeEvent\n // AssetsController (default-asset seeding when a new network is added)\n | NetworkControllerNetworkAddedEvent\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\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 trace callback. When provided, the controller invokes it with a\n * trace request when the first init/fetch completes (e.g. after unlock).\n * Use this to report first init fetch duration to Sentry (e.g. via addBreadcrumb or setMeasurement).\n */\n trace?: TraceCallback;\n /**\n * Optional Sentry (or compatible) reporter for **issue** events. When data source middlewares\n * fail, the controller constructs an `AssetsDataSourceError` and passes it here so you can call\n * `Sentry.captureException`. Span-only trace callbacks do not create Issues;\n * wire this if you need Sentry alerts on middleware failures.\n */\n captureException?: (error: Error) => 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 * Function returning whether onboarding is complete. When false,\n * RPC and staked balance data sources skip fetch and subscribe\n * (no on-chain calls until the user has finished onboarding).\n * Defaults to () => true.\n */\n isOnboarded?: () => boolean;\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 trace callback for first init/fetch measurement (duration). */\n readonly #trace?: TraceCallback;\n\n /** Optional reporter for Issue-style errors (e.g. Sentry.captureException). */\n readonly #captureException?: (error: Error) => void;\n\n /** Whether we have already reported first init fetch for this session (reset on #stop). */\n #firstInitFetchReported = false;\n\n /** Whether we have already reported state size for this session (reset on #stop). */\n #stateSizeReported = false;\n\n /**\n * Fire-and-forget trace helper. Swallows errors so telemetry never breaks the controller.\n *\n * @param name - Trace / span name visible in Sentry.\n * @param data - Key-value pairs attached as span data.\n * @param tags - Key-value pairs used for Sentry filtering.\n */\n #emitTrace(\n name: string,\n data: Record<string, number | string | boolean>,\n tags: Record<string, number | string | boolean> = {\n controller: 'AssetsController',\n },\n ): void {\n if (!this.#trace) {\n return;\n }\n this.#trace({ name, data, tags }, () => undefined)?.catch(() => {\n // Telemetry failure must not break.\n });\n }\n\n /**\n * Emit a state-size trace once on app start (first state update after unlock).\n */\n #emitStateSizeTrace(): void {\n if (!this.#trace || this.#stateSizeReported) {\n return;\n }\n this.#stateSizeReported = true;\n\n const {\n assetsBalance: balances,\n customAssets,\n assetsInfo,\n assetsPrice,\n } = this.state;\n\n // Count balance entries and collect unique asset IDs / chain IDs in one pass.\n let balanceEntries = 0;\n const uniqueAssets = new Set<string>();\n const uniqueNetworks = new Set<string>();\n\n for (const acct of Object.values(balances)) {\n const assetIds = Object.keys(acct);\n balanceEntries += assetIds.length;\n for (const assetId of assetIds) {\n uniqueAssets.add(assetId);\n // CAIP-19 format: \"eip155:1/slip44:60\" — chainId is everything before \"/\"\n const slash = assetId.indexOf('/');\n if (slash > 0) {\n uniqueNetworks.add(assetId.slice(0, slash));\n }\n }\n }\n\n let customAssetEntries = 0;\n for (const ids of Object.values(customAssets)) {\n if (Array.isArray(ids)) {\n customAssetEntries += ids.length;\n }\n }\n\n this.#emitTrace(TRACE_STATE_SIZE, {\n balance_entries: balanceEntries,\n balance_accounts: Object.keys(balances).length,\n unique_asset_count: uniqueAssets.size,\n network_count: uniqueNetworks.size,\n metadata_entries: Object.keys(assetsInfo).length,\n price_entries: Object.keys(assetsPrice).length,\n custom_asset_entries: customAssetEntries,\n });\n }\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 * Snapshot of account IDs that were active when the last subscription/fetch\n * cycle ran. Used by #handleAccountTreeStateChange to skip redundant\n * re-subscriptions when the account set hasn't actually changed.\n */\n #lastKnownAccountIds: ReadonlySet<string> = 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 #getSelectedAccounts(): InternalAccount[] {\n const accounts = this.messenger.call(\n 'AccountTreeController:getAccountsFromSelectedAccountGroup',\n );\n if (accounts.length > 0) {\n return accounts;\n }\n const selectedAccount = this.messenger.call(\n 'AccountsController:getSelectedAccount',\n );\n if (selectedAccount) {\n return [selectedAccount];\n }\n return [];\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 in priority order for chain-claiming and cleanup.\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 #customAssetGraduationMiddleware: CustomAssetGraduationMiddleware;\n\n readonly #rpcFallbackMiddleware: RpcFallbackMiddleware;\n\n readonly #tokenDataSource: TokenDataSource;\n\n #unsubscribeBasicFunctionality: (() => void) | null = null;\n\n readonly #queryApiClient: ApiPlatformClient;\n\n readonly #onActiveChainsUpdated: (\n dataSourceName: string,\n chains: 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 trace,\n captureException,\n accountsApiDataSourceConfig,\n priceDataSourceConfig,\n stakedBalanceDataSourceConfig,\n isOnboarded,\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.#trace = trace;\n this.#captureException = captureException;\n this.#queryApiClient = queryApiClient;\n const rpcConfig = rpcDataSourceConfig ?? {};\n\n this.#initializeNativeAssetsMap(queryApiClient);\n\n this.#onActiveChainsUpdated = (\n dataSourceName: string,\n chains: ChainId[],\n previousChains: ChainId[],\n ): void => {\n try {\n this.#handleActiveChainsUpdate(dataSourceName, chains, previousChains);\n } catch (error) {\n log('Failed to handle active chains update', {\n dataSourceName,\n error,\n });\n }\n };\n\n this.#backendWebsocketDataSource = new BackendWebsocketDataSource({\n messenger: this.messenger,\n queryApiClient,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n isNativeAsset: (assetId: Caip19AssetId): boolean =>\n this.#isNativeAsset(assetId),\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 getNativeAssetForChain: (chainId: ChainId): Caip19AssetId =>\n this.#getNativeAssetMap()[chainId] ??\n `${chainId}/erc20:${ZERO_ADDRESS}`,\n ...rpcConfig,\n isOnboarded: rpcConfig.isOnboarded ?? isOnboarded,\n });\n this.#stakedBalanceDataSource = new StakedBalanceDataSource({\n messenger: this.messenger,\n onActiveChainsUpdated: this.#onActiveChainsUpdated,\n ...stakedBalanceDataSourceConfig,\n });\n this.#tokenDataSource = new TokenDataSource(this.messenger, {\n queryApiClient,\n getNativeAssetIds: (): Caip19AssetId[] => {\n return this.#getNativeAssetIdsForEnabledChains();\n },\n });\n this.#priceDataSource = new PriceDataSource({\n queryApiClient,\n getSelectedCurrency: (): SupportedCurrency => this.state.selectedCurrency,\n ...priceDataSourceConfig,\n });\n this.#detectionMiddleware = new DetectionMiddleware();\n this.#customAssetGraduationMiddleware = new CustomAssetGraduationMiddleware(\n {\n getSelectedAccountId: (): AccountId | undefined => {\n try {\n return this.#getSelectedAccounts()[0]?.id;\n } catch {\n return undefined;\n }\n },\n removeCustomAsset: (accountId, assetId): void =>\n this.removeCustomAsset(accountId, assetId),\n },\n );\n this.#rpcFallbackMiddleware = new RpcFallbackMiddleware({\n rpcDataSource: this.#rpcDataSource,\n });\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 #initializeNativeAssetsMap(queryApiClient: ApiPlatformClient): void {\n queryApiClient.queryClient\n .fetchQuery({\n queryKey: NATIVE_ASSETS_QUERY_KEY,\n queryFn: buildNativeAssetsFromApi,\n staleTime: Infinity,\n gcTime: Infinity,\n })\n .catch((error) => {\n // Failure to populate native asset cache is non-fatal;\n // #isNativeAsset falls back to the seed data from buildNativeAssetsFromConstant.\n log(\n 'Failed to populate native asset cache, falling back to seed data',\n error,\n );\n });\n\n // Seed the cache synchronously so that synchronous consumers (e.g.\n // #isNativeAsset, #resolveNativeAssetIds) have data available immediately.\n // When fetchQuery resolves (currently instant, future: API), it overwrites\n // this with potentially fresher data.\n const initialNativeAssetMap = buildNativeAssetsFromConstant();\n queryApiClient.setCachedData(\n NATIVE_ASSETS_QUERY_KEY,\n initialNativeAssetMap,\n );\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 // Catch the initial tree build. On returning users,\n // `selectedAccountGroupChange` does NOT fire when the persisted group\n // is unchanged, and `accountTreeChange` doesn't fire either (init()\n // rebuilds from persisted accounts without publishing it).\n // The base-controller `:stateChange` event is guaranteed to fire\n // when init() calls this.update(). #start() is idempotent so\n // repeated fires are safe.\n this.messenger.subscribe('AccountTreeController:stateChange', () => {\n this.#handleAccountTreeStateChange();\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 // When a new network is added (e.g. the user finally adds Monad\n // to NetworkController), seed the default tracked assets for it\n // — but only if the chain is in our defaults registry. This is\n // what makes mUSD show up on Monad the moment the network is\n // configured, without waiting for it to also be enabled in\n // NetworkEnablementController.\n this.messenger.subscribe(\n 'NetworkController:networkAdded',\n (networkConfiguration) => {\n this.#handleNetworkAdded(networkConfiguration.chainId);\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 // Subscribe to unapproved transactions - TXs that need confirmation\n // Ensures that balances for the account making transaction are updated (e.g. for gas estimations)\n this.messenger.subscribe(\n 'TransactionController:unapprovedTransactionAdded',\n (transactionMeta: TransactionMeta) => {\n this.#onUnapprovedTransactionAdded(transactionMeta);\n },\n );\n }\n\n #onUnapprovedTransactionAdded(transactionMeta: TransactionMeta): void {\n const hexChainId = transactionMeta.chainId;\n if (!hexChainId) {\n return;\n }\n\n const caipChainId = `eip155:${parseInt(hexChainId, 16)}` as ChainId;\n const fromAddress = transactionMeta.txParams.from?.toLowerCase();\n if (!fromAddress) {\n return;\n }\n\n const matchedAccount = this.#getSelectedAccounts().find(\n (account) => account.address.toLowerCase() === fromAddress,\n );\n if (!matchedAccount) {\n return;\n }\n\n this.getAssets([matchedAccount], {\n chainIds: [caipChainId],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to refresh assets after unapproved transaction added', {\n error,\n });\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 /**\n * Handle AccountTreeController state changes.\n * If already running, re-subscribe only when the set of selected accounts\n * has actually changed (e.g. a snap account was added after initial startup).\n * This guards against the many tree mutations that don't affect which\n * accounts are selected — without this check every tree update would\n * trigger a redundant full re-subscribe + forceUpdate fetch.\n * If not running yet, delegate to #start() for the normal start flow.\n */\n #handleAccountTreeStateChange(): void {\n const shouldRun = this.#uiOpen && this.#keyringUnlocked;\n if (!shouldRun) {\n return;\n }\n if (this.#activeSubscriptions.size > 0) {\n const accounts = this.#getSelectedAccounts();\n const currentIds = new Set(accounts.map((a) => a.id));\n\n const accountsChanged =\n currentIds.size !== this.#lastKnownAccountIds.size ||\n [...currentIds].some((id) => !this.#lastKnownAccountIds.has(id));\n\n if (!accountsChanged) {\n return;\n }\n\n log('Account tree changed with new accounts, re-subscribing', {\n previousCount: this.#lastKnownAccountIds.size,\n currentCount: currentIds.size,\n });\n\n this.#lastKnownAccountIds = currentIds;\n this.#subscribeAssets();\n this.#ensureNativeBalancesDefaultZero();\n this.#ensureDefaultTrackedAssetsSeeded();\n this.getAssets(accounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch assets after tree change', error);\n });\n } else {\n this.#start();\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.#getSelectedAccounts().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.#getSelectedAccounts(), {\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: AssetsDataSource[],\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 = performance.now();\n try {\n return await middleware(ctx, next);\n } finally {\n inclusive[i] = performance.now() - start;\n }\n }) as Middleware,\n );\n\n const middlewareErrors: string[] = [];\n const chain = wrapped.reduceRight<NextFunction>(\n (next, middleware, index) =>\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 const sourceName = names[index] ?? `middleware_${index}`;\n middlewareErrors.push(sourceName);\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 if (result.durationByDataSource) {\n for (const [key, ms] of Object.entries(result.durationByDataSource)) {\n durationByDataSource[key] = ms;\n }\n }\n\n // Emit per-source timing traces for the Assets Health dashboard\n for (const [sourceName, durationMs] of Object.entries(\n durationByDataSource,\n )) {\n this.#emitTrace(TRACE_DATA_SOURCE_TIMING, {\n source: sourceName,\n duration_ms: durationMs,\n chain_count: request.chainIds.length,\n account_count: request.accountsWithSupportedChains.length,\n });\n }\n\n // Failed middlewares: Issues (optional) + perf/Dashboard spans\n if (middlewareErrors.length > 0) {\n const failedSources = middlewareErrors.join(',');\n const assetsError = new AssetsDataSourceError({\n failedSources,\n errorCount: middlewareErrors.length,\n chainCount: request.chainIds.length,\n });\n try {\n this.#captureException?.(assetsError);\n } catch {\n // Never let telemetry throw.\n }\n this.#emitTrace(\n TRACE_DATA_SOURCE_ERROR,\n {\n failed_sources: failedSources,\n error_count: middlewareErrors.length,\n chain_count: request.chainIds.length,\n },\n {\n controller: 'AssetsController',\n severity: 'error',\n error_type: assetsError.name,\n },\n );\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 = performance.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 // Fast pipeline: accountsApi + stakedBalance → detection → token + price.\n // Snap and RPC are excluded here due to their latency (snap triggers account\n // creation, RPC is slow on many chains). Results are committed to state\n // immediately so the UI can display balances without waiting for them.\n //\n // Both the fast and background pipelines use 'merge' mode because neither\n // alone represents the full set of data sources. Using 'full' in either\n // would wipe balances from the sources handled by the other pipeline.\n const fastSources = this.#isBasicFunctionality()\n ? [\n createParallelBalanceMiddleware([\n this.#accountsApiDataSource,\n this.#stakedBalanceDataSource,\n ]),\n // Graduation must run BEFORE the RPC fallback so it only sees\n // AccountsApi/Websocket balances. RPC intentionally carries\n // custom assets and must never trigger graduation.\n this.#customAssetGraduationMiddleware,\n this.#rpcFallbackMiddleware,\n this.#detectionMiddleware,\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ]\n : [this.#stakedBalanceDataSource, this.#detectionMiddleware];\n const { response, durationByDataSource } = await this.#executeMiddlewares(\n fastSources,\n request,\n );\n // The fast pipeline only contains a subset of data sources (AccountsApi +\n // StakedBalance), so it must always merge to avoid wiping Snap/RPC\n // balances that the background pipeline hasn't yet replaced.\n await this.#updateState({ ...response, updateMode: 'merge' });\n\n // Background pipeline: snap and RPC run in parallel after the fast path\n // commits to state. Their balances are merged together before detection.\n // Token + price enrichment matches the pre-split behavior: only when basic\n // functionality is on (RPC-only mode must not call token/price APIs).\n const slowSources = this.#isBasicFunctionality()\n ? [this.#snapDataSource, this.#rpcDataSource]\n : [this.#rpcDataSource];\n\n this.#executeMiddlewares(\n [\n createParallelBalanceMiddleware(slowSources),\n this.#detectionMiddleware,\n ...(this.#isBasicFunctionality()\n ? [\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n ]\n : []),\n ],\n request,\n )\n .then(({ response: slowResponse }) =>\n this.#updateState({ ...slowResponse, updateMode: 'merge' }),\n )\n .catch((error) => log('Background pipeline failed', { error }));\n\n const durationMs = performance.now() - startTime;\n\n // Emit trace for every full fetch (Assets Health dashboard)\n this.#emitTrace(TRACE_FULL_FETCH, {\n duration_ms: durationMs,\n chain_count: chainIds.length,\n account_count: accounts.length,\n basic_functionality: this.#isBasicFunctionality(),\n asset_count: response.assetsBalance\n ? Object.values(response.assetsBalance).reduce(\n (sum, acct) => sum + Object.keys(acct).length,\n 0,\n )\n : 0,\n price_count: response.assetsPrice\n ? Object.keys(response.assetsPrice).length\n : 0,\n ...durationByDataSource,\n });\n\n if (!this.#firstInitFetchReported) {\n this.#firstInitFetchReported = true;\n this.#emitTrace(TRACE_FIRST_INIT_FETCH, {\n duration_ms: durationMs,\n chain_ids: JSON.stringify(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 { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n return formatExchangeRatesForBridge({\n assetsInfo: this.state.assetsInfo,\n assetsPrice: this.state.assetsPrice,\n selectedCurrency: this.state.selectedCurrency,\n nativeAssetIdentifiers: this.#getNativeAssetMap(),\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.#getSelectedAccounts();\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: this.#getNativeAssetMap(),\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 (this.#isNativeAsset(normalizedAssetId)) {\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.#getSelectedAccounts().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 // Re-evaluate subscriptions so the supplemental RPC poll picks up the\n // new customAsset on chains another data source already owns.\n this.#subscribeAssets();\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 // Re-evaluate subscriptions so the supplemental RPC poll for that chain\n // is torn down when no more customAssets remain there.\n this.#subscribeAssets();\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 if (!this.#isBasicFunctionality()) {\n return;\n }\n\n this.getAssets(this.#getSelectedAccounts(), {\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 request = this.#buildDataRequest(accounts, chainIds, {\n dataTypes: ['price'],\n });\n const subscribeReq: SubscriptionRequest = {\n request,\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 * Reads the native asset map (CAIP-2 chain ID -> CAIP-19 native asset ID)\n * from the QueryClient cache. Populated at construction via fetchQuery.\n *\n * @returns Cached map, or empty object if not yet populated.\n */\n #getNativeAssetMap(): Record<ChainId, Caip19AssetId> {\n return (\n this.#queryApiClient.getCachedData<Record<ChainId, Caip19AssetId>>(\n NATIVE_ASSETS_QUERY_KEY,\n ) ?? {}\n );\n }\n\n /**\n * Checks whether the given CAIP-19 asset ID represents a native asset\n * according to the cached native asset map, the asset namespace, and the asset reference.\n *\n * @param assetId - The CAIP-19 asset ID to check (case-insensitive).\n * @returns True if the asset ID is a native asset.\n */\n #isNativeAsset(assetId: Caip19AssetId): boolean {\n const parsed = parseCaipAssetType(assetId);\n\n if (parsed.assetNamespace === 'slip44') {\n return true;\n }\n\n const lower = assetId.toLowerCase();\n const isInNativeAssetMap = Object.values(this.#getNativeAssetMap()).some(\n (id) => id.toLowerCase() === lower,\n );\n if (isInNativeAssetMap) {\n return true;\n }\n\n if (\n parsed.assetNamespace === 'erc20' &&\n parsed.assetReference === ZERO_ADDRESS\n ) {\n return true;\n }\n\n return false;\n }\n\n /**\n * Resolves native asset IDs (CAIP-19) for the given chains by looking them up\n * in the cached native asset map.\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 nativeAssetMap = this.#getNativeAssetMap();\n const ids: Caip19AssetId[] = [];\n for (const chainId of chains) {\n const nativeId = nativeAssetMap[chainId];\n if (nativeId) {\n ids.push(nativeId);\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.#getSelectedAccounts();\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 /**\n * Seed selected accounts with zero-balance entries for every\n * controller-managed default tracked asset (e.g. mUSD on mainnet,\n * Linea, Monad). Also re-asserts the metadata in `assetsInfo` so the\n * defaults survive a future state migration that strips them.\n *\n * `chainsToSeed` lets callers narrow the work to a specific subset\n * of chains — useful when reacting to a `NetworkEnablementController`\n * change to seed only the chain that was just added (e.g. when the\n * user finally turns on Monad). When `undefined`, every chain in the\n * default tracked assets registry is seeded.\n *\n * Existing entries are never clobbered.\n *\n * @param chainsToSeed - Optional subset of CAIP-2 chain ids to seed.\n * If omitted, all default tracked chains are seeded.\n */\n #ensureDefaultTrackedAssetsSeeded(chainsToSeed?: ChainId[]): void {\n const accounts = this.#getSelectedAccounts();\n if (accounts.length === 0) {\n return;\n }\n\n // Default tracked assets are ERC-20s on EVM chains today. Restrict\n // seeding to accounts that have at least one EVM scope so we don't\n // pollute non-EVM accounts. The wildcard `eip155:0` scope counts.\n const evmAccounts = accounts.filter((account) =>\n (account.scopes ?? []).some((scope) => scope.startsWith('eip155:')),\n );\n if (evmAccounts.length === 0) {\n return;\n }\n\n // Gate seeding on whether the chain is currently enabled. This is\n // why mUSD appears on mainnet/Linea immediately (those are normally\n // enabled at startup) but only appears on Monad once the user\n // turns the Monad network on.\n const candidateChains = chainsToSeed ?? [...this.#enabledChains];\n const targetChains = candidateChains.filter((chainId) =>\n CHAINS_WITH_DEFAULT_TRACKED_ASSETS.has(chainId),\n );\n if (targetChains.length === 0) {\n return;\n }\n this.update((state) => {\n const balances = state.assetsBalance as Record<\n string,\n Record<string, AssetBalance>\n >;\n const metadata = state.assetsInfo as Record<string, AssetMetadata>;\n\n for (const chainId of targetChains) {\n const defaultAssetIds =\n DEFAULT_TRACKED_ASSETS_BY_CHAIN.get(chainId) ?? [];\n for (const assetId of defaultAssetIds) {\n // Re-seed metadata if state was hydrated from a prior version\n // that didn't include defaults.\n if (!metadata[assetId]) {\n const seed = getDefaultAssetMetadata(assetId);\n if (seed) {\n metadata[assetId] = seed;\n }\n }\n\n for (const account of evmAccounts) {\n balances[account.id] ??= {};\n const accountBalances = balances[account.id];\n accountBalances[assetId] ??= { 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\n const existing = metadata[key] as FungibleAssetMetadata | undefined;\n const incoming = value as FungibleAssetMetadata;\n\n // When the API returns no symbol or name for this token, it has no\n // real knowledge of it (e.g. a user-deployed token not indexed by\n // the API). Preserve richer metadata already in state (e.g. from\n // pendingMetadata set by addCustomAsset) so that the correct\n // decimals/symbol/name/image are not overwritten with empty values.\n if (existing && !incoming.symbol && !incoming.name) {\n metadata[key] = {\n ...existing,\n ...incoming,\n symbol: existing.symbol,\n name: existing.name,\n decimals: existing.decimals ?? incoming.decimals,\n image: existing.image ?? incoming.image,\n };\n } else {\n metadata[key] = value;\n }\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.#getSelectedAccounts().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 // Emit state size trace (throttled to avoid JSON.stringify on every update)\n this.#emitStateSizeTrace();\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 // Skip native tokens on Tempo networks\n if (this.#shouldHideNativeToken(assetChainId, metadata)) {\n continue;\n }\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 * Determines if a native token should be hidden on specific networks.\n *\n * @param chainId - The CAIP-2 chain ID (e.g., \"eip155:42431\").\n * @param metadata - The asset metadata.\n * @returns True if the token should be hidden, false otherwise.\n */\n #shouldHideNativeToken(chainId: ChainId, metadata: AssetMetadata): boolean {\n // Check if it's a chain that should skip native tokens\n if (\n !CHAIN_IDS_WITH_NO_NATIVE_TOKEN.includes(\n chainId as (typeof CHAIN_IDS_WITH_NO_NATIVE_TOKEN)[number],\n )\n ) {\n return false;\n }\n\n return metadata.type === 'native';\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 * Idempotent — returns early if accounts/chains are not yet available or\n * subscriptions are already active.\n */\n #start(): void {\n const accounts = this.#getSelectedAccounts();\n const chainIds = [...this.#enabledChains];\n\n if (accounts.length === 0 || chainIds.length === 0) {\n return;\n }\n\n if (this.#activeSubscriptions.size > 0) {\n return;\n }\n\n log('Starting asset tracking', {\n selectedAccountCount: accounts.length,\n enabledChainCount: chainIds.length,\n });\n\n this.#lastKnownAccountIds = new Set(accounts.map((a) => a.id));\n this.#subscribeAssets();\n this.#ensureNativeBalancesDefaultZero();\n this.#ensureDefaultTrackedAssetsSeeded();\n this.getAssets(accounts, {\n chainIds,\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 this.#stateSizeReported = false;\n this.#lastKnownAccountIds = new Set();\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 continue;\n }\n // Subscription keys take the form `ds:<SourceName>` for the regular\n // subscription or `ds:<SourceName>:<suffix>` for supplemental\n // subscriptions (e.g. `ds:RpcDataSource:custom`). Split on `:` and\n // pick the source-name segment so both shapes resolve correctly.\n const [, sourceId] = subscriptionKey.split(':');\n const source = allSources.find((ds) => ds.getName() === sourceId);\n if (source) {\n // Unsubscribe by the actual key — `#unsubscribeDataSource` only\n // knows the regular `ds:<SourceName>` shape and would miss\n // supplemental subscriptions, leaking their polling timers.\n this.#unsubscribeBySubscriptionKey(source, subscriptionKey);\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 const accounts = this.#getSelectedAccounts();\n const enabledChains = [...this.#enabledChains];\n if (accounts.length === 0 || enabledChains.length === 0) {\n return;\n }\n\n // Subscribe to balance updates (batched by data source)\n this.#subscribeAssetsBalance(accounts, enabledChains);\n\n // Subscribe to staked balance updates (separate from regular balance chain-claiming)\n this.#subscribeStakedBalance(accounts, enabledChains);\n\n // Subscribe to price updates for all assets held by selected accounts\n this.subscribeAssetsPrice(accounts, 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 // When basic functionality is on, use all balance data sources; when off, RPC only.\n const balanceDataSources = this.#isBasicFunctionality()\n ? this.#allBalanceDataSources\n : [this.#rpcDataSource];\n\n let rpcAssignedChains: Set<ChainId> = new Set();\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 if (source === this.#rpcDataSource) {\n rpcAssignedChains = new Set(assignedChains);\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 // Supplemental RPC subscription for customAssets on chains another data\n // source claimed during regular handoff. RPC is the sole balance fetcher\n // for customAssets, so we must always poll them — even when (e.g.)\n // AccountsApi is already covering the chain for normal balances. The\n // supplemental subscription runs in `customAssetsOnly` mode so it does\n // NOT double-poll the regular tracked balances.\n this.#subscribeRpcCustomAssetsSupplement(\n accounts,\n chainToAccounts,\n rpcAssignedChains,\n );\n }\n\n /**\n * Subscribe RPC to chains where the user has customAssets but another\n * balance data source already owns the chain in regular handoff. Uses a\n * separate subscription key (`customAssetsOnly` mode) so the regular RPC\n * subscription, if any, is unaffected.\n *\n * @param accounts - Accounts to consider for customAssets.\n * @param chainToAccounts - Map of chain → accounts (built by caller).\n * @param rpcAssignedChains - Chains RPC was assigned in the regular handoff.\n */\n #subscribeRpcCustomAssetsSupplement(\n accounts: InternalAccount[],\n chainToAccounts: Map<ChainId, InternalAccount[]>,\n rpcAssignedChains: Set<ChainId>,\n ): void {\n const rpc = this.#rpcDataSource;\n const rpcAvailableChains = new Set(rpc.getActiveChainsSync());\n\n // Collect chains that have customAssets for at least one of the given\n // accounts and are NOT already covered by the regular RPC subscription.\n const supplementalChainSet = new Set<ChainId>();\n const accountsWithCustomAssets = new Set<string>();\n for (const account of accounts) {\n const customForAccount = this.state.customAssets[account.id] ?? [];\n if (customForAccount.length === 0) {\n continue;\n }\n accountsWithCustomAssets.add(account.id);\n for (const assetId of customForAccount) {\n let chainId: ChainId;\n try {\n chainId = extractChainId(assetId);\n } catch {\n continue;\n }\n if (rpcAssignedChains.has(chainId)) {\n continue;\n }\n if (!rpcAvailableChains.has(chainId)) {\n continue;\n }\n if (!chainToAccounts.has(chainId)) {\n continue;\n }\n supplementalChainSet.add(chainId);\n }\n }\n\n const supplementalKey = `ds:${rpc.getName()}:custom`;\n if (supplementalChainSet.size === 0) {\n this.#unsubscribeBySubscriptionKey(rpc, supplementalKey);\n return;\n }\n\n const supplementalChains = [...supplementalChainSet];\n const supplementalAccounts = accounts.filter((account) =>\n accountsWithCustomAssets.has(account.id),\n );\n if (supplementalAccounts.length === 0) {\n this.#unsubscribeBySubscriptionKey(rpc, supplementalKey);\n return;\n }\n\n this.#subscribeDataSource(rpc, supplementalAccounts, supplementalChains, {\n subscriptionKey: supplementalKey,\n customAssetsOnly: true,\n });\n }\n\n /**\n * Unsubscribe a data source by an explicit subscription key.\n *\n * @param source - The data source instance.\n * @param subscriptionKey - The subscription key to unsubscribe.\n */\n #unsubscribeBySubscriptionKey(\n source: AbstractDataSource<string, DataSourceState>,\n subscriptionKey: string,\n ): void {\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n if (existingSubscription) {\n source.unsubscribe(subscriptionKey).catch(() => undefined);\n existingSubscription.unsubscribe();\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 * @param options - Optional subscription overrides.\n * @param options.subscriptionKey - Custom subscription key (default: `ds:<sourceId>`).\n * @param options.customAssetsOnly - When true, only poll customAssets for these chains.\n */\n #subscribeDataSource(\n source: AbstractDataSource<string, DataSourceState>,\n accounts: InternalAccount[],\n chains: ChainId[],\n options: { subscriptionKey?: string; customAssetsOnly?: boolean } = {},\n ): void {\n const sourceId = source.getName();\n const subscriptionKey = options.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 customAssetsOnly: options.customAssetsOnly === true,\n });\n\n const subscribeReq: SubscriptionRequest = {\n request: this.#buildDataRequest(accounts, chains, {\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n updateInterval: this.#defaultUpdateInterval,\n ...(options.customAssetsOnly === true\n ? { customAssetsOnly: true }\n : {}),\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 this.#emitTrace(TRACE_SUBSCRIPTION_ERROR, {\n source: sourceId,\n error_message: String(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 // Force update will pass in additional chains for wildcard account scope\n // Enables updates for chains that are not selected (e.g. for unapproved transactions)\n const forceUpdateChains = partial.forceUpdate ? chainIds : undefined;\n const accountsWithSupportedChains: AccountWithSupportedChains[] =\n accounts.map((account) => ({\n account,\n supportedChains: this.#getEnabledChainsForAccount(\n account,\n forceUpdateChains,\n ).filter((chain) => chainIdSet.has(chain)),\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 * @param additionalChains - Optional extra chains to include for wildcard scopes.\n * @returns Array of ChainIds that the account supports and are enabled.\n */\n #getEnabledChainsForAccount(\n account: InternalAccount,\n additionalChains?: ChainId[],\n ): 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 // Add enabled chains\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 // Add additional chains\n for (const chain of additionalChains ?? []) {\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.#getSelectedAccounts();\n\n log('Account group changed', {\n accountCount: accounts.length,\n accountIds: accounts.map((a) => a.id),\n });\n\n this.#lastKnownAccountIds = new Set(accounts.map((a) => a.id));\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 this.#ensureDefaultTrackedAssetsSeeded();\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.#getSelectedAccounts().length > 0) {\n await this.getAssets(this.#getSelectedAccounts(), {\n chainIds: addedChains,\n forceUpdate: true,\n updateMode: 'merge',\n });\n }\n\n this.#ensureNativeBalancesDefaultZero();\n // Seed default tracked assets (mUSD) for any chain the user has\n // *just* enabled. This is what makes mUSD appear on Monad after\n // the user finally adds it to NetworkEnablementController.\n if (addedChains.length > 0) {\n this.#ensureDefaultTrackedAssetsSeeded(addedChains);\n }\n }\n\n /**\n * Handle a `NetworkController:networkAdded` event. When the user\n * adds a network (e.g. Monad) to NetworkController, seed the\n * controller-managed default tracked assets for that chain (mUSD on\n * Monad) into state immediately — without waiting for the chain to\n * also be enabled in NetworkEnablementController. No-op for chains\n * that aren't in the defaults registry.\n *\n * @param hexChainId - Hex chain id of the newly-added network\n * configuration (e.g. `0x279f`).\n */\n #handleNetworkAdded(hexChainId: Hex): void {\n let caipChainId: ChainId;\n try {\n caipChainId = `eip155:${parseInt(hexChainId, 16)}` as ChainId;\n } catch {\n return;\n }\n\n if (!CHAINS_WITH_DEFAULT_TRACKED_ASSETS.has(caipChainId)) {\n return;\n }\n\n log('Network added — seeding default tracked assets', {\n hexChainId,\n caipChainId,\n });\n\n this.#ensureDefaultTrackedAssetsSeeded([caipChainId]);\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 * Runs detection, then (when basic functionality is enabled) token metadata and price enrichment before updating state.\n * When basic functionality is disabled (RPC-only mode), only detection runs; token and price APIs are not used.\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 const updateStart = performance.now();\n log('Assets updated from data source', {\n sourceId,\n hasBalance: Boolean(response.assetsBalance),\n hasPrice: Boolean(response.assetsPrice),\n });\n\n const resolvedRequest: DataRequest = request ?? {\n accountsWithSupportedChains: [],\n chainIds: [],\n dataTypes: ['balance', 'metadata', 'price'],\n };\n\n // RPC-only mode (basic functionality off): never run token/price APIs. Strip\n // those data types so downstream middleware cannot treat them as requested.\n const pipelineRequest: DataRequest = this.#isBasicFunctionality()\n ? resolvedRequest\n : {\n ...resolvedRequest,\n dataTypes: resolvedRequest.dataTypes.filter(\n (dt) => dt !== 'metadata' && dt !== 'price',\n ),\n };\n\n // Graduate custom assets only when AccountsAPI / Websocket reports them.\n // RPC already fetches custom assets on purpose, and Snap handles non-EVM\n // chains the rule does not apply to, so skip the middleware for those.\n const shouldGraduateCustomAssets =\n sourceId === 'AccountsApiDataSource' ||\n sourceId === 'BackendWebsocketDataSource';\n\n const enrichmentSources: AssetsDataSource[] = [\n ...(shouldGraduateCustomAssets\n ? [this.#customAssetGraduationMiddleware]\n : []),\n this.#detectionMiddleware,\n ];\n if (this.#isBasicFunctionality()) {\n enrichmentSources.push(\n createParallelMiddleware([\n this.#tokenDataSource,\n this.#priceDataSource,\n ]),\n );\n }\n\n const { response: enrichedResponse } = await this.#executeMiddlewares(\n enrichmentSources,\n pipelineRequest,\n response,\n );\n\n await this.#updateState(enrichedResponse);\n\n this.#emitTrace(TRACE_UPDATE_PIPELINE, {\n source: sourceId,\n duration_ms: performance.now() - updateStart,\n has_balance: Boolean(response.assetsBalance),\n has_price: Boolean(response.assetsPrice),\n has_metadata: Boolean(enrichedResponse.assetsInfo),\n balance_account_count: response.assetsBalance\n ? Object.keys(response.assetsBalance).length\n : 0,\n });\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"]}