@metamask/assets-controllers 27.2.0 → 28.0.0

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 (123) hide show
  1. package/CHANGELOG.md +17 -1
  2. package/dist/AssetsContractController.js +6 -6
  3. package/dist/AssetsContractController.mjs +5 -5
  4. package/dist/CurrencyRateController.js +3 -3
  5. package/dist/CurrencyRateController.mjs +2 -2
  6. package/dist/NftController.js +3 -5
  7. package/dist/NftController.mjs +2 -4
  8. package/dist/NftDetectionController.js +4 -4
  9. package/dist/NftDetectionController.mjs +3 -3
  10. package/dist/Standards/ERC20Standard.js +3 -3
  11. package/dist/Standards/ERC20Standard.mjs +2 -2
  12. package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.js +3 -3
  13. package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.mjs +2 -2
  14. package/dist/Standards/NftStandards/ERC721/ERC721Standard.js +3 -3
  15. package/dist/Standards/NftStandards/ERC721/ERC721Standard.mjs +2 -2
  16. package/dist/TokenDetectionController.js +3 -3
  17. package/dist/TokenDetectionController.mjs +2 -2
  18. package/dist/TokenListController.js +4 -4
  19. package/dist/TokenListController.mjs +3 -3
  20. package/dist/TokenRatesController.js +4 -4
  21. package/dist/TokenRatesController.mjs +3 -3
  22. package/dist/TokensController.js +6 -6
  23. package/dist/TokensController.mjs +5 -5
  24. package/dist/assetsUtil.js +2 -8
  25. package/dist/assetsUtil.mjs +1 -7
  26. package/dist/{chunk-IFJWX5RY.mjs → chunk-4PAJC7WC.mjs} +93 -132
  27. package/dist/chunk-4PAJC7WC.mjs.map +1 -0
  28. package/dist/{chunk-WIVSZVF4.mjs → chunk-AC5HGIHS.mjs} +3 -3
  29. package/dist/chunk-AC5HGIHS.mjs.map +1 -0
  30. package/dist/{chunk-355MWUUQ.js → chunk-BM7EPTR6.js} +3 -3
  31. package/dist/{chunk-4YU7CT4O.js → chunk-C4JDCQG2.js} +7 -7
  32. package/dist/{chunk-ZKJSG4TA.mjs → chunk-CNICDMRI.mjs} +4 -4
  33. package/dist/chunk-CNICDMRI.mjs.map +1 -0
  34. package/dist/{chunk-VY7TUOK2.js → chunk-D3N35ABH.js} +3 -3
  35. package/dist/{chunk-RELRWIJN.js → chunk-D6BDJYSK.js} +2 -68
  36. package/dist/chunk-D6BDJYSK.js.map +1 -0
  37. package/dist/{chunk-C4PQK53K.js → chunk-DYH5P3VY.js} +5 -2
  38. package/dist/chunk-DYH5P3VY.js.map +1 -0
  39. package/dist/{chunk-NUBQRDKI.js → chunk-E2SLNUD4.js} +11 -11
  40. package/dist/{chunk-VHCPQS4X.js → chunk-GCWWC4X3.js} +5 -5
  41. package/dist/{chunk-KRTEFV4Q.mjs → chunk-IURNPMBB.mjs} +2 -2
  42. package/dist/{chunk-WSOHBBCO.js → chunk-JEGLHDNA.js} +42 -56
  43. package/dist/chunk-JEGLHDNA.js.map +1 -0
  44. package/dist/{chunk-GLZ66IRX.mjs → chunk-KDMDWUVX.mjs} +41 -55
  45. package/dist/chunk-KDMDWUVX.mjs.map +1 -0
  46. package/dist/{chunk-MQ7TIWBK.mjs → chunk-KS4MOA2T.mjs} +2 -2
  47. package/dist/{chunk-H6TOSWUM.js → chunk-LRKDZWS6.js} +3 -3
  48. package/dist/{chunk-CEENXWOB.mjs → chunk-N5H5KDJZ.mjs} +2 -2
  49. package/dist/{chunk-2IZUMKMY.mjs → chunk-NHX4QGOD.mjs} +3 -3
  50. package/dist/{chunk-TCO22VIO.mjs → chunk-PWZE6KJV.mjs} +5 -2
  51. package/dist/chunk-PWZE6KJV.mjs.map +1 -0
  52. package/dist/{chunk-KSGMD24G.js → chunk-Q6TLNHYC.js} +7 -7
  53. package/dist/{chunk-523YP4Z3.mjs → chunk-QQAG6ECS.mjs} +3 -3
  54. package/dist/{chunk-XEZJ5XMV.js → chunk-QVSSEGA5.js} +7 -7
  55. package/dist/chunk-QVSSEGA5.js.map +1 -0
  56. package/dist/{chunk-23F5W3A2.mjs → chunk-QWACHXRH.mjs} +2 -2
  57. package/dist/{chunk-KCUSDOIE.mjs → chunk-RDWXY4OF.mjs} +4 -4
  58. package/dist/{chunk-PD624ZNK.js → chunk-RUPB4ELK.js} +6 -6
  59. package/dist/{chunk-6NO7Z5DL.js → chunk-TGKUZNML.js} +5 -5
  60. package/dist/chunk-TGKUZNML.js.map +1 -0
  61. package/dist/{chunk-AQIXUBTK.js → chunk-UNKIHWZH.js} +93 -132
  62. package/dist/chunk-UNKIHWZH.js.map +1 -0
  63. package/dist/{chunk-76KGJJJV.mjs → chunk-ZM42FNAQ.mjs} +5 -5
  64. package/dist/{chunk-E5PIGGYR.mjs → chunk-ZPAX2HLQ.mjs} +1 -67
  65. package/dist/chunk-ZPAX2HLQ.mjs.map +1 -0
  66. package/dist/crypto-compare.js +2 -2
  67. package/dist/crypto-compare.mjs +1 -1
  68. package/dist/index.js +15 -15
  69. package/dist/index.mjs +18 -18
  70. package/dist/token-service.js +3 -3
  71. package/dist/token-service.mjs +2 -2
  72. package/dist/tsconfig.build.tsbuildinfo +1 -1
  73. package/dist/types/CurrencyRateController.d.ts +7 -7
  74. package/dist/types/CurrencyRateController.d.ts.map +1 -1
  75. package/dist/types/NftController.d.ts +20 -95
  76. package/dist/types/NftController.d.ts.map +1 -1
  77. package/dist/types/NftDetectionController.d.ts +184 -0
  78. package/dist/types/NftDetectionController.d.ts.map +1 -1
  79. package/dist/types/Standards/NftStandards/ERC721/ERC721Standard.d.ts.map +1 -1
  80. package/dist/types/TokenBalancesController.d.ts +10 -10
  81. package/dist/types/TokenBalancesController.d.ts.map +1 -1
  82. package/dist/types/TokenDetectionController.d.ts +9 -9
  83. package/dist/types/TokenDetectionController.d.ts.map +1 -1
  84. package/dist/types/TokenListController.d.ts +12 -12
  85. package/dist/types/TokenListController.d.ts.map +1 -1
  86. package/dist/types/TokensController.d.ts +10 -10
  87. package/dist/types/TokensController.d.ts.map +1 -1
  88. package/dist/types/assetsUtil.d.ts +1 -21
  89. package/dist/types/assetsUtil.d.ts.map +1 -1
  90. package/dist/types/crypto-compare.d.ts.map +1 -1
  91. package/dist/types/token-prices-service/abstract-token-prices-service.d.ts +3 -3
  92. package/dist/types/token-prices-service/abstract-token-prices-service.d.ts.map +1 -1
  93. package/dist/types/token-prices-service/codefi-v2.d.ts +2 -2
  94. package/dist/types/token-prices-service/codefi-v2.d.ts.map +1 -1
  95. package/package.json +12 -12
  96. package/dist/chunk-6NO7Z5DL.js.map +0 -1
  97. package/dist/chunk-AQIXUBTK.js.map +0 -1
  98. package/dist/chunk-C4PQK53K.js.map +0 -1
  99. package/dist/chunk-E5PIGGYR.mjs.map +0 -1
  100. package/dist/chunk-GLZ66IRX.mjs.map +0 -1
  101. package/dist/chunk-IFJWX5RY.mjs.map +0 -1
  102. package/dist/chunk-RELRWIJN.js.map +0 -1
  103. package/dist/chunk-TCO22VIO.mjs.map +0 -1
  104. package/dist/chunk-WIVSZVF4.mjs.map +0 -1
  105. package/dist/chunk-WSOHBBCO.js.map +0 -1
  106. package/dist/chunk-XEZJ5XMV.js.map +0 -1
  107. package/dist/chunk-ZKJSG4TA.mjs.map +0 -1
  108. /package/dist/{chunk-355MWUUQ.js.map → chunk-BM7EPTR6.js.map} +0 -0
  109. /package/dist/{chunk-4YU7CT4O.js.map → chunk-C4JDCQG2.js.map} +0 -0
  110. /package/dist/{chunk-VY7TUOK2.js.map → chunk-D3N35ABH.js.map} +0 -0
  111. /package/dist/{chunk-NUBQRDKI.js.map → chunk-E2SLNUD4.js.map} +0 -0
  112. /package/dist/{chunk-VHCPQS4X.js.map → chunk-GCWWC4X3.js.map} +0 -0
  113. /package/dist/{chunk-KRTEFV4Q.mjs.map → chunk-IURNPMBB.mjs.map} +0 -0
  114. /package/dist/{chunk-MQ7TIWBK.mjs.map → chunk-KS4MOA2T.mjs.map} +0 -0
  115. /package/dist/{chunk-H6TOSWUM.js.map → chunk-LRKDZWS6.js.map} +0 -0
  116. /package/dist/{chunk-CEENXWOB.mjs.map → chunk-N5H5KDJZ.mjs.map} +0 -0
  117. /package/dist/{chunk-2IZUMKMY.mjs.map → chunk-NHX4QGOD.mjs.map} +0 -0
  118. /package/dist/{chunk-KSGMD24G.js.map → chunk-Q6TLNHYC.js.map} +0 -0
  119. /package/dist/{chunk-523YP4Z3.mjs.map → chunk-QQAG6ECS.mjs.map} +0 -0
  120. /package/dist/{chunk-23F5W3A2.mjs.map → chunk-QWACHXRH.mjs.map} +0 -0
  121. /package/dist/{chunk-KCUSDOIE.mjs.map → chunk-RDWXY4OF.mjs.map} +0 -0
  122. /package/dist/{chunk-PD624ZNK.js.map → chunk-RUPB4ELK.js.map} +0 -0
  123. /package/dist/{chunk-76KGJJJV.mjs.map → chunk-ZM42FNAQ.mjs.map} +0 -0
