PHP WebShell
Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/coins/utxo
Просмотр файла: transaction.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
/**
* @prettier
*/
require("mocha");
const _ = require("lodash");
const assert = require("assert");
const utxolib = require("@bitgo/utxo-lib");
const nock = require("nock");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const abstract_utxo_1 = require("@bitgo/abstract-utxo");
const util_1 = require("./util");
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_test_1 = require("@bitgo/sdk-test");
const src_1 = require("../../../../../src");
function getScriptTypes2Of3() {
return [...utxo_lib_1.bitgo.outputScripts.scriptTypes2Of3, 'taprootKeyPathSpend'];
}
describe(`UTXO coin signTransaction`, async function () {
const bgUrl = sdk_core_1.common.Environments[sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' }).getEnv()].uri;
const coin = (0, util_1.getUtxoCoin)('btc');
const wallet = (0, util_1.getUtxoWallet)(coin, { id: '5b34252f1bf349930e34020a00000000', coin: coin.getChain() });
const rootWalletKeys = (0, util_1.getDefaultWalletKeys)();
const userPrv = rootWalletKeys.user.toBase58();
const pubs = util_1.keychainsBase58.map((v) => v.pub);
function validatePsbt(txHex, targetSigCount, targetNonceCount) {
const psbt = utxolib.bitgo.createPsbtFromHex(txHex, coin.network);
psbt.data.inputs.forEach((input, index) => {
const parsed = utxolib.bitgo.parsePsbtInput(input);
if (parsed.scriptType === 'taprootKeyPathSpend') {
assert.ok(targetNonceCount);
const nonce = psbt.getProprietaryKeyVals(index, {
identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,
subtype: utxolib.bitgo.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,
});
assert.strictEqual(nonce.length, targetNonceCount);
}
const expectedSigCount = parsed.scriptType === 'p2shP2pk' || targetSigCount === 0 ? undefined : 1;
assert.strictEqual(parsed.signatures?.length, expectedSigCount);
});
}
function validateTx(txHex, unspents, targetSigCount) {
const tx = utxolib.bitgo.createTransactionFromHex(txHex, coin.network);
unspents.forEach((u, i) => {
const sigCount = utxolib.bitgo.getStrictSignatureCount(tx.ins[i]);
const expectedSigCount = utxolib.bitgo.isWalletUnspent(u) && !!targetSigCount ? 1 : 0;
assert.strictEqual(sigCount, expectedSigCount);
});
}
async function signTransaction(tx, useSigningSteps, unspents) {
const isPsbt = tx instanceof utxolib.bitgo.UtxoPsbt;
const isTxWithTaprootKeyPathSpend = isPsbt && utxolib.bitgo.isTransactionWithKeyPathSpendInput(tx);
const txHex = tx.toHex();
function nockSignPsbt(psbtHex) {
const psbt = utxolib.bitgo.createPsbtFromHex(psbtHex, coin.network);
return nock(bgUrl)
.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/signpsbt`, (body) => body.psbt)
.reply(200, { psbt: psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo).toHex() });
}
if (!useSigningSteps) {
let scope;
if (tx instanceof utxolib.bitgo.UtxoPsbt && isTxWithTaprootKeyPathSpend) {
scope = nockSignPsbt(tx.clone().setAllInputsMusig2NonceHD(rootWalletKeys.bitgo).toHex());
}
const psbt = await coin.signTransaction({
txPrebuild: {
txHex,
txInfo: isPsbt ? undefined : { unspents },
walletId: isTxWithTaprootKeyPathSpend ? wallet.id() : undefined,
},
prv: userPrv,
pubs: isPsbt ? undefined : pubs,
});
assert.ok('txHex' in psbt);
if (isPsbt) {
validatePsbt(psbt.txHex, 1, 2);
}
else {
assert(unspents);
validateTx(psbt.txHex, unspents, 1);
}
if (scope) {
assert.strictEqual(scope.isDone(), true);
}
return;
}
const signerNoncePsbt = await coin.signTransaction({
txPrebuild: { txHex },
prv: userPrv,
signingStep: 'signerNonce',
});
assert.ok('txHex' in signerNoncePsbt);
if (isPsbt) {
validatePsbt(signerNoncePsbt.txHex, 0, isTxWithTaprootKeyPathSpend ? 1 : undefined);
}
else {
assert(unspents);
validateTx(signerNoncePsbt.txHex, unspents, 0);
}
let scope;
if (isTxWithTaprootKeyPathSpend) {
scope = nockSignPsbt(signerNoncePsbt.txHex);
}
const cosignerNoncePsbt = await coin.signTransaction({
txPrebuild: { ...signerNoncePsbt, walletId: wallet.id() },
signingStep: 'cosignerNonce',
});
assert.ok('txHex' in cosignerNoncePsbt);
if (isPsbt) {
validatePsbt(cosignerNoncePsbt.txHex, 0, isTxWithTaprootKeyPathSpend ? 2 : undefined);
}
else {
assert(unspents);
validateTx(cosignerNoncePsbt.txHex, unspents, 0);
}
if (scope) {
assert.strictEqual(scope.isDone(), true);
}
const signerSigPsbt = await coin.signTransaction({
txPrebuild: { ...cosignerNoncePsbt, txInfo: isPsbt ? undefined : { unspents } },
prv: userPrv,
pubs: isPsbt ? undefined : pubs,
signingStep: 'signerSignature',
});
assert.ok('txHex' in signerSigPsbt);
if (isPsbt) {
validatePsbt(signerSigPsbt.txHex, 1, isTxWithTaprootKeyPathSpend ? 2 : undefined);
}
else {
assert(unspents);
validateTx(signerSigPsbt.txHex, unspents, 1);
}
}
it('success when called like customSigningFunction flow - PSBT with taprootKeyPathSpend inputs', async function () {
const inputs = utxo_lib_1.testutil.inputScriptTypes.map((scriptType) => ({
scriptType,
value: BigInt(1000),
}));
const unspentSum = inputs.reduce((prev, curr) => prev + curr.value, BigInt(0));
const outputs = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];
const psbt = utxo_lib_1.testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');
for (const v of [false, true]) {
await signTransaction(psbt, v);
}
});
it('success when called like customSigningFunction flow - PSBT without taprootKeyPathSpend inputs', async function () {
const inputs = utxo_lib_1.testutil.inputScriptTypes
.filter((v) => v !== 'taprootKeyPathSpend')
.map((scriptType) => ({
scriptType,
value: BigInt(1000),
}));
const unspentSum = inputs.reduce((prev, cur) => prev + cur.value, BigInt(0));
const outputs = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];
const psbt = utxo_lib_1.testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');
for (const v of [false, true]) {
await signTransaction(psbt, v);
}
});
it('success when called like customSigningFunction flow - Network Tx', async function () {
const inputs = utxo_lib_1.testutil.txnInputScriptTypes
.filter((v) => v !== 'p2shP2pk')
.map((scriptType) => ({
scriptType,
value: BigInt(1000),
}));
const unspentSum = inputs.reduce((prev, curr) => prev + curr.value, BigInt(0));
const outputs = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];
const txBuilder = utxo_lib_1.testutil.constructTxnBuilder(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');
const unspents = inputs.map((v, i) => utxo_lib_1.testutil.toTxnUnspent(v, i, coin.network, rootWalletKeys));
for (const v of [false, true]) {
await signTransaction(txBuilder.buildIncomplete(), v, unspents);
}
});
it('fails when called like customSigningFunction flow - PSBT cache miss', async function () {
const inputs = [{ scriptType: 'taprootKeyPathSpend', value: BigInt(1000) }];
const unspentSum = inputs.reduce((prev, curr) => prev + curr.value, BigInt(0));
const outputs = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];
const psbt = utxo_lib_1.testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');
await assert.rejects(async () => {
await coin.signTransaction({
txPrebuild: { txHex: psbt.toHex() },
prv: userPrv,
signingStep: 'signerSignature',
});
}, {
message: `Psbt is missing from txCache (cache size 0).
This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`,
});
});
it('fails when unsupported locking script is used', async function () {
const inputs = [
{ scriptType: 'p2wsh', value: BigInt(1000) },
{ scriptType: 'p2trMusig2', value: BigInt(1000) },
];
const unspentSum = inputs.reduce((prev, curr) => prev + curr.value, BigInt(0));
const outputs = [{ scriptType: 'p2sh', value: unspentSum - BigInt(500) }];
const psbt = utxo_lib_1.testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');
// override the 1st PSBT input with unsupported 2 of 2 multi-sig locking script.
const unspent = utxo_lib_1.testutil.toUnspent(inputs[0], 0, coin.network, rootWalletKeys);
if (!utxolib.bitgo.isWalletUnspent(unspent)) {
throw new Error('invalid unspent');
}
const { publicKeys } = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);
const script2Of2 = utxolib.payments.p2ms({ m: 2, pubkeys: [publicKeys[0], publicKeys[1]] });
psbt.data.inputs[0].witnessScript = script2Of2.output;
await assert.rejects(async () => {
await coin.signTransaction({
txPrebuild: { txHex: psbt.toHex() },
prv: userPrv,
});
}, {
message: `length mismatch`,
});
});
});
function run(coin, inputScripts, txFormat, amountType = 'number') {
describe(`Transaction Stages ${coin.getChain()} (${amountType}) scripts=${inputScripts.join(',')} txFormat=${txFormat}`, function () {
const bgUrl = sdk_core_1.common.Environments[sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' }).getEnv()].uri;
const isTransactionWithKeyPathSpend = inputScripts.some((s) => s === 'taprootKeyPathSpend');
const isTransactionWithReplayProtection = inputScripts.some((s) => s === 'p2shP2pk');
const isTransactionWithP2tr = inputScripts.some((s) => s === 'p2tr');
const isTransactionWithP2trMusig2 = inputScripts.some((s) => s === 'p2trMusig2');
const value = (amountType === 'bigint' ? BigInt('10999999800000001') : 1e8);
const wallet = (0, util_1.getUtxoWallet)(coin, { id: '5b34252f1bf349930e34020a00000000', coin: coin.getChain() });
const walletKeys = (0, util_1.getDefaultWalletKeys)();
const fullSign = !(isTransactionWithReplayProtection || isTransactionWithKeyPathSpend);
function getUnspentsForPsbt() {
return inputScripts.map((t, index) => {
return utxo_lib_1.testutil.toUnspent({ scriptType: t, value: t === 'p2shP2pk' ? BigInt(1000) : BigInt(value) }, index, coin.network, walletKeys);
});
}
function toTxnInputScriptType(type) {
return type === 'p2shP2pk' ? 'replayProtection' : type === 'taprootKeyPathSpend' ? 'p2trMusig2' : type;
}
function getUnspents() {
return inputScripts.map((type, i) => (0, util_1.mockUnspent)(coin.network, walletKeys, toTxnInputScriptType(type), i, value));
}
function getOutputAddress(rootWalletKeys) {
return coin.generateAddress({
keychains: rootWalletKeys.triple.map((k) => ({ pub: k.neutered().toBase58() })),
}).address;
}
function getSignParams(prebuildHex, signer, cosigner) {
const txInfo = {
unspents: txFormat === 'psbt' ? undefined : getUnspents(),
};
return {
txPrebuild: {
walletId: isTransactionWithKeyPathSpend ? wallet.id() : undefined,
txHex: prebuildHex,
txInfo,
},
prv: signer.toBase58(),
pubs: walletKeys.triple.map((k) => k.neutered().toBase58()),
cosignerPub: cosigner.neutered().toBase58(),
};
}
async function createHalfSignedTransaction(prebuild, signer, cosigner) {
let scope;
if (prebuild instanceof utxolib.bitgo.UtxoPsbt && isTransactionWithKeyPathSpend) {
const psbt = prebuild.clone().setAllInputsMusig2NonceHD(cosigner);
scope = nock(bgUrl)
.post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/signpsbt`, (body) => body.psbt)
.reply(200, { psbt: psbt.toHex() });
}
// half-sign with the user key
const result = (await wallet.signTransaction(getSignParams(prebuild.toBuffer().toString('hex'), signer, cosigner)));
if (scope) {
assert.strictEqual(scope.isDone(), true);
}
return result;
}
async function createFullSignedTransaction(halfSigned, signer, cosigner) {
return (await wallet.signTransaction({
...getSignParams(halfSigned.txHex, signer, cosigner),
isLastSignature: true,
}));
}
function createPrebuildPsbt() {
const inputs = inputScripts.map((t) => ({
scriptType: t,
value: t === 'p2shP2pk' ? BigInt(1000) : BigInt(value),
}));
const unspentSum = inputs.reduce((prev, curr) => prev + curr.value, BigInt(0));
const outputs = [
{ address: getOutputAddress((0, util_1.getWalletKeys)('test')), value: unspentSum - BigInt(1000) },
];
const psbt = utxo_lib_1.testutil.constructPsbt(inputs, outputs, coin.network, walletKeys, 'unsigned');
utxolib.bitgo.addXpubsToPsbt(psbt, walletKeys);
return psbt;
}
async function getTransactionStages() {
const prebuild = txFormat === 'psbt'
? createPrebuildPsbt()
: (0, util_1.createPrebuildTransaction)(coin.network, getUnspents(), getOutputAddress(walletKeys));
const halfSignedUserBitGo = await createHalfSignedTransaction(prebuild, walletKeys.user, walletKeys.bitgo);
const fullSignedUserBitGo = fullSign && !isTransactionWithP2trMusig2
? await createFullSignedTransaction(halfSignedUserBitGo, walletKeys.bitgo, walletKeys.user)
: undefined;
const halfSignedUserBackup = !isTransactionWithKeyPathSpend && !(txFormat === 'psbt' && isTransactionWithP2tr)
? await createHalfSignedTransaction(prebuild, walletKeys.user, walletKeys.backup)
: undefined;
const fullSignedUserBackup = fullSign && halfSignedUserBackup
? await createFullSignedTransaction(halfSignedUserBackup, walletKeys.backup, walletKeys.user)
: undefined;
return {
prebuild,
halfSignedUserBackup,
halfSignedUserBitGo,
fullSignedUserBackup,
fullSignedUserBitGo,
};
}
let transactionStages;
before('prepare', async function () {
transactionStages = await getTransactionStages();
});
afterEach(nock.cleanAll);
it('match fixtures', async function () {
if (txFormat === 'psbt') {
// TODO (maybe) - once full PSBT support is added to abstract-utxo module, custom JSON representation of PSBT can be created and tested here.
// signatures of taprootKeyPathSpends are random since random nature of MuSig2 nonce, so psbt hex comparison also wont work.
return this.skip();
}
function toTransactionStagesObj(stages) {
return _.mapValues(stages, (v) => v === undefined || v instanceof utxolib.bitgo.UtxoPsbt
? undefined
: v instanceof utxolib.bitgo.UtxoTransaction
? (0, util_1.transactionToObj)(v)
: (0, util_1.transactionHexToObj)(v.txHex, coin.network, amountType));
}
(0, util_1.shouldEqualJSON)(toTransactionStagesObj(transactionStages), await (0, util_1.getFixture)(coin, `transactions-${inputScripts.map((t) => toTxnInputScriptType(t)).join('-')}`, toTransactionStagesObj(transactionStages)));
});
function testPsbtValidSignatures(tx, signedBy) {
const psbt = utxolib.bitgo.createPsbtFromHex(tx.txHex, coin.network);
const unspents = getUnspentsForPsbt();
psbt.data.inputs.forEach((input, index) => {
const unspent = unspents[index];
if (!utxolib.bitgo.isWalletUnspent(unspent)) {
assert.ok(utxolib.bitgo.getPsbtInputScriptType(input), 'p2shP2pk');
return;
}
const pubkeys = walletKeys.deriveForChainAndIndex(unspent.chain, unspent.index).publicKeys;
pubkeys.forEach((pk, pkIndex) => {
psbt.validateSignaturesOfInputCommon(index, pk).should.eql(signedBy.includes(walletKeys.triple[pkIndex]));
});
});
}
function testValidSignatures(tx, signedBy, sign) {
if (txFormat === 'psbt' && sign === 'halfsigned') {
testPsbtValidSignatures(tx, signedBy);
return;
}
const unspents = txFormat === 'psbt'
? getUnspentsForPsbt().map((u) => ({ ...u, value: utxo_lib_1.bitgo.toTNumber(u.value, amountType) }))
: getUnspents();
const prevOutputs = unspents.map((u) => ({
script: utxolib.address.toOutputScript(u.address, coin.network),
value: u.value,
}));
const transaction = utxolib.bitgo.createTransactionFromBuffer(Buffer.from(tx.txHex, 'hex'), coin.network, { amountType });
transaction.ins.forEach((input, index) => {
if (inputScripts[index] === 'p2shP2pk') {
assert(coin.isBitGoTaintedUnspent(unspents[index]));
return;
}
const unspent = unspents[index];
const pubkeys = walletKeys.deriveForChainAndIndex(unspent.chain, unspent.index).publicKeys;
pubkeys.forEach((pk, pkIndex) => {
utxolib.bitgo
.verifySignature(transaction, index, prevOutputs[index].value, {
publicKey: pk,
}, prevOutputs)
.should.eql(signedBy.includes(walletKeys.triple[pkIndex]));
});
});
}
async function testExplainTx(stageName, txHex, unspents, pubs) {
const explanation = await coin.explainTransaction({
txHex,
txInfo: {
unspents,
},
pubs,
});
explanation.should.have.properties('displayOrder', 'id', 'outputs', 'changeOutputs', 'changeAmount', 'outputAmount', 'inputSignatures', 'signatures');
const expectedSignatureCount = stageName === 'prebuild' || pubs === undefined
? 0
: stageName.startsWith('halfSigned')
? 1
: stageName.startsWith('fullSigned')
? 2
: undefined;
explanation.inputSignatures.should.eql(
// FIXME(BG-35154): implement signature verification for replay protection inputs
inputScripts.map((type) => (type === 'p2shP2pk' ? 0 : expectedSignatureCount)));
explanation.signatures.should.eql(expectedSignatureCount);
explanation.changeAmount.should.eql('0'); // no change addresses given
let expectedOutputAmount = BigInt((txFormat === 'psbt' ? getUnspentsForPsbt() : getUnspents()).length) * BigInt(value);
inputScripts.forEach((type) => {
if (type === 'p2shP2pk') {
// replayProtection unspents have value 1000
expectedOutputAmount -= BigInt(value);
expectedOutputAmount += BigInt(1000);
}
});
expectedOutputAmount -= BigInt(1000); // fee of 1000
explanation.outputAmount.should.eql(expectedOutputAmount.toString());
}
it('have valid signature for half-signed transaction', function () {
if (transactionStages.halfSignedUserBackup) {
testValidSignatures(transactionStages.halfSignedUserBackup, [walletKeys.user], 'halfsigned');
}
testValidSignatures(transactionStages.halfSignedUserBitGo, [walletKeys.user], 'halfsigned');
});
it('have valid signatures for full-signed transaction', function () {
if (!fullSign) {
return this.skip();
}
if (transactionStages.fullSignedUserBackup) {
testValidSignatures(transactionStages.fullSignedUserBackup, [walletKeys.user, walletKeys.backup], 'fullsigned');
}
if (transactionStages.fullSignedUserBitGo) {
testValidSignatures(transactionStages.fullSignedUserBitGo, [walletKeys.user, walletKeys.bitgo], 'fullsigned');
}
});
it('have correct results for explainTransaction', async function () {
for (const [stageName, stageTx] of Object.entries(transactionStages)) {
if (!stageTx) {
continue;
}
const txHex = stageTx instanceof utxolib.bitgo.UtxoPsbt || stageTx instanceof utxolib.bitgo.UtxoTransaction
? stageTx.toBuffer().toString('hex')
: stageTx.txHex;
const pubs = walletKeys.triple.map((k) => k.neutered().toBase58());
const unspents = txFormat === 'psbt'
? getUnspentsForPsbt().map((u) => ({ ...u, value: utxo_lib_1.bitgo.toTNumber(u.value, amountType) }))
: getUnspents();
await testExplainTx(stageName, txHex, unspents, pubs);
await testExplainTx(stageName, txHex, unspents);
}
});
});
}
function runWithAmountType(coin, inputScripts, txFormat) {
const amountType = coin.amountType;
if (amountType === 'bigint') {
run(coin, inputScripts, txFormat, amountType);
}
else {
run(coin, inputScripts, txFormat, amountType);
}
}
util_1.utxoCoins.forEach((coin) => getScriptTypes2Of3().forEach((type) => {
['legacy', 'psbt'].forEach((txFormat) => {
if ((type === 'taprootKeyPathSpend' || type === 'p2trMusig2') && txFormat !== 'psbt') {
return;
}
if (coin.supportsAddressType(type === 'taprootKeyPathSpend' ? 'p2trMusig2' : type)) {
runWithAmountType(coin, [type, type], txFormat);
if ((0, abstract_utxo_1.getReplayProtectionAddresses)(coin.network).length) {
runWithAmountType(coin, ['p2shP2pk', type], txFormat);
}
}
});
}));
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../../../../test/v2/unit/coins/utxo/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport 'mocha';\nimport * as _ from 'lodash';\nimport * as assert from 'assert';\nimport * as utxolib from '@bitgo/utxo-lib';\nimport * as nock from 'nock';\nimport { BIP32Interface, bitgo, testutil } from '@bitgo/utxo-lib';\n\nimport { AbstractUtxoCoin, getReplayProtectionAddresses } from '@bitgo/abstract-utxo';\n\nimport {\n  utxoCoins,\n  shouldEqualJSON,\n  getFixture,\n  getUtxoWallet,\n  mockUnspent,\n  InputScriptType,\n  TransactionObj,\n  transactionToObj,\n  transactionHexToObj,\n  createPrebuildTransaction,\n  getDefaultWalletKeys,\n  getUtxoCoin,\n  keychainsBase58,\n  getWalletKeys,\n} from './util';\n\nimport {\n  common,\n  FullySignedTransaction,\n  HalfSignedUtxoTransaction,\n  Triple,\n  WalletSignTransactionOptions,\n} from '@bitgo/sdk-core';\nimport { TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../../../../src';\n\ntype Unspent<TNumber extends number | bigint = number> = bitgo.Unspent<TNumber>;\ntype WalletUnspent<TNumber extends number | bigint = number> = bitgo.WalletUnspent<TNumber>;\n\nfunction getScriptTypes2Of3() {\n  return [...bitgo.outputScripts.scriptTypes2Of3, 'taprootKeyPathSpend'] as const;\n}\n\ndescribe(`UTXO coin signTransaction`, async function () {\n  const bgUrl = common.Environments[TestBitGo.decorate(BitGo, { env: 'mock' }).getEnv()].uri;\n\n  const coin = getUtxoCoin('btc');\n  const wallet = getUtxoWallet(coin, { id: '5b34252f1bf349930e34020a00000000', coin: coin.getChain() });\n  const rootWalletKeys = getDefaultWalletKeys();\n  const userPrv = rootWalletKeys.user.toBase58();\n  const pubs = keychainsBase58.map((v) => v.pub) as Triple<string>;\n\n  function validatePsbt(txHex: string, targetSigCount: 0 | 1, targetNonceCount?: 1 | 2) {\n    const psbt = utxolib.bitgo.createPsbtFromHex(txHex, coin.network);\n    psbt.data.inputs.forEach((input, index) => {\n      const parsed = utxolib.bitgo.parsePsbtInput(input);\n      if (parsed.scriptType === 'taprootKeyPathSpend') {\n        assert.ok(targetNonceCount);\n        const nonce = psbt.getProprietaryKeyVals(index, {\n          identifier: utxolib.bitgo.PSBT_PROPRIETARY_IDENTIFIER,\n          subtype: utxolib.bitgo.ProprietaryKeySubtype.MUSIG2_PUB_NONCE,\n        });\n        assert.strictEqual(nonce.length, targetNonceCount);\n      }\n      const expectedSigCount = parsed.scriptType === 'p2shP2pk' || targetSigCount === 0 ? undefined : 1;\n      assert.strictEqual(parsed.signatures?.length, expectedSigCount);\n    });\n  }\n\n  function validateTx(txHex: string, unspents: Unspent<bigint>[], targetSigCount: 0 | 1) {\n    const tx = utxolib.bitgo.createTransactionFromHex(txHex, coin.network);\n    unspents.forEach((u, i) => {\n      const sigCount = utxolib.bitgo.getStrictSignatureCount(tx.ins[i]);\n      const expectedSigCount = utxolib.bitgo.isWalletUnspent(u) && !!targetSigCount ? 1 : 0;\n      assert.strictEqual(sigCount, expectedSigCount);\n    });\n  }\n\n  async function signTransaction(\n    tx: utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>,\n    useSigningSteps: boolean,\n    unspents?: Unspent<bigint>[]\n  ) {\n    const isPsbt = tx instanceof utxolib.bitgo.UtxoPsbt;\n    const isTxWithTaprootKeyPathSpend = isPsbt && utxolib.bitgo.isTransactionWithKeyPathSpendInput(tx);\n    const txHex = tx.toHex();\n\n    function nockSignPsbt(psbtHex: string): nock.Scope {\n      const psbt = utxolib.bitgo.createPsbtFromHex(psbtHex, coin.network);\n      return nock(bgUrl)\n        .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/signpsbt`, (body) => body.psbt)\n        .reply(200, { psbt: psbt.setAllInputsMusig2NonceHD(rootWalletKeys.bitgo).toHex() });\n    }\n\n    if (!useSigningSteps) {\n      let scope: nock.Scope | undefined;\n      if (tx instanceof utxolib.bitgo.UtxoPsbt && isTxWithTaprootKeyPathSpend) {\n        scope = nockSignPsbt(tx.clone().setAllInputsMusig2NonceHD(rootWalletKeys.bitgo).toHex());\n      }\n      const psbt = await coin.signTransaction({\n        txPrebuild: {\n          txHex,\n          txInfo: isPsbt ? undefined : { unspents },\n          walletId: isTxWithTaprootKeyPathSpend ? wallet.id() : undefined,\n        },\n        prv: userPrv,\n        pubs: isPsbt ? undefined : pubs,\n      });\n      assert.ok('txHex' in psbt);\n      if (isPsbt) {\n        validatePsbt(psbt.txHex, 1, 2);\n      } else {\n        assert(unspents);\n        validateTx(psbt.txHex, unspents, 1);\n      }\n      if (scope) {\n        assert.strictEqual(scope.isDone(), true);\n      }\n      return;\n    }\n\n    const signerNoncePsbt = await coin.signTransaction({\n      txPrebuild: { txHex },\n      prv: userPrv,\n      signingStep: 'signerNonce',\n    });\n    assert.ok('txHex' in signerNoncePsbt);\n    if (isPsbt) {\n      validatePsbt(signerNoncePsbt.txHex, 0, isTxWithTaprootKeyPathSpend ? 1 : undefined);\n    } else {\n      assert(unspents);\n      validateTx(signerNoncePsbt.txHex, unspents, 0);\n    }\n\n    let scope: nock.Scope | undefined;\n    if (isTxWithTaprootKeyPathSpend) {\n      scope = nockSignPsbt(signerNoncePsbt.txHex);\n    }\n\n    const cosignerNoncePsbt = await coin.signTransaction({\n      txPrebuild: { ...signerNoncePsbt, walletId: wallet.id() },\n      signingStep: 'cosignerNonce',\n    });\n    assert.ok('txHex' in cosignerNoncePsbt);\n    if (isPsbt) {\n      validatePsbt(cosignerNoncePsbt.txHex, 0, isTxWithTaprootKeyPathSpend ? 2 : undefined);\n    } else {\n      assert(unspents);\n      validateTx(cosignerNoncePsbt.txHex, unspents, 0);\n    }\n\n    if (scope) {\n      assert.strictEqual(scope.isDone(), true);\n    }\n\n    const signerSigPsbt = await coin.signTransaction({\n      txPrebuild: { ...cosignerNoncePsbt, txInfo: isPsbt ? undefined : { unspents } },\n      prv: userPrv,\n      pubs: isPsbt ? undefined : pubs,\n      signingStep: 'signerSignature',\n    });\n    assert.ok('txHex' in signerSigPsbt);\n    if (isPsbt) {\n      validatePsbt(signerSigPsbt.txHex, 1, isTxWithTaprootKeyPathSpend ? 2 : undefined);\n    } else {\n      assert(unspents);\n      validateTx(signerSigPsbt.txHex, unspents, 1);\n    }\n  }\n\n  it('success when called like customSigningFunction flow - PSBT with taprootKeyPathSpend inputs', async function () {\n    const inputs: testutil.Input[] = testutil.inputScriptTypes.map((scriptType) => ({\n      scriptType,\n      value: BigInt(1000),\n    }));\n    const unspentSum = inputs.reduce((prev: bigint, curr) => prev + curr.value, BigInt(0));\n    const outputs: testutil.Output[] = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];\n    const psbt = testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');\n\n    for (const v of [false, true]) {\n      await signTransaction(psbt, v);\n    }\n  });\n\n  it('success when called like customSigningFunction flow - PSBT without taprootKeyPathSpend inputs', async function () {\n    const inputs: testutil.Input[] = testutil.inputScriptTypes\n      .filter((v) => v !== 'taprootKeyPathSpend')\n      .map((scriptType) => ({\n        scriptType,\n        value: BigInt(1000),\n      }));\n    const unspentSum = inputs.reduce((prev: bigint, cur) => prev + cur.value, BigInt(0));\n    const outputs: testutil.Output[] = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];\n    const psbt = testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');\n\n    for (const v of [false, true]) {\n      await signTransaction(psbt, v);\n    }\n  });\n\n  it('success when called like customSigningFunction flow - Network Tx', async function () {\n    const inputs: testutil.TxnInput<bigint>[] = testutil.txnInputScriptTypes\n      .filter((v) => v !== 'p2shP2pk')\n      .map((scriptType) => ({\n        scriptType,\n        value: BigInt(1000),\n      }));\n    const unspentSum = inputs.reduce((prev: bigint, curr) => prev + curr.value, BigInt(0));\n    const outputs: testutil.TxnOutput<bigint>[] = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];\n    const txBuilder = testutil.constructTxnBuilder(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');\n    const unspents = inputs.map((v, i) => testutil.toTxnUnspent(v, i, coin.network, rootWalletKeys));\n\n    for (const v of [false, true]) {\n      await signTransaction(txBuilder.buildIncomplete(), v, unspents);\n    }\n  });\n\n  it('fails when called like customSigningFunction flow - PSBT cache miss', async function () {\n    const inputs: testutil.Input[] = [{ scriptType: 'taprootKeyPathSpend', value: BigInt(1000) }];\n    const unspentSum = inputs.reduce((prev: bigint, curr) => prev + curr.value, BigInt(0));\n    const outputs: testutil.Output[] = [{ scriptType: 'p2sh', value: unspentSum - BigInt(1000) }];\n    const psbt = testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');\n\n    await assert.rejects(\n      async () => {\n        await coin.signTransaction({\n          txPrebuild: { txHex: psbt.toHex() },\n          prv: userPrv,\n          signingStep: 'signerSignature',\n        });\n      },\n      {\n        message: `Psbt is missing from txCache (cache size 0).\n            This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`,\n      }\n    );\n  });\n\n  it('fails when unsupported locking script is used', async function () {\n    const inputs: testutil.Input[] = [\n      { scriptType: 'p2wsh', value: BigInt(1000) },\n      { scriptType: 'p2trMusig2', value: BigInt(1000) },\n    ];\n    const unspentSum = inputs.reduce((prev: bigint, curr) => prev + curr.value, BigInt(0));\n    const outputs: testutil.Output[] = [{ scriptType: 'p2sh', value: unspentSum - BigInt(500) }];\n    const psbt = testutil.constructPsbt(inputs, outputs, coin.network, rootWalletKeys, 'unsigned');\n\n    // override the 1st PSBT input with unsupported 2 of 2 multi-sig locking script.\n    const unspent = testutil.toUnspent(inputs[0], 0, coin.network, rootWalletKeys);\n    if (!utxolib.bitgo.isWalletUnspent(unspent)) {\n      throw new Error('invalid unspent');\n    }\n    const { publicKeys } = rootWalletKeys.deriveForChainAndIndex(unspent.chain, unspent.index);\n    const script2Of2 = utxolib.payments.p2ms({ m: 2, pubkeys: [publicKeys[0], publicKeys[1]] });\n    psbt.data.inputs[0].witnessScript = script2Of2.output;\n\n    await assert.rejects(\n      async () => {\n        await coin.signTransaction({\n          txPrebuild: { txHex: psbt.toHex() },\n          prv: userPrv,\n        });\n      },\n      {\n        message: `length mismatch`,\n      }\n    );\n  });\n});\n\nfunction run<TNumber extends number | bigint = number>(\n  coin: AbstractUtxoCoin,\n  inputScripts: testutil.InputScriptType[],\n  txFormat: 'legacy' | 'psbt',\n  amountType: 'number' | 'bigint' = 'number'\n) {\n  describe(`Transaction Stages ${coin.getChain()} (${amountType}) scripts=${inputScripts.join(\n    ','\n  )} txFormat=${txFormat}`, function () {\n    const bgUrl = common.Environments[TestBitGo.decorate(BitGo, { env: 'mock' }).getEnv()].uri;\n\n    const isTransactionWithKeyPathSpend = inputScripts.some((s) => s === 'taprootKeyPathSpend');\n    const isTransactionWithReplayProtection = inputScripts.some((s) => s === 'p2shP2pk');\n    const isTransactionWithP2tr = inputScripts.some((s) => s === 'p2tr');\n    const isTransactionWithP2trMusig2 = inputScripts.some((s) => s === 'p2trMusig2');\n\n    const value = (amountType === 'bigint' ? BigInt('10999999800000001') : 1e8) as TNumber;\n    const wallet = getUtxoWallet(coin, { id: '5b34252f1bf349930e34020a00000000', coin: coin.getChain() });\n    const walletKeys = getDefaultWalletKeys();\n\n    const fullSign = !(isTransactionWithReplayProtection || isTransactionWithKeyPathSpend);\n\n    function getUnspentsForPsbt(): Unspent<bigint>[] {\n      return inputScripts.map((t, index) => {\n        return testutil.toUnspent(\n          { scriptType: t, value: t === 'p2shP2pk' ? BigInt(1000) : BigInt(value) },\n          index,\n          coin.network,\n          walletKeys\n        );\n      });\n    }\n\n    function toTxnInputScriptType(type: testutil.InputScriptType): InputScriptType {\n      return type === 'p2shP2pk' ? 'replayProtection' : type === 'taprootKeyPathSpend' ? 'p2trMusig2' : type;\n    }\n\n    function getUnspents(): Unspent<TNumber>[] {\n      return inputScripts.map((type, i) =>\n        mockUnspent<TNumber>(coin.network, walletKeys, toTxnInputScriptType(type), i, value)\n      );\n    }\n\n    function getOutputAddress(rootWalletKeys: utxolib.bitgo.RootWalletKeys): string {\n      return coin.generateAddress({\n        keychains: rootWalletKeys.triple.map((k) => ({ pub: k.neutered().toBase58() })),\n      }).address;\n    }\n\n    function getSignParams(\n      prebuildHex: string,\n      signer: BIP32Interface,\n      cosigner: BIP32Interface\n    ): WalletSignTransactionOptions {\n      const txInfo = {\n        unspents: txFormat === 'psbt' ? undefined : getUnspents(),\n      };\n      return {\n        txPrebuild: {\n          walletId: isTransactionWithKeyPathSpend ? wallet.id() : undefined,\n          txHex: prebuildHex,\n          txInfo,\n        },\n        prv: signer.toBase58(),\n        pubs: walletKeys.triple.map((k) => k.neutered().toBase58()),\n        cosignerPub: cosigner.neutered().toBase58(),\n      } as WalletSignTransactionOptions;\n    }\n\n    async function createHalfSignedTransaction(\n      prebuild: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoPsbt,\n      signer: BIP32Interface,\n      cosigner: BIP32Interface\n    ): Promise<HalfSignedUtxoTransaction> {\n      let scope: nock.Scope | undefined;\n      if (prebuild instanceof utxolib.bitgo.UtxoPsbt && isTransactionWithKeyPathSpend) {\n        const psbt = prebuild.clone().setAllInputsMusig2NonceHD(cosigner);\n        scope = nock(bgUrl)\n          .post(`/api/v2/${wallet.coin()}/wallet/${wallet.id()}/tx/signpsbt`, (body) => body.psbt)\n          .reply(200, { psbt: psbt.toHex() });\n      }\n\n      // half-sign with the user key\n      const result = (await wallet.signTransaction(\n        getSignParams(prebuild.toBuffer().toString('hex'), signer, cosigner)\n      )) as Promise<HalfSignedUtxoTransaction>;\n\n      if (scope) {\n        assert.strictEqual(scope.isDone(), true);\n      }\n\n      return result;\n    }\n\n    async function createFullSignedTransaction(\n      halfSigned: HalfSignedUtxoTransaction,\n      signer: BIP32Interface,\n      cosigner: BIP32Interface\n    ): Promise<FullySignedTransaction> {\n      return (await wallet.signTransaction({\n        ...getSignParams(halfSigned.txHex, signer, cosigner),\n        isLastSignature: true,\n      })) as FullySignedTransaction;\n    }\n\n    type TransactionStages = {\n      prebuild: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoPsbt;\n      halfSignedUserBackup?: HalfSignedUtxoTransaction;\n      halfSignedUserBitGo: HalfSignedUtxoTransaction;\n      fullSignedUserBackup?: FullySignedTransaction;\n      fullSignedUserBitGo?: FullySignedTransaction;\n    };\n\n    type TransactionObjStages = Record<keyof TransactionStages, TransactionObj>;\n\n    function createPrebuildPsbt() {\n      const inputs = inputScripts.map(\n        (t): testutil.Input => ({\n          scriptType: t,\n          value: t === 'p2shP2pk' ? BigInt(1000) : BigInt(value),\n        })\n      );\n      const unspentSum = inputs.reduce((prev: bigint, curr) => prev + curr.value, BigInt(0));\n      const outputs: testutil.Output[] = [\n        { address: getOutputAddress(getWalletKeys('test')), value: unspentSum - BigInt(1000) },\n      ];\n      const psbt = testutil.constructPsbt(inputs, outputs, coin.network, walletKeys, 'unsigned');\n      utxolib.bitgo.addXpubsToPsbt(psbt, walletKeys);\n      return psbt;\n    }\n\n    async function getTransactionStages(): Promise<TransactionStages> {\n      const prebuild =\n        txFormat === 'psbt'\n          ? createPrebuildPsbt()\n          : createPrebuildTransaction<TNumber>(coin.network, getUnspents(), getOutputAddress(walletKeys));\n\n      const halfSignedUserBitGo = await createHalfSignedTransaction(prebuild, walletKeys.user, walletKeys.bitgo);\n      const fullSignedUserBitGo =\n        fullSign && !isTransactionWithP2trMusig2\n          ? await createFullSignedTransaction(halfSignedUserBitGo, walletKeys.bitgo, walletKeys.user)\n          : undefined;\n\n      const halfSignedUserBackup =\n        !isTransactionWithKeyPathSpend && !(txFormat === 'psbt' && isTransactionWithP2tr)\n          ? await createHalfSignedTransaction(prebuild, walletKeys.user, walletKeys.backup)\n          : undefined;\n      const fullSignedUserBackup =\n        fullSign && halfSignedUserBackup\n          ? await createFullSignedTransaction(halfSignedUserBackup, walletKeys.backup, walletKeys.user)\n          : undefined;\n\n      return {\n        prebuild,\n        halfSignedUserBackup,\n        halfSignedUserBitGo,\n        fullSignedUserBackup,\n        fullSignedUserBitGo,\n      };\n    }\n\n    let transactionStages: TransactionStages;\n\n    before('prepare', async function () {\n      transactionStages = await getTransactionStages();\n    });\n\n    afterEach(nock.cleanAll);\n\n    it('match fixtures', async function (this: Mocha.Context) {\n      if (txFormat === 'psbt') {\n        // TODO (maybe) - once full PSBT support is added to abstract-utxo module, custom JSON representation of PSBT can be created and tested here.\n        // signatures of taprootKeyPathSpends are random since random nature of MuSig2 nonce, so psbt hex comparison also wont work.\n\n        return this.skip();\n      }\n\n      function toTransactionStagesObj(stages: TransactionStages): TransactionObjStages {\n        return _.mapValues(stages, (v) =>\n          v === undefined || v instanceof utxolib.bitgo.UtxoPsbt\n            ? undefined\n            : v instanceof utxolib.bitgo.UtxoTransaction\n            ? transactionToObj<TNumber>(v)\n            : transactionHexToObj(v.txHex, coin.network, amountType)\n        ) as TransactionObjStages;\n      }\n\n      shouldEqualJSON(\n        toTransactionStagesObj(transactionStages),\n        await getFixture(\n          coin,\n          `transactions-${inputScripts.map((t) => toTxnInputScriptType(t)).join('-')}`,\n          toTransactionStagesObj(transactionStages)\n        )\n      );\n    });\n\n    function testPsbtValidSignatures(tx: HalfSignedUtxoTransaction, signedBy: BIP32Interface[]) {\n      const psbt = utxolib.bitgo.createPsbtFromHex(tx.txHex, coin.network);\n      const unspents = getUnspentsForPsbt();\n      psbt.data.inputs.forEach((input, index) => {\n        const unspent = unspents[index];\n        if (!utxolib.bitgo.isWalletUnspent(unspent)) {\n          assert.ok(utxolib.bitgo.getPsbtInputScriptType(input), 'p2shP2pk');\n          return;\n        }\n        const pubkeys = walletKeys.deriveForChainAndIndex(unspent.chain, unspent.index).publicKeys;\n        pubkeys.forEach((pk, pkIndex) => {\n          psbt.validateSignaturesOfInputCommon(index, pk).should.eql(signedBy.includes(walletKeys.triple[pkIndex]));\n        });\n      });\n    }\n\n    function testValidSignatures(\n      tx: HalfSignedUtxoTransaction | FullySignedTransaction,\n      signedBy: BIP32Interface[],\n      sign: 'halfsigned' | 'fullsigned'\n    ) {\n      if (txFormat === 'psbt' && sign === 'halfsigned') {\n        testPsbtValidSignatures(tx, signedBy);\n        return;\n      }\n      const unspents =\n        txFormat === 'psbt'\n          ? getUnspentsForPsbt().map((u) => ({ ...u, value: bitgo.toTNumber(u.value, amountType) as TNumber }))\n          : getUnspents();\n      const prevOutputs = unspents.map(\n        (u): utxolib.TxOutput<TNumber> => ({\n          script: utxolib.address.toOutputScript(u.address, coin.network),\n          value: u.value,\n        })\n      );\n\n      const transaction = utxolib.bitgo.createTransactionFromBuffer<TNumber>(\n        Buffer.from(tx.txHex, 'hex'),\n        coin.network,\n        { amountType }\n      );\n      transaction.ins.forEach((input, index) => {\n        if (inputScripts[index] === 'p2shP2pk') {\n          assert(coin.isBitGoTaintedUnspent(unspents[index]));\n          return;\n        }\n\n        const unspent = unspents[index] as WalletUnspent<TNumber>;\n        const pubkeys = walletKeys.deriveForChainAndIndex(unspent.chain, unspent.index).publicKeys;\n\n        pubkeys.forEach((pk, pkIndex) => {\n          utxolib.bitgo\n            .verifySignature<TNumber>(\n              transaction,\n              index,\n              prevOutputs[index].value,\n              {\n                publicKey: pk,\n              },\n              prevOutputs\n            )\n            .should.eql(signedBy.includes(walletKeys.triple[pkIndex]));\n        });\n      });\n    }\n\n    async function testExplainTx(\n      stageName: string,\n      txHex: string,\n      unspents?: utxolib.bitgo.Unspent<TNumber>[],\n      pubs?: Triple<string>\n    ): Promise<void> {\n      const explanation = await coin.explainTransaction<TNumber>({\n        txHex,\n        txInfo: {\n          unspents,\n        },\n        pubs,\n      });\n\n      explanation.should.have.properties(\n        'displayOrder',\n        'id',\n        'outputs',\n        'changeOutputs',\n        'changeAmount',\n        'outputAmount',\n        'inputSignatures',\n        'signatures'\n      );\n\n      const expectedSignatureCount =\n        stageName === 'prebuild' || pubs === undefined\n          ? 0\n          : stageName.startsWith('halfSigned')\n          ? 1\n          : stageName.startsWith('fullSigned')\n          ? 2\n          : undefined;\n\n      explanation.inputSignatures.should.eql(\n        // FIXME(BG-35154): implement signature verification for replay protection inputs\n        inputScripts.map((type) => (type === 'p2shP2pk' ? 0 : expectedSignatureCount))\n      );\n      explanation.signatures.should.eql(expectedSignatureCount);\n      explanation.changeAmount.should.eql('0'); // no change addresses given\n      let expectedOutputAmount =\n        BigInt((txFormat === 'psbt' ? getUnspentsForPsbt() : getUnspents()).length) * BigInt(value);\n      inputScripts.forEach((type) => {\n        if (type === 'p2shP2pk') {\n          // replayProtection unspents have value 1000\n          expectedOutputAmount -= BigInt(value);\n          expectedOutputAmount += BigInt(1000);\n        }\n      });\n      expectedOutputAmount -= BigInt(1000); // fee of 1000\n      explanation.outputAmount.should.eql(expectedOutputAmount.toString());\n    }\n\n    it('have valid signature for half-signed transaction', function () {\n      if (transactionStages.halfSignedUserBackup) {\n        testValidSignatures(transactionStages.halfSignedUserBackup, [walletKeys.user], 'halfsigned');\n      }\n      testValidSignatures(transactionStages.halfSignedUserBitGo, [walletKeys.user], 'halfsigned');\n    });\n\n    it('have valid signatures for full-signed transaction', function () {\n      if (!fullSign) {\n        return this.skip();\n      }\n      if (transactionStages.fullSignedUserBackup) {\n        testValidSignatures(transactionStages.fullSignedUserBackup, [walletKeys.user, walletKeys.backup], 'fullsigned');\n      }\n      if (transactionStages.fullSignedUserBitGo) {\n        testValidSignatures(transactionStages.fullSignedUserBitGo, [walletKeys.user, walletKeys.bitgo], 'fullsigned');\n      }\n    });\n\n    it('have correct results for explainTransaction', async function () {\n      for (const [stageName, stageTx] of Object.entries(transactionStages)) {\n        if (!stageTx) {\n          continue;\n        }\n\n        const txHex =\n          stageTx instanceof utxolib.bitgo.UtxoPsbt || stageTx instanceof utxolib.bitgo.UtxoTransaction\n            ? stageTx.toBuffer().toString('hex')\n            : stageTx.txHex;\n\n        const pubs = walletKeys.triple.map((k) => k.neutered().toBase58()) as Triple<string>;\n        const unspents =\n          txFormat === 'psbt'\n            ? getUnspentsForPsbt().map((u) => ({ ...u, value: bitgo.toTNumber(u.value, amountType) as TNumber }))\n            : getUnspents();\n        await testExplainTx(stageName, txHex, unspents, pubs);\n        await testExplainTx(stageName, txHex, unspents);\n      }\n    });\n  });\n}\n\nfunction runWithAmountType(\n  coin: AbstractUtxoCoin,\n  inputScripts: testutil.InputScriptType[],\n  txFormat: 'legacy' | 'psbt'\n) {\n  const amountType = coin.amountType;\n  if (amountType === 'bigint') {\n    run<bigint>(coin, inputScripts, txFormat, amountType);\n  } else {\n    run(coin, inputScripts, txFormat, amountType);\n  }\n}\n\nutxoCoins.forEach((coin) =>\n  getScriptTypes2Of3().forEach((type) => {\n    (['legacy', 'psbt'] as const).forEach((txFormat) => {\n      if ((type === 'taprootKeyPathSpend' || type === 'p2trMusig2') && txFormat !== 'psbt') {\n        return;\n      }\n      if (coin.supportsAddressType(type === 'taprootKeyPathSpend' ? 'p2trMusig2' : type)) {\n        runWithAmountType(coin, [type, type], txFormat);\n\n        if (getReplayProtectionAddresses(coin.network).length) {\n          runWithAmountType(coin, ['p2shP2pk', type], txFormat);\n        }\n      }\n    });\n  })\n);\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!