-
Notifications
You must be signed in to change notification settings - Fork 146
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: check utxo ids for inscriptions, ref #4920
- Loading branch information
1 parent
c4fb072
commit d566426
Showing
8 changed files
with
224 additions
and
5 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,127 @@ | ||
import { useCallback, useState } from 'react'; | ||
|
||
import * as btc from '@scure/btc-signer'; | ||
import { bytesToHex } from '@stacks/common'; | ||
|
||
import { isUndefined } from '@shared/utils'; | ||
|
||
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 { getNumberOfInscriptionOnUtxoUsingOrdinalsCom } from '../ordinals/inscriptions.query'; | ||
|
||
class PreventTransactionError extends Error { | ||
constructor(message: string) { | ||
super(message); | ||
this.name = 'PreventTransactionError'; | ||
} | ||
} | ||
|
||
interface FilterOutIntentionalInscriptionsSpendArgs { | ||
inputs: btc.TransactionInput[]; | ||
intentionalSpendUtxoIds: string[]; | ||
} | ||
export function filterOutIntentionalUtxoSpend({ | ||
inputs, | ||
intentionalSpendUtxoIds, | ||
}: FilterOutIntentionalInscriptionsSpendArgs): btc.TransactionInput[] { | ||
return inputs.filter(input => { | ||
if (!input.txid) throw new Error('Transaction ID is missing in the input'); | ||
const inputTxid = bytesToHex(input.txid); | ||
|
||
return intentionalSpendUtxoIds.every(id => { | ||
return id !== inputTxid; | ||
}); | ||
}); | ||
} | ||
|
||
export function useCheckInscribedUtxos(blockTxAction?: () => void) { | ||
const client = useBitcoinClient(); | ||
const analytics = useAnalytics(); | ||
const [isLoading, setIsLoading] = useState(false); | ||
const { isTestnet } = useCurrentNetworkState(); | ||
|
||
const preventTransaction = useCallback(() => { | ||
if (blockTxAction) return blockTxAction(); | ||
throw new PreventTransactionError( | ||
'Transaction is prevented due to inscribed utxos in the transaction. Please contact support for more information.' | ||
); | ||
}, [blockTxAction]); | ||
|
||
const checkIfUtxosListIncludesInscribed = useCallback( | ||
async (inputs: btc.TransactionInput[]) => { | ||
setIsLoading(true); | ||
const txids = inputs.map(input => { | ||
if (!input.txid) throw new Error('Transaction ID is missing in the input'); | ||
return bytesToHex(input.txid); | ||
}); | ||
|
||
try { | ||
// no need to check for inscriptions on testnet | ||
if (isTestnet) { | ||
return false; | ||
} | ||
|
||
if (txids.length === 0) { | ||
throw new Error('Utxos list cannot be empty'); | ||
} | ||
|
||
const responses = await Promise.all( | ||
txids.map(id => client.bestinslotInscriptionsApi.getInscriptionsByTransactionId(id)) | ||
); | ||
|
||
const hasInscribedUtxo = responses.some(resp => { | ||
return resp.data.length > 0; | ||
}); | ||
|
||
if (hasInscribedUtxo) { | ||
void analytics.track('utxos_includes_inscribed_one', { | ||
txids, | ||
}); | ||
preventTransaction(); | ||
return true; | ||
} | ||
|
||
return false; | ||
} catch (e) { | ||
if (e instanceof PreventTransactionError) { | ||
throw e; | ||
} | ||
|
||
void analytics.track('error_checking_utxos_from_bestinslot', { | ||
txids, | ||
}); | ||
|
||
const ordinalsComResponses = await Promise.all( | ||
txids.map(async (id, index) => { | ||
const inscriptionIndex = inputs[index].index; | ||
if (isUndefined(inscriptionIndex)) { | ||
throw new Error('Inscription index is missing in the input'); | ||
} | ||
const num = await getNumberOfInscriptionOnUtxoUsingOrdinalsCom(id, inscriptionIndex); | ||
return num > 0; | ||
}) | ||
); | ||
|
||
const hasInscribedUtxo = ordinalsComResponses.some(resp => resp); | ||
|
||
// if there are inscribed utxos in the transaction, and no error => prevent the transaction | ||
if (hasInscribedUtxo) { | ||
preventTransaction(); | ||
return true; | ||
} | ||
|
||
return true; | ||
} finally { | ||
setIsLoading(false); | ||
} | ||
}, | ||
[analytics, client.bestinslotInscriptionsApi, isTestnet, preventTransaction] | ||
); | ||
|
||
return { | ||
checkIfUtxosListIncludesInscribed, | ||
isLoading, | ||
}; | ||
} |