@@ -1,12 +1,12 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
2
2
 
3
- var _chunk6NO7Z5DLjs = require('./chunk-6NO7Z5DL.js');
3
+ var _chunkTGKUZNMLjs = require('./chunk-TGKUZNML.js');
4
4
 
5
5
 
6
- var _chunkVHCPQS4Xjs = require('./chunk-VHCPQS4X.js');
6
+ var _chunkGCWWC4X3js = require('./chunk-GCWWC4X3.js');
7
7
 
8
8
 
9
- var _chunk355MWUUQjs = require('./chunk-355MWUUQ.js');
9
+ var _chunkBM7EPTR6js = require('./chunk-BM7EPTR6.js');
10
10
 
11
11
  // src/AssetsContractController.ts
12
12
  var _contracts = require('@ethersproject/contracts');
@@ -112,7 +112,7 @@ var AssetsContractController = class extends _basecontroller.BaseControllerV1 {
112
112
  */
113
113
  getERC20Standard(networkClientId) {
114
114
  const provider = this.getProvider(networkClientId);
115
- return new (0, _chunk355MWUUQjs.ERC20Standard)(provider);
115
+ return new (0, _chunkBM7EPTR6js.ERC20Standard)(provider);
116
116
  }
117
117
  /**
118
118
  * Get a ERC721Standard instance using the relevant provider instance.
@@ -122,7 +122,7 @@ var AssetsContractController = class extends _basecontroller.BaseControllerV1 {
122
122
  */
123
123
  getERC721Standard(networkClientId) {
124
124
  const provider = this.getProvider(networkClientId);
125
- return new (0, _chunk6NO7Z5DLjs.ERC721Standard)(provider);
125
+ return new (0, _chunkTGKUZNMLjs.ERC721Standard)(provider);
126
126
  }
127
127
  /**
128
128
  * Get a ERC1155Standard instance using the relevant provider instance.
@@ -132,7 +132,7 @@ var AssetsContractController = class extends _basecontroller.BaseControllerV1 {
132
132
  */
133
133
  getERC1155Standard(networkClientId) {
134
134
  const provider = this.getProvider(networkClientId);
135
- return new (0, _chunkVHCPQS4Xjs.ERC1155Standard)(provider);
135
+ return new (0, _chunkGCWWC4X3js.ERC1155Standard)(provider);
136
136
  }
137
137
  /**
138
138
  * Get balance or count for current account on specific asset contract.
@@ -358,4 +358,4 @@ var AssetsContractController_default = AssetsContractController;
358
358
 
359
359
 
360
360
  exports.SINGLE_CALL_BALANCES_ADDRESS_BY_CHAINID = SINGLE_CALL_BALANCES_ADDRESS_BY_CHAINID; exports.MISSING_PROVIDER_ERROR = MISSING_PROVIDER_ERROR; exports.AssetsContractController = AssetsContractController; exports.AssetsContractController_default = AssetsContractController_default;
361
- //# sourceMappingURL=chunk-4YU7CT4O.js.map
361
+ //# sourceMappingURL=chunk-C4JDCQG2.js.map
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  isTokenDetectionSupportedForNetwork
3
- } from "./chunk-E5PIGGYR.mjs";
3
+ } from "./chunk-ZPAX2HLQ.mjs";
4
4
  import {
5
5
  __privateAdd,
6
6
  __privateGet,
@@ -229,7 +229,7 @@ registerEventListeners_fn = function() {
229
229
  const isDetectionChangedFromPreferences = __privateGet(this, _isDetectionEnabledFromPreferences) !== useTokenDetection;
230
230
  __privateSet(this, _selectedAddress, newSelectedAddress);
231
231
  __privateSet(this, _isDetectionEnabledFromPreferences, useTokenDetection);
232
- if (useTokenDetection && (isSelectedAddressChanged || isDetectionChangedFromPreferences)) {
232
+ if (isSelectedAddressChanged || isDetectionChangedFromPreferences) {
233
233
  await __privateMethod(this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
234
234
  selectedAddress: __privateGet(this, _selectedAddress)
235
235
  });
@@ -240,7 +240,7 @@ registerEventListeners_fn = function() {
240
240
  "AccountsController:selectedAccountChange",
241
241
  async ({ address: newSelectedAddress }) => {
242
242
  const isSelectedAddressChanged = __privateGet(this, _selectedAddress) !== newSelectedAddress;
243
- if (isSelectedAddressChanged && __privateGet(this, _isDetectionEnabledFromPreferences)) {
243
+ if (isSelectedAddressChanged) {
244
244
  __privateSet(this, _selectedAddress, newSelectedAddress);
245
245
  await __privateMethod(this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
246
246
  selectedAddress: __privateGet(this, _selectedAddress)
@@ -408,4 +408,4 @@ export {
408
408
  TokenDetectionController,
409
409
  TokenDetectionController_default
410
410
  };
411
- //# sourceMappingURL=chunk-ZKJSG4TA.mjs.map
411
+ //# sourceMappingURL=chunk-CNICDMRI.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/TokenDetectionController.ts"],"sourcesContent":["import type {\n AccountsControllerGetSelectedAccountAction,\n AccountsControllerSelectedAccountChangeEvent,\n} from '@metamask/accounts-controller';\nimport type {\n RestrictedControllerMessenger,\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n} from '@metamask/base-controller';\nimport contractMap from '@metamask/contract-metadata';\nimport { ChainId, safelyExecute } from '@metamask/controller-utils';\nimport type {\n KeyringControllerGetStateAction,\n KeyringControllerLockEvent,\n KeyringControllerUnlockEvent,\n} from '@metamask/keyring-controller';\nimport type {\n NetworkClientId,\n NetworkControllerGetNetworkClientByIdAction,\n NetworkControllerGetNetworkConfigurationByNetworkClientId,\n NetworkControllerGetStateAction,\n NetworkControllerNetworkDidChangeEvent,\n} from '@metamask/network-controller';\nimport { StaticIntervalPollingController } from '@metamask/polling-controller';\nimport type {\n PreferencesControllerGetStateAction,\n PreferencesControllerStateChangeEvent,\n} from '@metamask/preferences-controller';\nimport type { Hex } from '@metamask/utils';\n\nimport type { AssetsContractController } from './AssetsContractController';\nimport { isTokenDetectionSupportedForNetwork } from './assetsUtil';\nimport type {\n GetTokenListState,\n TokenListMap,\n TokenListStateChange,\n} from './TokenListController';\nimport type { Token } from './TokenRatesController';\nimport type {\n TokensControllerAddDetectedTokensAction,\n TokensControllerGetStateAction,\n} from './TokensController';\n\nconst DEFAULT_INTERVAL = 180000;\n\n/**\n * Compare 2 given strings and return boolean\n * eg: \"foo\" and \"FOO\" => true\n * eg: \"foo\" and \"bar\" => false\n * eg: \"foo\" and 123 => false\n *\n * @param value1 - first string to compare\n * @param value2 - first string to compare\n * @returns true if 2 strings are identical when they are lowercase\n */\nexport function isEqualCaseInsensitive(\n value1: string,\n value2: string,\n): boolean {\n if (typeof value1 !== 'string' || typeof value2 !== 'string') {\n return false;\n }\n return value1.toLowerCase() === value2.toLowerCase();\n}\n\ntype LegacyToken = {\n name: string;\n logo: `${string}.svg`;\n symbol: string;\n decimals: number;\n erc20?: boolean;\n erc721?: boolean;\n};\n\ntype TokenDetectionMap = {\n [P in keyof TokenListMap]: Omit<TokenListMap[P], 'occurrences'>;\n};\n\nexport const STATIC_MAINNET_TOKEN_LIST = Object.entries<LegacyToken>(\n contractMap,\n).reduce<TokenDetectionMap>((acc, [base, contract]) => {\n const { logo, erc20, erc721, ...tokenMetadata } = contract;\n return {\n ...acc,\n [base.toLowerCase()]: {\n ...tokenMetadata,\n address: base.toLowerCase(),\n iconUrl: `images/contract/${logo}`,\n aggregators: [],\n },\n };\n}, {});\n\nexport const controllerName = 'TokenDetectionController';\n\nexport type TokenDetectionState = Record<never, never>;\n\nexport type TokenDetectionControllerGetStateAction = ControllerGetStateAction<\n typeof controllerName,\n TokenDetectionState\n>;\n\nexport type TokenDetectionControllerActions =\n TokenDetectionControllerGetStateAction;\n\nexport type AllowedActions =\n | AccountsControllerGetSelectedAccountAction\n | NetworkControllerGetNetworkClientByIdAction\n | NetworkControllerGetNetworkConfigurationByNetworkClientId\n | NetworkControllerGetStateAction\n | GetTokenListState\n | KeyringControllerGetStateAction\n | PreferencesControllerGetStateAction\n | TokensControllerGetStateAction\n | TokensControllerAddDetectedTokensAction;\n\nexport type TokenDetectionControllerStateChangeEvent =\n ControllerStateChangeEvent<typeof controllerName, TokenDetectionState>;\n\nexport type TokenDetectionControllerEvents =\n TokenDetectionControllerStateChangeEvent;\n\nexport type AllowedEvents =\n | AccountsControllerSelectedAccountChangeEvent\n | NetworkControllerNetworkDidChangeEvent\n | TokenListStateChange\n | KeyringControllerLockEvent\n | KeyringControllerUnlockEvent\n | PreferencesControllerStateChangeEvent;\n\nexport type TokenDetectionControllerMessenger = RestrictedControllerMessenger<\n typeof controllerName,\n TokenDetectionControllerActions | AllowedActions,\n TokenDetectionControllerEvents | AllowedEvents,\n AllowedActions['type'],\n AllowedEvents['type']\n>;\n\n/**\n * Controller that passively polls on a set interval for Tokens auto detection\n * @property intervalId - Polling interval used to fetch new token rates\n * @property selectedAddress - Vault selected address\n * @property networkClientId - The network client ID of the current selected network\n * @property disabled - Boolean to track if network requests are blocked\n * @property isUnlocked - Boolean to track if the keyring state is unlocked\n * @property isDetectionEnabledFromPreferences - Boolean to track if detection is enabled from PreferencesController\n * @property isDetectionEnabledForNetwork - Boolean to track if detected is enabled for current network\n */\nexport class TokenDetectionController extends StaticIntervalPollingController<\n typeof controllerName,\n TokenDetectionState,\n TokenDetectionControllerMessenger\n> {\n #intervalId?: ReturnType<typeof setTimeout>;\n\n #selectedAddress: string;\n\n #networkClientId: NetworkClientId;\n\n #tokenList: TokenDetectionMap = {};\n\n #disabled: boolean;\n\n #isUnlocked: boolean;\n\n #isDetectionEnabledFromPreferences: boolean;\n\n #isDetectionEnabledForNetwork: boolean;\n\n readonly #getBalancesInSingleCall: AssetsContractController['getBalancesInSingleCall'];\n\n readonly #trackMetaMetricsEvent: (options: {\n event: string;\n category: string;\n properties: {\n tokens: string[];\n token_standard: string;\n asset_type: string;\n };\n }) => void;\n\n /**\n * Creates a TokenDetectionController instance.\n *\n * @param options - The controller options.\n * @param options.messenger - The controller messaging system.\n * @param options.disabled - If set to true, all network requests are blocked.\n * @param options.interval - Polling interval used to fetch new token rates\n * @param options.selectedAddress - Vault selected address\n * @param options.getBalancesInSingleCall - Gets the balances of a list of tokens for the given address.\n * @param options.trackMetaMetricsEvent - Sets options for MetaMetrics event tracking.\n */\n constructor({\n selectedAddress,\n interval = DEFAULT_INTERVAL,\n disabled = true,\n getBalancesInSingleCall,\n trackMetaMetricsEvent,\n messenger,\n }: {\n selectedAddress?: string;\n interval?: number;\n disabled?: boolean;\n getBalancesInSingleCall: AssetsContractController['getBalancesInSingleCall'];\n trackMetaMetricsEvent: (options: {\n event: string;\n category: string;\n properties: {\n tokens: string[];\n token_standard: string;\n asset_type: string;\n };\n }) => void;\n messenger: TokenDetectionControllerMessenger;\n }) {\n super({\n name: controllerName,\n messenger,\n state: {},\n metadata: {},\n });\n\n this.#disabled = disabled;\n this.setIntervalLength(interval);\n\n this.#selectedAddress =\n selectedAddress ??\n this.messagingSystem.call('AccountsController:getSelectedAccount')\n .address;\n\n const { chainId, networkClientId } =\n this.#getCorrectChainIdAndNetworkClientId();\n this.#networkClientId = networkClientId;\n\n const { useTokenDetection: defaultUseTokenDetection } =\n this.messagingSystem.call('PreferencesController:getState');\n this.#isDetectionEnabledFromPreferences = defaultUseTokenDetection;\n this.#isDetectionEnabledForNetwork =\n isTokenDetectionSupportedForNetwork(chainId);\n\n this.#getBalancesInSingleCall = getBalancesInSingleCall;\n\n this.#trackMetaMetricsEvent = trackMetaMetricsEvent;\n\n const { isUnlocked } = this.messagingSystem.call(\n 'KeyringController:getState',\n );\n this.#isUnlocked = isUnlocked;\n\n this.#registerEventListeners();\n }\n\n /**\n * Constructor helper for registering this controller's messaging system subscriptions to controller events.\n */\n #registerEventListeners() {\n this.messagingSystem.subscribe('KeyringController:unlock', async () => {\n this.#isUnlocked = true;\n await this.#restartTokenDetection();\n });\n\n this.messagingSystem.subscribe('KeyringController:lock', () => {\n this.#isUnlocked = false;\n this.#stopPolling();\n });\n\n this.messagingSystem.subscribe(\n 'TokenListController:stateChange',\n async ({ tokenList }) => {\n const hasTokens = Object.keys(tokenList).length;\n\n if (hasTokens) {\n await this.#restartTokenDetection();\n }\n },\n );\n\n this.messagingSystem.subscribe(\n 'PreferencesController:stateChange',\n async ({ selectedAddress: newSelectedAddress, useTokenDetection }) => {\n const isSelectedAddressChanged =\n this.#selectedAddress !== newSelectedAddress;\n const isDetectionChangedFromPreferences =\n this.#isDetectionEnabledFromPreferences !== useTokenDetection;\n\n this.#selectedAddress = newSelectedAddress;\n this.#isDetectionEnabledFromPreferences = useTokenDetection;\n\n if (isSelectedAddressChanged || isDetectionChangedFromPreferences) {\n await this.#restartTokenDetection({\n selectedAddress: this.#selectedAddress,\n });\n }\n },\n );\n\n this.messagingSystem.subscribe(\n 'AccountsController:selectedAccountChange',\n async ({ address: newSelectedAddress }) => {\n const isSelectedAddressChanged =\n this.#selectedAddress !== newSelectedAddress;\n if (isSelectedAddressChanged) {\n this.#selectedAddress = newSelectedAddress;\n await this.#restartTokenDetection({\n selectedAddress: this.#selectedAddress,\n });\n }\n },\n );\n\n this.messagingSystem.subscribe(\n 'NetworkController:networkDidChange',\n async ({ selectedNetworkClientId }) => {\n const isNetworkClientIdChanged =\n this.#networkClientId !== selectedNetworkClientId;\n\n const { chainId: newChainId } =\n this.#getCorrectChainIdAndNetworkClientId(selectedNetworkClientId);\n this.#isDetectionEnabledForNetwork =\n isTokenDetectionSupportedForNetwork(newChainId);\n\n if (isNetworkClientIdChanged && this.#isDetectionEnabledForNetwork) {\n this.#networkClientId = selectedNetworkClientId;\n await this.#restartTokenDetection({\n networkClientId: this.#networkClientId,\n });\n }\n },\n );\n }\n\n /**\n * Allows controller to make active and passive polling requests\n */\n enable(): void {\n this.#disabled = false;\n }\n\n /**\n * Blocks controller from making network calls\n */\n disable(): void {\n this.#disabled = true;\n }\n\n /**\n * Internal isActive state\n * @type {boolean}\n */\n get isActive(): boolean {\n return !this.#disabled && this.#isUnlocked;\n }\n\n /**\n * Start polling for detected tokens.\n */\n async start(): Promise<void> {\n this.enable();\n await this.#startPolling();\n }\n\n /**\n * Stop polling for detected tokens.\n */\n stop(): void {\n this.disable();\n this.#stopPolling();\n }\n\n #stopPolling(): void {\n if (this.#intervalId) {\n clearInterval(this.#intervalId);\n }\n }\n\n /**\n * Starts a new polling interval.\n */\n async #startPolling(): Promise<void> {\n if (!this.isActive) {\n return;\n }\n this.#stopPolling();\n await this.detectTokens();\n this.#intervalId = setInterval(async () => {\n await this.detectTokens();\n }, this.getIntervalLength());\n }\n\n #getCorrectChainIdAndNetworkClientId(networkClientId?: NetworkClientId): {\n chainId: Hex;\n networkClientId: NetworkClientId;\n } {\n if (networkClientId) {\n const networkConfiguration = this.messagingSystem.call(\n 'NetworkController:getNetworkConfigurationByNetworkClientId',\n networkClientId,\n );\n if (networkConfiguration) {\n return {\n chainId: networkConfiguration.chainId,\n networkClientId,\n };\n }\n }\n const { selectedNetworkClientId } = this.messagingSystem.call(\n 'NetworkController:getState',\n );\n const {\n configuration: { chainId },\n } = this.messagingSystem.call(\n 'NetworkController:getNetworkClientById',\n selectedNetworkClientId,\n );\n return {\n chainId,\n networkClientId: selectedNetworkClientId,\n };\n }\n\n async _executePoll(\n networkClientId: NetworkClientId,\n options: { address: string },\n ): Promise<void> {\n if (!this.isActive) {\n return;\n }\n await this.detectTokens({\n networkClientId,\n selectedAddress: options.address,\n });\n }\n\n /**\n * Restart token detection polling period and call detectNewTokens\n * in case of address change or user session initialization.\n *\n * @param options - Options for restart token detection.\n * @param options.selectedAddress - the selectedAddress against which to detect for token balances\n * @param options.networkClientId - The ID of the network client to use.\n */\n async #restartTokenDetection({\n selectedAddress,\n networkClientId,\n }: {\n selectedAddress?: string;\n networkClientId?: NetworkClientId;\n } = {}): Promise<void> {\n await this.detectTokens({\n networkClientId,\n selectedAddress,\n });\n this.setIntervalLength(DEFAULT_INTERVAL);\n }\n\n /**\n * For each token in the token list provided by the TokenListController, checks the token's balance for the selected account address on the active network.\n * On mainnet, if token detection is disabled in preferences, ERC20 token auto detection will be triggered for each contract address in the legacy token list from the @metamask/contract-metadata repo.\n *\n * @param options - Options for token detection.\n * @param options.networkClientId - The ID of the network client to use.\n * @param options.selectedAddress - the selectedAddress against which to detect for token balances.\n */\n async detectTokens({\n networkClientId,\n selectedAddress,\n }: {\n networkClientId?: NetworkClientId;\n selectedAddress?: string;\n } = {}): Promise<void> {\n if (!this.isActive) {\n return;\n }\n\n const addressAgainstWhichToDetect =\n selectedAddress ?? this.#selectedAddress;\n const { chainId, networkClientId: selectedNetworkClientId } =\n this.#getCorrectChainIdAndNetworkClientId(networkClientId);\n const chainIdAgainstWhichToDetect = chainId;\n const networkClientIdAgainstWhichToDetect = selectedNetworkClientId;\n\n if (!isTokenDetectionSupportedForNetwork(chainIdAgainstWhichToDetect)) {\n return;\n }\n if (\n !this.#isDetectionEnabledFromPreferences &&\n chainIdAgainstWhichToDetect !== ChainId.mainnet\n ) {\n return;\n }\n const isTokenDetectionInactiveInMainnet =\n !this.#isDetectionEnabledFromPreferences &&\n chainIdAgainstWhichToDetect === ChainId.mainnet;\n const { tokensChainsCache } = this.messagingSystem.call(\n 'TokenListController:getState',\n );\n this.#tokenList = isTokenDetectionInactiveInMainnet\n ? STATIC_MAINNET_TOKEN_LIST\n : tokensChainsCache[chainIdAgainstWhichToDetect]?.data ?? {};\n\n for (const tokensSlice of this.#getSlicesOfTokensToDetect({\n chainId: chainIdAgainstWhichToDetect,\n selectedAddress: addressAgainstWhichToDetect,\n })) {\n await this.#addDetectedTokens({\n tokensSlice,\n selectedAddress: addressAgainstWhichToDetect,\n networkClientId: networkClientIdAgainstWhichToDetect,\n chainId: chainIdAgainstWhichToDetect,\n });\n }\n }\n\n #getSlicesOfTokensToDetect({\n chainId,\n selectedAddress,\n }: {\n chainId: Hex;\n selectedAddress: string;\n }): string[][] {\n const { allTokens, allDetectedTokens, allIgnoredTokens } =\n this.messagingSystem.call('TokensController:getState');\n const [tokensAddresses, detectedTokensAddresses, ignoredTokensAddresses] = [\n allTokens,\n allDetectedTokens,\n allIgnoredTokens,\n ].map((tokens) =>\n (tokens[chainId]?.[selectedAddress] ?? []).map((value) =>\n typeof value === 'string' ? value : value.address,\n ),\n );\n\n const tokensToDetect: string[] = [];\n for (const tokenAddress of Object.keys(this.#tokenList)) {\n if (\n [\n tokensAddresses,\n detectedTokensAddresses,\n ignoredTokensAddresses,\n ].every(\n (addresses) =>\n !addresses.find((address) =>\n isEqualCaseInsensitive(address, tokenAddress),\n ),\n )\n ) {\n tokensToDetect.push(tokenAddress);\n }\n }\n\n const slicesOfTokensToDetect = [];\n for (let i = 0, size = 1000; i < tokensToDetect.length; i += size) {\n slicesOfTokensToDetect.push(tokensToDetect.slice(i, i + size));\n }\n\n return slicesOfTokensToDetect;\n }\n\n async #addDetectedTokens({\n tokensSlice,\n selectedAddress,\n networkClientId,\n chainId,\n }: {\n tokensSlice: string[];\n selectedAddress: string;\n networkClientId: NetworkClientId;\n chainId: Hex;\n }): Promise<void> {\n await safelyExecute(async () => {\n const balances = await this.#getBalancesInSingleCall(\n selectedAddress,\n tokensSlice,\n networkClientId,\n );\n\n const tokensWithBalance: Token[] = [];\n const eventTokensDetails: string[] = [];\n for (const nonZeroTokenAddress of Object.keys(balances)) {\n const { decimals, symbol, aggregators, iconUrl, name } =\n this.#tokenList[nonZeroTokenAddress];\n eventTokensDetails.push(`${symbol} - ${nonZeroTokenAddress}`);\n tokensWithBalance.push({\n address: nonZeroTokenAddress,\n decimals,\n symbol,\n aggregators,\n image: iconUrl,\n isERC721: false,\n name,\n });\n }\n\n if (tokensWithBalance.length) {\n this.#trackMetaMetricsEvent({\n event: 'Token Detected',\n category: 'Wallet',\n properties: {\n tokens: eventTokensDetails,\n token_standard: 'ERC20',\n asset_type: 'TOKEN',\n },\n });\n\n await this.messagingSystem.call(\n 'TokensController:addDetectedTokens',\n tokensWithBalance,\n {\n selectedAddress,\n chainId,\n },\n );\n }\n });\n }\n}\n\nexport default TokenDetectionController;\n"],"mappings":"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","names":[]}
@@ -1,6 +1,6 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true});
2
2
 
