From a89b2f54878865e5ef0e704e8be19cf0d3c9dc9c Mon Sep 17 00:00:00 2001 From: alter-eggo Date: Thu, 18 Apr 2024 16:56:01 +0400 Subject: [PATCH] fix: check utxo fallback error, closes #5250 --- src/app/query/bitcoin/bitcoin-client.ts | 4 +- .../bitcoin/transaction/use-check-utxos.ts | 23 ++++--- src/shared/constants.ts | 2 +- tests/specs/send/send-btc.spec.ts | 62 ++++++++++++++++++- 4 files changed, 74 insertions(+), 17 deletions(-) diff --git a/src/app/query/bitcoin/bitcoin-client.ts b/src/app/query/bitcoin/bitcoin-client.ts index 5f86e5633bb..347bb9de872 100644 --- a/src/app/query/bitcoin/bitcoin-client.ts +++ b/src/app/query/bitcoin/bitcoin-client.ts @@ -54,12 +54,12 @@ interface BestinslotInscription { byte_size: number; } -export interface BestinslotInscriptionByIdResponse { +interface BestinslotInscriptionByIdResponse { data: BestinslotInscription; block_height: number; } -export interface BestinslotInscriptionsByTxIdResponse { +interface BestinslotInscriptionsByTxIdResponse { data: { inscription_id: string }[]; blockHeight: number; } diff --git a/src/app/query/bitcoin/transaction/use-check-utxos.ts b/src/app/query/bitcoin/transaction/use-check-utxos.ts index 8ecbda64b4b..570112898b1 100644 --- a/src/app/query/bitcoin/transaction/use-check-utxos.ts +++ b/src/app/query/bitcoin/transaction/use-check-utxos.ts @@ -10,10 +10,7 @@ import { useAnalytics } from '@app/common/hooks/analytics/use-analytics'; import { useBitcoinClient } from '@app/store/common/api-clients.hooks'; import { useCurrentNetworkState } from '@app/store/networks/networks.hooks'; -import type { - BestinslotInscriptionByIdResponse, - BestinslotInscriptionsByTxIdResponse, -} from '../bitcoin-client'; +import type { BitcoinClient } from '../bitcoin-client'; import { getNumberOfInscriptionOnUtxoUsingOrdinalsCom } from '../ordinals/inscriptions.query'; class PreventTransactionError extends Error { @@ -45,32 +42,35 @@ export function filterOutIntentionalUtxoSpend({ interface CheckInscribedUtxosByBestinslotArgs { inputs: btc.TransactionInput[]; txids: string[]; - getInscriptionsByTransactionId(id: string): Promise; - getInscriptionById(id: string): Promise; + client: BitcoinClient; } async function checkInscribedUtxosByBestinslot({ inputs, txids, - getInscriptionsByTransactionId, - getInscriptionById, + client, }: CheckInscribedUtxosByBestinslotArgs): Promise { /** * @description Get the list of inscriptions moving on a transaction * @see https://docs.bestinslot.xyz/reference/api-reference/ordinals-and-brc-20-and-bitmap-v3-api-mainnet+testnet/inscriptions */ - const inscriptionIdsList = await Promise.all(txids.map(id => getInscriptionsByTransactionId(id))); + const inscriptionIdsList = await Promise.all( + txids.map(id => client.BestinslotApi.getInscriptionsByTransactionId(id)) + ); const inscriptionIds = inscriptionIdsList.flatMap(inscription => inscription.data.map(data => data.inscription_id) ); - const inscriptionsList = await Promise.all(inscriptionIds.map(id => getInscriptionById(id))); + const inscriptionsList = await Promise.all( + inscriptionIds.map(id => client.BestinslotApi.getInscriptionById(id)) + ); const hasInscribedUtxos = inscriptionsList.some(resp => { return inputs.some(input => { if (!input.txid) throw new Error('Transaction ID is missing in the input'); const idWithIndex = `${bytesToHex(input.txid)}:${input.index}`; + console.log({ resp }); return resp.data.satpoint.includes(idWithIndex); }); }); @@ -147,8 +147,7 @@ export function useCheckInscribedUtxos(blockTxAction?: () => void) { const hasInscribedUtxo = await checkInscribedUtxosByBestinslot({ inputs, txids, - getInscriptionsByTransactionId: client.BestinslotApi.getInscriptionsByTransactionId, - getInscriptionById: client.BestinslotApi.getInscriptionById, + client, }); if (hasInscribedUtxo) { diff --git a/src/shared/constants.ts b/src/shared/constants.ts index a6819aca2b7..9b5b770fcb8 100644 --- a/src/shared/constants.ts +++ b/src/shared/constants.ts @@ -76,7 +76,7 @@ export interface NetworkConfiguration { } export const BESTINSLOT_API_BASE_URL_MAINNET = 'https://leatherapi.bestinslot.xyz/v3'; -export const BESTINSLOT_API_BASE_URL_TESTNET = 'https://testnet.api.bestinslot.xyz/v3'; +export const BESTINSLOT_API_BASE_URL_TESTNET = 'https://leatherapi_testnet.bestinslot.xyz/v3'; export const HIRO_API_BASE_URL_MAINNET = 'https://api.hiro.so'; export const HIRO_API_BASE_URL_TESTNET = 'https://api.testnet.hiro.so'; diff --git a/tests/specs/send/send-btc.spec.ts b/tests/specs/send/send-btc.spec.ts index 4ef10d57b09..07189ad99a2 100644 --- a/tests/specs/send/send-btc.spec.ts +++ b/tests/specs/send/send-btc.spec.ts @@ -8,6 +8,8 @@ import { BtcFeeType } from '@shared/models/fees/bitcoin-fees.model'; import { test } from '../../fixtures/fixtures'; +test.describe.configure({ mode: 'serial' }); + test.describe('send btc', () => { test.beforeEach(async ({ extensionId, globalPage, homePage, onboardingPage, sendPage }) => { await globalPage.setupAndUseApiCalls(extensionId); @@ -100,9 +102,65 @@ test.describe('send btc', () => { await sendPage.clickInfoCardButton(); - const isErrorPageVisible = await sendPage.broadcastErrorTitle.isVisible(); + await test.expect(sendPage.broadcastErrorTitle).toBeVisible(); + }); + + test('that fallbacks to other api provider if main fails', async ({ sendPage }) => { + let output = ''; + let id = ''; + let index = ''; + + await sendPage.page.route('**/ordinals-explorer.generative.xyz/**', async route => { + return route.fulfill({ + status: 500, + contentType: 'text/html', + body: mockOrdinalsComApiHtmlResponse, + }); + }); + + sendPage.page.on('request', async request => { + if (request.url().includes('ordinals-explorer.generative.xyz')) { + const url = request.url(); + output = url.split('/').pop() || ''; + id = output.split(':')[0]; + index = output.split(':')[1]; + } + }); + + await sendPage.page.route( + '**/leatherapi.bestinslot.xyz/v3/inscription/in_transaction**', + async route => { + return route.fulfill({ + status: 200, + contentType: 'application/json', + body: JSON.stringify({ + data: [{ txid: id, output, index, satpoint: output }], + }), + }); + } + ); + + await sendPage.page.route( + '**/leatherapi.bestinslot.xyz/v3/inscription/single_info_id**', + async route => { + return route.fulfill({ + status: 200, + contentType: 'application/json', + body: JSON.stringify({ + data: { txid: id, output, index, satpoint: output }, + }), + }); + } + ); + await sendPage.amountInput.fill('0.00006'); + await sendPage.recipientInput.fill(TEST_TESTNET_ACCOUNT_2_BTC_ADDRESS); + + await sendPage.previewSendTxButton.click(); + await sendPage.feesListItem.filter({ hasText: BtcFeeType.High }).click(); + + await sendPage.clickInfoCardButton(); - test.expect(isErrorPageVisible).toBeTruthy(); + await test.expect(sendPage.broadcastErrorTitle).toBeVisible(); }); }); });