@metamask-previews/assets-controller 0.0.0-preview-e09bf49f → 0.0.0-preview-52f4a2ca

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (70) hide show
  1. package/CHANGELOG.md +3 -0
  2. package/dist/AssetsController.cjs +26 -32
  3. package/dist/AssetsController.cjs.map +1 -1
  4. package/dist/AssetsController.d.cts +5 -1
  5. package/dist/AssetsController.d.cts.map +1 -1
  6. package/dist/AssetsController.d.mts +5 -1
  7. package/dist/AssetsController.d.mts.map +1 -1
  8. package/dist/AssetsController.mjs +26 -32
  9. package/dist/AssetsController.mjs.map +1 -1
  10. package/dist/data-sources/PriceDataSource.cjs +12 -19
  11. package/dist/data-sources/PriceDataSource.cjs.map +1 -1
  12. package/dist/data-sources/PriceDataSource.d.cts.map +1 -1
  13. package/dist/data-sources/PriceDataSource.d.mts.map +1 -1
  14. package/dist/data-sources/PriceDataSource.mjs +12 -19
  15. package/dist/data-sources/PriceDataSource.mjs.map +1 -1
  16. package/dist/data-sources/RpcDataSource.cjs +232 -42
  17. package/dist/data-sources/RpcDataSource.cjs.map +1 -1
  18. package/dist/data-sources/RpcDataSource.d.cts +5 -1
  19. package/dist/data-sources/RpcDataSource.d.cts.map +1 -1
  20. package/dist/data-sources/RpcDataSource.d.mts +5 -1
  21. package/dist/data-sources/RpcDataSource.d.mts.map +1 -1
  22. package/dist/data-sources/RpcDataSource.mjs +229 -42
  23. package/dist/data-sources/RpcDataSource.mjs.map +1 -1
  24. package/dist/data-sources/evm-rpc-services/index.cjs.map +1 -1
  25. package/dist/data-sources/evm-rpc-services/index.d.cts +1 -1
  26. package/dist/data-sources/evm-rpc-services/index.d.cts.map +1 -1
  27. package/dist/data-sources/evm-rpc-services/index.d.mts +1 -1
  28. package/dist/data-sources/evm-rpc-services/index.d.mts.map +1 -1
  29. package/dist/data-sources/evm-rpc-services/index.mjs.map +1 -1
  30. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.cjs +32 -27
  31. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.cjs.map +1 -1
  32. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.d.cts +12 -5
  33. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.d.cts.map +1 -1
  34. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.d.mts +12 -5
  35. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.d.mts.map +1 -1
  36. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.mjs +32 -27
  37. package/dist/data-sources/evm-rpc-services/services/BalanceFetcher.mjs.map +1 -1
  38. package/dist/data-sources/evm-rpc-services/services/TokenDetector.cjs +23 -12
  39. package/dist/data-sources/evm-rpc-services/services/TokenDetector.cjs.map +1 -1
  40. package/dist/data-sources/evm-rpc-services/services/TokenDetector.d.cts +8 -3
  41. package/dist/data-sources/evm-rpc-services/services/TokenDetector.d.cts.map +1 -1
  42. package/dist/data-sources/evm-rpc-services/services/TokenDetector.d.mts +8 -3
  43. package/dist/data-sources/evm-rpc-services/services/TokenDetector.d.mts.map +1 -1
  44. package/dist/data-sources/evm-rpc-services/services/TokenDetector.mjs +23 -12
  45. package/dist/data-sources/evm-rpc-services/services/TokenDetector.mjs.map +1 -1
  46. package/dist/data-sources/evm-rpc-services/services/index.cjs.map +1 -1
  47. package/dist/data-sources/evm-rpc-services/services/index.d.cts +2 -2
  48. package/dist/data-sources/evm-rpc-services/services/index.d.cts.map +1 -1
  49. package/dist/data-sources/evm-rpc-services/services/index.d.mts +2 -2
  50. package/dist/data-sources/evm-rpc-services/services/index.d.mts.map +1 -1
  51. package/dist/data-sources/evm-rpc-services/services/index.mjs.map +1 -1
  52. package/dist/data-sources/evm-rpc-services/types/index.cjs.map +1 -1
  53. package/dist/data-sources/evm-rpc-services/types/index.d.cts +1 -1
  54. package/dist/data-sources/evm-rpc-services/types/index.d.cts.map +1 -1
  55. package/dist/data-sources/evm-rpc-services/types/index.d.mts +1 -1
  56. package/dist/data-sources/evm-rpc-services/types/index.d.mts.map +1 -1
  57. package/dist/data-sources/evm-rpc-services/types/index.mjs.map +1 -1
  58. package/dist/data-sources/evm-rpc-services/types/state.cjs.map +1 -1
  59. package/dist/data-sources/evm-rpc-services/types/state.d.cts +9 -24
  60. package/dist/data-sources/evm-rpc-services/types/state.d.cts.map +1 -1
  61. package/dist/data-sources/evm-rpc-services/types/state.d.mts +9 -24
  62. package/dist/data-sources/evm-rpc-services/types/state.d.mts.map +1 -1
  63. package/dist/data-sources/evm-rpc-services/types/state.mjs.map +1 -1
  64. package/dist/types.cjs.map +1 -1
  65. package/dist/types.d.cts +40 -6
  66. package/dist/types.d.cts.map +1 -1
  67. package/dist/types.d.mts +40 -6
  68. package/dist/types.d.mts.map +1 -1
  69. package/dist/types.mjs.map +1 -1
  70. package/package.json +1 -1
package/CHANGELOG.md CHANGED
@@ -21,6 +21,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
21
21
 
22
22
  ### Changed
23
23
 