3
- var _chunkRELRWIJNjs = require('./chunk-RELRWIJN.js');
3
+ var _chunkD6BDJYSKjs = require('./chunk-D6BDJYSK.js');
4
4
 
5
5
  // src/token-service.ts
6
6
  var _controllerutils = require('@metamask/controller-utils');
@@ -27,7 +27,7 @@ async function fetchTokenListByChainId(chainId, abortSignal, { timeout = default
27
27
  return void 0;
28
28
  }
29
29
  async function fetchTokenMetadata(chainId, tokenAddress, abortSignal, { timeout = defaultTimeout } = {}) {
30
- if (!_chunkRELRWIJNjs.isTokenListSupportedForNetwork.call(void 0, chainId)) {
30
+ if (!_chunkD6BDJYSKjs.isTokenListSupportedForNetwork.call(void 0, chainId)) {
31
31
  throw new Error(TOKEN_METADATA_NO_SUPPORT_ERROR);
32
32
  }
33
33
  const tokenMetadataURL = getTokenMetadataURL(chainId, tokenAddress);
@@ -71,4 +71,4 @@ async function parseJsonResponse(apiResponse) {
71
71
 
72
72
 
73
73
  exports.TOKEN_END_POINT_API = TOKEN_END_POINT_API; exports.TOKEN_METADATA_NO_SUPPORT_ERROR = TOKEN_METADATA_NO_SUPPORT_ERROR; exports.fetchTokenListByChainId = fetchTokenListByChainId; exports.fetchTokenMetadata = fetchTokenMetadata;
74
- //# sourceMappingURL=chunk-VY7TUOK2.js.map
74
+ //# sourceMappingURL=chunk-D3N35ABH.js.map
@@ -144,69 +144,6 @@ async function reduceInBatchesSerially({
144
144
  const finalResult = workingResult;
145
145
  return finalResult;
146
146
  }
147
- function mapOpenSeaNftV2ToV1(nft) {
148
- return {
149
- token_id: nft.identifier,
150
- num_sales: null,
151
- background_color: null,
152
- image_url: nft.image_url ?? null,
153
- image_preview_url: null,
154
- image_thumbnail_url: null,
155
- image_original_url: null,
156
- animation_url: null,
157
- animation_original_url: null,
158
- name: nft.name,
159
- description: nft.description,
160
- external_link: null,
161
- asset_contract: {
162
- address: nft.contract,
163
- asset_contract_type: null,
164
- created_date: null,
165
- schema_name: nft.token_standard.toUpperCase(),
166
- symbol: null,
167
- total_supply: null,
168
- description: nft.description,
169
- external_link: null,
170
- collection: {
171
- name: nft.collection,
172
- image_url: null
173
- }
174
- },
175
- creator: {
176
- user: { username: "" },
177
- profile_img_url: "",
178
- address: ""
179
- },
180
- last_sale: null
181
- };
182
- }
183
- function mapOpenSeaDetailedNftV2ToV1(nft) {
184
- const mapped = mapOpenSeaNftV2ToV1(nft);
185
- return {
186
- ...mapped,
187
- animation_url: nft.animation_url ?? null,
188
- creator: {
189
- ...mapped.creator,
190
- address: nft.creator
191
- }
192
- };
193
- }
194
- function mapOpenSeaContractV2ToV1(contract, collection) {
195
- return {
196
- address: contract.address,
197
- asset_contract_type: null,
198
- created_date: null,
199
- schema_name: contract.contract_standard.toUpperCase(),
200
- symbol: null,
201
- total_supply: collection?.total_supply?.toString() ?? contract.total_supply?.toString() ?? null,
202
- description: collection?.description ?? null,
203
- external_link: collection?.project_url ?? null,
204
- collection: {
205
- name: collection?.name ?? contract.name,
206
- image_url: collection?.image_url
207
- }
208
- };
209
- }
210
147
  async function fetchTokenContractExchangeRates({
211
148
  tokenPricesService,
212
149
  nativeCurrency,
@@ -261,8 +198,5 @@ async function fetchTokenContractExchangeRates({
261
198
 
262
199
 
263
200
 
264
-
265
-
266
-
267
- exports.TOKEN_PRICES_BATCH_SIZE = TOKEN_PRICES_BATCH_SIZE; exports.compareNftMetadata = compareNftMetadata; exports.formatAggregatorNames = formatAggregatorNames; exports.formatIconUrlWithProxy = formatIconUrlWithProxy; exports.SupportedTokenDetectionNetworks = SupportedTokenDetectionNetworks; exports.isTokenDetectionSupportedForNetwork = isTokenDetectionSupportedForNetwork; exports.isTokenListSupportedForNetwork = isTokenListSupportedForNetwork; exports.removeIpfsProtocolPrefix = removeIpfsProtocolPrefix; exports.getIpfsCIDv1AndPath = getIpfsCIDv1AndPath; exports.getFormattedIpfsUrl = getFormattedIpfsUrl; exports.addUrlProtocolPrefix = addUrlProtocolPrefix; exports.ethersBigNumberToBN = ethersBigNumberToBN; exports.divideIntoBatches = divideIntoBatches; exports.reduceInBatchesSerially = reduceInBatchesSerially; exports.mapOpenSeaNftV2ToV1 = mapOpenSeaNftV2ToV1; exports.mapOpenSeaDetailedNftV2ToV1 = mapOpenSeaDetailedNftV2ToV1; exports.mapOpenSeaContractV2ToV1 = mapOpenSeaContractV2ToV1; exports.fetchTokenContractExchangeRates = fetchTokenContractExchangeRates;
268
- //# sourceMappingURL=chunk-RELRWIJN.js.map
201
+ exports.TOKEN_PRICES_BATCH_SIZE = TOKEN_PRICES_BATCH_SIZE; exports.compareNftMetadata = compareNftMetadata; exports.formatAggregatorNames = formatAggregatorNames; exports.formatIconUrlWithProxy = formatIconUrlWithProxy; exports.SupportedTokenDetectionNetworks = SupportedTokenDetectionNetworks; exports.isTokenDetectionSupportedForNetwork = isTokenDetectionSupportedForNetwork; exports.isTokenListSupportedForNetwork = isTokenListSupportedForNetwork; exports.removeIpfsProtocolPrefix = removeIpfsProtocolPrefix; exports.getIpfsCIDv1AndPath = getIpfsCIDv1AndPath; exports.getFormattedIpfsUrl = getFormattedIpfsUrl; exports.addUrlProtocolPrefix = addUrlProtocolPrefix; exports.ethersBigNumberToBN = ethersBigNumberToBN; exports.divideIntoBatches = divideIntoBatches; exports.reduceInBatchesSerially = reduceInBatchesSerially; exports.fetchTokenContractExchangeRates = fetchTokenContractExchangeRates;
202
+ //# sourceMappingURL=chunk-D6BDJYSK.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/assetsUtil.ts"],"names":["SupportedTokenDetectionNetworks"],"mappings":";AACA;AAAA,EACE;AAAA,EACA;AAAA,OACK;AAEP,SAAS,gBAAgB;AACzB,OAAO,QAAQ;AACf,SAAS,WAAW;AAUb,IAAM,0BAA0B;AAWhC,SAAS,mBAAmB,gBAA6B,KAAU;AACxE,QAAM,OAA8B;AAAA,IAClC;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACA,QAAM,kBAAkB,KAAK,OAAO,CAAC,OAAO,QAAQ;AAClD,QAAI,eAAe,GAAG,KAAK,eAAe,GAAG,MAAM,IAAI,GAAG,GAAG;AAC3D,aAAO,QAAQ;AAAA,IACjB;AACA,WAAO;AAAA,EACT,GAAG,CAAC;AACJ,SAAO,kBAAkB;AAC3B;AAEA,IAAM,sBAA8C;AAAA,EAClD,MAAM;AAAA,EACN,QAAQ;AAAA,EACR,KAAK;AAAA,EACL,WAAW;AAAA,EACX,WAAW;AAAA,EACX,SAAS;AAAA,EACT,UAAU;AAAA,EACV,KAAK;AAAA,EACL,QAAQ;AAAA,EACR,QAAQ;AAAA,EACR,QAAQ;AAAA,EACR,WAAW;AAAA,EACX,OAAO;AAAA,EACP,SAAS;AAAA,EACT,YAAY;AAAA,EACZ,eAAe;AAAA,EACf,iBAAiB;AAAA,EACjB,UAAU;AAAA,EACV,WAAW;AAAA,EACX,QAAQ;AAAA,EACR,aAAa;AAAA,EACb,uBAAuB;AAAA,EACvB,uBAAuB;AAAA,EACvB,WAAW;AAAA,EACX,cAAc;AAAA,EACd,QAAQ;AACV;AAQO,IAAM,wBAAwB,CAAC,gBAA0B;AAC9D,SAAO,YAAY;AAAA,IACjB,CAAC,QACC,oBAAoB,GAAG,KACvB,GAAG,IAAI,CAAC,EAAE,YAAY,CAAC,GAAG,IAAI,UAAU,GAAG,IAAI,MAAM,CAAC;AAAA,EAC1D;AACF;AAUO,IAAM,yBAAyB,CAAC;AAAA,EACrC;AAAA,EACA;AACF,MAGM;AACJ,QAAM,iBAAiB,oBAAoB,OAAO,EAAE,SAAS;AAC7D,SAAO,0DAA0D,cAAc,IAAI,aAAa,YAAY,CAAC;AAC/G;AAKO,IAAK,kCAAL,kBAAKA,qCAAL;AACL,EAAAA,iCAAA,aAAU;AACV,EAAAA,iCAAA,SAAM;AACN,EAAAA,iCAAA,aAAU;AACV,EAAAA,iCAAA,UAAO;AACP,EAAAA,iCAAA,YAAS;AACT,EAAAA,iCAAA,kBAAe;AACf,EAAAA,iCAAA,mBAAgB;AAChB,EAAAA,iCAAA,cAAW;AACX,EAAAA,iCAAA,cAAW;AACX,EAAAA,iCAAA,UAAO;AACP,EAAAA,iCAAA,YAAS;AAXC,SAAAA;AAAA,GAAA;AAoBL,SAAS,oCAAoC,SAAuB;AACzE,SAAO,OAAO,OAAY,+BAA+B,EAAE,SAAS,OAAO;AAC7E;AASO,SAAS,+BAA+B,SAAuB;AACpE,SAAO,oCAAoC,OAAO;AACpD;AASO,SAAS,yBAAyB,SAAiB;AACxD,MAAI,QAAQ,WAAW,cAAc,GAAG;AACtC,WAAO,QAAQ,QAAQ,gBAAgB,EAAE;AAAA,EAC3C,WAAW,QAAQ,WAAW,SAAS,GAAG;AACxC,WAAO,QAAQ,QAAQ,WAAW,EAAE;AAAA,EACtC;AAEA,QAAM,IAAI,MAAM,mDAAmD;AACrE;AASO,SAAS,oBAAoB,SAGlC;AACA,QAAM,MAAM,yBAAyB,OAAO;AAI5C,QAAM,QAAQ,IAAI,QAAQ,GAAG;AAC7B,QAAM,MAAM,UAAU,KAAK,IAAI,UAAU,GAAG,KAAK,IAAI;AACrD,QAAM,OAAO,UAAU,KAAK,IAAI,UAAU,KAAK,IAAI;AAInD,SAAO;AAAA,IACL,KAAK,IAAI,MAAM,GAAG,EAAE,KAAK,EAAE,SAAS;AAAA,IACpC;AAAA,EACF;AACF;AAUO,SAAS,oBACd,aACA,SACA,oBACQ;AACR,QAAM,EAAE,MAAM,UAAU,OAAO,IAAI,IAAI,IAAI,qBAAqB,WAAW,CAAC;AAC5E,MAAI,oBAAoB;AACtB,UAAM,EAAE,KAAK,KAAK,IAAI,oBAAoB,OAAO;AACjD,WAAO,GAAG,QAAQ,KAAK,GAAG,SAAS,IAAI,GAAG,QAAQ,EAAE;AAAA,EACtD;AACA,QAAM,aAAa,yBAAyB,OAAO;AACnD,SAAO,GAAG,MAAM,SAAS,UAAU;AACrC;AAQO,SAAS,qBAAqB,WAA2B;AAC9D,MAAI,CAAC,UAAU,MAAM,6BAA6B,GAAG;AACnD,WAAO,WAAW,SAAS;AAAA,EAC7B;AACA,SAAO;AACT;AAQO,SAAS,oBAAoB,WAA0B;AAC5D,SAAO,IAAI,GAAG,SAAS,UAAU,YAAY,CAAC,GAAG,KAAK;AACxD;AAWO,SAAS,kBACd,QACA,EAAE,UAAU,GACD;AACX,QAAM,UAAU,CAAC;AACjB,WAAS,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAK,WAAW;AACjD,YAAQ,KAAK,OAAO,MAAM,GAAG,IAAI,SAAS,CAAC;AAAA,EAC7C;AACA,SAAO;AACT;AAiBA,eAAsB,wBAGpB;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,GASoB;AAClB,QAAM,UAAU,kBAAkB,QAAQ,EAAE,UAAU,CAAC;AACvD,MAAI,gBAAgB;AACpB,aAAW,CAAC,OAAO,KAAK,KAAK,QAAQ,QAAQ,GAAG;AAC9C,oBAAgB,MAAM,UAAU,eAAe,OAAO,KAAK;AAAA,EAC7D;AAGA,QAAM,cAAc;AACpB,SAAO;AACT;AAYA,eAAsB,gCAAgC;AAAA,EACpD;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,GAKmC;AACjC,QAAM,qBACJ,mBAAmB,yBAAyB,OAAO;AACrD,QAAM,sBACJ,mBAAmB,0BAA0B,cAAc;AAE7D,MAAI,CAAC,sBAAsB,CAAC,qBAAqB;AAC/C,WAAO,CAAC;AAAA,EACV;AAEA,QAAM,4BAA4B,MAAM,wBAGtC;AAAA,IACA,QAAQ,CAAC,GAAG,cAAc,EAAE,KAAK;AAAA,IACjC,WAAW;AAAA,IACX,WAAW,OAAO,8BAA8B,UAAU;AACxD,YAAM,oCACJ,MAAM,mBAAmB,iBAAiB;AAAA,QACxC,gBAAgB;AAAA,QAChB;AAAA,QACA,UAAU;AAAA,MACZ,CAAC;AAEH,aAAO;AAAA,QACL,GAAG;AAAA,QACH,GAAG;AAAA,MACL;AAAA,IACF;AAAA,IACA,eAAe,CAAC;AAAA,EAClB,CAAC;AAED,SAAO,OAAO,QAAQ,yBAAyB,EAAE;AAAA,IAC/C,CAAC,KAAK,CAAC,cAAc,UAAU,MAAM;AACnC,aAAO;AAAA,QACL,GAAG;AAAA,QACH,CAAC,qBAAqB,YAAY,CAAC,GAAG,YAAY;AAAA,MACpD;AAAA,IACF;AAAA,IACA,CAAC;AAAA,EACH;AACF","sourcesContent":["import type { BigNumber } from '@ethersproject/bignumber';\nimport {\n convertHexToDecimal,\n toChecksumHexAddress,\n} from '@metamask/controller-utils';\nimport type { Hex } from '@metamask/utils';\nimport { remove0x } from '@metamask/utils';\nimport BN from 'bn.js';\nimport { CID } from 'multiformats/cid';\n\nimport type { Nft, NftMetadata } from './NftController';\nimport type { AbstractTokenPricesService } from './token-prices-service';\nimport { type ContractExchangeRates } from './TokenRatesController';\n\n/**\n * The maximum number of token addresses that should be sent to the Price API in\n * a single request.\n */\nexport const TOKEN_PRICES_BATCH_SIZE = 30;\n\n/**\n * Compares nft metadata entries to any nft entry.\n * We need this method when comparing a new fetched nft metadata, in case a entry changed to a defined value,\n * there's a need to update the nft in state.\n *\n * @param newNftMetadata - Nft metadata object.\n * @param nft - Nft object to compare with.\n * @returns Whether there are differences.\n */\nexport function compareNftMetadata(newNftMetadata: NftMetadata, nft: Nft) {\n const keys: (keyof NftMetadata)[] = [\n 'image',\n 'backgroundColor',\n 'imagePreview',\n 'imageThumbnail',\n 'imageOriginal',\n 'animation',\n 'animationOriginal',\n 'externalLink',\n 'tokenURI',\n ];\n const differentValues = keys.reduce((value, key) => {\n if (newNftMetadata[key] && newNftMetadata[key] !== nft[key]) {\n return value + 1;\n }\n return value;\n }, 0);\n return differentValues > 0;\n}\n\nconst aggregatorNameByKey: Record<string, string> = {\n aave: 'Aave',\n bancor: 'Bancor',\n cmc: 'CMC',\n cryptocom: 'Crypto.com',\n coinGecko: 'CoinGecko',\n oneInch: '1inch',\n paraswap: 'Paraswap',\n pmm: 'PMM',\n zapper: 'Zapper',\n zerion: 'Zerion',\n zeroEx: '0x',\n synthetix: 'Synthetix',\n yearn: 'Yearn',\n apeswap: 'ApeSwap',\n binanceDex: 'BinanceDex',\n pancakeTop100: 'PancakeTop100',\n pancakeExtended: 'PancakeExtended',\n balancer: 'Balancer',\n quickswap: 'QuickSwap',\n matcha: 'Matcha',\n pangolinDex: 'PangolinDex',\n pangolinDexStableCoin: 'PangolinDexStableCoin',\n pangolinDexAvaxBridge: 'PangolinDexAvaxBridge',\n traderJoe: 'TraderJoe',\n airswapLight: 'AirswapLight',\n kleros: 'Kleros',\n};\n\n/**\n * Formats aggregator names to presentable format.\n *\n * @param aggregators - List of token list names in camelcase.\n * @returns Formatted aggregator names.\n */\nexport const formatAggregatorNames = (aggregators: string[]) => {\n return aggregators.map(\n (key) =>\n aggregatorNameByKey[key] ||\n `${key[0].toUpperCase()}${key.substring(1, key.length)}`,\n );\n};\n\n/**\n * Format token list assets to use image proxy from Codefi.\n *\n * @param params - Object that contains chainID and tokenAddress.\n * @param params.chainId - ChainID of network in 0x-prefixed hexadecimal format.\n * @param params.tokenAddress - Address of token in mixed or lowercase.\n * @returns Formatted image url\n */\nexport const formatIconUrlWithProxy = ({\n chainId,\n tokenAddress,\n}: {\n chainId: Hex;\n tokenAddress: string;\n}) => {\n const chainIdDecimal = convertHexToDecimal(chainId).toString();\n return `https://static.metafi.codefi.network/api/v1/tokenIcons/${chainIdDecimal}/${tokenAddress.toLowerCase()}.png`;\n};\n\n/**\n * Networks where token detection is supported - Values are in decimal format\n */\nexport enum SupportedTokenDetectionNetworks {\n mainnet = '0x1', // decimal: 1\n bsc = '0x38', // decimal: 56\n polygon = '0x89', // decimal: 137\n avax = '0xa86a', // decimal: 43114\n aurora = '0x4e454152', // decimal: 1313161554\n linea_goerli = '0xe704', // decimal: 59140\n linea_mainnet = '0xe708', // decimal: 59144\n arbitrum = '0xa4b1', // decimal: 42161\n optimism = '0xa', // decimal: 10\n base = '0x2105', // decimal: 8453\n zksync = '0x144', // decimal: 324\n}\n\n/**\n * Check if token detection is enabled for certain networks.\n *\n * @param chainId - ChainID of network\n * @returns Whether the current network supports token detection\n */\nexport function isTokenDetectionSupportedForNetwork(chainId: Hex): boolean {\n return Object.values<Hex>(SupportedTokenDetectionNetworks).includes(chainId);\n}\n\n/**\n * Check if token list polling is enabled for a given network.\n * Currently this method is used to support e2e testing for consumers of this package.\n *\n * @param chainId - ChainID of network\n * @returns Whether the current network supports tokenlists\n */\nexport function isTokenListSupportedForNetwork(chainId: Hex): boolean {\n return isTokenDetectionSupportedForNetwork(chainId);\n}\n\n/**\n * Removes IPFS protocol prefix from input string.\n *\n * @param ipfsUrl - An IPFS url (e.g. ipfs://{content id})\n * @returns IPFS content identifier and (possibly) path in a string\n * @throws Will throw if the url passed is not IPFS.\n */\nexport function removeIpfsProtocolPrefix(ipfsUrl: string) {\n if (ipfsUrl.startsWith('ipfs://ipfs/')) {\n return ipfsUrl.replace('ipfs://ipfs/', '');\n } else if (ipfsUrl.startsWith('ipfs://')) {\n return ipfsUrl.replace('ipfs://', '');\n }\n // this method should not be used with non-ipfs urls (i.e. startsWith('ipfs://') === true)\n throw new Error('this method should not be used with non ipfs urls');\n}\n\n/**\n * Extracts content identifier and path from an input string.\n *\n * @param ipfsUrl - An IPFS URL minus the IPFS protocol prefix\n * @returns IFPS content identifier (cid) and sub path as string.\n * @throws Will throw if the url passed is not ipfs.\n */\nexport function getIpfsCIDv1AndPath(ipfsUrl: string): {\n cid: string;\n path?: string;\n} {\n const url = removeIpfsProtocolPrefix(ipfsUrl);\n\n // check if there is a path\n // (CID is everything preceding first forward slash, path is everything after)\n const index = url.indexOf('/');\n const cid = index !== -1 ? url.substring(0, index) : url;\n const path = index !== -1 ? url.substring(index) : undefined;\n\n // We want to ensure that the CID is v1 (https://docs.ipfs.io/concepts/content-addressing/#identifier-formats)\n // because most cid v0s appear to be incompatible with IPFS subdomains\n return {\n cid: CID.parse(cid).toV1().toString(),\n path,\n };\n}\n\n/**\n * Formats URL correctly for use retrieving assets hosted on IPFS.\n *\n * @param ipfsGateway - The users preferred IPFS gateway (full URL or just host).\n * @param ipfsUrl - The IFPS URL pointed at the asset.\n * @param subdomainSupported - Boolean indicating whether the URL should be formatted with subdomains or not.\n * @returns A formatted URL, with the user's preferred IPFS gateway and format (subdomain or not), pointing to an asset hosted on IPFS.\n */\nexport function getFormattedIpfsUrl(\n ipfsGateway: string,\n ipfsUrl: string,\n subdomainSupported: boolean,\n): string {\n const { host, protocol, origin } = new URL(addUrlProtocolPrefix(ipfsGateway));\n if (subdomainSupported) {\n const { cid, path } = getIpfsCIDv1AndPath(ipfsUrl);\n return `${protocol}//${cid}.ipfs.${host}${path ?? ''}`;\n }\n const cidAndPath = removeIpfsProtocolPrefix(ipfsUrl);\n return `${origin}/ipfs/${cidAndPath}`;\n}\n\n/**\n * Adds URL protocol prefix to input URL string if missing.\n *\n * @param urlString - An IPFS URL.\n * @returns A URL with a https:// prepended.\n */\nexport function addUrlProtocolPrefix(urlString: string): string {\n if (!urlString.match(/(^http:\\/\\/)|(^https:\\/\\/)/u)) {\n return `https://${urlString}`;\n }\n return urlString;\n}\n\n/**\n * Converts an Ethers BigNumber to a BN.\n *\n * @param bigNumber - An Ethers BigNumber instance.\n * @returns A BN object.\n */\nexport function ethersBigNumberToBN(bigNumber: BigNumber): BN {\n return new BN(remove0x(bigNumber.toHexString()), 'hex');\n}\n\n/**\n * Partitions a list of values into groups that are at most `batchSize` in\n * length.\n *\n * @param values - The list of values.\n * @param args - The remaining arguments.\n * @param args.batchSize - The desired maximum number of values per batch.\n * @returns The list of batches.\n */\nexport function divideIntoBatches<Value>(\n values: Value[],\n { batchSize }: { batchSize: number },\n): Value[][] {\n const batches = [];\n for (let i = 0; i < values.length; i += batchSize) {\n batches.push(values.slice(i, i + batchSize));\n }\n return batches;\n}\n\n/**\n * Constructs an object from processing batches of the given values\n * sequentially.\n *\n * @param args - The arguments to this function.\n * @param args.values - A list of values to iterate over.\n * @param args.batchSize - The maximum number of values in each batch.\n * @param args.eachBatch - A function to call for each batch. This function is\n * similar to the function that `Array.prototype.reduce` takes, in that it\n * receives the object that is being built, each batch in the list of batches\n * and the index, and should return an updated version of the object.\n * @param args.initialResult - The initial value of the final data structure,\n * i.e., the value that will be fed into the first call of `eachBatch`.\n * @returns The built object.\n */\nexport async function reduceInBatchesSerially<\n Value,\n Result extends Record<PropertyKey, unknown>,\n>({\n values,\n batchSize,\n eachBatch,\n initialResult,\n}: {\n values: Value[];\n batchSize: number;\n eachBatch: (\n workingResult: Partial<Result>,\n batch: Value[],\n index: number,\n ) => Partial<Result> | Promise<Partial<Result>>;\n initialResult: Partial<Result>;\n}): Promise<Result> {\n const batches = divideIntoBatches(values, { batchSize });\n let workingResult = initialResult;\n for (const [index, batch] of batches.entries()) {\n workingResult = await eachBatch(workingResult, batch, index);\n }\n // There's no way around this — we have to assume that in the end, the result\n // matches the intended type.\n const finalResult = workingResult as Result;\n return finalResult;\n}\n\n/**\n * Retrieves token prices for a set of contract addresses in a specific currency and chainId.\n *\n * @param args - The arguments to function.\n * @param args.tokenPricesService - An object in charge of retrieving token prices.\n * @param args.nativeCurrency - The native currency to request price in.\n * @param args.tokenAddresses - The list of contract addresses.\n * @param args.chainId - The chainId of the tokens.\n * @returns The prices for the requested tokens.\n */\nexport async function fetchTokenContractExchangeRates({\n tokenPricesService,\n nativeCurrency,\n tokenAddresses,\n chainId,\n}: {\n tokenPricesService: AbstractTokenPricesService;\n nativeCurrency: string;\n tokenAddresses: Hex[];\n chainId: Hex;\n}): Promise<ContractExchangeRates> {\n const isChainIdSupported =\n tokenPricesService.validateChainIdSupported(chainId);\n const isCurrencySupported =\n tokenPricesService.validateCurrencySupported(nativeCurrency);\n\n if (!isChainIdSupported || !isCurrencySupported) {\n return {};\n }\n\n const tokenPricesByTokenAddress = await reduceInBatchesSerially<\n Hex,\n Awaited<ReturnType<AbstractTokenPricesService['fetchTokenPrices']>>\n >({\n values: [...tokenAddresses].sort(),\n batchSize: TOKEN_PRICES_BATCH_SIZE,\n eachBatch: async (allTokenPricesByTokenAddress, batch) => {\n const tokenPricesByTokenAddressForBatch =\n await tokenPricesService.fetchTokenPrices({\n tokenAddresses: batch,\n chainId,\n currency: nativeCurrency,\n });\n\n return {\n ...allTokenPricesByTokenAddress,\n ...tokenPricesByTokenAddressForBatch,\n };\n },\n initialResult: {},\n });\n\n return Object.entries(tokenPricesByTokenAddress).reduce(\n (obj, [tokenAddress, tokenPrice]) => {\n return {\n ...obj,\n [toChecksumHexAddress(tokenAddress)]: tokenPrice?.value,\n };\n },\n {},\n );\n}\n"]}
@@ -1,7 +1,10 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true});// src/crypto-compare.ts
2
2
  var _controllerutils = require('@metamask/controller-utils');
3
+ var nativeSymbolOverrides = /* @__PURE__ */ new Map([["MNT", "MANTLE"]]);
3
4
  function getPricingURL(currentCurrency, nativeCurrency, includeUSDRate) {
4
- return `https://min-api.cryptocompare.com/data/price?fsym=${nativeCurrency.toUpperCase()}&tsyms=${currentCurrency.toUpperCase()}${includeUSDRate && currentCurrency.toUpperCase() !== "USD" ? ",USD" : ""}`;
5
+ nativeCurrency = nativeCurrency.toUpperCase();
6
+ const fsym = nativeSymbolOverrides.get(nativeCurrency) ?? nativeCurrency;
7
+ return `https://min-api.cryptocompare.com/data/price?fsym=${fsym}&tsyms=${currentCurrency.toUpperCase()}${includeUSDRate && currentCurrency.toUpperCase() !== "USD" ? ",USD" : ""}`;
5
8
  }
6
9
  async function fetchExchangeRate(currency, nativeCurrency, includeUSDRate) {
7
10
  const json = await _controllerutils.handleFetch.call(void 0,
@@ -29,4 +32,4 @@ async function fetchExchangeRate(currency, nativeCurrency, includeUSDRate) {
29
32
 
30
33
 
31
34
  exports.fetchExchangeRate = fetchExchangeRate;
32
- //# sourceMappingURL=chunk-C4PQK53K.js.map
35
+ //# sourceMappingURL=chunk-DYH5P3VY.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/crypto-compare.ts"],"names":[],"mappings":";AAAA,SAAS,mBAAmB;AAM5B,IAAM,wBAAwB,oBAAI,IAAI,CAAC,CAAC,OAAO,QAAQ,CAAC,CAAC;AAazD,SAAS,cACP,iBACA,gBACA,gBACA;AACA,mBAAiB,eAAe,YAAY;AAC5C,QAAM,OAAO,sBAAsB,IAAI,cAAc,KAAK;AAC1D,SACE,qDACG,IAAI,UAAU,gBAAgB,YAAY,CAAC,GAC3C,kBAAkB,gBAAgB,YAAY,MAAM,QAAQ,SAAS,EAAE;AAE9E;AAUA,eAAsB,kBACpB,UACA,gBACA,gBAIC;AACD,QAAM,OAAO,MAAM;AAAA,IACjB,cAAc,UAAU,gBAAgB,cAAc;AAAA,EACxD;AAUA,MAAI,KAAK,aAAa,SAAS;AAC7B,UAAM,IAAI,MAAM,KAAK,OAAO;AAAA,EAC9B;AAEA,QAAM,iBAAiB,OAAO,KAAK,SAAS,YAAY,CAAC,CAAC;AAE1D,QAAM,oBAAoB,OAAO,KAAK,GAAG;AACzC,MAAI,CAAC,OAAO,SAAS,cAAc,GAAG;AACpC,UAAM,IAAI;AAAA,MACR,wBAAwB,SAAS,YAAY,CAAC,KAC5C,KAAK,SAAS,YAAY,CAAC,CAC7B;AAAA,IACF;AAAA,EACF;AAEA,MAAI,kBAAkB,CAAC,OAAO,SAAS,iBAAiB,GAAG;AACzD,UAAM,IAAI,MAAM,2CAA2C,KAAK,GAAG,EAAE;AAAA,EACvE;AAEA,SAAO;AAAA,IACL;AAAA,IACA;AAAA,EACF;AACF","sourcesContent":["import { handleFetch } from '@metamask/controller-utils';\n\n/**\n * A map from native currency symbol to CryptoCompare identifier.\n * This is only needed when the values don't match.\n */\nconst nativeSymbolOverrides = new Map([['MNT', 'MANTLE']]);\n\n/**\n * Get the CryptoCompare API URL for getting the conversion rate from the given native currency to\n * the given currency. Optionally, the conversion rate from the native currency to USD can also be\n * included in the response.\n *\n * @param currentCurrency - The currency to get a conversion rate for.\n * @param nativeCurrency - The native currency to convert from.\n * @param includeUSDRate - Whether or not the native currency to USD conversion rate should be\n * included in the response as well.\n * @returns The API URL for getting the conversion rate.\n */\nfunction getPricingURL(\n currentCurrency: string,\n nativeCurrency: string,\n includeUSDRate?: boolean,\n) {\n nativeCurrency = nativeCurrency.toUpperCase();\n const fsym = nativeSymbolOverrides.get(nativeCurrency) ?? nativeCurrency;\n return (\n `https://min-api.cryptocompare.com/data/price?fsym=` +\n `${fsym}&tsyms=${currentCurrency.toUpperCase()}` +\n `${includeUSDRate && currentCurrency.toUpperCase() !== 'USD' ? ',USD' : ''}`\n );\n}\n\n/**\n * Fetches the exchange rate for a given currency.\n *\n * @param currency - ISO 4217 currency code.\n * @param nativeCurrency - Symbol for base asset.\n * @param includeUSDRate - Whether to add the USD rate to the fetch.\n * @returns Promise resolving to exchange rate for given currency.\n */\nexport async function fetchExchangeRate(\n currency: string,\n nativeCurrency: string,\n includeUSDRate?: boolean,\n): Promise<{\n conversionRate: number;\n usdConversionRate: number;\n}> {\n const json = await handleFetch(\n getPricingURL(currency, nativeCurrency, includeUSDRate),\n );\n\n /*\n Example expected error response (if pair is not found)\n {\n Response: \"Error\",\n Message: \"cccagg_or_exchange market does not exist for this coin pair (ETH-<NON_EXISTENT_TOKEN>)\",\n HasWarning: false,\n }\n */\n if (json.Response === 'Error') {\n throw new Error(json.Message);\n }\n\n const conversionRate = Number(json[currency.toUpperCase()]);\n\n const usdConversionRate = Number(json.USD);\n if (!Number.isFinite(conversionRate)) {\n throw new Error(\n `Invalid response for ${currency.toUpperCase()}: ${\n json[currency.toUpperCase()]\n }`,\n );\n }\n\n if (includeUSDRate && !Number.isFinite(usdConversionRate)) {\n throw new Error(`Invalid response for usdConversionRate: ${json.USD}`);\n }\n\n return {\n conversionRate,\n usdConversionRate,\n };\n}\n"]}
@@ -1,17 +1,17 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
2
2
 
3
- var _chunkVHCPQS4Xjs = require('./chunk-VHCPQS4X.js');
3
+ var _chunkGCWWC4X3js = require('./chunk-GCWWC4X3.js');
4
4
 
5
5
 
6
- var _chunk355MWUUQjs = require('./chunk-355MWUUQ.js');
6
+ var _chunkBM7EPTR6js = require('./chunk-BM7EPTR6.js');
7
7
 
8
8
 
9
9
 
10
- var _chunkVY7TUOK2js = require('./chunk-VY7TUOK2.js');
10
+ var _chunkD3N35ABHjs = require('./chunk-D3N35ABH.js');
11
11
 
12
12
 
13
13
 
14
- var _chunkRELRWIJNjs = require('./chunk-RELRWIJN.js');
14
+ var _chunkD6BDJYSKjs = require('./chunk-D6BDJYSK.js');
15
15
 
16
16
  // src/TokensController.ts
17
17
  var _contracts = require('@ethersproject/contracts');
@@ -135,14 +135,14 @@ var TokensController = class extends _basecontroller.BaseControllerV1 {
135
135
  */
136
136
  async fetchTokenMetadata(tokenAddress) {
137
137
  try {
138
- const token = await _chunkVY7TUOK2js.fetchTokenMetadata.call(void 0,
138
+ const token = await _chunkD3N35ABHjs.fetchTokenMetadata.call(void 0,
139
139
  this.config.chainId,
140
140
  tokenAddress,
141
141
  this.abortController.signal
142
142
  );
143
143
  return token;
144
144
  } catch (error) {
145
- if (error instanceof Error && error.message.includes(_chunkVY7TUOK2js.TOKEN_METADATA_NO_SUPPORT_ERROR)) {
145
+ if (error instanceof Error && error.message.includes(_chunkD3N35ABHjs.TOKEN_METADATA_NO_SUPPORT_ERROR)) {
146
146
  return void 0;
147
147
  }
148
148
  throw error;
@@ -202,12 +202,12 @@ var TokensController = class extends _basecontroller.BaseControllerV1 {
202
202
  address,
203
203
  symbol,
204
204
  decimals,
205
- image: image || _chunkRELRWIJNjs.formatIconUrlWithProxy.call(void 0, {
205
+ image: image || _chunkD6BDJYSKjs.formatIconUrlWithProxy.call(void 0, {
206
206
  chainId: currentChainId,
207
207
  tokenAddress: address
208
208
  }),
209
209
  isERC721,
210
- aggregators: _chunkRELRWIJNjs.formatAggregatorNames.call(void 0, tokenMetadata?.aggregators || []),
210
+ aggregators: _chunkD6BDJYSKjs.formatAggregatorNames.call(void 0, tokenMetadata?.aggregators || []),
211
211
  name
212
212
  };
213
213
  const previousIndex = newTokens.findIndex(
@@ -532,7 +532,7 @@ var TokensController = class extends _basecontroller.BaseControllerV1 {
532
532
  }
533
533
  const provider = this._getProvider(networkClientId);
534
534
  const isErc1155 = await _controllerutils.safelyExecute.call(void 0,
535
- () => new (0, _chunkVHCPQS4Xjs.ERC1155Standard)(provider).contractSupportsBase1155Interface(
535
+ () => new (0, _chunkGCWWC4X3js.ERC1155Standard)(provider).contractSupportsBase1155Interface(
536
536
  asset.address
537
537
  )
538
538
  );
@@ -541,7 +541,7 @@ var TokensController = class extends _basecontroller.BaseControllerV1 {
541
541
  `Contract ${asset.address} must match type ${type}, but was detected as ${_controllerutils.ERC1155}`
542
542
  );
543
543
  }
544
- const erc20 = new (0, _chunk355MWUUQjs.ERC20Standard)(provider);
544
+ const erc20 = new (0, _chunkBM7EPTR6js.ERC20Standard)(provider);
545
545
  const [contractName, contractSymbol, contractDecimals] = await Promise.all([
546
546
  _controllerutils.safelyExecute.call(void 0, () => erc20.getTokenName(asset.address)),
547
547
  _controllerutils.safelyExecute.call(void 0, () => erc20.getTokenSymbol(asset.address)),
@@ -701,4 +701,4 @@ var TokensController_default = TokensController;
701
701
 
702
702
 
703
703
  exports.getDefaultTokensState = getDefaultTokensState; exports.TokensController = TokensController; exports.TokensController_default = TokensController_default;
704
- //# sourceMappingURL=chunk-NUBQRDKI.js.map
704
+ //# sourceMappingURL=chunk-E2SLNUD4.js.map
@@ -1,7 +1,7 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true});
2
2
 
3
3
 
4
- var _chunkRELRWIJNjs = require('./chunk-RELRWIJN.js');
4
+ var _chunkD6BDJYSKjs = require('./chunk-D6BDJYSK.js');
5
5
 
6
6
  // src/Standards/NftStandards/ERC1155/ERC1155Standard.ts
7
7
  var _contracts = require('@ethersproject/contracts');
@@ -73,7 +73,7 @@ var ERC1155Standard = class {
73
73
  async getBalanceOf(contractAddress, address, tokenId) {
74
74
  const contract = new (0, _contracts.Contract)(contractAddress, _metamaskethabis.abiERC1155, this.provider);
75
75
  const balance = await contract.balanceOf(address, tokenId);
76
- return _chunkRELRWIJNjs.ethersBigNumberToBN.call(void 0, balance);
76
+ return _chunkD6BDJYSKjs.ethersBigNumberToBN.call(void 0, balance);
77
77
  }
78
78
  /**
79
79
  * Transfer single ERC1155 token.
@@ -184,7 +184,7 @@ var ERC1155Standard = class {
184
184
  _controllerutils.safelyExecute.call(void 0, () => this.getAssetName(address)),
185
185
  tokenId ? _controllerutils.safelyExecute.call(void 0,
186
186
  () => this.getTokenURI(address, tokenId).then(
187
- (uri) => uri.startsWith("ipfs://") ? _chunkRELRWIJNjs.getFormattedIpfsUrl.call(void 0, ipfsGateway, uri, true) : uri
187
+ (uri) => uri.startsWith("ipfs://") ? _chunkD6BDJYSKjs.getFormattedIpfsUrl.call(void 0, ipfsGateway, uri, true) : uri
188
188
  )
189
189
  ) : void 0
190
190
  ]);
@@ -194,7 +194,7 @@ var ERC1155Standard = class {
194
194
  const object = await response.json();
195
195
  image = object?.image;
196
196
  if (image?.startsWith("ipfs://")) {
197
- image = _chunkRELRWIJNjs.getFormattedIpfsUrl.call(void 0, ipfsGateway, image, true);
197
+ image = _chunkD6BDJYSKjs.getFormattedIpfsUrl.call(void 0, ipfsGateway, image, true);
198
198
  }
199
199
  } catch {
200
200
  }
@@ -212,4 +212,4 @@ var ERC1155Standard = class {
212
212
 
213
213
 
214
214
  exports.ERC1155Standard = ERC1155Standard;
215
- //# sourceMappingURL=chunk-VHCPQS4X.js.map
215
+ //# sourceMappingURL=chunk-GCWWC4X3.js.map
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  isTokenListSupportedForNetwork
3
- } from "./chunk-E5PIGGYR.mjs";
3
+ } from "./chunk-ZPAX2HLQ.mjs";
4
4
 
5
5
  // src/token-service.ts
6
6
  import { convertHexToDecimal, timeoutFetch } from "@metamask/controller-utils";
@@ -71,4 +71,4 @@ export {
71
71
  fetchTokenListByChainId,
72
72
  fetchTokenMetadata
73
73
  };
74
- //# sourceMappingURL=chunk-KRTEFV4Q.mjs.map
74
+ //# sourceMappingURL=chunk-IURNPMBB.mjs.map
@@ -1,10 +1,4 @@
1
- "use strict";Object.defineProperty(exports, "__esModule", {value: true});
2
-
3
- var _chunkRELRWIJNjs = require('./chunk-RELRWIJN.js');
4
-
5
- // src/NftDetectionController.ts
6
-
7
-
1
+ "use strict";Object.defineProperty(exports, "__esModule", {value: true});// src/NftDetectionController.ts
8
2
 
9
3
 
10
4
 
@@ -13,6 +7,13 @@ var _chunkRELRWIJNjs = require('./chunk-RELRWIJN.js');
13
7
  var _controllerutils = require('@metamask/controller-utils');
14
8
  var _pollingcontroller = require('@metamask/polling-controller');
15
9
  var DEFAULT_INTERVAL = 18e4;
10
+ var BlockaidResultType = /* @__PURE__ */ ((BlockaidResultType2) => {
11
+ BlockaidResultType2["Benign"] = "Benign";
12
+ BlockaidResultType2["Spam"] = "Spam";
13
+ BlockaidResultType2["Warning"] = "Warning";
14
+ BlockaidResultType2["Malicious"] = "Malicious";
15
+ return BlockaidResultType2;
16
+ })(BlockaidResultType || {});
16
17
  var NftDetectionController = class extends _pollingcontroller.StaticIntervalPollingControllerV1 {
17
18
  /**
18
19
  * Creates an NftDetectionController instance.
@@ -88,7 +89,7 @@ var NftDetectionController = class extends _pollingcontroller.StaticIntervalPoll
88
89
  address,
89
90
  next
90
91
  }) {
91
- return `${_controllerutils.OPENSEA_PROXY_URL}/chain/${"ethereum" /* ethereum */}/account/${address}/nfts?limit=200&next=${next ?? ""}`;
92
+ return `${_controllerutils.NFT_API_BASE_URL}/users/${address}/tokens?chainIds=1&limit=200&includeTopBid=true&continuation=${next ?? ""}`;
92
93
  }
93
94
  async getOwnerNfts(address) {
94
95
  let nftApiResponse;
@@ -97,32 +98,21 @@ var NftDetectionController = class extends _pollingcontroller.StaticIntervalPoll
97
98
  do {
98
99
  nftApiResponse = await _controllerutils.fetchWithErrorHandling.call(void 0, {
99
100
  url: this.getOwnerNftApi({ address, next }),
101
+ options: {
102
+ headers: {
103
+ Version: "1"
104
+ }
105
+ },
100
106
  timeout: 15e3
101
107
  });
102
108
  if (!nftApiResponse) {
103
109
  return nfts;
104
110
  }
105
- const newNfts = await Promise.all(
106
- nftApiResponse.nfts.map(async (nftV2) => {
107
- const nftV1 = _chunkRELRWIJNjs.mapOpenSeaNftV2ToV1.call(void 0, nftV2);
108
- if (!nftV1.image_url && nftV2.metadata_url) {
109
- const nftDetails = await _controllerutils.safelyExecute.call(void 0,
110
- () => _controllerutils.timeoutFetch.call(void 0,
111
- this.getNftApi({
112
- contractAddress: nftV2.contract,
113
- tokenId: nftV2.identifier
114
- }),
115
- void 0,
116
- 1e3
117
- ).then((r) => r.json())
118
- );
119
- nftV1.image_original_url = nftDetails?.nft?.image_url ?? null;
120
- }
121
- return nftV1;
122
- })
111
+ const newNfts = nftApiResponse.tokens.filter(
112
+ (elm) => elm.token.isSpam === false && (elm.blockaidResult?.result_type ? elm.blockaidResult?.result_type === "Benign" /* Benign */ : true)
123
113
  );
124
114
  nfts = [...nfts, ...newNfts];
125
- } while (next = nftApiResponse.next);
115
+ } while (next = nftApiResponse.continuation);
126
116
  return nfts;
127
117
  }
128
118
  async _executePoll(networkClientId, options) {
@@ -182,50 +172,45 @@ var NftDetectionController = class extends _pollingcontroller.StaticIntervalPoll
182
172
  const apiNfts = await this.getOwnerNfts(userAddress);
183
173
  const addNftPromises = apiNfts.map(async (nft) => {
184
174
  const {
185
- token_id,
186
- num_sales,
187
- background_color,
188
- image_url,
189
- image_preview_url,
190
- image_thumbnail_url,
191
- image_original_url,
192
- animation_url,
193
- animation_original_url,
175
+ tokenId: token_id,
176
+ contract,
177
+ kind,
178
+ image: image_url,
179
+ imageSmall: image_thumbnail_url,
180
+ metadata: { imageOriginal: image_original_url } = {},
194
181
  name,
195
182
  description,
196
- external_link,
197
- creator,
198
- asset_contract: { address, schema_name },
199
- last_sale
200
- } = nft;
183
+ attributes,
184
+ topBid,
185
+ lastSale,
186
+ rarityRank,
187
+ rarityScore,
188
+ collection
189
+ } = nft.token;
201
190
  let ignored;
202
191
  const { ignoredNfts } = this.getNftState();
203
192
  if (ignoredNfts.length) {
204
193
  ignored = ignoredNfts.find((c) => {
205
- return c.address === _controllerutils.toChecksumHexAddress.call(void 0, address) && c.tokenId === token_id;
194
+ return c.address === _controllerutils.toChecksumHexAddress.call(void 0, contract) && c.tokenId === token_id;
206
195
  });
207
196
  }
208
197
  if (!ignored) {
209
198
  const nftMetadata = Object.assign(
210
199
  {},
211
200
  { name },
212
- creator && { creator },
213
201
  description && { description },
214
202
  image_url && { image: image_url },
215
- num_sales && { numberOfSales: num_sales },
216
- background_color && { backgroundColor: background_color },
217
- image_preview_url && { imagePreview: image_preview_url },
218
203
  image_thumbnail_url && { imageThumbnail: image_thumbnail_url },
219
204
  image_original_url && { imageOriginal: image_original_url },
220
- animation_url && { animation: animation_url },
221
- animation_original_url && {
222
- animationOriginal: animation_original_url
223
- },
224
- schema_name && { standard: schema_name },
225
- external_link && { externalLink: external_link },
226
- last_sale && { lastSale: last_sale }
205
+ kind && { standard: kind.toUpperCase() },
206
+ lastSale && { lastSale },
207
+ attributes && { attributes },
208
+ topBid && { topBid },
209
+ rarityRank && { rarityRank },
210
+ rarityScore && { rarityScore },
211
+ collection && { collection }
227
212
  );
228
- await this.addNft(address, token_id, {
213
+ await this.addNft(contract, token_id, {
229
214
  nftMetadata,
230
215
  userAddress,
231
216
  source: "detected" /* Detected */,
@@ -241,5 +226,6 @@ var NftDetectionController_default = NftDetectionController;
241
226
 
242
227
 
243
228
 
244
- exports.NftDetectionController = NftDetectionController; exports.NftDetectionController_default = NftDetectionController_default;
245
- //# sourceMappingURL=chunk-WSOHBBCO.js.map
229
+
230
+ exports.BlockaidResultType = BlockaidResultType; exports.NftDetectionController = NftDetectionController; exports.NftDetectionController_default = NftDetectionController_default;
231
+ //# sourceMappingURL=chunk-JEGLHDNA.js.map