@metamask/assets-controllers 26.0.0 → 27.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.
- package/CHANGELOG.md +22 -1
- package/dist/AccountTrackerController.js +9 -236
- package/dist/AccountTrackerController.js.map +1 -1
- package/dist/AccountTrackerController.mjs +10 -0
- package/dist/AccountTrackerController.mjs.map +1 -0
- package/dist/AssetsContractController.js +17 -384
- package/dist/AssetsContractController.js.map +1 -1
- package/dist/AssetsContractController.mjs +18 -0
- package/dist/AssetsContractController.mjs.map +1 -0
- package/dist/CurrencyRateController.js +10 -157
- package/dist/CurrencyRateController.js.map +1 -1
- package/dist/CurrencyRateController.mjs +11 -0
- package/dist/CurrencyRateController.mjs.map +1 -0
- package/dist/NftController.js +15 -1041
- package/dist/NftController.js.map +1 -1
- package/dist/NftController.mjs +16 -0
- package/dist/NftController.mjs.map +1 -0
- package/dist/NftDetectionController.js +12 -213
- package/dist/NftDetectionController.js.map +1 -1
- package/dist/NftDetectionController.mjs +13 -0
- package/dist/NftDetectionController.mjs.map +1 -0
- package/dist/Standards/ERC20Standard.js +8 -142
- package/dist/Standards/ERC20Standard.js.map +1 -1
- package/dist/Standards/ERC20Standard.mjs +9 -0
- package/dist/Standards/ERC20Standard.mjs.map +1 -0
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.js +8 -217
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.js.map +1 -1
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.mjs +9 -0
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.mjs.map +1 -0
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.js +8 -174
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.js.map +1 -1
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.mjs +9 -0
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.mjs.map +1 -0
- package/dist/TokenBalancesController.js +11 -141
- package/dist/TokenBalancesController.js.map +1 -1
- package/dist/TokenBalancesController.mjs +12 -0
- package/dist/TokenBalancesController.mjs.map +1 -0
- package/dist/TokenDetectionController.js +16 -362
- package/dist/TokenDetectionController.js.map +1 -1
- package/dist/TokenDetectionController.mjs +17 -0
- package/dist/TokenDetectionController.mjs.map +1 -0
- package/dist/TokenListController.js +13 -250
- package/dist/TokenListController.js.map +1 -1
- package/dist/TokenListController.mjs +14 -0
- package/dist/TokenListController.mjs.map +1 -0
- package/dist/TokenRatesController.js +11 -328
- package/dist/TokenRatesController.js.map +1 -1
- package/dist/TokenRatesController.mjs +12 -0
- package/dist/TokenRatesController.mjs.map +1 -0
- package/dist/TokensController.js +15 -629
- package/dist/TokensController.js.map +1 -1
- package/dist/TokensController.mjs +16 -0
- package/dist/TokensController.mjs.map +1 -0
- package/dist/assetsUtil.js +41 -384
- package/dist/assetsUtil.js.map +1 -1
- package/dist/assetsUtil.mjs +42 -0
- package/dist/assetsUtil.mjs.map +1 -0
- package/dist/chunk-23F5W3A2.mjs +157 -0
- package/dist/chunk-23F5W3A2.mjs.map +1 -0
- package/dist/chunk-27KOXCQK.mjs +1 -0
- package/dist/chunk-27KOXCQK.mjs.map +1 -0
- package/dist/chunk-3FMWI46J.mjs +361 -0
- package/dist/chunk-3FMWI46J.mjs.map +1 -0
- package/dist/chunk-3FOTFZIL.mjs +411 -0
- package/dist/chunk-3FOTFZIL.mjs.map +1 -0
- package/dist/chunk-4LY47RPI.mjs +139 -0
- package/dist/chunk-4LY47RPI.mjs.map +1 -0
- package/dist/chunk-5C7ZVZNM.mjs +267 -0
- package/dist/chunk-5C7ZVZNM.mjs.map +1 -0
- package/dist/chunk-64EHFYLM.mjs +245 -0
- package/dist/chunk-64EHFYLM.mjs.map +1 -0
- package/dist/chunk-6C2ZIK7A.js +245 -0
- package/dist/chunk-6C2ZIK7A.js.map +1 -0
- package/dist/chunk-6DTCHPBB.mjs +178 -0
- package/dist/chunk-6DTCHPBB.mjs.map +1 -0
- package/dist/chunk-74IP34EL.js +1177 -0
- package/dist/chunk-74IP34EL.js.map +1 -0
- package/dist/chunk-BZI7P3TD.js +115 -0
- package/dist/chunk-BZI7P3TD.js.map +1 -0
- package/dist/chunk-C4PQK53K.js +32 -0
- package/dist/chunk-C4PQK53K.js.map +1 -0
- package/dist/chunk-CP3HC7AQ.mjs +12 -0
- package/dist/chunk-CP3HC7AQ.mjs.map +1 -0
- package/dist/chunk-GHZX5DE4.js +215 -0
- package/dist/chunk-GHZX5DE4.js.map +1 -0
- package/dist/chunk-GMKIOYCB.js +361 -0
- package/dist/chunk-GMKIOYCB.js.map +1 -0
- package/dist/chunk-H6TOSWUM.js +157 -0
- package/dist/chunk-H6TOSWUM.js.map +1 -0
- package/dist/chunk-HWFBJFHS.js +284 -0
- package/dist/chunk-HWFBJFHS.js.map +1 -0
- package/dist/chunk-J2JQQNHN.mjs +115 -0
- package/dist/chunk-J2JQQNHN.mjs.map +1 -0
- package/dist/chunk-J6HPEQL3.mjs +704 -0
- package/dist/chunk-J6HPEQL3.mjs.map +1 -0
- package/dist/chunk-JC4WAN2J.mjs +1177 -0
- package/dist/chunk-JC4WAN2J.mjs.map +1 -0
- package/dist/chunk-LD4GC7OR.js +139 -0
- package/dist/chunk-LD4GC7OR.js.map +1 -0
- package/dist/chunk-NG2UOKDD.js +178 -0
- package/dist/chunk-NG2UOKDD.js.map +1 -0
- package/dist/chunk-NGIXA5M5.mjs +418 -0
- package/dist/chunk-NGIXA5M5.mjs.map +1 -0
- package/dist/chunk-NLNXQHAU.js +704 -0
- package/dist/chunk-NLNXQHAU.js.map +1 -0
- package/dist/chunk-OBUR4TXH.js +1 -0
- package/dist/chunk-OBUR4TXH.js.map +1 -0
- package/dist/chunk-PAJTKWEC.mjs +246 -0
- package/dist/chunk-PAJTKWEC.mjs.map +1 -0
- package/dist/chunk-PRIXT2R6.js +411 -0
- package/dist/chunk-PRIXT2R6.js.map +1 -0
- package/dist/chunk-PUFSYRJZ.js +74 -0
- package/dist/chunk-PUFSYRJZ.js.map +1 -0
- package/dist/chunk-QHRPRO5U.mjs +74 -0
- package/dist/chunk-QHRPRO5U.mjs.map +1 -0
- package/dist/chunk-QR4CX2JT.mjs +284 -0
- package/dist/chunk-QR4CX2JT.mjs.map +1 -0
- package/dist/chunk-RGHTNZQ6.js +12 -0
- package/dist/chunk-RGHTNZQ6.js.map +1 -0
- package/dist/chunk-TCO22VIO.mjs +32 -0
- package/dist/chunk-TCO22VIO.mjs.map +1 -0
- package/dist/chunk-TDKCVCGP.mjs +215 -0
- package/dist/chunk-TDKCVCGP.mjs.map +1 -0
- package/dist/chunk-UAH5YURZ.js +343 -0
- package/dist/chunk-UAH5YURZ.js.map +1 -0
- package/dist/chunk-V4ZO3F2S.js +246 -0
- package/dist/chunk-V4ZO3F2S.js.map +1 -0
- package/dist/chunk-VDJBJAUB.js +418 -0
- package/dist/chunk-VDJBJAUB.js.map +1 -0
- package/dist/chunk-WRQ7POD7.mjs +343 -0
- package/dist/chunk-WRQ7POD7.mjs.map +1 -0
- package/dist/chunk-XUI43LEZ.mjs +30 -0
- package/dist/chunk-XUI43LEZ.mjs.map +1 -0
- package/dist/chunk-YD3NRMFC.js +267 -0
- package/dist/chunk-YD3NRMFC.js.map +1 -0
- package/dist/chunk-Z4BLTVTB.js +30 -0
- package/dist/chunk-Z4BLTVTB.js.map +1 -0
- package/dist/constants.js +7 -9
- package/dist/constants.js.map +1 -1
- package/dist/constants.mjs +8 -0
- package/dist/constants.mjs.map +1 -0
- package/dist/crypto-compare.js +7 -66
- package/dist/crypto-compare.js.map +1 -1
- package/dist/crypto-compare.mjs +8 -0
- package/dist/crypto-compare.mjs.map +1 -0
- package/dist/index.js +72 -39
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +73 -0
- package/dist/index.mjs.map +1 -0
- package/dist/token-prices-service/abstract-token-prices-service.js +1 -3
- package/dist/token-prices-service/abstract-token-prices-service.js.map +1 -1
- package/dist/token-prices-service/abstract-token-prices-service.mjs +1 -0
- package/dist/token-prices-service/abstract-token-prices-service.mjs.map +1 -0
- package/dist/token-prices-service/codefi-v2.js +11 -353
- package/dist/token-prices-service/codefi-v2.js.map +1 -1
- package/dist/token-prices-service/codefi-v2.mjs +12 -0
- package/dist/token-prices-service/codefi-v2.mjs.map +1 -0
- package/dist/token-prices-service/index.js +8 -5
- package/dist/token-prices-service/index.js.map +1 -1
- package/dist/token-prices-service/index.mjs +9 -0
- package/dist/token-prices-service/index.mjs.map +1 -0
- package/dist/token-service.js +14 -133
- package/dist/token-service.js.map +1 -1
- package/dist/token-service.mjs +15 -0
- package/dist/token-service.mjs.map +1 -0
- package/dist/tsconfig.build.tsbuildinfo +1 -0
- package/dist/types/AccountTrackerController.d.ts.map +1 -0
- package/dist/types/AssetsContractController.d.ts.map +1 -0
- package/dist/types/CurrencyRateController.d.ts.map +1 -0
- package/dist/types/NftController.d.ts.map +1 -0
- package/dist/types/NftDetectionController.d.ts.map +1 -0
- package/dist/types/Standards/ERC20Standard.d.ts.map +1 -0
- package/dist/types/Standards/NftStandards/ERC1155/ERC1155Standard.d.ts.map +1 -0
- package/dist/types/Standards/NftStandards/ERC721/ERC721Standard.d.ts.map +1 -0
- package/dist/types/TokenBalancesController.d.ts.map +1 -0
- package/dist/types/TokenDetectionController.d.ts.map +1 -0
- package/dist/types/TokenListController.d.ts.map +1 -0
- package/dist/types/TokenRatesController.d.ts.map +1 -0
- package/dist/{TokensController.d.ts → types/TokensController.d.ts} +10 -3
- package/dist/types/TokensController.d.ts.map +1 -0
- package/dist/types/assetsUtil.d.ts.map +1 -0
- package/dist/types/constants.d.ts.map +1 -0
- package/dist/types/crypto-compare.d.ts.map +1 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/token-prices-service/abstract-token-prices-service.d.ts.map +1 -0
- package/dist/types/token-prices-service/codefi-v2.d.ts.map +1 -0
- package/dist/types/token-prices-service/index.d.ts.map +1 -0
- package/dist/types/token-service.d.ts.map +1 -0
- package/package.json +24 -14
- package/dist/AccountTrackerController.d.ts.map +0 -1
- package/dist/AssetsContractController.d.ts.map +0 -1
- package/dist/CurrencyRateController.d.ts.map +0 -1
- package/dist/NftController.d.ts.map +0 -1
- package/dist/NftDetectionController.d.ts.map +0 -1
- package/dist/Standards/ERC20Standard.d.ts.map +0 -1
- package/dist/Standards/NftStandards/ERC1155/ERC1155Standard.d.ts.map +0 -1
- package/dist/Standards/NftStandards/ERC721/ERC721Standard.d.ts.map +0 -1
- package/dist/TokenBalancesController.d.ts.map +0 -1
- package/dist/TokenDetectionController.d.ts.map +0 -1
- package/dist/TokenListController.d.ts.map +0 -1
- package/dist/TokenRatesController.d.ts.map +0 -1
- package/dist/TokensController.d.ts.map +0 -1
- package/dist/assetsUtil.d.ts.map +0 -1
- package/dist/constants.d.ts.map +0 -1
- package/dist/crypto-compare.d.ts.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/token-prices-service/abstract-token-prices-service.d.ts.map +0 -1
- package/dist/token-prices-service/codefi-v2.d.ts.map +0 -1
- package/dist/token-prices-service/index.d.ts.map +0 -1
- package/dist/token-service.d.ts.map +0 -1
- /package/dist/{AccountTrackerController.d.ts → types/AccountTrackerController.d.ts} +0 -0
- /package/dist/{AssetsContractController.d.ts → types/AssetsContractController.d.ts} +0 -0
- /package/dist/{CurrencyRateController.d.ts → types/CurrencyRateController.d.ts} +0 -0
- /package/dist/{NftController.d.ts → types/NftController.d.ts} +0 -0
- /package/dist/{NftDetectionController.d.ts → types/NftDetectionController.d.ts} +0 -0
- /package/dist/{Standards → types/Standards}/ERC20Standard.d.ts +0 -0
- /package/dist/{Standards → types/Standards}/NftStandards/ERC1155/ERC1155Standard.d.ts +0 -0
- /package/dist/{Standards → types/Standards}/NftStandards/ERC721/ERC721Standard.d.ts +0 -0
- /package/dist/{TokenBalancesController.d.ts → types/TokenBalancesController.d.ts} +0 -0
- /package/dist/{TokenDetectionController.d.ts → types/TokenDetectionController.d.ts} +0 -0
- /package/dist/{TokenListController.d.ts → types/TokenListController.d.ts} +0 -0
- /package/dist/{TokenRatesController.d.ts → types/TokenRatesController.d.ts} +0 -0
- /package/dist/{assetsUtil.d.ts → types/assetsUtil.d.ts} +0 -0
- /package/dist/{constants.d.ts → types/constants.d.ts} +0 -0
- /package/dist/{crypto-compare.d.ts → types/crypto-compare.d.ts} +0 -0
- /package/dist/{index.d.ts → types/index.d.ts} +0 -0
- /package/dist/{token-prices-service → types/token-prices-service}/abstract-token-prices-service.d.ts +0 -0
- /package/dist/{token-prices-service → types/token-prices-service}/codefi-v2.d.ts +0 -0
- /package/dist/{token-prices-service → types/token-prices-service}/index.d.ts +0 -0
- /package/dist/{token-service.d.ts → types/token-service.d.ts} +0 -0
|
@@ -0,0 +1,411 @@
|
|
|
1
|
+
"use strict";Object.defineProperty(exports, "__esModule", {value: true}); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
2
|
+
|
|
3
|
+
var _chunkYD3NRMFCjs = require('./chunk-YD3NRMFC.js');
|
|
4
|
+
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
var _chunkZ4BLTVTBjs = require('./chunk-Z4BLTVTB.js');
|
|
10
|
+
|
|
11
|
+
// src/TokenDetectionController.ts
|
|
12
|
+
var _contractmetadata = require('@metamask/contract-metadata'); var _contractmetadata2 = _interopRequireDefault(_contractmetadata);
|
|
13
|
+
var _controllerutils = require('@metamask/controller-utils');
|
|
14
|
+
var _pollingcontroller = require('@metamask/polling-controller');
|
|
15
|
+
var DEFAULT_INTERVAL = 18e4;
|
|
16
|
+
function isEqualCaseInsensitive(value1, value2) {
|
|
17
|
+
if (typeof value1 !== "string" || typeof value2 !== "string") {
|
|
18
|
+
return false;
|
|
19
|
+
}
|
|
20
|
+
return value1.toLowerCase() === value2.toLowerCase();
|
|
21
|
+
}
|
|
22
|
+
var STATIC_MAINNET_TOKEN_LIST = Object.entries(
|
|
23
|
+
_contractmetadata2.default
|
|
24
|
+
).reduce((acc, [base, contract]) => {
|
|
25
|
+
const { logo, erc20, erc721, ...tokenMetadata } = contract;
|
|
26
|
+
return {
|
|
27
|
+
...acc,
|
|
28
|
+
[base.toLowerCase()]: {
|
|
29
|
+
...tokenMetadata,
|
|
30
|
+
address: base.toLowerCase(),
|
|
31
|
+
iconUrl: `images/contract/${logo}`,
|
|
32
|
+
aggregators: []
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
}, {});
|
|
36
|
+
var controllerName = "TokenDetectionController";
|
|
37
|
+
var _intervalId, _selectedAddress, _networkClientId, _tokenList, _disabled, _isUnlocked, _isDetectionEnabledFromPreferences, _isDetectionEnabledForNetwork, _getBalancesInSingleCall, _trackMetaMetricsEvent, _registerEventListeners, registerEventListeners_fn, _stopPolling, stopPolling_fn, _startPolling, startPolling_fn, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn, _restartTokenDetection, restartTokenDetection_fn, _getSlicesOfTokensToDetect, getSlicesOfTokensToDetect_fn, _addDetectedTokens, addDetectedTokens_fn;
|
|
38
|
+
var TokenDetectionController = class extends _pollingcontroller.StaticIntervalPollingController {
|
|
39
|
+
/**
|
|
40
|
+
* Creates a TokenDetectionController instance.
|
|
41
|
+
*
|
|
42
|
+
* @param options - The controller options.
|
|
43
|
+
* @param options.messenger - The controller messaging system.
|
|
44
|
+
* @param options.disabled - If set to true, all network requests are blocked.
|
|
45
|
+
* @param options.interval - Polling interval used to fetch new token rates
|
|
46
|
+
* @param options.selectedAddress - Vault selected address
|
|
47
|
+
* @param options.getBalancesInSingleCall - Gets the balances of a list of tokens for the given address.
|
|
48
|
+
* @param options.trackMetaMetricsEvent - Sets options for MetaMetrics event tracking.
|
|
49
|
+
*/
|
|
50
|
+
constructor({
|
|
51
|
+
selectedAddress,
|
|
52
|
+
interval = DEFAULT_INTERVAL,
|
|
53
|
+
disabled = true,
|
|
54
|
+
getBalancesInSingleCall,
|
|
55
|
+
trackMetaMetricsEvent,
|
|
56
|
+
messenger
|
|
57
|
+
}) {
|
|
58
|
+
super({
|
|
59
|
+
name: controllerName,
|
|
60
|
+
messenger,
|
|
61
|
+
state: {},
|
|
62
|
+
metadata: {}
|
|
63
|
+
});
|
|
64
|
+
/**
|
|
65
|
+
* Constructor helper for registering this controller's messaging system subscriptions to controller events.
|
|
66
|
+
*/
|
|
67
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _registerEventListeners);
|
|
68
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _stopPolling);
|
|
69
|
+
/**
|
|
70
|
+
* Starts a new polling interval.
|
|
71
|
+
*/
|
|
72
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _startPolling);
|
|
73
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _getCorrectChainIdAndNetworkClientId);
|
|
74
|
+
/**
|
|
75
|
+
* Restart token detection polling period and call detectNewTokens
|
|
76
|
+
* in case of address change or user session initialization.
|
|
77
|
+
*
|
|
78
|
+
* @param options - Options for restart token detection.
|
|
79
|
+
* @param options.selectedAddress - the selectedAddress against which to detect for token balances
|
|
80
|
+
* @param options.networkClientId - The ID of the network client to use.
|
|
81
|
+
*/
|
|
82
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _restartTokenDetection);
|
|
83
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _getSlicesOfTokensToDetect);
|
|
84
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _addDetectedTokens);
|
|
85
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _intervalId, void 0);
|
|
86
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _selectedAddress, void 0);
|
|
87
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _networkClientId, void 0);
|
|
88
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _tokenList, {});
|
|
89
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _disabled, void 0);
|
|
90
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _isUnlocked, void 0);
|
|
91
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _isDetectionEnabledFromPreferences, void 0);
|
|
92
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _isDetectionEnabledForNetwork, void 0);
|
|
93
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _getBalancesInSingleCall, void 0);
|
|
94
|
+
_chunkZ4BLTVTBjs.__privateAdd.call(void 0, this, _trackMetaMetricsEvent, void 0);
|
|
95
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _disabled, disabled);
|
|
96
|
+
this.setIntervalLength(interval);
|
|
97
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _selectedAddress, selectedAddress ?? this.messagingSystem.call("AccountsController:getSelectedAccount").address);
|
|
98
|
+
const { chainId, networkClientId } = _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn).call(this);
|
|
99
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _networkClientId, networkClientId);
|
|
100
|
+
const { useTokenDetection: defaultUseTokenDetection } = this.messagingSystem.call("PreferencesController:getState");
|
|
101
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _isDetectionEnabledFromPreferences, defaultUseTokenDetection);
|
|
102
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _isDetectionEnabledForNetwork, _chunkYD3NRMFCjs.isTokenDetectionSupportedForNetwork.call(void 0, chainId));
|
|
103
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _getBalancesInSingleCall, getBalancesInSingleCall);
|
|
104
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _trackMetaMetricsEvent, trackMetaMetricsEvent);
|
|
105
|
+
const { isUnlocked } = this.messagingSystem.call(
|
|
106
|
+
"KeyringController:getState"
|
|
107
|
+
);
|
|
108
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _isUnlocked, isUnlocked);
|
|
109
|
+
_chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _registerEventListeners, registerEventListeners_fn).call(this);
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Allows controller to make active and passive polling requests
|
|
113
|
+
*/
|
|
114
|
+
enable() {
|
|
115
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _disabled, false);
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Blocks controller from making network calls
|
|
119
|
+
*/
|
|
120
|
+
disable() {
|
|
121
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _disabled, true);
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Internal isActive state
|
|
125
|
+
* @type {boolean}
|
|
126
|
+
*/
|
|
127
|
+
get isActive() {
|
|
128
|
+
return !_chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _disabled) && _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _isUnlocked);
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Start polling for detected tokens.
|
|
132
|
+
*/
|
|
133
|
+
async start() {
|
|
134
|
+
this.enable();
|
|
135
|
+
await _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _startPolling, startPolling_fn).call(this);
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Stop polling for detected tokens.
|
|
139
|
+
*/
|
|
140
|
+
stop() {
|
|
141
|
+
this.disable();
|
|
142
|
+
_chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _stopPolling, stopPolling_fn).call(this);
|
|
143
|
+
}
|
|
144
|
+
async _executePoll(networkClientId, options) {
|
|
145
|
+
if (!this.isActive) {
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
await this.detectTokens({
|
|
149
|
+
networkClientId,
|
|
150
|
+
selectedAddress: options.address
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* 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.
|
|
155
|
+
* 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.
|
|
156
|
+
*
|
|
157
|
+
* @param options - Options for token detection.
|
|
158
|
+
* @param options.networkClientId - The ID of the network client to use.
|
|
159
|
+
* @param options.selectedAddress - the selectedAddress against which to detect for token balances.
|
|
160
|
+
*/
|
|
161
|
+
async detectTokens({
|
|
162
|
+
networkClientId,
|
|
163
|
+
selectedAddress
|
|
164
|
+
} = {}) {
|
|
165
|
+
if (!this.isActive) {
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
const addressAgainstWhichToDetect = selectedAddress ?? _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _selectedAddress);
|
|
169
|
+
const { chainId, networkClientId: selectedNetworkClientId } = _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn).call(this, networkClientId);
|
|
170
|
+
const chainIdAgainstWhichToDetect = chainId;
|
|
171
|
+
const networkClientIdAgainstWhichToDetect = selectedNetworkClientId;
|
|
172
|
+
if (!_chunkYD3NRMFCjs.isTokenDetectionSupportedForNetwork.call(void 0, chainIdAgainstWhichToDetect)) {
|
|
173
|
+
return;
|
|
174
|
+
}
|
|
175
|
+
if (!_chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _isDetectionEnabledFromPreferences) && chainIdAgainstWhichToDetect !== _controllerutils.ChainId.mainnet) {
|
|
176
|
+
return;
|
|
177
|
+
}
|
|
178
|
+
const isTokenDetectionInactiveInMainnet = !_chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _isDetectionEnabledFromPreferences) && chainIdAgainstWhichToDetect === _controllerutils.ChainId.mainnet;
|
|
179
|
+
const { tokensChainsCache } = this.messagingSystem.call(
|
|
180
|
+
"TokenListController:getState"
|
|
181
|
+
);
|
|
182
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _tokenList, isTokenDetectionInactiveInMainnet ? STATIC_MAINNET_TOKEN_LIST : tokensChainsCache[chainIdAgainstWhichToDetect]?.data ?? {});
|
|
183
|
+
for (const tokensSlice of _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _getSlicesOfTokensToDetect, getSlicesOfTokensToDetect_fn).call(this, {
|
|
184
|
+
chainId: chainIdAgainstWhichToDetect,
|
|
185
|
+
selectedAddress: addressAgainstWhichToDetect
|
|
186
|
+
})) {
|
|
187
|
+
await _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _addDetectedTokens, addDetectedTokens_fn).call(this, {
|
|
188
|
+
tokensSlice,
|
|
189
|
+
selectedAddress: addressAgainstWhichToDetect,
|
|
190
|
+
networkClientId: networkClientIdAgainstWhichToDetect,
|
|
191
|
+
chainId: chainIdAgainstWhichToDetect
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
};
|
|
196
|
+
_intervalId = new WeakMap();
|
|
197
|
+
_selectedAddress = new WeakMap();
|
|
198
|
+
_networkClientId = new WeakMap();
|
|
199
|
+
_tokenList = new WeakMap();
|
|
200
|
+
_disabled = new WeakMap();
|
|
201
|
+
_isUnlocked = new WeakMap();
|
|
202
|
+
_isDetectionEnabledFromPreferences = new WeakMap();
|
|
203
|
+
_isDetectionEnabledForNetwork = new WeakMap();
|
|
204
|
+
_getBalancesInSingleCall = new WeakMap();
|
|
205
|
+
_trackMetaMetricsEvent = new WeakMap();
|
|
206
|
+
_registerEventListeners = new WeakSet();
|
|
207
|
+
registerEventListeners_fn = function() {
|
|
208
|
+
this.messagingSystem.subscribe("KeyringController:unlock", async () => {
|
|
209
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _isUnlocked, true);
|
|
210
|
+
await _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _restartTokenDetection, restartTokenDetection_fn).call(this);
|
|
211
|
+
});
|
|
212
|
+
this.messagingSystem.subscribe("KeyringController:lock", () => {
|
|
213
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _isUnlocked, false);
|
|
214
|
+
_chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _stopPolling, stopPolling_fn).call(this);
|
|
215
|
+
});
|
|
216
|
+
this.messagingSystem.subscribe(
|
|
217
|
+
"TokenListController:stateChange",
|
|
218
|
+
async ({ tokenList }) => {
|
|
219
|
+
const hasTokens = Object.keys(tokenList).length;
|
|
220
|
+
if (hasTokens) {
|
|
221
|
+
await _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _restartTokenDetection, restartTokenDetection_fn).call(this);
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
);
|
|
225
|
+
this.messagingSystem.subscribe(
|
|
226
|
+
"PreferencesController:stateChange",
|
|
227
|
+
async ({ selectedAddress: newSelectedAddress, useTokenDetection }) => {
|
|
228
|
+
const isSelectedAddressChanged = _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _selectedAddress) !== newSelectedAddress;
|
|
229
|
+
const isDetectionChangedFromPreferences = _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _isDetectionEnabledFromPreferences) !== useTokenDetection;
|
|
230
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _selectedAddress, newSelectedAddress);
|
|
231
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _isDetectionEnabledFromPreferences, useTokenDetection);
|
|
232
|
+
if (useTokenDetection && (isSelectedAddressChanged || isDetectionChangedFromPreferences)) {
|
|
233
|
+
await _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
|
|
234
|
+
selectedAddress: _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _selectedAddress)
|
|
235
|
+
});
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
);
|
|
239
|
+
this.messagingSystem.subscribe(
|
|
240
|
+
"AccountsController:selectedAccountChange",
|
|
241
|
+
async ({ address: newSelectedAddress }) => {
|
|
242
|
+
const isSelectedAddressChanged = _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _selectedAddress) !== newSelectedAddress;
|
|
243
|
+
if (isSelectedAddressChanged && _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _isDetectionEnabledFromPreferences)) {
|
|
244
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _selectedAddress, newSelectedAddress);
|
|
245
|
+
await _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
|
|
246
|
+
selectedAddress: _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _selectedAddress)
|
|
247
|
+
});
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
);
|
|
251
|
+
this.messagingSystem.subscribe(
|
|
252
|
+
"NetworkController:networkDidChange",
|
|
253
|
+
async ({ selectedNetworkClientId }) => {
|
|
254
|
+
const isNetworkClientIdChanged = _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _networkClientId) !== selectedNetworkClientId;
|
|
255
|
+
const { chainId: newChainId } = _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _getCorrectChainIdAndNetworkClientId, getCorrectChainIdAndNetworkClientId_fn).call(this, selectedNetworkClientId);
|
|
256
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _isDetectionEnabledForNetwork, _chunkYD3NRMFCjs.isTokenDetectionSupportedForNetwork.call(void 0, newChainId));
|
|
257
|
+
if (isNetworkClientIdChanged && _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _isDetectionEnabledForNetwork)) {
|
|
258
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _networkClientId, selectedNetworkClientId);
|
|
259
|
+
await _chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _restartTokenDetection, restartTokenDetection_fn).call(this, {
|
|
260
|
+
networkClientId: _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _networkClientId)
|
|
261
|
+
});
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
);
|
|
265
|
+
};
|
|
266
|
+
_stopPolling = new WeakSet();
|
|
267
|
+
stopPolling_fn = function() {
|
|
268
|
+
if (_chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _intervalId)) {
|
|
269
|
+
clearInterval(_chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _intervalId));
|
|
270
|
+
}
|
|
271
|
+
};
|
|
272
|
+
_startPolling = new WeakSet();
|
|
273
|
+
startPolling_fn = async function() {
|
|
274
|
+
if (!this.isActive) {
|
|
275
|
+
return;
|
|
276
|
+
}
|
|
277
|
+
_chunkZ4BLTVTBjs.__privateMethod.call(void 0, this, _stopPolling, stopPolling_fn).call(this);
|
|
278
|
+
await this.detectTokens();
|
|
279
|
+
_chunkZ4BLTVTBjs.__privateSet.call(void 0, this, _intervalId, setInterval(async () => {
|
|
280
|
+
await this.detectTokens();
|
|
281
|
+
}, this.getIntervalLength()));
|
|
282
|
+
};
|
|
283
|
+
_getCorrectChainIdAndNetworkClientId = new WeakSet();
|
|
284
|
+
getCorrectChainIdAndNetworkClientId_fn = function(networkClientId) {
|
|
285
|
+
if (networkClientId) {
|
|
286
|
+
const networkConfiguration = this.messagingSystem.call(
|
|
287
|
+
"NetworkController:getNetworkConfigurationByNetworkClientId",
|
|
288
|
+
networkClientId
|
|
289
|
+
);
|
|
290
|
+
if (networkConfiguration) {
|
|
291
|
+
return {
|
|
292
|
+
chainId: networkConfiguration.chainId,
|
|
293
|
+
networkClientId
|
|
294
|
+
};
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
const { selectedNetworkClientId } = this.messagingSystem.call(
|
|
298
|
+
"NetworkController:getState"
|
|
299
|
+
);
|
|
300
|
+
const {
|
|
301
|
+
configuration: { chainId }
|
|
302
|
+
} = this.messagingSystem.call(
|
|
303
|
+
"NetworkController:getNetworkClientById",
|
|
304
|
+
selectedNetworkClientId
|
|
305
|
+
);
|
|
306
|
+
return {
|
|
307
|
+
chainId,
|
|
308
|
+
networkClientId: selectedNetworkClientId
|
|
309
|
+
};
|
|
310
|
+
};
|
|
311
|
+
_restartTokenDetection = new WeakSet();
|
|
312
|
+
restartTokenDetection_fn = async function({
|
|
313
|
+
selectedAddress,
|
|
314
|
+
networkClientId
|
|
315
|
+
} = {}) {
|
|
316
|
+
await this.detectTokens({
|
|
317
|
+
networkClientId,
|
|
318
|
+
selectedAddress
|
|
319
|
+
});
|
|
320
|
+
this.setIntervalLength(DEFAULT_INTERVAL);
|
|
321
|
+
};
|
|
322
|
+
_getSlicesOfTokensToDetect = new WeakSet();
|
|
323
|
+
getSlicesOfTokensToDetect_fn = function({
|
|
324
|
+
chainId,
|
|
325
|
+
selectedAddress
|
|
326
|
+
}) {
|
|
327
|
+
const { allTokens, allDetectedTokens, allIgnoredTokens } = this.messagingSystem.call("TokensController:getState");
|
|
328
|
+
const [tokensAddresses, detectedTokensAddresses, ignoredTokensAddresses] = [
|
|
329
|
+
allTokens,
|
|
330
|
+
allDetectedTokens,
|
|
331
|
+
allIgnoredTokens
|
|
332
|
+
].map(
|
|
333
|
+
(tokens) => (tokens[chainId]?.[selectedAddress] ?? []).map(
|
|
334
|
+
(value) => typeof value === "string" ? value : value.address
|
|
335
|
+
)
|
|
336
|
+
);
|
|
337
|
+
const tokensToDetect = [];
|
|
338
|
+
for (const tokenAddress of Object.keys(_chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _tokenList))) {
|
|
339
|
+
if ([
|
|
340
|
+
tokensAddresses,
|
|
341
|
+
detectedTokensAddresses,
|
|
342
|
+
ignoredTokensAddresses
|
|
343
|
+
].every(
|
|
344
|
+
(addresses) => !addresses.find(
|
|
345
|
+
(address) => isEqualCaseInsensitive(address, tokenAddress)
|
|
346
|
+
)
|
|
347
|
+
)) {
|
|
348
|
+
tokensToDetect.push(tokenAddress);
|
|
349
|
+
}
|
|
350
|
+
}
|
|
351
|
+
const slicesOfTokensToDetect = [];
|
|
352
|
+
for (let i = 0, size = 1e3; i < tokensToDetect.length; i += size) {
|
|
353
|
+
slicesOfTokensToDetect.push(tokensToDetect.slice(i, i + size));
|
|
354
|
+
}
|
|
355
|
+
return slicesOfTokensToDetect;
|
|
356
|
+
};
|
|
357
|
+
_addDetectedTokens = new WeakSet();
|
|
358
|
+
addDetectedTokens_fn = async function({
|
|
359
|
+
tokensSlice,
|
|
360
|
+
selectedAddress,
|
|
361
|
+
networkClientId,
|
|
362
|
+
chainId
|
|
363
|
+
}) {
|
|
364
|
+
await _controllerutils.safelyExecute.call(void 0, async () => {
|
|
365
|
+
const balances = await _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _getBalancesInSingleCall).call(this, selectedAddress, tokensSlice, networkClientId);
|
|
366
|
+
const tokensWithBalance = [];
|
|
367
|
+
const eventTokensDetails = [];
|
|
368
|
+
for (const nonZeroTokenAddress of Object.keys(balances)) {
|
|
369
|
+
const { decimals, symbol, aggregators, iconUrl, name } = _chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _tokenList)[nonZeroTokenAddress];
|
|
370
|
+
eventTokensDetails.push(`${symbol} - ${nonZeroTokenAddress}`);
|
|
371
|
+
tokensWithBalance.push({
|
|
372
|
+
address: nonZeroTokenAddress,
|
|
373
|
+
decimals,
|
|
374
|
+
symbol,
|
|
375
|
+
aggregators,
|
|
376
|
+
image: iconUrl,
|
|
377
|
+
isERC721: false,
|
|
378
|
+
name
|
|
379
|
+
});
|
|
380
|
+
}
|
|
381
|
+
if (tokensWithBalance.length) {
|
|
382
|
+
_chunkZ4BLTVTBjs.__privateGet.call(void 0, this, _trackMetaMetricsEvent).call(this, {
|
|
383
|
+
event: "Token Detected",
|
|
384
|
+
category: "Wallet",
|
|
385
|
+
properties: {
|
|
386
|
+
tokens: eventTokensDetails,
|
|
387
|
+
token_standard: "ERC20",
|
|
388
|
+
asset_type: "TOKEN"
|
|
389
|
+
}
|
|
390
|
+
});
|
|
391
|
+
await this.messagingSystem.call(
|
|
392
|
+
"TokensController:addDetectedTokens",
|
|
393
|
+
tokensWithBalance,
|
|
394
|
+
{
|
|
395
|
+
selectedAddress,
|
|
396
|
+
chainId
|
|
397
|
+
}
|
|
398
|
+
);
|
|
399
|
+
}
|
|
400
|
+
});
|
|
401
|
+
};
|
|
402
|
+
var TokenDetectionController_default = TokenDetectionController;
|
|
403
|
+
|
|
404
|
+
|
|
405
|
+
|
|
406
|
+
|
|
407
|
+
|
|
408
|
+
|
|
409
|
+
|
|
410
|
+
exports.isEqualCaseInsensitive = isEqualCaseInsensitive; exports.STATIC_MAINNET_TOKEN_LIST = STATIC_MAINNET_TOKEN_LIST; exports.controllerName = controllerName; exports.TokenDetectionController = TokenDetectionController; exports.TokenDetectionController_default = TokenDetectionController_default;
|
|
411
|
+
//# sourceMappingURL=chunk-PRIXT2R6.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/TokenDetectionController.ts"],"names":[],"mappings":"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","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 (\n useTokenDetection &&\n (isSelectedAddressChanged || isDetectionChangedFromPreferences)\n ) {\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 (\n isSelectedAddressChanged &&\n this.#isDetectionEnabledFromPreferences\n ) {\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"]}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
"use strict";Object.defineProperty(exports, "__esModule", {value: true});
|
|
2
|
+
|
|
3
|
+
var _chunkYD3NRMFCjs = require('./chunk-YD3NRMFC.js');
|
|
4
|
+
|
|
5
|
+
// src/token-service.ts
|
|
6
|
+
var _controllerutils = require('@metamask/controller-utils');
|
|
7
|
+
var TOKEN_END_POINT_API = "https://token-api.metaswap.codefi.network";
|
|
8
|
+
var TOKEN_METADATA_NO_SUPPORT_ERROR = "TokenService Error: Network does not support fetchTokenMetadata";
|
|
9
|
+
function getTokensURL(chainId) {
|
|
10
|
+
return `${TOKEN_END_POINT_API}/tokens/${_controllerutils.convertHexToDecimal.call(void 0,
|
|
11
|
+
chainId
|
|
12
|
+
)}?occurrenceFloor=3&includeNativeAssets=false&includeDuplicateSymbolAssets=false&includeTokenFees=false&includeAssetType=false`;
|
|
13
|
+
}
|
|
14
|
+
function getTokenMetadataURL(chainId, tokenAddress) {
|
|
15
|
+
return `${TOKEN_END_POINT_API}/token/${_controllerutils.convertHexToDecimal.call(void 0,
|
|
16
|
+
chainId
|
|
17
|
+
)}?address=${tokenAddress}`;
|
|
18
|
+
}
|
|
19
|
+
var tenSecondsInMilliseconds = 1e4;
|
|
20
|
+
var defaultTimeout = tenSecondsInMilliseconds;
|
|
21
|
+
async function fetchTokenListByChainId(chainId, abortSignal, { timeout = defaultTimeout } = {}) {
|
|
22
|
+
const tokenURL = getTokensURL(chainId);
|
|
23
|
+
const response = await queryApi(tokenURL, abortSignal, timeout);
|
|
24
|
+
if (response) {
|
|
25
|
+
return parseJsonResponse(response);
|
|
26
|
+
}
|
|
27
|
+
return void 0;
|
|
28
|
+
}
|
|
29
|
+
async function fetchTokenMetadata(chainId, tokenAddress, abortSignal, { timeout = defaultTimeout } = {}) {
|
|
30
|
+
if (!_chunkYD3NRMFCjs.isTokenListSupportedForNetwork.call(void 0, chainId)) {
|
|
31
|
+
throw new Error(TOKEN_METADATA_NO_SUPPORT_ERROR);
|
|
32
|
+
}
|
|
33
|
+
const tokenMetadataURL = getTokenMetadataURL(chainId, tokenAddress);
|
|
34
|
+
const response = await queryApi(tokenMetadataURL, abortSignal, timeout);
|
|
35
|
+
if (response) {
|
|
36
|
+
return parseJsonResponse(response);
|
|
37
|
+
}
|
|
38
|
+
return void 0;
|
|
39
|
+
}
|
|
40
|
+
async function queryApi(apiURL, abortSignal, timeout) {
|
|
41
|
+
const fetchOptions = {
|
|
42
|
+
referrer: apiURL,
|
|
43
|
+
referrerPolicy: "no-referrer-when-downgrade",
|
|
44
|
+
method: "GET",
|
|
45
|
+
mode: "cors",
|
|
46
|
+
signal: abortSignal,
|
|
47
|
+
cache: "default"
|
|
48
|
+
};
|
|
49
|
+
fetchOptions.headers = new window.Headers();
|
|
50
|
+
fetchOptions.headers.set("Content-Type", "application/json");
|
|
51
|
+
try {
|
|
52
|
+
return await _controllerutils.timeoutFetch.call(void 0, apiURL, fetchOptions, timeout);
|
|
53
|
+
} catch (error) {
|
|
54
|
+
if (error instanceof Error && error.name === "AbortError") {
|
|
55
|
+
console.log("Request is aborted");
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
return void 0;
|
|
59
|
+
}
|
|
60
|
+
async function parseJsonResponse(apiResponse) {
|
|
61
|
+
const responseObj = await apiResponse.json();
|
|
62
|
+
if (responseObj?.error) {
|
|
63
|
+
throw new Error(`TokenService Error: ${responseObj.error}`);
|
|
64
|
+
}
|
|
65
|
+
return responseObj;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
|
|
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-PUFSYRJZ.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/token-service.ts"],"names":[],"mappings":";;;;;AAAA,SAAS,qBAAqB,oBAAoB;AAK3C,IAAM,sBAAsB;AAC5B,IAAM,kCACX;AAQF,SAAS,aAAa,SAAc;AAClC,SAAO,GAAG,mBAAmB,WAAW;AAAA,IACtC;AAAA,EACF,CAAC;AACH;AASA,SAAS,oBAAoB,SAAc,cAAsB;AAC/D,SAAO,GAAG,mBAAmB,UAAU;AAAA,IACrC;AAAA,EACF,CAAC,YAAY,YAAY;AAC3B;AAEA,IAAM,2BAA2B;AAIjC,IAAM,iBAAiB;AAYvB,eAAsB,wBACpB,SACA,aACA,EAAE,UAAU,eAAe,IAAI,CAAC,GACd;AAClB,QAAM,WAAW,aAAa,OAAO;AACrC,QAAM,WAAW,MAAM,SAAS,UAAU,aAAa,OAAO;AAC9D,MAAI,UAAU;AACZ,WAAO,kBAAkB,QAAQ;AAAA,EACnC;AACA,SAAO;AACT;AAaA,eAAsB,mBACpB,SACA,cACA,aACA,EAAE,UAAU,eAAe,IAAI,CAAC,GACR;AACxB,MAAI,CAAC,+BAA+B,OAAO,GAAG;AAC5C,UAAM,IAAI,MAAM,+BAA+B;AAAA,EACjD;AACA,QAAM,mBAAmB,oBAAoB,SAAS,YAAY;AAClE,QAAM,WAAW,MAAM,SAAS,kBAAkB,aAAa,OAAO;AACtE,MAAI,UAAU;AACZ,WAAO,kBAAkB,QAAQ;AAAA,EACnC;AACA,SAAO;AACT;AAUA,eAAe,SACb,QACA,aACA,SAC+B;AAC/B,QAAM,eAA4B;AAAA,IAChC,UAAU;AAAA,IACV,gBAAgB;AAAA,IAChB,QAAQ;AAAA,IACR,MAAM;AAAA,IACN,QAAQ;AAAA,IACR,OAAO;AAAA,EACT;AACA,eAAa,UAAU,IAAI,OAAO,QAAQ;AAC1C,eAAa,QAAQ,IAAI,gBAAgB,kBAAkB;AAC3D,MAAI;AACF,WAAO,MAAM,aAAa,QAAQ,cAAc,OAAO;AAAA,EACzD,SAAS,OAAO;AACd,QAAI,iBAAiB,SAAS,MAAM,SAAS,cAAc;AACzD,cAAQ,IAAI,oBAAoB;AAAA,IAClC;AAAA,EACF;AACA,SAAO;AACT;AASA,eAAe,kBAAkB,aAAyC;AACxE,QAAM,cAAc,MAAM,YAAY,KAAK;AAE3C,MAAI,aAAa,OAAO;AACtB,UAAM,IAAI,MAAM,uBAAuB,YAAY,KAAK,EAAE;AAAA,EAC5D;AACA,SAAO;AACT","sourcesContent":["import { convertHexToDecimal, timeoutFetch } from '@metamask/controller-utils';\nimport type { Hex } from '@metamask/utils';\n\nimport { isTokenListSupportedForNetwork } from './assetsUtil';\n\nexport const TOKEN_END_POINT_API = 'https://token-api.metaswap.codefi.network';\nexport const TOKEN_METADATA_NO_SUPPORT_ERROR =\n 'TokenService Error: Network does not support fetchTokenMetadata';\n\n/**\n * Get the tokens URL for a specific network.\n *\n * @param chainId - The chain ID of the network the tokens requested are on.\n * @returns The tokens URL.\n */\nfunction getTokensURL(chainId: Hex) {\n return `${TOKEN_END_POINT_API}/tokens/${convertHexToDecimal(\n chainId,\n )}?occurrenceFloor=3&includeNativeAssets=false&includeDuplicateSymbolAssets=false&includeTokenFees=false&includeAssetType=false`;\n}\n\n/**\n * Get the token metadata URL for the given network and token.\n *\n * @param chainId - The chain ID of the network the token is on.\n * @param tokenAddress - The token address.\n * @returns The token metadata URL.\n */\nfunction getTokenMetadataURL(chainId: Hex, tokenAddress: string) {\n return `${TOKEN_END_POINT_API}/token/${convertHexToDecimal(\n chainId,\n )}?address=${tokenAddress}`;\n}\n\nconst tenSecondsInMilliseconds = 10_000;\n\n// Token list averages 1.6 MB in size\n// timeoutFetch by default has a 500ms timeout, which will almost always timeout given the response size.\nconst defaultTimeout = tenSecondsInMilliseconds;\n\n/**\n * Fetch the list of token metadata for a given network. This request is cancellable using the\n * abort signal passed in.\n *\n * @param chainId - The chain ID of the network the requested tokens are on.\n * @param abortSignal - The abort signal used to cancel the request if necessary.\n * @param options - Additional fetch options.\n * @param options.timeout - The fetch timeout.\n * @returns The token list, or `undefined` if the request was cancelled.\n */\nexport async function fetchTokenListByChainId(\n chainId: Hex,\n abortSignal: AbortSignal,\n { timeout = defaultTimeout } = {},\n): Promise<unknown> {\n const tokenURL = getTokensURL(chainId);\n const response = await queryApi(tokenURL, abortSignal, timeout);\n if (response) {\n return parseJsonResponse(response);\n }\n return undefined;\n}\n\n/**\n * Fetch metadata for the token address provided for a given network. This request is cancellable\n * using the abort signal passed in.\n *\n * @param chainId - The chain ID of the network the token is on.\n * @param tokenAddress - The address of the token to fetch metadata for.\n * @param abortSignal - The abort signal used to cancel the request if necessary.\n * @param options - Additional fetch options.\n * @param options.timeout - The fetch timeout.\n * @returns The token metadata, or `undefined` if the request was either aborted or failed.\n */\nexport async function fetchTokenMetadata<T>(\n chainId: Hex,\n tokenAddress: string,\n abortSignal: AbortSignal,\n { timeout = defaultTimeout } = {},\n): Promise<T | undefined> {\n if (!isTokenListSupportedForNetwork(chainId)) {\n throw new Error(TOKEN_METADATA_NO_SUPPORT_ERROR);\n }\n const tokenMetadataURL = getTokenMetadataURL(chainId, tokenAddress);\n const response = await queryApi(tokenMetadataURL, abortSignal, timeout);\n if (response) {\n return parseJsonResponse(response) as Promise<T>;\n }\n return undefined;\n}\n\n/**\n * Perform fetch request against the api.\n *\n * @param apiURL - The URL of the API to fetch.\n * @param abortSignal - The abort signal used to cancel the request if necessary.\n * @param timeout - The fetch timeout.\n * @returns Promise resolving request response.\n */\nasync function queryApi(\n apiURL: string,\n abortSignal: AbortSignal,\n timeout: number,\n): Promise<Response | undefined> {\n const fetchOptions: RequestInit = {\n referrer: apiURL,\n referrerPolicy: 'no-referrer-when-downgrade',\n method: 'GET',\n mode: 'cors',\n signal: abortSignal,\n cache: 'default',\n };\n fetchOptions.headers = new window.Headers();\n fetchOptions.headers.set('Content-Type', 'application/json');\n try {\n return await timeoutFetch(apiURL, fetchOptions, timeout);\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n console.log('Request is aborted');\n }\n }\n return undefined;\n}\n\n/**\n * Parse an API response and return the response JSON data.\n *\n * @param apiResponse - The API response to parse.\n * @returns The response JSON data.\n * @throws Will throw if the response includes an error.\n */\nasync function parseJsonResponse(apiResponse: Response): Promise<unknown> {\n const responseObj = await apiResponse.json();\n // api may return errors as json without setting an error http status code\n if (responseObj?.error) {\n throw new Error(`TokenService Error: ${responseObj.error}`);\n }\n return responseObj;\n}\n"]}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import {
|
|
2
|
+
isTokenListSupportedForNetwork
|
|
3
|
+
} from "./chunk-5C7ZVZNM.mjs";
|
|
4
|
+
|
|
5
|
+
// src/token-service.ts
|
|
6
|
+
import { convertHexToDecimal, timeoutFetch } from "@metamask/controller-utils";
|
|
7
|
+
var TOKEN_END_POINT_API = "https://token-api.metaswap.codefi.network";
|
|
8
|
+
var TOKEN_METADATA_NO_SUPPORT_ERROR = "TokenService Error: Network does not support fetchTokenMetadata";
|
|
9
|
+
function getTokensURL(chainId) {
|
|
10
|
+
return `${TOKEN_END_POINT_API}/tokens/${convertHexToDecimal(
|
|
11
|
+
chainId
|
|
12
|
+
)}?occurrenceFloor=3&includeNativeAssets=false&includeDuplicateSymbolAssets=false&includeTokenFees=false&includeAssetType=false`;
|
|
13
|
+
}
|
|
14
|
+
function getTokenMetadataURL(chainId, tokenAddress) {
|
|
15
|
+
return `${TOKEN_END_POINT_API}/token/${convertHexToDecimal(
|
|
16
|
+
chainId
|
|
17
|
+
)}?address=${tokenAddress}`;
|
|
18
|
+
}
|
|
19
|
+
var tenSecondsInMilliseconds = 1e4;
|
|
20
|
+
var defaultTimeout = tenSecondsInMilliseconds;
|
|
21
|
+
async function fetchTokenListByChainId(chainId, abortSignal, { timeout = defaultTimeout } = {}) {
|
|
22
|
+
const tokenURL = getTokensURL(chainId);
|
|
23
|
+
const response = await queryApi(tokenURL, abortSignal, timeout);
|
|
24
|
+
if (response) {
|
|
25
|
+
return parseJsonResponse(response);
|
|
26
|
+
}
|
|
27
|
+
return void 0;
|
|
28
|
+
}
|
|
29
|
+
async function fetchTokenMetadata(chainId, tokenAddress, abortSignal, { timeout = defaultTimeout } = {}) {
|
|
30
|
+
if (!isTokenListSupportedForNetwork(chainId)) {
|
|
31
|
+
throw new Error(TOKEN_METADATA_NO_SUPPORT_ERROR);
|
|
32
|
+
}
|
|
33
|
+
const tokenMetadataURL = getTokenMetadataURL(chainId, tokenAddress);
|
|
34
|
+
const response = await queryApi(tokenMetadataURL, abortSignal, timeout);
|
|
35
|
+
if (response) {
|
|
36
|
+
return parseJsonResponse(response);
|
|
37
|
+
}
|
|
38
|
+
return void 0;
|
|
39
|
+
}
|
|
40
|
+
async function queryApi(apiURL, abortSignal, timeout) {
|
|
41
|
+
const fetchOptions = {
|
|
42
|
+
referrer: apiURL,
|
|
43
|
+
referrerPolicy: "no-referrer-when-downgrade",
|
|
44
|
+
method: "GET",
|
|
45
|
+
mode: "cors",
|
|
46
|
+
signal: abortSignal,
|
|
47
|
+
cache: "default"
|
|
48
|
+
};
|
|
49
|
+
fetchOptions.headers = new window.Headers();
|
|
50
|
+
fetchOptions.headers.set("Content-Type", "application/json");
|
|
51
|
+
try {
|
|
52
|
+
return await timeoutFetch(apiURL, fetchOptions, timeout);
|
|
53
|
+
} catch (error) {
|
|
54
|
+
if (error instanceof Error && error.name === "AbortError") {
|
|
55
|
+
console.log("Request is aborted");
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
return void 0;
|
|
59
|
+
}
|
|
60
|
+
async function parseJsonResponse(apiResponse) {
|
|
61
|
+
const responseObj = await apiResponse.json();
|
|
62
|
+
if (responseObj?.error) {
|
|
63
|
+
throw new Error(`TokenService Error: ${responseObj.error}`);
|
|
64
|
+
}
|
|
65
|
+
return responseObj;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
export {
|
|
69
|
+
TOKEN_END_POINT_API,
|
|
70
|
+
TOKEN_METADATA_NO_SUPPORT_ERROR,
|
|
71
|
+
fetchTokenListByChainId,
|
|
72
|
+
fetchTokenMetadata
|
|
73
|
+
};
|
|
74
|
+
//# sourceMappingURL=chunk-QHRPRO5U.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/token-service.ts"],"sourcesContent":["import { convertHexToDecimal, timeoutFetch } from '@metamask/controller-utils';\nimport type { Hex } from '@metamask/utils';\n\nimport { isTokenListSupportedForNetwork } from './assetsUtil';\n\nexport const TOKEN_END_POINT_API = 'https://token-api.metaswap.codefi.network';\nexport const TOKEN_METADATA_NO_SUPPORT_ERROR =\n 'TokenService Error: Network does not support fetchTokenMetadata';\n\n/**\n * Get the tokens URL for a specific network.\n *\n * @param chainId - The chain ID of the network the tokens requested are on.\n * @returns The tokens URL.\n */\nfunction getTokensURL(chainId: Hex) {\n return `${TOKEN_END_POINT_API}/tokens/${convertHexToDecimal(\n chainId,\n )}?occurrenceFloor=3&includeNativeAssets=false&includeDuplicateSymbolAssets=false&includeTokenFees=false&includeAssetType=false`;\n}\n\n/**\n * Get the token metadata URL for the given network and token.\n *\n * @param chainId - The chain ID of the network the token is on.\n * @param tokenAddress - The token address.\n * @returns The token metadata URL.\n */\nfunction getTokenMetadataURL(chainId: Hex, tokenAddress: string) {\n return `${TOKEN_END_POINT_API}/token/${convertHexToDecimal(\n chainId,\n )}?address=${tokenAddress}`;\n}\n\nconst tenSecondsInMilliseconds = 10_000;\n\n// Token list averages 1.6 MB in size\n// timeoutFetch by default has a 500ms timeout, which will almost always timeout given the response size.\nconst defaultTimeout = tenSecondsInMilliseconds;\n\n/**\n * Fetch the list of token metadata for a given network. This request is cancellable using the\n * abort signal passed in.\n *\n * @param chainId - The chain ID of the network the requested tokens are on.\n * @param abortSignal - The abort signal used to cancel the request if necessary.\n * @param options - Additional fetch options.\n * @param options.timeout - The fetch timeout.\n * @returns The token list, or `undefined` if the request was cancelled.\n */\nexport async function fetchTokenListByChainId(\n chainId: Hex,\n abortSignal: AbortSignal,\n { timeout = defaultTimeout } = {},\n): Promise<unknown> {\n const tokenURL = getTokensURL(chainId);\n const response = await queryApi(tokenURL, abortSignal, timeout);\n if (response) {\n return parseJsonResponse(response);\n }\n return undefined;\n}\n\n/**\n * Fetch metadata for the token address provided for a given network. This request is cancellable\n * using the abort signal passed in.\n *\n * @param chainId - The chain ID of the network the token is on.\n * @param tokenAddress - The address of the token to fetch metadata for.\n * @param abortSignal - The abort signal used to cancel the request if necessary.\n * @param options - Additional fetch options.\n * @param options.timeout - The fetch timeout.\n * @returns The token metadata, or `undefined` if the request was either aborted or failed.\n */\nexport async function fetchTokenMetadata<T>(\n chainId: Hex,\n tokenAddress: string,\n abortSignal: AbortSignal,\n { timeout = defaultTimeout } = {},\n): Promise<T | undefined> {\n if (!isTokenListSupportedForNetwork(chainId)) {\n throw new Error(TOKEN_METADATA_NO_SUPPORT_ERROR);\n }\n const tokenMetadataURL = getTokenMetadataURL(chainId, tokenAddress);\n const response = await queryApi(tokenMetadataURL, abortSignal, timeout);\n if (response) {\n return parseJsonResponse(response) as Promise<T>;\n }\n return undefined;\n}\n\n/**\n * Perform fetch request against the api.\n *\n * @param apiURL - The URL of the API to fetch.\n * @param abortSignal - The abort signal used to cancel the request if necessary.\n * @param timeout - The fetch timeout.\n * @returns Promise resolving request response.\n */\nasync function queryApi(\n apiURL: string,\n abortSignal: AbortSignal,\n timeout: number,\n): Promise<Response | undefined> {\n const fetchOptions: RequestInit = {\n referrer: apiURL,\n referrerPolicy: 'no-referrer-when-downgrade',\n method: 'GET',\n mode: 'cors',\n signal: abortSignal,\n cache: 'default',\n };\n fetchOptions.headers = new window.Headers();\n fetchOptions.headers.set('Content-Type', 'application/json');\n try {\n return await timeoutFetch(apiURL, fetchOptions, timeout);\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n console.log('Request is aborted');\n }\n }\n return undefined;\n}\n\n/**\n * Parse an API response and return the response JSON data.\n *\n * @param apiResponse - The API response to parse.\n * @returns The response JSON data.\n * @throws Will throw if the response includes an error.\n */\nasync function parseJsonResponse(apiResponse: Response): Promise<unknown> {\n const responseObj = await apiResponse.json();\n // api may return errors as json without setting an error http status code\n if (responseObj?.error) {\n throw new Error(`TokenService Error: ${responseObj.error}`);\n }\n return responseObj;\n}\n"],"mappings":";;;;;AAAA,SAAS,qBAAqB,oBAAoB;AAK3C,IAAM,sBAAsB;AAC5B,IAAM,kCACX;AAQF,SAAS,aAAa,SAAc;AAClC,SAAO,GAAG,mBAAmB,WAAW;AAAA,IACtC;AAAA,EACF,CAAC;AACH;AASA,SAAS,oBAAoB,SAAc,cAAsB;AAC/D,SAAO,GAAG,mBAAmB,UAAU;AAAA,IACrC;AAAA,EACF,CAAC,YAAY,YAAY;AAC3B;AAEA,IAAM,2BAA2B;AAIjC,IAAM,iBAAiB;AAYvB,eAAsB,wBACpB,SACA,aACA,EAAE,UAAU,eAAe,IAAI,CAAC,GACd;AAClB,QAAM,WAAW,aAAa,OAAO;AACrC,QAAM,WAAW,MAAM,SAAS,UAAU,aAAa,OAAO;AAC9D,MAAI,UAAU;AACZ,WAAO,kBAAkB,QAAQ;AAAA,EACnC;AACA,SAAO;AACT;AAaA,eAAsB,mBACpB,SACA,cACA,aACA,EAAE,UAAU,eAAe,IAAI,CAAC,GACR;AACxB,MAAI,CAAC,+BAA+B,OAAO,GAAG;AAC5C,UAAM,IAAI,MAAM,+BAA+B;AAAA,EACjD;AACA,QAAM,mBAAmB,oBAAoB,SAAS,YAAY;AAClE,QAAM,WAAW,MAAM,SAAS,kBAAkB,aAAa,OAAO;AACtE,MAAI,UAAU;AACZ,WAAO,kBAAkB,QAAQ;AAAA,EACnC;AACA,SAAO;AACT;AAUA,eAAe,SACb,QACA,aACA,SAC+B;AAC/B,QAAM,eAA4B;AAAA,IAChC,UAAU;AAAA,IACV,gBAAgB;AAAA,IAChB,QAAQ;AAAA,IACR,MAAM;AAAA,IACN,QAAQ;AAAA,IACR,OAAO;AAAA,EACT;AACA,eAAa,UAAU,IAAI,OAAO,QAAQ;AAC1C,eAAa,QAAQ,IAAI,gBAAgB,kBAAkB;AAC3D,MAAI;AACF,WAAO,MAAM,aAAa,QAAQ,cAAc,OAAO;AAAA,EACzD,SAAS,OAAO;AACd,QAAI,iBAAiB,SAAS,MAAM,SAAS,cAAc;AACzD,cAAQ,IAAI,oBAAoB;AAAA,IAClC;AAAA,EACF;AACA,SAAO;AACT;AASA,eAAe,kBAAkB,aAAyC;AACxE,QAAM,cAAc,MAAM,YAAY,KAAK;AAE3C,MAAI,aAAa,OAAO;AACtB,UAAM,IAAI,MAAM,uBAAuB,YAAY,KAAK,EAAE;AAAA,EAC5D;AACA,SAAO;AACT;","names":[]}
|