24
+ - Convert asset balances to human-readable format using token decimals across all data sources (`RpcDataSource`, `AccountsApiDataSource`, `BackendWebsocketDataSource`) ([#7752](https://github.com/MetaMask/core/pull/7752))
25
+ - Store native token metadata (type, symbol, name, decimals) in `assetsMetadata` derived from `NetworkController` chain status ([#7752](https://github.com/MetaMask/core/pull/7752))
26
+ - `AccountsApiDataSource` now includes `assetsMetadata` in response with token info from V5 API ([#7752](https://github.com/MetaMask/core/pull/7752))
24
27
  - Bump `@metamask/keyring-controller` from `^25.0.0` to `^25.1.0` ([#7713](https://github.com/MetaMask/core/pull/7713))
25
28
  - Refactor `MulticallClient` to use viem for ABI encoding/decoding instead of manual implementation
26
29
  - Refactor `RpcDataSource` to delegate polling to `BalanceFetcher` and `TokenDetector` services ([#7709](https://github.com/MetaMask/core/pull/7709))
@@ -13,7 +13,7 @@ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (
13
13
  var __importDefault = (this && this.__importDefault) || function (mod) {
14
14
  return (mod && mod.__esModule) ? mod : { "default": mod };
15
15
  };
16
- var _AssetsController_instances, _AssetsController_defaultUpdateInterval, _AssetsController_controllerMutex, _AssetsController_activeSubscriptions, _AssetsController_enabledChains, _AssetsController_selectedAccounts_get, _AssetsController_assetsPrice, _AssetsController_dataSources, _AssetsController_initializeState, _AssetsController_extractEnabledChains, _AssetsController_normalizeChainReference, _AssetsController_subscribeToEvents, _AssetsController_registerActionHandlers, _AssetsController_executeMiddlewares, _AssetsController_assignChainsToDataSources, _AssetsController_updateState, _AssetsController_getAssetsFromState, _AssetsController_tokenStandardToAssetType, _AssetsController_start, _AssetsController_stop, _AssetsController_subscribeToDataSources, _AssetsController_subscribeAssetsBalance, _AssetsController_buildChainToAccountsMap, _AssetsController_getAccountsForChains, _AssetsController_subscribeToDataSource, _AssetsController_unsubscribeDataSource, _AssetsController_getEnabledChainsForAccount, _AssetsController_handleAccountGroupChanged, _AssetsController_handleEnabledNetworksChanged, _AssetsController_handleSubscriptionUpdate;
16
+ var _AssetsController_instances, _AssetsController_defaultUpdateInterval, _AssetsController_controllerMutex, _AssetsController_activeSubscriptions, _AssetsController_enabledChains, _AssetsController_selectedAccounts_get, _AssetsController_dataSources, _AssetsController_initializeState, _AssetsController_extractEnabledChains, _AssetsController_normalizeChainReference, _AssetsController_subscribeToEvents, _AssetsController_registerActionHandlers, _AssetsController_executeMiddlewares, _AssetsController_assignChainsToDataSources, _AssetsController_updateState, _AssetsController_getAssetsFromState, _AssetsController_tokenStandardToAssetType, _AssetsController_start, _AssetsController_stop, _AssetsController_subscribeToDataSources, _AssetsController_subscribeAssetsBalance, _AssetsController_buildChainToAccountsMap, _AssetsController_getAccountsForChains, _AssetsController_subscribeToDataSource, _AssetsController_unsubscribeDataSource, _AssetsController_getEnabledChainsForAccount, _AssetsController_handleAccountGroupChanged, _AssetsController_handleEnabledNetworksChanged, _AssetsController_handleSubscriptionUpdate;
17
17
  Object.defineProperty(exports, "__esModule", { value: true });
18
18
  exports.AssetsController = exports.getDefaultAssetsControllerState = void 0;
19
19
  const base_controller_1 = require("@metamask/base-controller");
@@ -33,12 +33,13 @@ const log = (0, logger_1.createModuleLogger)(logger_1.projectLogger, CONTROLLER_
33
33
  /**
34
34
  * Returns the default state for AssetsController.
35
35
  *
36
- * @returns The default AssetsController state with empty metadata, balance, and customAssets maps.
36
+ * @returns The default AssetsController state with empty metadata, balance, price, and customAssets maps.
37
37
  */
38
38
  function getDefaultAssetsControllerState() {
39
39
  return {
40
40
  assetsMetadata: {},
41
41
  assetsBalance: {},
42
+ assetsPrice: {},
42
43
  customAssets: {},
43
44
  };
44
45
  }
@@ -59,6 +60,12 @@ const stateMetadata = {
59
60
  includeInDebugSnapshot: false,
60
61
  usedInUi: true,
61
62
  },
63
+ assetsPrice: {
64
+ persist: false,
65
+ includeInStateLogs: false,
66
+ includeInDebugSnapshot: false,
67
+ usedInUi: true,
68
+ },
62
69
  customAssets: {
63
70
  persist: true,
64
71
  includeInStateLogs: false,
@@ -177,8 +184,6 @@ class AssetsController extends base_controller_1.BaseController {
177
184
  _AssetsController_activeSubscriptions.set(this, new Map());
178
185
  /** Currently enabled chains from NetworkEnablementController */
179
186
  _AssetsController_enabledChains.set(this, new Set());
180
- /** Price data for assets (in-memory only, not persisted) */
181
- _AssetsController_assetsPrice.set(this, {});
182
187
  /**
183
188
  * Registered data sources with their available chains.
184
189
  * Updated continuously and independently from subscription flows.
@@ -492,7 +497,7 @@ class AssetsController extends base_controller_1.BaseController {
492
497
  }
493
498
  }
494
499
  exports.AssetsController = AssetsController;
495
- _AssetsController_defaultUpdateInterval = new WeakMap(), _AssetsController_controllerMutex = new WeakMap(), _AssetsController_activeSubscriptions = new WeakMap(), _AssetsController_enabledChains = new WeakMap(), _AssetsController_assetsPrice = new WeakMap(), _AssetsController_dataSources = new WeakMap(), _AssetsController_instances = new WeakSet(), _AssetsController_selectedAccounts_get = function _AssetsController_selectedAccounts_get() {
500
+ _AssetsController_defaultUpdateInterval = new WeakMap(), _AssetsController_controllerMutex = new WeakMap(), _AssetsController_activeSubscriptions = new WeakMap(), _AssetsController_enabledChains = new WeakMap(), _AssetsController_dataSources = new WeakMap(), _AssetsController_instances = new WeakSet(), _AssetsController_selectedAccounts_get = function _AssetsController_selectedAccounts_get() {
496
501
  return this.messenger.call('AccountTreeController:getAccountsFromSelectedAccountGroup');
497
502
  }, _AssetsController_initializeState = function _AssetsController_initializeState() {
498
503
  const { enabledNetworkMap } = this.messenger.call('NetworkEnablementController:getState');
@@ -615,15 +620,9 @@ async function _AssetsController_updateState(response) {
615
620
  const releaseLock = await __classPrivateFieldGet(this, _AssetsController_controllerMutex, "f").acquire();
616
621
  try {
617
622
  const previousState = this.state;
618
- const previousPrices = { ...__classPrivateFieldGet(this, _AssetsController_assetsPrice, "f") };
623
+ const previousPrices = { ...this.state.assetsPrice };
619
624
  // Use detectedAssets from response (assets without metadata)
620
625
  const detectedAssets = normalizedResponse.detectedAssets ?? {};
621
- // Update prices in memory (not persisted in state)
622
- if (normalizedResponse.assetsPrice) {
623
- for (const [key, value] of Object.entries(normalizedResponse.assetsPrice)) {
624
- __classPrivateFieldGet(this, _AssetsController_assetsPrice, "f")[key] = value;
625
- }
626
- }
627
626
  // Track actual changes for logging
628
627
  const changedBalances = [];
629
628
  const changedMetadata = [];
@@ -631,6 +630,7 @@ async function _AssetsController_updateState(response) {
631
630
  // Use type assertions to avoid deep type instantiation issues with Draft<Json>
632
631
  const metadata = state.assetsMetadata;
633
632
  const balances = state.assetsBalance;
633
+ const prices = state.assetsPrice;
634
634
  if (normalizedResponse.assetsMetadata) {
635
635
  for (const [key, value] of Object.entries(normalizedResponse.assetsMetadata)) {
636
636
  if (!(0, lodash_1.isEqual)(previousState.assetsMetadata[key], value)) {
@@ -663,6 +663,12 @@ async function _AssetsController_updateState(response) {
663
663
  Object.assign(balances[accountId], accountBalances);
664
664
  }
665
665
  }
666
+ // Update prices in state
667
+ if (normalizedResponse.assetsPrice) {
668
+ for (const [key, value] of Object.entries(normalizedResponse.assetsPrice)) {
669
+ prices[key] = value;
670
+ }
671
+ }
666
672
  });
667
673
  // Calculate changed prices
668
674
  const changedPriceAssets = normalizedResponse.assetsPrice
@@ -748,16 +754,16 @@ async function _AssetsController_updateState(response) {
748
754
  continue;
749
755
  }
750
756
  const typedBalance = balance;
751
- const priceRaw = __classPrivateFieldGet(this, _AssetsController_assetsPrice, "f")[typedAssetId];
757
+ const priceRaw = this.state.assetsPrice[typedAssetId];
752
758
  const price = priceRaw ?? {
753
759
  price: 0,
754
760
  lastUpdated: 0,
755
761
  };
756
- // Compute fiat value using BigNumber for precision with large balances
762
+ // Compute fiat value using BigNumber for precision
763
+ // Note: typedBalance.amount is already in human-readable format (e.g., "1" for 1 ETH)
764
+ // so we do NOT divide by 10^decimals here
757
765
  const balanceAmount = new bignumber_js_1.default(typedBalance.amount || '0');
758
- const divisor = new bignumber_js_1.default(10).pow(metadata.decimals);
759
- const normalizedAmount = balanceAmount.dividedBy(divisor);
760
- const fiatValue = normalizedAmount
766
+ const fiatValue = balanceAmount
761
767
  .multipliedBy(price.price || 0)
762
768
  .toNumber();
763
769
  const asset = {
@@ -1005,21 +1011,9 @@ async function _AssetsController_handleAccountGroupChanged() {
1005
1011
  addedChains,
1006
1012
  removedChains,
1007
1013
  });
1008
- // Clean up state for disabled chains
1009
- if (removedChains.length > 0) {
1010
- const removedChainsSet = new Set(removedChains);
1011
- this.update((state) => {
1012
- const balances = state.assetsBalance;
1013
- for (const accountId of Object.keys(balances)) {
1014
- for (const assetId of Object.keys(balances[accountId])) {
1015
- const assetChainId = extractChainId(assetId);
1016
- if (removedChainsSet.has(assetChainId)) {
1017
- delete balances[accountId][assetId];
1018
- }
1019
- }
1020
- }
1021
- });
1022
- }
1014
+ // Note: We intentionally do NOT delete balance data for disabled chains.
1015
+ // Users may want to see historical balances even if the network is currently disabled.
1016
+ // The data will simply not be updated until the network is re-enabled.
1023
1017
  // Refresh subscriptions for new chain set
1024
1018
  __classPrivateFieldGet(this, _AssetsController_instances, "m", _AssetsController_subscribeToDataSources).call(this);
1025
1019
  // Do one-time fetch for newly enabled chains
@@ -1 +1 @@
1
- {"version":3,"file":"AssetsController.cjs","sourceRoot":"","sources":["../src/AssetsController.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n AccountTreeControllerGetAccountsFromSelectedAccountGroupAction,\n AccountTreeControllerSelectedAccountGroupChangeEvent,\n} from '@metamask/account-tree-controller';\nimport { BaseController } from '@metamask/base-controller';\nimport type {\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n StateMetadata,\n} from '@metamask/base-controller';\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 NetworkEnablementControllerGetStateAction,\n NetworkEnablementControllerEvents,\n NetworkEnablementControllerState,\n} from '@metamask/network-enablement-controller';\nimport type { Json } from '@metamask/utils';\nimport { parseCaipAssetType } from '@metamask/utils';\nimport { Mutex } from 'async-mutex';\nimport BigNumberJS from 'bignumber.js';\nimport { isEqual } from 'lodash';\n\nimport type { AccountsApiDataSourceGetAssetsMiddlewareAction } from './data-sources/AccountsApiDataSource';\nimport type {\n PriceDataSourceGetAssetsMiddlewareAction,\n PriceDataSourceFetchAction,\n PriceDataSourceSubscribeAction,\n PriceDataSourceUnsubscribeAction,\n} from './data-sources/PriceDataSource';\nimport type { RpcDataSourceGetAssetsMiddlewareAction } from './data-sources/RpcDataSource';\nimport type { SnapDataSourceGetAssetsMiddlewareAction } from './data-sources/SnapDataSource';\nimport type { TokenDataSourceGetAssetsMiddlewareAction } from './data-sources/TokenDataSource';\nimport { projectLogger, createModuleLogger } from './logger';\nimport type { DetectionMiddlewareGetAssetsMiddlewareAction } from './middlewares/DetectionMiddleware';\nimport type {\n AccountId,\n ChainId,\n Caip19AssetId,\n AssetMetadata,\n AssetPrice,\n AssetBalance,\n AssetType,\n DataType,\n DataRequest,\n DataResponse,\n NextFunction,\n Middleware,\n DataSourceDefinition,\n SubscriptionResponse,\n Asset,\n AssetsControllerStateInternal,\n} from './types';\nimport { normalizeAssetId } from './utils';\n\n// ============================================================================\n// CONTROLLER CONSTANTS\n// ============================================================================\n\nconst CONTROLLER_NAME = 'AssetsController' as const;\n\n/** Default polling interval hint for data sources (30 seconds) */\nconst DEFAULT_POLLING_INTERVAL_MS = 30_000;\n\nconst log = createModuleLogger(projectLogger, CONTROLLER_NAME);\n\n// ============================================================================\n// STATE TYPES\n// ============================================================================\n\n/**\n * State structure for AssetsController.\n *\n * All values are JSON-serializable. The type is widened to satisfy\n * StateConstraint from BaseController, but the actual runtime values\n * conform to AssetMetadata, AssetPrice, and AssetBalance interfaces.\n *\n * @see AssetsControllerStateInternal for the semantic type structure\n */\nexport type AssetsControllerState = {\n /** Shared metadata for all assets (stored once per asset) */\n assetsMetadata: { [assetId: string]: Json };\n /** Per-account balance data */\n assetsBalance: { [accountId: string]: { [assetId: string]: Json } };\n /** Custom assets added by users per account (CAIP-19 asset IDs) */\n customAssets: { [accountId: string]: string[] };\n};\n\n/**\n * Returns the default state for AssetsController.\n *\n * @returns The default AssetsController state with empty metadata, balance, and customAssets maps.\n */\nexport function getDefaultAssetsControllerState(): AssetsControllerState {\n return {\n assetsMetadata: {},\n assetsBalance: {},\n customAssets: {},\n };\n}\n\n// ============================================================================\n// MESSENGER TYPES\n// ============================================================================\n\nexport type AssetsControllerGetStateAction = ControllerGetStateAction<\n typeof CONTROLLER_NAME,\n AssetsControllerState\n>;\n\nexport type AssetsControllerGetAssetsAction = {\n type: `${typeof CONTROLLER_NAME}:getAssets`;\n handler: AssetsController['getAssets'];\n};\n\nexport type AssetsControllerGetAssetsBalanceAction = {\n type: `${typeof CONTROLLER_NAME}:getAssetsBalance`;\n handler: AssetsController['getAssetsBalance'];\n};\n\nexport type AssetsControllerGetAssetMetadataAction = {\n type: `${typeof CONTROLLER_NAME}:getAssetMetadata`;\n handler: AssetsController['getAssetMetadata'];\n};\n\nexport type AssetsControllerGetAssetsPriceAction = {\n type: `${typeof CONTROLLER_NAME}:getAssetsPrice`;\n handler: AssetsController['getAssetsPrice'];\n};\n\nexport type AssetsControllerActiveChainsUpdateAction = {\n type: `${typeof CONTROLLER_NAME}:activeChainsUpdate`;\n handler: AssetsController['handleActiveChainsUpdate'];\n};\n\nexport type AssetsControllerAssetsUpdateAction = {\n type: `${typeof CONTROLLER_NAME}:assetsUpdate`;\n handler: AssetsController['handleAssetsUpdate'];\n};\n\nexport type AssetsControllerAddCustomAssetAction = {\n type: `${typeof CONTROLLER_NAME}:addCustomAsset`;\n handler: AssetsController['addCustomAsset'];\n};\n\nexport type AssetsControllerRemoveCustomAssetAction = {\n type: `${typeof CONTROLLER_NAME}:removeCustomAsset`;\n handler: AssetsController['removeCustomAsset'];\n};\n\nexport type AssetsControllerGetCustomAssetsAction = {\n type: `${typeof CONTROLLER_NAME}:getCustomAssets`;\n handler: AssetsController['getCustomAssets'];\n};\n\nexport type AssetsControllerActions =\n | AssetsControllerGetStateAction\n | AssetsControllerGetAssetsAction\n | AssetsControllerGetAssetsBalanceAction\n | AssetsControllerGetAssetMetadataAction\n | AssetsControllerGetAssetsPriceAction\n | AssetsControllerActiveChainsUpdateAction\n | AssetsControllerAssetsUpdateAction\n | AssetsControllerAddCustomAssetAction\n | AssetsControllerRemoveCustomAssetAction\n | AssetsControllerGetCustomAssetsAction;\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 | AccountTreeControllerGetAccountsFromSelectedAccountGroupAction\n | NetworkEnablementControllerGetStateAction\n // Data source middlewares\n | AccountsApiDataSourceGetAssetsMiddlewareAction\n | SnapDataSourceGetAssetsMiddlewareAction\n | RpcDataSourceGetAssetsMiddlewareAction\n // Enrichment middlewares\n | TokenDataSourceGetAssetsMiddlewareAction\n | PriceDataSourceGetAssetsMiddlewareAction\n | PriceDataSourceFetchAction\n | PriceDataSourceSubscribeAction\n | PriceDataSourceUnsubscribeAction\n | DetectionMiddlewareGetAssetsMiddlewareAction;\n\n/**\n * App lifecycle event: fired when app becomes active (opened/foregrounded)\n */\nexport type AppStateControllerAppOpenedEvent = {\n type: 'AppStateController:appOpened';\n payload: [];\n};\n\n/**\n * App lifecycle event: fired when app becomes inactive (closed/backgrounded)\n */\nexport type AppStateControllerAppClosedEvent = {\n type: 'AppStateController:appClosed';\n payload: [];\n};\n\ntype AllowedEvents =\n | AccountTreeControllerSelectedAccountGroupChangeEvent\n | NetworkEnablementControllerEvents\n | AppStateControllerAppOpenedEvent\n | AppStateControllerAppClosedEvent\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent;\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};\n\n// ============================================================================\n// STATE METADATA\n// ============================================================================\n\nconst stateMetadata: StateMetadata<AssetsControllerState> = {\n assetsMetadata: {\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 customAssets: {\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.assetsMetadata) {\n normalized.assetsMetadata = {};\n for (const [assetId, metadata] of Object.entries(response.assetsMetadata)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsMetadata[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 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. **App Lifecycle Management**: Listens to app open/close events via messenger\n * to start/stop subscriptions automatically, conserving resources when app is closed.\n *\n * ## App Lifecycle\n *\n * - **App Opened** (`AppStateController:appOpened`): Starts subscriptions, fetches initial data\n * - **App Closed** (`AppStateController:appClosed`): Stops all subscriptions to conserve resources\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 /** Default update interval hint passed to data sources */\n readonly #defaultUpdateInterval: number;\n\n readonly #controllerMutex = new Mutex();\n\n /**\n * Active balance subscriptions keyed by account ID.\n * Each account has one logical subscription that may span multiple data sources.\n * For example, if WebSocket covers chains A,B and RPC covers chain C,\n * the account subscribes to both data sources for its chains.\n */\n readonly #activeSubscriptions: Map<string, SubscriptionResponse> = new Map();\n\n /** Currently enabled chains from NetworkEnablementController */\n #enabledChains: Set<ChainId> = new Set();\n\n /**\n * Get the currently selected accounts from AccountTreeController.\n * This includes all accounts in the same group as the selected account\n * (EVM, Bitcoin, Solana, Tron, etc. that belong to the same logical account group).\n *\n * @returns Array of InternalAccount objects from the selected account group.\n */\n get #selectedAccounts(): InternalAccount[] {\n return this.messenger.call(\n 'AccountTreeController:getAccountsFromSelectedAccountGroup',\n );\n }\n\n /** Price data for assets (in-memory only, not persisted) */\n #assetsPrice: Record<Caip19AssetId, AssetPrice> = {};\n\n /**\n * Registered data sources with their available chains.\n * Updated continuously and independently from subscription flows.\n * Key: sourceId, Value: Set of currently available chainIds\n */\n readonly #dataSources: Map<string, Set<ChainId>> = new Map();\n\n constructor({\n messenger,\n state = {},\n defaultUpdateInterval = DEFAULT_POLLING_INTERVAL_MS,\n }: AssetsControllerOptions) {\n super({\n name: CONTROLLER_NAME,\n messenger,\n metadata: stateMetadata,\n state: {\n ...getDefaultAssetsControllerState(),\n ...state,\n },\n });\n\n this.#defaultUpdateInterval = defaultUpdateInterval;\n\n log('Initializing AssetsController', {\n defaultUpdateInterval,\n });\n\n this.#initializeState();\n this.#subscribeToEvents();\n this.#registerActionHandlers();\n\n // Register data sources (order = subscription priority)\n this.registerDataSources([\n 'BackendWebsocketDataSource', // Real-time push updates\n 'AccountsApiDataSource', // HTTP polling fallback\n 'SnapDataSource', // Solana/Bitcoin/Tron snaps\n 'RpcDataSource', // Direct blockchain queries\n ]);\n }\n\n // ============================================================================\n // INITIALIZATION\n // ============================================================================\n\n #initializeState(): void {\n const { enabledNetworkMap } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n this.#enabledChains = this.#extractEnabledChains(enabledNetworkMap);\n\n log('Initialized state', {\n enabledNetworkMap,\n enabledChains: this.#enabledChains,\n });\n }\n\n /**\n * Extract enabled chains from enabledNetworkMap.\n * Returns CAIP-2 chain IDs for all enabled networks across all namespaces.\n *\n * Note: For EIP155 (EVM) chains, the reference is normalized to decimal format\n * to ensure consistency with CAIP-2 standard and API responses.\n *\n * @param enabledNetworkMap - The enabled network map from NetworkEnablementController.\n * @returns Set of CAIP-2 chain IDs for all enabled networks.\n */\n #extractEnabledChains(\n enabledNetworkMap: NetworkEnablementControllerState['enabledNetworkMap'],\n ): Set<ChainId> {\n const chains = new Set<ChainId>();\n\n for (const [namespace, networks] of Object.entries(enabledNetworkMap)) {\n for (const [reference, isEnabled] of Object.entries(networks)) {\n if (isEnabled) {\n // Check if reference is already a full CAIP-2 chain ID (contains colon)\n if (reference.includes(':')) {\n // Already a full chain ID, use as-is\n chains.add(reference as ChainId);\n } else {\n // Normalize EIP155 chain references from hex to decimal (CAIP-2 standard)\n const normalizedReference = this.#normalizeChainReference(\n namespace,\n reference,\n );\n chains.add(`${namespace}:${normalizedReference}`);\n }\n }\n }\n }\n return chains;\n }\n\n /**\n * Normalize chain reference to CAIP-2 standard format.\n * For EIP155, converts hex chain IDs to decimal.\n *\n * @param namespace - The chain namespace (e.g., \"eip155\").\n * @param reference - The chain reference (e.g., \"0x1\" or \"1\").\n * @returns The normalized chain reference in decimal format.\n */\n #normalizeChainReference(namespace: string, reference: string): string {\n if (namespace === 'eip155' && reference.startsWith('0x')) {\n // Convert hex to decimal for EIP155 chains\n return parseInt(reference, 16).toString();\n }\n return reference;\n }\n\n #subscribeToEvents(): void {\n // Subscribe to account group changes (when user switches between account groups like Account 1 -> Account 2)\n this.messenger.subscribe(\n 'AccountTreeController:selectedAccountGroupChange',\n () => {\n this.#handleAccountGroupChanged().catch(console.error);\n },\n );\n\n // Subscribe to network enablement changes (only enabledNetworkMap)\n this.messenger.subscribe(\n 'NetworkEnablementController:stateChange',\n ({ enabledNetworkMap }) => {\n this.#handleEnabledNetworksChanged(enabledNetworkMap).catch(\n console.error,\n );\n },\n );\n\n // App lifecycle: start when opened, stop when closed\n this.messenger.subscribe('AppStateController:appOpened', () =>\n this.#start(),\n );\n this.messenger.subscribe('AppStateController:appClosed', () =>\n this.#stop(),\n );\n\n // Keyring lifecycle: start when unlocked, stop when locked\n this.messenger.subscribe('KeyringController:unlock', () => this.#start());\n this.messenger.subscribe('KeyringController:lock', () => this.#stop());\n }\n\n #registerActionHandlers(): void {\n this.messenger.registerActionHandler(\n 'AssetsController:getAssets',\n this.getAssets.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getAssetsBalance',\n this.getAssetsBalance.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getAssetMetadata',\n this.getAssetMetadata.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getAssetsPrice',\n this.getAssetsPrice.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:activeChainsUpdate',\n this.handleActiveChainsUpdate.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:assetsUpdate',\n this.handleAssetsUpdate.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:addCustomAsset',\n this.addCustomAsset.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:removeCustomAsset',\n this.removeCustomAsset.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getCustomAssets',\n this.getCustomAssets.bind(this),\n );\n }\n\n // ============================================================================\n // DATA SOURCE MANAGEMENT\n // ============================================================================\n\n /**\n * Register data sources with the controller.\n * Order of the array determines subscription order.\n *\n * Data sources report chain changes by calling `AssetsController:activeChainsUpdate` action.\n *\n * @param dataSourceIds - Array of data source identifiers to register.\n */\n registerDataSources(dataSourceIds: DataSourceDefinition[]): void {\n for (const id of dataSourceIds) {\n log('Registering data source', { id });\n\n // Initialize available chains tracking for this source\n this.#dataSources.set(id, new Set());\n }\n }\n\n // ============================================================================\n // DATA SOURCE CHAIN MANAGEMENT\n // ============================================================================\n\n /**\n * Handle when a data source's active chains change.\n * Active chains are chains that are both supported AND available.\n * Updates centralized chain tracking and triggers re-selection if needed.\n *\n * Data sources should call this via `AssetsController:activeChainsUpdate` action.\n *\n * @param dataSourceId - The identifier of the data source reporting the change.\n * @param activeChains - Array of currently active chain IDs for this source.\n */\n handleActiveChainsUpdate(\n dataSourceId: string,\n activeChains: ChainId[],\n ): void {\n log('Data source active chains changed', {\n dataSourceId,\n chainCount: activeChains.length,\n chains: activeChains,\n });\n\n const previousChains = this.#dataSources.get(dataSourceId) ?? new Set();\n const newChains = new Set(activeChains);\n\n // Update centralized available chains tracking\n this.#dataSources.set(dataSourceId, newChains);\n\n // Check for changes\n const addedChains = activeChains.filter(\n (chain) => !previousChains.has(chain),\n );\n const removedChains = Array.from(previousChains).filter(\n (chain) => !newChains.has(chain),\n );\n\n if (addedChains.length > 0 || removedChains.length > 0) {\n // Refresh subscriptions to use updated data source availability\n this.#subscribeToDataSources();\n }\n\n // If chains were added and we have selected accounts, do one-time fetch\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n const addedEnabledChains = addedChains.filter((chain) =>\n this.#enabledChains.has(chain),\n );\n if (addedEnabledChains.length > 0) {\n log('Fetching balances for newly added chains', { addedEnabledChains });\n this.getAssets(this.#selectedAccounts, {\n chainIds: addedEnabledChains,\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch balance for added chains', { error });\n });\n }\n }\n }\n\n // ============================================================================\n // MIDDLEWARE EXECUTION\n // ============================================================================\n\n /**\n * Execute middlewares with request/response context.\n *\n * @param middlewares - Middlewares to execute in order.\n * @param request - The data request.\n * @param initialResponse - Optional initial response (for enriching existing data).\n * @returns The final DataResponse after all middlewares have processed.\n */\n async #executeMiddlewares(\n middlewares: Middleware[],\n request: DataRequest,\n initialResponse: DataResponse = {},\n ): Promise<DataResponse> {\n const chain = middlewares.reduceRight<NextFunction>(\n (next, middleware) =>\n async (\n ctx,\n ): Promise<{\n request: DataRequest;\n response: DataResponse;\n getAssetsState: () => AssetsControllerStateInternal;\n }> => {\n try {\n return await middleware(ctx, next);\n } catch (error) {\n console.error('[AssetsController] Middleware failed:', error);\n return next(ctx);\n }\n },\n async (ctx) => ctx,\n );\n\n const result = await chain({\n request,\n response: initialResponse,\n getAssetsState: () => this.state as AssetsControllerStateInternal,\n });\n return result.response;\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 },\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 // 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 response = await this.#executeMiddlewares(\n [\n this.messenger.call('AccountsApiDataSource:getAssetsMiddleware'),\n this.messenger.call('SnapDataSource:getAssetsMiddleware'),\n this.messenger.call('RpcDataSource:getAssetsMiddleware'),\n this.messenger.call('DetectionMiddleware:getAssetsMiddleware'),\n this.messenger.call('TokenDataSource:getAssetsMiddleware'),\n this.messenger.call('PriceDataSource:getAssetsMiddleware'),\n ],\n {\n accounts,\n chainIds,\n assetTypes,\n dataTypes,\n customAssets: customAssets.length > 0 ? customAssets : undefined,\n forceUpdate: true,\n },\n );\n await this.#updateState(response);\n }\n\n return this.#getAssetsFromState(accounts, chainIds, assetTypes);\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.assetsMetadata[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 // 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 *\n * @param accountId - The account ID to add the custom asset for.\n * @param assetId - The CAIP-19 asset ID to add.\n */\n async addCustomAsset(\n accountId: AccountId,\n assetId: Caip19AssetId,\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\n // Fetch data for the newly added custom asset\n const account = this.#selectedAccounts.find((a) => a.id === accountId);\n if (account) {\n const chainId = extractChainId(normalizedAssetId);\n await this.getAssets([account], {\n chainIds: [chainId],\n forceUpdate: true,\n });\n }\n }\n\n /**\n * Remove a custom asset from an account.\n *\n * @param accountId - The account ID to remove the custom asset from.\n * @param assetId - The CAIP-19 asset ID to remove.\n */\n removeCustomAsset(accountId: AccountId, assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Removing custom asset', { accountId, assetId: normalizedAssetId });\n\n this.update((state) => {\n const customAssets = state.customAssets as Record<string, string[]>;\n if (customAssets[accountId]) {\n customAssets[accountId] = customAssets[accountId].filter(\n (id) => id !== normalizedAssetId,\n );\n\n // Clean up empty arrays\n if (customAssets[accountId].length === 0) {\n delete customAssets[accountId];\n }\n }\n });\n }\n\n /**\n * Get all custom assets for an account.\n *\n * @param accountId - The account ID to get custom assets for.\n * @returns Array of CAIP-19 asset IDs for the account's custom assets.\n */\n getCustomAssets(accountId: AccountId): Caip19AssetId[] {\n return (this.state.customAssets[accountId] ?? []) as Caip19AssetId[];\n }\n\n // ============================================================================\n // SUBSCRIPTIONS\n // ============================================================================\n\n /**\n * Assign chains to data sources based on availability.\n * Returns a map of sourceId -> chains to handle.\n *\n * @param requestedChains - Array of chain IDs to assign to data sources.\n * @returns Map of sourceId to array of assigned chain IDs.\n */\n #assignChainsToDataSources(\n requestedChains: ChainId[],\n ): Map<string, ChainId[]> {\n const assignment = new Map<string, ChainId[]>();\n const remainingChains = new Set(requestedChains);\n\n for (const sourceId of this.#dataSources.keys()) {\n // Get available chains for this data source\n const availableChains = this.#dataSources.get(sourceId);\n if (!availableChains || availableChains.size === 0) {\n continue;\n }\n\n const chainsForThisSource: ChainId[] = [];\n\n for (const chainId of remainingChains) {\n // Check if this chain is available on this source\n if (availableChains.has(chainId)) {\n chainsForThisSource.push(chainId);\n remainingChains.delete(chainId);\n }\n }\n\n if (chainsForThisSource.length > 0) {\n assignment.set(sourceId, chainsForThisSource);\n log('Assigned chains to data source', {\n sourceId,\n chains: chainsForThisSource,\n });\n }\n }\n\n return assignment;\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 * @param options - Subscription options.\n * @param options.updateInterval - Polling interval in ms.\n */\n subscribeAssetsPrice(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n options: { updateInterval?: number } = {},\n ): void {\n const { updateInterval = this.#defaultUpdateInterval } = options;\n const subscriptionKey = 'ds:PriceDataSource';\n\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n // Fire-and-forget - errors are handled internally by PriceDataSource\n this.messenger\n .call('PriceDataSource:subscribe', {\n request: {\n accounts,\n chainIds,\n dataTypes: ['price'],\n updateInterval,\n },\n subscriptionId: subscriptionKey,\n isUpdate,\n })\n .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\n // Fire-and-forget - errors are handled internally by PriceDataSource\n this.messenger\n .call('PriceDataSource:unsubscribe', subscriptionKey)\n .catch(console.error);\n\n existingSubscription.unsubscribe();\n }\n\n // ============================================================================\n // STATE MANAGEMENT\n // ============================================================================\n\n async #updateState(response: DataResponse): Promise<void> {\n // Normalize asset IDs (checksum EVM addresses) before storing in state\n const normalizedResponse = normalizeResponse(response);\n\n const releaseLock = await this.#controllerMutex.acquire();\n\n try {\n const previousState = this.state;\n const previousPrices = { ...this.#assetsPrice };\n // Use detectedAssets from response (assets without metadata)\n const detectedAssets: Record<AccountId, Caip19AssetId[]> =\n normalizedResponse.detectedAssets ?? {};\n\n // Update prices in memory (not persisted in state)\n if (normalizedResponse.assetsPrice) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsPrice,\n )) {\n this.#assetsPrice[key as Caip19AssetId] = value;\n }\n }\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 Draft<Json>\n const metadata = state.assetsMetadata as unknown as Record<\n string,\n unknown\n >;\n const balances = state.assetsBalance as unknown as Record<\n string,\n Record<string, unknown>\n >;\n\n if (normalizedResponse.assetsMetadata) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsMetadata,\n )) {\n if (\n !isEqual(\n previousState.assetsMetadata[key as Caip19AssetId],\n value,\n )\n ) {\n changedMetadata.push(key);\n }\n metadata[key] = value;\n }\n }\n\n if (normalizedResponse.assetsBalance) {\n for (const [accountId, accountBalances] of Object.entries(\n normalizedResponse.assetsBalance,\n )) {\n const previousBalances =\n previousState.assetsBalance[accountId] ?? {};\n\n if (!balances[accountId]) {\n balances[accountId] = {};\n }\n\n for (const [assetId, balance] of Object.entries(accountBalances)) {\n const previousBalance = previousBalances[\n assetId as Caip19AssetId\n ] as { amount: string } | undefined;\n const balanceData = balance as { amount: string };\n const newAmount = balanceData.amount;\n const oldAmount = previousBalance?.amount;\n\n // Track if balance actually changed\n if (oldAmount !== newAmount) {\n changedBalances.push({\n accountId,\n assetId,\n oldAmount,\n newAmount,\n });\n }\n }\n\n Object.assign(balances[accountId], accountBalances);\n }\n }\n });\n\n // Calculate changed prices\n const changedPriceAssets: string[] = normalizedResponse.assetsPrice\n ? Object.keys(normalizedResponse.assetsPrice).filter(\n (assetId) =>\n !isEqual(\n previousPrices[assetId as Caip19AssetId],\n normalizedResponse.assetsPrice?.[assetId as Caip19AssetId],\n ),\n )\n : [];\n\n // Log only actual changes\n if (\n changedBalances.length > 0 ||\n changedMetadata.length > 0 ||\n changedPriceAssets.length > 0\n ) {\n log('State updated', {\n changedBalances:\n changedBalances.length > 0 ? changedBalances : undefined,\n changedMetadataCount:\n changedMetadata.length > 0 ? changedMetadata.length : undefined,\n changedPricesCount:\n changedPriceAssets.length > 0\n ? changedPriceAssets.length\n : undefined,\n newAssets:\n Object.keys(detectedAssets).length > 0\n ? Object.entries(detectedAssets).map(([accountId, assets]) => ({\n accountId,\n assets,\n }))\n : undefined,\n });\n }\n\n // Publish balance changed events\n for (const change of changedBalances) {\n this.messenger.publish('AssetsController:balanceChanged', {\n accountId: change.accountId,\n assetId: change.assetId as Caip19AssetId,\n previousAmount: change.oldAmount ?? '0',\n newAmount: change.newAmount,\n });\n }\n\n // Publish price changed event with full price data\n if (changedPriceAssets.length > 0 && normalizedResponse.assetsPrice) {\n // Build prices object with only changed prices\n const changedPrices: Record<Caip19AssetId, AssetPrice> = {};\n for (const assetId of changedPriceAssets) {\n const price =\n normalizedResponse.assetsPrice[assetId as Caip19AssetId];\n if (price) {\n changedPrices[assetId as Caip19AssetId] = price;\n }\n }\n this.messenger.publish('AssetsController:priceChanged', {\n prices: changedPrices,\n });\n }\n\n // Publish assets detected events\n for (const [accountId, assetIds] of Object.entries(detectedAssets)) {\n if (assetIds.length > 0) {\n this.messenger.publish('AssetsController:assetsDetected', {\n accountId,\n assetIds,\n });\n }\n }\n } finally {\n releaseLock();\n }\n }\n\n #getAssetsFromState(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n assetTypes: AssetType[],\n ): Record<AccountId, Record<Caip19AssetId, Asset>> {\n const result: Record<AccountId, Record<Caip19AssetId, Asset>> = {};\n // Convert to Sets for O(1) lookups\n const chainIdSet = new Set(chainIds);\n const assetTypeSet = new Set(assetTypes);\n\n for (const account of accounts) {\n result[account.id] = {};\n\n const accountBalances = this.state.assetsBalance[account.id] ?? {};\n\n for (const [assetId, balance] of Object.entries(accountBalances)) {\n const typedAssetId = assetId as Caip19AssetId;\n const assetChainId = extractChainId(typedAssetId);\n\n if (!chainIdSet.has(assetChainId)) {\n continue;\n }\n\n const metadataRaw = this.state.assetsMetadata[typedAssetId];\n\n // Skip assets without metadata\n if (!metadataRaw) {\n continue;\n }\n\n const metadata = metadataRaw as AssetMetadata;\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 as AssetBalance;\n const priceRaw = this.#assetsPrice[typedAssetId];\n const price: AssetPrice = priceRaw ?? {\n price: 0,\n lastUpdated: 0,\n };\n\n // Compute fiat value using BigNumber for precision with large balances\n const balanceAmount = new BigNumberJS(typedBalance.amount || '0');\n const divisor = new BigNumberJS(10).pow(metadata.decimals);\n const normalizedAmount = balanceAmount.dividedBy(divisor);\n const fiatValue = normalizedAmount\n .multipliedBy(price.price || 0)\n .toNumber();\n\n const asset: Asset = {\n id: typedAssetId,\n chainId: assetChainId,\n balance: typedBalance,\n metadata,\n price,\n fiatValue,\n };\n\n result[account.id][typedAssetId] = asset;\n }\n }\n\n return result;\n }\n\n /**\n * Maps a token standard to its corresponding asset type.\n *\n * @param tokenStandard - The token standard from metadata.\n * @returns The corresponding asset type.\n */\n #tokenStandardToAssetType(tokenStandard: string): AssetType {\n switch (tokenStandard) {\n case 'native':\n case 'erc20':\n case 'spl':\n return 'fungible';\n case 'erc721':\n return 'nft';\n case 'erc1155':\n // ERC1155 can be either fungible or non-fungible, treat as collectible\n return 'collectible';\n default:\n // Unknown standards default to fungible\n return 'fungible';\n }\n }\n\n // ============================================================================\n // START / STOP\n // ============================================================================\n\n /**\n * Start asset tracking: subscribe to updates and fetch current balances.\n * Called when app opens, account changes, or keyring unlocks.\n */\n #start(): void {\n log('Starting asset tracking', {\n selectedAccountCount: this.#selectedAccounts.length,\n enabledChainCount: this.#enabledChains.size,\n });\n\n this.#subscribeToDataSources();\n if (this.#selectedAccounts.length > 0) {\n this.getAssets(this.#selectedAccounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch assets', error);\n });\n }\n }\n\n /**\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 // 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 // Convert to array first to avoid modifying map during iteration\n const subscriptionKeys = [...this.#activeSubscriptions.keys()];\n for (const subscriptionKey of subscriptionKeys) {\n // Extract sourceId from subscription key (format: \"ds:${sourceId}\")\n if (subscriptionKey.startsWith('ds:')) {\n const sourceId = subscriptionKey.slice(3);\n this.#unsubscribeDataSource(sourceId);\n }\n }\n this.#activeSubscriptions.clear();\n }\n\n /**\n * Subscribe to asset updates for all selected accounts.\n */\n #subscribeToDataSources(): void {\n if (this.#selectedAccounts.length === 0) {\n return;\n }\n\n // Subscribe to balance updates (batched by data source)\n this.#subscribeAssetsBalance();\n\n // Subscribe to price updates for all assets held by selected accounts\n this.subscribeAssetsPrice(this.#selectedAccounts, [...this.#enabledChains]);\n }\n\n /**\n * Subscribe to balance updates for all selected accounts.\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 #subscribeAssetsBalance(): void {\n // Step 1: Build chain -> accounts mapping based on account scopes and enabled networks\n const chainToAccounts = this.#buildChainToAccountsMap(\n this.#selectedAccounts,\n this.#enabledChains,\n );\n\n // Step 2: Split by data source active chains (ordered by priority)\n // Get all chains that need to be subscribed\n const remainingChains = new Set(chainToAccounts.keys());\n\n // Assign chains to data sources based on availability (ordered by priority)\n const chainAssignment = this.#assignChainsToDataSources(\n Array.from(remainingChains),\n );\n\n log('Subscribe - chain assignment', {\n totalChains: remainingChains.size,\n dataSourceAssignments: Array.from(chainAssignment.entries()).map(\n ([sourceId, chains]) => ({ sourceId, chainCount: chains.length }),\n ),\n });\n\n // Subscribe to each data source with its assigned chains and relevant accounts\n for (const sourceId of this.#dataSources.keys()) {\n const assignedChains = chainAssignment.get(sourceId);\n\n if (!assignedChains || assignedChains.length === 0) {\n // Unsubscribe from data sources with no assigned chains\n this.#unsubscribeDataSource(sourceId);\n continue;\n }\n\n // Collect unique accounts that need any of the assigned chains\n const accountsForSource = this.#getAccountsForChains(\n assignedChains,\n chainToAccounts,\n );\n\n if (accountsForSource.length === 0) {\n continue;\n }\n\n // Subscribe with ONE call per data source\n this.#subscribeToDataSource(sourceId, accountsForSource, assignedChains);\n }\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\n for (const account of accounts) {\n const accountChains = this.#getEnabledChainsForAccount(account);\n\n for (const chainId of accountChains) {\n if (!chainsToSubscribe.has(chainId)) {\n continue;\n }\n\n const existingAccounts = chainToAccounts.get(chainId) ?? [];\n existingAccounts.push(account);\n chainToAccounts.set(chainId, existingAccounts);\n }\n }\n\n return chainToAccounts;\n }\n\n /**\n * Get unique accounts that need any of the specified chains.\n *\n * @param chains - Array of chain IDs to find accounts for.\n * @param chainToAccounts - Map of chainId to accounts.\n * @returns Array of unique accounts that need any of the specified chains.\n */\n #getAccountsForChains(\n chains: ChainId[],\n chainToAccounts: Map<ChainId, InternalAccount[]>,\n ): InternalAccount[] {\n const accountIds = new Set<string>();\n const accounts: InternalAccount[] = [];\n\n for (const chainId of chains) {\n const chainAccounts = chainToAccounts.get(chainId) ?? [];\n for (const account of chainAccounts) {\n if (!accountIds.has(account.id)) {\n accountIds.add(account.id);\n accounts.push(account);\n }\n }\n }\n\n return accounts;\n }\n\n /**\n * Subscribe to a specific data source with accounts and chains.\n * Uses the data source ID as the subscription key for batching.\n *\n * @param sourceId - The data source identifier.\n * @param accounts - Array of accounts to subscribe for.\n * @param chains - Array of chain IDs to subscribe for.\n */\n #subscribeToDataSource(\n sourceId: string,\n accounts: InternalAccount[],\n chains: ChainId[],\n ): void {\n const subscriptionKey = `ds:${sourceId}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n log('Subscribe to data source', {\n sourceId,\n subscriptionKey,\n isUpdate,\n accountCount: accounts.length,\n chainCount: chains.length,\n });\n\n // Call data source subscribe action via Messenger (fire-and-forget)\n (async (): Promise<void> => {\n try {\n await (this.messenger.call as CallableFunction)(\n `${sourceId}:subscribe`,\n {\n request: {\n accounts,\n chainIds: chains,\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n updateInterval: this.#defaultUpdateInterval,\n },\n subscriptionId: subscriptionKey,\n isUpdate,\n },\n );\n } catch (error) {\n console.error(\n `[AssetsController] Failed to subscribe to '${sourceId}':`,\n error,\n );\n }\n })().catch(console.error);\n\n // Track subscription\n const subscription: SubscriptionResponse = {\n chains,\n accountId: subscriptionKey,\n assetTypes: ['fungible'],\n dataTypes: ['balance', 'price'],\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 sourceId - The data source identifier to unsubscribe from.\n */\n #unsubscribeDataSource(sourceId: string): void {\n const subscriptionKey = `ds:${sourceId}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n\n if (existingSubscription) {\n // Fire-and-forget unsubscribe call\n (async (): Promise<void> => {\n try {\n await (this.messenger.call as CallableFunction)(\n `${sourceId}:unsubscribe`,\n subscriptionKey,\n );\n } catch {\n // Ignore errors - source may not have been subscribed\n }\n })().catch(() => {\n // Ignore errors - source may not have been subscribed\n });\n existingSubscription.unsubscribe();\n }\n }\n\n // ============================================================================\n // HELPERS\n // ============================================================================\n\n /**\n * Get the chains that an account supports based on its scopes.\n * Returns the intersection of the account's scopes and the enabled chains.\n *\n * @param account - The account to get supported chains for.\n * @returns Array of ChainIds that the account supports and are enabled.\n */\n #getEnabledChainsForAccount(account: InternalAccount): ChainId[] {\n // Account scopes are CAIP-2 chain IDs like \"eip155:1\", \"solana:mainnet\", \"bip122:...\"\n const scopes = account.scopes ?? [];\n const result: ChainId[] = [];\n\n for (const scope of scopes) {\n const [namespace, reference] = (scope as string).split(':');\n\n // Wildcard scope (e.g., \"eip155:0\" means all enabled chains in that namespace)\n if (reference === '0') {\n for (const chain of this.#enabledChains) {\n if (chain.startsWith(`${namespace}:`)) {\n result.push(chain);\n }\n }\n } else if (namespace === 'eip155' && reference?.startsWith('0x')) {\n // Normalize hex to decimal for EIP155\n result.push(`eip155:${parseInt(reference, 16)}` as ChainId);\n } else {\n result.push(scope);\n }\n }\n\n return result;\n }\n\n // ============================================================================\n // EVENT HANDLERS\n // ============================================================================\n\n async #handleAccountGroupChanged(): Promise<void> {\n const accounts = this.#selectedAccounts;\n\n log('Account group changed', {\n accountCount: accounts.length,\n accountIds: accounts.map((a) => a.id),\n });\n\n // Subscribe and fetch for the new account group\n this.#subscribeToDataSources();\n if (accounts.length > 0) {\n await this.getAssets(accounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n });\n }\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 // Clean up state for disabled chains\n if (removedChains.length > 0) {\n const removedChainsSet = new Set(removedChains);\n this.update((state) => {\n const balances = state.assetsBalance as unknown as Record<\n string,\n Record<string, unknown>\n >;\n for (const accountId of Object.keys(balances)) {\n for (const assetId of Object.keys(balances[accountId])) {\n const assetChainId = extractChainId(assetId as Caip19AssetId);\n if (removedChainsSet.has(assetChainId)) {\n delete balances[accountId][assetId];\n }\n }\n }\n });\n }\n\n // Refresh subscriptions for new chain set\n this.#subscribeToDataSources();\n\n // Do one-time fetch for newly enabled chains\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n await this.getAssets(this.#selectedAccounts, {\n chainIds: addedChains,\n forceUpdate: true,\n });\n }\n }\n\n /**\n * Handle assets updated from a data source.\n * Called via `AssetsController:assetsUpdate` action by data sources.\n *\n * @param response - The data response with updated assets\n * @param sourceId - The data source ID reporting the update\n */\n async handleAssetsUpdate(\n response: DataResponse,\n sourceId: string,\n ): Promise<void> {\n log('Assets updated from data source', {\n sourceId,\n hasBalance: Boolean(response.assetsBalance),\n hasPrice: Boolean(response.assetsPrice),\n });\n await this.#handleSubscriptionUpdate(response, sourceId);\n }\n\n /**\n * Handle an async update from a data source subscription.\n * Enriches response with token metadata before updating state.\n *\n * @param response - The data response from the data source.\n * @param _sourceId - The source ID (unused but kept for logging context).\n * @param request - Optional original request for context.\n */\n async #handleSubscriptionUpdate(\n response: DataResponse,\n _sourceId?: string,\n request?: DataRequest,\n ): Promise<void> {\n // Run through enrichment middlewares (Event Stack: Detection → Token → Price)\n // Include 'metadata' in dataTypes so TokenDataSource runs to enrich detected assets\n const enrichedResponse = await this.#executeMiddlewares(\n [\n this.messenger.call('DetectionMiddleware:getAssetsMiddleware'),\n this.messenger.call('TokenDataSource:getAssetsMiddleware'),\n this.messenger.call('PriceDataSource:getAssetsMiddleware'),\n ],\n request ?? {\n accounts: [],\n chainIds: [],\n dataTypes: ['balance', 'metadata', 'price'],\n },\n response,\n );\n\n // Update state\n await this.#updateState(enrichedResponse);\n }\n\n // ============================================================================\n // CLEANUP\n // ============================================================================\n\n destroy(): void {\n log('Destroying AssetsController', {\n dataSourceCount: this.#dataSources.size,\n subscriptionCount: this.#activeSubscriptions.size,\n });\n\n // Clear data sources\n this.#dataSources.clear();\n\n // Stop all active subscriptions\n this.#stop();\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:activeChainsUpdate',\n );\n this.messenger.unregisterActionHandler('AssetsController:assetsUpdate');\n this.messenger.unregisterActionHandler('AssetsController:addCustomAsset');\n this.messenger.unregisterActionHandler(\n 'AssetsController:removeCustomAsset',\n );\n this.messenger.unregisterActionHandler('AssetsController:getCustomAssets');\n }\n}\n"]}
1
+ {"version":3,"file":"AssetsController.cjs","sourceRoot":"","sources":["../src/AssetsController.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;AAIA,+DAA2D;AAkB3D,2CAAqD;AACrD,6CAAoC;AACpC,gEAAuC;AACvC,mCAAiC;AAYjC,yCAA6D;AAoB7D,uCAA2C;AAE3C,+EAA+E;AAC/E,uBAAuB;AACvB,+EAA+E;AAE/E,MAAM,eAAe,GAAG,kBAA2B,CAAC;AAEpD,kEAAkE;AAClE,MAAM,2BAA2B,GAAG,KAAM,CAAC;AAE3C,MAAM,GAAG,GAAG,IAAA,2BAAkB,EAAC,sBAAa,EAAE,eAAe,CAAC,CAAC;AA0B/D;;;;GAIG;AACH,SAAgB,+BAA+B;IAC7C,OAAO;QACL,cAAc,EAAE,EAAE;QAClB,aAAa,EAAE,EAAE;QACjB,WAAW,EAAE,EAAE;QACf,YAAY,EAAE,EAAE;KACjB,CAAC;AACJ,CAAC;AAPD,0EAOC;AA6JD,+EAA+E;AAC/E,iBAAiB;AACjB,+EAA+E;AAE/E,MAAM,aAAa,GAAyC;IAC1D,cAAc,EAAE;QACd,OAAO,EAAE,IAAI;QACb,kBAAkB,EAAE,KAAK;QACzB,sBAAsB,EAAE,KAAK;QAC7B,QAAQ,EAAE,IAAI;KACf;IACD,aAAa,EAAE;QACb,OAAO,EAAE,IAAI;QACb,kBAAkB,EAAE,KAAK;QACzB,sBAAsB,EAAE,KAAK;QAC7B,QAAQ,EAAE,IAAI;KACf;IACD,WAAW,EAAE;QACX,OAAO,EAAE,KAAK;QACd,kBAAkB,EAAE,KAAK;QACzB,sBAAsB,EAAE,KAAK;QAC7B,QAAQ,EAAE,IAAI;KACf;IACD,YAAY,EAAE;QACZ,OAAO,EAAE,IAAI;QACb,kBAAkB,EAAE,KAAK;QACzB,sBAAsB,EAAE,KAAK;QAC7B,QAAQ,EAAE,IAAI;KACf;CACF,CAAC;AAEF,+EAA+E;AAC/E,mBAAmB;AACnB,+EAA+E;AAE/E,SAAS,cAAc,CAAC,OAAsB;IAC5C,MAAM,MAAM,GAAG,IAAA,0BAAkB,EAAC,OAAO,CAAC,CAAC;IAC3C,OAAO,MAAM,CAAC,OAAO,CAAC;AACxB,CAAC;AAED;;;;;;;GAOG;AACH,SAAS,iBAAiB,CAAC,QAAsB;IAC/C,MAAM,UAAU,GAAiB,EAAE,CAAC;IAEpC,IAAI,QAAQ,CAAC,cAAc,EAAE,CAAC;QAC5B,UAAU,CAAC,cAAc,GAAG,EAAE,CAAC;QAC/B,KAAK,MAAM,CAAC,OAAO,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAC,EAAE,CAAC;YAC1E,MAAM,YAAY,GAAG,IAAA,wBAAgB,EAAC,OAAwB,CAAC,CAAC;YAChE,UAAU,CAAC,cAAc,CAAC,YAAY,CAAC,GAAG,QAAQ,CAAC;QACrD,CAAC;IACH,CAAC;IAED,IAAI,QAAQ,CAAC,WAAW,EAAE,CAAC;QACzB,UAAU,CAAC,WAAW,GAAG,EAAE,CAAC;QAC5B,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;YACpE,MAAM,YAAY,GAAG,IAAA,wBAAgB,EAAC,OAAwB,CAAC,CAAC;YAChE,UAAU,CAAC,WAAW,CAAC,YAAY,CAAC,GAAG,KAAK,CAAC;QAC/C,CAAC;IACH,CAAC;IAED,IAAI,QAAQ,CAAC,aAAa,EAAE,CAAC;QAC3B,UAAU,CAAC,aAAa,GAAG,EAAE,CAAC;QAC9B,KAAK,MAAM,CAAC,SAAS,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAChD,QAAQ,CAAC,aAAa,CACvB,EAAE,CAAC;YACF,UAAU,CAAC,aAAa,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC;YACzC,KAAK,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;gBAC1D,MAAM,YAAY,GAAG,IAAA,wBAAgB,EAAC,OAAwB,CAAC,CAAC;gBAChE,UAAU,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC,YAAY,CAAC,GAAG,OAAO,CAAC;YAC9D,CAAC;QACH,CAAC;IACH,CAAC;IAED,oDAAoD;IACpD,IAAI,QAAQ,CAAC,cAAc,EAAE,CAAC;QAC5B,UAAU,CAAC,cAAc,GAAG,EAAE,CAAC;QAC/B,KAAK,MAAM,CAAC,SAAS,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAChD,QAAQ,CAAC,cAAc,CACxB,EAAE,CAAC;YACF,UAAU,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAC9D,IAAA,wBAAgB,EAAC,OAAO,CAAC,CAC1B,CAAC;QACJ,CAAC;IACH,CAAC;IAED,uDAAuD;IACvD,IAAI,QAAQ,CAAC,MAAM,EAAE,CAAC;QACpB,UAAU,CAAC,MAAM,GAAG,EAAE,GAAG,QAAQ,CAAC,MAAM,EAAE,CAAC;IAC7C,CAAC;IAED,OAAO,UAAU,CAAC;AACpB,CAAC;AAED,+EAA+E;AAC/E,4BAA4B;AAC5B,+EAA+E;AAE/E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BG;AACH,MAAa,gBAAiB,SAAQ,gCAIrC;IAqCC,YAAY,EACV,SAAS,EACT,KAAK,GAAG,EAAE,EACV,qBAAqB,GAAG,2BAA2B,GAC3B;QACxB,KAAK,CAAC;YACJ,IAAI,EAAE,eAAe;YACrB,SAAS;YACT,QAAQ,EAAE,aAAa;YACvB,KAAK,EAAE;gBACL,GAAG,+BAA+B,EAAE;gBACpC,GAAG,KAAK;aACT;SACF,CAAC,CAAC;;QAjDL,0DAA0D;QACjD,0DAA+B;QAE/B,4CAAmB,IAAI,mBAAK,EAAE,EAAC;QAExC;;;;;WAKG;QACM,gDAA0D,IAAI,GAAG,EAAE,EAAC;QAE7E,gEAAgE;QAChE,0CAA+B,IAAI,GAAG,EAAE,EAAC;QAezC;;;;WAIG;QACM,wCAA0C,IAAI,GAAG,EAAE,EAAC;QAiB3D,uBAAA,IAAI,2CAA0B,qBAAqB,MAAA,CAAC;QAEpD,GAAG,CAAC,+BAA+B,EAAE;YACnC,qBAAqB;SACtB,CAAC,CAAC;QAEH,uBAAA,IAAI,sEAAiB,MAArB,IAAI,CAAmB,CAAC;QACxB,uBAAA,IAAI,wEAAmB,MAAvB,IAAI,CAAqB,CAAC;QAC1B,uBAAA,IAAI,6EAAwB,MAA5B,IAAI,CAA0B,CAAC;QAE/B,wDAAwD;QACxD,IAAI,CAAC,mBAAmB,CAAC;YACvB,4BAA4B,EAAE,yBAAyB;YACvD,uBAAuB,EAAE,wBAAwB;YACjD,gBAAgB,EAAE,4BAA4B;YAC9C,eAAe,EAAE,4BAA4B;SAC9C,CAAC,CAAC;IACL,CAAC;IAqJD,+EAA+E;IAC/E,yBAAyB;IACzB,+EAA+E;IAE/E;;;;;;;OAOG;IACH,mBAAmB,CAAC,aAAqC;QACvD,KAAK,MAAM,EAAE,IAAI,aAAa,EAAE,CAAC;YAC/B,GAAG,CAAC,yBAAyB,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC;YAEvC,uDAAuD;YACvD,uBAAA,IAAI,qCAAa,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,GAAG,EAAE,CAAC,CAAC;QACvC,CAAC;IACH,CAAC;IAED,+EAA+E;IAC/E,+BAA+B;IAC/B,+EAA+E;IAE/E;;;;;;;;;OASG;IACH,wBAAwB,CACtB,YAAoB,EACpB,YAAuB;QAEvB,GAAG,CAAC,mCAAmC,EAAE;YACvC,YAAY;YACZ,UAAU,EAAE,YAAY,CAAC,MAAM;YAC/B,MAAM,EAAE,YAAY;SACrB,CAAC,CAAC;QAEH,MAAM,cAAc,GAAG,uBAAA,IAAI,qCAAa,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,IAAI,GAAG,EAAE,CAAC;QACxE,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,YAAY,CAAC,CAAC;QAExC,+CAA+C;QAC/C,uBAAA,IAAI,qCAAa,CAAC,GAAG,CAAC,YAAY,EAAE,SAAS,CAAC,CAAC;QAE/C,oBAAoB;QACpB,MAAM,WAAW,GAAG,YAAY,CAAC,MAAM,CACrC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,CACtC,CAAC;QACF,MAAM,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,MAAM,CACrD,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CACjC,CAAC;QAEF,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvD,gEAAgE;YAChE,uBAAA,IAAI,6EAAwB,MAA5B,IAAI,CAA0B,CAAC;QACjC,CAAC;QAED,wEAAwE;QACxE,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,IAAI,uBAAA,IAAI,2EAAkB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAChE,MAAM,kBAAkB,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CACtD,uBAAA,IAAI,uCAAe,CAAC,GAAG,CAAC,KAAK,CAAC,CAC/B,CAAC;YACF,IAAI,kBAAkB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAClC,GAAG,CAAC,0CAA0C,EAAE,EAAE,kBAAkB,EAAE,CAAC,CAAC;gBACxE,IAAI,CAAC,SAAS,CAAC,uBAAA,IAAI,2EAAkB,EAAE;oBACrC,QAAQ,EAAE,kBAAkB;oBAC5B,WAAW,EAAE,IAAI;iBAClB,CAAC,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;oBACjB,GAAG,CAAC,0CAA0C,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC7D,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IA8CD,+EAA+E;IAC/E,4BAA4B;IAC5B,+EAA+E;IAE/E,KAAK,CAAC,SAAS,CACb,QAA2B,EAC3B,OAKC;QAED,MAAM,QAAQ,GAAG,OAAO,EAAE,QAAQ,IAAI,CAAC,GAAG,uBAAA,IAAI,uCAAe,CAAC,CAAC;QAC/D,MAAM,UAAU,GAAG,OAAO,EAAE,UAAU,IAAI,CAAC,UAAU,CAAC,CAAC;QACvD,MAAM,SAAS,GAAG,OAAO,EAAE,SAAS,IAAI,CAAC,SAAS,EAAE,UAAU,EAAE,OAAO,CAAC,CAAC;QAEzE,mDAAmD;QACnD,MAAM,YAAY,GAAoB,EAAE,CAAC;QACzC,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;YAC/B,MAAM,mBAAmB,GAAG,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;YAC7D,YAAY,CAAC,IAAI,CAAC,GAAG,mBAAmB,CAAC,CAAC;QAC5C,CAAC;QAED,IAAI,OAAO,EAAE,WAAW,EAAE,CAAC;YACzB,MAAM,QAAQ,GAAG,MAAM,uBAAA,IAAI,yEAAoB,MAAxB,IAAI,EACzB;gBACE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,2CAA2C,CAAC;gBAChE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,oCAAoC,CAAC;gBACzD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,mCAAmC,CAAC;gBACxD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,yCAAyC,CAAC;gBAC9D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,qCAAqC,CAAC;gBAC1D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,qCAAqC,CAAC;aAC3D,EACD;gBACE,QAAQ;gBACR,QAAQ;gBACR,UAAU;gBACV,SAAS;gBACT,YAAY,EAAE,YAAY,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,SAAS;gBAChE,WAAW,EAAE,IAAI;aAClB,CACF,CAAC;YACF,MAAM,uBAAA,IAAI,kEAAa,MAAjB,IAAI,EAAc,QAAQ,CAAC,CAAC;QACpC,CAAC;QAED,OAAO,uBAAA,IAAI,yEAAoB,MAAxB,IAAI,EAAqB,QAAQ,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;IAClE,CAAC;IAED,KAAK,CAAC,gBAAgB,CACpB,QAA2B,EAC3B,OAIC;QAED,mDAAmD;QACnD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE;YAC5C,QAAQ,EAAE,OAAO,EAAE,QAAQ;YAC3B,UAAU,EAAE,OAAO,EAAE,UAAU;YAC/B,WAAW,EAAE,OAAO,EAAE,WAAW;YACjC,SAAS,EAAE,CAAC,SAAS,EAAE,UAAU,CAAC;SACnC,CAAC,CAAC;QAEH,2CAA2C;QAC3C,MAAM,MAAM,GAA2D,EAAE,CAAC;QAC1E,KAAK,MAAM,CAAC,SAAS,EAAE,aAAa,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YAChE,MAAM,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC;YACvB,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE,CAAC;gBAC7D,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;oBAClB,MAAM,CAAC,SAAS,CAAC,CAAC,OAAwB,CAAC,GAAG,KAAK,CAAC,OAAO,CAAC;gBAC9D,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,gBAAgB,CAAC,OAAsB;QACrC,OAAO,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,OAAO,CAA8B,CAAC;IACzE,CAAC;IAED,KAAK,CAAC,cAAc,CAClB,QAA2B,EAC3B,OAIC;QAED,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE;YAC5C,QAAQ,EAAE,OAAO,EAAE,QAAQ;YAC3B,UAAU,EAAE,OAAO,EAAE,UAAU;YAC/B,WAAW,EAAE,OAAO,EAAE,WAAW;YACjC,SAAS,EAAE,CAAC,OAAO,CAAC;SACrB,CAAC,CAAC;QAEH,qEAAqE;QACrE,MAAM,MAAM,GAAsC,EAAE,CAAC;QACrD,KAAK,MAAM,aAAa,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;YAClD,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE,CAAC;gBAC7D,IAAI,KAAK,CAAC,KAAK,EAAE,CAAC;oBAChB,MAAM,CAAC,OAAwB,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;gBACjD,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,+EAA+E;IAC/E,2BAA2B;IAC3B,+EAA+E;IAE/E;;;;;;OAMG;IACH,KAAK,CAAC,cAAc,CAClB,SAAoB,EACpB,OAAsB;QAEtB,MAAM,iBAAiB,GAAG,IAAA,wBAAgB,EAAC,OAAO,CAAC,CAAC;QAEpD,GAAG,CAAC,qBAAqB,EAAE,EAAE,SAAS,EAAE,OAAO,EAAE,iBAAiB,EAAE,CAAC,CAAC;QAEtE,IAAI,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE;YACpB,MAAM,YAAY,GAAG,KAAK,CAAC,YAAwC,CAAC;YACpE,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC7B,YAAY,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC;YAC/B,CAAC;YAED,kCAAkC;YAClC,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,iBAAiB,CAAC,EAAE,CAAC;gBACzD,YAAY,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAClD,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,8CAA8C;QAC9C,MAAM,OAAO,GAAG,uBAAA,IAAI,2EAAkB,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,SAAS,CAAC,CAAC;QACvE,IAAI,OAAO,EAAE,CAAC;YACZ,MAAM,OAAO,GAAG,cAAc,CAAC,iBAAiB,CAAC,CAAC;YAClD,MAAM,IAAI,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,EAAE;gBAC9B,QAAQ,EAAE,CAAC,OAAO,CAAC;gBACnB,WAAW,EAAE,IAAI;aAClB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,iBAAiB,CAAC,SAAoB,EAAE,OAAsB;QAC5D,MAAM,iBAAiB,GAAG,IAAA,wBAAgB,EAAC,OAAO,CAAC,CAAC;QAEpD,GAAG,CAAC,uBAAuB,EAAE,EAAE,SAAS,EAAE,OAAO,EAAE,iBAAiB,EAAE,CAAC,CAAC;QAExE,IAAI,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE;YACpB,MAAM,YAAY,GAAG,KAAK,CAAC,YAAwC,CAAC;YACpE,IAAI,YAAY,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC5B,YAAY,CAAC,SAAS,CAAC,GAAG,YAAY,CAAC,SAAS,CAAC,CAAC,MAAM,CACtD,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,KAAK,iBAAiB,CACjC,CAAC;gBAEF,wBAAwB;gBACxB,IAAI,YAAY,CAAC,SAAS,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBACzC,OAAO,YAAY,CAAC,SAAS,CAAC,CAAC;gBACjC,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,SAAoB;QAClC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,SAAS,CAAC,IAAI,EAAE,CAAoB,CAAC;IACvE,CAAC;IAgDD;;;;;;;;OAQG;IACH,oBAAoB,CAClB,QAA2B,EAC3B,QAAmB,EACnB,UAAuC,EAAE;QAEzC,MAAM,EAAE,cAAc,GAAG,uBAAA,IAAI,+CAAuB,EAAE,GAAG,OAAO,CAAC;QACjE,MAAM,eAAe,GAAG,oBAAoB,CAAC;QAE7C,MAAM,oBAAoB,GAAG,uBAAA,IAAI,6CAAqB,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QAC5E,MAAM,QAAQ,GAAG,oBAAoB,KAAK,SAAS,CAAC;QAEpD,qEAAqE;QACrE,IAAI,CAAC,SAAS;aACX,IAAI,CAAC,2BAA2B,EAAE;YACjC,OAAO,EAAE;gBACP,QAAQ;gBACR,QAAQ;gBACR,SAAS,EAAE,CAAC,OAAO,CAAC;gBACpB,cAAc;aACf;YACD,cAAc,EAAE,eAAe;YAC/B,QAAQ;SACT,CAAC;aACD,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAExB,qBAAqB;QACrB,MAAM,YAAY,GAAyB;YACzC,MAAM,EAAE,QAAQ;YAChB,SAAS,EAAE,eAAe;YAC1B,UAAU,EAAE,CAAC,UAAU,CAAC;YACxB,SAAS,EAAE,CAAC,OAAO,CAAC;YACpB,WAAW,EAAE,GAAG,EAAE;gBAChB,uBAAA,IAAI,6CAAqB,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;YACpD,CAAC;SACF,CAAC;QAEF,uBAAA,IAAI,6CAAqB,CAAC,GAAG,CAAC,eAAe,EAAE,YAAY,CAAC,CAAC;IAC/D,CAAC;IAED;;OAEG;IACH,sBAAsB;QACpB,MAAM,eAAe,GAAG,oBAAoB,CAAC;QAC7C,MAAM,oBAAoB,GAAG,uBAAA,IAAI,6CAAqB,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QAE5E,IAAI,CAAC,oBAAoB,EAAE,CAAC;YAC1B,OAAO;QACT,CAAC;QAED,qEAAqE;QACrE,IAAI,CAAC,SAAS;aACX,IAAI,CAAC,6BAA6B,EAAE,eAAe,CAAC;aACpD,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAExB,oBAAoB,CAAC,WAAW,EAAE,CAAC;IACrC,CAAC;IAuoBD;;;;;;OAMG;IACH,KAAK,CAAC,kBAAkB,CACtB,QAAsB,EACtB,QAAgB;QAEhB,GAAG,CAAC,iCAAiC,EAAE;YACrC,QAAQ;YACR,UAAU,EAAE,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAC;YAC3C,QAAQ,EAAE,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC;SACxC,CAAC,CAAC;QACH,MAAM,uBAAA,IAAI,+EAA0B,MAA9B,IAAI,EAA2B,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAC3D,CAAC;IAmCD,+EAA+E;IAC/E,UAAU;IACV,+EAA+E;IAE/E,OAAO;QACL,GAAG,CAAC,6BAA6B,EAAE;YACjC,eAAe,EAAE,uBAAA,IAAI,qCAAa,CAAC,IAAI;YACvC,iBAAiB,EAAE,uBAAA,IAAI,6CAAqB,CAAC,IAAI;SAClD,CAAC,CAAC;QAEH,qBAAqB;QACrB,uBAAA,IAAI,qCAAa,CAAC,KAAK,EAAE,CAAC;QAE1B,gCAAgC;QAChC,uBAAA,IAAI,2DAAM,MAAV,IAAI,CAAQ,CAAC;QAEb,6BAA6B;QAC7B,IAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,4BAA4B,CAAC,CAAC;QACrE,IAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,mCAAmC,CAAC,CAAC;QAC5E,IAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,mCAAmC,CAAC,CAAC;QAC5E,IAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,iCAAiC,CAAC,CAAC;QAC1E,IAAI,CAAC,SAAS,CAAC,uBAAuB,CACpC,qCAAqC,CACtC,CAAC;QACF,IAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,+BAA+B,CAAC,CAAC;QACxE,IAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,iCAAiC,CAAC,CAAC;QAC1E,IAAI,CAAC,SAAS,CAAC,uBAAuB,CACpC,oCAAoC,CACrC,CAAC;QACF,IAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,kCAAkC,CAAC,CAAC;IAC7E,CAAC;CACF;AAj2CD,4CAi2CC;;IAp0CG,OAAO,IAAI,CAAC,SAAS,CAAC,IAAI,CACxB,2DAA2D,CAC5D,CAAC;AACJ,CAAC;IAgDC,MAAM,EAAE,iBAAiB,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAC/C,sCAAsC,CACvC,CAAC;IACF,uBAAA,IAAI,mCAAkB,uBAAA,IAAI,2EAAsB,MAA1B,IAAI,EAAuB,iBAAiB,CAAC,MAAA,CAAC;IAEpE,GAAG,CAAC,mBAAmB,EAAE;QACvB,iBAAiB;QACjB,aAAa,EAAE,uBAAA,IAAI,uCAAe;KACnC,CAAC,CAAC;AACL,CAAC,2FAaC,iBAAwE;IAExE,MAAM,MAAM,GAAG,IAAI,GAAG,EAAW,CAAC;IAElC,KAAK,MAAM,CAAC,SAAS,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,iBAAiB,CAAC,EAAE,CAAC;QACtE,KAAK,MAAM,CAAC,SAAS,EAAE,SAAS,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC9D,IAAI,SAAS,EAAE,CAAC;gBACd,wEAAwE;gBACxE,IAAI,SAAS,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;oBAC5B,qCAAqC;oBACrC,MAAM,CAAC,GAAG,CAAC,SAAoB,CAAC,CAAC;gBACnC,CAAC;qBAAM,CAAC;oBACN,0EAA0E;oBAC1E,MAAM,mBAAmB,GAAG,uBAAA,IAAI,8EAAyB,MAA7B,IAAI,EAC9B,SAAS,EACT,SAAS,CACV,CAAC;oBACF,MAAM,CAAC,GAAG,CAAC,GAAG,SAAS,IAAI,mBAAmB,EAAE,CAAC,CAAC;gBACpD,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC,iGAUwB,SAAiB,EAAE,SAAiB;IAC3D,IAAI,SAAS,KAAK,QAAQ,IAAI,SAAS,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;QACzD,2CAA2C;QAC3C,OAAO,QAAQ,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC5C,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;IAGC,6GAA6G;IAC7G,IAAI,CAAC,SAAS,CAAC,SAAS,CACtB,kDAAkD,EAClD,GAAG,EAAE;QACH,uBAAA,IAAI,gFAA2B,MAA/B,IAAI,CAA6B,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IACzD,CAAC,CACF,CAAC;IAEF,mEAAmE;IACnE,IAAI,CAAC,SAAS,CAAC,SAAS,CACtB,yCAAyC,EACzC,CAAC,EAAE,iBAAiB,EAAE,EAAE,EAAE;QACxB,uBAAA,IAAI,mFAA8B,MAAlC,IAAI,EAA+B,iBAAiB,CAAC,CAAC,KAAK,CACzD,OAAO,CAAC,KAAK,CACd,CAAC;IACJ,CAAC,CACF,CAAC;IAEF,qDAAqD;IACrD,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8BAA8B,EAAE,GAAG,EAAE,CAC5D,uBAAA,IAAI,4DAAO,MAAX,IAAI,CAAS,CACd,CAAC;IACF,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8BAA8B,EAAE,GAAG,EAAE,CAC5D,uBAAA,IAAI,2DAAM,MAAV,IAAI,CAAQ,CACb,CAAC;IAEF,2DAA2D;IAC3D,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,0BAA0B,EAAE,GAAG,EAAE,CAAC,uBAAA,IAAI,4DAAO,MAAX,IAAI,CAAS,CAAC,CAAC;IAC1E,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,wBAAwB,EAAE,GAAG,EAAE,CAAC,uBAAA,IAAI,2DAAM,MAAV,IAAI,CAAQ,CAAC,CAAC;AACzE,CAAC;IAGC,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,4BAA4B,EAC5B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAC1B,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,mCAAmC,EACnC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CACjC,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,mCAAmC,EACnC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CACjC,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,iCAAiC,EACjC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAC/B,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,qCAAqC,EACrC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,IAAI,CAAC,CACzC,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,+BAA+B,EAC/B,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CACnC,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,iCAAiC,EACjC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAC/B,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,oCAAoC,EACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAClC,CAAC;IAEF,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAClC,kCAAkC,EAClC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAChC,CAAC;AACJ,CAAC;AAmFD,+EAA+E;AAC/E,uBAAuB;AACvB,+EAA+E;AAE/E;;;;;;;GAOG;AACH,KAAK,+CACH,WAAyB,EACzB,OAAoB,EACpB,kBAAgC,EAAE;IAElC,MAAM,KAAK,GAAG,WAAW,CAAC,WAAW,CACnC,CAAC,IAAI,EAAE,UAAU,EAAE,EAAE,CACnB,KAAK,EACH,GAAG,EAKF,EAAE;QACH,IAAI,CAAC;YACH,OAAO,MAAM,UAAU,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QACrC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,uCAAuC,EAAE,KAAK,CAAC,CAAC;YAC9D,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC;QACnB,CAAC;IACH,CAAC,EACH,KAAK,EAAE,GAAG,EAAE,EAAE,CAAC,GAAG,CACnB,CAAC;IAEF,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC;QACzB,OAAO;QACP,QAAQ,EAAE,eAAe;QACzB,cAAc,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,KAAsC;KAClE,CAAC,CAAC;IACH,OAAO,MAAM,CAAC,QAAQ,CAAC;AACzB,CAAC,qGA2MC,eAA0B;IAE1B,MAAM,UAAU,GAAG,IAAI,GAAG,EAAqB,CAAC;IAChD,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC,CAAC;IAEjD,KAAK,MAAM,QAAQ,IAAI,uBAAA,IAAI,qCAAa,CAAC,IAAI,EAAE,EAAE,CAAC;QAChD,4CAA4C;QAC5C,MAAM,eAAe,GAAG,uBAAA,IAAI,qCAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACxD,IAAI,CAAC,eAAe,IAAI,eAAe,CAAC,IAAI,KAAK,CAAC,EAAE,CAAC;YACnD,SAAS;QACX,CAAC;QAED,MAAM,mBAAmB,GAAc,EAAE,CAAC;QAE1C,KAAK,MAAM,OAAO,IAAI,eAAe,EAAE,CAAC;YACtC,kDAAkD;YAClD,IAAI,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC;gBACjC,mBAAmB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBAClC,eAAe,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAClC,CAAC;QACH,CAAC;QAED,IAAI,mBAAmB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACnC,UAAU,CAAC,GAAG,CAAC,QAAQ,EAAE,mBAAmB,CAAC,CAAC;YAC9C,GAAG,CAAC,gCAAgC,EAAE;gBACpC,QAAQ;gBACR,MAAM,EAAE,mBAAmB;aAC5B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,UAAU,CAAC;AACpB,CAAC;AAqED,+EAA+E;AAC/E,mBAAmB;AACnB,+EAA+E;AAE/E,KAAK,wCAAc,QAAsB;IACvC,uEAAuE;IACvE,MAAM,kBAAkB,GAAG,iBAAiB,CAAC,QAAQ,CAAC,CAAC;IAEvD,MAAM,WAAW,GAAG,MAAM,uBAAA,IAAI,yCAAiB,CAAC,OAAO,EAAE,CAAC;IAE1D,IAAI,CAAC;QACH,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC;QACjC,MAAM,cAAc,GAAG,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;QACrD,6DAA6D;QAC7D,MAAM,cAAc,GAClB,kBAAkB,CAAC,cAAc,IAAI,EAAE,CAAC;QAE1C,mCAAmC;QACnC,MAAM,eAAe,GAKf,EAAE,CAAC;QACT,MAAM,eAAe,GAAa,EAAE,CAAC;QAErC,IAAI,CAAC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE;YACpB,+EAA+E;YAC/E,MAAM,QAAQ,GAAG,KAAK,CAAC,cAGtB,CAAC;YACF,MAAM,QAAQ,GAAG,KAAK,CAAC,aAGtB,CAAC;YACF,MAAM,MAAM,GAAG,KAAK,CAAC,WAAiD,CAAC;YAEvE,IAAI,kBAAkB,CAAC,cAAc,EAAE,CAAC;gBACtC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CACvC,kBAAkB,CAAC,cAAc,CAClC,EAAE,CAAC;oBACF,IACE,CAAC,IAAA,gBAAO,EACN,aAAa,CAAC,cAAc,CAAC,GAAoB,CAAC,EAClD,KAAK,CACN,EACD,CAAC;wBACD,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;oBAC5B,CAAC;oBACD,QAAQ,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;gBACxB,CAAC;YACH,CAAC;YAED,IAAI,kBAAkB,CAAC,aAAa,EAAE,CAAC;gBACrC,KAAK,MAAM,CAAC,SAAS,EAAE,eAAe,CAAC,IAAI,MAAM,CAAC,OAAO,CACvD,kBAAkB,CAAC,aAAa,CACjC,EAAE,CAAC;oBACF,MAAM,gBAAgB,GACpB,aAAa,CAAC,aAAa,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC;oBAE/C,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;wBACzB,QAAQ,CAAC,SAAS,CAAC,GAAG,EAAE,CAAC;oBAC3B,CAAC;oBAED,KAAK,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC,EAAE,CAAC;wBACjE,MAAM,eAAe,GAAG,gBAAgB,CACtC,OAAwB,CACS,CAAC;wBACpC,MAAM,WAAW,GAAG,OAA6B,CAAC;wBAClD,MAAM,SAAS,GAAG,WAAW,CAAC,MAAM,CAAC;wBACrC,MAAM,SAAS,GAAG,eAAe,EAAE,MAAM,CAAC;wBAE1C,oCAAoC;wBACpC,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;4BAC5B,eAAe,CAAC,IAAI,CAAC;gCACnB,SAAS;gCACT,OAAO;gCACP,SAAS;gCACT,SAAS;6BACV,CAAC,CAAC;wBACL,CAAC;oBACH,CAAC;oBAED,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,eAAe,CAAC,CAAC;gBACtD,CAAC;YACH,CAAC;YAED,yBAAyB;YACzB,IAAI,kBAAkB,CAAC,WAAW,EAAE,CAAC;gBACnC,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CACvC,kBAAkB,CAAC,WAAW,CAC/B,EAAE,CAAC;oBACF,MAAM,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;gBACtB,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,2BAA2B;QAC3B,MAAM,kBAAkB,GAAa,kBAAkB,CAAC,WAAW;YACjE,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,CAAC,WAAW,CAAC,CAAC,MAAM,CAChD,CAAC,OAAO,EAAE,EAAE,CACV,CAAC,IAAA,gBAAO,EACN,cAAc,CAAC,OAAwB,CAAC,EACxC,kBAAkB,CAAC,WAAW,EAAE,CAAC,OAAwB,CAAC,CAC3D,CACJ;YACH,CAAC,CAAC,EAAE,CAAC;QAEP,0BAA0B;QAC1B,IACE,eAAe,CAAC,MAAM,GAAG,CAAC;YAC1B,eAAe,CAAC,MAAM,GAAG,CAAC;YAC1B,kBAAkB,CAAC,MAAM,GAAG,CAAC,EAC7B,CAAC;YACD,GAAG,CAAC,eAAe,EAAE;gBACnB,eAAe,EACb,eAAe,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,SAAS;gBAC1D,oBAAoB,EAClB,eAAe,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS;gBACjE,kBAAkB,EAChB,kBAAkB,CAAC,MAAM,GAAG,CAAC;oBAC3B,CAAC,CAAC,kBAAkB,CAAC,MAAM;oBAC3B,CAAC,CAAC,SAAS;gBACf,SAAS,EACP,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,MAAM,GAAG,CAAC;oBACpC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC;wBAC3D,SAAS;wBACT,MAAM;qBACP,CAAC,CAAC;oBACL,CAAC,CAAC,SAAS;aAChB,CAAC,CAAC;QACL,CAAC;QAED,iCAAiC;QACjC,KAAK,MAAM,MAAM,IAAI,eAAe,EAAE,CAAC;YACrC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,iCAAiC,EAAE;gBACxD,SAAS,EAAE,MAAM,CAAC,SAAS;gBAC3B,OAAO,EAAE,MAAM,CAAC,OAAwB;gBACxC,cAAc,EAAE,MAAM,CAAC,SAAS,IAAI,GAAG;gBACvC,SAAS,EAAE,MAAM,CAAC,SAAS;aAC5B,CAAC,CAAC;QACL,CAAC;QAED,mDAAmD;QACnD,IAAI,kBAAkB,CAAC,MAAM,GAAG,CAAC,IAAI,kBAAkB,CAAC,WAAW,EAAE,CAAC;YACpE,+CAA+C;YAC/C,MAAM,aAAa,GAAsC,EAAE,CAAC;YAC5D,KAAK,MAAM,OAAO,IAAI,kBAAkB,EAAE,CAAC;gBACzC,MAAM,KAAK,GACT,kBAAkB,CAAC,WAAW,CAAC,OAAwB,CAAC,CAAC;gBAC3D,IAAI,KAAK,EAAE,CAAC;oBACV,aAAa,CAAC,OAAwB,CAAC,GAAG,KAAK,CAAC;gBAClD,CAAC;YACH,CAAC;YACD,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,+BAA+B,EAAE;gBACtD,MAAM,EAAE,aAAa;aACtB,CAAC,CAAC;QACL,CAAC;QAED,iCAAiC;QACjC,KAAK,MAAM,CAAC,SAAS,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC;YACnE,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACxB,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,iCAAiC,EAAE;oBACxD,SAAS;oBACT,QAAQ;iBACT,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;YAAS,CAAC;QACT,WAAW,EAAE,CAAC;IAChB,CAAC;AACH,CAAC,uFAGC,QAA2B,EAC3B,QAAmB,EACnB,UAAuB;IAEvB,MAAM,MAAM,GAAoD,EAAE,CAAC;IACnE,mCAAmC;IACnC,MAAM,UAAU,GAAG,IAAI,GAAG,CAAC,QAAQ,CAAC,CAAC;IACrC,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,UAAU,CAAC,CAAC;IAEzC,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC;QAExB,MAAM,eAAe,GAAG,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,OAAO,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QAEnE,KAAK,MAAM,CAAC,OAAO,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC,EAAE,CAAC;YACjE,MAAM,YAAY,GAAG,OAAwB,CAAC;YAC9C,MAAM,YAAY,GAAG,cAAc,CAAC,YAAY,CAAC,CAAC;YAElD,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,YAAY,CAAC,EAAE,CAAC;gBAClC,SAAS;YACX,CAAC;YAED,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,YAAY,CAAC,CAAC;YAE5D,+BAA+B;YAC/B,IAAI,CAAC,WAAW,EAAE,CAAC;gBACjB,SAAS;YACX,CAAC;YAED,MAAM,QAAQ,GAAG,WAA4B,CAAC;YAE9C,uBAAuB;YACvB,MAAM,cAAc,GAAG,uBAAA,IAAI,+EAA0B,MAA9B,IAAI,EAA2B,QAAQ,CAAC,IAAI,CAAC,CAAC;YACrE,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,CAAC;gBACtC,SAAS;YACX,CAAC;YAED,MAAM,YAAY,GAAG,OAAuB,CAAC;YAC7C,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,YAAY,CAAe,CAAC;YACpE,MAAM,KAAK,GAAe,QAAQ,IAAI;gBACpC,KAAK,EAAE,CAAC;gBACR,WAAW,EAAE,CAAC;aACf,CAAC;YAEF,mDAAmD;YACnD,sFAAsF;YACtF,0CAA0C;YAC1C,MAAM,aAAa,GAAG,IAAI,sBAAW,CAAC,YAAY,CAAC,MAAM,IAAI,GAAG,CAAC,CAAC;YAClE,MAAM,SAAS,GAAG,aAAa;iBAC5B,YAAY,CAAC,KAAK,CAAC,KAAK,IAAI,CAAC,CAAC;iBAC9B,QAAQ,EAAE,CAAC;YAEd,MAAM,KAAK,GAAU;gBACnB,EAAE,EAAE,YAAY;gBAChB,OAAO,EAAE,YAAY;gBACrB,OAAO,EAAE,YAAY;gBACrB,QAAQ;gBACR,KAAK;gBACL,SAAS;aACV,CAAC;YAEF,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,YAAY,CAAC,GAAG,KAAK,CAAC;QAC3C,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC,mGAQyB,aAAqB;IAC7C,QAAQ,aAAa,EAAE,CAAC;QACtB,KAAK,QAAQ,CAAC;QACd,KAAK,OAAO,CAAC;QACb,KAAK,KAAK;YACR,OAAO,UAAU,CAAC;QACpB,KAAK,QAAQ;YACX,OAAO,KAAK,CAAC;QACf,KAAK,SAAS;YACZ,uEAAuE;YACvE,OAAO,aAAa,CAAC;QACvB;YACE,wCAAwC;YACxC,OAAO,UAAU,CAAC;IACtB,CAAC;AACH,CAAC;IAWC,GAAG,CAAC,yBAAyB,EAAE;QAC7B,oBAAoB,EAAE,uBAAA,IAAI,2EAAkB,CAAC,MAAM;QACnD,iBAAiB,EAAE,uBAAA,IAAI,uCAAe,CAAC,IAAI;KAC5C,CAAC,CAAC;IAEH,uBAAA,IAAI,6EAAwB,MAA5B,IAAI,CAA0B,CAAC;IAC/B,IAAI,uBAAA,IAAI,2EAAkB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtC,IAAI,CAAC,SAAS,CAAC,uBAAA,IAAI,2EAAkB,EAAE;YACrC,QAAQ,EAAE,CAAC,GAAG,uBAAA,IAAI,uCAAe,CAAC;YAClC,WAAW,EAAE,IAAI;SAClB,CAAC,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;YACjB,GAAG,CAAC,wBAAwB,EAAE,KAAK,CAAC,CAAC;QACvC,CAAC,CAAC,CAAC;IACL,CAAC;AACH,CAAC;IAOC,GAAG,CAAC,yBAAyB,EAAE;QAC7B,uBAAuB,EAAE,uBAAA,IAAI,6CAAqB,CAAC,IAAI;QACvD,oBAAoB,EAAE,uBAAA,IAAI,6CAAqB,CAAC,GAAG,CAAC,oBAAoB,CAAC;KAC1E,CAAC,CAAC;IAEH,6DAA6D;IAC7D,IAAI,CAAC,sBAAsB,EAAE,CAAC;IAE9B,8EAA8E;IAC9E,sDAAsD;IACtD,iEAAiE;IACjE,MAAM,gBAAgB,GAAG,CAAC,GAAG,uBAAA,IAAI,6CAAqB,CAAC,IAAI,EAAE,CAAC,CAAC;IAC/D,KAAK,MAAM,eAAe,IAAI,gBAAgB,EAAE,CAAC;QAC/C,oEAAoE;QACpE,IAAI,eAAe,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;YACtC,MAAM,QAAQ,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC1C,uBAAA,IAAI,4EAAuB,MAA3B,IAAI,EAAwB,QAAQ,CAAC,CAAC;QACxC,CAAC;IACH,CAAC;IACD,uBAAA,IAAI,6CAAqB,CAAC,KAAK,EAAE,CAAC;AACpC,CAAC;IAMC,IAAI,uBAAA,IAAI,2EAAkB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACxC,OAAO;IACT,CAAC;IAED,wDAAwD;IACxD,uBAAA,IAAI,6EAAwB,MAA5B,IAAI,CAA0B,CAAC;IAE/B,sEAAsE;IACtE,IAAI,CAAC,oBAAoB,CAAC,uBAAA,IAAI,2EAAkB,EAAE,CAAC,GAAG,uBAAA,IAAI,uCAAe,CAAC,CAAC,CAAC;AAC9E,CAAC;IAcC,uFAAuF;IACvF,MAAM,eAAe,GAAG,uBAAA,IAAI,8EAAyB,MAA7B,IAAI,EAC1B,uBAAA,IAAI,2EAAkB,EACtB,uBAAA,IAAI,uCAAe,CACpB,CAAC;IAEF,mEAAmE;IACnE,4CAA4C;IAC5C,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC,CAAC;IAExD,4EAA4E;IAC5E,MAAM,eAAe,GAAG,uBAAA,IAAI,gFAA2B,MAA/B,IAAI,EAC1B,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,CAC5B,CAAC;IAEF,GAAG,CAAC,8BAA8B,EAAE;QAClC,WAAW,EAAE,eAAe,CAAC,IAAI;QACjC,qBAAqB,EAAE,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAC9D,CAAC,CAAC,QAAQ,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,CAAC,CAClE;KACF,CAAC,CAAC;IAEH,+EAA+E;IAC/E,KAAK,MAAM,QAAQ,IAAI,uBAAA,IAAI,qCAAa,CAAC,IAAI,EAAE,EAAE,CAAC;QAChD,MAAM,cAAc,GAAG,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAErD,IAAI,CAAC,cAAc,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACnD,wDAAwD;YACxD,uBAAA,IAAI,4EAAuB,MAA3B,IAAI,EAAwB,QAAQ,CAAC,CAAC;YACtC,SAAS;QACX,CAAC;QAED,+DAA+D;QAC/D,MAAM,iBAAiB,GAAG,uBAAA,IAAI,2EAAsB,MAA1B,IAAI,EAC5B,cAAc,EACd,eAAe,CAChB,CAAC;QAEF,IAAI,iBAAiB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACnC,SAAS;QACX,CAAC;QAED,0CAA0C;QAC1C,uBAAA,IAAI,4EAAuB,MAA3B,IAAI,EAAwB,QAAQ,EAAE,iBAAiB,EAAE,cAAc,CAAC,CAAC;IAC3E,CAAC;AACH,CAAC,iGAWC,QAA2B,EAC3B,iBAA+B;IAE/B,MAAM,eAAe,GAAG,IAAI,GAAG,EAA8B,CAAC;IAE9D,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE,CAAC;QAC/B,MAAM,aAAa,GAAG,uBAAA,IAAI,iFAA4B,MAAhC,IAAI,EAA6B,OAAO,CAAC,CAAC;QAEhE,KAAK,MAAM,OAAO,IAAI,aAAa,EAAE,CAAC;YACpC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC;gBACpC,SAAS;YACX,CAAC;YAED,MAAM,gBAAgB,GAAG,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;YAC5D,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC/B,eAAe,CAAC,GAAG,CAAC,OAAO,EAAE,gBAAgB,CAAC,CAAC;QACjD,CAAC;IACH,CAAC;IAED,OAAO,eAAe,CAAC;AACzB,CAAC,2FAUC,MAAiB,EACjB,eAAgD;IAEhD,MAAM,UAAU,GAAG,IAAI,GAAG,EAAU,CAAC;IACrC,MAAM,QAAQ,GAAsB,EAAE,CAAC;IAEvC,KAAK,MAAM,OAAO,IAAI,MAAM,EAAE,CAAC;QAC7B,MAAM,aAAa,GAAG,eAAe,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;QACzD,KAAK,MAAM,OAAO,IAAI,aAAa,EAAE,CAAC;YACpC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC;gBAChC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBAC3B,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACzB,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,QAAQ,CAAC;AAClB,CAAC,6FAWC,QAAgB,EAChB,QAA2B,EAC3B,MAAiB;IAEjB,MAAM,eAAe,GAAG,MAAM,QAAQ,EAAE,CAAC;IACzC,MAAM,oBAAoB,GAAG,uBAAA,IAAI,6CAAqB,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;IAC5E,MAAM,QAAQ,GAAG,oBAAoB,KAAK,SAAS,CAAC;IAEpD,GAAG,CAAC,0BAA0B,EAAE;QAC9B,QAAQ;QACR,eAAe;QACf,QAAQ;QACR,YAAY,EAAE,QAAQ,CAAC,MAAM;QAC7B,UAAU,EAAE,MAAM,CAAC,MAAM;KAC1B,CAAC,CAAC;IAEH,oEAAoE;IACpE,CAAC,KAAK,IAAmB,EAAE;QACzB,IAAI,CAAC;YACH,MAAO,IAAI,CAAC,SAAS,CAAC,IAAyB,CAC7C,GAAG,QAAQ,YAAY,EACvB;gBACE,OAAO,EAAE;oBACP,QAAQ;oBACR,QAAQ,EAAE,MAAM;oBAChB,UAAU,EAAE,CAAC,UAAU,CAAC;oBACxB,SAAS,EAAE,CAAC,SAAS,CAAC;oBACtB,cAAc,EAAE,uBAAA,IAAI,+CAAuB;iBAC5C;gBACD,cAAc,EAAE,eAAe;gBAC/B,QAAQ;aACT,CACF,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CACX,8CAA8C,QAAQ,IAAI,EAC1D,KAAK,CACN,CAAC;QACJ,CAAC;IACH,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;IAE1B,qBAAqB;IACrB,MAAM,YAAY,GAAyB;QACzC,MAAM;QACN,SAAS,EAAE,eAAe;QAC1B,UAAU,EAAE,CAAC,UAAU,CAAC;QACxB,SAAS,EAAE,CAAC,SAAS,EAAE,OAAO,CAAC;QAC/B,WAAW,EAAE,GAAG,EAAE;YAChB,uBAAA,IAAI,6CAAqB,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QACpD,CAAC;KACF,CAAC;IAEF,uBAAA,IAAI,6CAAqB,CAAC,GAAG,CAAC,eAAe,EAAE,YAAY,CAAC,CAAC;AAC/D,CAAC,6FAOsB,QAAgB;IACrC,MAAM,eAAe,GAAG,MAAM,QAAQ,EAAE,CAAC;IACzC,MAAM,oBAAoB,GAAG,uBAAA,IAAI,6CAAqB,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;IAE5E,IAAI,oBAAoB,EAAE,CAAC;QACzB,mCAAmC;QACnC,CAAC,KAAK,IAAmB,EAAE;YACzB,IAAI,CAAC;gBACH,MAAO,IAAI,CAAC,SAAS,CAAC,IAAyB,CAC7C,GAAG,QAAQ,cAAc,EACzB,eAAe,CAChB,CAAC;YACJ,CAAC;YAAC,MAAM,CAAC;gBACP,sDAAsD;YACxD,CAAC;QACH,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE;YACd,sDAAsD;QACxD,CAAC,CAAC,CAAC;QACH,oBAAoB,CAAC,WAAW,EAAE,CAAC;IACrC,CAAC;AACH,CAAC,uGAa2B,OAAwB;IAClD,sFAAsF;IACtF,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,IAAI,EAAE,CAAC;IACpC,MAAM,MAAM,GAAc,EAAE,CAAC;IAE7B,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;QAC3B,MAAM,CAAC,SAAS,EAAE,SAAS,CAAC,GAAI,KAAgB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAE5D,+EAA+E;QAC/E,IAAI,SAAS,KAAK,GAAG,EAAE,CAAC;YACtB,KAAK,MAAM,KAAK,IAAI,uBAAA,IAAI,uCAAe,EAAE,CAAC;gBACxC,IAAI,KAAK,CAAC,UAAU,CAAC,GAAG,SAAS,GAAG,CAAC,EAAE,CAAC;oBACtC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBACrB,CAAC;YACH,CAAC;QACH,CAAC;aAAM,IAAI,SAAS,KAAK,QAAQ,IAAI,SAAS,EAAE,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;YACjE,sCAAsC;YACtC,MAAM,CAAC,IAAI,CAAC,UAAU,QAAQ,CAAC,SAAS,EAAE,EAAE,CAAC,EAAa,CAAC,CAAC;QAC9D,CAAC;aAAM,CAAC;YACN,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,+EAA+E;AAC/E,iBAAiB;AACjB,+EAA+E;AAE/E,KAAK;IACH,MAAM,QAAQ,GAAG,uBAAA,IAAI,2EAAkB,CAAC;IAExC,GAAG,CAAC,uBAAuB,EAAE;QAC3B,YAAY,EAAE,QAAQ,CAAC,MAAM;QAC7B,UAAU,EAAE,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;KACtC,CAAC,CAAC;IAEH,gDAAgD;IAChD,uBAAA,IAAI,6EAAwB,MAA5B,IAAI,CAA0B,CAAC;IAC/B,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACxB,MAAM,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE;YAC7B,QAAQ,EAAE,CAAC,GAAG,uBAAA,IAAI,uCAAe,CAAC;YAClC,WAAW,EAAE,IAAI;SAClB,CAAC,CAAC;IACL,CAAC;AACH,CAAC,mDAED,KAAK,yDACH,iBAAwE;IAExE,MAAM,cAAc,GAAG,uBAAA,IAAI,uCAAe,CAAC;IAC3C,uBAAA,IAAI,mCAAkB,uBAAA,IAAI,2EAAsB,MAA1B,IAAI,EAAuB,iBAAiB,CAAC,MAAA,CAAC;IAEpE,6DAA6D;IAC7D,MAAM,WAAW,GAAc,EAAE,CAAC;IAClC,KAAK,MAAM,KAAK,IAAI,uBAAA,IAAI,uCAAe,EAAE,CAAC;QACxC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YAC/B,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,CAAC;IACH,CAAC;IAED,gEAAgE;IAChE,MAAM,aAAa,GAAc,EAAE,CAAC;IACpC,KAAK,MAAM,KAAK,IAAI,cAAc,EAAE,CAAC;QACnC,IAAI,CAAC,uBAAA,IAAI,uCAAe,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YACpC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC;IACH,CAAC;IAED,GAAG,CAAC,0BAA0B,EAAE;QAC9B,aAAa,EAAE,cAAc,CAAC,IAAI;QAClC,QAAQ,EAAE,uBAAA,IAAI,uCAAe,CAAC,IAAI;QAClC,WAAW;QACX,aAAa;KACd,CAAC,CAAC;IAEH,yEAAyE;IACzE,uFAAuF;IACvF,uEAAuE;IAEvE,0CAA0C;IAC1C,uBAAA,IAAI,6EAAwB,MAA5B,IAAI,CAA0B,CAAC;IAE/B,6CAA6C;IAC7C,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,IAAI,uBAAA,IAAI,2EAAkB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAChE,MAAM,IAAI,CAAC,SAAS,CAAC,uBAAA,IAAI,2EAAkB,EAAE;YAC3C,QAAQ,EAAE,WAAW;YACrB,WAAW,EAAE,IAAI;SAClB,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAqBD;;;;;;;GAOG;AACH,KAAK,qDACH,QAAsB,EACtB,SAAkB,EAClB,OAAqB;IAErB,8EAA8E;IAC9E,oFAAoF;IACpF,MAAM,gBAAgB,GAAG,MAAM,uBAAA,IAAI,yEAAoB,MAAxB,IAAI,EACjC;QACE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,yCAAyC,CAAC;QAC9D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,qCAAqC,CAAC;QAC1D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,qCAAqC,CAAC;KAC3D,EACD,OAAO,IAAI;QACT,QAAQ,EAAE,EAAE;QACZ,QAAQ,EAAE,EAAE;QACZ,SAAS,EAAE,CAAC,SAAS,EAAE,UAAU,EAAE,OAAO,CAAC;KAC5C,EACD,QAAQ,CACT,CAAC;IAEF,eAAe;IACf,MAAM,uBAAA,IAAI,kEAAa,MAAjB,IAAI,EAAc,gBAAgB,CAAC,CAAC;AAC5C,CAAC","sourcesContent":["import type {\n AccountTreeControllerGetAccountsFromSelectedAccountGroupAction,\n AccountTreeControllerSelectedAccountGroupChangeEvent,\n} from '@metamask/account-tree-controller';\nimport { BaseController } from '@metamask/base-controller';\nimport type {\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n StateMetadata,\n} from '@metamask/base-controller';\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 NetworkEnablementControllerGetStateAction,\n NetworkEnablementControllerEvents,\n NetworkEnablementControllerState,\n} from '@metamask/network-enablement-controller';\nimport type { Json } from '@metamask/utils';\nimport { parseCaipAssetType } from '@metamask/utils';\nimport { Mutex } from 'async-mutex';\nimport BigNumberJS from 'bignumber.js';\nimport { isEqual } from 'lodash';\n\nimport type { AccountsApiDataSourceGetAssetsMiddlewareAction } from './data-sources/AccountsApiDataSource';\nimport type {\n PriceDataSourceGetAssetsMiddlewareAction,\n PriceDataSourceFetchAction,\n PriceDataSourceSubscribeAction,\n PriceDataSourceUnsubscribeAction,\n} from './data-sources/PriceDataSource';\nimport type { RpcDataSourceGetAssetsMiddlewareAction } from './data-sources/RpcDataSource';\nimport type { SnapDataSourceGetAssetsMiddlewareAction } from './data-sources/SnapDataSource';\nimport type { TokenDataSourceGetAssetsMiddlewareAction } from './data-sources/TokenDataSource';\nimport { projectLogger, createModuleLogger } from './logger';\nimport type { DetectionMiddlewareGetAssetsMiddlewareAction } from './middlewares/DetectionMiddleware';\nimport type {\n AccountId,\n ChainId,\n Caip19AssetId,\n AssetMetadata,\n AssetPrice,\n AssetBalance,\n AssetType,\n DataType,\n DataRequest,\n DataResponse,\n NextFunction,\n Middleware,\n DataSourceDefinition,\n SubscriptionResponse,\n Asset,\n AssetsControllerStateInternal,\n} from './types';\nimport { normalizeAssetId } from './utils';\n\n// ============================================================================\n// CONTROLLER CONSTANTS\n// ============================================================================\n\nconst CONTROLLER_NAME = 'AssetsController' as const;\n\n/** Default polling interval hint for data sources (30 seconds) */\nconst DEFAULT_POLLING_INTERVAL_MS = 30_000;\n\nconst log = createModuleLogger(projectLogger, CONTROLLER_NAME);\n\n// ============================================================================\n// STATE TYPES\n// ============================================================================\n\n/**\n * State structure for AssetsController.\n *\n * All values are JSON-serializable. The type is widened to satisfy\n * StateConstraint from BaseController, but the actual runtime values\n * conform to AssetMetadata, AssetPrice, and AssetBalance interfaces.\n *\n * @see AssetsControllerStateInternal for the semantic type structure\n */\nexport type AssetsControllerState = {\n /** Shared metadata for all assets (stored once per asset) */\n assetsMetadata: { [assetId: string]: Json };\n /** Per-account balance data */\n assetsBalance: { [accountId: string]: { [assetId: string]: Json } };\n /** Price data for assets */\n assetsPrice: { [assetId: string]: Json };\n /** Custom assets added by users per account (CAIP-19 asset IDs) */\n customAssets: { [accountId: string]: string[] };\n};\n\n/**\n * Returns the default state for AssetsController.\n *\n * @returns The default AssetsController state with empty metadata, balance, price, and customAssets maps.\n */\nexport function getDefaultAssetsControllerState(): AssetsControllerState {\n return {\n assetsMetadata: {},\n assetsBalance: {},\n assetsPrice: {},\n customAssets: {},\n };\n}\n\n// ============================================================================\n// MESSENGER TYPES\n// ============================================================================\n\nexport type AssetsControllerGetStateAction = ControllerGetStateAction<\n typeof CONTROLLER_NAME,\n AssetsControllerState\n>;\n\nexport type AssetsControllerGetAssetsAction = {\n type: `${typeof CONTROLLER_NAME}:getAssets`;\n handler: AssetsController['getAssets'];\n};\n\nexport type AssetsControllerGetAssetsBalanceAction = {\n type: `${typeof CONTROLLER_NAME}:getAssetsBalance`;\n handler: AssetsController['getAssetsBalance'];\n};\n\nexport type AssetsControllerGetAssetMetadataAction = {\n type: `${typeof CONTROLLER_NAME}:getAssetMetadata`;\n handler: AssetsController['getAssetMetadata'];\n};\n\nexport type AssetsControllerGetAssetsPriceAction = {\n type: `${typeof CONTROLLER_NAME}:getAssetsPrice`;\n handler: AssetsController['getAssetsPrice'];\n};\n\nexport type AssetsControllerActiveChainsUpdateAction = {\n type: `${typeof CONTROLLER_NAME}:activeChainsUpdate`;\n handler: AssetsController['handleActiveChainsUpdate'];\n};\n\nexport type AssetsControllerAssetsUpdateAction = {\n type: `${typeof CONTROLLER_NAME}:assetsUpdate`;\n handler: AssetsController['handleAssetsUpdate'];\n};\n\nexport type AssetsControllerAddCustomAssetAction = {\n type: `${typeof CONTROLLER_NAME}:addCustomAsset`;\n handler: AssetsController['addCustomAsset'];\n};\n\nexport type AssetsControllerRemoveCustomAssetAction = {\n type: `${typeof CONTROLLER_NAME}:removeCustomAsset`;\n handler: AssetsController['removeCustomAsset'];\n};\n\nexport type AssetsControllerGetCustomAssetsAction = {\n type: `${typeof CONTROLLER_NAME}:getCustomAssets`;\n handler: AssetsController['getCustomAssets'];\n};\n\nexport type AssetsControllerActions =\n | AssetsControllerGetStateAction\n | AssetsControllerGetAssetsAction\n | AssetsControllerGetAssetsBalanceAction\n | AssetsControllerGetAssetMetadataAction\n | AssetsControllerGetAssetsPriceAction\n | AssetsControllerActiveChainsUpdateAction\n | AssetsControllerAssetsUpdateAction\n | AssetsControllerAddCustomAssetAction\n | AssetsControllerRemoveCustomAssetAction\n | AssetsControllerGetCustomAssetsAction;\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 | AccountTreeControllerGetAccountsFromSelectedAccountGroupAction\n | NetworkEnablementControllerGetStateAction\n // Data source middlewares\n | AccountsApiDataSourceGetAssetsMiddlewareAction\n | SnapDataSourceGetAssetsMiddlewareAction\n | RpcDataSourceGetAssetsMiddlewareAction\n // Enrichment middlewares\n | TokenDataSourceGetAssetsMiddlewareAction\n | PriceDataSourceGetAssetsMiddlewareAction\n | PriceDataSourceFetchAction\n | PriceDataSourceSubscribeAction\n | PriceDataSourceUnsubscribeAction\n | DetectionMiddlewareGetAssetsMiddlewareAction;\n\n/**\n * App lifecycle event: fired when app becomes active (opened/foregrounded)\n */\nexport type AppStateControllerAppOpenedEvent = {\n type: 'AppStateController:appOpened';\n payload: [];\n};\n\n/**\n * App lifecycle event: fired when app becomes inactive (closed/backgrounded)\n */\nexport type AppStateControllerAppClosedEvent = {\n type: 'AppStateController:appClosed';\n payload: [];\n};\n\ntype AllowedEvents =\n | AccountTreeControllerSelectedAccountGroupChangeEvent\n | NetworkEnablementControllerEvents\n | AppStateControllerAppOpenedEvent\n | AppStateControllerAppClosedEvent\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent;\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};\n\n// ============================================================================\n// STATE METADATA\n// ============================================================================\n\nconst stateMetadata: StateMetadata<AssetsControllerState> = {\n assetsMetadata: {\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};\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.assetsMetadata) {\n normalized.assetsMetadata = {};\n for (const [assetId, metadata] of Object.entries(response.assetsMetadata)) {\n const normalizedId = normalizeAssetId(assetId as Caip19AssetId);\n normalized.assetsMetadata[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 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. **App Lifecycle Management**: Listens to app open/close events via messenger\n * to start/stop subscriptions automatically, conserving resources when app is closed.\n *\n * ## App Lifecycle\n *\n * - **App Opened** (`AppStateController:appOpened`): Starts subscriptions, fetches initial data\n * - **App Closed** (`AppStateController:appClosed`): Stops all subscriptions to conserve resources\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 /** Default update interval hint passed to data sources */\n readonly #defaultUpdateInterval: number;\n\n readonly #controllerMutex = new Mutex();\n\n /**\n * Active balance subscriptions keyed by account ID.\n * Each account has one logical subscription that may span multiple data sources.\n * For example, if WebSocket covers chains A,B and RPC covers chain C,\n * the account subscribes to both data sources for its chains.\n */\n readonly #activeSubscriptions: Map<string, SubscriptionResponse> = new Map();\n\n /** Currently enabled chains from NetworkEnablementController */\n #enabledChains: Set<ChainId> = new Set();\n\n /**\n * Get the currently selected accounts from AccountTreeController.\n * This includes all accounts in the same group as the selected account\n * (EVM, Bitcoin, Solana, Tron, etc. that belong to the same logical account group).\n *\n * @returns Array of InternalAccount objects from the selected account group.\n */\n get #selectedAccounts(): InternalAccount[] {\n return this.messenger.call(\n 'AccountTreeController:getAccountsFromSelectedAccountGroup',\n );\n }\n\n /**\n * Registered data sources with their available chains.\n * Updated continuously and independently from subscription flows.\n * Key: sourceId, Value: Set of currently available chainIds\n */\n readonly #dataSources: Map<string, Set<ChainId>> = new Map();\n\n constructor({\n messenger,\n state = {},\n defaultUpdateInterval = DEFAULT_POLLING_INTERVAL_MS,\n }: AssetsControllerOptions) {\n super({\n name: CONTROLLER_NAME,\n messenger,\n metadata: stateMetadata,\n state: {\n ...getDefaultAssetsControllerState(),\n ...state,\n },\n });\n\n this.#defaultUpdateInterval = defaultUpdateInterval;\n\n log('Initializing AssetsController', {\n defaultUpdateInterval,\n });\n\n this.#initializeState();\n this.#subscribeToEvents();\n this.#registerActionHandlers();\n\n // Register data sources (order = subscription priority)\n this.registerDataSources([\n 'BackendWebsocketDataSource', // Real-time push updates\n 'AccountsApiDataSource', // HTTP polling fallback\n 'SnapDataSource', // Solana/Bitcoin/Tron snaps\n 'RpcDataSource', // Direct blockchain queries\n ]);\n }\n\n // ============================================================================\n // INITIALIZATION\n // ============================================================================\n\n #initializeState(): void {\n const { enabledNetworkMap } = this.messenger.call(\n 'NetworkEnablementController:getState',\n );\n this.#enabledChains = this.#extractEnabledChains(enabledNetworkMap);\n\n log('Initialized state', {\n enabledNetworkMap,\n enabledChains: this.#enabledChains,\n });\n }\n\n /**\n * Extract enabled chains from enabledNetworkMap.\n * Returns CAIP-2 chain IDs for all enabled networks across all namespaces.\n *\n * Note: For EIP155 (EVM) chains, the reference is normalized to decimal format\n * to ensure consistency with CAIP-2 standard and API responses.\n *\n * @param enabledNetworkMap - The enabled network map from NetworkEnablementController.\n * @returns Set of CAIP-2 chain IDs for all enabled networks.\n */\n #extractEnabledChains(\n enabledNetworkMap: NetworkEnablementControllerState['enabledNetworkMap'],\n ): Set<ChainId> {\n const chains = new Set<ChainId>();\n\n for (const [namespace, networks] of Object.entries(enabledNetworkMap)) {\n for (const [reference, isEnabled] of Object.entries(networks)) {\n if (isEnabled) {\n // Check if reference is already a full CAIP-2 chain ID (contains colon)\n if (reference.includes(':')) {\n // Already a full chain ID, use as-is\n chains.add(reference as ChainId);\n } else {\n // Normalize EIP155 chain references from hex to decimal (CAIP-2 standard)\n const normalizedReference = this.#normalizeChainReference(\n namespace,\n reference,\n );\n chains.add(`${namespace}:${normalizedReference}`);\n }\n }\n }\n }\n return chains;\n }\n\n /**\n * Normalize chain reference to CAIP-2 standard format.\n * For EIP155, converts hex chain IDs to decimal.\n *\n * @param namespace - The chain namespace (e.g., \"eip155\").\n * @param reference - The chain reference (e.g., \"0x1\" or \"1\").\n * @returns The normalized chain reference in decimal format.\n */\n #normalizeChainReference(namespace: string, reference: string): string {\n if (namespace === 'eip155' && reference.startsWith('0x')) {\n // Convert hex to decimal for EIP155 chains\n return parseInt(reference, 16).toString();\n }\n return reference;\n }\n\n #subscribeToEvents(): void {\n // Subscribe to account group changes (when user switches between account groups like Account 1 -> Account 2)\n this.messenger.subscribe(\n 'AccountTreeController:selectedAccountGroupChange',\n () => {\n this.#handleAccountGroupChanged().catch(console.error);\n },\n );\n\n // Subscribe to network enablement changes (only enabledNetworkMap)\n this.messenger.subscribe(\n 'NetworkEnablementController:stateChange',\n ({ enabledNetworkMap }) => {\n this.#handleEnabledNetworksChanged(enabledNetworkMap).catch(\n console.error,\n );\n },\n );\n\n // App lifecycle: start when opened, stop when closed\n this.messenger.subscribe('AppStateController:appOpened', () =>\n this.#start(),\n );\n this.messenger.subscribe('AppStateController:appClosed', () =>\n this.#stop(),\n );\n\n // Keyring lifecycle: start when unlocked, stop when locked\n this.messenger.subscribe('KeyringController:unlock', () => this.#start());\n this.messenger.subscribe('KeyringController:lock', () => this.#stop());\n }\n\n #registerActionHandlers(): void {\n this.messenger.registerActionHandler(\n 'AssetsController:getAssets',\n this.getAssets.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getAssetsBalance',\n this.getAssetsBalance.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getAssetMetadata',\n this.getAssetMetadata.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getAssetsPrice',\n this.getAssetsPrice.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:activeChainsUpdate',\n this.handleActiveChainsUpdate.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:assetsUpdate',\n this.handleAssetsUpdate.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:addCustomAsset',\n this.addCustomAsset.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:removeCustomAsset',\n this.removeCustomAsset.bind(this),\n );\n\n this.messenger.registerActionHandler(\n 'AssetsController:getCustomAssets',\n this.getCustomAssets.bind(this),\n );\n }\n\n // ============================================================================\n // DATA SOURCE MANAGEMENT\n // ============================================================================\n\n /**\n * Register data sources with the controller.\n * Order of the array determines subscription order.\n *\n * Data sources report chain changes by calling `AssetsController:activeChainsUpdate` action.\n *\n * @param dataSourceIds - Array of data source identifiers to register.\n */\n registerDataSources(dataSourceIds: DataSourceDefinition[]): void {\n for (const id of dataSourceIds) {\n log('Registering data source', { id });\n\n // Initialize available chains tracking for this source\n this.#dataSources.set(id, new Set());\n }\n }\n\n // ============================================================================\n // DATA SOURCE CHAIN MANAGEMENT\n // ============================================================================\n\n /**\n * Handle when a data source's active chains change.\n * Active chains are chains that are both supported AND available.\n * Updates centralized chain tracking and triggers re-selection if needed.\n *\n * Data sources should call this via `AssetsController:activeChainsUpdate` action.\n *\n * @param dataSourceId - The identifier of the data source reporting the change.\n * @param activeChains - Array of currently active chain IDs for this source.\n */\n handleActiveChainsUpdate(\n dataSourceId: string,\n activeChains: ChainId[],\n ): void {\n log('Data source active chains changed', {\n dataSourceId,\n chainCount: activeChains.length,\n chains: activeChains,\n });\n\n const previousChains = this.#dataSources.get(dataSourceId) ?? new Set();\n const newChains = new Set(activeChains);\n\n // Update centralized available chains tracking\n this.#dataSources.set(dataSourceId, newChains);\n\n // Check for changes\n const addedChains = activeChains.filter(\n (chain) => !previousChains.has(chain),\n );\n const removedChains = Array.from(previousChains).filter(\n (chain) => !newChains.has(chain),\n );\n\n if (addedChains.length > 0 || removedChains.length > 0) {\n // Refresh subscriptions to use updated data source availability\n this.#subscribeToDataSources();\n }\n\n // If chains were added and we have selected accounts, do one-time fetch\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n const addedEnabledChains = addedChains.filter((chain) =>\n this.#enabledChains.has(chain),\n );\n if (addedEnabledChains.length > 0) {\n log('Fetching balances for newly added chains', { addedEnabledChains });\n this.getAssets(this.#selectedAccounts, {\n chainIds: addedEnabledChains,\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch balance for added chains', { error });\n });\n }\n }\n }\n\n // ============================================================================\n // MIDDLEWARE EXECUTION\n // ============================================================================\n\n /**\n * Execute middlewares with request/response context.\n *\n * @param middlewares - Middlewares to execute in order.\n * @param request - The data request.\n * @param initialResponse - Optional initial response (for enriching existing data).\n * @returns The final DataResponse after all middlewares have processed.\n */\n async #executeMiddlewares(\n middlewares: Middleware[],\n request: DataRequest,\n initialResponse: DataResponse = {},\n ): Promise<DataResponse> {\n const chain = middlewares.reduceRight<NextFunction>(\n (next, middleware) =>\n async (\n ctx,\n ): Promise<{\n request: DataRequest;\n response: DataResponse;\n getAssetsState: () => AssetsControllerStateInternal;\n }> => {\n try {\n return await middleware(ctx, next);\n } catch (error) {\n console.error('[AssetsController] Middleware failed:', error);\n return next(ctx);\n }\n },\n async (ctx) => ctx,\n );\n\n const result = await chain({\n request,\n response: initialResponse,\n getAssetsState: () => this.state as AssetsControllerStateInternal,\n });\n return result.response;\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 },\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 // 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 response = await this.#executeMiddlewares(\n [\n this.messenger.call('AccountsApiDataSource:getAssetsMiddleware'),\n this.messenger.call('SnapDataSource:getAssetsMiddleware'),\n this.messenger.call('RpcDataSource:getAssetsMiddleware'),\n this.messenger.call('DetectionMiddleware:getAssetsMiddleware'),\n this.messenger.call('TokenDataSource:getAssetsMiddleware'),\n this.messenger.call('PriceDataSource:getAssetsMiddleware'),\n ],\n {\n accounts,\n chainIds,\n assetTypes,\n dataTypes,\n customAssets: customAssets.length > 0 ? customAssets : undefined,\n forceUpdate: true,\n },\n );\n await this.#updateState(response);\n }\n\n return this.#getAssetsFromState(accounts, chainIds, assetTypes);\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.assetsMetadata[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 // 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 *\n * @param accountId - The account ID to add the custom asset for.\n * @param assetId - The CAIP-19 asset ID to add.\n */\n async addCustomAsset(\n accountId: AccountId,\n assetId: Caip19AssetId,\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\n // Fetch data for the newly added custom asset\n const account = this.#selectedAccounts.find((a) => a.id === accountId);\n if (account) {\n const chainId = extractChainId(normalizedAssetId);\n await this.getAssets([account], {\n chainIds: [chainId],\n forceUpdate: true,\n });\n }\n }\n\n /**\n * Remove a custom asset from an account.\n *\n * @param accountId - The account ID to remove the custom asset from.\n * @param assetId - The CAIP-19 asset ID to remove.\n */\n removeCustomAsset(accountId: AccountId, assetId: Caip19AssetId): void {\n const normalizedAssetId = normalizeAssetId(assetId);\n\n log('Removing custom asset', { accountId, assetId: normalizedAssetId });\n\n this.update((state) => {\n const customAssets = state.customAssets as Record<string, string[]>;\n if (customAssets[accountId]) {\n customAssets[accountId] = customAssets[accountId].filter(\n (id) => id !== normalizedAssetId,\n );\n\n // Clean up empty arrays\n if (customAssets[accountId].length === 0) {\n delete customAssets[accountId];\n }\n }\n });\n }\n\n /**\n * Get all custom assets for an account.\n *\n * @param accountId - The account ID to get custom assets for.\n * @returns Array of CAIP-19 asset IDs for the account's custom assets.\n */\n getCustomAssets(accountId: AccountId): Caip19AssetId[] {\n return (this.state.customAssets[accountId] ?? []) as Caip19AssetId[];\n }\n\n // ============================================================================\n // SUBSCRIPTIONS\n // ============================================================================\n\n /**\n * Assign chains to data sources based on availability.\n * Returns a map of sourceId -> chains to handle.\n *\n * @param requestedChains - Array of chain IDs to assign to data sources.\n * @returns Map of sourceId to array of assigned chain IDs.\n */\n #assignChainsToDataSources(\n requestedChains: ChainId[],\n ): Map<string, ChainId[]> {\n const assignment = new Map<string, ChainId[]>();\n const remainingChains = new Set(requestedChains);\n\n for (const sourceId of this.#dataSources.keys()) {\n // Get available chains for this data source\n const availableChains = this.#dataSources.get(sourceId);\n if (!availableChains || availableChains.size === 0) {\n continue;\n }\n\n const chainsForThisSource: ChainId[] = [];\n\n for (const chainId of remainingChains) {\n // Check if this chain is available on this source\n if (availableChains.has(chainId)) {\n chainsForThisSource.push(chainId);\n remainingChains.delete(chainId);\n }\n }\n\n if (chainsForThisSource.length > 0) {\n assignment.set(sourceId, chainsForThisSource);\n log('Assigned chains to data source', {\n sourceId,\n chains: chainsForThisSource,\n });\n }\n }\n\n return assignment;\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 * @param options - Subscription options.\n * @param options.updateInterval - Polling interval in ms.\n */\n subscribeAssetsPrice(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n options: { updateInterval?: number } = {},\n ): void {\n const { updateInterval = this.#defaultUpdateInterval } = options;\n const subscriptionKey = 'ds:PriceDataSource';\n\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n // Fire-and-forget - errors are handled internally by PriceDataSource\n this.messenger\n .call('PriceDataSource:subscribe', {\n request: {\n accounts,\n chainIds,\n dataTypes: ['price'],\n updateInterval,\n },\n subscriptionId: subscriptionKey,\n isUpdate,\n })\n .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\n // Fire-and-forget - errors are handled internally by PriceDataSource\n this.messenger\n .call('PriceDataSource:unsubscribe', subscriptionKey)\n .catch(console.error);\n\n existingSubscription.unsubscribe();\n }\n\n // ============================================================================\n // STATE MANAGEMENT\n // ============================================================================\n\n async #updateState(response: DataResponse): Promise<void> {\n // Normalize asset IDs (checksum EVM addresses) before storing in state\n const normalizedResponse = normalizeResponse(response);\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 Draft<Json>\n const metadata = state.assetsMetadata as unknown as Record<\n string,\n unknown\n >;\n const balances = state.assetsBalance as unknown as Record<\n string,\n Record<string, unknown>\n >;\n const prices = state.assetsPrice as unknown as Record<string, unknown>;\n\n if (normalizedResponse.assetsMetadata) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsMetadata,\n )) {\n if (\n !isEqual(\n previousState.assetsMetadata[key as Caip19AssetId],\n value,\n )\n ) {\n changedMetadata.push(key);\n }\n metadata[key] = value;\n }\n }\n\n if (normalizedResponse.assetsBalance) {\n for (const [accountId, accountBalances] of Object.entries(\n normalizedResponse.assetsBalance,\n )) {\n const previousBalances =\n previousState.assetsBalance[accountId] ?? {};\n\n if (!balances[accountId]) {\n balances[accountId] = {};\n }\n\n for (const [assetId, balance] of Object.entries(accountBalances)) {\n const previousBalance = previousBalances[\n assetId as Caip19AssetId\n ] as { amount: string } | undefined;\n const balanceData = balance as { amount: string };\n const newAmount = balanceData.amount;\n const oldAmount = previousBalance?.amount;\n\n // Track if balance actually changed\n if (oldAmount !== newAmount) {\n changedBalances.push({\n accountId,\n assetId,\n oldAmount,\n newAmount,\n });\n }\n }\n\n Object.assign(balances[accountId], accountBalances);\n }\n }\n\n // Update prices in state\n if (normalizedResponse.assetsPrice) {\n for (const [key, value] of Object.entries(\n normalizedResponse.assetsPrice,\n )) {\n prices[key] = value;\n }\n }\n });\n\n // Calculate changed prices\n const changedPriceAssets: string[] = normalizedResponse.assetsPrice\n ? Object.keys(normalizedResponse.assetsPrice).filter(\n (assetId) =>\n !isEqual(\n previousPrices[assetId as Caip19AssetId],\n normalizedResponse.assetsPrice?.[assetId as Caip19AssetId],\n ),\n )\n : [];\n\n // Log only actual changes\n if (\n changedBalances.length > 0 ||\n changedMetadata.length > 0 ||\n changedPriceAssets.length > 0\n ) {\n log('State updated', {\n changedBalances:\n changedBalances.length > 0 ? changedBalances : undefined,\n changedMetadataCount:\n changedMetadata.length > 0 ? changedMetadata.length : undefined,\n changedPricesCount:\n changedPriceAssets.length > 0\n ? changedPriceAssets.length\n : undefined,\n newAssets:\n Object.keys(detectedAssets).length > 0\n ? Object.entries(detectedAssets).map(([accountId, assets]) => ({\n accountId,\n assets,\n }))\n : undefined,\n });\n }\n\n // Publish balance changed events\n for (const change of changedBalances) {\n this.messenger.publish('AssetsController:balanceChanged', {\n accountId: change.accountId,\n assetId: change.assetId as Caip19AssetId,\n previousAmount: change.oldAmount ?? '0',\n newAmount: change.newAmount,\n });\n }\n\n // Publish price changed event with full price data\n if (changedPriceAssets.length > 0 && normalizedResponse.assetsPrice) {\n // Build prices object with only changed prices\n const changedPrices: Record<Caip19AssetId, AssetPrice> = {};\n for (const assetId of changedPriceAssets) {\n const price =\n normalizedResponse.assetsPrice[assetId as Caip19AssetId];\n if (price) {\n changedPrices[assetId as Caip19AssetId] = price;\n }\n }\n this.messenger.publish('AssetsController:priceChanged', {\n prices: changedPrices,\n });\n }\n\n // Publish assets detected events\n for (const [accountId, assetIds] of Object.entries(detectedAssets)) {\n if (assetIds.length > 0) {\n this.messenger.publish('AssetsController:assetsDetected', {\n accountId,\n assetIds,\n });\n }\n }\n } finally {\n releaseLock();\n }\n }\n\n #getAssetsFromState(\n accounts: InternalAccount[],\n chainIds: ChainId[],\n assetTypes: AssetType[],\n ): Record<AccountId, Record<Caip19AssetId, Asset>> {\n const result: Record<AccountId, Record<Caip19AssetId, Asset>> = {};\n // Convert to Sets for O(1) lookups\n const chainIdSet = new Set(chainIds);\n const assetTypeSet = new Set(assetTypes);\n\n for (const account of accounts) {\n result[account.id] = {};\n\n const accountBalances = this.state.assetsBalance[account.id] ?? {};\n\n for (const [assetId, balance] of Object.entries(accountBalances)) {\n const typedAssetId = assetId as Caip19AssetId;\n const assetChainId = extractChainId(typedAssetId);\n\n if (!chainIdSet.has(assetChainId)) {\n continue;\n }\n\n const metadataRaw = this.state.assetsMetadata[typedAssetId];\n\n // Skip assets without metadata\n if (!metadataRaw) {\n continue;\n }\n\n const metadata = metadataRaw as AssetMetadata;\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 as AssetBalance;\n const priceRaw = this.state.assetsPrice[typedAssetId] as AssetPrice;\n const price: AssetPrice = priceRaw ?? {\n price: 0,\n lastUpdated: 0,\n };\n\n // Compute fiat value using BigNumber for precision\n // Note: typedBalance.amount is already in human-readable format (e.g., \"1\" for 1 ETH)\n // so we do NOT divide by 10^decimals here\n const balanceAmount = new BigNumberJS(typedBalance.amount || '0');\n const fiatValue = balanceAmount\n .multipliedBy(price.price || 0)\n .toNumber();\n\n const asset: Asset = {\n id: typedAssetId,\n chainId: assetChainId,\n balance: typedBalance,\n metadata,\n price,\n fiatValue,\n };\n\n result[account.id][typedAssetId] = asset;\n }\n }\n\n return result;\n }\n\n /**\n * Maps a token standard to its corresponding asset type.\n *\n * @param tokenStandard - The token standard from metadata.\n * @returns The corresponding asset type.\n */\n #tokenStandardToAssetType(tokenStandard: string): AssetType {\n switch (tokenStandard) {\n case 'native':\n case 'erc20':\n case 'spl':\n return 'fungible';\n case 'erc721':\n return 'nft';\n case 'erc1155':\n // ERC1155 can be either fungible or non-fungible, treat as collectible\n return 'collectible';\n default:\n // Unknown standards default to fungible\n return 'fungible';\n }\n }\n\n // ============================================================================\n // START / STOP\n // ============================================================================\n\n /**\n * Start asset tracking: subscribe to updates and fetch current balances.\n * Called when app opens, account changes, or keyring unlocks.\n */\n #start(): void {\n log('Starting asset tracking', {\n selectedAccountCount: this.#selectedAccounts.length,\n enabledChainCount: this.#enabledChains.size,\n });\n\n this.#subscribeToDataSources();\n if (this.#selectedAccounts.length > 0) {\n this.getAssets(this.#selectedAccounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n }).catch((error) => {\n log('Failed to fetch assets', error);\n });\n }\n }\n\n /**\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 // 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 // Convert to array first to avoid modifying map during iteration\n const subscriptionKeys = [...this.#activeSubscriptions.keys()];\n for (const subscriptionKey of subscriptionKeys) {\n // Extract sourceId from subscription key (format: \"ds:${sourceId}\")\n if (subscriptionKey.startsWith('ds:')) {\n const sourceId = subscriptionKey.slice(3);\n this.#unsubscribeDataSource(sourceId);\n }\n }\n this.#activeSubscriptions.clear();\n }\n\n /**\n * Subscribe to asset updates for all selected accounts.\n */\n #subscribeToDataSources(): void {\n if (this.#selectedAccounts.length === 0) {\n return;\n }\n\n // Subscribe to balance updates (batched by data source)\n this.#subscribeAssetsBalance();\n\n // Subscribe to price updates for all assets held by selected accounts\n this.subscribeAssetsPrice(this.#selectedAccounts, [...this.#enabledChains]);\n }\n\n /**\n * Subscribe to balance updates for all selected accounts.\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 #subscribeAssetsBalance(): void {\n // Step 1: Build chain -> accounts mapping based on account scopes and enabled networks\n const chainToAccounts = this.#buildChainToAccountsMap(\n this.#selectedAccounts,\n this.#enabledChains,\n );\n\n // Step 2: Split by data source active chains (ordered by priority)\n // Get all chains that need to be subscribed\n const remainingChains = new Set(chainToAccounts.keys());\n\n // Assign chains to data sources based on availability (ordered by priority)\n const chainAssignment = this.#assignChainsToDataSources(\n Array.from(remainingChains),\n );\n\n log('Subscribe - chain assignment', {\n totalChains: remainingChains.size,\n dataSourceAssignments: Array.from(chainAssignment.entries()).map(\n ([sourceId, chains]) => ({ sourceId, chainCount: chains.length }),\n ),\n });\n\n // Subscribe to each data source with its assigned chains and relevant accounts\n for (const sourceId of this.#dataSources.keys()) {\n const assignedChains = chainAssignment.get(sourceId);\n\n if (!assignedChains || assignedChains.length === 0) {\n // Unsubscribe from data sources with no assigned chains\n this.#unsubscribeDataSource(sourceId);\n continue;\n }\n\n // Collect unique accounts that need any of the assigned chains\n const accountsForSource = this.#getAccountsForChains(\n assignedChains,\n chainToAccounts,\n );\n\n if (accountsForSource.length === 0) {\n continue;\n }\n\n // Subscribe with ONE call per data source\n this.#subscribeToDataSource(sourceId, accountsForSource, assignedChains);\n }\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\n for (const account of accounts) {\n const accountChains = this.#getEnabledChainsForAccount(account);\n\n for (const chainId of accountChains) {\n if (!chainsToSubscribe.has(chainId)) {\n continue;\n }\n\n const existingAccounts = chainToAccounts.get(chainId) ?? [];\n existingAccounts.push(account);\n chainToAccounts.set(chainId, existingAccounts);\n }\n }\n\n return chainToAccounts;\n }\n\n /**\n * Get unique accounts that need any of the specified chains.\n *\n * @param chains - Array of chain IDs to find accounts for.\n * @param chainToAccounts - Map of chainId to accounts.\n * @returns Array of unique accounts that need any of the specified chains.\n */\n #getAccountsForChains(\n chains: ChainId[],\n chainToAccounts: Map<ChainId, InternalAccount[]>,\n ): InternalAccount[] {\n const accountIds = new Set<string>();\n const accounts: InternalAccount[] = [];\n\n for (const chainId of chains) {\n const chainAccounts = chainToAccounts.get(chainId) ?? [];\n for (const account of chainAccounts) {\n if (!accountIds.has(account.id)) {\n accountIds.add(account.id);\n accounts.push(account);\n }\n }\n }\n\n return accounts;\n }\n\n /**\n * Subscribe to a specific data source with accounts and chains.\n * Uses the data source ID as the subscription key for batching.\n *\n * @param sourceId - The data source identifier.\n * @param accounts - Array of accounts to subscribe for.\n * @param chains - Array of chain IDs to subscribe for.\n */\n #subscribeToDataSource(\n sourceId: string,\n accounts: InternalAccount[],\n chains: ChainId[],\n ): void {\n const subscriptionKey = `ds:${sourceId}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n const isUpdate = existingSubscription !== undefined;\n\n log('Subscribe to data source', {\n sourceId,\n subscriptionKey,\n isUpdate,\n accountCount: accounts.length,\n chainCount: chains.length,\n });\n\n // Call data source subscribe action via Messenger (fire-and-forget)\n (async (): Promise<void> => {\n try {\n await (this.messenger.call as CallableFunction)(\n `${sourceId}:subscribe`,\n {\n request: {\n accounts,\n chainIds: chains,\n assetTypes: ['fungible'],\n dataTypes: ['balance'],\n updateInterval: this.#defaultUpdateInterval,\n },\n subscriptionId: subscriptionKey,\n isUpdate,\n },\n );\n } catch (error) {\n console.error(\n `[AssetsController] Failed to subscribe to '${sourceId}':`,\n error,\n );\n }\n })().catch(console.error);\n\n // Track subscription\n const subscription: SubscriptionResponse = {\n chains,\n accountId: subscriptionKey,\n assetTypes: ['fungible'],\n dataTypes: ['balance', 'price'],\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 sourceId - The data source identifier to unsubscribe from.\n */\n #unsubscribeDataSource(sourceId: string): void {\n const subscriptionKey = `ds:${sourceId}`;\n const existingSubscription = this.#activeSubscriptions.get(subscriptionKey);\n\n if (existingSubscription) {\n // Fire-and-forget unsubscribe call\n (async (): Promise<void> => {\n try {\n await (this.messenger.call as CallableFunction)(\n `${sourceId}:unsubscribe`,\n subscriptionKey,\n );\n } catch {\n // Ignore errors - source may not have been subscribed\n }\n })().catch(() => {\n // Ignore errors - source may not have been subscribed\n });\n existingSubscription.unsubscribe();\n }\n }\n\n // ============================================================================\n // HELPERS\n // ============================================================================\n\n /**\n * Get the chains that an account supports based on its scopes.\n * Returns the intersection of the account's scopes and the enabled chains.\n *\n * @param account - The account to get supported chains for.\n * @returns Array of ChainIds that the account supports and are enabled.\n */\n #getEnabledChainsForAccount(account: InternalAccount): ChainId[] {\n // Account scopes are CAIP-2 chain IDs like \"eip155:1\", \"solana:mainnet\", \"bip122:...\"\n const scopes = account.scopes ?? [];\n const result: ChainId[] = [];\n\n for (const scope of scopes) {\n const [namespace, reference] = (scope as string).split(':');\n\n // Wildcard scope (e.g., \"eip155:0\" means all enabled chains in that namespace)\n if (reference === '0') {\n for (const chain of this.#enabledChains) {\n if (chain.startsWith(`${namespace}:`)) {\n result.push(chain);\n }\n }\n } else if (namespace === 'eip155' && reference?.startsWith('0x')) {\n // Normalize hex to decimal for EIP155\n result.push(`eip155:${parseInt(reference, 16)}` as ChainId);\n } else {\n result.push(scope);\n }\n }\n\n return result;\n }\n\n // ============================================================================\n // EVENT HANDLERS\n // ============================================================================\n\n async #handleAccountGroupChanged(): Promise<void> {\n const accounts = this.#selectedAccounts;\n\n log('Account group changed', {\n accountCount: accounts.length,\n accountIds: accounts.map((a) => a.id),\n });\n\n // Subscribe and fetch for the new account group\n this.#subscribeToDataSources();\n if (accounts.length > 0) {\n await this.getAssets(accounts, {\n chainIds: [...this.#enabledChains],\n forceUpdate: true,\n });\n }\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.#subscribeToDataSources();\n\n // Do one-time fetch for newly enabled chains\n if (addedChains.length > 0 && this.#selectedAccounts.length > 0) {\n await this.getAssets(this.#selectedAccounts, {\n chainIds: addedChains,\n forceUpdate: true,\n });\n }\n }\n\n /**\n * Handle assets updated from a data source.\n * Called via `AssetsController:assetsUpdate` action by data sources.\n *\n * @param response - The data response with updated assets\n * @param sourceId - The data source ID reporting the update\n */\n async handleAssetsUpdate(\n response: DataResponse,\n sourceId: string,\n ): Promise<void> {\n log('Assets updated from data source', {\n sourceId,\n hasBalance: Boolean(response.assetsBalance),\n hasPrice: Boolean(response.assetsPrice),\n });\n await this.#handleSubscriptionUpdate(response, sourceId);\n }\n\n /**\n * Handle an async update from a data source subscription.\n * Enriches response with token metadata before updating state.\n *\n * @param response - The data response from the data source.\n * @param _sourceId - The source ID (unused but kept for logging context).\n * @param request - Optional original request for context.\n */\n async #handleSubscriptionUpdate(\n response: DataResponse,\n _sourceId?: string,\n request?: DataRequest,\n ): Promise<void> {\n // Run through enrichment middlewares (Event Stack: Detection → Token → Price)\n // Include 'metadata' in dataTypes so TokenDataSource runs to enrich detected assets\n const enrichedResponse = await this.#executeMiddlewares(\n [\n this.messenger.call('DetectionMiddleware:getAssetsMiddleware'),\n this.messenger.call('TokenDataSource:getAssetsMiddleware'),\n this.messenger.call('PriceDataSource:getAssetsMiddleware'),\n ],\n request ?? {\n accounts: [],\n chainIds: [],\n dataTypes: ['balance', 'metadata', 'price'],\n },\n response,\n );\n\n // Update state\n await this.#updateState(enrichedResponse);\n }\n\n // ============================================================================\n // CLEANUP\n // ============================================================================\n\n destroy(): void {\n log('Destroying AssetsController', {\n dataSourceCount: this.#dataSources.size,\n subscriptionCount: this.#activeSubscriptions.size,\n });\n\n // Clear data sources\n this.#dataSources.clear();\n\n // Stop all active subscriptions\n this.#stop();\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:activeChainsUpdate',\n );\n this.messenger.unregisterActionHandler('AssetsController:assetsUpdate');\n this.messenger.unregisterActionHandler('AssetsController:addCustomAsset');\n this.messenger.unregisterActionHandler(\n 'AssetsController:removeCustomAsset',\n );\n this.messenger.unregisterActionHandler('AssetsController:getCustomAssets');\n }\n}\n"]}