@metamask-previews/transaction-pay-controller 3.0.0-preview-e983f054 → 3.0.0-preview-e958bbad

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
@@ -11,6 +11,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
11
11
 
12
12
  - Calculate totals even if no quotes received ([#7042](https://github.com/MetaMask/core/pull/7042))
13
13
 
14
+ ### Fixed
15
+
16
+ - Fix bridging to native Polygon ([#7053](https://github.com/MetaMask/core/pull/7053))
17
+ - Use original quote if bridge quote fails to refresh during submit.
18
+ - Only refresh quotes if transaction status is unapproved.
19
+
14
20
  ## [3.0.0]
15
21
 
16
22
  ### Changed
@@ -177,7 +177,12 @@ async function getSufficientSingleBridgeQuote(quoteRequest, index, request) {
177
177
  }
178
178
  }
179
179
  if (new bignumber_js_1.BigNumber(currentSourceAmount).isGreaterThanOrEqualTo(sourceBalanceRaw)) {
180
- log('Reached balance limit', targetTokenAddress);
180
+ log('Reached balance limit', {
181
+ targetTokenAddress,
182
+ sourceBalanceRaw,
183
+ currentSourceAmount,
184
+ attempt: i + 1,
185
+ });
181
186
  break;
182
187
  }
183
188
  const newSourceAmount = originalSourceAmount.multipliedBy(1 + buffer + bufferStep * (i + 1));
@@ -304,10 +309,13 @@ function getFeatureFlags(messenger) {
304
309
  * @returns Normalized transaction pay quote.
305
310
  */
306
311
  function normalizeQuote(quote, request, messenger, transaction) {
307
- 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));
309
- if (sourceFiatRate === undefined || targetFiatRate === undefined) {
310
- throw new Error('Fiat rate not found for source or target token');
312
+ const sourceFiatRate = (0, token_1.getTokenFiatRate)(messenger, request.sourceTokenAddress, request.sourceChainId);
313
+ if (sourceFiatRate === undefined) {
314
+ throw new Error(`Fiat rate not found for source token - Chain ID: ${request.sourceChainId}, Address: ${request.sourceTokenAddress}`);
315
+ }
316
+ const targetFiatRate = (0, token_1.getTokenFiatRate)(messenger, request.targetTokenAddress, request.targetChainId);
317
+ if (targetFiatRate === undefined) {
318
+ throw new Error(`Fiat rate not found for target token - Chain ID: ${request.targetChainId}, Address: ${request.targetTokenAddress}`);
311
319
  }
312
320
  const targetAmountMinimumFiat = calculateFiatValue(quote.quote.minDestTokenAmount, quote.quote.destAsset.decimals, targetFiatRate.fiatRate, targetFiatRate.usdRate);
313
321
  const sourceAmountFiat = calculateFiatValue(quote.quote.srcTokenAmount, quote.quote.srcAsset.decimals, sourceFiatRate.fiatRate, sourceFiatRate.usdRate);
@@ -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.confirmations_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 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', {\n targetTokenAddress,\n sourceBalanceRaw,\n currentSourceAmount,\n attempt: i + 1,\n });\n\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.confirmations_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 sourceFiatRate = getTokenFiatRate(\n messenger,\n request.sourceTokenAddress,\n request.sourceChainId,\n );\n\n if (sourceFiatRate === undefined) {\n throw new Error(\n `Fiat rate not found for source token - Chain ID: ${request.sourceChainId}, Address: ${request.sourceTokenAddress}`,\n );\n }\n\n const targetFiatRate = getTokenFiatRate(\n messenger,\n request.targetTokenAddress,\n request.targetChainId,\n );\n\n if (targetFiatRate === undefined) {\n throw new Error(\n `Fiat rate not found for target token - Chain ID: ${request.targetChainId}, Address: ${request.targetTokenAddress}`,\n );\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"]}
@@ -171,7 +171,12 @@ async function getSufficientSingleBridgeQuote(quoteRequest, index, request) {
171
171
  }
172
172
  }
173
173
  if (new BigNumber(currentSourceAmount).isGreaterThanOrEqualTo(sourceBalanceRaw)) {
174
- log('Reached balance limit', targetTokenAddress);
174
+ log('Reached balance limit', {
175
+ targetTokenAddress,
176
+ sourceBalanceRaw,
177
+ currentSourceAmount,
178
+ attempt: i + 1,
179
+ });
175
180
  break;
176
181
  }
177
182
  const newSourceAmount = originalSourceAmount.multipliedBy(1 + buffer + bufferStep * (i + 1));
@@ -298,10 +303,13 @@ function getFeatureFlags(messenger) {
298
303
  * @returns Normalized transaction pay quote.
299
304
  */
300
305
  function normalizeQuote(quote, request, messenger, transaction) {
301
- const targetFiatRate = getTokenFiatRate(messenger, quote.quote.destAsset.address, toHex(quote.quote.destChainId));
302
- const sourceFiatRate = getTokenFiatRate(messenger, request.sourceTokenAddress, toHex(quote.quote.srcChainId));
303
- if (sourceFiatRate === undefined || targetFiatRate === undefined) {
304
- throw new Error('Fiat rate not found for source or target token');
306
+ const sourceFiatRate = getTokenFiatRate(messenger, request.sourceTokenAddress, request.sourceChainId);
307
+ if (sourceFiatRate === undefined) {
308
+ throw new Error(`Fiat rate not found for source token - Chain ID: ${request.sourceChainId}, Address: ${request.sourceTokenAddress}`);
309
+ }
310
+ const targetFiatRate = getTokenFiatRate(messenger, request.targetTokenAddress, request.targetChainId);
311
+ if (targetFiatRate === undefined) {
312
+ throw new Error(`Fiat rate not found for target token - Chain ID: ${request.targetChainId}, Address: ${request.targetTokenAddress}`);
305
313
  }
306
314
  const targetAmountMinimumFiat = calculateFiatValue(quote.quote.minDestTokenAmount, quote.quote.destAsset.decimals, targetFiatRate.fiatRate, targetFiatRate.usdRate);
307
315
  const sourceAmountFiat = calculateFiatValue(quote.quote.srcTokenAmount, quote.quote.srcAsset.decimals, sourceFiatRate.fiatRate, sourceFiatRate.usdRate);
@@ -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.confirmations_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 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', {\n targetTokenAddress,\n sourceBalanceRaw,\n currentSourceAmount,\n attempt: i + 1,\n });\n\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.confirmations_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 sourceFiatRate = getTokenFiatRate(\n messenger,\n request.sourceTokenAddress,\n request.sourceChainId,\n );\n\n if (sourceFiatRate === undefined) {\n throw new Error(\n `Fiat rate not found for source token - Chain ID: ${request.sourceChainId}, Address: ${request.sourceTokenAddress}`,\n );\n }\n\n const targetFiatRate = getTokenFiatRate(\n messenger,\n request.targetTokenAddress,\n request.targetChainId,\n );\n\n if (targetFiatRate === undefined) {\n throw new Error(\n `Fiat rate not found for target token - Chain ID: ${request.targetChainId}, Address: ${request.targetTokenAddress}`,\n );\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"]}
@@ -31,9 +31,15 @@ async function submitBridgeQuotes(request) {
31
31
  let index = 0;
32
32
  for (const quote of quotes) {
33
33
  log('Submitting bridge', index, quote);
34
- const finalQuote = index > 0
35
- ? await (0, bridge_quotes_1.refreshQuote)(quote, messenger, transaction)
36
- : quote.original;
34
+ let finalQuote = quote.original;
35
+ if (index > 0) {
36
+ try {
37
+ finalQuote = await (0, bridge_quotes_1.refreshQuote)(quote, messenger, transaction);
38
+ }
39
+ catch (error) {
40
+ log('Failed to refresh subsequent quote before submit', error);
41
+ }
42
+ }
37
43
  await submitBridgeTransaction(request, finalQuote);
38
44
  index += 1;
39
45
  }
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-submit.cjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":"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","sourcesContent":["import { StatusTypes } from '@metamask/bridge-controller';\nimport type { QuoteMetadata } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport type { BridgeHistoryItem } from '@metamask/bridge-status-controller';\nimport { toHex } from '@metamask/controller-utils';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { cloneDeep } from 'lodash';\n\nimport { refreshQuote } from './bridge-quotes';\nimport type { TransactionPayBridgeQuote } from './types';\nimport { projectLogger } from '../../logger';\nimport type {\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport {\n updateTransaction,\n waitForTransactionConfirmed,\n} from '../../utils/transaction';\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\nexport type SubmitBridgeQuotesRequest = {\n from: Hex;\n isSmartTransaction: (chainId: Hex) => boolean;\n messenger: TransactionPayControllerMessenger;\n quotes: TransactionPayQuote<TransactionPayBridgeQuote>[];\n transaction: TransactionMeta;\n};\n\n/**\n * Submit multiple bridge quotes sequentially and wait for their completion.\n *\n * @param request - The request object containing necessary data.\n * @returns An object containing the transaction hash if available.\n */\nexport async function submitBridgeQuotes(\n request: SubmitBridgeQuotesRequest,\n): Promise<void> {\n const { quotes, messenger, transaction } = request;\n\n if (!quotes?.length) {\n log('No quotes found');\n return;\n }\n\n // Currently we only support a single source meaning we only check the first quote.\n const isSameChain =\n quotes[0].original.quote.srcChainId ===\n quotes[0].original.quote.destChainId;\n\n if (isSameChain) {\n log(\n 'Ignoring quotes as source is same chain',\n quotes[0].original.quote.srcChainId,\n );\n return;\n }\n\n let index = 0;\n\n for (const quote of quotes) {\n log('Submitting bridge', index, quote);\n\n const finalQuote =\n index > 0\n ? await refreshQuote(quote, messenger as never, transaction)\n : quote.original;\n\n await submitBridgeTransaction(request, finalQuote);\n\n index += 1;\n }\n}\n\n/**\n * Submit a bridge transaction and wait for it to complete.\n *\n * @param request - The request object containing necessary data.\n * @param originalQuote - The original quote to submit.\n */\nasync function submitBridgeTransaction(\n request: SubmitBridgeQuotesRequest,\n originalQuote: QuoteResponse,\n): Promise<void> {\n const { isSmartTransaction, messenger, from, transaction } = request;\n const quote = cloneDeep(originalQuote);\n const sourceChainId = toHex(quote.quote.srcChainId);\n const isSTX = isSmartTransaction(sourceChainId);\n const requiredTransactionIds: string[] = [];\n\n const bridgeTransactionIdCollector = collectTransactionIds(\n sourceChainId,\n from,\n messenger,\n (id) =>\n updateTransaction(\n {\n transactionId: transaction.id,\n messenger,\n note: 'Add required transaction ID',\n },\n (transactionMeta) => {\n if (!transactionMeta.requiredTransactionIds) {\n transactionMeta.requiredTransactionIds = [];\n }\n\n transactionMeta?.requiredTransactionIds.push(id);\n requiredTransactionIds.push(id);\n },\n ),\n );\n\n const tokenAmountValues = {\n amount: '0',\n valueInCurrency: null,\n usd: null,\n };\n\n const metadata: QuoteMetadata = {\n gasFee: {\n effective: tokenAmountValues,\n max: tokenAmountValues,\n total: tokenAmountValues,\n },\n totalNetworkFee: tokenAmountValues,\n totalMaxNetworkFee: tokenAmountValues,\n toTokenAmount: tokenAmountValues,\n minToTokenAmount: tokenAmountValues,\n adjustedReturn: tokenAmountValues,\n sentAmount: tokenAmountValues,\n swapRate: '0',\n cost: tokenAmountValues,\n };\n\n const result = await messenger.call(\n 'BridgeStatusController:submitTx',\n from,\n { ...quote, ...metadata },\n isSTX,\n );\n\n bridgeTransactionIdCollector.end();\n\n log('Bridge transaction submitted', {\n requiredTransactionIds,\n result,\n });\n\n await Promise.all(\n requiredTransactionIds.map((id) =>\n waitForTransactionConfirmed(id, messenger),\n ),\n );\n\n log('All required transactions confirmed', requiredTransactionIds);\n\n const { id: bridgeTransactionId } = result;\n\n log('Waiting for bridge completion', bridgeTransactionId);\n\n await waitForBridgeCompletion(bridgeTransactionId, messenger);\n}\n\n/**\n * Wait for a bridge transaction to complete.\n *\n * @param bridgeTransactionId - The bridge transaction ID.\n * @param messenger - The controller messenger.\n */\nasync function waitForBridgeCompletion(\n bridgeTransactionId: string,\n messenger: TransactionPayControllerMessenger,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n const isComplete = (item?: BridgeHistoryItem, fn?: () => void) => {\n const status = item?.status?.status;\n\n log('Checking bridge status', status ?? 'missing');\n\n if (status === StatusTypes.COMPLETE) {\n fn?.();\n resolve();\n return true;\n }\n\n if (status === StatusTypes.FAILED) {\n fn?.();\n reject(new Error('Bridge failed'));\n return true;\n }\n\n return false;\n };\n\n const initialState = messenger.call('BridgeStatusController:getState');\n\n const initialTx = initialState.txHistory[bridgeTransactionId];\n\n if (isComplete(initialTx)) {\n return;\n }\n\n const handler = (bridgeHistory?: BridgeHistoryItem) => {\n const unsubscribe = () =>\n messenger.unsubscribe('BridgeStatusController:stateChange', handler);\n\n isComplete(bridgeHistory, unsubscribe);\n };\n\n messenger.subscribe(\n 'BridgeStatusController:stateChange',\n handler,\n (state) => state.txHistory[bridgeTransactionId],\n );\n });\n}\n\n/**\n * Collect all new transactions until `end` is called.\n *\n * @param chainId - The chain ID to filter transactions by.\n * @param from - The address to filter transactions by.\n * @param messenger - The controller messenger.\n * @param onTransaction - Callback called with each matching transaction ID.\n * @returns An object with an `end` method to stop collecting transactions.\n */\nfunction collectTransactionIds(\n chainId: Hex,\n from: Hex,\n messenger: TransactionPayControllerMessenger,\n onTransaction: (transactionId: string) => void,\n): { end: () => void } {\n const listener = (tx: TransactionMeta) => {\n if (\n tx.chainId !== chainId ||\n tx.txParams.from.toLowerCase() !== from.toLowerCase()\n ) {\n return;\n }\n\n onTransaction(tx.id);\n };\n\n messenger.subscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n\n const end = () => {\n messenger.unsubscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n };\n\n return { end };\n}\n"]}
1
+ {"version":3,"file":"bridge-submit.cjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":";;;AAAA,mEAA0D;AAI1D,iEAAmD;AAGnD,2CAAqD;AACrD,mCAAmC;AAEnC,uDAA+C;AAE/C,6CAA6C;AAK7C,6DAGiC;AAEjC,MAAM,GAAG,GAAG,IAAA,0BAAkB,EAAC,sBAAa,EAAE,iBAAiB,CAAC,CAAC;AAUjE;;;;;GAKG;AACI,KAAK,UAAU,kBAAkB,CACtC,OAAkC;IAElC,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,WAAW,EAAE,GAAG,OAAO,CAAC;IAEnD,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE;QACnB,GAAG,CAAC,iBAAiB,CAAC,CAAC;QACvB,OAAO;KACR;IAED,mFAAmF;IACnF,MAAM,WAAW,GACf,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU;QACnC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,WAAW,CAAC;IAEvC,IAAI,WAAW,EAAE;QACf,GAAG,CACD,yCAAyC,EACzC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,UAAU,CACpC,CAAC;QACF,OAAO;KACR;IAED,IAAI,KAAK,GAAG,CAAC,CAAC;IAEd,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE;QAC1B,GAAG,CAAC,mBAAmB,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC;QAEvC,IAAI,UAAU,GAAG,KAAK,CAAC,QAAQ,CAAC;QAEhC,IAAI,KAAK,GAAG,CAAC,EAAE;YACb,IAAI;gBACF,UAAU,GAAG,MAAM,IAAA,4BAAY,EAAC,KAAK,EAAE,SAAS,EAAE,WAAW,CAAC,CAAC;aAChE;YAAC,OAAO,KAAK,EAAE;gBACd,GAAG,CAAC,kDAAkD,EAAE,KAAK,CAAC,CAAC;aAChE;SACF;QAED,MAAM,uBAAuB,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAEnD,KAAK,IAAI,CAAC,CAAC;KACZ;AACH,CAAC;AA1CD,gDA0CC;AAED;;;;;GAKG;AACH,KAAK,UAAU,uBAAuB,CACpC,OAAkC,EAClC,aAA4B;IAE5B,MAAM,EAAE,kBAAkB,EAAE,SAAS,EAAE,IAAI,EAAE,WAAW,EAAE,GAAG,OAAO,CAAC;IACrE,MAAM,KAAK,GAAG,IAAA,kBAAS,EAAC,aAAa,CAAC,CAAC;IACvC,MAAM,aAAa,GAAG,IAAA,wBAAK,EAAC,KAAK,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;IACpD,MAAM,KAAK,GAAG,kBAAkB,CAAC,aAAa,CAAC,CAAC;IAChD,MAAM,sBAAsB,GAAa,EAAE,CAAC;IAE5C,MAAM,4BAA4B,GAAG,qBAAqB,CACxD,aAAa,EACb,IAAI,EACJ,SAAS,EACT,CAAC,EAAE,EAAE,EAAE,CACL,IAAA,+BAAiB,EACf;QACE,aAAa,EAAE,WAAW,CAAC,EAAE;QAC7B,SAAS;QACT,IAAI,EAAE,6BAA6B;KACpC,EACD,CAAC,eAAe,EAAE,EAAE;QAClB,IAAI,CAAC,eAAe,CAAC,sBAAsB,EAAE;YAC3C,eAAe,CAAC,sBAAsB,GAAG,EAAE,CAAC;SAC7C;QAED,eAAe,EAAE,sBAAsB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjD,sBAAsB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAClC,CAAC,CACF,CACJ,CAAC;IAEF,MAAM,iBAAiB,GAAG;QACxB,MAAM,EAAE,GAAG;QACX,eAAe,EAAE,IAAI;QACrB,GAAG,EAAE,IAAI;KACV,CAAC;IAEF,MAAM,QAAQ,GAAkB;QAC9B,MAAM,EAAE;YACN,SAAS,EAAE,iBAAiB;YAC5B,GAAG,EAAE,iBAAiB;YACtB,KAAK,EAAE,iBAAiB;SACzB;QACD,eAAe,EAAE,iBAAiB;QAClC,kBAAkB,EAAE,iBAAiB;QACrC,aAAa,EAAE,iBAAiB;QAChC,gBAAgB,EAAE,iBAAiB;QACnC,cAAc,EAAE,iBAAiB;QACjC,UAAU,EAAE,iBAAiB;QAC7B,QAAQ,EAAE,GAAG;QACb,IAAI,EAAE,iBAAiB;KACxB,CAAC;IAEF,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,IAAI,CACjC,iCAAiC,EACjC,IAAI,EACJ,EAAE,GAAG,KAAK,EAAE,GAAG,QAAQ,EAAE,EACzB,KAAK,CACN,CAAC;IAEF,4BAA4B,CAAC,GAAG,EAAE,CAAC;IAEnC,GAAG,CAAC,8BAA8B,EAAE;QAClC,sBAAsB;QACtB,MAAM;KACP,CAAC,CAAC;IAEH,MAAM,OAAO,CAAC,GAAG,CACf,sBAAsB,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,CAChC,IAAA,yCAA2B,EAAC,EAAE,EAAE,SAAS,CAAC,CAC3C,CACF,CAAC;IAEF,GAAG,CAAC,qCAAqC,EAAE,sBAAsB,CAAC,CAAC;IAEnE,MAAM,EAAE,EAAE,EAAE,mBAAmB,EAAE,GAAG,MAAM,CAAC;IAE3C,GAAG,CAAC,+BAA+B,EAAE,mBAAmB,CAAC,CAAC;IAE1D,MAAM,uBAAuB,CAAC,mBAAmB,EAAE,SAAS,CAAC,CAAC;AAChE,CAAC;AAED;;;;;GAKG;AACH,KAAK,UAAU,uBAAuB,CACpC,mBAA2B,EAC3B,SAA4C;IAE5C,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,MAAM,UAAU,GAAG,CAAC,IAAwB,EAAE,EAAe,EAAE,EAAE;YAC/D,MAAM,MAAM,GAAG,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC;YAEpC,GAAG,CAAC,wBAAwB,EAAE,MAAM,IAAI,SAAS,CAAC,CAAC;YAEnD,IAAI,MAAM,KAAK,+BAAW,CAAC,QAAQ,EAAE;gBACnC,EAAE,EAAE,EAAE,CAAC;gBACP,OAAO,EAAE,CAAC;gBACV,OAAO,IAAI,CAAC;aACb;YAED,IAAI,MAAM,KAAK,+BAAW,CAAC,MAAM,EAAE;gBACjC,EAAE,EAAE,EAAE,CAAC;gBACP,MAAM,CAAC,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC,CAAC;gBACnC,OAAO,IAAI,CAAC;aACb;YAED,OAAO,KAAK,CAAC;QACf,CAAC,CAAC;QAEF,MAAM,YAAY,GAAG,SAAS,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;QAEvE,MAAM,SAAS,GAAG,YAAY,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;QAE9D,IAAI,UAAU,CAAC,SAAS,CAAC,EAAE;YACzB,OAAO;SACR;QAED,MAAM,OAAO,GAAG,CAAC,aAAiC,EAAE,EAAE;YACpD,MAAM,WAAW,GAAG,GAAG,EAAE,CACvB,SAAS,CAAC,WAAW,CAAC,oCAAoC,EAAE,OAAO,CAAC,CAAC;YAEvE,UAAU,CAAC,aAAa,EAAE,WAAW,CAAC,CAAC;QACzC,CAAC,CAAC;QAEF,SAAS,CAAC,SAAS,CACjB,oCAAoC,EACpC,OAAO,EACP,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAChD,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;GAQG;AACH,SAAS,qBAAqB,CAC5B,OAAY,EACZ,IAAS,EACT,SAA4C,EAC5C,aAA8C;IAE9C,MAAM,QAAQ,GAAG,CAAC,EAAmB,EAAE,EAAE;QACvC,IACE,EAAE,CAAC,OAAO,KAAK,OAAO;YACtB,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,IAAI,CAAC,WAAW,EAAE,EACrD;YACA,OAAO;SACR;QAED,aAAa,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;IACvB,CAAC,CAAC;IAEF,SAAS,CAAC,SAAS,CACjB,kDAAkD,EAClD,QAAQ,CACT,CAAC;IAEF,MAAM,GAAG,GAAG,GAAG,EAAE;QACf,SAAS,CAAC,WAAW,CACnB,kDAAkD,EAClD,QAAQ,CACT,CAAC;IACJ,CAAC,CAAC;IAEF,OAAO,EAAE,GAAG,EAAE,CAAC;AACjB,CAAC","sourcesContent":["import { StatusTypes } from '@metamask/bridge-controller';\nimport type { QuoteMetadata } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport type { BridgeHistoryItem } from '@metamask/bridge-status-controller';\nimport { toHex } from '@metamask/controller-utils';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { cloneDeep } from 'lodash';\n\nimport { refreshQuote } from './bridge-quotes';\nimport type { TransactionPayBridgeQuote } from './types';\nimport { projectLogger } from '../../logger';\nimport type {\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport {\n updateTransaction,\n waitForTransactionConfirmed,\n} from '../../utils/transaction';\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\nexport type SubmitBridgeQuotesRequest = {\n from: Hex;\n isSmartTransaction: (chainId: Hex) => boolean;\n messenger: TransactionPayControllerMessenger;\n quotes: TransactionPayQuote<TransactionPayBridgeQuote>[];\n transaction: TransactionMeta;\n};\n\n/**\n * Submit multiple bridge quotes sequentially and wait for their completion.\n *\n * @param request - The request object containing necessary data.\n * @returns An object containing the transaction hash if available.\n */\nexport async function submitBridgeQuotes(\n request: SubmitBridgeQuotesRequest,\n): Promise<void> {\n const { quotes, messenger, transaction } = request;\n\n if (!quotes?.length) {\n log('No quotes found');\n return;\n }\n\n // Currently we only support a single source meaning we only check the first quote.\n const isSameChain =\n quotes[0].original.quote.srcChainId ===\n quotes[0].original.quote.destChainId;\n\n if (isSameChain) {\n log(\n 'Ignoring quotes as source is same chain',\n quotes[0].original.quote.srcChainId,\n );\n return;\n }\n\n let index = 0;\n\n for (const quote of quotes) {\n log('Submitting bridge', index, quote);\n\n let finalQuote = quote.original;\n\n if (index > 0) {\n try {\n finalQuote = await refreshQuote(quote, messenger, transaction);\n } catch (error) {\n log('Failed to refresh subsequent quote before submit', error);\n }\n }\n\n await submitBridgeTransaction(request, finalQuote);\n\n index += 1;\n }\n}\n\n/**\n * Submit a bridge transaction and wait for it to complete.\n *\n * @param request - The request object containing necessary data.\n * @param originalQuote - The original quote to submit.\n */\nasync function submitBridgeTransaction(\n request: SubmitBridgeQuotesRequest,\n originalQuote: QuoteResponse,\n): Promise<void> {\n const { isSmartTransaction, messenger, from, transaction } = request;\n const quote = cloneDeep(originalQuote);\n const sourceChainId = toHex(quote.quote.srcChainId);\n const isSTX = isSmartTransaction(sourceChainId);\n const requiredTransactionIds: string[] = [];\n\n const bridgeTransactionIdCollector = collectTransactionIds(\n sourceChainId,\n from,\n messenger,\n (id) =>\n updateTransaction(\n {\n transactionId: transaction.id,\n messenger,\n note: 'Add required transaction ID',\n },\n (transactionMeta) => {\n if (!transactionMeta.requiredTransactionIds) {\n transactionMeta.requiredTransactionIds = [];\n }\n\n transactionMeta?.requiredTransactionIds.push(id);\n requiredTransactionIds.push(id);\n },\n ),\n );\n\n const tokenAmountValues = {\n amount: '0',\n valueInCurrency: null,\n usd: null,\n };\n\n const metadata: QuoteMetadata = {\n gasFee: {\n effective: tokenAmountValues,\n max: tokenAmountValues,\n total: tokenAmountValues,\n },\n totalNetworkFee: tokenAmountValues,\n totalMaxNetworkFee: tokenAmountValues,\n toTokenAmount: tokenAmountValues,\n minToTokenAmount: tokenAmountValues,\n adjustedReturn: tokenAmountValues,\n sentAmount: tokenAmountValues,\n swapRate: '0',\n cost: tokenAmountValues,\n };\n\n const result = await messenger.call(\n 'BridgeStatusController:submitTx',\n from,\n { ...quote, ...metadata },\n isSTX,\n );\n\n bridgeTransactionIdCollector.end();\n\n log('Bridge transaction submitted', {\n requiredTransactionIds,\n result,\n });\n\n await Promise.all(\n requiredTransactionIds.map((id) =>\n waitForTransactionConfirmed(id, messenger),\n ),\n );\n\n log('All required transactions confirmed', requiredTransactionIds);\n\n const { id: bridgeTransactionId } = result;\n\n log('Waiting for bridge completion', bridgeTransactionId);\n\n await waitForBridgeCompletion(bridgeTransactionId, messenger);\n}\n\n/**\n * Wait for a bridge transaction to complete.\n *\n * @param bridgeTransactionId - The bridge transaction ID.\n * @param messenger - The controller messenger.\n */\nasync function waitForBridgeCompletion(\n bridgeTransactionId: string,\n messenger: TransactionPayControllerMessenger,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n const isComplete = (item?: BridgeHistoryItem, fn?: () => void) => {\n const status = item?.status?.status;\n\n log('Checking bridge status', status ?? 'missing');\n\n if (status === StatusTypes.COMPLETE) {\n fn?.();\n resolve();\n return true;\n }\n\n if (status === StatusTypes.FAILED) {\n fn?.();\n reject(new Error('Bridge failed'));\n return true;\n }\n\n return false;\n };\n\n const initialState = messenger.call('BridgeStatusController:getState');\n\n const initialTx = initialState.txHistory[bridgeTransactionId];\n\n if (isComplete(initialTx)) {\n return;\n }\n\n const handler = (bridgeHistory?: BridgeHistoryItem) => {\n const unsubscribe = () =>\n messenger.unsubscribe('BridgeStatusController:stateChange', handler);\n\n isComplete(bridgeHistory, unsubscribe);\n };\n\n messenger.subscribe(\n 'BridgeStatusController:stateChange',\n handler,\n (state) => state.txHistory[bridgeTransactionId],\n );\n });\n}\n\n/**\n * Collect all new transactions until `end` is called.\n *\n * @param chainId - The chain ID to filter transactions by.\n * @param from - The address to filter transactions by.\n * @param messenger - The controller messenger.\n * @param onTransaction - Callback called with each matching transaction ID.\n * @returns An object with an `end` method to stop collecting transactions.\n */\nfunction collectTransactionIds(\n chainId: Hex,\n from: Hex,\n messenger: TransactionPayControllerMessenger,\n onTransaction: (transactionId: string) => void,\n): { end: () => void } {\n const listener = (tx: TransactionMeta) => {\n if (\n tx.chainId !== chainId ||\n tx.txParams.from.toLowerCase() !== from.toLowerCase()\n ) {\n return;\n }\n\n onTransaction(tx.id);\n };\n\n messenger.subscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n\n const end = () => {\n messenger.unsubscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n };\n\n return { end };\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-submit.d.cts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AACxE,OAAO,KAAK,EAAE,GAAG,EAAE,wBAAwB;AAK3C,OAAO,KAAK,EAAE,yBAAyB,EAAE,oBAAgB;AAEzD,OAAO,KAAK,EACV,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAQrB,MAAM,MAAM,yBAAyB,GAAG;IACtC,IAAI,EAAE,GAAG,CAAC;IACV,kBAAkB,EAAE,CAAC,OAAO,EAAE,GAAG,KAAK,OAAO,CAAC;IAC9C,SAAS,EAAE,iCAAiC,CAAC;IAC7C,MAAM,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC;IACzD,WAAW,EAAE,eAAe,CAAC;CAC9B,CAAC;AAEF;;;;;GAKG;AACH,wBAAsB,kBAAkB,CACtC,OAAO,EAAE,yBAAyB,GACjC,OAAO,CAAC,IAAI,CAAC,CAmCf"}
1
+ {"version":3,"file":"bridge-submit.d.cts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AACxE,OAAO,KAAK,EAAE,GAAG,EAAE,wBAAwB;AAK3C,OAAO,KAAK,EAAE,yBAAyB,EAAE,oBAAgB;AAEzD,OAAO,KAAK,EACV,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAQrB,MAAM,MAAM,yBAAyB,GAAG;IACtC,IAAI,EAAE,GAAG,CAAC;IACV,kBAAkB,EAAE,CAAC,OAAO,EAAE,GAAG,KAAK,OAAO,CAAC;IAC9C,SAAS,EAAE,iCAAiC,CAAC;IAC7C,MAAM,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC;IACzD,WAAW,EAAE,eAAe,CAAC;CAC9B,CAAC;AAEF;;;;;GAKG;AACH,wBAAsB,kBAAkB,CACtC,OAAO,EAAE,yBAAyB,GACjC,OAAO,CAAC,IAAI,CAAC,CAwCf"}
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-submit.d.mts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AACxE,OAAO,KAAK,EAAE,GAAG,EAAE,wBAAwB;AAK3C,OAAO,KAAK,EAAE,yBAAyB,EAAE,oBAAgB;AAEzD,OAAO,KAAK,EACV,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAQrB,MAAM,MAAM,yBAAyB,GAAG;IACtC,IAAI,EAAE,GAAG,CAAC;IACV,kBAAkB,EAAE,CAAC,OAAO,EAAE,GAAG,KAAK,OAAO,CAAC;IAC9C,SAAS,EAAE,iCAAiC,CAAC;IAC7C,MAAM,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC;IACzD,WAAW,EAAE,eAAe,CAAC;CAC9B,CAAC;AAEF;;;;;GAKG;AACH,wBAAsB,kBAAkB,CACtC,OAAO,EAAE,yBAAyB,GACjC,OAAO,CAAC,IAAI,CAAC,CAmCf"}
1
+ {"version":3,"file":"bridge-submit.d.mts","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,eAAe,EAAE,yCAAyC;AACxE,OAAO,KAAK,EAAE,GAAG,EAAE,wBAAwB;AAK3C,OAAO,KAAK,EAAE,yBAAyB,EAAE,oBAAgB;AAEzD,OAAO,KAAK,EACV,iCAAiC,EACjC,mBAAmB,EACpB,wBAAoB;AAQrB,MAAM,MAAM,yBAAyB,GAAG;IACtC,IAAI,EAAE,GAAG,CAAC;IACV,kBAAkB,EAAE,CAAC,OAAO,EAAE,GAAG,KAAK,OAAO,CAAC;IAC9C,SAAS,EAAE,iCAAiC,CAAC;IAC7C,MAAM,EAAE,mBAAmB,CAAC,yBAAyB,CAAC,EAAE,CAAC;IACzD,WAAW,EAAE,eAAe,CAAC;CAC9B,CAAC;AAEF;;;;;GAKG;AACH,wBAAsB,kBAAkB,CACtC,OAAO,EAAE,yBAAyB,GACjC,OAAO,CAAC,IAAI,CAAC,CAwCf"}
@@ -29,9 +29,15 @@ export async function submitBridgeQuotes(request) {
29
29
  let index = 0;
30
30
  for (const quote of quotes) {
31
31
  log('Submitting bridge', index, quote);
32
- const finalQuote = index > 0
33
- ? await refreshQuote(quote, messenger, transaction)
34
- : quote.original;
32
+ let finalQuote = quote.original;
33
+ if (index > 0) {
34
+ try {
35
+ finalQuote = await refreshQuote(quote, messenger, transaction);
36
+ }
37
+ catch (error) {
38
+ log('Failed to refresh subsequent quote before submit', error);
39
+ }
40
+ }
35
41
  await submitBridgeTransaction(request, finalQuote);
36
42
  index += 1;
37
43
  }
@@ -1 +1 @@
1
- {"version":3,"file":"bridge-submit.mjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":"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","sourcesContent":["import { StatusTypes } from '@metamask/bridge-controller';\nimport type { QuoteMetadata } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport type { BridgeHistoryItem } from '@metamask/bridge-status-controller';\nimport { toHex } from '@metamask/controller-utils';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { cloneDeep } from 'lodash';\n\nimport { refreshQuote } from './bridge-quotes';\nimport type { TransactionPayBridgeQuote } from './types';\nimport { projectLogger } from '../../logger';\nimport type {\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport {\n updateTransaction,\n waitForTransactionConfirmed,\n} from '../../utils/transaction';\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\nexport type SubmitBridgeQuotesRequest = {\n from: Hex;\n isSmartTransaction: (chainId: Hex) => boolean;\n messenger: TransactionPayControllerMessenger;\n quotes: TransactionPayQuote<TransactionPayBridgeQuote>[];\n transaction: TransactionMeta;\n};\n\n/**\n * Submit multiple bridge quotes sequentially and wait for their completion.\n *\n * @param request - The request object containing necessary data.\n * @returns An object containing the transaction hash if available.\n */\nexport async function submitBridgeQuotes(\n request: SubmitBridgeQuotesRequest,\n): Promise<void> {\n const { quotes, messenger, transaction } = request;\n\n if (!quotes?.length) {\n log('No quotes found');\n return;\n }\n\n // Currently we only support a single source meaning we only check the first quote.\n const isSameChain =\n quotes[0].original.quote.srcChainId ===\n quotes[0].original.quote.destChainId;\n\n if (isSameChain) {\n log(\n 'Ignoring quotes as source is same chain',\n quotes[0].original.quote.srcChainId,\n );\n return;\n }\n\n let index = 0;\n\n for (const quote of quotes) {\n log('Submitting bridge', index, quote);\n\n const finalQuote =\n index > 0\n ? await refreshQuote(quote, messenger as never, transaction)\n : quote.original;\n\n await submitBridgeTransaction(request, finalQuote);\n\n index += 1;\n }\n}\n\n/**\n * Submit a bridge transaction and wait for it to complete.\n *\n * @param request - The request object containing necessary data.\n * @param originalQuote - The original quote to submit.\n */\nasync function submitBridgeTransaction(\n request: SubmitBridgeQuotesRequest,\n originalQuote: QuoteResponse,\n): Promise<void> {\n const { isSmartTransaction, messenger, from, transaction } = request;\n const quote = cloneDeep(originalQuote);\n const sourceChainId = toHex(quote.quote.srcChainId);\n const isSTX = isSmartTransaction(sourceChainId);\n const requiredTransactionIds: string[] = [];\n\n const bridgeTransactionIdCollector = collectTransactionIds(\n sourceChainId,\n from,\n messenger,\n (id) =>\n updateTransaction(\n {\n transactionId: transaction.id,\n messenger,\n note: 'Add required transaction ID',\n },\n (transactionMeta) => {\n if (!transactionMeta.requiredTransactionIds) {\n transactionMeta.requiredTransactionIds = [];\n }\n\n transactionMeta?.requiredTransactionIds.push(id);\n requiredTransactionIds.push(id);\n },\n ),\n );\n\n const tokenAmountValues = {\n amount: '0',\n valueInCurrency: null,\n usd: null,\n };\n\n const metadata: QuoteMetadata = {\n gasFee: {\n effective: tokenAmountValues,\n max: tokenAmountValues,\n total: tokenAmountValues,\n },\n totalNetworkFee: tokenAmountValues,\n totalMaxNetworkFee: tokenAmountValues,\n toTokenAmount: tokenAmountValues,\n minToTokenAmount: tokenAmountValues,\n adjustedReturn: tokenAmountValues,\n sentAmount: tokenAmountValues,\n swapRate: '0',\n cost: tokenAmountValues,\n };\n\n const result = await messenger.call(\n 'BridgeStatusController:submitTx',\n from,\n { ...quote, ...metadata },\n isSTX,\n );\n\n bridgeTransactionIdCollector.end();\n\n log('Bridge transaction submitted', {\n requiredTransactionIds,\n result,\n });\n\n await Promise.all(\n requiredTransactionIds.map((id) =>\n waitForTransactionConfirmed(id, messenger),\n ),\n );\n\n log('All required transactions confirmed', requiredTransactionIds);\n\n const { id: bridgeTransactionId } = result;\n\n log('Waiting for bridge completion', bridgeTransactionId);\n\n await waitForBridgeCompletion(bridgeTransactionId, messenger);\n}\n\n/**\n * Wait for a bridge transaction to complete.\n *\n * @param bridgeTransactionId - The bridge transaction ID.\n * @param messenger - The controller messenger.\n */\nasync function waitForBridgeCompletion(\n bridgeTransactionId: string,\n messenger: TransactionPayControllerMessenger,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n const isComplete = (item?: BridgeHistoryItem, fn?: () => void) => {\n const status = item?.status?.status;\n\n log('Checking bridge status', status ?? 'missing');\n\n if (status === StatusTypes.COMPLETE) {\n fn?.();\n resolve();\n return true;\n }\n\n if (status === StatusTypes.FAILED) {\n fn?.();\n reject(new Error('Bridge failed'));\n return true;\n }\n\n return false;\n };\n\n const initialState = messenger.call('BridgeStatusController:getState');\n\n const initialTx = initialState.txHistory[bridgeTransactionId];\n\n if (isComplete(initialTx)) {\n return;\n }\n\n const handler = (bridgeHistory?: BridgeHistoryItem) => {\n const unsubscribe = () =>\n messenger.unsubscribe('BridgeStatusController:stateChange', handler);\n\n isComplete(bridgeHistory, unsubscribe);\n };\n\n messenger.subscribe(\n 'BridgeStatusController:stateChange',\n handler,\n (state) => state.txHistory[bridgeTransactionId],\n );\n });\n}\n\n/**\n * Collect all new transactions until `end` is called.\n *\n * @param chainId - The chain ID to filter transactions by.\n * @param from - The address to filter transactions by.\n * @param messenger - The controller messenger.\n * @param onTransaction - Callback called with each matching transaction ID.\n * @returns An object with an `end` method to stop collecting transactions.\n */\nfunction collectTransactionIds(\n chainId: Hex,\n from: Hex,\n messenger: TransactionPayControllerMessenger,\n onTransaction: (transactionId: string) => void,\n): { end: () => void } {\n const listener = (tx: TransactionMeta) => {\n if (\n tx.chainId !== chainId ||\n tx.txParams.from.toLowerCase() !== from.toLowerCase()\n ) {\n return;\n }\n\n onTransaction(tx.id);\n };\n\n messenger.subscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n\n const end = () => {\n messenger.unsubscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n };\n\n return { end };\n}\n"]}
1
+ {"version":3,"file":"bridge-submit.mjs","sourceRoot":"","sources":["../../../src/strategy/bridge/bridge-submit.ts"],"names":[],"mappings":"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","sourcesContent":["import { StatusTypes } from '@metamask/bridge-controller';\nimport type { QuoteMetadata } from '@metamask/bridge-controller';\nimport type { QuoteResponse } from '@metamask/bridge-controller';\nimport type { BridgeHistoryItem } from '@metamask/bridge-status-controller';\nimport { toHex } from '@metamask/controller-utils';\nimport type { TransactionMeta } from '@metamask/transaction-controller';\nimport type { Hex } from '@metamask/utils';\nimport { createModuleLogger } from '@metamask/utils';\nimport { cloneDeep } from 'lodash';\n\nimport { refreshQuote } from './bridge-quotes';\nimport type { TransactionPayBridgeQuote } from './types';\nimport { projectLogger } from '../../logger';\nimport type {\n TransactionPayControllerMessenger,\n TransactionPayQuote,\n} from '../../types';\nimport {\n updateTransaction,\n waitForTransactionConfirmed,\n} from '../../utils/transaction';\n\nconst log = createModuleLogger(projectLogger, 'bridge-strategy');\n\nexport type SubmitBridgeQuotesRequest = {\n from: Hex;\n isSmartTransaction: (chainId: Hex) => boolean;\n messenger: TransactionPayControllerMessenger;\n quotes: TransactionPayQuote<TransactionPayBridgeQuote>[];\n transaction: TransactionMeta;\n};\n\n/**\n * Submit multiple bridge quotes sequentially and wait for their completion.\n *\n * @param request - The request object containing necessary data.\n * @returns An object containing the transaction hash if available.\n */\nexport async function submitBridgeQuotes(\n request: SubmitBridgeQuotesRequest,\n): Promise<void> {\n const { quotes, messenger, transaction } = request;\n\n if (!quotes?.length) {\n log('No quotes found');\n return;\n }\n\n // Currently we only support a single source meaning we only check the first quote.\n const isSameChain =\n quotes[0].original.quote.srcChainId ===\n quotes[0].original.quote.destChainId;\n\n if (isSameChain) {\n log(\n 'Ignoring quotes as source is same chain',\n quotes[0].original.quote.srcChainId,\n );\n return;\n }\n\n let index = 0;\n\n for (const quote of quotes) {\n log('Submitting bridge', index, quote);\n\n let finalQuote = quote.original;\n\n if (index > 0) {\n try {\n finalQuote = await refreshQuote(quote, messenger, transaction);\n } catch (error) {\n log('Failed to refresh subsequent quote before submit', error);\n }\n }\n\n await submitBridgeTransaction(request, finalQuote);\n\n index += 1;\n }\n}\n\n/**\n * Submit a bridge transaction and wait for it to complete.\n *\n * @param request - The request object containing necessary data.\n * @param originalQuote - The original quote to submit.\n */\nasync function submitBridgeTransaction(\n request: SubmitBridgeQuotesRequest,\n originalQuote: QuoteResponse,\n): Promise<void> {\n const { isSmartTransaction, messenger, from, transaction } = request;\n const quote = cloneDeep(originalQuote);\n const sourceChainId = toHex(quote.quote.srcChainId);\n const isSTX = isSmartTransaction(sourceChainId);\n const requiredTransactionIds: string[] = [];\n\n const bridgeTransactionIdCollector = collectTransactionIds(\n sourceChainId,\n from,\n messenger,\n (id) =>\n updateTransaction(\n {\n transactionId: transaction.id,\n messenger,\n note: 'Add required transaction ID',\n },\n (transactionMeta) => {\n if (!transactionMeta.requiredTransactionIds) {\n transactionMeta.requiredTransactionIds = [];\n }\n\n transactionMeta?.requiredTransactionIds.push(id);\n requiredTransactionIds.push(id);\n },\n ),\n );\n\n const tokenAmountValues = {\n amount: '0',\n valueInCurrency: null,\n usd: null,\n };\n\n const metadata: QuoteMetadata = {\n gasFee: {\n effective: tokenAmountValues,\n max: tokenAmountValues,\n total: tokenAmountValues,\n },\n totalNetworkFee: tokenAmountValues,\n totalMaxNetworkFee: tokenAmountValues,\n toTokenAmount: tokenAmountValues,\n minToTokenAmount: tokenAmountValues,\n adjustedReturn: tokenAmountValues,\n sentAmount: tokenAmountValues,\n swapRate: '0',\n cost: tokenAmountValues,\n };\n\n const result = await messenger.call(\n 'BridgeStatusController:submitTx',\n from,\n { ...quote, ...metadata },\n isSTX,\n );\n\n bridgeTransactionIdCollector.end();\n\n log('Bridge transaction submitted', {\n requiredTransactionIds,\n result,\n });\n\n await Promise.all(\n requiredTransactionIds.map((id) =>\n waitForTransactionConfirmed(id, messenger),\n ),\n );\n\n log('All required transactions confirmed', requiredTransactionIds);\n\n const { id: bridgeTransactionId } = result;\n\n log('Waiting for bridge completion', bridgeTransactionId);\n\n await waitForBridgeCompletion(bridgeTransactionId, messenger);\n}\n\n/**\n * Wait for a bridge transaction to complete.\n *\n * @param bridgeTransactionId - The bridge transaction ID.\n * @param messenger - The controller messenger.\n */\nasync function waitForBridgeCompletion(\n bridgeTransactionId: string,\n messenger: TransactionPayControllerMessenger,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n const isComplete = (item?: BridgeHistoryItem, fn?: () => void) => {\n const status = item?.status?.status;\n\n log('Checking bridge status', status ?? 'missing');\n\n if (status === StatusTypes.COMPLETE) {\n fn?.();\n resolve();\n return true;\n }\n\n if (status === StatusTypes.FAILED) {\n fn?.();\n reject(new Error('Bridge failed'));\n return true;\n }\n\n return false;\n };\n\n const initialState = messenger.call('BridgeStatusController:getState');\n\n const initialTx = initialState.txHistory[bridgeTransactionId];\n\n if (isComplete(initialTx)) {\n return;\n }\n\n const handler = (bridgeHistory?: BridgeHistoryItem) => {\n const unsubscribe = () =>\n messenger.unsubscribe('BridgeStatusController:stateChange', handler);\n\n isComplete(bridgeHistory, unsubscribe);\n };\n\n messenger.subscribe(\n 'BridgeStatusController:stateChange',\n handler,\n (state) => state.txHistory[bridgeTransactionId],\n );\n });\n}\n\n/**\n * Collect all new transactions until `end` is called.\n *\n * @param chainId - The chain ID to filter transactions by.\n * @param from - The address to filter transactions by.\n * @param messenger - The controller messenger.\n * @param onTransaction - Callback called with each matching transaction ID.\n * @returns An object with an `end` method to stop collecting transactions.\n */\nfunction collectTransactionIds(\n chainId: Hex,\n from: Hex,\n messenger: TransactionPayControllerMessenger,\n onTransaction: (transactionId: string) => void,\n): { end: () => void } {\n const listener = (tx: TransactionMeta) => {\n if (\n tx.chainId !== chainId ||\n tx.txParams.from.toLowerCase() !== from.toLowerCase()\n ) {\n return;\n }\n\n onTransaction(tx.id);\n };\n\n messenger.subscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n\n const end = () => {\n messenger.unsubscribe(\n 'TransactionController:unapprovedTransactionAdded',\n listener,\n );\n };\n\n return { end };\n}\n"]}
@@ -1,6 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.refreshQuotes = exports.updateQuotes = void 0;
4
+ const transaction_controller_1 = require("@metamask/transaction-controller");
4
5
  const utils_1 = require("@metamask/utils");
5
6
  const strategy_1 = require("./strategy.cjs");
6
7
  const totals_1 = require("./totals.cjs");
@@ -12,14 +13,18 @@ const log = (0, utils_1.createModuleLogger)(logger_1.projectLogger, 'quotes');
12
13
  * Update the quotes for a specific transaction.
13
14
  *
14
15
  * @param request - Request parameters.
16
+ * @returns Boolean indicating if the quotes were updated.
15
17
  */
16
18
  async function updateQuotes(request) {
17
19
  const { messenger, transactionData, transactionId, updateTransactionData } = request;
18
20
  const transaction = (0, transaction_1.getTransaction)(transactionId, messenger);
19
- log('Updating quotes', { transactionId });
20
21
  if (!transaction || !transactionData) {
21
22
  throw new Error('Transaction not found');
22
23
  }
24
+ if (transaction?.status !== transaction_controller_1.TransactionStatus.unapproved) {
25
+ return false;
26
+ }
27
+ log('Updating quotes', { transactionId });
23
28
  const { paymentToken, sourceAmounts, tokens } = transactionData;
24
29
  const requests = buildQuoteRequests({
25
30
  from: transaction.txParams.from,
@@ -58,6 +63,7 @@ async function updateQuotes(request) {
58
63
  data.isLoading = false;
59
64
  });
60
65
  }
66
+ return true;
61
67
  }
62
68
  exports.updateQuotes = updateQuotes;
63
69
  /**
@@ -115,18 +121,15 @@ async function refreshQuotes(messenger, updateTransactionData) {
115
121
  if (!isExpired) {
116
122
  continue;
117
123
  }
118
- log('Refreshing expired quotes', {
119
- transactionId,
120
- strategy: strategyName,
121
- refreshInterval,
122
- });
123
- await updateQuotes({
124
+ const isUpdated = await updateQuotes({
124
125
  messenger,
125
126
  transactionData,
126
127
  transactionId,
127
128
  updateTransactionData,
128
129
  });
129
- log('Refreshed quotes', { transactionId, strategy: strategyName });
130
+ if (isUpdated) {
131
+ log('Refreshed quotes', { transactionId, strategy: strategyName });
132
+ }
130
133
  }
131
134
  }
132
135
  exports.refreshQuotes = refreshQuotes;
@@ -1 +1 @@
1
- {"version":3,"file":"quotes.cjs","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":";;;AAGA,2CAAqD;AAErD,6CAA4D;AAC5D,yCAA2C;AAC3C,mDAAkE;AAClE,0CAA0C;AAa1C,MAAM,wBAAwB,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,aAAa;AAEzD,MAAM,GAAG,GAAG,IAAA,0BAAkB,EAAC,sBAAa,EAAE,QAAQ,CAAC,CAAC;AASxD;;;;GAIG;AACI,KAAK,UAAU,YAAY,CAAC,OAA4B;IAC7D,MAAM,EAAE,SAAS,EAAE,eAAe,EAAE,aAAa,EAAE,qBAAqB,EAAE,GACxE,OAAO,CAAC;IAEV,MAAM,WAAW,GAAG,IAAA,4BAAc,EAAC,aAAa,EAAE,SAAS,CAAC,CAAC;IAE7D,GAAG,CAAC,iBAAiB,EAAE,EAAE,aAAa,EAAE,CAAC,CAAC;IAE1C,IAAI,CAAC,WAAW,IAAI,CAAC,eAAe,EAAE;QACpC,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC;KAC1C;IAED,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,EAAE,GAAG,eAAe,CAAC;IAEhE,MAAM,QAAQ,GAAG,kBAAkB,CAAC;QAClC,IAAI,EAAE,WAAW,CAAC,QAAQ,CAAC,IAAW;QACtC,YAAY;QACZ,aAAa;QACb,MAAM;QACN,aAAa;KACd,CAAC,CAAC;IAEH,qBAAqB,CAAC,aAAa,EAAE,CAAC,IAAI,EAAE,EAAE;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,IAAI;QACF,MAAM,EAAE,iBAAiB,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CACnD,WAAW,EACX,QAAQ,EACR,SAAS,CACV,CAAC;QAEF,MAAM,MAAM,GAAG,IAAA,wBAAe,EAAC;YAC7B,MAAM,EAAE,MAAwC;YAChD,SAAS;YACT,MAAM;YACN,WAAW;SACZ,CAAC,CAAC;QAEH,GAAG,CAAC,mBAAmB,EAAE,EAAE,aAAa,EAAE,MAAM,EAAE,CAAC,CAAC;QAEpD,eAAe,CAAC;YACd,iBAAiB;YACjB,SAAS,EAAE,SAAkB;YAC7B,YAAY;YACZ,MAAM;YACN,aAAa;SACd,CAAC,CAAC;QAEH,qBAAqB,CAAC,aAAa,EAAE,CAAC,IAAI,EAAE,EAAE;YAC5C,IAAI,CAAC,MAAM,GAAG,MAAe,CAAC;YAC9B,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YACpC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACvB,CAAC,CAAC,CAAC;KACJ;YAAS;QACR,qBAAqB,CAAC,aAAa,EAAE,CAAC,IAAI,EAAE,EAAE;YAC5C,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACzB,CAAC,CAAC,CAAC;KACJ;AACH,CAAC;AA5DD,oCA4DC;AAED;;;;;;;;;GASG;AACH,SAAS,eAAe,CAAC,EACvB,iBAAiB,EACjB,SAAS,EACT,YAAY,EACZ,MAAM,EACN,aAAa,GAOd;IACC,IAAI,CAAC,YAAY,EAAE;QACjB,OAAO;KACR;IAED,IAAA,+BAAiB,EACf;QACE,aAAa;QACb,SAAS,EAAE,SAAkB;QAC7B,IAAI,EAAE,6BAA6B;KACpC,EACD,CAAC,EAAmB,EAAE,EAAE;QACtB,EAAE,CAAC,iBAAiB,GAAG,iBAAiB,CAAC;QACzC,EAAE,CAAC,wBAAwB,GAAG,EAAE,CAAC;QAEjC,EAAE,CAAC,WAAW,GAAG;YACf,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG;YACvC,OAAO,EAAE,YAAY,CAAC,OAAO;YAC7B,cAAc,EAAE,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG;YAC7C,YAAY,EAAE,YAAY,CAAC,OAAO;YAClC,SAAS,EAAE,MAAM,CAAC,KAAK,CAAC,GAAG;SAC5B,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC;AAED;;;;;GAKG;AACI,KAAK,UAAU,aAAa,CACjC,SAA4C,EAC5C,qBAAoD;IAEpD,MAAM,KAAK,GAAG,SAAS,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;IAClE,MAAM,cAAc,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC;IAE1D,KAAK,MAAM,aAAa,IAAI,cAAc,EAAE;QAC1C,MAAM,eAAe,GAAG,KAAK,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC;QAC7D,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,iBAAiB,EAAE,GAAG,eAAe,CAAC;QAEjE,IAAI,SAAS,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE;YAChC,SAAS;SACV;QAED,MAAM,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC;QACxC,MAAM,QAAQ,GAAG,IAAA,4BAAiB,EAAC,YAAY,CAAC,CAAC;QAEjD,MAAM,eAAe,GACnB,CAAC,MAAM,QAAQ,CAAC,kBAAkB,EAAE,CAAC;YACnC,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa;YACxC,SAAS;SACV,CAAC,CAAC,IAAI,wBAAwB,CAAC;QAElC,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,iBAAiB,IAAI,CAAC,CAAC,GAAG,eAAe,CAAC;QAE1E,IAAI,CAAC,SAAS,EAAE;YACd,SAAS;SACV;QAED,GAAG,CAAC,2BAA2B,EAAE;YAC/B,aAAa;YACb,QAAQ,EAAE,YAAY;YACtB,eAAe;SAChB,CAAC,CAAC;QAEH,MAAM,YAAY,CAAC;YACjB,SAAS;YACT,eAAe;YACf,aAAa;YACb,qBAAqB;SACtB,CAAC,CAAC;QAEH,GAAG,CAAC,kBAAkB,EAAE,EAAE,aAAa,EAAE,QAAQ,EAAE,YAAY,EAAE,CAAC,CAAC;KACpE;AACH,CAAC;AA7CD,sCA6CC;AAED;;;;;;;;;;GAUG;AACH,SAAS,kBAAkB,CAAC,EAC1B,IAAI,EACJ,YAAY,EACZ,aAAa,EACb,MAAM,EACN,aAAa,GAOd;IACC,IAAI,CAAC,YAAY,EAAE;QACjB,OAAO,EAAE,CAAC;KACX;IAED,MAAM,QAAQ,GAAG,CAAC,aAAa,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,YAAY,EAAE,EAAE;QAC1D,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CACvB,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,YAAY,CAAC,kBAAkB,CACtB,CAAC;QAEjC,OAAO;YACL,IAAI;YACJ,gBAAgB,EAAE,YAAY,CAAC,UAAU;YACzC,iBAAiB,EAAE,YAAY,CAAC,eAAe;YAC/C,aAAa,EAAE,YAAY,CAAC,OAAO;YACnC,kBAAkB,EAAE,YAAY,CAAC,OAAO;YACxC,mBAAmB,EAAE,KAAK,CAAC,iBAAiB,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS;YACpE,aAAa,EAAE,KAAK,CAAC,OAAO;YAC5B,kBAAkB,EAAE,KAAK,CAAC,OAAO;SAClC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE;QACpB,GAAG,CAAC,mBAAmB,EAAE,EAAE,aAAa,EAAE,CAAC,CAAC;KAC7C;IAED,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED;;;;;;;GAOG;AACH,KAAK,UAAU,SAAS,CACtB,WAA4B,EAC5B,QAAwB,EACxB,SAA4C;IAE5C,MAAM,EAAE,EAAE,EAAE,aAAa,EAAE,GAAG,WAAW,CAAC;IAC1C,MAAM,QAAQ,GAAG,MAAM,IAAA,sBAAW,EAAC,SAAkB,EAAE,WAAW,CAAC,CAAC;IACpE,IAAI,MAAM,GAA4C,EAAE,CAAC;IAEzD,IAAI;QACF,MAAM,GAAG,QAAQ,EAAE,MAAM;YACvB,CAAC,CAAE,CAAC,MAAM,QAAQ,CAAC,SAAS,CAAC;gBACzB,SAAS;gBACT,QAAQ;gBACR,WAAW;aACZ,CAAC,CAAiC;YACrC,CAAC,CAAC,EAAE,CAAC;KACR;IAAC,OAAO,KAAK,EAAE;QACd,GAAG,CAAC,uBAAuB,EAAE,EAAE,KAAK,EAAE,aAAa,EAAE,CAAC,CAAC;KACxD;IAED,GAAG,CAAC,SAAS,EAAE,EAAE,aAAa,EAAE,MAAM,EAAE,CAAC,CAAC;IAE1C,MAAM,iBAAiB,GACrB,MAAM,EAAE,MAAM,IAAI,QAAQ,CAAC,oBAAoB;QAC7C,CAAC,CAAC,MAAM,QAAQ,CAAC,oBAAoB,CAAC;YAClC,SAAS;YACT,MAAM;SACP,CAAC;QACJ,CAAC,CAAC,EAAE,CAAC;IAET,GAAG,CAAC,oBAAoB,EAAE,EAAE,aAAa,EAAE,iBAAiB,EAAE,CAAC,CAAC;IAEhE,OAAO;QACL,iBAAiB;QACjB,MAAM;KACP,CAAC;AACJ,CAAC","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 TransactionPaySourceAmount,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\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 const requests = buildQuoteRequests({\n from: transaction.txParams.from as Hex,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n try {\n const { batchTransactions, quotes } = await getQuotes(\n transaction,\n requests,\n messenger,\n );\n\n const totals = calculateTotals({\n quotes: quotes as TransactionPayQuote<unknown>[],\n messenger,\n tokens,\n transaction,\n });\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 });\n } finally {\n updateTransactionData(transactionId, (data) => {\n data.isLoading = false;\n });\n }\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 | undefined;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n if (!paymentToken) {\n return;\n }\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 */\nexport async 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 await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n}\n\n/**\n * Build quote requests required to retrieve quotes.\n *\n * @param request - Request parameters.\n * @param request.from - Address from which the transaction is sent.\n * @param request.paymentToken - Payment token used for the transaction.\n * @param request.sourceAmounts - Source amounts for the transaction.\n * @param request.tokens - Required tokens for the transaction.\n * @param request.transactionId - ID of the transaction.\n * @returns Array of quote requests.\n */\nfunction buildQuoteRequests({\n from,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n}: {\n from: Hex;\n paymentToken: TransactionPaymentToken | undefined;\n sourceAmounts: TransactionPaySourceAmount[] | undefined;\n tokens: TransactionPayRequiredToken[];\n transactionId: string;\n}): QuoteRequest[] {\n if (!paymentToken) {\n return [];\n }\n\n const requests = (sourceAmounts ?? []).map((sourceAmount) => {\n const token = tokens.find(\n (t) => t.address === sourceAmount.targetTokenAddress,\n ) as TransactionPayRequiredToken;\n\n return {\n from,\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 return requests;\n}\n\n/**\n * Retrieve quotes for a transaction.\n *\n * @param transaction - Transaction metadata.\n * @param requests - Quote requests.\n * @param messenger - Controller messenger.\n * @returns An object containing batch transactions and quotes.\n */\nasync function getQuotes(\n transaction: TransactionMeta,\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n) {\n const { id: transactionId } = transaction;\n const strategy = await getStrategy(messenger as never, transaction);\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\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 return {\n batchTransactions,\n quotes,\n };\n}\n"]}
1
+ {"version":3,"file":"quotes.cjs","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n TransactionStatus,\n type BatchTransaction,\n} 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 TransactionPaySourceAmount,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\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 * @returns Boolean indicating if the quotes were updated.\n */\nexport async function updateQuotes(\n request: UpdateQuotesRequest,\n): Promise<boolean> {\n const { messenger, transactionData, transactionId, updateTransactionData } =\n request;\n\n const transaction = getTransaction(transactionId, messenger);\n\n if (!transaction || !transactionData) {\n throw new Error('Transaction not found');\n }\n\n if (transaction?.status !== TransactionStatus.unapproved) {\n return false;\n }\n\n log('Updating quotes', { transactionId });\n\n const { paymentToken, sourceAmounts, tokens } = transactionData;\n\n const requests = buildQuoteRequests({\n from: transaction.txParams.from as Hex,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n try {\n const { batchTransactions, quotes } = await getQuotes(\n transaction,\n requests,\n messenger,\n );\n\n const totals = calculateTotals({\n quotes: quotes as TransactionPayQuote<unknown>[],\n messenger,\n tokens,\n transaction,\n });\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 });\n } finally {\n updateTransactionData(transactionId, (data) => {\n data.isLoading = false;\n });\n }\n\n return true;\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 | undefined;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n if (!paymentToken) {\n return;\n }\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 */\nexport async 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 const isUpdated = await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n if (isUpdated) {\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n }\n}\n\n/**\n * Build quote requests required to retrieve quotes.\n *\n * @param request - Request parameters.\n * @param request.from - Address from which the transaction is sent.\n * @param request.paymentToken - Payment token used for the transaction.\n * @param request.sourceAmounts - Source amounts for the transaction.\n * @param request.tokens - Required tokens for the transaction.\n * @param request.transactionId - ID of the transaction.\n * @returns Array of quote requests.\n */\nfunction buildQuoteRequests({\n from,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n}: {\n from: Hex;\n paymentToken: TransactionPaymentToken | undefined;\n sourceAmounts: TransactionPaySourceAmount[] | undefined;\n tokens: TransactionPayRequiredToken[];\n transactionId: string;\n}): QuoteRequest[] {\n if (!paymentToken) {\n return [];\n }\n\n const requests = (sourceAmounts ?? []).map((sourceAmount) => {\n const token = tokens.find(\n (t) => t.address === sourceAmount.targetTokenAddress,\n ) as TransactionPayRequiredToken;\n\n return {\n from,\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 return requests;\n}\n\n/**\n * Retrieve quotes for a transaction.\n *\n * @param transaction - Transaction metadata.\n * @param requests - Quote requests.\n * @param messenger - Controller messenger.\n * @returns An object containing batch transactions and quotes.\n */\nasync function getQuotes(\n transaction: TransactionMeta,\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n) {\n const { id: transactionId } = transaction;\n const strategy = await getStrategy(messenger as never, transaction);\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\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 return {\n batchTransactions,\n quotes,\n };\n}\n"]}
@@ -9,8 +9,9 @@ export type UpdateQuotesRequest = {
9
9
  * Update the quotes for a specific transaction.
10
10
  *
11
11
  * @param request - Request parameters.
12
+ * @returns Boolean indicating if the quotes were updated.
12
13
  */
13
- export declare function updateQuotes(request: UpdateQuotesRequest): Promise<void>;
14
+ export declare function updateQuotes(request: UpdateQuotesRequest): Promise<boolean>;
14
15
  /**
15
16
  * Refresh quotes for all transactions if expired.
16
17
  *
@@ -1 +1 @@
1
- {"version":3,"file":"quotes.d.cts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAMjC,6BAA6B,EAC9B,qBAAiB;AAMlB,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,iBA4D9D;AAkDD;;;;;GAKG;AACH,wBAAsB,aAAa,CACjC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,iBA2CrD"}
1
+ {"version":3,"file":"quotes.d.cts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAMjC,6BAA6B,EAC9B,qBAAiB;AAMlB,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;;;;;GAKG;AACH,wBAAsB,YAAY,CAChC,OAAO,EAAE,mBAAmB,GAC3B,OAAO,CAAC,OAAO,CAAC,CAkElB;AAkDD;;;;;GAKG;AACH,wBAAsB,aAAa,CACjC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,iBAuCrD"}
@@ -9,8 +9,9 @@ export type UpdateQuotesRequest = {
9
9
  * Update the quotes for a specific transaction.
10
10
  *
11
11
  * @param request - Request parameters.
12
+ * @returns Boolean indicating if the quotes were updated.
12
13
  */
13
- export declare function updateQuotes(request: UpdateQuotesRequest): Promise<void>;
14
+ export declare function updateQuotes(request: UpdateQuotesRequest): Promise<boolean>;
14
15
  /**
15
16
  * Refresh quotes for all transactions if expired.
16
17
  *
@@ -1 +1 @@
1
- {"version":3,"file":"quotes.d.mts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAMjC,6BAA6B,EAC9B,qBAAiB;AAMlB,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,iBA4D9D;AAkDD;;;;;GAKG;AACH,wBAAsB,aAAa,CACjC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,iBA2CrD"}
1
+ {"version":3,"file":"quotes.d.mts","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AAYA,OAAO,KAAK,EAEV,eAAe,EACf,iCAAiC,EAMjC,6BAA6B,EAC9B,qBAAiB;AAMlB,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;;;;;GAKG;AACH,wBAAsB,YAAY,CAChC,OAAO,EAAE,mBAAmB,GAC3B,OAAO,CAAC,OAAO,CAAC,CAkElB;AAkDD;;;;;GAKG;AACH,wBAAsB,aAAa,CACjC,SAAS,EAAE,iCAAiC,EAC5C,qBAAqB,EAAE,6BAA6B,iBAuCrD"}
@@ -1,3 +1,4 @@
1
+ import { TransactionStatus } from "@metamask/transaction-controller";
1
2
  import { createModuleLogger } from "@metamask/utils";
2
3
  import { getStrategy, getStrategyByName } from "./strategy.mjs";
3
4
  import { calculateTotals } from "./totals.mjs";
@@ -9,14 +10,18 @@ const log = createModuleLogger(projectLogger, 'quotes');
9
10
  * Update the quotes for a specific transaction.
10
11
  *
11
12
  * @param request - Request parameters.
13
+ * @returns Boolean indicating if the quotes were updated.
12
14
  */
13
15
  export async function updateQuotes(request) {
14
16
  const { messenger, transactionData, transactionId, updateTransactionData } = request;
15
17
  const transaction = getTransaction(transactionId, messenger);
16
- log('Updating quotes', { transactionId });
17
18
  if (!transaction || !transactionData) {
18
19
  throw new Error('Transaction not found');
19
20
  }
21
+ if (transaction?.status !== TransactionStatus.unapproved) {
22
+ return false;
23
+ }
24
+ log('Updating quotes', { transactionId });
20
25
  const { paymentToken, sourceAmounts, tokens } = transactionData;
21
26
  const requests = buildQuoteRequests({
22
27
  from: transaction.txParams.from,
@@ -55,6 +60,7 @@ export async function updateQuotes(request) {
55
60
  data.isLoading = false;
56
61
  });
57
62
  }
63
+ return true;
58
64
  }
59
65
  /**
60
66
  * Sync batch transactions to the transaction meta.
@@ -111,18 +117,15 @@ export async function refreshQuotes(messenger, updateTransactionData) {
111
117
  if (!isExpired) {
112
118
  continue;
113
119
  }
114
- log('Refreshing expired quotes', {
115
- transactionId,
116
- strategy: strategyName,
117
- refreshInterval,
118
- });
119
- await updateQuotes({
120
+ const isUpdated = await updateQuotes({
120
121
  messenger,
121
122
  transactionData,
122
123
  transactionId,
123
124
  updateTransactionData,
124
125
  });
125
- log('Refreshed quotes', { transactionId, strategy: strategyName });
126
+ if (isUpdated) {
127
+ log('Refreshed quotes', { transactionId, strategy: strategyName });
128
+ }
126
129
  }
127
130
  }
128
131
  /**
@@ -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 TransactionPaySourceAmount,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\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 const requests = buildQuoteRequests({\n from: transaction.txParams.from as Hex,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n try {\n const { batchTransactions, quotes } = await getQuotes(\n transaction,\n requests,\n messenger,\n );\n\n const totals = calculateTotals({\n quotes: quotes as TransactionPayQuote<unknown>[],\n messenger,\n tokens,\n transaction,\n });\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 });\n } finally {\n updateTransactionData(transactionId, (data) => {\n data.isLoading = false;\n });\n }\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 | undefined;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n if (!paymentToken) {\n return;\n }\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 */\nexport async 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 await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n}\n\n/**\n * Build quote requests required to retrieve quotes.\n *\n * @param request - Request parameters.\n * @param request.from - Address from which the transaction is sent.\n * @param request.paymentToken - Payment token used for the transaction.\n * @param request.sourceAmounts - Source amounts for the transaction.\n * @param request.tokens - Required tokens for the transaction.\n * @param request.transactionId - ID of the transaction.\n * @returns Array of quote requests.\n */\nfunction buildQuoteRequests({\n from,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n}: {\n from: Hex;\n paymentToken: TransactionPaymentToken | undefined;\n sourceAmounts: TransactionPaySourceAmount[] | undefined;\n tokens: TransactionPayRequiredToken[];\n transactionId: string;\n}): QuoteRequest[] {\n if (!paymentToken) {\n return [];\n }\n\n const requests = (sourceAmounts ?? []).map((sourceAmount) => {\n const token = tokens.find(\n (t) => t.address === sourceAmount.targetTokenAddress,\n ) as TransactionPayRequiredToken;\n\n return {\n from,\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 return requests;\n}\n\n/**\n * Retrieve quotes for a transaction.\n *\n * @param transaction - Transaction metadata.\n * @param requests - Quote requests.\n * @param messenger - Controller messenger.\n * @returns An object containing batch transactions and quotes.\n */\nasync function getQuotes(\n transaction: TransactionMeta,\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n) {\n const { id: transactionId } = transaction;\n const strategy = await getStrategy(messenger as never, transaction);\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\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 return {\n batchTransactions,\n quotes,\n };\n}\n"]}
1
+ {"version":3,"file":"quotes.mjs","sourceRoot":"","sources":["../../src/utils/quotes.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,iBAAiB,EAElB,yCAAyC;AAG1C,OAAO,EAAE,kBAAkB,EAAE,wBAAwB;AAErD,OAAO,EAAE,WAAW,EAAE,iBAAiB,EAAE,uBAAmB;AAC5D,OAAO,EAAE,eAAe,EAAE,qBAAiB;AAC3C,OAAO,EAAE,cAAc,EAAE,iBAAiB,EAAE,0BAAsB;AAClE,OAAO,EAAE,aAAa,EAAE,sBAAkB;AAa1C,MAAM,wBAAwB,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,aAAa;AAEzD,MAAM,GAAG,GAAG,kBAAkB,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAC;AASxD;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,YAAY,CAChC,OAA4B;IAE5B,MAAM,EAAE,SAAS,EAAE,eAAe,EAAE,aAAa,EAAE,qBAAqB,EAAE,GACxE,OAAO,CAAC;IAEV,MAAM,WAAW,GAAG,cAAc,CAAC,aAAa,EAAE,SAAS,CAAC,CAAC;IAE7D,IAAI,CAAC,WAAW,IAAI,CAAC,eAAe,EAAE;QACpC,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC;KAC1C;IAED,IAAI,WAAW,EAAE,MAAM,KAAK,iBAAiB,CAAC,UAAU,EAAE;QACxD,OAAO,KAAK,CAAC;KACd;IAED,GAAG,CAAC,iBAAiB,EAAE,EAAE,aAAa,EAAE,CAAC,CAAC;IAE1C,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,EAAE,GAAG,eAAe,CAAC;IAEhE,MAAM,QAAQ,GAAG,kBAAkB,CAAC;QAClC,IAAI,EAAE,WAAW,CAAC,QAAQ,CAAC,IAAW;QACtC,YAAY;QACZ,aAAa;QACb,MAAM;QACN,aAAa;KACd,CAAC,CAAC;IAEH,qBAAqB,CAAC,aAAa,EAAE,CAAC,IAAI,EAAE,EAAE;QAC5C,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,IAAI;QACF,MAAM,EAAE,iBAAiB,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CACnD,WAAW,EACX,QAAQ,EACR,SAAS,CACV,CAAC;QAEF,MAAM,MAAM,GAAG,eAAe,CAAC;YAC7B,MAAM,EAAE,MAAwC;YAChD,SAAS;YACT,MAAM;YACN,WAAW;SACZ,CAAC,CAAC;QAEH,GAAG,CAAC,mBAAmB,EAAE,EAAE,aAAa,EAAE,MAAM,EAAE,CAAC,CAAC;QAEpD,eAAe,CAAC;YACd,iBAAiB;YACjB,SAAS,EAAE,SAAkB;YAC7B,YAAY;YACZ,MAAM;YACN,aAAa;SACd,CAAC,CAAC;QAEH,qBAAqB,CAAC,aAAa,EAAE,CAAC,IAAI,EAAE,EAAE;YAC5C,IAAI,CAAC,MAAM,GAAG,MAAe,CAAC;YAC9B,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YACpC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACvB,CAAC,CAAC,CAAC;KACJ;YAAS;QACR,qBAAqB,CAAC,aAAa,EAAE,CAAC,IAAI,EAAE,EAAE;YAC5C,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACzB,CAAC,CAAC,CAAC;KACJ;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;;;;;;;GASG;AACH,SAAS,eAAe,CAAC,EACvB,iBAAiB,EACjB,SAAS,EACT,YAAY,EACZ,MAAM,EACN,aAAa,GAOd;IACC,IAAI,CAAC,YAAY,EAAE;QACjB,OAAO;KACR;IAED,iBAAiB,CACf;QACE,aAAa;QACb,SAAS,EAAE,SAAkB;QAC7B,IAAI,EAAE,6BAA6B;KACpC,EACD,CAAC,EAAmB,EAAE,EAAE;QACtB,EAAE,CAAC,iBAAiB,GAAG,iBAAiB,CAAC;QACzC,EAAE,CAAC,wBAAwB,GAAG,EAAE,CAAC;QAEjC,EAAE,CAAC,WAAW,GAAG;YACf,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG;YACvC,OAAO,EAAE,YAAY,CAAC,OAAO;YAC7B,cAAc,EAAE,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,GAAG;YAC7C,YAAY,EAAE,YAAY,CAAC,OAAO;YAClC,SAAS,EAAE,MAAM,CAAC,KAAK,CAAC,GAAG;SAC5B,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC;AAED;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,aAAa,CACjC,SAA4C,EAC5C,qBAAoD;IAEpD,MAAM,KAAK,GAAG,SAAS,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;IAClE,MAAM,cAAc,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC;IAE1D,KAAK,MAAM,aAAa,IAAI,cAAc,EAAE;QAC1C,MAAM,eAAe,GAAG,KAAK,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC;QAC7D,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,iBAAiB,EAAE,GAAG,eAAe,CAAC;QAEjE,IAAI,SAAS,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE;YAChC,SAAS;SACV;QAED,MAAM,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC;QACxC,MAAM,QAAQ,GAAG,iBAAiB,CAAC,YAAY,CAAC,CAAC;QAEjD,MAAM,eAAe,GACnB,CAAC,MAAM,QAAQ,CAAC,kBAAkB,EAAE,CAAC;YACnC,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa;YACxC,SAAS;SACV,CAAC,CAAC,IAAI,wBAAwB,CAAC;QAElC,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,iBAAiB,IAAI,CAAC,CAAC,GAAG,eAAe,CAAC;QAE1E,IAAI,CAAC,SAAS,EAAE;YACd,SAAS;SACV;QAED,MAAM,SAAS,GAAG,MAAM,YAAY,CAAC;YACnC,SAAS;YACT,eAAe;YACf,aAAa;YACb,qBAAqB;SACtB,CAAC,CAAC;QAEH,IAAI,SAAS,EAAE;YACb,GAAG,CAAC,kBAAkB,EAAE,EAAE,aAAa,EAAE,QAAQ,EAAE,YAAY,EAAE,CAAC,CAAC;SACpE;KACF;AACH,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAS,kBAAkB,CAAC,EAC1B,IAAI,EACJ,YAAY,EACZ,aAAa,EACb,MAAM,EACN,aAAa,GAOd;IACC,IAAI,CAAC,YAAY,EAAE;QACjB,OAAO,EAAE,CAAC;KACX;IAED,MAAM,QAAQ,GAAG,CAAC,aAAa,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,YAAY,EAAE,EAAE;QAC1D,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CACvB,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,KAAK,YAAY,CAAC,kBAAkB,CACtB,CAAC;QAEjC,OAAO;YACL,IAAI;YACJ,gBAAgB,EAAE,YAAY,CAAC,UAAU;YACzC,iBAAiB,EAAE,YAAY,CAAC,eAAe;YAC/C,aAAa,EAAE,YAAY,CAAC,OAAO;YACnC,kBAAkB,EAAE,YAAY,CAAC,OAAO;YACxC,mBAAmB,EAAE,KAAK,CAAC,iBAAiB,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS;YACpE,aAAa,EAAE,KAAK,CAAC,OAAO;YAC5B,kBAAkB,EAAE,KAAK,CAAC,OAAO;SAClC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE;QACpB,GAAG,CAAC,mBAAmB,EAAE,EAAE,aAAa,EAAE,CAAC,CAAC;KAC7C;IAED,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED;;;;;;;GAOG;AACH,KAAK,UAAU,SAAS,CACtB,WAA4B,EAC5B,QAAwB,EACxB,SAA4C;IAE5C,MAAM,EAAE,EAAE,EAAE,aAAa,EAAE,GAAG,WAAW,CAAC;IAC1C,MAAM,QAAQ,GAAG,MAAM,WAAW,CAAC,SAAkB,EAAE,WAAW,CAAC,CAAC;IACpE,IAAI,MAAM,GAA4C,EAAE,CAAC;IAEzD,IAAI;QACF,MAAM,GAAG,QAAQ,EAAE,MAAM;YACvB,CAAC,CAAE,CAAC,MAAM,QAAQ,CAAC,SAAS,CAAC;gBACzB,SAAS;gBACT,QAAQ;gBACR,WAAW;aACZ,CAAC,CAAiC;YACrC,CAAC,CAAC,EAAE,CAAC;KACR;IAAC,OAAO,KAAK,EAAE;QACd,GAAG,CAAC,uBAAuB,EAAE,EAAE,KAAK,EAAE,aAAa,EAAE,CAAC,CAAC;KACxD;IAED,GAAG,CAAC,SAAS,EAAE,EAAE,aAAa,EAAE,MAAM,EAAE,CAAC,CAAC;IAE1C,MAAM,iBAAiB,GACrB,MAAM,EAAE,MAAM,IAAI,QAAQ,CAAC,oBAAoB;QAC7C,CAAC,CAAC,MAAM,QAAQ,CAAC,oBAAoB,CAAC;YAClC,SAAS;YACT,MAAM;SACP,CAAC;QACJ,CAAC,CAAC,EAAE,CAAC;IAET,GAAG,CAAC,oBAAoB,EAAE,EAAE,aAAa,EAAE,iBAAiB,EAAE,CAAC,CAAC;IAEhE,OAAO;QACL,iBAAiB;QACjB,MAAM;KACP,CAAC;AACJ,CAAC","sourcesContent":["import {\n TransactionStatus,\n type BatchTransaction,\n} 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 TransactionPaySourceAmount,\n TransactionPayTotals,\n TransactionPaymentToken,\n UpdateTransactionDataCallback,\n} from '../types';\n\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 * @returns Boolean indicating if the quotes were updated.\n */\nexport async function updateQuotes(\n request: UpdateQuotesRequest,\n): Promise<boolean> {\n const { messenger, transactionData, transactionId, updateTransactionData } =\n request;\n\n const transaction = getTransaction(transactionId, messenger);\n\n if (!transaction || !transactionData) {\n throw new Error('Transaction not found');\n }\n\n if (transaction?.status !== TransactionStatus.unapproved) {\n return false;\n }\n\n log('Updating quotes', { transactionId });\n\n const { paymentToken, sourceAmounts, tokens } = transactionData;\n\n const requests = buildQuoteRequests({\n from: transaction.txParams.from as Hex,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n });\n\n updateTransactionData(transactionId, (data) => {\n data.isLoading = true;\n });\n\n try {\n const { batchTransactions, quotes } = await getQuotes(\n transaction,\n requests,\n messenger,\n );\n\n const totals = calculateTotals({\n quotes: quotes as TransactionPayQuote<unknown>[],\n messenger,\n tokens,\n transaction,\n });\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 });\n } finally {\n updateTransactionData(transactionId, (data) => {\n data.isLoading = false;\n });\n }\n\n return true;\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 | undefined;\n totals: TransactionPayTotals;\n transactionId: string;\n}) {\n if (!paymentToken) {\n return;\n }\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 */\nexport async 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 const isUpdated = await updateQuotes({\n messenger,\n transactionData,\n transactionId,\n updateTransactionData,\n });\n\n if (isUpdated) {\n log('Refreshed quotes', { transactionId, strategy: strategyName });\n }\n }\n}\n\n/**\n * Build quote requests required to retrieve quotes.\n *\n * @param request - Request parameters.\n * @param request.from - Address from which the transaction is sent.\n * @param request.paymentToken - Payment token used for the transaction.\n * @param request.sourceAmounts - Source amounts for the transaction.\n * @param request.tokens - Required tokens for the transaction.\n * @param request.transactionId - ID of the transaction.\n * @returns Array of quote requests.\n */\nfunction buildQuoteRequests({\n from,\n paymentToken,\n sourceAmounts,\n tokens,\n transactionId,\n}: {\n from: Hex;\n paymentToken: TransactionPaymentToken | undefined;\n sourceAmounts: TransactionPaySourceAmount[] | undefined;\n tokens: TransactionPayRequiredToken[];\n transactionId: string;\n}): QuoteRequest[] {\n if (!paymentToken) {\n return [];\n }\n\n const requests = (sourceAmounts ?? []).map((sourceAmount) => {\n const token = tokens.find(\n (t) => t.address === sourceAmount.targetTokenAddress,\n ) as TransactionPayRequiredToken;\n\n return {\n from,\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 return requests;\n}\n\n/**\n * Retrieve quotes for a transaction.\n *\n * @param transaction - Transaction metadata.\n * @param requests - Quote requests.\n * @param messenger - Controller messenger.\n * @returns An object containing batch transactions and quotes.\n */\nasync function getQuotes(\n transaction: TransactionMeta,\n requests: QuoteRequest[],\n messenger: TransactionPayControllerMessenger,\n) {\n const { id: transactionId } = transaction;\n const strategy = await getStrategy(messenger as never, transaction);\n let quotes: TransactionPayQuote<Json>[] | undefined = [];\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 return {\n batchTransactions,\n quotes,\n };\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@metamask-previews/transaction-pay-controller",
3
- "version": "3.0.0-preview-e983f054",
3
+ "version": "3.0.0-preview-e958bbad",
4
4
  "description": "Manages alternate payment strategies to provide required funds for transactions in MetaMask",
5
5
  "keywords": [
6
6
  "MetaMask",
@@ -62,7 +62,7 @@
62
62
  "lodash": "^4.17.21"
63
63
  },
64
64
  "devDependencies": {
65
- "@metamask/assets-controllers": "^87.0.0",
65
+ "@metamask/assets-controllers": "^87.1.0",
66
66
  "@metamask/auto-changelog": "^3.4.4",
67
67
  "@metamask/bridge-controller": "^59.0.0",
68
68
  "@metamask/bridge-status-controller": "^59.0.0",