@metamask-previews/transaction-pay-controller 0.0.0-preview-24bcd769 → 1.0.0-preview-1ae9d4fd

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 CHANGED
@@ -7,8 +7,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
7
7
 
8
8
  ## [Unreleased]
9
9
 
10
+ ## [1.0.0]
11
+
10
12
  ### Added
11
13
 
12
14
  - Initial release ([#6820](https://github.com/MetaMask/core/pull/6820))
13
15
 
14
- [Unreleased]: https://github.com/MetaMask/core/
16
+ [Unreleased]: https://github.com/MetaMask/core/compare/@metamask/transaction-pay-controller@1.0.0...HEAD
17
+ [1.0.0]: https://github.com/MetaMask/core/releases/tag/@metamask/transaction-pay-controller@1.0.0
@@ -31,15 +31,9 @@ const log = (0, utils_1.createModuleLogger)(logger_1.projectLogger, 'bridge-stra
31
31
  async function getBridgeQuotes(request) {
32
32
  log('Fetching quotes', request);
33
33
  const { requests, messenger, transaction } = request;
34
- try {
35
- const finalRequests = getFinalRequests(requests, messenger);
36
- const quotes = await Promise.all(finalRequests.map((r, index) => getSufficientSingleBridgeQuote(r, index, request)));
37
- return quotes.map((quote, index) => normalizeQuote(quote, finalRequests[index], messenger, transaction));
38
- }
39
- catch (error) {
40
- log('Error fetching quotes', { error });
41
- throw new Error(`Failed to fetch bridge quotes: ${String(error)}`);
42
- }
34
+ const finalRequests = getFinalRequests(requests, messenger);
35
+ const quotes = await Promise.all(finalRequests.map((r, index) => getSufficientSingleBridgeQuote(r, index, request)));
36
+ return quotes.map((quote, index) => normalizeQuote(quote, finalRequests[index], messenger, transaction));
43
37
  }
44
38
  exports.getBridgeQuotes = getBridgeQuotes;
45
39
  /**
@@ -305,7 +299,7 @@ function getFeatureFlags(messenger) {
305
299
  */
306
300
  function normalizeQuote(quote, request, messenger, transaction) {
307
301
  const targetFiatRate = (0, token_1.getTokenFiatRate)(messenger, quote.quote.destAsset.address, (0, controller_utils_1.toHex)(quote.quote.destChainId));
308
- const sourceFiatRate = (0, token_1.getTokenFiatRate)(messenger, request.sourceTokenAddress, (0, controller_utils_1.toHex)(quote.quote.srcChainId));
302
+ const sourceFiatRate = (0, token_1.getTokenFiatRate)(messenger, quote.quote.srcAsset.address, (0, controller_utils_1.toHex)(quote.quote.srcChainId));
309
303
  if (sourceFiatRate === undefined || targetFiatRate === undefined) {
310
304
  throw new Error('Fiat rate not found for source or target token');
311
305
  }
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-quotes.cjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n FeatureId,\n type GenericQuoteRequest,\n} from '@metamask/bridge-controller';\nimport type { TxData } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport { toChecksumHexAddress, toHex } from '@metamask/controller-utils';\nimport { TransactionType } from '@metamask/transaction-controller';\nimport type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\nimport { orderBy } from 'lodash';\n\nimport type {\n BridgeFeatureFlags,\n TransactionPayBridgeQuote,\n BridgeQuoteRequest,\n} from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetBatchRequest,\n PayStrategyGetQuotesRequest,\n PayStrategyGetRefreshIntervalRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getTokenFiatRate } from '../../utils/token';\n\nconst ERROR_MESSAGE_NO_QUOTES = 'No quotes found';\nconst ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM = 'All quotes under minimum';\nconst ATTEMPTS_MAX_DEFAULT = 5;\nconst BUFFER_INITIAL_DEFAULT = 0.04;\nconst BUFFER_STEP_DEFAULT = 0.04;\nconst BUFFER_SUBSEQUENT_DEFAULT = 0.05;\nconst SLIPPAGE_DEFAULT = 0.005;\n\nconst FEATURE_ID_BY_TRANSACTION_TYPE = new Map<TransactionType, FeatureId>([\n [TransactionType.perpsDeposit, FeatureId.PERPS],\n]);\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\n/**\n * Fetch bridge quotes for multiple requests.\n *\n * @param request - Request object.\n * @returns An array of bridge quotes.\n */\nexport async function getBridgeQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<TransactionPayBridgeQuote>[]> {\n log('Fetching quotes', request);\n\n const { requests, messenger, transaction } = request;\n\n try {\n const finalRequests = getFinalRequests(requests, messenger);\n\n const quotes = await Promise.all(\n finalRequests.map((r, index) =>\n getSufficientSingleBridgeQuote(r, index, request),\n ),\n );\n\n return quotes.map((quote, index) =>\n normalizeQuote(quote, finalRequests[index], messenger, transaction),\n );\n } catch (error) {\n log('Error fetching quotes', { error });\n throw new Error(`Failed to fetch bridge quotes: ${String(error)}`);\n }\n}\n\n/**\n * Get bridge batch transactions if needed by the quotes.\n *\n * @param request - Request object.\n * @returns Array of batch transactions.\n */\nexport async function getBridgeBatchTransactions(\n request: PayStrategyGetBatchRequest<TransactionPayBridgeQuote>,\n): Promise<BatchTransaction[]> {\n const { quotes } = request;\n const firstQuote = quotes[0]?.original?.quote;\n\n if (firstQuote?.srcChainId !== firstQuote?.destChainId) {\n log('No batch transactions needed for bridge quotes');\n return [];\n }\n\n return quotes\n .map((q) => q.original)\n .flatMap((quote) => {\n const result = [];\n\n if (quote.approval) {\n result.push({\n ...getBatchTransaction(quote.approval),\n type: TransactionType.swapApproval,\n });\n }\n\n result.push({\n ...getBatchTransaction(quote.trade as TxData),\n type: TransactionType.swap,\n });\n\n return result;\n });\n}\n\n/**\n * Get the refresh interval for bridge quotes.\n *\n * @param request - Request object.\n * @returns Refresh interval in milliseconds.\n */\nexport function getBridgeRefreshInterval(\n request: PayStrategyGetRefreshIntervalRequest,\n): number | undefined {\n const { chainId, messenger } = request;\n\n const bridgeFeatureFlags = messenger.call(\n 'RemoteFeatureFlagController:getState',\n ).remoteFeatureFlags.bridgeConfigV2 as BridgeFeatureFlags | undefined;\n\n const chainInterval =\n bridgeFeatureFlags?.chains?.[parseInt(chainId, 16)]?.refreshRate;\n\n const globalInterval = bridgeFeatureFlags?.refreshRate;\n\n return chainInterval ?? globalInterval;\n}\n\n/**\n * Get a fresh quote for a previously fetched bridge quote to avoid expiration.\n *\n * @param quote - Original quote.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Fresh quote response.\n */\nexport async function refreshQuote(\n quote: TransactionPayQuote<TransactionPayBridgeQuote>,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const newQuote = await getSingleBridgeQuote(\n { ...quote.original.request, attemptsMax: 1 },\n messenger,\n transaction,\n );\n\n log('Refreshed quote', { old: quote, new: newQuote });\n\n return newQuote;\n}\n\n/**\n * Convert a quote trade or approval to a batch transaction.\n *\n * @param transaction - Quote trade or approval.\n * @returns Batch transaction.\n */\nfunction getBatchTransaction(transaction: TxData): BatchTransaction {\n const data = transaction.data as Hex;\n const gas = transaction.gasLimit ? toHex(transaction.gasLimit) : undefined;\n const to = transaction.to as Hex;\n const value = transaction.value as Hex;\n\n return {\n data,\n gas,\n isAfter: false,\n to,\n value,\n };\n}\n\n/**\n * Retry fetching a single bridge quote until it meets the minimum target amount.\n *\n * @param quoteRequest - Original quote request.\n * @param index - Index of the request in the array.\n * @param request - Full quotes request.\n * @returns The sufficient bridge quote.\n */\nasync function getSufficientSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n index: number,\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayBridgeQuote> {\n const {\n attemptsMax,\n bufferInitial,\n bufferStep,\n bufferSubsequent,\n sourceBalanceRaw,\n sourceTokenAmount,\n targetAmountMinimum,\n targetTokenAddress,\n } = quoteRequest;\n\n const sourceAmountValue = new BigNumber(sourceTokenAmount);\n const buffer = index === 0 ? bufferInitial : bufferSubsequent;\n const originalSourceAmount = sourceAmountValue.div(1 + buffer);\n const start = Date.now();\n\n let currentSourceAmount = sourceTokenAmount;\n\n for (let i = 0; i < attemptsMax; i++) {\n const currentRequest = {\n ...quoteRequest,\n sourceTokenAmount: currentSourceAmount,\n };\n\n try {\n log('Attempt', {\n attempt: i + 1,\n attemptsMax,\n bufferInitial,\n bufferStep,\n currentSourceAmount,\n target: targetTokenAddress,\n });\n\n const result = await getSingleBridgeQuote(\n currentRequest,\n request.messenger,\n request.transaction,\n );\n\n const dust = new BigNumber(result.quote.minDestTokenAmount)\n .minus(targetAmountMinimum)\n .toString(10);\n\n log('Found valid quote', {\n attempt: i + 1,\n target: targetTokenAddress,\n targetAmount: result.quote.minDestTokenAmount,\n goalAmount: targetAmountMinimum,\n dust,\n quote: result,\n });\n\n return {\n ...result,\n metrics: {\n attempts: i + 1,\n buffer: buffer + bufferStep * i,\n latency: Date.now() - start,\n },\n };\n } catch (error) {\n const errorMessage = (error as { message: string }).message;\n\n if (errorMessage !== ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM) {\n throw error;\n }\n }\n\n if (\n new BigNumber(currentSourceAmount).isGreaterThanOrEqualTo(\n sourceBalanceRaw,\n )\n ) {\n log('Reached balance limit', targetTokenAddress);\n break;\n }\n\n const newSourceAmount = originalSourceAmount.multipliedBy(\n 1 + buffer + bufferStep * (i + 1),\n );\n\n currentSourceAmount = newSourceAmount.isLessThan(sourceBalanceRaw)\n ? newSourceAmount.toFixed(0)\n : sourceBalanceRaw;\n }\n\n log('All attempts failed', targetTokenAddress);\n\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n}\n\n/**\n * Fetch a single bridge quote.\n *\n * @param quoteRequest - Quote request parameters.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns The bridge quote.\n */\nasync function getSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const {\n from,\n slippage,\n sourceChainId,\n sourceTokenAddress,\n sourceTokenAmount,\n targetChainId,\n targetTokenAddress,\n } = quoteRequest;\n\n const { type } = transaction;\n const featureId = FEATURE_ID_BY_TRANSACTION_TYPE.get(type as TransactionType);\n\n const bridgeRequest: GenericQuoteRequest = {\n destChainId: targetChainId,\n destTokenAddress: toChecksumHexAddress(targetTokenAddress),\n destWalletAddress: from,\n gasIncluded: false,\n gasIncluded7702: false,\n insufficientBal: false,\n slippage: slippage * 100,\n srcChainId: sourceChainId,\n srcTokenAddress: toChecksumHexAddress(sourceTokenAddress),\n srcTokenAmount: sourceTokenAmount,\n walletAddress: from,\n };\n\n const quotes = await messenger.call(\n 'BridgeController:fetchQuotes',\n bridgeRequest,\n undefined,\n featureId,\n );\n\n if (!quotes.length) {\n throw new Error(ERROR_MESSAGE_NO_QUOTES);\n }\n\n const result = getBestQuote(quotes, quoteRequest);\n\n return {\n ...result,\n request: quoteRequest,\n };\n}\n\n/**\n * Select the best quote from a list of quotes.\n *\n * @param quotes - List of quotes.\n * @param request - Original quote request.\n * @returns The best quote.\n */\nfunction getBestQuote(\n quotes: QuoteResponse[],\n request: BridgeQuoteRequest,\n): QuoteResponse {\n const fastestQuotes = orderBy(\n quotes,\n (quote) => quote.estimatedProcessingTimeInSeconds,\n 'asc',\n ).slice(0, 3);\n\n const quotesOverMinimumTarget = fastestQuotes.filter((quote) =>\n new BigNumber(quote.quote.minDestTokenAmount).isGreaterThanOrEqualTo(\n request.targetAmountMinimum,\n ),\n );\n\n log('Finding best quote', {\n allQuotes: quotes,\n fastestQuotes,\n quotesOverMinimumTarget,\n });\n\n if (!quotesOverMinimumTarget.length) {\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n }\n\n const cheapestQuote = orderBy(\n quotesOverMinimumTarget,\n (quote) => BigNumber(quote.quote.minDestTokenAmount).toNumber(),\n 'desc',\n )[0];\n\n return cheapestQuote;\n}\n\n/**\n * Get the final bridge quote requests.\n * Subtracts subsequent source amounts from the available balance.\n *\n * @param requests - List of bridge quote requests.\n * @param messenger - Controller messenger.\n * @returns The final bridge quote requests.\n */\nfunction getFinalRequests(\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n): BridgeQuoteRequest[] {\n const featureFlags = getFeatureFlags(messenger);\n\n return requests\n .map((request) => ({ ...request, ...featureFlags }))\n .map((request, index) => {\n const isFirstRequest = index === 0;\n const attemptsMax = isFirstRequest ? request.attemptsMax : 1;\n\n const sourceBalanceRaw = requests\n .reduce((acc, value, j) => {\n const isSameSource =\n value.sourceTokenAddress.toLowerCase() ===\n request.sourceTokenAddress.toLowerCase() &&\n value.sourceChainId === request.sourceChainId;\n\n if (isFirstRequest && j > index && isSameSource) {\n return acc.minus(value.sourceTokenAmount);\n }\n\n return acc;\n }, new BigNumber(request.sourceBalanceRaw))\n .toFixed(0);\n\n return {\n ...request,\n attemptsMax,\n sourceBalanceRaw,\n };\n });\n}\n\n/**\n * Get feature flags for bridge quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlags = messenger.call('RemoteFeatureFlagController:getState')\n .remoteFeatureFlags.confirmation_pay as Record<string, number> | undefined;\n\n return {\n attemptsMax: featureFlags?.attemptsMax ?? ATTEMPTS_MAX_DEFAULT,\n bufferInitial: featureFlags?.bufferInitial ?? BUFFER_INITIAL_DEFAULT,\n bufferStep: featureFlags?.bufferStep ?? BUFFER_STEP_DEFAULT,\n bufferSubsequent:\n featureFlags?.bufferSubsequent ?? BUFFER_SUBSEQUENT_DEFAULT,\n slippage: featureFlags?.slippage ?? SLIPPAGE_DEFAULT,\n };\n}\n\n/**\n * Convert a bridge specific quote response to a normalized transaction pay quote.\n *\n * @param quote - Bridge quote response.\n * @param request - Request\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Normalized transaction pay quote.\n */\nfunction normalizeQuote(\n quote: TransactionPayBridgeQuote,\n request: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): TransactionPayQuote<TransactionPayBridgeQuote> {\n const targetFiatRate = getTokenFiatRate(\n messenger,\n quote.quote.destAsset.address as Hex,\n toHex(quote.quote.destChainId),\n );\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n request.sourceTokenAddress,\n toHex(quote.quote.srcChainId),\n );\n\n if (sourceFiatRate === undefined || targetFiatRate === undefined) {\n throw new Error('Fiat rate not found for source or target token');\n }\n\n const targetAmountMinimumFiat = calculateFiatValue(\n quote.quote.minDestTokenAmount,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const sourceAmountFiat = calculateFiatValue(\n quote.quote.srcTokenAmount,\n quote.quote.srcAsset.decimals,\n sourceFiatRate.fiatRate,\n sourceFiatRate.usdRate,\n );\n\n const targetAmountGoal = calculateFiatValue(\n request.targetAmountMinimum,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const targetNetwork = calculateTransactionGasCost(transaction, messenger);\n const sourceNetwork = calculateSourceNetworkFee(quote, messenger);\n\n return {\n estimatedDuration: quote.estimatedProcessingTimeInSeconds,\n dust: {\n fiat: new BigNumber(targetAmountMinimumFiat.fiat)\n .minus(targetAmountGoal.fiat)\n .toString(10),\n usd: new BigNumber(targetAmountMinimumFiat.usd)\n .minus(targetAmountGoal.usd)\n .toString(10),\n },\n fees: {\n provider: {\n fiat: new BigNumber(sourceAmountFiat.fiat)\n .minus(targetAmountMinimumFiat.fiat)\n .toString(10),\n usd: new BigNumber(sourceAmountFiat.usd)\n .minus(targetAmountMinimumFiat.usd)\n .toString(10),\n },\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Bridge,\n };\n}\n\n/**\n * Calculate fiat value from amount and fiat rates.\n *\n * @param amount - Amount to convert.\n * @param decimals - Token decimals.\n * @param fiatRateFiat - Fiat rate.\n * @param fiatRateUsd - USD rate.\n * @returns Fiat value.\n */\nfunction calculateFiatValue(\n amount: string,\n decimals: number,\n fiatRateFiat: string,\n fiatRateUsd: string,\n): FiatValue {\n const amountHuman = new BigNumber(amount).shiftedBy(-decimals);\n const usd = amountHuman.multipliedBy(fiatRateUsd).toString(10);\n const fiat = amountHuman.multipliedBy(fiatRateFiat).toString(10);\n\n return { fiat, usd };\n}\n\n/**\n * Calculate the source network fee for a bridge quote.\n *\n * @param quote - Bridge quote response.\n * @param messenger - Controller messenger.\n * @returns Estimated gas cost for the source network.\n */\nfunction calculateSourceNetworkFee(\n quote: TransactionPayBridgeQuote,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { approval, trade } = quote;\n\n const approvalCost = approval\n ? calculateTransactionCost(approval as TxData, messenger)\n : { fiat: '0', usd: '0' };\n\n const tradeCost = calculateTransactionCost(trade as TxData, messenger);\n\n return {\n fiat: new BigNumber(approvalCost.fiat).plus(tradeCost.fiat).toString(10),\n usd: new BigNumber(approvalCost.usd).plus(tradeCost.usd).toString(10),\n };\n}\n\n/**\n * Calculate the source gas cost for a transaction.\n *\n * @param transaction - Transaction parameters.\n * @param messenger - Controller messenger\n * @returns Estimated gas cost for a bridge transaction.\n */\nfunction calculateTransactionCost(\n transaction: TxData,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { effectiveGas, gasLimit } = transaction;\n\n return calculateGasCost({\n ...transaction,\n gas: effectiveGas || gasLimit || '0x0',\n messenger,\n });\n}\n"]}
1
+ {"version":3,"file":"bridge-quotes.cjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n FeatureId,\n type GenericQuoteRequest,\n} from '@metamask/bridge-controller';\nimport type { TxData } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport { toChecksumHexAddress, toHex } from '@metamask/controller-utils';\nimport { TransactionType } from '@metamask/transaction-controller';\nimport type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\nimport { orderBy } from 'lodash';\n\nimport type {\n BridgeFeatureFlags,\n TransactionPayBridgeQuote,\n BridgeQuoteRequest,\n} from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetBatchRequest,\n PayStrategyGetQuotesRequest,\n PayStrategyGetRefreshIntervalRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getTokenFiatRate } from '../../utils/token';\n\nconst ERROR_MESSAGE_NO_QUOTES = 'No quotes found';\nconst ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM = 'All quotes under minimum';\nconst ATTEMPTS_MAX_DEFAULT = 5;\nconst BUFFER_INITIAL_DEFAULT = 0.04;\nconst BUFFER_STEP_DEFAULT = 0.04;\nconst BUFFER_SUBSEQUENT_DEFAULT = 0.05;\nconst SLIPPAGE_DEFAULT = 0.005;\n\nconst FEATURE_ID_BY_TRANSACTION_TYPE = new Map<TransactionType, FeatureId>([\n [TransactionType.perpsDeposit, FeatureId.PERPS],\n]);\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\n/**\n * Fetch bridge quotes for multiple requests.\n *\n * @param request - Request object.\n * @returns An array of bridge quotes.\n */\nexport async function getBridgeQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<TransactionPayBridgeQuote>[]> {\n log('Fetching quotes', request);\n\n const { requests, messenger, transaction } = request;\n\n const finalRequests = getFinalRequests(requests, messenger);\n\n const quotes = await Promise.all(\n finalRequests.map((r, index) =>\n getSufficientSingleBridgeQuote(r, index, request),\n ),\n );\n\n return quotes.map((quote, index) =>\n normalizeQuote(quote, finalRequests[index], messenger, transaction),\n );\n}\n\n/**\n * Get bridge batch transactions if needed by the quotes.\n *\n * @param request - Request object.\n * @returns Array of batch transactions.\n */\nexport async function getBridgeBatchTransactions(\n request: PayStrategyGetBatchRequest<TransactionPayBridgeQuote>,\n): Promise<BatchTransaction[]> {\n const { quotes } = request;\n const firstQuote = quotes[0]?.original?.quote;\n\n if (firstQuote?.srcChainId !== firstQuote?.destChainId) {\n log('No batch transactions needed for bridge quotes');\n return [];\n }\n\n return quotes\n .map((q) => q.original)\n .flatMap((quote) => {\n const result = [];\n\n if (quote.approval) {\n result.push({\n ...getBatchTransaction(quote.approval),\n type: TransactionType.swapApproval,\n });\n }\n\n result.push({\n ...getBatchTransaction(quote.trade as TxData),\n type: TransactionType.swap,\n });\n\n return result;\n });\n}\n\n/**\n * Get the refresh interval for bridge quotes.\n *\n * @param request - Request object.\n * @returns Refresh interval in milliseconds.\n */\nexport function getBridgeRefreshInterval(\n request: PayStrategyGetRefreshIntervalRequest,\n): number | undefined {\n const { chainId, messenger } = request;\n\n const bridgeFeatureFlags = messenger.call(\n 'RemoteFeatureFlagController:getState',\n ).remoteFeatureFlags.bridgeConfigV2 as BridgeFeatureFlags | undefined;\n\n const chainInterval =\n bridgeFeatureFlags?.chains?.[parseInt(chainId, 16)]?.refreshRate;\n\n const globalInterval = bridgeFeatureFlags?.refreshRate;\n\n return chainInterval ?? globalInterval;\n}\n\n/**\n * Get a fresh quote for a previously fetched bridge quote to avoid expiration.\n *\n * @param quote - Original quote.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Fresh quote response.\n */\nexport async function refreshQuote(\n quote: TransactionPayQuote<TransactionPayBridgeQuote>,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const newQuote = await getSingleBridgeQuote(\n { ...quote.original.request, attemptsMax: 1 },\n messenger,\n transaction,\n );\n\n log('Refreshed quote', { old: quote, new: newQuote });\n\n return newQuote;\n}\n\n/**\n * Convert a quote trade or approval to a batch transaction.\n *\n * @param transaction - Quote trade or approval.\n * @returns Batch transaction.\n */\nfunction getBatchTransaction(transaction: TxData): BatchTransaction {\n const data = transaction.data as Hex;\n const gas = transaction.gasLimit ? toHex(transaction.gasLimit) : undefined;\n const to = transaction.to as Hex;\n const value = transaction.value as Hex;\n\n return {\n data,\n gas,\n isAfter: false,\n to,\n value,\n };\n}\n\n/**\n * Retry fetching a single bridge quote until it meets the minimum target amount.\n *\n * @param quoteRequest - Original quote request.\n * @param index - Index of the request in the array.\n * @param request - Full quotes request.\n * @returns The sufficient bridge quote.\n */\nasync function getSufficientSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n index: number,\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayBridgeQuote> {\n const {\n attemptsMax,\n bufferInitial,\n bufferStep,\n bufferSubsequent,\n sourceBalanceRaw,\n sourceTokenAmount,\n targetAmountMinimum,\n targetTokenAddress,\n } = quoteRequest;\n\n const sourceAmountValue = new BigNumber(sourceTokenAmount);\n const buffer = index === 0 ? bufferInitial : bufferSubsequent;\n const originalSourceAmount = sourceAmountValue.div(1 + buffer);\n const start = Date.now();\n\n let currentSourceAmount = sourceTokenAmount;\n\n for (let i = 0; i < attemptsMax; i++) {\n const currentRequest = {\n ...quoteRequest,\n sourceTokenAmount: currentSourceAmount,\n };\n\n try {\n log('Attempt', {\n attempt: i + 1,\n attemptsMax,\n bufferInitial,\n bufferStep,\n currentSourceAmount,\n target: targetTokenAddress,\n });\n\n const result = await getSingleBridgeQuote(\n currentRequest,\n request.messenger,\n request.transaction,\n );\n\n const dust = new BigNumber(result.quote.minDestTokenAmount)\n .minus(targetAmountMinimum)\n .toString(10);\n\n log('Found valid quote', {\n attempt: i + 1,\n target: targetTokenAddress,\n targetAmount: result.quote.minDestTokenAmount,\n goalAmount: targetAmountMinimum,\n dust,\n quote: result,\n });\n\n return {\n ...result,\n metrics: {\n attempts: i + 1,\n buffer: buffer + bufferStep * i,\n latency: Date.now() - start,\n },\n };\n } catch (error) {\n const errorMessage = (error as { message: string }).message;\n\n if (errorMessage !== ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM) {\n throw error;\n }\n }\n\n if (\n new BigNumber(currentSourceAmount).isGreaterThanOrEqualTo(\n sourceBalanceRaw,\n )\n ) {\n log('Reached balance limit', targetTokenAddress);\n break;\n }\n\n const newSourceAmount = originalSourceAmount.multipliedBy(\n 1 + buffer + bufferStep * (i + 1),\n );\n\n currentSourceAmount = newSourceAmount.isLessThan(sourceBalanceRaw)\n ? newSourceAmount.toFixed(0)\n : sourceBalanceRaw;\n }\n\n log('All attempts failed', targetTokenAddress);\n\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n}\n\n/**\n * Fetch a single bridge quote.\n *\n * @param quoteRequest - Quote request parameters.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns The bridge quote.\n */\nasync function getSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const {\n from,\n slippage,\n sourceChainId,\n sourceTokenAddress,\n sourceTokenAmount,\n targetChainId,\n targetTokenAddress,\n } = quoteRequest;\n\n const { type } = transaction;\n const featureId = FEATURE_ID_BY_TRANSACTION_TYPE.get(type as TransactionType);\n\n const bridgeRequest: GenericQuoteRequest = {\n destChainId: targetChainId,\n destTokenAddress: toChecksumHexAddress(targetTokenAddress),\n destWalletAddress: from,\n gasIncluded: false,\n gasIncluded7702: false,\n insufficientBal: false,\n slippage: slippage * 100,\n srcChainId: sourceChainId,\n srcTokenAddress: toChecksumHexAddress(sourceTokenAddress),\n srcTokenAmount: sourceTokenAmount,\n walletAddress: from,\n };\n\n const quotes = await messenger.call(\n 'BridgeController:fetchQuotes',\n bridgeRequest,\n undefined,\n featureId,\n );\n\n if (!quotes.length) {\n throw new Error(ERROR_MESSAGE_NO_QUOTES);\n }\n\n const result = getBestQuote(quotes, quoteRequest);\n\n return {\n ...result,\n request: quoteRequest,\n };\n}\n\n/**\n * Select the best quote from a list of quotes.\n *\n * @param quotes - List of quotes.\n * @param request - Original quote request.\n * @returns The best quote.\n */\nfunction getBestQuote(\n quotes: QuoteResponse[],\n request: BridgeQuoteRequest,\n): QuoteResponse {\n const fastestQuotes = orderBy(\n quotes,\n (quote) => quote.estimatedProcessingTimeInSeconds,\n 'asc',\n ).slice(0, 3);\n\n const quotesOverMinimumTarget = fastestQuotes.filter((quote) =>\n new BigNumber(quote.quote.minDestTokenAmount).isGreaterThanOrEqualTo(\n request.targetAmountMinimum,\n ),\n );\n\n log('Finding best quote', {\n allQuotes: quotes,\n fastestQuotes,\n quotesOverMinimumTarget,\n });\n\n if (!quotesOverMinimumTarget.length) {\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n }\n\n const cheapestQuote = orderBy(\n quotesOverMinimumTarget,\n (quote) => BigNumber(quote.quote.minDestTokenAmount).toNumber(),\n 'desc',\n )[0];\n\n return cheapestQuote;\n}\n\n/**\n * Get the final bridge quote requests.\n * Subtracts subsequent source amounts from the available balance.\n *\n * @param requests - List of bridge quote requests.\n * @param messenger - Controller messenger.\n * @returns The final bridge quote requests.\n */\nfunction getFinalRequests(\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n): BridgeQuoteRequest[] {\n const featureFlags = getFeatureFlags(messenger);\n\n return requests\n .map((request) => ({ ...request, ...featureFlags }))\n .map((request, index) => {\n const isFirstRequest = index === 0;\n const attemptsMax = isFirstRequest ? request.attemptsMax : 1;\n\n const sourceBalanceRaw = requests\n .reduce((acc, value, j) => {\n const isSameSource =\n value.sourceTokenAddress.toLowerCase() ===\n request.sourceTokenAddress.toLowerCase() &&\n value.sourceChainId === request.sourceChainId;\n\n if (isFirstRequest && j > index && isSameSource) {\n return acc.minus(value.sourceTokenAmount);\n }\n\n return acc;\n }, new BigNumber(request.sourceBalanceRaw))\n .toFixed(0);\n\n return {\n ...request,\n attemptsMax,\n sourceBalanceRaw,\n };\n });\n}\n\n/**\n * Get feature flags for bridge quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlags = messenger.call('RemoteFeatureFlagController:getState')\n .remoteFeatureFlags.confirmation_pay as Record<string, number> | undefined;\n\n return {\n attemptsMax: featureFlags?.attemptsMax ?? ATTEMPTS_MAX_DEFAULT,\n bufferInitial: featureFlags?.bufferInitial ?? BUFFER_INITIAL_DEFAULT,\n bufferStep: featureFlags?.bufferStep ?? BUFFER_STEP_DEFAULT,\n bufferSubsequent:\n featureFlags?.bufferSubsequent ?? BUFFER_SUBSEQUENT_DEFAULT,\n slippage: featureFlags?.slippage ?? SLIPPAGE_DEFAULT,\n };\n}\n\n/**\n * Convert a bridge specific quote response to a normalized transaction pay quote.\n *\n * @param quote - Bridge quote response.\n * @param request - Request\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Normalized transaction pay quote.\n */\nfunction normalizeQuote(\n quote: TransactionPayBridgeQuote,\n request: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): TransactionPayQuote<TransactionPayBridgeQuote> {\n const targetFiatRate = getTokenFiatRate(\n messenger,\n quote.quote.destAsset.address as Hex,\n toHex(quote.quote.destChainId),\n );\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n quote.quote.srcAsset.address as Hex,\n toHex(quote.quote.srcChainId),\n );\n\n if (sourceFiatRate === undefined || targetFiatRate === undefined) {\n throw new Error('Fiat rate not found for source or target token');\n }\n\n const targetAmountMinimumFiat = calculateFiatValue(\n quote.quote.minDestTokenAmount,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const sourceAmountFiat = calculateFiatValue(\n quote.quote.srcTokenAmount,\n quote.quote.srcAsset.decimals,\n sourceFiatRate.fiatRate,\n sourceFiatRate.usdRate,\n );\n\n const targetAmountGoal = calculateFiatValue(\n request.targetAmountMinimum,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const targetNetwork = calculateTransactionGasCost(transaction, messenger);\n const sourceNetwork = calculateSourceNetworkFee(quote, messenger);\n\n return {\n estimatedDuration: quote.estimatedProcessingTimeInSeconds,\n dust: {\n fiat: new BigNumber(targetAmountMinimumFiat.fiat)\n .minus(targetAmountGoal.fiat)\n .toString(10),\n usd: new BigNumber(targetAmountMinimumFiat.usd)\n .minus(targetAmountGoal.usd)\n .toString(10),\n },\n fees: {\n provider: {\n fiat: new BigNumber(sourceAmountFiat.fiat)\n .minus(targetAmountMinimumFiat.fiat)\n .toString(10),\n usd: new BigNumber(sourceAmountFiat.usd)\n .minus(targetAmountMinimumFiat.usd)\n .toString(10),\n },\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Bridge,\n };\n}\n\n/**\n * Calculate fiat value from amount and fiat rates.\n *\n * @param amount - Amount to convert.\n * @param decimals - Token decimals.\n * @param fiatRateFiat - Fiat rate.\n * @param fiatRateUsd - USD rate.\n * @returns Fiat value.\n */\nfunction calculateFiatValue(\n amount: string,\n decimals: number,\n fiatRateFiat: string,\n fiatRateUsd: string,\n): FiatValue {\n const amountHuman = new BigNumber(amount).shiftedBy(-decimals);\n const usd = amountHuman.multipliedBy(fiatRateUsd).toString(10);\n const fiat = amountHuman.multipliedBy(fiatRateFiat).toString(10);\n\n return { fiat, usd };\n}\n\n/**\n * Calculate the source network fee for a bridge quote.\n *\n * @param quote - Bridge quote response.\n * @param messenger - Controller messenger.\n * @returns Estimated gas cost for the source network.\n */\nfunction calculateSourceNetworkFee(\n quote: TransactionPayBridgeQuote,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { approval, trade } = quote;\n\n const approvalCost = approval\n ? calculateTransactionCost(approval as TxData, messenger)\n : { fiat: '0', usd: '0' };\n\n const tradeCost = calculateTransactionCost(trade as TxData, messenger);\n\n return {\n fiat: new BigNumber(approvalCost.fiat).plus(tradeCost.fiat).toString(10),\n usd: new BigNumber(approvalCost.usd).plus(tradeCost.usd).toString(10),\n };\n}\n\n/**\n * Calculate the source gas cost for a transaction.\n *\n * @param transaction - Transaction parameters.\n * @param messenger - Controller messenger\n * @returns Estimated gas cost for a bridge transaction.\n */\nfunction calculateTransactionCost(\n transaction: TxData,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { effectiveGas, gasLimit } = transaction;\n\n return calculateGasCost({\n ...transaction,\n gas: effectiveGas || gasLimit || '0x0',\n messenger,\n });\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-quotes.d.cts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,gBAAgB,EAAE,yCAAyC;AACzE,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AAMxE,OAAO,KAAK,EAEV,yBAAyB,EAE1B,oBAAgB;AAGjB,OAAO,KAAK,EAEV,0BAA0B,EAC1B,2BAA2B,EAC3B,oCAAoC,EAEpC,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAkBrB;;;;;GAKG;AACH,wBAAsB,eAAe,CACnC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC,CAqB3D;AAED;;;;;GAKG;AACH,wBAAsB,0BAA0B,CAC9C,OAAO,EAAE,0BAA0B,CAAC,yBAAyB,CAAC,GAC7D,OAAO,CAAC,gBAAgB,EAAE,CAAC,CA4B7B;AAED;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,OAAO,EAAE,oCAAoC,GAC5C,MAAM,GAAG,SAAS,CAapB;AAED;;;;;;;GAOG;AACH,wBAAsB,YAAY,CAChC,KAAK,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EACrD,SAAS,EAAE,iCAAiC,EAC5C,WAAW,EAAE,eAAe,GAC3B,OAAO,CAAC,yBAAyB,CAAC,CAUpC"}
1
+ {"version":3,"file":"bridge-quotes.d.cts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,gBAAgB,EAAE,yCAAyC;AACzE,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AAMxE,OAAO,KAAK,EAEV,yBAAyB,EAE1B,oBAAgB;AAGjB,OAAO,KAAK,EAEV,0BAA0B,EAC1B,2BAA2B,EAC3B,oCAAoC,EAEpC,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAkBrB;;;;;GAKG;AACH,wBAAsB,eAAe,CACnC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC,CAgB3D;AAED;;;;;GAKG;AACH,wBAAsB,0BAA0B,CAC9C,OAAO,EAAE,0BAA0B,CAAC,yBAAyB,CAAC,GAC7D,OAAO,CAAC,gBAAgB,EAAE,CAAC,CA4B7B;AAED;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,OAAO,EAAE,oCAAoC,GAC5C,MAAM,GAAG,SAAS,CAapB;AAED;;;;;;;GAOG;AACH,wBAAsB,YAAY,CAChC,KAAK,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EACrD,SAAS,EAAE,iCAAiC,EAC5C,WAAW,EAAE,eAAe,GAC3B,OAAO,CAAC,yBAAyB,CAAC,CAUpC"}
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-quotes.d.mts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,gBAAgB,EAAE,yCAAyC;AACzE,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AAMxE,OAAO,KAAK,EAEV,yBAAyB,EAE1B,oBAAgB;AAGjB,OAAO,KAAK,EAEV,0BAA0B,EAC1B,2BAA2B,EAC3B,oCAAoC,EAEpC,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAkBrB;;;;;GAKG;AACH,wBAAsB,eAAe,CACnC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC,CAqB3D;AAED;;;;;GAKG;AACH,wBAAsB,0BAA0B,CAC9C,OAAO,EAAE,0BAA0B,CAAC,yBAAyB,CAAC,GAC7D,OAAO,CAAC,gBAAgB,EAAE,CAAC,CA4B7B;AAED;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,OAAO,EAAE,oCAAoC,GAC5C,MAAM,GAAG,SAAS,CAapB;AAED;;;;;;;GAOG;AACH,wBAAsB,YAAY,CAChC,KAAK,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EACrD,SAAS,EAAE,iCAAiC,EAC5C,WAAW,EAAE,eAAe,GAC3B,OAAO,CAAC,yBAAyB,CAAC,CAUpC"}
1
+ {"version":3,"file":"bridge-quotes.d.mts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,gBAAgB,EAAE,yCAAyC;AACzE,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AAMxE,OAAO,KAAK,EAEV,yBAAyB,EAE1B,oBAAgB;AAGjB,OAAO,KAAK,EAEV,0BAA0B,EAC1B,2BAA2B,EAC3B,oCAAoC,EAEpC,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAkBrB;;;;;GAKG;AACH,wBAAsB,eAAe,CACnC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC,CAgB3D;AAED;;;;;GAKG;AACH,wBAAsB,0BAA0B,CAC9C,OAAO,EAAE,0BAA0B,CAAC,yBAAyB,CAAC,GAC7D,OAAO,CAAC,gBAAgB,EAAE,CAAC,CA4B7B;AAED;;;;;GAKG;AACH,wBAAgB,wBAAwB,CACtC,OAAO,EAAE,oCAAoC,GAC5C,MAAM,GAAG,SAAS,CAapB;AAED;;;;;;;GAOG;AACH,wBAAsB,YAAY,CAChC,KAAK,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EACrD,SAAS,EAAE,iCAAiC,EAC5C,WAAW,EAAE,eAAe,GAC3B,OAAO,CAAC,yBAAyB,CAAC,CAUpC"}
@@ -29,15 +29,9 @@ const log = createModuleLogger(projectLogger, 'bridge-strategy');
29
29
  export async function getBridgeQuotes(request) {
30
30
  log('Fetching quotes', request);
31
31
  const { requests, messenger, transaction } = request;
32
- try {
33
- const finalRequests = getFinalRequests(requests, messenger);
34
- const quotes = await Promise.all(finalRequests.map((r, index) => getSufficientSingleBridgeQuote(r, index, request)));
35
- return quotes.map((quote, index) => normalizeQuote(quote, finalRequests[index], messenger, transaction));
36
- }
37
- catch (error) {
38
- log('Error fetching quotes', { error });
39
- throw new Error(`Failed to fetch bridge quotes: ${String(error)}`);
40
- }
32
+ const finalRequests = getFinalRequests(requests, messenger);
33
+ const quotes = await Promise.all(finalRequests.map((r, index) => getSufficientSingleBridgeQuote(r, index, request)));
34
+ return quotes.map((quote, index) => normalizeQuote(quote, finalRequests[index], messenger, transaction));
41
35
  }
42
36
  /**
43
37
  * Get bridge batch transactions if needed by the quotes.
@@ -299,7 +293,7 @@ function getFeatureFlags(messenger) {
299
293
  */
300
294
  function normalizeQuote(quote, request, messenger, transaction) {
301
295
  const targetFiatRate = getTokenFiatRate(messenger, quote.quote.destAsset.address, toHex(quote.quote.destChainId));
302
- const sourceFiatRate = getTokenFiatRate(messenger, request.sourceTokenAddress, toHex(quote.quote.srcChainId));
296
+ const sourceFiatRate = getTokenFiatRate(messenger, quote.quote.srcAsset.address, toHex(quote.quote.srcChainId));
303
297
  if (sourceFiatRate === undefined || targetFiatRate === undefined) {
304
298
  throw new Error('Fiat rate not found for source or target token');
305
299
  }
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-quotes.mjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n FeatureId,\n type GenericQuoteRequest,\n} from '@metamask/bridge-controller';\nimport type { TxData } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport { toChecksumHexAddress, toHex } from '@metamask/controller-utils';\nimport { TransactionType } from '@metamask/transaction-controller';\nimport type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\nimport { orderBy } from 'lodash';\n\nimport type {\n BridgeFeatureFlags,\n TransactionPayBridgeQuote,\n BridgeQuoteRequest,\n} from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetBatchRequest,\n PayStrategyGetQuotesRequest,\n PayStrategyGetRefreshIntervalRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getTokenFiatRate } from '../../utils/token';\n\nconst ERROR_MESSAGE_NO_QUOTES = 'No quotes found';\nconst ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM = 'All quotes under minimum';\nconst ATTEMPTS_MAX_DEFAULT = 5;\nconst BUFFER_INITIAL_DEFAULT = 0.04;\nconst BUFFER_STEP_DEFAULT = 0.04;\nconst BUFFER_SUBSEQUENT_DEFAULT = 0.05;\nconst SLIPPAGE_DEFAULT = 0.005;\n\nconst FEATURE_ID_BY_TRANSACTION_TYPE = new Map<TransactionType, FeatureId>([\n [TransactionType.perpsDeposit, FeatureId.PERPS],\n]);\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\n/**\n * Fetch bridge quotes for multiple requests.\n *\n * @param request - Request object.\n * @returns An array of bridge quotes.\n */\nexport async function getBridgeQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<TransactionPayBridgeQuote>[]> {\n log('Fetching quotes', request);\n\n const { requests, messenger, transaction } = request;\n\n try {\n const finalRequests = getFinalRequests(requests, messenger);\n\n const quotes = await Promise.all(\n finalRequests.map((r, index) =>\n getSufficientSingleBridgeQuote(r, index, request),\n ),\n );\n\n return quotes.map((quote, index) =>\n normalizeQuote(quote, finalRequests[index], messenger, transaction),\n );\n } catch (error) {\n log('Error fetching quotes', { error });\n throw new Error(`Failed to fetch bridge quotes: ${String(error)}`);\n }\n}\n\n/**\n * Get bridge batch transactions if needed by the quotes.\n *\n * @param request - Request object.\n * @returns Array of batch transactions.\n */\nexport async function getBridgeBatchTransactions(\n request: PayStrategyGetBatchRequest<TransactionPayBridgeQuote>,\n): Promise<BatchTransaction[]> {\n const { quotes } = request;\n const firstQuote = quotes[0]?.original?.quote;\n\n if (firstQuote?.srcChainId !== firstQuote?.destChainId) {\n log('No batch transactions needed for bridge quotes');\n return [];\n }\n\n return quotes\n .map((q) => q.original)\n .flatMap((quote) => {\n const result = [];\n\n if (quote.approval) {\n result.push({\n ...getBatchTransaction(quote.approval),\n type: TransactionType.swapApproval,\n });\n }\n\n result.push({\n ...getBatchTransaction(quote.trade as TxData),\n type: TransactionType.swap,\n });\n\n return result;\n });\n}\n\n/**\n * Get the refresh interval for bridge quotes.\n *\n * @param request - Request object.\n * @returns Refresh interval in milliseconds.\n */\nexport function getBridgeRefreshInterval(\n request: PayStrategyGetRefreshIntervalRequest,\n): number | undefined {\n const { chainId, messenger } = request;\n\n const bridgeFeatureFlags = messenger.call(\n 'RemoteFeatureFlagController:getState',\n ).remoteFeatureFlags.bridgeConfigV2 as BridgeFeatureFlags | undefined;\n\n const chainInterval =\n bridgeFeatureFlags?.chains?.[parseInt(chainId, 16)]?.refreshRate;\n\n const globalInterval = bridgeFeatureFlags?.refreshRate;\n\n return chainInterval ?? globalInterval;\n}\n\n/**\n * Get a fresh quote for a previously fetched bridge quote to avoid expiration.\n *\n * @param quote - Original quote.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Fresh quote response.\n */\nexport async function refreshQuote(\n quote: TransactionPayQuote<TransactionPayBridgeQuote>,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const newQuote = await getSingleBridgeQuote(\n { ...quote.original.request, attemptsMax: 1 },\n messenger,\n transaction,\n );\n\n log('Refreshed quote', { old: quote, new: newQuote });\n\n return newQuote;\n}\n\n/**\n * Convert a quote trade or approval to a batch transaction.\n *\n * @param transaction - Quote trade or approval.\n * @returns Batch transaction.\n */\nfunction getBatchTransaction(transaction: TxData): BatchTransaction {\n const data = transaction.data as Hex;\n const gas = transaction.gasLimit ? toHex(transaction.gasLimit) : undefined;\n const to = transaction.to as Hex;\n const value = transaction.value as Hex;\n\n return {\n data,\n gas,\n isAfter: false,\n to,\n value,\n };\n}\n\n/**\n * Retry fetching a single bridge quote until it meets the minimum target amount.\n *\n * @param quoteRequest - Original quote request.\n * @param index - Index of the request in the array.\n * @param request - Full quotes request.\n * @returns The sufficient bridge quote.\n */\nasync function getSufficientSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n index: number,\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayBridgeQuote> {\n const {\n attemptsMax,\n bufferInitial,\n bufferStep,\n bufferSubsequent,\n sourceBalanceRaw,\n sourceTokenAmount,\n targetAmountMinimum,\n targetTokenAddress,\n } = quoteRequest;\n\n const sourceAmountValue = new BigNumber(sourceTokenAmount);\n const buffer = index === 0 ? bufferInitial : bufferSubsequent;\n const originalSourceAmount = sourceAmountValue.div(1 + buffer);\n const start = Date.now();\n\n let currentSourceAmount = sourceTokenAmount;\n\n for (let i = 0; i < attemptsMax; i++) {\n const currentRequest = {\n ...quoteRequest,\n sourceTokenAmount: currentSourceAmount,\n };\n\n try {\n log('Attempt', {\n attempt: i + 1,\n attemptsMax,\n bufferInitial,\n bufferStep,\n currentSourceAmount,\n target: targetTokenAddress,\n });\n\n const result = await getSingleBridgeQuote(\n currentRequest,\n request.messenger,\n request.transaction,\n );\n\n const dust = new BigNumber(result.quote.minDestTokenAmount)\n .minus(targetAmountMinimum)\n .toString(10);\n\n log('Found valid quote', {\n attempt: i + 1,\n target: targetTokenAddress,\n targetAmount: result.quote.minDestTokenAmount,\n goalAmount: targetAmountMinimum,\n dust,\n quote: result,\n });\n\n return {\n ...result,\n metrics: {\n attempts: i + 1,\n buffer: buffer + bufferStep * i,\n latency: Date.now() - start,\n },\n };\n } catch (error) {\n const errorMessage = (error as { message: string }).message;\n\n if (errorMessage !== ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM) {\n throw error;\n }\n }\n\n if (\n new BigNumber(currentSourceAmount).isGreaterThanOrEqualTo(\n sourceBalanceRaw,\n )\n ) {\n log('Reached balance limit', targetTokenAddress);\n break;\n }\n\n const newSourceAmount = originalSourceAmount.multipliedBy(\n 1 + buffer + bufferStep * (i + 1),\n );\n\n currentSourceAmount = newSourceAmount.isLessThan(sourceBalanceRaw)\n ? newSourceAmount.toFixed(0)\n : sourceBalanceRaw;\n }\n\n log('All attempts failed', targetTokenAddress);\n\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n}\n\n/**\n * Fetch a single bridge quote.\n *\n * @param quoteRequest - Quote request parameters.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns The bridge quote.\n */\nasync function getSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const {\n from,\n slippage,\n sourceChainId,\n sourceTokenAddress,\n sourceTokenAmount,\n targetChainId,\n targetTokenAddress,\n } = quoteRequest;\n\n const { type } = transaction;\n const featureId = FEATURE_ID_BY_TRANSACTION_TYPE.get(type as TransactionType);\n\n const bridgeRequest: GenericQuoteRequest = {\n destChainId: targetChainId,\n destTokenAddress: toChecksumHexAddress(targetTokenAddress),\n destWalletAddress: from,\n gasIncluded: false,\n gasIncluded7702: false,\n insufficientBal: false,\n slippage: slippage * 100,\n srcChainId: sourceChainId,\n srcTokenAddress: toChecksumHexAddress(sourceTokenAddress),\n srcTokenAmount: sourceTokenAmount,\n walletAddress: from,\n };\n\n const quotes = await messenger.call(\n 'BridgeController:fetchQuotes',\n bridgeRequest,\n undefined,\n featureId,\n );\n\n if (!quotes.length) {\n throw new Error(ERROR_MESSAGE_NO_QUOTES);\n }\n\n const result = getBestQuote(quotes, quoteRequest);\n\n return {\n ...result,\n request: quoteRequest,\n };\n}\n\n/**\n * Select the best quote from a list of quotes.\n *\n * @param quotes - List of quotes.\n * @param request - Original quote request.\n * @returns The best quote.\n */\nfunction getBestQuote(\n quotes: QuoteResponse[],\n request: BridgeQuoteRequest,\n): QuoteResponse {\n const fastestQuotes = orderBy(\n quotes,\n (quote) => quote.estimatedProcessingTimeInSeconds,\n 'asc',\n ).slice(0, 3);\n\n const quotesOverMinimumTarget = fastestQuotes.filter((quote) =>\n new BigNumber(quote.quote.minDestTokenAmount).isGreaterThanOrEqualTo(\n request.targetAmountMinimum,\n ),\n );\n\n log('Finding best quote', {\n allQuotes: quotes,\n fastestQuotes,\n quotesOverMinimumTarget,\n });\n\n if (!quotesOverMinimumTarget.length) {\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n }\n\n const cheapestQuote = orderBy(\n quotesOverMinimumTarget,\n (quote) => BigNumber(quote.quote.minDestTokenAmount).toNumber(),\n 'desc',\n )[0];\n\n return cheapestQuote;\n}\n\n/**\n * Get the final bridge quote requests.\n * Subtracts subsequent source amounts from the available balance.\n *\n * @param requests - List of bridge quote requests.\n * @param messenger - Controller messenger.\n * @returns The final bridge quote requests.\n */\nfunction getFinalRequests(\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n): BridgeQuoteRequest[] {\n const featureFlags = getFeatureFlags(messenger);\n\n return requests\n .map((request) => ({ ...request, ...featureFlags }))\n .map((request, index) => {\n const isFirstRequest = index === 0;\n const attemptsMax = isFirstRequest ? request.attemptsMax : 1;\n\n const sourceBalanceRaw = requests\n .reduce((acc, value, j) => {\n const isSameSource =\n value.sourceTokenAddress.toLowerCase() ===\n request.sourceTokenAddress.toLowerCase() &&\n value.sourceChainId === request.sourceChainId;\n\n if (isFirstRequest && j > index && isSameSource) {\n return acc.minus(value.sourceTokenAmount);\n }\n\n return acc;\n }, new BigNumber(request.sourceBalanceRaw))\n .toFixed(0);\n\n return {\n ...request,\n attemptsMax,\n sourceBalanceRaw,\n };\n });\n}\n\n/**\n * Get feature flags for bridge quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlags = messenger.call('RemoteFeatureFlagController:getState')\n .remoteFeatureFlags.confirmation_pay as Record<string, number> | undefined;\n\n return {\n attemptsMax: featureFlags?.attemptsMax ?? ATTEMPTS_MAX_DEFAULT,\n bufferInitial: featureFlags?.bufferInitial ?? BUFFER_INITIAL_DEFAULT,\n bufferStep: featureFlags?.bufferStep ?? BUFFER_STEP_DEFAULT,\n bufferSubsequent:\n featureFlags?.bufferSubsequent ?? BUFFER_SUBSEQUENT_DEFAULT,\n slippage: featureFlags?.slippage ?? SLIPPAGE_DEFAULT,\n };\n}\n\n/**\n * Convert a bridge specific quote response to a normalized transaction pay quote.\n *\n * @param quote - Bridge quote response.\n * @param request - Request\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Normalized transaction pay quote.\n */\nfunction normalizeQuote(\n quote: TransactionPayBridgeQuote,\n request: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): TransactionPayQuote<TransactionPayBridgeQuote> {\n const targetFiatRate = getTokenFiatRate(\n messenger,\n quote.quote.destAsset.address as Hex,\n toHex(quote.quote.destChainId),\n );\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n request.sourceTokenAddress,\n toHex(quote.quote.srcChainId),\n );\n\n if (sourceFiatRate === undefined || targetFiatRate === undefined) {\n throw new Error('Fiat rate not found for source or target token');\n }\n\n const targetAmountMinimumFiat = calculateFiatValue(\n quote.quote.minDestTokenAmount,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const sourceAmountFiat = calculateFiatValue(\n quote.quote.srcTokenAmount,\n quote.quote.srcAsset.decimals,\n sourceFiatRate.fiatRate,\n sourceFiatRate.usdRate,\n );\n\n const targetAmountGoal = calculateFiatValue(\n request.targetAmountMinimum,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const targetNetwork = calculateTransactionGasCost(transaction, messenger);\n const sourceNetwork = calculateSourceNetworkFee(quote, messenger);\n\n return {\n estimatedDuration: quote.estimatedProcessingTimeInSeconds,\n dust: {\n fiat: new BigNumber(targetAmountMinimumFiat.fiat)\n .minus(targetAmountGoal.fiat)\n .toString(10),\n usd: new BigNumber(targetAmountMinimumFiat.usd)\n .minus(targetAmountGoal.usd)\n .toString(10),\n },\n fees: {\n provider: {\n fiat: new BigNumber(sourceAmountFiat.fiat)\n .minus(targetAmountMinimumFiat.fiat)\n .toString(10),\n usd: new BigNumber(sourceAmountFiat.usd)\n .minus(targetAmountMinimumFiat.usd)\n .toString(10),\n },\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Bridge,\n };\n}\n\n/**\n * Calculate fiat value from amount and fiat rates.\n *\n * @param amount - Amount to convert.\n * @param decimals - Token decimals.\n * @param fiatRateFiat - Fiat rate.\n * @param fiatRateUsd - USD rate.\n * @returns Fiat value.\n */\nfunction calculateFiatValue(\n amount: string,\n decimals: number,\n fiatRateFiat: string,\n fiatRateUsd: string,\n): FiatValue {\n const amountHuman = new BigNumber(amount).shiftedBy(-decimals);\n const usd = amountHuman.multipliedBy(fiatRateUsd).toString(10);\n const fiat = amountHuman.multipliedBy(fiatRateFiat).toString(10);\n\n return { fiat, usd };\n}\n\n/**\n * Calculate the source network fee for a bridge quote.\n *\n * @param quote - Bridge quote response.\n * @param messenger - Controller messenger.\n * @returns Estimated gas cost for the source network.\n */\nfunction calculateSourceNetworkFee(\n quote: TransactionPayBridgeQuote,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { approval, trade } = quote;\n\n const approvalCost = approval\n ? calculateTransactionCost(approval as TxData, messenger)\n : { fiat: '0', usd: '0' };\n\n const tradeCost = calculateTransactionCost(trade as TxData, messenger);\n\n return {\n fiat: new BigNumber(approvalCost.fiat).plus(tradeCost.fiat).toString(10),\n usd: new BigNumber(approvalCost.usd).plus(tradeCost.usd).toString(10),\n };\n}\n\n/**\n * Calculate the source gas cost for a transaction.\n *\n * @param transaction - Transaction parameters.\n * @param messenger - Controller messenger\n * @returns Estimated gas cost for a bridge transaction.\n */\nfunction calculateTransactionCost(\n transaction: TxData,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { effectiveGas, gasLimit } = transaction;\n\n return calculateGasCost({\n ...transaction,\n gas: effectiveGas || gasLimit || '0x0',\n messenger,\n });\n}\n"]}
1
+ {"version":3,"file":"bridge-quotes.mjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n FeatureId,\n type GenericQuoteRequest,\n} from '@metamask/bridge-controller';\nimport type { TxData } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport { toChecksumHexAddress, toHex } from '@metamask/controller-utils';\nimport { TransactionType } from '@metamask/transaction-controller';\nimport type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\nimport { orderBy } from 'lodash';\n\nimport type {\n BridgeFeatureFlags,\n TransactionPayBridgeQuote,\n BridgeQuoteRequest,\n} from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetBatchRequest,\n PayStrategyGetQuotesRequest,\n PayStrategyGetRefreshIntervalRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getTokenFiatRate } from '../../utils/token';\n\nconst ERROR_MESSAGE_NO_QUOTES = 'No quotes found';\nconst ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM = 'All quotes under minimum';\nconst ATTEMPTS_MAX_DEFAULT = 5;\nconst BUFFER_INITIAL_DEFAULT = 0.04;\nconst BUFFER_STEP_DEFAULT = 0.04;\nconst BUFFER_SUBSEQUENT_DEFAULT = 0.05;\nconst SLIPPAGE_DEFAULT = 0.005;\n\nconst FEATURE_ID_BY_TRANSACTION_TYPE = new Map<TransactionType, FeatureId>([\n [TransactionType.perpsDeposit, FeatureId.PERPS],\n]);\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\n/**\n * Fetch bridge quotes for multiple requests.\n *\n * @param request - Request object.\n * @returns An array of bridge quotes.\n */\nexport async function getBridgeQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<TransactionPayBridgeQuote>[]> {\n log('Fetching quotes', request);\n\n const { requests, messenger, transaction } = request;\n\n const finalRequests = getFinalRequests(requests, messenger);\n\n const quotes = await Promise.all(\n finalRequests.map((r, index) =>\n getSufficientSingleBridgeQuote(r, index, request),\n ),\n );\n\n return quotes.map((quote, index) =>\n normalizeQuote(quote, finalRequests[index], messenger, transaction),\n );\n}\n\n/**\n * Get bridge batch transactions if needed by the quotes.\n *\n * @param request - Request object.\n * @returns Array of batch transactions.\n */\nexport async function getBridgeBatchTransactions(\n request: PayStrategyGetBatchRequest<TransactionPayBridgeQuote>,\n): Promise<BatchTransaction[]> {\n const { quotes } = request;\n const firstQuote = quotes[0]?.original?.quote;\n\n if (firstQuote?.srcChainId !== firstQuote?.destChainId) {\n log('No batch transactions needed for bridge quotes');\n return [];\n }\n\n return quotes\n .map((q) => q.original)\n .flatMap((quote) => {\n const result = [];\n\n if (quote.approval) {\n result.push({\n ...getBatchTransaction(quote.approval),\n type: TransactionType.swapApproval,\n });\n }\n\n result.push({\n ...getBatchTransaction(quote.trade as TxData),\n type: TransactionType.swap,\n });\n\n return result;\n });\n}\n\n/**\n * Get the refresh interval for bridge quotes.\n *\n * @param request - Request object.\n * @returns Refresh interval in milliseconds.\n */\nexport function getBridgeRefreshInterval(\n request: PayStrategyGetRefreshIntervalRequest,\n): number | undefined {\n const { chainId, messenger } = request;\n\n const bridgeFeatureFlags = messenger.call(\n 'RemoteFeatureFlagController:getState',\n ).remoteFeatureFlags.bridgeConfigV2 as BridgeFeatureFlags | undefined;\n\n const chainInterval =\n bridgeFeatureFlags?.chains?.[parseInt(chainId, 16)]?.refreshRate;\n\n const globalInterval = bridgeFeatureFlags?.refreshRate;\n\n return chainInterval ?? globalInterval;\n}\n\n/**\n * Get a fresh quote for a previously fetched bridge quote to avoid expiration.\n *\n * @param quote - Original quote.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Fresh quote response.\n */\nexport async function refreshQuote(\n quote: TransactionPayQuote<TransactionPayBridgeQuote>,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const newQuote = await getSingleBridgeQuote(\n { ...quote.original.request, attemptsMax: 1 },\n messenger,\n transaction,\n );\n\n log('Refreshed quote', { old: quote, new: newQuote });\n\n return newQuote;\n}\n\n/**\n * Convert a quote trade or approval to a batch transaction.\n *\n * @param transaction - Quote trade or approval.\n * @returns Batch transaction.\n */\nfunction getBatchTransaction(transaction: TxData): BatchTransaction {\n const data = transaction.data as Hex;\n const gas = transaction.gasLimit ? toHex(transaction.gasLimit) : undefined;\n const to = transaction.to as Hex;\n const value = transaction.value as Hex;\n\n return {\n data,\n gas,\n isAfter: false,\n to,\n value,\n };\n}\n\n/**\n * Retry fetching a single bridge quote until it meets the minimum target amount.\n *\n * @param quoteRequest - Original quote request.\n * @param index - Index of the request in the array.\n * @param request - Full quotes request.\n * @returns The sufficient bridge quote.\n */\nasync function getSufficientSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n index: number,\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayBridgeQuote> {\n const {\n attemptsMax,\n bufferInitial,\n bufferStep,\n bufferSubsequent,\n sourceBalanceRaw,\n sourceTokenAmount,\n targetAmountMinimum,\n targetTokenAddress,\n } = quoteRequest;\n\n const sourceAmountValue = new BigNumber(sourceTokenAmount);\n const buffer = index === 0 ? bufferInitial : bufferSubsequent;\n const originalSourceAmount = sourceAmountValue.div(1 + buffer);\n const start = Date.now();\n\n let currentSourceAmount = sourceTokenAmount;\n\n for (let i = 0; i < attemptsMax; i++) {\n const currentRequest = {\n ...quoteRequest,\n sourceTokenAmount: currentSourceAmount,\n };\n\n try {\n log('Attempt', {\n attempt: i + 1,\n attemptsMax,\n bufferInitial,\n bufferStep,\n currentSourceAmount,\n target: targetTokenAddress,\n });\n\n const result = await getSingleBridgeQuote(\n currentRequest,\n request.messenger,\n request.transaction,\n );\n\n const dust = new BigNumber(result.quote.minDestTokenAmount)\n .minus(targetAmountMinimum)\n .toString(10);\n\n log('Found valid quote', {\n attempt: i + 1,\n target: targetTokenAddress,\n targetAmount: result.quote.minDestTokenAmount,\n goalAmount: targetAmountMinimum,\n dust,\n quote: result,\n });\n\n return {\n ...result,\n metrics: {\n attempts: i + 1,\n buffer: buffer + bufferStep * i,\n latency: Date.now() - start,\n },\n };\n } catch (error) {\n const errorMessage = (error as { message: string }).message;\n\n if (errorMessage !== ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM) {\n throw error;\n }\n }\n\n if (\n new BigNumber(currentSourceAmount).isGreaterThanOrEqualTo(\n sourceBalanceRaw,\n )\n ) {\n log('Reached balance limit', targetTokenAddress);\n break;\n }\n\n const newSourceAmount = originalSourceAmount.multipliedBy(\n 1 + buffer + bufferStep * (i + 1),\n );\n\n currentSourceAmount = newSourceAmount.isLessThan(sourceBalanceRaw)\n ? newSourceAmount.toFixed(0)\n : sourceBalanceRaw;\n }\n\n log('All attempts failed', targetTokenAddress);\n\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n}\n\n/**\n * Fetch a single bridge quote.\n *\n * @param quoteRequest - Quote request parameters.\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns The bridge quote.\n */\nasync function getSingleBridgeQuote(\n quoteRequest: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): Promise<TransactionPayBridgeQuote> {\n const {\n from,\n slippage,\n sourceChainId,\n sourceTokenAddress,\n sourceTokenAmount,\n targetChainId,\n targetTokenAddress,\n } = quoteRequest;\n\n const { type } = transaction;\n const featureId = FEATURE_ID_BY_TRANSACTION_TYPE.get(type as TransactionType);\n\n const bridgeRequest: GenericQuoteRequest = {\n destChainId: targetChainId,\n destTokenAddress: toChecksumHexAddress(targetTokenAddress),\n destWalletAddress: from,\n gasIncluded: false,\n gasIncluded7702: false,\n insufficientBal: false,\n slippage: slippage * 100,\n srcChainId: sourceChainId,\n srcTokenAddress: toChecksumHexAddress(sourceTokenAddress),\n srcTokenAmount: sourceTokenAmount,\n walletAddress: from,\n };\n\n const quotes = await messenger.call(\n 'BridgeController:fetchQuotes',\n bridgeRequest,\n undefined,\n featureId,\n );\n\n if (!quotes.length) {\n throw new Error(ERROR_MESSAGE_NO_QUOTES);\n }\n\n const result = getBestQuote(quotes, quoteRequest);\n\n return {\n ...result,\n request: quoteRequest,\n };\n}\n\n/**\n * Select the best quote from a list of quotes.\n *\n * @param quotes - List of quotes.\n * @param request - Original quote request.\n * @returns The best quote.\n */\nfunction getBestQuote(\n quotes: QuoteResponse[],\n request: BridgeQuoteRequest,\n): QuoteResponse {\n const fastestQuotes = orderBy(\n quotes,\n (quote) => quote.estimatedProcessingTimeInSeconds,\n 'asc',\n ).slice(0, 3);\n\n const quotesOverMinimumTarget = fastestQuotes.filter((quote) =>\n new BigNumber(quote.quote.minDestTokenAmount).isGreaterThanOrEqualTo(\n request.targetAmountMinimum,\n ),\n );\n\n log('Finding best quote', {\n allQuotes: quotes,\n fastestQuotes,\n quotesOverMinimumTarget,\n });\n\n if (!quotesOverMinimumTarget.length) {\n throw new Error(ERROR_MESSAGE_ALL_QUOTES_UNDER_MINIMUM);\n }\n\n const cheapestQuote = orderBy(\n quotesOverMinimumTarget,\n (quote) => BigNumber(quote.quote.minDestTokenAmount).toNumber(),\n 'desc',\n )[0];\n\n return cheapestQuote;\n}\n\n/**\n * Get the final bridge quote requests.\n * Subtracts subsequent source amounts from the available balance.\n *\n * @param requests - List of bridge quote requests.\n * @param messenger - Controller messenger.\n * @returns The final bridge quote requests.\n */\nfunction getFinalRequests(\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n): BridgeQuoteRequest[] {\n const featureFlags = getFeatureFlags(messenger);\n\n return requests\n .map((request) => ({ ...request, ...featureFlags }))\n .map((request, index) => {\n const isFirstRequest = index === 0;\n const attemptsMax = isFirstRequest ? request.attemptsMax : 1;\n\n const sourceBalanceRaw = requests\n .reduce((acc, value, j) => {\n const isSameSource =\n value.sourceTokenAddress.toLowerCase() ===\n request.sourceTokenAddress.toLowerCase() &&\n value.sourceChainId === request.sourceChainId;\n\n if (isFirstRequest && j > index && isSameSource) {\n return acc.minus(value.sourceTokenAmount);\n }\n\n return acc;\n }, new BigNumber(request.sourceBalanceRaw))\n .toFixed(0);\n\n return {\n ...request,\n attemptsMax,\n sourceBalanceRaw,\n };\n });\n}\n\n/**\n * Get feature flags for bridge quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlags = messenger.call('RemoteFeatureFlagController:getState')\n .remoteFeatureFlags.confirmation_pay as Record<string, number> | undefined;\n\n return {\n attemptsMax: featureFlags?.attemptsMax ?? ATTEMPTS_MAX_DEFAULT,\n bufferInitial: featureFlags?.bufferInitial ?? BUFFER_INITIAL_DEFAULT,\n bufferStep: featureFlags?.bufferStep ?? BUFFER_STEP_DEFAULT,\n bufferSubsequent:\n featureFlags?.bufferSubsequent ?? BUFFER_SUBSEQUENT_DEFAULT,\n slippage: featureFlags?.slippage ?? SLIPPAGE_DEFAULT,\n };\n}\n\n/**\n * Convert a bridge specific quote response to a normalized transaction pay quote.\n *\n * @param quote - Bridge quote response.\n * @param request - Request\n * @param messenger - Controller messenger.\n * @param transaction - Transaction metadata.\n * @returns Normalized transaction pay quote.\n */\nfunction normalizeQuote(\n quote: TransactionPayBridgeQuote,\n request: BridgeQuoteRequest,\n messenger: TransactionPayControllerMessenger,\n transaction: TransactionMeta,\n): TransactionPayQuote<TransactionPayBridgeQuote> {\n const targetFiatRate = getTokenFiatRate(\n messenger,\n quote.quote.destAsset.address as Hex,\n toHex(quote.quote.destChainId),\n );\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n quote.quote.srcAsset.address as Hex,\n toHex(quote.quote.srcChainId),\n );\n\n if (sourceFiatRate === undefined || targetFiatRate === undefined) {\n throw new Error('Fiat rate not found for source or target token');\n }\n\n const targetAmountMinimumFiat = calculateFiatValue(\n quote.quote.minDestTokenAmount,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const sourceAmountFiat = calculateFiatValue(\n quote.quote.srcTokenAmount,\n quote.quote.srcAsset.decimals,\n sourceFiatRate.fiatRate,\n sourceFiatRate.usdRate,\n );\n\n const targetAmountGoal = calculateFiatValue(\n request.targetAmountMinimum,\n quote.quote.destAsset.decimals,\n targetFiatRate.fiatRate,\n targetFiatRate.usdRate,\n );\n\n const targetNetwork = calculateTransactionGasCost(transaction, messenger);\n const sourceNetwork = calculateSourceNetworkFee(quote, messenger);\n\n return {\n estimatedDuration: quote.estimatedProcessingTimeInSeconds,\n dust: {\n fiat: new BigNumber(targetAmountMinimumFiat.fiat)\n .minus(targetAmountGoal.fiat)\n .toString(10),\n usd: new BigNumber(targetAmountMinimumFiat.usd)\n .minus(targetAmountGoal.usd)\n .toString(10),\n },\n fees: {\n provider: {\n fiat: new BigNumber(sourceAmountFiat.fiat)\n .minus(targetAmountMinimumFiat.fiat)\n .toString(10),\n usd: new BigNumber(sourceAmountFiat.usd)\n .minus(targetAmountMinimumFiat.usd)\n .toString(10),\n },\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Bridge,\n };\n}\n\n/**\n * Calculate fiat value from amount and fiat rates.\n *\n * @param amount - Amount to convert.\n * @param decimals - Token decimals.\n * @param fiatRateFiat - Fiat rate.\n * @param fiatRateUsd - USD rate.\n * @returns Fiat value.\n */\nfunction calculateFiatValue(\n amount: string,\n decimals: number,\n fiatRateFiat: string,\n fiatRateUsd: string,\n): FiatValue {\n const amountHuman = new BigNumber(amount).shiftedBy(-decimals);\n const usd = amountHuman.multipliedBy(fiatRateUsd).toString(10);\n const fiat = amountHuman.multipliedBy(fiatRateFiat).toString(10);\n\n return { fiat, usd };\n}\n\n/**\n * Calculate the source network fee for a bridge quote.\n *\n * @param quote - Bridge quote response.\n * @param messenger - Controller messenger.\n * @returns Estimated gas cost for the source network.\n */\nfunction calculateSourceNetworkFee(\n quote: TransactionPayBridgeQuote,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { approval, trade } = quote;\n\n const approvalCost = approval\n ? calculateTransactionCost(approval as TxData, messenger)\n : { fiat: '0', usd: '0' };\n\n const tradeCost = calculateTransactionCost(trade as TxData, messenger);\n\n return {\n fiat: new BigNumber(approvalCost.fiat).plus(tradeCost.fiat).toString(10),\n usd: new BigNumber(approvalCost.usd).plus(tradeCost.usd).toString(10),\n };\n}\n\n/**\n * Calculate the source gas cost for a transaction.\n *\n * @param transaction - Transaction parameters.\n * @param messenger - Controller messenger\n * @returns Estimated gas cost for a bridge transaction.\n */\nfunction calculateTransactionCost(\n transaction: TxData,\n messenger: TransactionPayControllerMessenger,\n): FiatValue {\n const { effectiveGas, gasLimit } = transaction;\n\n return calculateGasCost({\n ...transaction,\n gas: effectiveGas || gasLimit || '0x0',\n messenger,\n });\n}\n"]}
@@ -20,20 +20,14 @@ const log = (0, utils_1.createModuleLogger)(logger_1.projectLogger, 'relay-strat
20
20
  async function getRelayQuotes(request) {
21
21
  const { requests } = request;
22
22
  log('Fetching quotes', requests);
23
- try {
24
- const result = requests
25
- // Ignore gas fee token requests
26
- .filter((r) => r.targetAmountMinimum !== '0')
27
- .map((r) => normalizeRequest(r));
28
- const normalizedRequests = result.map((r) => r.request);
29
- const isSkipTransaction = result.some((r) => r.isSkipTransaction);
30
- log('Normalized requests', { normalizedRequests, isSkipTransaction });
31
- return await Promise.all(normalizedRequests.map((r) => getSingleQuote(r, isSkipTransaction, request)));
32
- }
33
- catch (error) {
34
- log('Error fetching quotes', { error });
35
- throw new Error(`Failed to fetch Relay quotes: ${String(error)}`);
36
- }
23
+ const result = requests
24
+ // Ignore gas fee token requests
25
+ .filter((r) => r.targetAmountMinimum !== '0')
26
+ .map((r) => normalizeRequest(r));
27
+ const normalizedRequests = result.map((r) => r.request);
28
+ const isSkipTransaction = result.some((r) => r.isSkipTransaction);
29
+ log('Normalized requests', { normalizedRequests, isSkipTransaction });
30
+ return await Promise.all(normalizedRequests.map((r) => getSingleQuote(r, isSkipTransaction, request)));
37
31
  }
38
32
  exports.getRelayQuotes = getRelayQuotes;
39
33
  /**
@@ -1 +1 @@
1
- {"version":3,"file":"relay-quotes.cjs","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import { successfulFetch, toHex } from '@metamask/controller-utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\n\nimport {\n ARBITRUM_USDC_ADDRESS,\n CHAIN_ID_ARBITRUM,\n CHAIN_ID_POLYGON,\n RELAY_URL_QUOTE,\n} from './constants';\nimport type { RelayQuote } from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { NATIVE_TOKEN_ADDRESS } from '../../constants';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetQuotesRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getNativeToken, getTokenFiatRate } from '../../utils/token';\n\nconst log = createModuleLogger(projectLogger, 'relay-strategy');\n\n/**\n * Fetches Relay quotes.\n *\n * @param request - Request object.\n * @returns Array of quotes.\n */\nexport async function getRelayQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>[]> {\n const { requests } = request;\n\n log('Fetching quotes', requests);\n\n try {\n const result = requests\n // Ignore gas fee token requests\n .filter((r) => r.targetAmountMinimum !== '0')\n .map((r) => normalizeRequest(r));\n\n const normalizedRequests = result.map((r) => r.request);\n const isSkipTransaction = result.some((r) => r.isSkipTransaction);\n\n log('Normalized requests', { normalizedRequests, isSkipTransaction });\n\n return await Promise.all(\n normalizedRequests.map((r) =>\n getSingleQuote(r, isSkipTransaction, request),\n ),\n );\n } catch (error) {\n log('Error fetching quotes', { error });\n throw new Error(`Failed to fetch Relay quotes: ${String(error)}`);\n }\n}\n\n/**\n * Fetches a single Relay quote.\n *\n * @param request - Quote request.\n * @param isSkipTransaction - Whether to skip the transaction.\n * @param fullRequest - Full quotes request.\n * @returns Single quote.\n */\nasync function getSingleQuote(\n request: QuoteRequest,\n isSkipTransaction: boolean,\n fullRequest: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>> {\n const { messenger } = fullRequest;\n\n try {\n const body = {\n amount: request.targetAmountMinimum,\n destinationChainId: Number(request.targetChainId),\n destinationCurrency: request.targetTokenAddress,\n originChainId: Number(request.sourceChainId),\n originCurrency: request.sourceTokenAddress,\n recipient: request.from,\n tradeType: 'EXPECTED_OUTPUT',\n user: request.from,\n };\n\n const url = getFeatureFlags(messenger).relayQuoteUrl;\n\n const response = await successfulFetch(url, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(body),\n });\n\n const quote = (await response.json()) as RelayQuote;\n quote.skipTransaction = isSkipTransaction;\n\n log('Fetched relay quote', { quote, url });\n\n return normalizeQuote(quote, request, fullRequest);\n } catch (e) {\n log('Error fetching relay quote', e);\n throw e;\n }\n}\n\n/**\n * Normalizes requests for Relay.\n *\n * @param request - Quote request to normalize.\n * @returns Normalized request.\n */\nfunction normalizeRequest(request: QuoteRequest) {\n const isHyperliquidDeposit =\n request.targetChainId === CHAIN_ID_ARBITRUM &&\n request.targetTokenAddress.toLowerCase() ===\n ARBITRUM_USDC_ADDRESS.toLowerCase();\n\n const isPolygonNativeSource =\n request.sourceChainId === CHAIN_ID_POLYGON &&\n request.sourceTokenAddress === getNativeToken(request.sourceChainId);\n\n const requestOutput: QuoteRequest = {\n ...request,\n sourceTokenAddress: isPolygonNativeSource\n ? NATIVE_TOKEN_ADDRESS\n : request.sourceTokenAddress,\n targetChainId: isHyperliquidDeposit ? toHex(1337) : request.targetChainId,\n targetTokenAddress: isHyperliquidDeposit\n ? '0x00000000000000000000000000000000'\n : request.targetTokenAddress,\n targetAmountMinimum: isHyperliquidDeposit\n ? new BigNumber(request.targetAmountMinimum).shiftedBy(2).toString(10)\n : request.targetAmountMinimum,\n };\n\n if (isHyperliquidDeposit) {\n log('Converting Arbitrum Hyperliquid deposit to direct deposit', {\n originalRequest: request,\n normalizedRequest: requestOutput,\n });\n }\n\n return {\n request: requestOutput,\n isSkipTransaction: isHyperliquidDeposit,\n };\n}\n\n/**\n * Normalizes a Relay quote into a TransactionPayQuote.\n *\n * @param quote - Relay quote.\n * @param request - Original quote request.\n * @param fullRequest - Full quotes request.\n * @returns Normalized quote.\n */\nfunction normalizeQuote(\n quote: RelayQuote,\n request: QuoteRequest,\n fullRequest: PayStrategyGetQuotesRequest,\n): TransactionPayQuote<RelayQuote> {\n const { messenger, transaction } = fullRequest;\n const { details } = quote;\n const { currencyIn, currencyOut } = details;\n const params = quote.steps[0].items[0].data;\n\n const { usdToFiatRate } = getFiatRates(messenger, request);\n\n const dust = getFiatValueFromUsd(\n calculateDustUsd(quote, request),\n usdToFiatRate,\n );\n\n const provider = getFiatValueFromUsd(\n new BigNumber(currencyIn.amountUsd).minus(currencyOut.amountUsd),\n usdToFiatRate,\n );\n\n const sourceNetwork = calculateGasCost({\n ...params,\n maxFeePerGas: undefined,\n maxPriorityFeePerGas: undefined,\n messenger,\n });\n\n const targetNetwork = quote.skipTransaction\n ? {\n usd: '0',\n fiat: '0',\n }\n : calculateTransactionGasCost(transaction, messenger);\n\n return {\n dust,\n estimatedDuration: details.timeEstimate,\n fees: {\n provider,\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Relay,\n };\n}\n\n/**\n * Calculate dust USD value.\n *\n * @param quote - Relay quote.\n * @param request - Quote request.\n * @returns Dust value in USD and fiat.\n */\nfunction calculateDustUsd(quote: RelayQuote, request: QuoteRequest) {\n const { currencyOut } = quote.details;\n const { amountUsd, amountFormatted, minimumAmount } = currencyOut;\n const { decimals: targetDecimals } = currencyOut.currency;\n\n const targetUsdRate = new BigNumber(amountUsd).dividedBy(amountFormatted);\n\n const dustRaw = new BigNumber(minimumAmount).minus(\n request.targetAmountMinimum,\n );\n\n return dustRaw.shiftedBy(-targetDecimals).multipliedBy(targetUsdRate);\n}\n\n/**\n * Converts USD value to fiat value.\n *\n * @param usdValue - USD value.\n * @param usdToFiatRate - USD to fiat rate.\n * @returns Fiat value.\n */\nfunction getFiatValueFromUsd(\n usdValue: BigNumber,\n usdToFiatRate: BigNumber,\n): FiatValue {\n const fiatValue = usdValue.multipliedBy(usdToFiatRate);\n\n return {\n usd: usdValue.toString(10),\n fiat: fiatValue.toString(10),\n };\n}\n\n/**\n * Calculates USD to fiat rate.\n *\n * @param messenger - Controller messenger.\n * @param request - Quote request.\n * @returns USD to fiat rate.\n */\nfunction getFiatRates(\n messenger: TransactionPayControllerMessenger,\n request: QuoteRequest,\n) {\n const { sourceChainId, sourceTokenAddress } = request;\n\n const finalSourceTokenAddress =\n sourceChainId === CHAIN_ID_POLYGON &&\n sourceTokenAddress === NATIVE_TOKEN_ADDRESS\n ? getNativeToken(sourceChainId)\n : sourceTokenAddress;\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n finalSourceTokenAddress,\n sourceChainId,\n );\n\n if (!sourceFiatRate) {\n throw new Error('Source token fiat rate not found');\n }\n\n const usdToFiatRate = new BigNumber(sourceFiatRate.fiatRate).dividedBy(\n sourceFiatRate.usdRate,\n );\n\n return { sourceFiatRate, usdToFiatRate };\n}\n\n/**\n * Gets feature flags for Relay quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlagState = messenger.call(\n 'RemoteFeatureFlagController:getState',\n );\n\n const featureFlags = featureFlagState.remoteFeatureFlags?.confirmation_pay as\n | Record<string, string>\n | undefined;\n\n const relayQuoteUrl = featureFlags?.relayQuoteUrl ?? RELAY_URL_QUOTE;\n\n return {\n relayQuoteUrl,\n };\n}\n"]}
1
+ {"version":3,"file":"relay-quotes.cjs","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import { successfulFetch, toHex } from '@metamask/controller-utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\n\nimport {\n ARBITRUM_USDC_ADDRESS,\n CHAIN_ID_ARBITRUM,\n CHAIN_ID_POLYGON,\n RELAY_URL_QUOTE,\n} from './constants';\nimport type { RelayQuote } from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { NATIVE_TOKEN_ADDRESS } from '../../constants';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetQuotesRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getNativeToken, getTokenFiatRate } from '../../utils/token';\n\nconst log = createModuleLogger(projectLogger, 'relay-strategy');\n\n/**\n * Fetches Relay quotes.\n *\n * @param request - Request object.\n * @returns Array of quotes.\n */\nexport async function getRelayQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>[]> {\n const { requests } = request;\n\n log('Fetching quotes', requests);\n\n const result = requests\n // Ignore gas fee token requests\n .filter((r) => r.targetAmountMinimum !== '0')\n .map((r) => normalizeRequest(r));\n\n const normalizedRequests = result.map((r) => r.request);\n const isSkipTransaction = result.some((r) => r.isSkipTransaction);\n\n log('Normalized requests', { normalizedRequests, isSkipTransaction });\n\n return await Promise.all(\n normalizedRequests.map((r) =>\n getSingleQuote(r, isSkipTransaction, request),\n ),\n );\n}\n\n/**\n * Fetches a single Relay quote.\n *\n * @param request - Quote request.\n * @param isSkipTransaction - Whether to skip the transaction.\n * @param fullRequest - Full quotes request.\n * @returns Single quote.\n */\nasync function getSingleQuote(\n request: QuoteRequest,\n isSkipTransaction: boolean,\n fullRequest: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>> {\n const { messenger } = fullRequest;\n\n try {\n const body = {\n amount: request.targetAmountMinimum,\n destinationChainId: Number(request.targetChainId),\n destinationCurrency: request.targetTokenAddress,\n originChainId: Number(request.sourceChainId),\n originCurrency: request.sourceTokenAddress,\n recipient: request.from,\n tradeType: 'EXPECTED_OUTPUT',\n user: request.from,\n };\n\n const url = getFeatureFlags(messenger).relayQuoteUrl;\n\n const response = await successfulFetch(url, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(body),\n });\n\n const quote = (await response.json()) as RelayQuote;\n quote.skipTransaction = isSkipTransaction;\n\n log('Fetched relay quote', { quote, url });\n\n return normalizeQuote(quote, request, fullRequest);\n } catch (e) {\n log('Error fetching relay quote', e);\n throw e;\n }\n}\n\n/**\n * Normalizes requests for Relay.\n *\n * @param request - Quote request to normalize.\n * @returns Normalized request.\n */\nfunction normalizeRequest(request: QuoteRequest) {\n const isHyperliquidDeposit =\n request.targetChainId === CHAIN_ID_ARBITRUM &&\n request.targetTokenAddress.toLowerCase() ===\n ARBITRUM_USDC_ADDRESS.toLowerCase();\n\n const isPolygonNativeSource =\n request.sourceChainId === CHAIN_ID_POLYGON &&\n request.sourceTokenAddress === getNativeToken(request.sourceChainId);\n\n const requestOutput: QuoteRequest = {\n ...request,\n sourceTokenAddress: isPolygonNativeSource\n ? NATIVE_TOKEN_ADDRESS\n : request.sourceTokenAddress,\n targetChainId: isHyperliquidDeposit ? toHex(1337) : request.targetChainId,\n targetTokenAddress: isHyperliquidDeposit\n ? '0x00000000000000000000000000000000'\n : request.targetTokenAddress,\n targetAmountMinimum: isHyperliquidDeposit\n ? new BigNumber(request.targetAmountMinimum).shiftedBy(2).toString(10)\n : request.targetAmountMinimum,\n };\n\n if (isHyperliquidDeposit) {\n log('Converting Arbitrum Hyperliquid deposit to direct deposit', {\n originalRequest: request,\n normalizedRequest: requestOutput,\n });\n }\n\n return {\n request: requestOutput,\n isSkipTransaction: isHyperliquidDeposit,\n };\n}\n\n/**\n * Normalizes a Relay quote into a TransactionPayQuote.\n *\n * @param quote - Relay quote.\n * @param request - Original quote request.\n * @param fullRequest - Full quotes request.\n * @returns Normalized quote.\n */\nfunction normalizeQuote(\n quote: RelayQuote,\n request: QuoteRequest,\n fullRequest: PayStrategyGetQuotesRequest,\n): TransactionPayQuote<RelayQuote> {\n const { messenger, transaction } = fullRequest;\n const { details } = quote;\n const { currencyIn, currencyOut } = details;\n const params = quote.steps[0].items[0].data;\n\n const { usdToFiatRate } = getFiatRates(messenger, request);\n\n const dust = getFiatValueFromUsd(\n calculateDustUsd(quote, request),\n usdToFiatRate,\n );\n\n const provider = getFiatValueFromUsd(\n new BigNumber(currencyIn.amountUsd).minus(currencyOut.amountUsd),\n usdToFiatRate,\n );\n\n const sourceNetwork = calculateGasCost({\n ...params,\n maxFeePerGas: undefined,\n maxPriorityFeePerGas: undefined,\n messenger,\n });\n\n const targetNetwork = quote.skipTransaction\n ? {\n usd: '0',\n fiat: '0',\n }\n : calculateTransactionGasCost(transaction, messenger);\n\n return {\n dust,\n estimatedDuration: details.timeEstimate,\n fees: {\n provider,\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Relay,\n };\n}\n\n/**\n * Calculate dust USD value.\n *\n * @param quote - Relay quote.\n * @param request - Quote request.\n * @returns Dust value in USD and fiat.\n */\nfunction calculateDustUsd(quote: RelayQuote, request: QuoteRequest) {\n const { currencyOut } = quote.details;\n const { amountUsd, amountFormatted, minimumAmount } = currencyOut;\n const { decimals: targetDecimals } = currencyOut.currency;\n\n const targetUsdRate = new BigNumber(amountUsd).dividedBy(amountFormatted);\n\n const dustRaw = new BigNumber(minimumAmount).minus(\n request.targetAmountMinimum,\n );\n\n return dustRaw.shiftedBy(-targetDecimals).multipliedBy(targetUsdRate);\n}\n\n/**\n * Converts USD value to fiat value.\n *\n * @param usdValue - USD value.\n * @param usdToFiatRate - USD to fiat rate.\n * @returns Fiat value.\n */\nfunction getFiatValueFromUsd(\n usdValue: BigNumber,\n usdToFiatRate: BigNumber,\n): FiatValue {\n const fiatValue = usdValue.multipliedBy(usdToFiatRate);\n\n return {\n usd: usdValue.toString(10),\n fiat: fiatValue.toString(10),\n };\n}\n\n/**\n * Calculates USD to fiat rate.\n *\n * @param messenger - Controller messenger.\n * @param request - Quote request.\n * @returns USD to fiat rate.\n */\nfunction getFiatRates(\n messenger: TransactionPayControllerMessenger,\n request: QuoteRequest,\n) {\n const { sourceChainId, sourceTokenAddress } = request;\n\n const finalSourceTokenAddress =\n sourceChainId === CHAIN_ID_POLYGON &&\n sourceTokenAddress === NATIVE_TOKEN_ADDRESS\n ? getNativeToken(sourceChainId)\n : sourceTokenAddress;\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n finalSourceTokenAddress,\n sourceChainId,\n );\n\n if (!sourceFiatRate) {\n throw new Error('Source token fiat rate not found');\n }\n\n const usdToFiatRate = new BigNumber(sourceFiatRate.fiatRate).dividedBy(\n sourceFiatRate.usdRate,\n );\n\n return { sourceFiatRate, usdToFiatRate };\n}\n\n/**\n * Gets feature flags for Relay quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlagState = messenger.call(\n 'RemoteFeatureFlagController:getState',\n );\n\n const featureFlags = featureFlagState.remoteFeatureFlags?.confirmation_pay as\n | Record<string, string>\n | undefined;\n\n const relayQuoteUrl = featureFlags?.relayQuoteUrl ?? RELAY_URL_QUOTE;\n\n return {\n relayQuoteUrl,\n };\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"relay-quotes.d.cts","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"AAUA,OAAO,KAAK,EAAE,UAAU,EAAE,oBAAgB;AAI1C,OAAO,KAAK,EAEV,2BAA2B,EAG3B,mBAAmB,EACpB,wBAAoB;AAMrB;;;;;GAKG;AACH,wBAAsB,cAAc,CAClC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,UAAU,CAAC,EAAE,CAAC,CAyB5C"}
1
+ {"version":3,"file":"relay-quotes.d.cts","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"AAUA,OAAO,KAAK,EAAE,UAAU,EAAE,oBAAgB;AAI1C,OAAO,KAAK,EAEV,2BAA2B,EAG3B,mBAAmB,EACpB,wBAAoB;AAMrB;;;;;GAKG;AACH,wBAAsB,cAAc,CAClC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,UAAU,CAAC,EAAE,CAAC,CAoB5C"}
@@ -1 +1 @@
1
- {"version":3,"file":"relay-quotes.d.mts","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"AAUA,OAAO,KAAK,EAAE,UAAU,EAAE,oBAAgB;AAI1C,OAAO,KAAK,EAEV,2BAA2B,EAG3B,mBAAmB,EACpB,wBAAoB;AAMrB;;;;;GAKG;AACH,wBAAsB,cAAc,CAClC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,UAAU,CAAC,EAAE,CAAC,CAyB5C"}
1
+ {"version":3,"file":"relay-quotes.d.mts","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"AAUA,OAAO,KAAK,EAAE,UAAU,EAAE,oBAAgB;AAI1C,OAAO,KAAK,EAEV,2BAA2B,EAG3B,mBAAmB,EACpB,wBAAoB;AAMrB;;;;;GAKG;AACH,wBAAsB,cAAc,CAClC,OAAO,EAAE,2BAA2B,GACnC,OAAO,CAAC,mBAAmB,CAAC,UAAU,CAAC,EAAE,CAAC,CAoB5C"}
@@ -17,20 +17,14 @@ const log = createModuleLogger(projectLogger, 'relay-strategy');
17
17
  export async function getRelayQuotes(request) {
18
18
  const { requests } = request;
19
19
  log('Fetching quotes', requests);
20
- try {
21
- const result = requests
22
- // Ignore gas fee token requests
23
- .filter((r) => r.targetAmountMinimum !== '0')
24
- .map((r) => normalizeRequest(r));
25
- const normalizedRequests = result.map((r) => r.request);
26
- const isSkipTransaction = result.some((r) => r.isSkipTransaction);
27
- log('Normalized requests', { normalizedRequests, isSkipTransaction });
28
- return await Promise.all(normalizedRequests.map((r) => getSingleQuote(r, isSkipTransaction, request)));
29
- }
30
- catch (error) {
31
- log('Error fetching quotes', { error });
32
- throw new Error(`Failed to fetch Relay quotes: ${String(error)}`);
33
- }
20
+ const result = requests
21
+ // Ignore gas fee token requests
22
+ .filter((r) => r.targetAmountMinimum !== '0')
23
+ .map((r) => normalizeRequest(r));
24
+ const normalizedRequests = result.map((r) => r.request);
25
+ const isSkipTransaction = result.some((r) => r.isSkipTransaction);
26
+ log('Normalized requests', { normalizedRequests, isSkipTransaction });
27
+ return await Promise.all(normalizedRequests.map((r) => getSingleQuote(r, isSkipTransaction, request)));
34
28
  }
35
29
  /**
36
30
  * Fetches a single Relay quote.
@@ -1 +1 @@
1
- {"version":3,"file":"relay-quotes.mjs","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import { successfulFetch, toHex } from '@metamask/controller-utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\n\nimport {\n ARBITRUM_USDC_ADDRESS,\n CHAIN_ID_ARBITRUM,\n CHAIN_ID_POLYGON,\n RELAY_URL_QUOTE,\n} from './constants';\nimport type { RelayQuote } from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { NATIVE_TOKEN_ADDRESS } from '../../constants';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetQuotesRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getNativeToken, getTokenFiatRate } from '../../utils/token';\n\nconst log = createModuleLogger(projectLogger, 'relay-strategy');\n\n/**\n * Fetches Relay quotes.\n *\n * @param request - Request object.\n * @returns Array of quotes.\n */\nexport async function getRelayQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>[]> {\n const { requests } = request;\n\n log('Fetching quotes', requests);\n\n try {\n const result = requests\n // Ignore gas fee token requests\n .filter((r) => r.targetAmountMinimum !== '0')\n .map((r) => normalizeRequest(r));\n\n const normalizedRequests = result.map((r) => r.request);\n const isSkipTransaction = result.some((r) => r.isSkipTransaction);\n\n log('Normalized requests', { normalizedRequests, isSkipTransaction });\n\n return await Promise.all(\n normalizedRequests.map((r) =>\n getSingleQuote(r, isSkipTransaction, request),\n ),\n );\n } catch (error) {\n log('Error fetching quotes', { error });\n throw new Error(`Failed to fetch Relay quotes: ${String(error)}`);\n }\n}\n\n/**\n * Fetches a single Relay quote.\n *\n * @param request - Quote request.\n * @param isSkipTransaction - Whether to skip the transaction.\n * @param fullRequest - Full quotes request.\n * @returns Single quote.\n */\nasync function getSingleQuote(\n request: QuoteRequest,\n isSkipTransaction: boolean,\n fullRequest: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>> {\n const { messenger } = fullRequest;\n\n try {\n const body = {\n amount: request.targetAmountMinimum,\n destinationChainId: Number(request.targetChainId),\n destinationCurrency: request.targetTokenAddress,\n originChainId: Number(request.sourceChainId),\n originCurrency: request.sourceTokenAddress,\n recipient: request.from,\n tradeType: 'EXPECTED_OUTPUT',\n user: request.from,\n };\n\n const url = getFeatureFlags(messenger).relayQuoteUrl;\n\n const response = await successfulFetch(url, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(body),\n });\n\n const quote = (await response.json()) as RelayQuote;\n quote.skipTransaction = isSkipTransaction;\n\n log('Fetched relay quote', { quote, url });\n\n return normalizeQuote(quote, request, fullRequest);\n } catch (e) {\n log('Error fetching relay quote', e);\n throw e;\n }\n}\n\n/**\n * Normalizes requests for Relay.\n *\n * @param request - Quote request to normalize.\n * @returns Normalized request.\n */\nfunction normalizeRequest(request: QuoteRequest) {\n const isHyperliquidDeposit =\n request.targetChainId === CHAIN_ID_ARBITRUM &&\n request.targetTokenAddress.toLowerCase() ===\n ARBITRUM_USDC_ADDRESS.toLowerCase();\n\n const isPolygonNativeSource =\n request.sourceChainId === CHAIN_ID_POLYGON &&\n request.sourceTokenAddress === getNativeToken(request.sourceChainId);\n\n const requestOutput: QuoteRequest = {\n ...request,\n sourceTokenAddress: isPolygonNativeSource\n ? NATIVE_TOKEN_ADDRESS\n : request.sourceTokenAddress,\n targetChainId: isHyperliquidDeposit ? toHex(1337) : request.targetChainId,\n targetTokenAddress: isHyperliquidDeposit\n ? '0x00000000000000000000000000000000'\n : request.targetTokenAddress,\n targetAmountMinimum: isHyperliquidDeposit\n ? new BigNumber(request.targetAmountMinimum).shiftedBy(2).toString(10)\n : request.targetAmountMinimum,\n };\n\n if (isHyperliquidDeposit) {\n log('Converting Arbitrum Hyperliquid deposit to direct deposit', {\n originalRequest: request,\n normalizedRequest: requestOutput,\n });\n }\n\n return {\n request: requestOutput,\n isSkipTransaction: isHyperliquidDeposit,\n };\n}\n\n/**\n * Normalizes a Relay quote into a TransactionPayQuote.\n *\n * @param quote - Relay quote.\n * @param request - Original quote request.\n * @param fullRequest - Full quotes request.\n * @returns Normalized quote.\n */\nfunction normalizeQuote(\n quote: RelayQuote,\n request: QuoteRequest,\n fullRequest: PayStrategyGetQuotesRequest,\n): TransactionPayQuote<RelayQuote> {\n const { messenger, transaction } = fullRequest;\n const { details } = quote;\n const { currencyIn, currencyOut } = details;\n const params = quote.steps[0].items[0].data;\n\n const { usdToFiatRate } = getFiatRates(messenger, request);\n\n const dust = getFiatValueFromUsd(\n calculateDustUsd(quote, request),\n usdToFiatRate,\n );\n\n const provider = getFiatValueFromUsd(\n new BigNumber(currencyIn.amountUsd).minus(currencyOut.amountUsd),\n usdToFiatRate,\n );\n\n const sourceNetwork = calculateGasCost({\n ...params,\n maxFeePerGas: undefined,\n maxPriorityFeePerGas: undefined,\n messenger,\n });\n\n const targetNetwork = quote.skipTransaction\n ? {\n usd: '0',\n fiat: '0',\n }\n : calculateTransactionGasCost(transaction, messenger);\n\n return {\n dust,\n estimatedDuration: details.timeEstimate,\n fees: {\n provider,\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Relay,\n };\n}\n\n/**\n * Calculate dust USD value.\n *\n * @param quote - Relay quote.\n * @param request - Quote request.\n * @returns Dust value in USD and fiat.\n */\nfunction calculateDustUsd(quote: RelayQuote, request: QuoteRequest) {\n const { currencyOut } = quote.details;\n const { amountUsd, amountFormatted, minimumAmount } = currencyOut;\n const { decimals: targetDecimals } = currencyOut.currency;\n\n const targetUsdRate = new BigNumber(amountUsd).dividedBy(amountFormatted);\n\n const dustRaw = new BigNumber(minimumAmount).minus(\n request.targetAmountMinimum,\n );\n\n return dustRaw.shiftedBy(-targetDecimals).multipliedBy(targetUsdRate);\n}\n\n/**\n * Converts USD value to fiat value.\n *\n * @param usdValue - USD value.\n * @param usdToFiatRate - USD to fiat rate.\n * @returns Fiat value.\n */\nfunction getFiatValueFromUsd(\n usdValue: BigNumber,\n usdToFiatRate: BigNumber,\n): FiatValue {\n const fiatValue = usdValue.multipliedBy(usdToFiatRate);\n\n return {\n usd: usdValue.toString(10),\n fiat: fiatValue.toString(10),\n };\n}\n\n/**\n * Calculates USD to fiat rate.\n *\n * @param messenger - Controller messenger.\n * @param request - Quote request.\n * @returns USD to fiat rate.\n */\nfunction getFiatRates(\n messenger: TransactionPayControllerMessenger,\n request: QuoteRequest,\n) {\n const { sourceChainId, sourceTokenAddress } = request;\n\n const finalSourceTokenAddress =\n sourceChainId === CHAIN_ID_POLYGON &&\n sourceTokenAddress === NATIVE_TOKEN_ADDRESS\n ? getNativeToken(sourceChainId)\n : sourceTokenAddress;\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n finalSourceTokenAddress,\n sourceChainId,\n );\n\n if (!sourceFiatRate) {\n throw new Error('Source token fiat rate not found');\n }\n\n const usdToFiatRate = new BigNumber(sourceFiatRate.fiatRate).dividedBy(\n sourceFiatRate.usdRate,\n );\n\n return { sourceFiatRate, usdToFiatRate };\n}\n\n/**\n * Gets feature flags for Relay quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlagState = messenger.call(\n 'RemoteFeatureFlagController:getState',\n );\n\n const featureFlags = featureFlagState.remoteFeatureFlags?.confirmation_pay as\n | Record<string, string>\n | undefined;\n\n const relayQuoteUrl = featureFlags?.relayQuoteUrl ?? RELAY_URL_QUOTE;\n\n return {\n relayQuoteUrl,\n };\n}\n"]}
1
+ {"version":3,"file":"relay-quotes.mjs","sourceRoot":"","sources":["../../../src/strategy/relay/relay-quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import { successfulFetch, toHex } from '@metamask/controller-utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { BigNumber } from 'bignumber.js';\n\nimport {\n ARBITRUM_USDC_ADDRESS,\n CHAIN_ID_ARBITRUM,\n CHAIN_ID_POLYGON,\n RELAY_URL_QUOTE,\n} from './constants';\nimport type { RelayQuote } from './types';\nimport { TransactionPayStrategy } from '../..';\nimport { NATIVE_TOKEN_ADDRESS } from '../../constants';\nimport { projectLogger } from '../../logger';\nimport type {\n FiatValue,\n PayStrategyGetQuotesRequest,\n QuoteRequest,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport { calculateGasCost, calculateTransactionGasCost } from '../../utils/gas';\nimport { getNativeToken, getTokenFiatRate } from '../../utils/token';\n\nconst log = createModuleLogger(projectLogger, 'relay-strategy');\n\n/**\n * Fetches Relay quotes.\n *\n * @param request - Request object.\n * @returns Array of quotes.\n */\nexport async function getRelayQuotes(\n request: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>[]> {\n const { requests } = request;\n\n log('Fetching quotes', requests);\n\n const result = requests\n // Ignore gas fee token requests\n .filter((r) => r.targetAmountMinimum !== '0')\n .map((r) => normalizeRequest(r));\n\n const normalizedRequests = result.map((r) => r.request);\n const isSkipTransaction = result.some((r) => r.isSkipTransaction);\n\n log('Normalized requests', { normalizedRequests, isSkipTransaction });\n\n return await Promise.all(\n normalizedRequests.map((r) =>\n getSingleQuote(r, isSkipTransaction, request),\n ),\n );\n}\n\n/**\n * Fetches a single Relay quote.\n *\n * @param request - Quote request.\n * @param isSkipTransaction - Whether to skip the transaction.\n * @param fullRequest - Full quotes request.\n * @returns Single quote.\n */\nasync function getSingleQuote(\n request: QuoteRequest,\n isSkipTransaction: boolean,\n fullRequest: PayStrategyGetQuotesRequest,\n): Promise<TransactionPayQuote<RelayQuote>> {\n const { messenger } = fullRequest;\n\n try {\n const body = {\n amount: request.targetAmountMinimum,\n destinationChainId: Number(request.targetChainId),\n destinationCurrency: request.targetTokenAddress,\n originChainId: Number(request.sourceChainId),\n originCurrency: request.sourceTokenAddress,\n recipient: request.from,\n tradeType: 'EXPECTED_OUTPUT',\n user: request.from,\n };\n\n const url = getFeatureFlags(messenger).relayQuoteUrl;\n\n const response = await successfulFetch(url, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(body),\n });\n\n const quote = (await response.json()) as RelayQuote;\n quote.skipTransaction = isSkipTransaction;\n\n log('Fetched relay quote', { quote, url });\n\n return normalizeQuote(quote, request, fullRequest);\n } catch (e) {\n log('Error fetching relay quote', e);\n throw e;\n }\n}\n\n/**\n * Normalizes requests for Relay.\n *\n * @param request - Quote request to normalize.\n * @returns Normalized request.\n */\nfunction normalizeRequest(request: QuoteRequest) {\n const isHyperliquidDeposit =\n request.targetChainId === CHAIN_ID_ARBITRUM &&\n request.targetTokenAddress.toLowerCase() ===\n ARBITRUM_USDC_ADDRESS.toLowerCase();\n\n const isPolygonNativeSource =\n request.sourceChainId === CHAIN_ID_POLYGON &&\n request.sourceTokenAddress === getNativeToken(request.sourceChainId);\n\n const requestOutput: QuoteRequest = {\n ...request,\n sourceTokenAddress: isPolygonNativeSource\n ? NATIVE_TOKEN_ADDRESS\n : request.sourceTokenAddress,\n targetChainId: isHyperliquidDeposit ? toHex(1337) : request.targetChainId,\n targetTokenAddress: isHyperliquidDeposit\n ? '0x00000000000000000000000000000000'\n : request.targetTokenAddress,\n targetAmountMinimum: isHyperliquidDeposit\n ? new BigNumber(request.targetAmountMinimum).shiftedBy(2).toString(10)\n : request.targetAmountMinimum,\n };\n\n if (isHyperliquidDeposit) {\n log('Converting Arbitrum Hyperliquid deposit to direct deposit', {\n originalRequest: request,\n normalizedRequest: requestOutput,\n });\n }\n\n return {\n request: requestOutput,\n isSkipTransaction: isHyperliquidDeposit,\n };\n}\n\n/**\n * Normalizes a Relay quote into a TransactionPayQuote.\n *\n * @param quote - Relay quote.\n * @param request - Original quote request.\n * @param fullRequest - Full quotes request.\n * @returns Normalized quote.\n */\nfunction normalizeQuote(\n quote: RelayQuote,\n request: QuoteRequest,\n fullRequest: PayStrategyGetQuotesRequest,\n): TransactionPayQuote<RelayQuote> {\n const { messenger, transaction } = fullRequest;\n const { details } = quote;\n const { currencyIn, currencyOut } = details;\n const params = quote.steps[0].items[0].data;\n\n const { usdToFiatRate } = getFiatRates(messenger, request);\n\n const dust = getFiatValueFromUsd(\n calculateDustUsd(quote, request),\n usdToFiatRate,\n );\n\n const provider = getFiatValueFromUsd(\n new BigNumber(currencyIn.amountUsd).minus(currencyOut.amountUsd),\n usdToFiatRate,\n );\n\n const sourceNetwork = calculateGasCost({\n ...params,\n maxFeePerGas: undefined,\n maxPriorityFeePerGas: undefined,\n messenger,\n });\n\n const targetNetwork = quote.skipTransaction\n ? {\n usd: '0',\n fiat: '0',\n }\n : calculateTransactionGasCost(transaction, messenger);\n\n return {\n dust,\n estimatedDuration: details.timeEstimate,\n fees: {\n provider,\n sourceNetwork,\n targetNetwork,\n },\n original: quote,\n request,\n strategy: TransactionPayStrategy.Relay,\n };\n}\n\n/**\n * Calculate dust USD value.\n *\n * @param quote - Relay quote.\n * @param request - Quote request.\n * @returns Dust value in USD and fiat.\n */\nfunction calculateDustUsd(quote: RelayQuote, request: QuoteRequest) {\n const { currencyOut } = quote.details;\n const { amountUsd, amountFormatted, minimumAmount } = currencyOut;\n const { decimals: targetDecimals } = currencyOut.currency;\n\n const targetUsdRate = new BigNumber(amountUsd).dividedBy(amountFormatted);\n\n const dustRaw = new BigNumber(minimumAmount).minus(\n request.targetAmountMinimum,\n );\n\n return dustRaw.shiftedBy(-targetDecimals).multipliedBy(targetUsdRate);\n}\n\n/**\n * Converts USD value to fiat value.\n *\n * @param usdValue - USD value.\n * @param usdToFiatRate - USD to fiat rate.\n * @returns Fiat value.\n */\nfunction getFiatValueFromUsd(\n usdValue: BigNumber,\n usdToFiatRate: BigNumber,\n): FiatValue {\n const fiatValue = usdValue.multipliedBy(usdToFiatRate);\n\n return {\n usd: usdValue.toString(10),\n fiat: fiatValue.toString(10),\n };\n}\n\n/**\n * Calculates USD to fiat rate.\n *\n * @param messenger - Controller messenger.\n * @param request - Quote request.\n * @returns USD to fiat rate.\n */\nfunction getFiatRates(\n messenger: TransactionPayControllerMessenger,\n request: QuoteRequest,\n) {\n const { sourceChainId, sourceTokenAddress } = request;\n\n const finalSourceTokenAddress =\n sourceChainId === CHAIN_ID_POLYGON &&\n sourceTokenAddress === NATIVE_TOKEN_ADDRESS\n ? getNativeToken(sourceChainId)\n : sourceTokenAddress;\n\n const sourceFiatRate = getTokenFiatRate(\n messenger,\n finalSourceTokenAddress,\n sourceChainId,\n );\n\n if (!sourceFiatRate) {\n throw new Error('Source token fiat rate not found');\n }\n\n const usdToFiatRate = new BigNumber(sourceFiatRate.fiatRate).dividedBy(\n sourceFiatRate.usdRate,\n );\n\n return { sourceFiatRate, usdToFiatRate };\n}\n\n/**\n * Gets feature flags for Relay quotes.\n *\n * @param messenger - Controller messenger.\n * @returns Feature flags.\n */\nfunction getFeatureFlags(messenger: TransactionPayControllerMessenger) {\n const featureFlagState = messenger.call(\n 'RemoteFeatureFlagController:getState',\n );\n\n const featureFlags = featureFlagState.remoteFeatureFlags?.confirmation_pay as\n | Record<string, string>\n | undefined;\n\n const relayQuoteUrl = featureFlags?.relayQuoteUrl ?? RELAY_URL_QUOTE;\n\n return {\n relayQuoteUrl,\n };\n}\n"]}
@@ -25,8 +25,8 @@ async function updateQuotes(request) {
25
25
  if (!paymentToken) {
26
26
  return;
27
27
  }
28
- const requests = (sourceAmounts ?? []).map((sourceAmount) => {
29
- const token = tokens.find((t) => t.address === sourceAmount.targetTokenAddress);
28
+ const requests = (sourceAmounts ?? []).map((sourceAmount, i) => {
29
+ const token = tokens[i];
30
30
  return {
31
31
  from: transaction.txParams.from,
32
32
  sourceBalanceRaw: paymentToken.balanceRaw,
@@ -1 +1 @@
1
- {"version":3,"file":"quotes.cjs","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex, Json } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\n\nimport { getStrategy, getStrategyByName } from './strategy';\nimport { calculateTotals } from './totals';\nimport { getTransaction, updateTransaction } from './transaction';\nimport { projectLogger } from '../logger';\nimport type {\n QuoteRequest,\n TransactionData,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n TransactionPayRequiredToken,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\nconst QUOTES_CHECK_INTERVAL = 1 * 1000; // 1 Second\nconst DEFAULT_REFRESH_INTERVAL = 30 * 1000; // 30 Seconds\n\nconst log = createModuleLogger(projectLogger, 'quotes');\n\nexport type UpdateQuotesRequest = {\n messenger: TransactionPayControllerMessenger;\n transactionData: TransactionData | undefined;\n transactionId: string;\n updateTransactionData: UpdateTransactionDataCallback;\n};\n\n/**\n * Update the quotes for a specific transaction.\n *\n * @param request - Request parameters.\n */\nexport async function updateQuotes(request: UpdateQuotesRequest) {\n const { messenger, transactionData, transactionId, updateTransactionData } =\n request;\n\n const transaction = getTransaction(transactionId, messenger);\n\n log('Updating quotes', { transactionId });\n\n if (!transaction || !transactionData) {\n throw new Error('Transaction not found');\n }\n\n const { paymentToken, sourceAmounts, tokens } = transactionData;\n\n if (!paymentToken) {\n return;\n }\n\n const requests: QuoteRequest[] = (sourceAmounts ?? []).map((sourceAmount) => {\n const token = tokens.find(\n (t) => t.address === sourceAmount.targetTokenAddress,\n ) as TransactionPayRequiredToken;\n\n return {\n from: transaction.txParams.from as Hex,\n sourceBalanceRaw: paymentToken.balanceRaw,\n sourceTokenAmount: sourceAmount.sourceAmountRaw,\n sourceChainId: paymentToken.chainId,\n sourceTokenAddress: paymentToken.address,\n targetAmountMinimum: token.allowUnderMinimum ? '0' : token.amountRaw,\n targetChainId: token.chainId,\n targetTokenAddress: token.address,\n };\n });\n\n if (!requests?.length) {\n log('No quote requests', { transactionId });\n }\n\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\n\n const strategy = await getStrategy(messenger as never, transaction);\n\n try {\n quotes = requests?.length\n ? ((await strategy.getQuotes({\n messenger,\n requests,\n transaction,\n })) as TransactionPayQuote<Json>[])\n : [];\n } catch (error) {\n log('Error fetching quotes', { error, transactionId });\n }\n\n log('Updated', { transactionId, quotes });\n\n const batchTransactions =\n quotes?.length && strategy.getBatchTransactions\n ? await strategy.getBatchTransactions({\n messenger,\n quotes,\n })\n : [];\n\n log('Batch transactions', { transactionId, batchTransactions });\n\n const totals = calculateTotals(quotes as never, tokens, messenger);\n\n log('Calculated totals', { transactionId, totals });\n\n syncTransaction({\n batchTransactions,\n messenger: messenger as never,\n paymentToken,\n totals,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.quotes = quotes as never;\n data.quotesLastUpdated = Date.now();\n data.totals = totals;\n data.isLoading = false;\n });\n}\n\n/**\n * Poll quotes at regular intervals.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nexport function queueRefreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n setTimeout(() => {\n refreshQuotes(messenger, updateTransactionData)\n .finally(() => queueRefreshQuotes(messenger, updateTransactionData))\n .catch((error) => {\n log('Error polling quotes', { messenger, error });\n });\n }, QUOTES_CHECK_INTERVAL);\n}\n\n/**\n * Sync batch transactions to the transaction meta.\n *\n * @param request - Request object.\n * @param request.batchTransactions - Batch transactions to sync.\n * @param request.messenger - Messenger instance.\n * @param request.paymentToken - Payment token used.\n * @param request.totals - Calculated totals.\n * @param request.transactionId - ID of the transaction to sync.\n */\nfunction syncTransaction({\n batchTransactions,\n messenger,\n paymentToken,\n totals,\n transactionId,\n}: {\n batchTransactions: BatchTransaction[];\n messenger: TransactionPayControllerMessenger;\n paymentToken: TransactionPaymentToken;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n updateTransaction(\n {\n transactionId,\n messenger: messenger as never,\n note: 'Update transaction pay data',\n },\n (tx: TransactionMeta) => {\n tx.batchTransactions = batchTransactions;\n tx.batchTransactionsOptions = {};\n\n tx.metamaskPay = {\n bridgeFeeFiat: totals.fees.provider.usd,\n chainId: paymentToken.chainId,\n networkFeeFiat: totals.fees.sourceNetwork.usd,\n tokenAddress: paymentToken.address,\n totalFiat: totals.total.usd,\n };\n },\n );\n}\n\n/**\n * Refresh quotes for all transactions if expired.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nasync function refreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n const state = messenger.call('TransactionPayController:getState');\n const transactionIds = Object.keys(state.transactionData);\n\n for (const transactionId of transactionIds) {\n const transactionData = state.transactionData[transactionId];\n const { isLoading, quotes, quotesLastUpdated } = transactionData;\n\n if (isLoading || !quotes?.length) {\n continue;\n }\n\n const strategyName = quotes[0].strategy;\n const strategy = getStrategyByName(strategyName);\n\n const refreshInterval =\n (await strategy.getRefreshInterval?.({\n chainId: quotes[0].request.sourceChainId,\n messenger,\n })) ?? DEFAULT_REFRESH_INTERVAL;\n\n const isExpired = Date.now() - (quotesLastUpdated ?? 0) > refreshInterval;\n\n if (!isExpired) {\n continue;\n }\n\n log('Refreshing expired quotes', {\n transactionId,\n strategy: strategyName,\n refreshInterval,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n}\n"]}
1
+ {"version":3,"file":"quotes.cjs","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex, Json } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\n\nimport { getStrategy, getStrategyByName } from './strategy';\nimport { calculateTotals } from './totals';\nimport { getTransaction, updateTransaction } from './transaction';\nimport { projectLogger } from '../logger';\nimport type {\n QuoteRequest,\n TransactionData,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\nconst QUOTES_CHECK_INTERVAL = 1 * 1000; // 1 Second\nconst DEFAULT_REFRESH_INTERVAL = 30 * 1000; // 30 Seconds\n\nconst log = createModuleLogger(projectLogger, 'quotes');\n\nexport type UpdateQuotesRequest = {\n messenger: TransactionPayControllerMessenger;\n transactionData: TransactionData | undefined;\n transactionId: string;\n updateTransactionData: UpdateTransactionDataCallback;\n};\n\n/**\n * Update the quotes for a specific transaction.\n *\n * @param request - Request parameters.\n */\nexport async function updateQuotes(request: UpdateQuotesRequest) {\n const { messenger, transactionData, transactionId, updateTransactionData } =\n request;\n\n const transaction = getTransaction(transactionId, messenger);\n\n log('Updating quotes', { transactionId });\n\n if (!transaction || !transactionData) {\n throw new Error('Transaction not found');\n }\n\n const { paymentToken, sourceAmounts, tokens } = transactionData;\n\n if (!paymentToken) {\n return;\n }\n\n const requests: QuoteRequest[] = (sourceAmounts ?? []).map(\n (sourceAmount, i) => {\n const token = tokens[i];\n\n return {\n from: transaction.txParams.from as Hex,\n sourceBalanceRaw: paymentToken.balanceRaw,\n sourceTokenAmount: sourceAmount.sourceAmountRaw,\n sourceChainId: paymentToken.chainId,\n sourceTokenAddress: paymentToken.address,\n targetAmountMinimum: token.allowUnderMinimum ? '0' : token.amountRaw,\n targetChainId: token.chainId,\n targetTokenAddress: token.address,\n };\n },\n );\n\n if (!requests?.length) {\n log('No quote requests', { transactionId });\n }\n\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\n\n const strategy = await getStrategy(messenger as never, transaction);\n\n try {\n quotes = requests?.length\n ? ((await strategy.getQuotes({\n messenger,\n requests,\n transaction,\n })) as TransactionPayQuote<Json>[])\n : [];\n } catch (error) {\n log('Error fetching quotes', { error, transactionId });\n }\n\n log('Updated', { transactionId, quotes });\n\n const batchTransactions =\n quotes?.length && strategy.getBatchTransactions\n ? await strategy.getBatchTransactions({\n messenger,\n quotes,\n })\n : [];\n\n log('Batch transactions', { transactionId, batchTransactions });\n\n const totals = calculateTotals(quotes as never, tokens, messenger);\n\n log('Calculated totals', { transactionId, totals });\n\n syncTransaction({\n batchTransactions,\n messenger: messenger as never,\n paymentToken,\n totals,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.quotes = quotes as never;\n data.quotesLastUpdated = Date.now();\n data.totals = totals;\n data.isLoading = false;\n });\n}\n\n/**\n * Poll quotes at regular intervals.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nexport function queueRefreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n setTimeout(() => {\n refreshQuotes(messenger, updateTransactionData)\n .finally(() => queueRefreshQuotes(messenger, updateTransactionData))\n .catch((error) => {\n log('Error polling quotes', { messenger, error });\n });\n }, QUOTES_CHECK_INTERVAL);\n}\n\n/**\n * Sync batch transactions to the transaction meta.\n *\n * @param request - Request object.\n * @param request.batchTransactions - Batch transactions to sync.\n * @param request.messenger - Messenger instance.\n * @param request.paymentToken - Payment token used.\n * @param request.totals - Calculated totals.\n * @param request.transactionId - ID of the transaction to sync.\n */\nfunction syncTransaction({\n batchTransactions,\n messenger,\n paymentToken,\n totals,\n transactionId,\n}: {\n batchTransactions: BatchTransaction[];\n messenger: TransactionPayControllerMessenger;\n paymentToken: TransactionPaymentToken;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n updateTransaction(\n {\n transactionId,\n messenger: messenger as never,\n note: 'Update transaction pay data',\n },\n (tx: TransactionMeta) => {\n tx.batchTransactions = batchTransactions;\n tx.batchTransactionsOptions = {};\n\n tx.metamaskPay = {\n bridgeFeeFiat: totals.fees.provider.usd,\n chainId: paymentToken.chainId,\n networkFeeFiat: totals.fees.sourceNetwork.usd,\n tokenAddress: paymentToken.address,\n totalFiat: totals.total.usd,\n };\n },\n );\n}\n\n/**\n * Refresh quotes for all transactions if expired.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nasync function refreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n const state = messenger.call('TransactionPayController:getState');\n const transactionIds = Object.keys(state.transactionData);\n\n for (const transactionId of transactionIds) {\n const transactionData = state.transactionData[transactionId];\n const { isLoading, quotes, quotesLastUpdated } = transactionData;\n\n if (isLoading || !quotes?.length) {\n continue;\n }\n\n const strategyName = quotes[0].strategy;\n const strategy = getStrategyByName(strategyName);\n\n const refreshInterval =\n (await strategy.getRefreshInterval?.({\n chainId: quotes[0].request.sourceChainId,\n messenger,\n })) ?? DEFAULT_REFRESH_INTERVAL;\n\n const isExpired = Date.now() - (quotesLastUpdated ?? 0) > refreshInterval;\n\n if (!isExpired) {\n continue;\n }\n\n log('Refreshing expired quotes', {\n transactionId,\n strategy: strategyName,\n refreshInterval,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"quotes.d.cts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAKjC,6BAA6B,EAC9B,qBAAiB;AAOlB,MAAM,MAAM,mBAAmB,GAAG;IAChC,SAAS,EAAE,iCAAiC,CAAC;IAC7C,eAAe,EAAE,eAAe,GAAG,SAAS,CAAC;IAC7C,aAAa,EAAE,MAAM,CAAC;IACtB,qBAAqB,EAAE,6BAA6B,CAAC;CACtD,CAAC;AAEF;;;;GAIG;AACH,wBAAsB,YAAY,CAAC,OAAO,EAAE,mBAAmB,iBAqF9D;AAED;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAChC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,QASrD"}
1
+ {"version":3,"file":"quotes.d.cts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAIjC,6BAA6B,EAC9B,qBAAiB;AAOlB,MAAM,MAAM,mBAAmB,GAAG;IAChC,SAAS,EAAE,iCAAiC,CAAC;IAC7C,eAAe,EAAE,eAAe,GAAG,SAAS,CAAC;IAC7C,aAAa,EAAE,MAAM,CAAC;IACtB,qBAAqB,EAAE,6BAA6B,CAAC;CACtD,CAAC;AAEF;;;;GAIG;AACH,wBAAsB,YAAY,CAAC,OAAO,EAAE,mBAAmB,iBAqF9D;AAED;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAChC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,QASrD"}
@@ -1 +1 @@
1
- {"version":3,"file":"quotes.d.mts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAKjC,6BAA6B,EAC9B,qBAAiB;AAOlB,MAAM,MAAM,mBAAmB,GAAG;IAChC,SAAS,EAAE,iCAAiC,CAAC;IAC7C,eAAe,EAAE,eAAe,GAAG,SAAS,CAAC;IAC7C,aAAa,EAAE,MAAM,CAAC;IACtB,qBAAqB,EAAE,6BAA6B,CAAC;CACtD,CAAC;AAEF;;;;GAIG;AACH,wBAAsB,YAAY,CAAC,OAAO,EAAE,mBAAmB,iBAqF9D;AAED;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAChC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,QASrD"}
1
+ {"version":3,"file":"quotes.d.mts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAIjC,6BAA6B,EAC9B,qBAAiB;AAOlB,MAAM,MAAM,mBAAmB,GAAG;IAChC,SAAS,EAAE,iCAAiC,CAAC;IAC7C,eAAe,EAAE,eAAe,GAAG,SAAS,CAAC;IAC7C,aAAa,EAAE,MAAM,CAAC;IACtB,qBAAqB,EAAE,6BAA6B,CAAC;CACtD,CAAC;AAEF;;;;GAIG;AACH,wBAAsB,YAAY,CAAC,OAAO,EAAE,mBAAmB,iBAqF9D;AAED;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAChC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,QASrD"}
@@ -22,8 +22,8 @@ export async function updateQuotes(request) {
22
22
  if (!paymentToken) {
23
23
  return;
24
24
  }
25
- const requests = (sourceAmounts ?? []).map((sourceAmount) => {
26
- const token = tokens.find((t) => t.address === sourceAmount.targetTokenAddress);
25
+ const requests = (sourceAmounts ?? []).map((sourceAmount, i) => {
26
+ const token = tokens[i];
27
27
  return {
28
28
  from: transaction.txParams.from,
29
29
  sourceBalanceRaw: paymentToken.balanceRaw,
@@ -1 +1 @@
1
- {"version":3,"file":"quotes.mjs","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex, Json } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\n\nimport { getStrategy, getStrategyByName } from './strategy';\nimport { calculateTotals } from './totals';\nimport { getTransaction, updateTransaction } from './transaction';\nimport { projectLogger } from '../logger';\nimport type {\n QuoteRequest,\n TransactionData,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n TransactionPayRequiredToken,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\nconst QUOTES_CHECK_INTERVAL = 1 * 1000; // 1 Second\nconst DEFAULT_REFRESH_INTERVAL = 30 * 1000; // 30 Seconds\n\nconst log = createModuleLogger(projectLogger, 'quotes');\n\nexport type UpdateQuotesRequest = {\n messenger: TransactionPayControllerMessenger;\n transactionData: TransactionData | undefined;\n transactionId: string;\n updateTransactionData: UpdateTransactionDataCallback;\n};\n\n/**\n * Update the quotes for a specific transaction.\n *\n * @param request - Request parameters.\n */\nexport async function updateQuotes(request: UpdateQuotesRequest) {\n const { messenger, transactionData, transactionId, updateTransactionData } =\n request;\n\n const transaction = getTransaction(transactionId, messenger);\n\n log('Updating quotes', { transactionId });\n\n if (!transaction || !transactionData) {\n throw new Error('Transaction not found');\n }\n\n const { paymentToken, sourceAmounts, tokens } = transactionData;\n\n if (!paymentToken) {\n return;\n }\n\n const requests: QuoteRequest[] = (sourceAmounts ?? []).map((sourceAmount) => {\n const token = tokens.find(\n (t) => t.address === sourceAmount.targetTokenAddress,\n ) as TransactionPayRequiredToken;\n\n return {\n from: transaction.txParams.from as Hex,\n sourceBalanceRaw: paymentToken.balanceRaw,\n sourceTokenAmount: sourceAmount.sourceAmountRaw,\n sourceChainId: paymentToken.chainId,\n sourceTokenAddress: paymentToken.address,\n targetAmountMinimum: token.allowUnderMinimum ? '0' : token.amountRaw,\n targetChainId: token.chainId,\n targetTokenAddress: token.address,\n };\n });\n\n if (!requests?.length) {\n log('No quote requests', { transactionId });\n }\n\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\n\n const strategy = await getStrategy(messenger as never, transaction);\n\n try {\n quotes = requests?.length\n ? ((await strategy.getQuotes({\n messenger,\n requests,\n transaction,\n })) as TransactionPayQuote<Json>[])\n : [];\n } catch (error) {\n log('Error fetching quotes', { error, transactionId });\n }\n\n log('Updated', { transactionId, quotes });\n\n const batchTransactions =\n quotes?.length && strategy.getBatchTransactions\n ? await strategy.getBatchTransactions({\n messenger,\n quotes,\n })\n : [];\n\n log('Batch transactions', { transactionId, batchTransactions });\n\n const totals = calculateTotals(quotes as never, tokens, messenger);\n\n log('Calculated totals', { transactionId, totals });\n\n syncTransaction({\n batchTransactions,\n messenger: messenger as never,\n paymentToken,\n totals,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.quotes = quotes as never;\n data.quotesLastUpdated = Date.now();\n data.totals = totals;\n data.isLoading = false;\n });\n}\n\n/**\n * Poll quotes at regular intervals.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nexport function queueRefreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n setTimeout(() => {\n refreshQuotes(messenger, updateTransactionData)\n .finally(() => queueRefreshQuotes(messenger, updateTransactionData))\n .catch((error) => {\n log('Error polling quotes', { messenger, error });\n });\n }, QUOTES_CHECK_INTERVAL);\n}\n\n/**\n * Sync batch transactions to the transaction meta.\n *\n * @param request - Request object.\n * @param request.batchTransactions - Batch transactions to sync.\n * @param request.messenger - Messenger instance.\n * @param request.paymentToken - Payment token used.\n * @param request.totals - Calculated totals.\n * @param request.transactionId - ID of the transaction to sync.\n */\nfunction syncTransaction({\n batchTransactions,\n messenger,\n paymentToken,\n totals,\n transactionId,\n}: {\n batchTransactions: BatchTransaction[];\n messenger: TransactionPayControllerMessenger;\n paymentToken: TransactionPaymentToken;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n updateTransaction(\n {\n transactionId,\n messenger: messenger as never,\n note: 'Update transaction pay data',\n },\n (tx: TransactionMeta) => {\n tx.batchTransactions = batchTransactions;\n tx.batchTransactionsOptions = {};\n\n tx.metamaskPay = {\n bridgeFeeFiat: totals.fees.provider.usd,\n chainId: paymentToken.chainId,\n networkFeeFiat: totals.fees.sourceNetwork.usd,\n tokenAddress: paymentToken.address,\n totalFiat: totals.total.usd,\n };\n },\n );\n}\n\n/**\n * Refresh quotes for all transactions if expired.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nasync function refreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n const state = messenger.call('TransactionPayController:getState');\n const transactionIds = Object.keys(state.transactionData);\n\n for (const transactionId of transactionIds) {\n const transactionData = state.transactionData[transactionId];\n const { isLoading, quotes, quotesLastUpdated } = transactionData;\n\n if (isLoading || !quotes?.length) {\n continue;\n }\n\n const strategyName = quotes[0].strategy;\n const strategy = getStrategyByName(strategyName);\n\n const refreshInterval =\n (await strategy.getRefreshInterval?.({\n chainId: quotes[0].request.sourceChainId,\n messenger,\n })) ?? DEFAULT_REFRESH_INTERVAL;\n\n const isExpired = Date.now() - (quotesLastUpdated ?? 0) > refreshInterval;\n\n if (!isExpired) {\n continue;\n }\n\n log('Refreshing expired quotes', {\n transactionId,\n strategy: strategyName,\n refreshInterval,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n}\n"]}
1
+ {"version":3,"file":"quotes.mjs","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import type { BatchTransaction } from '@metamask/transaction-controller';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex, Json } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\n\nimport { getStrategy, getStrategyByName } from './strategy';\nimport { calculateTotals } from './totals';\nimport { getTransaction, updateTransaction } from './transaction';\nimport { projectLogger } from '../logger';\nimport type {\n QuoteRequest,\n TransactionData,\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\nconst QUOTES_CHECK_INTERVAL = 1 * 1000; // 1 Second\nconst DEFAULT_REFRESH_INTERVAL = 30 * 1000; // 30 Seconds\n\nconst log = createModuleLogger(projectLogger, 'quotes');\n\nexport type UpdateQuotesRequest = {\n messenger: TransactionPayControllerMessenger;\n transactionData: TransactionData | undefined;\n transactionId: string;\n updateTransactionData: UpdateTransactionDataCallback;\n};\n\n/**\n * Update the quotes for a specific transaction.\n *\n * @param request - Request parameters.\n */\nexport async function updateQuotes(request: UpdateQuotesRequest) {\n const { messenger, transactionData, transactionId, updateTransactionData } =\n request;\n\n const transaction = getTransaction(transactionId, messenger);\n\n log('Updating quotes', { transactionId });\n\n if (!transaction || !transactionData) {\n throw new Error('Transaction not found');\n }\n\n const { paymentToken, sourceAmounts, tokens } = transactionData;\n\n if (!paymentToken) {\n return;\n }\n\n const requests: QuoteRequest[] = (sourceAmounts ?? []).map(\n (sourceAmount, i) => {\n const token = tokens[i];\n\n return {\n from: transaction.txParams.from as Hex,\n sourceBalanceRaw: paymentToken.balanceRaw,\n sourceTokenAmount: sourceAmount.sourceAmountRaw,\n sourceChainId: paymentToken.chainId,\n sourceTokenAddress: paymentToken.address,\n targetAmountMinimum: token.allowUnderMinimum ? '0' : token.amountRaw,\n targetChainId: token.chainId,\n targetTokenAddress: token.address,\n };\n },\n );\n\n if (!requests?.length) {\n log('No quote requests', { transactionId });\n }\n\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\n\n const strategy = await getStrategy(messenger as never, transaction);\n\n try {\n quotes = requests?.length\n ? ((await strategy.getQuotes({\n messenger,\n requests,\n transaction,\n })) as TransactionPayQuote<Json>[])\n : [];\n } catch (error) {\n log('Error fetching quotes', { error, transactionId });\n }\n\n log('Updated', { transactionId, quotes });\n\n const batchTransactions =\n quotes?.length && strategy.getBatchTransactions\n ? await strategy.getBatchTransactions({\n messenger,\n quotes,\n })\n : [];\n\n log('Batch transactions', { transactionId, batchTransactions });\n\n const totals = calculateTotals(quotes as never, tokens, messenger);\n\n log('Calculated totals', { transactionId, totals });\n\n syncTransaction({\n batchTransactions,\n messenger: messenger as never,\n paymentToken,\n totals,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.quotes = quotes as never;\n data.quotesLastUpdated = Date.now();\n data.totals = totals;\n data.isLoading = false;\n });\n}\n\n/**\n * Poll quotes at regular intervals.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nexport function queueRefreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n setTimeout(() => {\n refreshQuotes(messenger, updateTransactionData)\n .finally(() => queueRefreshQuotes(messenger, updateTransactionData))\n .catch((error) => {\n log('Error polling quotes', { messenger, error });\n });\n }, QUOTES_CHECK_INTERVAL);\n}\n\n/**\n * Sync batch transactions to the transaction meta.\n *\n * @param request - Request object.\n * @param request.batchTransactions - Batch transactions to sync.\n * @param request.messenger - Messenger instance.\n * @param request.paymentToken - Payment token used.\n * @param request.totals - Calculated totals.\n * @param request.transactionId - ID of the transaction to sync.\n */\nfunction syncTransaction({\n batchTransactions,\n messenger,\n paymentToken,\n totals,\n transactionId,\n}: {\n batchTransactions: BatchTransaction[];\n messenger: TransactionPayControllerMessenger;\n paymentToken: TransactionPaymentToken;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n updateTransaction(\n {\n transactionId,\n messenger: messenger as never,\n note: 'Update transaction pay data',\n },\n (tx: TransactionMeta) => {\n tx.batchTransactions = batchTransactions;\n tx.batchTransactionsOptions = {};\n\n tx.metamaskPay = {\n bridgeFeeFiat: totals.fees.provider.usd,\n chainId: paymentToken.chainId,\n networkFeeFiat: totals.fees.sourceNetwork.usd,\n tokenAddress: paymentToken.address,\n totalFiat: totals.total.usd,\n };\n },\n );\n}\n\n/**\n * Refresh quotes for all transactions if expired.\n *\n * @param messenger - Messenger instance.\n * @param updateTransactionData - Callback to update transaction data.\n */\nasync function refreshQuotes(\n messenger: TransactionPayControllerMessenger,\n updateTransactionData: UpdateTransactionDataCallback,\n) {\n const state = messenger.call('TransactionPayController:getState');\n const transactionIds = Object.keys(state.transactionData);\n\n for (const transactionId of transactionIds) {\n const transactionData = state.transactionData[transactionId];\n const { isLoading, quotes, quotesLastUpdated } = transactionData;\n\n if (isLoading || !quotes?.length) {\n continue;\n }\n\n const strategyName = quotes[0].strategy;\n const strategy = getStrategyByName(strategyName);\n\n const refreshInterval =\n (await strategy.getRefreshInterval?.({\n chainId: quotes[0].request.sourceChainId,\n messenger,\n })) ?? DEFAULT_REFRESH_INTERVAL;\n\n const isExpired = Date.now() - (quotesLastUpdated ?? 0) > refreshInterval;\n\n if (!isExpired) {\n continue;\n }\n\n log('Refreshing expired quotes', {\n transactionId,\n strategy: strategyName,\n refreshInterval,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@metamask-previews/transaction-pay-controller",
3
- "version": "0.0.0-preview-24bcd769",
3
+ "version": "1.0.0-preview-1ae9d4fd",
4
4
  "description": "Manages alternate payment strategies to provide required funds for transactions in MetaMask",
5
5
  "keywords": [
6
6
  "MetaMask",
@@ -68,7 +68,7 @@
68
68
  "@metamask/gas-fee-controller": "^25.0.0",
69
69
  "@metamask/network-controller": "^25.0.0",
70
70
  "@metamask/remote-feature-flag-controller": "^2.0.0",
71
- "@metamask/transaction-controller": "^61.0.0",
71
+ "@metamask/transaction-controller": "^61.1.0",
72
72
  "@ts-bridge/cli": "^0.6.1",
73
73
  "@types/jest": "^27.4.1",
74
74
  "deepmerge": "^4.2.2",