@metamask-previews/assets-controllers 89.0.1-preview-152da47f → 89.0.1-preview-21a5ddac
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 +0 -8
- package/dist/CurrencyRateController.cjs +3 -3
- package/dist/CurrencyRateController.cjs.map +1 -1
- package/dist/CurrencyRateController.d.cts.map +1 -1
- package/dist/CurrencyRateController.d.mts.map +1 -1
- package/dist/CurrencyRateController.mjs +3 -3
- package/dist/CurrencyRateController.mjs.map +1 -1
- package/dist/TokenRatesController.cjs +278 -82
- package/dist/TokenRatesController.cjs.map +1 -1
- package/dist/TokenRatesController.d.cts +35 -9
- package/dist/TokenRatesController.d.cts.map +1 -1
- package/dist/TokenRatesController.d.mts +35 -9
- package/dist/TokenRatesController.d.mts.map +1 -1
- package/dist/TokenRatesController.mjs +279 -83
- package/dist/TokenRatesController.mjs.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/TokenSearchDiscoveryDataController.cjs +3 -2
- package/dist/TokenSearchDiscoveryDataController/TokenSearchDiscoveryDataController.cjs.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/TokenSearchDiscoveryDataController.d.cts.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/TokenSearchDiscoveryDataController.d.mts.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/TokenSearchDiscoveryDataController.mjs +3 -2
- package/dist/TokenSearchDiscoveryDataController/TokenSearchDiscoveryDataController.mjs.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/types.cjs.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/types.d.cts +2 -2
- package/dist/TokenSearchDiscoveryDataController/types.d.cts.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/types.d.mts +2 -2
- package/dist/TokenSearchDiscoveryDataController/types.d.mts.map +1 -1
- package/dist/TokenSearchDiscoveryDataController/types.mjs.map +1 -1
- package/dist/assetsUtil.cjs +5 -11
- package/dist/assetsUtil.cjs.map +1 -1
- package/dist/assetsUtil.d.cts +2 -2
- package/dist/assetsUtil.d.cts.map +1 -1
- package/dist/assetsUtil.d.mts +2 -2
- package/dist/assetsUtil.d.mts.map +1 -1
- package/dist/assetsUtil.mjs +5 -11
- package/dist/assetsUtil.mjs.map +1 -1
- package/dist/token-prices-service/abstract-token-prices-service.cjs.map +1 -1
- package/dist/token-prices-service/abstract-token-prices-service.d.cts +43 -17
- package/dist/token-prices-service/abstract-token-prices-service.d.cts.map +1 -1
- package/dist/token-prices-service/abstract-token-prices-service.d.mts +43 -17
- package/dist/token-prices-service/abstract-token-prices-service.d.mts.map +1 -1
- package/dist/token-prices-service/abstract-token-prices-service.mjs.map +1 -1
- package/dist/token-prices-service/codefi-v2.cjs +108 -175
- package/dist/token-prices-service/codefi-v2.cjs.map +1 -1
- package/dist/token-prices-service/codefi-v2.d.cts +10 -45
- package/dist/token-prices-service/codefi-v2.d.cts.map +1 -1
- package/dist/token-prices-service/codefi-v2.d.mts +10 -45
- package/dist/token-prices-service/codefi-v2.d.mts.map +1 -1
- package/dist/token-prices-service/codefi-v2.mjs +108 -175
- package/dist/token-prices-service/codefi-v2.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -10,15 +10,21 @@ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (
|
|
|
10
10
|
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
|
11
11
|
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
12
12
|
};
|
|
13
|
-
var _TokenRatesController_instances, _TokenRatesController_tokenPricesService, _TokenRatesController_disabled, _TokenRatesController_allTokens, _TokenRatesController_allDetectedTokens, _TokenRatesController_subscribeToTokensStateChange, _TokenRatesController_subscribeToNetworkStateChange, _TokenRatesController_getTokenAddresses, _TokenRatesController_getTokensControllerState, _TokenRatesController_fetchAndMapExchangeRatesForSupportedNativeCurrency, _TokenRatesController_fetchAndMapExchangeRatesForUnsupportedNativeCurrency;
|
|
13
|
+
var _TokenRatesController_instances, _TokenRatesController_handle, _TokenRatesController_pollState, _TokenRatesController_tokenPricesService, _TokenRatesController_inProcessExchangeRateUpdates, _TokenRatesController_disabled, _TokenRatesController_interval, _TokenRatesController_allTokens, _TokenRatesController_allDetectedTokens, _TokenRatesController_subscribeToTokensStateChange, _TokenRatesController_subscribeToNetworkStateChange, _TokenRatesController_getTokenAddresses, _TokenRatesController_getTokensControllerState, _TokenRatesController_stopPoll, _TokenRatesController_poll, _TokenRatesController_fetchAndMapExchangeRates, _TokenRatesController_fetchAndMapExchangeRatesForSupportedNativeCurrency, _TokenRatesController_fetchAndMapExchangeRatesForUnsupportedNativeCurrency;
|
|
14
14
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
15
|
exports.TokenRatesController = exports.getDefaultTokenRatesControllerState = exports.controllerName = void 0;
|
|
16
16
|
const controller_utils_1 = require("@metamask/controller-utils");
|
|
17
17
|
const polling_controller_1 = require("@metamask/polling-controller");
|
|
18
|
+
const utils_1 = require("@metamask/utils");
|
|
18
19
|
const lodash_1 = require("lodash");
|
|
19
20
|
const assetsUtil_1 = require("./assetsUtil.cjs");
|
|
20
21
|
const codefi_v2_1 = require("./token-prices-service/codefi-v2.cjs");
|
|
21
22
|
const DEFAULT_INTERVAL = 180000;
|
|
23
|
+
var PollState;
|
|
24
|
+
(function (PollState) {
|
|
25
|
+
PollState["Active"] = "Active";
|
|
26
|
+
PollState["Inactive"] = "Inactive";
|
|
27
|
+
})(PollState || (PollState = {}));
|
|
22
28
|
/**
|
|
23
29
|
* The name of the {@link TokenRatesController}.
|
|
24
30
|
*/
|
|
@@ -65,13 +71,18 @@ class TokenRatesController extends (0, polling_controller_1.StaticIntervalPollin
|
|
|
65
71
|
metadata: tokenRatesControllerMetadata,
|
|
66
72
|
});
|
|
67
73
|
_TokenRatesController_instances.add(this);
|
|
74
|
+
_TokenRatesController_handle.set(this, void 0);
|
|
75
|
+
_TokenRatesController_pollState.set(this, PollState.Inactive);
|
|
68
76
|
_TokenRatesController_tokenPricesService.set(this, void 0);
|
|
77
|
+
_TokenRatesController_inProcessExchangeRateUpdates.set(this, {});
|
|
69
78
|
_TokenRatesController_disabled.set(this, void 0);
|
|
79
|
+
_TokenRatesController_interval.set(this, void 0);
|
|
70
80
|
_TokenRatesController_allTokens.set(this, void 0);
|
|
71
81
|
_TokenRatesController_allDetectedTokens.set(this, void 0);
|
|
72
82
|
this.setIntervalLength(interval);
|
|
73
83
|
__classPrivateFieldSet(this, _TokenRatesController_tokenPricesService, tokenPricesService, "f");
|
|
74
84
|
__classPrivateFieldSet(this, _TokenRatesController_disabled, disabled, "f");
|
|
85
|
+
__classPrivateFieldSet(this, _TokenRatesController_interval, interval, "f");
|
|
75
86
|
const { allTokens, allDetectedTokens } = __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_getTokensControllerState).call(this);
|
|
76
87
|
__classPrivateFieldSet(this, _TokenRatesController_allTokens, allTokens, "f");
|
|
77
88
|
__classPrivateFieldSet(this, _TokenRatesController_allDetectedTokens, allDetectedTokens, "f");
|
|
@@ -90,52 +101,99 @@ class TokenRatesController extends (0, polling_controller_1.StaticIntervalPollin
|
|
|
90
101
|
disable() {
|
|
91
102
|
__classPrivateFieldSet(this, _TokenRatesController_disabled, true, "f");
|
|
92
103
|
}
|
|
104
|
+
/**
|
|
105
|
+
* Start (or restart) polling.
|
|
106
|
+
*
|
|
107
|
+
* @param chainId - The chain ID.
|
|
108
|
+
* @param nativeCurrency - The native currency.
|
|
109
|
+
*/
|
|
110
|
+
async start(chainId, nativeCurrency) {
|
|
111
|
+
__classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_stopPoll).call(this);
|
|
112
|
+
__classPrivateFieldSet(this, _TokenRatesController_pollState, PollState.Active, "f");
|
|
113
|
+
await __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_poll).call(this, chainId, nativeCurrency);
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Stop polling.
|
|
117
|
+
*/
|
|
118
|
+
stop() {
|
|
119
|
+
__classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_stopPoll).call(this);
|
|
120
|
+
__classPrivateFieldSet(this, _TokenRatesController_pollState, PollState.Inactive, "f");
|
|
121
|
+
}
|
|
93
122
|
/**
|
|
94
123
|
* Updates exchange rates for all tokens.
|
|
95
124
|
*
|
|
96
125
|
* @param chainIdAndNativeCurrency - The chain ID and native currency.
|
|
97
126
|
*/
|
|
98
127
|
async updateExchangeRates(chainIdAndNativeCurrency) {
|
|
128
|
+
await this.updateExchangeRatesByChainId(chainIdAndNativeCurrency);
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Updates exchange rates for all tokens.
|
|
132
|
+
*
|
|
133
|
+
* @param chainIds - The chain IDs.
|
|
134
|
+
* @returns A promise that resolves when all chain updates complete.
|
|
135
|
+
*/
|
|
136
|
+
/**
|
|
137
|
+
* Updates exchange rates for all tokens.
|
|
138
|
+
*
|
|
139
|
+
* @param chainIdAndNativeCurrency - The chain ID and native currency.
|
|
140
|
+
*/
|
|
141
|
+
async updateExchangeRatesByChainId(chainIdAndNativeCurrency) {
|
|
99
142
|
if (__classPrivateFieldGet(this, _TokenRatesController_disabled, "f")) {
|
|
100
143
|
return;
|
|
101
144
|
}
|
|
102
|
-
|
|
103
|
-
const
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
tokenAddress,
|
|
112
|
-
});
|
|
113
|
-
}
|
|
114
|
-
else {
|
|
115
|
-
(unsupportedAssetsByNativeCurrency[nativeCurrency] ?? (unsupportedAssetsByNativeCurrency[nativeCurrency] = [])).push({
|
|
116
|
-
chainId,
|
|
117
|
-
tokenAddress,
|
|
118
|
-
});
|
|
119
|
-
}
|
|
120
|
-
}
|
|
145
|
+
// Create a promise for each chainId to fetch exchange rates.
|
|
146
|
+
const updatePromises = chainIdAndNativeCurrency.map(async ({ chainId, nativeCurrency }) => {
|
|
147
|
+
const tokenAddresses = __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_getTokenAddresses).call(this, chainId);
|
|
148
|
+
// Build a unique key based on chainId, nativeCurrency, and the number of token addresses.
|
|
149
|
+
const updateKey = `${chainId}:${nativeCurrency}:${tokenAddresses.length}`;
|
|
150
|
+
if (updateKey in __classPrivateFieldGet(this, _TokenRatesController_inProcessExchangeRateUpdates, "f")) {
|
|
151
|
+
// Await any ongoing update to avoid redundant work.
|
|
152
|
+
await __classPrivateFieldGet(this, _TokenRatesController_inProcessExchangeRateUpdates, "f")[updateKey];
|
|
153
|
+
return null;
|
|
121
154
|
}
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
155
|
+
// Create a deferred promise to track this update.
|
|
156
|
+
const { promise: inProgressUpdate, resolve: updateSucceeded, reject: updateFailed, } = (0, utils_1.createDeferredPromise)({ suppressUnhandledRejection: true });
|
|
157
|
+
__classPrivateFieldGet(this, _TokenRatesController_inProcessExchangeRateUpdates, "f")[updateKey] = inProgressUpdate;
|
|
158
|
+
try {
|
|
159
|
+
const contractInformations = await __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_fetchAndMapExchangeRates).call(this, {
|
|
160
|
+
tokenAddresses,
|
|
161
|
+
chainId,
|
|
162
|
+
nativeCurrency,
|
|
163
|
+
});
|
|
164
|
+
// Each promise returns an object with the market data for the chain.
|
|
165
|
+
const marketData = {
|
|
166
|
+
[chainId]: {
|
|
167
|
+
...(contractInformations ?? {}),
|
|
168
|
+
},
|
|
169
|
+
};
|
|
170
|
+
updateSucceeded();
|
|
171
|
+
return marketData;
|
|
132
172
|
}
|
|
133
|
-
|
|
134
|
-
|
|
173
|
+
catch (error) {
|
|
174
|
+
updateFailed(error);
|
|
175
|
+
throw error;
|
|
176
|
+
}
|
|
177
|
+
finally {
|
|
178
|
+
// Cleanup the tracking for this update.
|
|
179
|
+
delete __classPrivateFieldGet(this, _TokenRatesController_inProcessExchangeRateUpdates, "f")[updateKey];
|
|
180
|
+
}
|
|
181
|
+
});
|
|
182
|
+
// Wait for all update promises to settle.
|
|
183
|
+
const results = await Promise.allSettled(updatePromises);
|
|
184
|
+
// Merge all successful market data updates into one object.
|
|
185
|
+
const combinedMarketData = results.reduce((acc, result) => {
|
|
186
|
+
if (result.status === 'fulfilled' && result.value) {
|
|
187
|
+
acc = { ...acc, ...result.value };
|
|
188
|
+
}
|
|
189
|
+
return acc;
|
|
190
|
+
}, {});
|
|
191
|
+
// Call this.update only once with the combined market data to reduce the number of state changes and re-renders
|
|
192
|
+
if (Object.keys(combinedMarketData).length > 0) {
|
|
135
193
|
this.update((state) => {
|
|
136
194
|
state.marketData = {
|
|
137
195
|
...state.marketData,
|
|
138
|
-
...
|
|
196
|
+
...combinedMarketData,
|
|
139
197
|
};
|
|
140
198
|
});
|
|
141
199
|
}
|
|
@@ -160,7 +218,7 @@ class TokenRatesController extends (0, polling_controller_1.StaticIntervalPollin
|
|
|
160
218
|
});
|
|
161
219
|
return acc;
|
|
162
220
|
}, []);
|
|
163
|
-
await this.
|
|
221
|
+
await this.updateExchangeRatesByChainId(chainIdAndNativeCurrency);
|
|
164
222
|
}
|
|
165
223
|
/**
|
|
166
224
|
* Reset the controller state to the default state.
|
|
@@ -172,7 +230,7 @@ class TokenRatesController extends (0, polling_controller_1.StaticIntervalPollin
|
|
|
172
230
|
}
|
|
173
231
|
}
|
|
174
232
|
exports.TokenRatesController = TokenRatesController;
|
|
175
|
-
_TokenRatesController_tokenPricesService = new WeakMap(), _TokenRatesController_disabled = new WeakMap(), _TokenRatesController_allTokens = new WeakMap(), _TokenRatesController_allDetectedTokens = new WeakMap(), _TokenRatesController_instances = new WeakSet(), _TokenRatesController_subscribeToTokensStateChange = function _TokenRatesController_subscribeToTokensStateChange() {
|
|
233
|
+
_TokenRatesController_handle = new WeakMap(), _TokenRatesController_pollState = new WeakMap(), _TokenRatesController_tokenPricesService = new WeakMap(), _TokenRatesController_inProcessExchangeRateUpdates = new WeakMap(), _TokenRatesController_disabled = new WeakMap(), _TokenRatesController_interval = new WeakMap(), _TokenRatesController_allTokens = new WeakMap(), _TokenRatesController_allDetectedTokens = new WeakMap(), _TokenRatesController_instances = new WeakSet(), _TokenRatesController_subscribeToTokensStateChange = function _TokenRatesController_subscribeToTokensStateChange() {
|
|
176
234
|
this.messenger.subscribe('TokensController:stateChange',
|
|
177
235
|
// TODO: Either fix this lint violation or explain why it's necessary to ignore.
|
|
178
236
|
// eslint-disable-next-line @typescript-eslint/no-misused-promises
|
|
@@ -203,12 +261,24 @@ _TokenRatesController_tokenPricesService = new WeakMap(), _TokenRatesController_
|
|
|
203
261
|
});
|
|
204
262
|
return acc;
|
|
205
263
|
}, []);
|
|
206
|
-
await this.
|
|
264
|
+
await this.updateExchangeRatesByChainId(chainIdAndNativeCurrency);
|
|
207
265
|
}, ({ allTokens, allDetectedTokens }) => {
|
|
208
266
|
return { allTokens, allDetectedTokens };
|
|
209
267
|
});
|
|
210
268
|
}, _TokenRatesController_subscribeToNetworkStateChange = function _TokenRatesController_subscribeToNetworkStateChange() {
|
|
211
|
-
this.messenger.subscribe('NetworkController:stateChange',
|
|
269
|
+
this.messenger.subscribe('NetworkController:stateChange',
|
|
270
|
+
// TODO: Either fix this lint violation or explain why it's necessary to ignore.
|
|
271
|
+
// eslint-disable-next-line @typescript-eslint/no-misused-promises
|
|
272
|
+
async ({ networkConfigurationsByChainId }, patches) => {
|
|
273
|
+
const chainIdAndNativeCurrency = Object.values(networkConfigurationsByChainId).map(({ chainId, nativeCurrency }) => {
|
|
274
|
+
return {
|
|
275
|
+
chainId: chainId,
|
|
276
|
+
nativeCurrency,
|
|
277
|
+
};
|
|
278
|
+
});
|
|
279
|
+
if (__classPrivateFieldGet(this, _TokenRatesController_pollState, "f") === PollState.Active) {
|
|
280
|
+
await this.updateExchangeRates(chainIdAndNativeCurrency);
|
|
281
|
+
}
|
|
212
282
|
// Remove state for deleted networks
|
|
213
283
|
for (const patch of patches) {
|
|
214
284
|
if (patch.op === 'remove' &&
|
|
@@ -224,63 +294,189 @@ _TokenRatesController_tokenPricesService = new WeakMap(), _TokenRatesController_
|
|
|
224
294
|
const getTokens = (allTokens) => Object.values(allTokens ?? {}).flatMap((tokens) => tokens.map(({ address }) => (0, controller_utils_1.toChecksumHexAddress)(address)));
|
|
225
295
|
const tokenAddresses = getTokens(__classPrivateFieldGet(this, _TokenRatesController_allTokens, "f")[chainId]);
|
|
226
296
|
const detectedTokenAddresses = getTokens(__classPrivateFieldGet(this, _TokenRatesController_allDetectedTokens, "f")[chainId]);
|
|
227
|
-
return [
|
|
228
|
-
...new Set([
|
|
229
|
-
...tokenAddresses,
|
|
230
|
-
...detectedTokenAddresses,
|
|
231
|
-
(0, codefi_v2_1.getNativeTokenAddress)(chainId),
|
|
232
|
-
]),
|
|
233
|
-
].sort();
|
|
297
|
+
return [...new Set([...tokenAddresses, ...detectedTokenAddresses])].sort();
|
|
234
298
|
}, _TokenRatesController_getTokensControllerState = function _TokenRatesController_getTokensControllerState() {
|
|
235
299
|
const { allTokens, allDetectedTokens } = this.messenger.call('TokensController:getState');
|
|
236
300
|
return {
|
|
237
301
|
allTokens,
|
|
238
302
|
allDetectedTokens,
|
|
239
303
|
};
|
|
240
|
-
},
|
|
241
|
-
|
|
242
|
-
|
|
304
|
+
}, _TokenRatesController_stopPoll = function _TokenRatesController_stopPoll() {
|
|
305
|
+
if (__classPrivateFieldGet(this, _TokenRatesController_handle, "f")) {
|
|
306
|
+
clearTimeout(__classPrivateFieldGet(this, _TokenRatesController_handle, "f"));
|
|
307
|
+
}
|
|
308
|
+
}, _TokenRatesController_poll =
|
|
309
|
+
/**
|
|
310
|
+
* Poll for exchange rate updates.
|
|
311
|
+
*
|
|
312
|
+
* @param chainId - The chain ID.
|
|
313
|
+
* @param nativeCurrency - The native currency.
|
|
314
|
+
*/
|
|
315
|
+
async function _TokenRatesController_poll(chainId, nativeCurrency) {
|
|
316
|
+
await (0, controller_utils_1.safelyExecute)(() => this.updateExchangeRates([{ chainId, nativeCurrency }]));
|
|
317
|
+
// Poll using recursive `setTimeout` instead of `setInterval` so that
|
|
318
|
+
// requests don't stack if they take longer than the polling interval
|
|
319
|
+
__classPrivateFieldSet(this, _TokenRatesController_handle, setTimeout(() => {
|
|
320
|
+
// TODO: Either fix this lint violation or explain why it's necessary to ignore.
|
|
321
|
+
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
|
322
|
+
__classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_poll).call(this, chainId, nativeCurrency);
|
|
323
|
+
}, __classPrivateFieldGet(this, _TokenRatesController_interval, "f")), "f");
|
|
324
|
+
}, _TokenRatesController_fetchAndMapExchangeRates =
|
|
325
|
+
/**
|
|
326
|
+
* Uses the token prices service to retrieve exchange rates for tokens in a
|
|
327
|
+
* particular currency.
|
|
328
|
+
*
|
|
329
|
+
* If the price API does not support the given chain ID, returns an empty
|
|
330
|
+
* object.
|
|
331
|
+
*
|
|
332
|
+
* If the price API does not support the given currency, retrieves exchange
|
|
333
|
+
* rates in a known currency instead, then converts those rates using the
|
|
334
|
+
* exchange rate between the known currency and desired currency.
|
|
335
|
+
*
|
|
336
|
+
* @param args - The arguments to this function.
|
|
337
|
+
* @param args.tokenAddresses - Addresses for tokens.
|
|
338
|
+
* @param args.chainId - The EIP-155 ID of the chain where the tokens live.
|
|
339
|
+
* @param args.nativeCurrency - The native currency in which to request
|
|
340
|
+
* exchange rates.
|
|
341
|
+
* @returns A map from token address to its exchange rate in the native
|
|
342
|
+
* currency, or an empty map if no exchange rates can be obtained for the
|
|
343
|
+
* chain ID.
|
|
344
|
+
*/
|
|
345
|
+
async function _TokenRatesController_fetchAndMapExchangeRates({ tokenAddresses, chainId, nativeCurrency, }) {
|
|
346
|
+
if (!__classPrivateFieldGet(this, _TokenRatesController_tokenPricesService, "f").validateChainIdSupported(chainId)) {
|
|
347
|
+
return tokenAddresses.reduce((obj, tokenAddress) => {
|
|
348
|
+
obj = {
|
|
349
|
+
...obj,
|
|
350
|
+
[tokenAddress]: undefined,
|
|
351
|
+
};
|
|
352
|
+
return obj;
|
|
353
|
+
}, {});
|
|
354
|
+
}
|
|
355
|
+
if (__classPrivateFieldGet(this, _TokenRatesController_tokenPricesService, "f").validateCurrencySupported(nativeCurrency)) {
|
|
356
|
+
return await __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_fetchAndMapExchangeRatesForSupportedNativeCurrency).call(this, {
|
|
357
|
+
tokenAddresses,
|
|
358
|
+
chainId,
|
|
359
|
+
nativeCurrency,
|
|
360
|
+
});
|
|
361
|
+
}
|
|
362
|
+
return await __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_fetchAndMapExchangeRatesForUnsupportedNativeCurrency).call(this, {
|
|
363
|
+
chainId,
|
|
364
|
+
tokenAddresses,
|
|
365
|
+
nativeCurrency,
|
|
366
|
+
});
|
|
367
|
+
}, _TokenRatesController_fetchAndMapExchangeRatesForSupportedNativeCurrency =
|
|
368
|
+
/**
|
|
369
|
+
* Retrieves prices in the given currency for the given tokens on the given
|
|
370
|
+
* chain. Ensures that token addresses are checksum addresses.
|
|
371
|
+
*
|
|
372
|
+
* @param args - The arguments to this function.
|
|
373
|
+
* @param args.tokenAddresses - Addresses for tokens.
|
|
374
|
+
* @param args.chainId - The EIP-155 ID of the chain where the tokens live.
|
|
375
|
+
* @param args.nativeCurrency - The native currency in which to request
|
|
376
|
+
* prices.
|
|
377
|
+
* @returns A map of the token addresses (as checksums) to their prices in the
|
|
378
|
+
* native currency.
|
|
379
|
+
*/
|
|
380
|
+
async function _TokenRatesController_fetchAndMapExchangeRatesForSupportedNativeCurrency({ tokenAddresses, chainId, nativeCurrency, }) {
|
|
381
|
+
let contractNativeInformations;
|
|
382
|
+
const tokenPricesByTokenAddress = await (0, assetsUtil_1.reduceInBatchesSerially)({
|
|
383
|
+
values: [...tokenAddresses].sort(),
|
|
243
384
|
batchSize: assetsUtil_1.TOKEN_PRICES_BATCH_SIZE,
|
|
244
|
-
eachBatch: async (
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
currency,
|
|
385
|
+
eachBatch: async (allTokenPricesByTokenAddress, batch) => {
|
|
386
|
+
const tokenPricesByTokenAddressForBatch = await __classPrivateFieldGet(this, _TokenRatesController_tokenPricesService, "f").fetchTokenPrices({
|
|
387
|
+
tokenAddresses: batch,
|
|
388
|
+
chainId,
|
|
389
|
+
currency: nativeCurrency,
|
|
249
390
|
});
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
391
|
+
return {
|
|
392
|
+
...allTokenPricesByTokenAddress,
|
|
393
|
+
...tokenPricesByTokenAddressForBatch,
|
|
394
|
+
};
|
|
254
395
|
},
|
|
255
|
-
initialResult:
|
|
396
|
+
initialResult: {},
|
|
397
|
+
});
|
|
398
|
+
contractNativeInformations = tokenPricesByTokenAddress;
|
|
399
|
+
// fetch for native token
|
|
400
|
+
if (tokenAddresses.length === 0) {
|
|
401
|
+
const contractNativeInformationsNative = await __classPrivateFieldGet(this, _TokenRatesController_tokenPricesService, "f").fetchTokenPrices({
|
|
402
|
+
tokenAddresses: [],
|
|
403
|
+
chainId,
|
|
404
|
+
currency: nativeCurrency,
|
|
405
|
+
});
|
|
406
|
+
contractNativeInformations = {
|
|
407
|
+
[(0, codefi_v2_1.getNativeTokenAddress)(chainId)]: {
|
|
408
|
+
currency: nativeCurrency,
|
|
409
|
+
...contractNativeInformationsNative[(0, codefi_v2_1.getNativeTokenAddress)(chainId)],
|
|
410
|
+
},
|
|
411
|
+
};
|
|
412
|
+
}
|
|
413
|
+
return Object.entries(contractNativeInformations).reduce((obj, [tokenAddress, token]) => {
|
|
414
|
+
obj = {
|
|
415
|
+
...obj,
|
|
416
|
+
[tokenAddress]: { ...token },
|
|
417
|
+
};
|
|
418
|
+
return obj;
|
|
419
|
+
}, {});
|
|
420
|
+
}, _TokenRatesController_fetchAndMapExchangeRatesForUnsupportedNativeCurrency =
|
|
421
|
+
/**
|
|
422
|
+
* If the price API does not support a given native currency, then we need to
|
|
423
|
+
* convert it to a fallback currency and feed that currency into the price
|
|
424
|
+
* API, then convert the prices to our desired native currency.
|
|
425
|
+
*
|
|
426
|
+
* @param args - The arguments to this function.
|
|
427
|
+
* @param args.chainId - The chain id to fetch prices for.
|
|
428
|
+
* @param args.tokenAddresses - Addresses for tokens.
|
|
429
|
+
* @param args.nativeCurrency - The native currency in which to request
|
|
430
|
+
* prices.
|
|
431
|
+
* @returns A map of the token addresses (as checksums) to their prices in the
|
|
432
|
+
* native currency.
|
|
433
|
+
*/
|
|
434
|
+
async function _TokenRatesController_fetchAndMapExchangeRatesForUnsupportedNativeCurrency({ chainId, tokenAddresses, nativeCurrency, }) {
|
|
435
|
+
const nativeTokenAddress = (0, codefi_v2_1.getNativeTokenAddress)(chainId);
|
|
436
|
+
// Step -1: First fetch native token priced in USD
|
|
437
|
+
const nativeTokenPriceMap = await __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_fetchAndMapExchangeRatesForSupportedNativeCurrency).call(this, {
|
|
438
|
+
tokenAddresses: [], // special-case: returns only native token
|
|
439
|
+
chainId,
|
|
440
|
+
nativeCurrency: 'usd',
|
|
256
441
|
});
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
442
|
+
// Step -2: Then fetch all tracked tokens priced in USD
|
|
443
|
+
const tokenPricesInUSD = await __classPrivateFieldGet(this, _TokenRatesController_instances, "m", _TokenRatesController_fetchAndMapExchangeRatesForSupportedNativeCurrency).call(this, {
|
|
444
|
+
tokenAddresses,
|
|
445
|
+
chainId,
|
|
446
|
+
nativeCurrency: 'usd',
|
|
447
|
+
});
|
|
448
|
+
const nativeTokenInfo = nativeTokenPriceMap[nativeTokenAddress];
|
|
449
|
+
const nativeTokenPriceInUSD = nativeTokenInfo?.price;
|
|
450
|
+
if (!nativeTokenPriceInUSD || nativeTokenPriceInUSD === 0) {
|
|
451
|
+
// If we can't price the native token in the fallback currency,
|
|
452
|
+
// we can't safely convert; return empty so callers know there is no data.
|
|
453
|
+
return {};
|
|
454
|
+
}
|
|
455
|
+
// Step -3: Convert USD prices to native currency
|
|
260
456
|
// Formula: price_in_native = token_usd / native_usd
|
|
261
|
-
const convertUSDToNative = (valueInUSD
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
for (const [tokenAddress, tokenData] of Object.entries(marketDataByTokenAddress)) {
|
|
270
|
-
(marketData[chainId] ?? (marketData[chainId] = {}))[tokenAddress] = {
|
|
457
|
+
const convertUSDToNative = (valueInUSD) => valueInUSD !== undefined && valueInUSD !== null
|
|
458
|
+
? valueInUSD / nativeTokenPriceInUSD
|
|
459
|
+
: undefined;
|
|
460
|
+
// Step -4 & -5: Apply conversion to all token fields and return
|
|
461
|
+
const tokenPricesInNative = Object.entries(tokenPricesInUSD).reduce((acc, [tokenAddress, tokenData]) => {
|
|
462
|
+
acc = {
|
|
463
|
+
...acc,
|
|
464
|
+
[tokenAddress]: {
|
|
271
465
|
...tokenData,
|
|
272
|
-
currency,
|
|
273
|
-
price: convertUSDToNative(tokenData.price
|
|
274
|
-
marketCap: convertUSDToNative(tokenData.marketCap
|
|
275
|
-
allTimeHigh: convertUSDToNative(tokenData.allTimeHigh
|
|
276
|
-
allTimeLow: convertUSDToNative(tokenData.allTimeLow
|
|
277
|
-
totalVolume: convertUSDToNative(tokenData.totalVolume
|
|
278
|
-
high1d: convertUSDToNative(tokenData.high1d
|
|
279
|
-
low1d: convertUSDToNative(tokenData.low1d
|
|
280
|
-
dilutedMarketCap: convertUSDToNative(tokenData.dilutedMarketCap
|
|
281
|
-
}
|
|
282
|
-
}
|
|
283
|
-
|
|
466
|
+
currency: nativeCurrency,
|
|
467
|
+
price: convertUSDToNative(tokenData.price),
|
|
468
|
+
marketCap: convertUSDToNative(tokenData.marketCap),
|
|
469
|
+
allTimeHigh: convertUSDToNative(tokenData.allTimeHigh),
|
|
470
|
+
allTimeLow: convertUSDToNative(tokenData.allTimeLow),
|
|
471
|
+
totalVolume: convertUSDToNative(tokenData.totalVolume),
|
|
472
|
+
high1d: convertUSDToNative(tokenData.high1d),
|
|
473
|
+
low1d: convertUSDToNative(tokenData.low1d),
|
|
474
|
+
dilutedMarketCap: convertUSDToNative(tokenData.dilutedMarketCap),
|
|
475
|
+
},
|
|
476
|
+
};
|
|
477
|
+
return acc;
|
|
478
|
+
}, {});
|
|
479
|
+
return tokenPricesInNative;
|
|
284
480
|
};
|
|
285
481
|
exports.default = TokenRatesController;
|
|
286
482
|
//# sourceMappingURL=TokenRatesController.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"TokenRatesController.cjs","sourceRoot":"","sources":["../src/TokenRatesController.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n StateMetadata,\n} from '@metamask/base-controller';\nimport { toChecksumHexAddress } from '@metamask/controller-utils';\nimport type { Messenger } from '@metamask/messenger';\nimport type {\n NetworkControllerGetStateAction,\n NetworkControllerStateChangeEvent,\n} from '@metamask/network-controller';\nimport { StaticIntervalPollingController } from '@metamask/polling-controller';\nimport type { Hex } from '@metamask/utils';\nimport { isEqual } from 'lodash';\n\nimport { reduceInBatchesSerially, TOKEN_PRICES_BATCH_SIZE } from './assetsUtil';\nimport type { AbstractTokenPricesService } from './token-prices-service/abstract-token-prices-service';\nimport { getNativeTokenAddress } from './token-prices-service/codefi-v2';\nimport type {\n TokensControllerGetStateAction,\n TokensControllerStateChangeEvent,\n TokensControllerState,\n} from './TokensController';\n\n/**\n * @type Token\n *\n * Token representation\n *\n * @property address - Hex address of the token contract\n * @property decimals - Number of decimals the token uses\n * @property symbol - Symbol of the token\n * @property aggregators - An array containing the token's aggregators\n * @property image - Image of the token, url or bit32 image\n * @property hasBalanceError - 'true' if there is an error while updating the token balance\n * @property isERC721 - 'true' if the token is a ERC721 token\n * @property name - Name of the token\n */\nexport type Token = {\n address: string;\n decimals: number;\n symbol: string;\n aggregators?: string[];\n image?: string;\n hasBalanceError?: boolean;\n isERC721?: boolean;\n name?: string;\n};\n\nconst DEFAULT_INTERVAL = 180000;\n\nexport type ContractExchangeRates = {\n [address: string]: number | undefined;\n};\n\nexport type MarketDataDetails = {\n tokenAddress: `0x${string}`;\n currency: string;\n allTimeHigh: number;\n allTimeLow: number;\n circulatingSupply: number;\n dilutedMarketCap: number;\n high1d: number;\n low1d: number;\n marketCap: number;\n marketCapPercentChange1d: number;\n price: number;\n priceChange1d: number;\n pricePercentChange1d: number;\n pricePercentChange1h: number;\n pricePercentChange1y: number;\n pricePercentChange7d: number;\n pricePercentChange14d: number;\n pricePercentChange30d: number;\n pricePercentChange200d: number;\n totalVolume: number;\n};\n\n/**\n * Represents a mapping of token contract addresses to their market data.\n */\nexport type ContractMarketData = Record<Hex, MarketDataDetails>;\n\ntype ChainIdAndNativeCurrency = {\n chainId: Hex;\n nativeCurrency: string;\n};\n\n/**\n * The external actions available to the {@link TokenRatesController}.\n */\nexport type AllowedActions =\n | TokensControllerGetStateAction\n | NetworkControllerGetStateAction;\n\n/**\n * The external events available to the {@link TokenRatesController}.\n */\nexport type AllowedEvents =\n | TokensControllerStateChangeEvent\n | NetworkControllerStateChangeEvent;\n\n/**\n * The name of the {@link TokenRatesController}.\n */\nexport const controllerName = 'TokenRatesController';\n\n/**\n * @type TokenRatesState\n *\n * Token rates controller state\n *\n * @property marketData - Market data for tokens, keyed by chain ID and then token contract address.\n */\nexport type TokenRatesControllerState = {\n marketData: Record<Hex, Record<Hex, MarketDataDetails>>;\n};\n\n/**\n * The action that can be performed to get the state of the {@link TokenRatesController}.\n */\nexport type TokenRatesControllerGetStateAction = ControllerGetStateAction<\n typeof controllerName,\n TokenRatesControllerState\n>;\n\n/**\n * The actions that can be performed using the {@link TokenRatesController}.\n */\nexport type TokenRatesControllerActions = TokenRatesControllerGetStateAction;\n\n/**\n * The event that {@link TokenRatesController} can emit.\n */\nexport type TokenRatesControllerStateChangeEvent = ControllerStateChangeEvent<\n typeof controllerName,\n TokenRatesControllerState\n>;\n\n/**\n * The events that {@link TokenRatesController} can emit.\n */\nexport type TokenRatesControllerEvents = TokenRatesControllerStateChangeEvent;\n\n/**\n * The messenger of the {@link TokenRatesController} for communication.\n */\nexport type TokenRatesControllerMessenger = Messenger<\n typeof controllerName,\n TokenRatesControllerActions | AllowedActions,\n TokenRatesControllerEvents | AllowedEvents\n>;\n\nconst tokenRatesControllerMetadata: StateMetadata<TokenRatesControllerState> = {\n marketData: {\n includeInStateLogs: false,\n persist: true,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n};\n\n/**\n * Get the default {@link TokenRatesController} state.\n *\n * @returns The default {@link TokenRatesController} state.\n */\nexport const getDefaultTokenRatesControllerState =\n (): TokenRatesControllerState => {\n return {\n marketData: {},\n };\n };\n\n/** The input to start polling for the {@link TokenRatesController} */\nexport type TokenRatesPollingInput = {\n chainIds: Hex[];\n};\n\n/**\n * Controller that passively polls on a set interval for token-to-fiat exchange rates\n * for tokens stored in the TokensController\n */\nexport class TokenRatesController extends StaticIntervalPollingController<TokenRatesPollingInput>()<\n typeof controllerName,\n TokenRatesControllerState,\n TokenRatesControllerMessenger\n> {\n readonly #tokenPricesService: AbstractTokenPricesService;\n\n #disabled: boolean;\n\n #allTokens: TokensControllerState['allTokens'];\n\n #allDetectedTokens: TokensControllerState['allDetectedTokens'];\n\n /**\n * Creates a TokenRatesController instance.\n *\n * @param options - The controller options.\n * @param options.interval - The polling interval in ms\n * @param options.disabled - Boolean to track if network requests are blocked\n * @param options.tokenPricesService - An object in charge of retrieving token price\n * @param options.messenger - The messenger instance for communication\n * @param options.state - Initial state to set on this controller\n */\n constructor({\n interval = DEFAULT_INTERVAL,\n disabled = false,\n tokenPricesService,\n messenger,\n state,\n }: {\n interval?: number;\n disabled?: boolean;\n tokenPricesService: AbstractTokenPricesService;\n messenger: TokenRatesControllerMessenger;\n state?: Partial<TokenRatesControllerState>;\n }) {\n super({\n name: controllerName,\n messenger,\n state: { ...getDefaultTokenRatesControllerState(), ...state },\n metadata: tokenRatesControllerMetadata,\n });\n\n this.setIntervalLength(interval);\n this.#tokenPricesService = tokenPricesService;\n this.#disabled = disabled;\n\n const { allTokens, allDetectedTokens } = this.#getTokensControllerState();\n this.#allTokens = allTokens;\n this.#allDetectedTokens = allDetectedTokens;\n\n this.#subscribeToTokensStateChange();\n\n this.#subscribeToNetworkStateChange();\n }\n\n #subscribeToTokensStateChange() {\n this.messenger.subscribe(\n 'TokensController:stateChange',\n // TODO: Either fix this lint violation or explain why it's necessary to ignore.\n // eslint-disable-next-line @typescript-eslint/no-misused-promises\n async ({ allTokens, allDetectedTokens }) => {\n if (this.#disabled) {\n return;\n }\n\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n\n const chainIds = [\n ...new Set([\n ...Object.keys(allTokens),\n ...Object.keys(allDetectedTokens),\n ]),\n ] as Hex[];\n\n const chainIdsToUpdate = chainIds.filter(\n (chainId) =>\n !isEqual(this.#allTokens[chainId], allTokens[chainId]) ||\n !isEqual(\n this.#allDetectedTokens[chainId],\n allDetectedTokens[chainId],\n ),\n );\n\n this.#allTokens = allTokens;\n this.#allDetectedTokens = allDetectedTokens;\n\n const chainIdAndNativeCurrency = chainIdsToUpdate.reduce<\n { chainId: Hex; nativeCurrency: string }[]\n >((acc, chainId) => {\n const networkConfiguration = networkConfigurationsByChainId[chainId];\n if (!networkConfiguration) {\n console.error(\n `TokenRatesController: No network configuration found for chainId ${chainId}`,\n );\n return acc;\n }\n acc.push({\n chainId,\n nativeCurrency: networkConfiguration.nativeCurrency,\n });\n return acc;\n }, []);\n\n await this.updateExchangeRates(chainIdAndNativeCurrency);\n },\n ({ allTokens, allDetectedTokens }) => {\n return { allTokens, allDetectedTokens };\n },\n );\n }\n\n #subscribeToNetworkStateChange() {\n this.messenger.subscribe(\n 'NetworkController:stateChange',\n (_state, patches) => {\n // Remove state for deleted networks\n for (const patch of patches) {\n if (\n patch.op === 'remove' &&\n patch.path[0] === 'networkConfigurationsByChainId'\n ) {\n const removedChainId = patch.path[1] as Hex;\n this.update((state) => {\n delete state.marketData[removedChainId];\n });\n }\n }\n },\n );\n }\n\n /**\n * Get the tokens for the given chain.\n *\n * @param chainId - The chain ID.\n * @returns The list of tokens addresses for the current chain\n */\n #getTokenAddresses(chainId: Hex): Hex[] {\n const getTokens = (allTokens: Record<Hex, { address: string }[]>) =>\n Object.values(allTokens ?? {}).flatMap((tokens) =>\n tokens.map(({ address }) => toChecksumHexAddress(address) as Hex),\n );\n\n const tokenAddresses = getTokens(this.#allTokens[chainId]);\n const detectedTokenAddresses = getTokens(this.#allDetectedTokens[chainId]);\n\n return [\n ...new Set([\n ...tokenAddresses,\n ...detectedTokenAddresses,\n getNativeTokenAddress(chainId),\n ]),\n ].sort();\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 #getTokensControllerState(): {\n allTokens: TokensControllerState['allTokens'];\n allDetectedTokens: TokensControllerState['allDetectedTokens'];\n } {\n const { allTokens, allDetectedTokens } = this.messenger.call(\n 'TokensController:getState',\n );\n\n return {\n allTokens,\n allDetectedTokens,\n };\n }\n\n /**\n * Updates exchange rates for all tokens.\n *\n * @param chainIdAndNativeCurrency - The chain ID and native currency.\n */\n async updateExchangeRates(\n chainIdAndNativeCurrency: ChainIdAndNativeCurrency[],\n ): Promise<void> {\n if (this.#disabled) {\n return;\n }\n\n const marketData: Record<Hex, Record<Hex, MarketDataDetails>> = {};\n const assetsByNativeCurrency: Record<\n string,\n {\n chainId: Hex;\n tokenAddress: Hex;\n }[]\n > = {};\n const unsupportedAssetsByNativeCurrency: Record<\n string,\n {\n chainId: Hex;\n tokenAddress: Hex;\n }[]\n > = {};\n for (const { chainId, nativeCurrency } of chainIdAndNativeCurrency) {\n if (this.#tokenPricesService.validateChainIdSupported(chainId)) {\n for (const tokenAddress of this.#getTokenAddresses(chainId)) {\n if (\n this.#tokenPricesService.validateCurrencySupported(nativeCurrency)\n ) {\n (assetsByNativeCurrency[nativeCurrency] ??= []).push({\n chainId,\n tokenAddress,\n });\n } else {\n (unsupportedAssetsByNativeCurrency[nativeCurrency] ??= []).push({\n chainId,\n tokenAddress,\n });\n }\n }\n }\n }\n\n const promises = [\n ...Object.entries(assetsByNativeCurrency).map(\n ([nativeCurrency, assets]) =>\n this.#fetchAndMapExchangeRatesForSupportedNativeCurrency(\n assets,\n nativeCurrency,\n marketData,\n ),\n ),\n ...Object.entries(unsupportedAssetsByNativeCurrency).map(\n ([nativeCurrency, assets]) =>\n this.#fetchAndMapExchangeRatesForUnsupportedNativeCurrency(\n assets,\n nativeCurrency,\n marketData,\n ),\n ),\n ];\n\n await Promise.allSettled(promises);\n\n const chainIds = new Set(\n Object.values(chainIdAndNativeCurrency).map((chain) => chain.chainId),\n );\n\n for (const chainId of chainIds) {\n if (!marketData[chainId]) {\n marketData[chainId] = {};\n }\n }\n\n if (Object.keys(marketData).length > 0) {\n this.update((state) => {\n state.marketData = {\n ...state.marketData,\n ...marketData,\n };\n });\n }\n }\n\n async #fetchAndMapExchangeRatesForSupportedNativeCurrency(\n assets: {\n chainId: Hex;\n tokenAddress: Hex;\n }[],\n currency: string,\n marketData: Record<Hex, Record<Hex, MarketDataDetails>> = {},\n ) {\n return await reduceInBatchesSerially<\n { chainId: Hex; tokenAddress: Hex },\n Record<Hex, Record<Hex, MarketDataDetails>>\n >({\n values: assets,\n batchSize: TOKEN_PRICES_BATCH_SIZE,\n eachBatch: async (partialMarketData, assetsBatch) => {\n const batchMarketData = await this.#tokenPricesService.fetchTokenPrices(\n {\n assets: assetsBatch,\n currency,\n },\n );\n\n for (const tokenPrice of batchMarketData) {\n (partialMarketData[tokenPrice.chainId] ??= {})[\n tokenPrice.tokenAddress\n ] = tokenPrice;\n }\n\n return partialMarketData;\n },\n initialResult: marketData,\n });\n }\n\n async #fetchAndMapExchangeRatesForUnsupportedNativeCurrency(\n assets: {\n chainId: Hex;\n tokenAddress: Hex;\n }[],\n currency: string,\n marketData: Record<Hex, Record<Hex, MarketDataDetails>>,\n ) {\n // Step -1: Then fetch all tracked tokens priced in USD\n const marketDataInUSD =\n await this.#fetchAndMapExchangeRatesForSupportedNativeCurrency(\n assets,\n 'usd', // Fallback currency when the native currency is not supported\n );\n\n // Formula: price_in_native = token_usd / native_usd\n const convertUSDToNative = (\n valueInUSD: number,\n nativeTokenPriceInUSD: number,\n ) => valueInUSD / nativeTokenPriceInUSD;\n\n // Step -2: Convert USD prices to native currency\n for (const [chainId, marketDataByTokenAddress] of Object.entries(\n marketDataInUSD,\n ) as [Hex, Record<Hex, MarketDataDetails>][]) {\n const nativeTokenPriceInUSD =\n marketDataByTokenAddress[getNativeTokenAddress(chainId)]?.price;\n\n // Return here if it's null, undefined or 0\n if (!nativeTokenPriceInUSD) {\n continue;\n }\n\n for (const [tokenAddress, tokenData] of Object.entries(\n marketDataByTokenAddress,\n ) as [Hex, MarketDataDetails][]) {\n (marketData[chainId] ??= {})[tokenAddress] = {\n ...tokenData,\n currency,\n price: convertUSDToNative(tokenData.price, nativeTokenPriceInUSD),\n marketCap: convertUSDToNative(\n tokenData.marketCap,\n nativeTokenPriceInUSD,\n ),\n allTimeHigh: convertUSDToNative(\n tokenData.allTimeHigh,\n nativeTokenPriceInUSD,\n ),\n allTimeLow: convertUSDToNative(\n tokenData.allTimeLow,\n nativeTokenPriceInUSD,\n ),\n totalVolume: convertUSDToNative(\n tokenData.totalVolume,\n nativeTokenPriceInUSD,\n ),\n high1d: convertUSDToNative(tokenData.high1d, nativeTokenPriceInUSD),\n low1d: convertUSDToNative(tokenData.low1d, nativeTokenPriceInUSD),\n dilutedMarketCap: convertUSDToNative(\n tokenData.dilutedMarketCap,\n nativeTokenPriceInUSD,\n ),\n };\n }\n }\n }\n\n /**\n * Updates token rates for the given networkClientId\n *\n * @param input - The input for the poll.\n * @param input.chainIds - The chain ids to poll token rates on.\n */\n async _executePoll({ chainIds }: TokenRatesPollingInput): Promise<void> {\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n\n const chainIdAndNativeCurrency = chainIds.reduce<\n { chainId: Hex; nativeCurrency: string }[]\n >((acc, chainId) => {\n const networkConfiguration = networkConfigurationsByChainId[chainId];\n if (!networkConfiguration) {\n console.error(\n `TokenRatesController: No network configuration found for chainId ${chainId}`,\n );\n return acc;\n }\n acc.push({\n chainId,\n nativeCurrency: networkConfiguration.nativeCurrency,\n });\n return acc;\n }, []);\n\n await this.updateExchangeRates(chainIdAndNativeCurrency);\n }\n\n /**\n * Reset the controller state to the default state.\n */\n resetState() {\n this.update(() => {\n return getDefaultTokenRatesControllerState();\n });\n }\n}\n\nexport default TokenRatesController;\n"]}
|
|
1
|
+
{"version":3,"file":"TokenRatesController.cjs","sourceRoot":"","sources":["../src/TokenRatesController.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n AccountsControllerGetAccountAction,\n AccountsControllerGetSelectedAccountAction,\n AccountsControllerSelectedEvmAccountChangeEvent,\n} from '@metamask/accounts-controller';\nimport type {\n ControllerGetStateAction,\n ControllerStateChangeEvent,\n StateMetadata,\n} from '@metamask/base-controller';\nimport {\n safelyExecute,\n toChecksumHexAddress,\n} from '@metamask/controller-utils';\nimport type { Messenger } from '@metamask/messenger';\nimport type {\n NetworkControllerGetNetworkClientByIdAction,\n NetworkControllerGetStateAction,\n NetworkControllerStateChangeEvent,\n} from '@metamask/network-controller';\nimport { StaticIntervalPollingController } from '@metamask/polling-controller';\nimport { createDeferredPromise, type Hex } from '@metamask/utils';\nimport { isEqual } from 'lodash';\n\nimport { reduceInBatchesSerially, TOKEN_PRICES_BATCH_SIZE } from './assetsUtil';\nimport type { AbstractTokenPricesService } from './token-prices-service/abstract-token-prices-service';\nimport { getNativeTokenAddress } from './token-prices-service/codefi-v2';\nimport type {\n TokensControllerGetStateAction,\n TokensControllerStateChangeEvent,\n TokensControllerState,\n} from './TokensController';\n\n/**\n * @type Token\n *\n * Token representation\n *\n * @property address - Hex address of the token contract\n * @property decimals - Number of decimals the token uses\n * @property symbol - Symbol of the token\n * @property aggregators - An array containing the token's aggregators\n * @property image - Image of the token, url or bit32 image\n * @property hasBalanceError - 'true' if there is an error while updating the token balance\n * @property isERC721 - 'true' if the token is a ERC721 token\n * @property name - Name of the token\n */\nexport type Token = {\n address: string;\n decimals: number;\n symbol: string;\n aggregators?: string[];\n image?: string;\n hasBalanceError?: boolean;\n isERC721?: boolean;\n name?: string;\n};\n\nconst DEFAULT_INTERVAL = 180000;\n\nexport type ContractExchangeRates = {\n [address: string]: number | undefined;\n};\n\nexport type MarketDataDetails = {\n tokenAddress: `0x${string}`;\n currency: string;\n allTimeHigh: number;\n allTimeLow: number;\n circulatingSupply: number;\n dilutedMarketCap: number;\n high1d: number;\n low1d: number;\n marketCap: number;\n marketCapPercentChange1d: number;\n price: number;\n priceChange1d: number;\n pricePercentChange1d: number;\n pricePercentChange1h: number;\n pricePercentChange1y: number;\n pricePercentChange7d: number;\n pricePercentChange14d: number;\n pricePercentChange30d: number;\n pricePercentChange200d: number;\n totalVolume: number;\n};\n\n/**\n * Represents a mapping of token contract addresses to their market data.\n */\nexport type ContractMarketData = Record<Hex, MarketDataDetails>;\n\nenum PollState {\n Active = 'Active',\n Inactive = 'Inactive',\n}\n\n/**\n * The external actions available to the {@link TokenRatesController}.\n */\nexport type AllowedActions =\n | TokensControllerGetStateAction\n | NetworkControllerGetNetworkClientByIdAction\n | NetworkControllerGetStateAction\n | AccountsControllerGetAccountAction\n | AccountsControllerGetSelectedAccountAction;\n\n/**\n * The external events available to the {@link TokenRatesController}.\n */\nexport type AllowedEvents =\n | TokensControllerStateChangeEvent\n | NetworkControllerStateChangeEvent\n | AccountsControllerSelectedEvmAccountChangeEvent;\n\n/**\n * The name of the {@link TokenRatesController}.\n */\nexport const controllerName = 'TokenRatesController';\n\n/**\n * @type TokenRatesState\n *\n * Token rates controller state\n *\n * @property marketData - Market data for tokens, keyed by chain ID and then token contract address.\n */\nexport type TokenRatesControllerState = {\n marketData: Record<Hex, Record<Hex, MarketDataDetails>>;\n};\n\n/**\n * The action that can be performed to get the state of the {@link TokenRatesController}.\n */\nexport type TokenRatesControllerGetStateAction = ControllerGetStateAction<\n typeof controllerName,\n TokenRatesControllerState\n>;\n\n/**\n * The actions that can be performed using the {@link TokenRatesController}.\n */\nexport type TokenRatesControllerActions = TokenRatesControllerGetStateAction;\n\n/**\n * The event that {@link TokenRatesController} can emit.\n */\nexport type TokenRatesControllerStateChangeEvent = ControllerStateChangeEvent<\n typeof controllerName,\n TokenRatesControllerState\n>;\n\n/**\n * The events that {@link TokenRatesController} can emit.\n */\nexport type TokenRatesControllerEvents = TokenRatesControllerStateChangeEvent;\n\n/**\n * The messenger of the {@link TokenRatesController} for communication.\n */\nexport type TokenRatesControllerMessenger = Messenger<\n typeof controllerName,\n TokenRatesControllerActions | AllowedActions,\n TokenRatesControllerEvents | AllowedEvents\n>;\n\nconst tokenRatesControllerMetadata: StateMetadata<TokenRatesControllerState> = {\n marketData: {\n includeInStateLogs: false,\n persist: true,\n includeInDebugSnapshot: false,\n usedInUi: true,\n },\n};\n\n/**\n * Get the default {@link TokenRatesController} state.\n *\n * @returns The default {@link TokenRatesController} state.\n */\nexport const getDefaultTokenRatesControllerState =\n (): TokenRatesControllerState => {\n return {\n marketData: {},\n };\n };\n\n/** The input to start polling for the {@link TokenRatesController} */\nexport type TokenRatesPollingInput = {\n chainIds: Hex[];\n};\n\n/**\n * Controller that passively polls on a set interval for token-to-fiat exchange rates\n * for tokens stored in the TokensController\n */\nexport class TokenRatesController extends StaticIntervalPollingController<TokenRatesPollingInput>()<\n typeof controllerName,\n TokenRatesControllerState,\n TokenRatesControllerMessenger\n> {\n #handle?: ReturnType<typeof setTimeout>;\n\n #pollState = PollState.Inactive;\n\n readonly #tokenPricesService: AbstractTokenPricesService;\n\n #inProcessExchangeRateUpdates: Record<`${Hex}:${string}`, Promise<void>> = {};\n\n #disabled: boolean;\n\n readonly #interval: number;\n\n #allTokens: TokensControllerState['allTokens'];\n\n #allDetectedTokens: TokensControllerState['allDetectedTokens'];\n\n /**\n * Creates a TokenRatesController instance.\n *\n * @param options - The controller options.\n * @param options.interval - The polling interval in ms\n * @param options.disabled - Boolean to track if network requests are blocked\n * @param options.tokenPricesService - An object in charge of retrieving token price\n * @param options.messenger - The messenger instance for communication\n * @param options.state - Initial state to set on this controller\n */\n constructor({\n interval = DEFAULT_INTERVAL,\n disabled = false,\n tokenPricesService,\n messenger,\n state,\n }: {\n interval?: number;\n disabled?: boolean;\n tokenPricesService: AbstractTokenPricesService;\n messenger: TokenRatesControllerMessenger;\n state?: Partial<TokenRatesControllerState>;\n }) {\n super({\n name: controllerName,\n messenger,\n state: { ...getDefaultTokenRatesControllerState(), ...state },\n metadata: tokenRatesControllerMetadata,\n });\n\n this.setIntervalLength(interval);\n this.#tokenPricesService = tokenPricesService;\n this.#disabled = disabled;\n this.#interval = interval;\n\n const { allTokens, allDetectedTokens } = this.#getTokensControllerState();\n this.#allTokens = allTokens;\n this.#allDetectedTokens = allDetectedTokens;\n\n this.#subscribeToTokensStateChange();\n\n this.#subscribeToNetworkStateChange();\n }\n\n #subscribeToTokensStateChange() {\n this.messenger.subscribe(\n 'TokensController:stateChange',\n // TODO: Either fix this lint violation or explain why it's necessary to ignore.\n // eslint-disable-next-line @typescript-eslint/no-misused-promises\n async ({ allTokens, allDetectedTokens }) => {\n if (this.#disabled) {\n return;\n }\n\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n\n const chainIds = [\n ...new Set([\n ...Object.keys(allTokens),\n ...Object.keys(allDetectedTokens),\n ]),\n ] as Hex[];\n\n const chainIdsToUpdate = chainIds.filter(\n (chainId) =>\n !isEqual(this.#allTokens[chainId], allTokens[chainId]) ||\n !isEqual(\n this.#allDetectedTokens[chainId],\n allDetectedTokens[chainId],\n ),\n );\n\n this.#allTokens = allTokens;\n this.#allDetectedTokens = allDetectedTokens;\n\n const chainIdAndNativeCurrency = chainIdsToUpdate.reduce<\n { chainId: Hex; nativeCurrency: string }[]\n >((acc, chainId) => {\n const networkConfiguration = networkConfigurationsByChainId[chainId];\n if (!networkConfiguration) {\n console.error(\n `TokenRatesController: No network configuration found for chainId ${chainId}`,\n );\n return acc;\n }\n acc.push({\n chainId,\n nativeCurrency: networkConfiguration.nativeCurrency,\n });\n return acc;\n }, []);\n\n await this.updateExchangeRatesByChainId(chainIdAndNativeCurrency);\n },\n ({ allTokens, allDetectedTokens }) => {\n return { allTokens, allDetectedTokens };\n },\n );\n }\n\n #subscribeToNetworkStateChange() {\n this.messenger.subscribe(\n 'NetworkController:stateChange',\n // TODO: Either fix this lint violation or explain why it's necessary to ignore.\n // eslint-disable-next-line @typescript-eslint/no-misused-promises\n async ({ networkConfigurationsByChainId }, patches) => {\n const chainIdAndNativeCurrency: {\n chainId: Hex;\n nativeCurrency: string;\n }[] = Object.values(networkConfigurationsByChainId).map(\n ({ chainId, nativeCurrency }) => {\n return {\n chainId: chainId as Hex,\n nativeCurrency,\n };\n },\n );\n\n if (this.#pollState === PollState.Active) {\n await this.updateExchangeRates(chainIdAndNativeCurrency);\n }\n\n // Remove state for deleted networks\n for (const patch of patches) {\n if (\n patch.op === 'remove' &&\n patch.path[0] === 'networkConfigurationsByChainId'\n ) {\n const removedChainId = patch.path[1] as Hex;\n this.update((state) => {\n delete state.marketData[removedChainId];\n });\n }\n }\n },\n );\n }\n\n /**\n * Get the tokens for the given chain.\n *\n * @param chainId - The chain ID.\n * @returns The list of tokens addresses for the current chain\n */\n #getTokenAddresses(chainId: Hex): Hex[] {\n const getTokens = (allTokens: Record<Hex, { address: string }[]>) =>\n Object.values(allTokens ?? {}).flatMap((tokens) =>\n tokens.map(({ address }) => toChecksumHexAddress(address) as Hex),\n );\n\n const tokenAddresses = getTokens(this.#allTokens[chainId]);\n const detectedTokenAddresses = getTokens(this.#allDetectedTokens[chainId]);\n\n return [...new Set([...tokenAddresses, ...detectedTokenAddresses])].sort();\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 * Start (or restart) polling.\n *\n * @param chainId - The chain ID.\n * @param nativeCurrency - The native currency.\n */\n async start(chainId: Hex, nativeCurrency: string) {\n this.#stopPoll();\n this.#pollState = PollState.Active;\n await this.#poll(chainId, nativeCurrency);\n }\n\n /**\n * Stop polling.\n */\n stop() {\n this.#stopPoll();\n this.#pollState = PollState.Inactive;\n }\n\n #getTokensControllerState(): {\n allTokens: TokensControllerState['allTokens'];\n allDetectedTokens: TokensControllerState['allDetectedTokens'];\n } {\n const { allTokens, allDetectedTokens } = this.messenger.call(\n 'TokensController:getState',\n );\n\n return {\n allTokens,\n allDetectedTokens,\n };\n }\n\n /**\n * Clear the active polling timer, if present.\n */\n #stopPoll() {\n if (this.#handle) {\n clearTimeout(this.#handle);\n }\n }\n\n /**\n * Poll for exchange rate updates.\n *\n * @param chainId - The chain ID.\n * @param nativeCurrency - The native currency.\n */\n async #poll(chainId: Hex, nativeCurrency: string) {\n await safelyExecute(() =>\n this.updateExchangeRates([{ chainId, nativeCurrency }]),\n );\n\n // Poll using recursive `setTimeout` instead of `setInterval` so that\n // requests don't stack if they take longer than the polling interval\n this.#handle = setTimeout(() => {\n // TODO: Either fix this lint violation or explain why it's necessary to ignore.\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\n this.#poll(chainId, nativeCurrency);\n }, this.#interval);\n }\n\n /**\n * Updates exchange rates for all tokens.\n *\n * @param chainIdAndNativeCurrency - The chain ID and native currency.\n */\n async updateExchangeRates(\n chainIdAndNativeCurrency: {\n chainId: Hex;\n nativeCurrency: string;\n }[],\n ) {\n await this.updateExchangeRatesByChainId(chainIdAndNativeCurrency);\n }\n\n /**\n * Updates exchange rates for all tokens.\n *\n * @param chainIds - The chain IDs.\n * @returns A promise that resolves when all chain updates complete.\n */\n /**\n * Updates exchange rates for all tokens.\n *\n * @param chainIdAndNativeCurrency - The chain ID and native currency.\n */\n async updateExchangeRatesByChainId(\n chainIdAndNativeCurrency: {\n chainId: Hex;\n nativeCurrency: string;\n }[],\n ): Promise<void> {\n if (this.#disabled) {\n return;\n }\n\n // Create a promise for each chainId to fetch exchange rates.\n const updatePromises = chainIdAndNativeCurrency.map(\n async ({ chainId, nativeCurrency }) => {\n const tokenAddresses = this.#getTokenAddresses(chainId);\n // Build a unique key based on chainId, nativeCurrency, and the number of token addresses.\n const updateKey: `${Hex}:${string}` = `${chainId}:${nativeCurrency}:${tokenAddresses.length}`;\n\n if (updateKey in this.#inProcessExchangeRateUpdates) {\n // Await any ongoing update to avoid redundant work.\n await this.#inProcessExchangeRateUpdates[updateKey];\n return null;\n }\n\n // Create a deferred promise to track this update.\n const {\n promise: inProgressUpdate,\n resolve: updateSucceeded,\n reject: updateFailed,\n } = createDeferredPromise({ suppressUnhandledRejection: true });\n this.#inProcessExchangeRateUpdates[updateKey] = inProgressUpdate;\n\n try {\n const contractInformations = await this.#fetchAndMapExchangeRates({\n tokenAddresses,\n chainId,\n nativeCurrency,\n });\n\n // Each promise returns an object with the market data for the chain.\n const marketData = {\n [chainId]: {\n ...(contractInformations ?? {}),\n },\n };\n\n updateSucceeded();\n return marketData;\n } catch (error: unknown) {\n updateFailed(error);\n throw error;\n } finally {\n // Cleanup the tracking for this update.\n delete this.#inProcessExchangeRateUpdates[updateKey];\n }\n },\n );\n\n // Wait for all update promises to settle.\n const results = await Promise.allSettled(updatePromises);\n\n // Merge all successful market data updates into one object.\n const combinedMarketData = results.reduce((acc, result) => {\n if (result.status === 'fulfilled' && result.value) {\n acc = { ...acc, ...result.value };\n }\n return acc;\n }, {});\n\n // Call this.update only once with the combined market data to reduce the number of state changes and re-renders\n if (Object.keys(combinedMarketData).length > 0) {\n this.update((state) => {\n state.marketData = {\n ...state.marketData,\n ...combinedMarketData,\n };\n });\n }\n }\n\n /**\n * Uses the token prices service to retrieve exchange rates for tokens in a\n * particular currency.\n *\n * If the price API does not support the given chain ID, returns an empty\n * object.\n *\n * If the price API does not support the given currency, retrieves exchange\n * rates in a known currency instead, then converts those rates using the\n * exchange rate between the known currency and desired currency.\n *\n * @param args - The arguments to this function.\n * @param args.tokenAddresses - Addresses for tokens.\n * @param args.chainId - The EIP-155 ID of the chain where the tokens live.\n * @param args.nativeCurrency - The native currency in which to request\n * exchange rates.\n * @returns A map from token address to its exchange rate in the native\n * currency, or an empty map if no exchange rates can be obtained for the\n * chain ID.\n */\n async #fetchAndMapExchangeRates({\n tokenAddresses,\n chainId,\n nativeCurrency,\n }: {\n tokenAddresses: Hex[];\n chainId: Hex;\n nativeCurrency: string;\n }): Promise<ContractMarketData> {\n if (!this.#tokenPricesService.validateChainIdSupported(chainId)) {\n return tokenAddresses.reduce((obj, tokenAddress) => {\n obj = {\n ...obj,\n [tokenAddress]: undefined,\n };\n\n return obj;\n }, {});\n }\n\n if (this.#tokenPricesService.validateCurrencySupported(nativeCurrency)) {\n return await this.#fetchAndMapExchangeRatesForSupportedNativeCurrency({\n tokenAddresses,\n chainId,\n nativeCurrency,\n });\n }\n\n return await this.#fetchAndMapExchangeRatesForUnsupportedNativeCurrency({\n chainId,\n tokenAddresses,\n nativeCurrency,\n });\n }\n\n /**\n * Updates token rates for the given networkClientId\n *\n * @param input - The input for the poll.\n * @param input.chainIds - The chain ids to poll token rates on.\n */\n async _executePoll({ chainIds }: TokenRatesPollingInput): Promise<void> {\n const { networkConfigurationsByChainId } = this.messenger.call(\n 'NetworkController:getState',\n );\n\n const chainIdAndNativeCurrency = chainIds.reduce<\n { chainId: Hex; nativeCurrency: string }[]\n >((acc, chainId) => {\n const networkConfiguration = networkConfigurationsByChainId[chainId];\n if (!networkConfiguration) {\n console.error(\n `TokenRatesController: No network configuration found for chainId ${chainId}`,\n );\n return acc;\n }\n acc.push({\n chainId,\n nativeCurrency: networkConfiguration.nativeCurrency,\n });\n return acc;\n }, []);\n\n await this.updateExchangeRatesByChainId(chainIdAndNativeCurrency);\n }\n\n /**\n * Retrieves prices in the given currency for the given tokens on the given\n * chain. Ensures that token addresses are checksum addresses.\n *\n * @param args - The arguments to this function.\n * @param args.tokenAddresses - Addresses for tokens.\n * @param args.chainId - The EIP-155 ID of the chain where the tokens live.\n * @param args.nativeCurrency - The native currency in which to request\n * prices.\n * @returns A map of the token addresses (as checksums) to their prices in the\n * native currency.\n */\n async #fetchAndMapExchangeRatesForSupportedNativeCurrency({\n tokenAddresses,\n chainId,\n nativeCurrency,\n }: {\n tokenAddresses: Hex[];\n chainId: Hex;\n nativeCurrency: string;\n }): Promise<ContractMarketData> {\n let contractNativeInformations;\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 this.#tokenPricesService.fetchTokenPrices({\n tokenAddresses: batch,\n chainId,\n currency: nativeCurrency,\n });\n\n return {\n ...allTokenPricesByTokenAddress,\n ...tokenPricesByTokenAddressForBatch,\n };\n },\n initialResult: {},\n });\n contractNativeInformations = tokenPricesByTokenAddress;\n\n // fetch for native token\n if (tokenAddresses.length === 0) {\n const contractNativeInformationsNative =\n await this.#tokenPricesService.fetchTokenPrices({\n tokenAddresses: [],\n chainId,\n currency: nativeCurrency,\n });\n\n contractNativeInformations = {\n [getNativeTokenAddress(chainId)]: {\n currency: nativeCurrency,\n ...contractNativeInformationsNative[getNativeTokenAddress(chainId)],\n },\n };\n }\n return Object.entries(contractNativeInformations).reduce(\n (obj, [tokenAddress, token]) => {\n obj = {\n ...obj,\n [tokenAddress]: { ...token },\n };\n\n return obj;\n },\n {},\n );\n }\n\n /**\n * If the price API does not support a given native currency, then we need to\n * convert it to a fallback currency and feed that currency into the price\n * API, then convert the prices to our desired native currency.\n *\n * @param args - The arguments to this function.\n * @param args.chainId - The chain id to fetch prices for.\n * @param args.tokenAddresses - Addresses for tokens.\n * @param args.nativeCurrency - The native currency in which to request\n * prices.\n * @returns A map of the token addresses (as checksums) to their prices in the\n * native currency.\n */\n async #fetchAndMapExchangeRatesForUnsupportedNativeCurrency({\n chainId,\n tokenAddresses,\n nativeCurrency,\n }: {\n chainId: Hex;\n tokenAddresses: Hex[];\n nativeCurrency: string;\n }): Promise<ContractMarketData> {\n const nativeTokenAddress = getNativeTokenAddress(chainId);\n\n // Step -1: First fetch native token priced in USD\n const nativeTokenPriceMap =\n await this.#fetchAndMapExchangeRatesForSupportedNativeCurrency({\n tokenAddresses: [] as Hex[], // special-case: returns only native token\n chainId,\n nativeCurrency: 'usd',\n });\n\n // Step -2: Then fetch all tracked tokens priced in USD\n const tokenPricesInUSD =\n await this.#fetchAndMapExchangeRatesForSupportedNativeCurrency({\n tokenAddresses,\n chainId,\n nativeCurrency: 'usd',\n });\n\n const nativeTokenInfo = nativeTokenPriceMap[nativeTokenAddress];\n const nativeTokenPriceInUSD = nativeTokenInfo?.price;\n\n if (!nativeTokenPriceInUSD || nativeTokenPriceInUSD === 0) {\n // If we can't price the native token in the fallback currency,\n // we can't safely convert; return empty so callers know there is no data.\n return {};\n }\n\n // Step -3: Convert USD prices to native currency\n // Formula: price_in_native = token_usd / native_usd\n const convertUSDToNative = (valueInUSD: number | undefined) =>\n valueInUSD !== undefined && valueInUSD !== null\n ? valueInUSD / nativeTokenPriceInUSD\n : undefined;\n\n // Step -4 & -5: Apply conversion to all token fields and return\n const tokenPricesInNative = Object.entries(tokenPricesInUSD).reduce(\n (acc, [tokenAddress, tokenData]) => {\n acc = {\n ...acc,\n [tokenAddress]: {\n ...tokenData,\n currency: nativeCurrency,\n price: convertUSDToNative(tokenData.price),\n marketCap: convertUSDToNative(tokenData.marketCap),\n allTimeHigh: convertUSDToNative(tokenData.allTimeHigh),\n allTimeLow: convertUSDToNative(tokenData.allTimeLow),\n totalVolume: convertUSDToNative(tokenData.totalVolume),\n high1d: convertUSDToNative(tokenData.high1d),\n low1d: convertUSDToNative(tokenData.low1d),\n dilutedMarketCap: convertUSDToNative(tokenData.dilutedMarketCap),\n },\n };\n return acc;\n },\n {} as ContractMarketData,\n );\n\n return tokenPricesInNative;\n }\n\n /**\n * Reset the controller state to the default state.\n */\n resetState() {\n this.update(() => {\n return getDefaultTokenRatesControllerState();\n });\n }\n}\n\nexport default TokenRatesController;\n"]}
|