PHP WebShell

Текущая директория: /opt/BitGoJS/modules/utxo-lib/dist/test/bitgo/psbt

Просмотр файла: Psbt.js

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const assert = require("assert");
const src_1 = require("../../../src");
const bitgo_1 = require("../../../src/bitgo");
const outputScripts_1 = require("../../../src/bitgo/outputScripts");
const testutil_1 = require("../../../src/testutil");
const transaction_util_1 = require("../../transaction_util");
const psbtUtil_1 = require("./psbtUtil");
const testutil_2 = require("../../../src/testutil");
const Musig2Util_1 = require("./Musig2Util");
const CHANGE_INDEX = 100;
const FEE = BigInt(100);
const network = src_1.networks.bitcoin;
const rootWalletKeys = (0, testutil_1.getDefaultWalletKeys)();
function getScriptTypes2Of3() {
    // FIXME(BG-66941): p2trMusig2 signing does not work in this test suite yet
    //  because the test suite is written with TransactionBuilder
    return bitgo_1.outputScripts.scriptTypes2Of3.filter((scriptType) => scriptType !== 'p2trMusig2');
}
const halfSignedInputs = ['p2sh', 'p2wsh', 'p2shP2wsh'].map((scriptType) => ({
    scriptType,
    value: BigInt(1000),
}));
const halfSignedOutputs = testutil_1.outputScriptTypes.map((scriptType) => ({ scriptType, value: BigInt(500) }));
const psbtInputs = testutil_1.inputScriptTypes.map((scriptType) => ({ scriptType, value: BigInt(1000) }));
const psbtOutputs = testutil_1.outputScriptTypes.map((scriptType) => ({ scriptType, value: BigInt(900) }));
describe('Psbt Misc', function () {
    function getTestPsbt() {
        return src_1.testutil.constructPsbt([{ scriptType: 'p2tr', value: BigInt(1000) }], [{ scriptType: 'p2sh', value: BigInt(900) }], network, rootWalletKeys, 'fullsigned');
    }
    it('fail to finalise p2tr sighash mismatch', function () {
        const psbt = getTestPsbt();
        assert(psbt.validateSignaturesOfAllInputs());
        const tapScriptSig = psbt.data.inputs[0].tapScriptSig;
        assert(tapScriptSig);
        tapScriptSig[0].signature = Buffer.concat([tapScriptSig[0].signature, Buffer.of(src_1.Transaction.SIGHASH_ALL)]);
        assert.throws(() => psbt.finalizeAllInputs(), (e) => e.message === 'signature sighash does not match input sighash type');
    });
    describe('isPsbtLite', function () {
        it('no inputs', function () {
            const psbt = src_1.testutil.constructPsbt([], [], network, rootWalletKeys, 'unsigned');
            assert.strictEqual((0, bitgo_1.isPsbtLite)(psbt), false);
        });
        it('all inputs are segwit', function () {
            const psbt = src_1.testutil.constructPsbt(psbtInputs.filter((s) => s.scriptType !== 'p2sh' && s.scriptType !== 'p2shP2pk'), psbtOutputs, network, rootWalletKeys, 'unsigned');
            assert.strictEqual((0, bitgo_1.isPsbtLite)(psbt), false);
        });
        it('some inputs are non-segwit', function () {
            const psbt = src_1.testutil.constructPsbt(psbtInputs, psbtOutputs, network, rootWalletKeys, 'unsigned');
            assert.strictEqual((0, bitgo_1.isPsbtLite)(psbt), false);
        });
        it('should be true if after clonePsbtWithoutNonWitnessUtxo', function () {
            const psbt = src_1.testutil.constructPsbt(psbtInputs, psbtOutputs, network, rootWalletKeys, 'unsigned');
            const clonedPsbt = (0, bitgo_1.clonePsbtWithoutNonWitnessUtxo)(psbt);
            assert.strictEqual((0, bitgo_1.isPsbtLite)(clonedPsbt), true);
        });
    });
});
describe('extractP2msOnlyHalfSignedTx failure', function () {
    it('invalid signature count', function () {
        const psbt = src_1.testutil.constructPsbt(halfSignedInputs, halfSignedOutputs, network, rootWalletKeys, 'unsigned');
        assert.throws(() => (0, bitgo_1.extractP2msOnlyHalfSignedTx)(psbt), (e) => e.message === 'unexpected signature count undefined');
    });
    it('empty inputs', function () {
        const psbt = src_1.testutil.constructPsbt([], [], network, rootWalletKeys, 'unsigned');
        assert.throws(() => (0, bitgo_1.extractP2msOnlyHalfSignedTx)(psbt), (e) => e.message === 'empty inputs or outputs');
    });
    it('unsupported script type', function () {
        const psbt = src_1.testutil.constructPsbt([{ scriptType: 'p2tr', value: BigInt(1000) }], [{ scriptType: 'p2sh', value: BigInt(900) }], network, rootWalletKeys, 'halfsigned');
        assert.throws(() => (0, bitgo_1.extractP2msOnlyHalfSignedTx)(psbt), (e) => e.message === 'unsupported script type taprootScriptPathSpend');
    });
});
function runExtractP2msOnlyHalfSignedTxTest(network, inputs, outputs) {
    const coin = (0, src_1.getNetworkName)(network);
    describe(`extractP2msOnlyHalfSignedTx success for ${coin}`, function () {
        it(`success for ${coin}`, function () {
            const signers = { signerName: 'user', cosignerName: 'backup' };
            const txnOutputs = outputs;
            const txnInputs = inputs
                .map((v) => v.scriptType === 'p2sh' || v.scriptType === 'p2shP2wsh' || v.scriptType === 'p2wsh'
                ? {
                    scriptType: v.scriptType,
                    value: v.value,
                }
                : undefined)
                .filter((v) => !!v);
            const psbt = src_1.testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'halfsigned', { signers });
            const halfSignedPsbtTx = (0, bitgo_1.extractP2msOnlyHalfSignedTx)(psbt);
            let txb = src_1.testutil.constructTxnBuilder(txnInputs, txnOutputs, network, rootWalletKeys, 'halfsigned', signers);
            const halfSignedTxbTx = txb.buildIncomplete();
            const unspents = (0, psbtUtil_1.toBigInt)(inputs.map((input, i) => src_1.testutil.toUnspent(input, i, network, rootWalletKeys)));
            (0, psbtUtil_1.assertEqualTransactions)(halfSignedPsbtTx, halfSignedTxbTx);
            (0, psbtUtil_1.validatePsbtParsing)(halfSignedPsbtTx, psbt, unspents, 'halfsigned');
            (0, psbtUtil_1.validatePsbtParsing)(halfSignedTxbTx, psbt, unspents, 'halfsigned');
            src_1.testutil.signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'fullsigned', { signers });
            const fullySignedPsbt = psbt.clone();
            const psbtTx = psbt.finalizeAllInputs().extractTransaction();
            const txnUnspents = txnInputs.map((v, i) => src_1.testutil.toTxnUnspent(v, i, network, rootWalletKeys));
            const prevOutputs = txnUnspents.map((u) => (0, bitgo_1.toOutput)(u, network));
            txb = (0, bitgo_1.createTransactionBuilderFromTransaction)(halfSignedTxbTx, prevOutputs);
            (0, testutil_1.signAllTxnInputs)(txb, txnInputs, rootWalletKeys, 'fullsigned', signers);
            const txbTx = txb.build();
            (0, psbtUtil_1.assertEqualTransactions)(psbtTx, txbTx);
            (0, psbtUtil_1.validatePsbtParsing)(psbtTx, fullySignedPsbt, unspents, 'fullsigned');
            (0, psbtUtil_1.validatePsbtParsing)(txbTx, fullySignedPsbt, unspents, 'fullsigned');
        });
    });
}
function runBuildSignSendFlowTest(network, inputs, outputs, { skipNonWitnessUtxo = false } = {}) {
    const coin = (0, src_1.getNetworkName)(network);
    function assertValidate(psbt) {
        psbt.data.inputs.forEach((input, i) => {
            assert.ok(psbt.validateSignaturesOfInputHD(i, rootWalletKeys['user']));
            if ((0, bitgo_1.getPsbtInputScriptType)(input) !== 'p2shP2pk') {
                assert.ok(psbt.validateSignaturesOfInputHD(i, rootWalletKeys['bitgo']));
            }
        });
        assert.ok(psbt.validateSignaturesOfAllInputs());
    }
    describe(`Build, sign & send flow for ${coin}`, function () {
        /**
         * Skip adding nonWitnessUtxos to psbts
         * ------------------------------------
         * In the instance that we want to doing a bulk sweep, for network and client performance reasons we are substituting
         * the nonWitnessUtxo for p2sh and p2shP2pk inputs with a witnessUtxo. We need the witnessUtxo so that we can half
         * sign the transaction locally with the user key. When we send the half signed to BitGo, the PSBT will be properly
         * populated such that the non-segwit inputs have the nonWitnessUtxo. This means when we send it to BitGo we should
         * remove the witnessUtxo so that it just has the partialSig and redeemScript.
         */
        it(`success for ${coin}${skipNonWitnessUtxo ? ' without nonWitnessUtxo for p2sh' : ''}`, function () {
            const parentPsbt = src_1.testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'unsigned', {
                signers: {
                    signerName: 'user',
                    cosignerName: 'bitgo',
                },
            });
            let psbt = skipNonWitnessUtxo ? (0, bitgo_1.clonePsbtWithoutNonWitnessUtxo)(parentPsbt) : parentPsbt;
            (0, bitgo_1.addXpubsToPsbt)(psbt, rootWalletKeys);
            psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user']);
            let psbtWithoutPrevTx = (0, bitgo_1.clonePsbtWithoutNonWitnessUtxo)(psbt);
            let hex = psbtWithoutPrevTx.toHex();
            let psbtAtHsm = (0, bitgo_1.createPsbtFromHex)(hex, network);
            psbtAtHsm.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo'], { deterministic: true });
            let hexAtHsm = psbtAtHsm.toHex();
            let psbtFromHsm = (0, bitgo_1.createPsbtFromHex)(hexAtHsm, network);
            (0, bitgo_1.deleteWitnessUtxoForNonSegwitInputs)(psbtFromHsm);
            psbt.combine(psbtFromHsm);
            src_1.testutil.signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned', {
                signers: {
                    signerName: 'user',
                    cosignerName: 'bitgo',
                },
                skipNonWitnessUtxo,
            });
            psbtWithoutPrevTx = (0, bitgo_1.clonePsbtWithoutNonWitnessUtxo)(psbt);
            hex = psbtWithoutPrevTx.toHex();
            psbtAtHsm = (0, bitgo_1.createPsbtFromHex)(hex, network);
            (0, bitgo_1.withUnsafeNonSegwit)(psbtAtHsm, () => {
                src_1.testutil.signAllPsbtInputs(psbtAtHsm, inputs, rootWalletKeys, 'fullsigned', {
                    signers: {
                        signerName: 'user',
                        cosignerName: 'bitgo',
                    },
                    deterministic: true,
                });
            });
            (0, bitgo_1.withUnsafeNonSegwit)(psbtAtHsm, () => {
                assertValidate(psbtAtHsm);
            });
            hexAtHsm = psbtAtHsm.toHex();
            psbtFromHsm = (0, bitgo_1.createPsbtFromHex)(hexAtHsm, network);
            (0, bitgo_1.deleteWitnessUtxoForNonSegwitInputs)(psbtFromHsm);
            if (skipNonWitnessUtxo) {
                psbt = parentPsbt;
            }
            psbt.combine(psbtFromHsm);
            assertValidate(psbt);
            assert.doesNotThrow(() => psbt.finalizeAllInputs().extractTransaction());
        });
    });
}
function runBuildPsbtWithSDK(network, inputs, outputs) {
    const coin = (0, src_1.getNetworkName)(network);
    it(`check that building a PSBT while skipping nonWitnessUtxo works - ${coin}`, async function () {
        const psbtWithNonWitness = src_1.testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'unsigned', {
            signers: {
                signerName: 'user',
                cosignerName: 'bitgo',
            },
        });
        const psbtWithoutNonWitness = src_1.testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'unsigned', {
            signers: {
                signerName: 'user',
                cosignerName: 'bitgo',
            },
            skipNonWitnessUtxo: true,
        });
        const clonedPsbt = (0, bitgo_1.clonePsbtWithoutNonWitnessUtxo)(psbtWithNonWitness);
        assert.deepStrictEqual(psbtWithoutNonWitness.toHex(), clonedPsbt.toHex());
    });
}
(0, src_1.getNetworkList)()
    .filter((v) => (0, src_1.isMainnet)(v) && v !== src_1.networks.bitcoinsv)
    .forEach((network) => {
    runExtractP2msOnlyHalfSignedTxTest(network, halfSignedInputs.filter((input) => (0, outputScripts_1.isSupportedScriptType)(network, input.scriptType)), halfSignedOutputs.filter((output) => (0, outputScripts_1.isSupportedScriptType)(network, output.scriptType)));
    const supportedPsbtInputs = psbtInputs.filter((input) => (0, outputScripts_1.isSupportedScriptType)(network, input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType));
    const supportedPsbtOutputs = psbtOutputs.filter((output) => (0, outputScripts_1.isSupportedScriptType)(network, output.scriptType));
    [false, true].forEach((skipNonWitnessUtxo) => runBuildSignSendFlowTest(network, supportedPsbtInputs, supportedPsbtOutputs, { skipNonWitnessUtxo }));
    runBuildPsbtWithSDK(network, supportedPsbtInputs, supportedPsbtOutputs);
});
describe('isTransactionWithKeyPathSpendInput', function () {
    describe('transaction input', function () {
        it('empty inputs', function () {
            const tx = src_1.testutil.constructTxnBuilder([], [], network, rootWalletKeys, 'unsigned').buildIncomplete();
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx), false);
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx.ins), false);
        });
        it('taprootKeyPath inputs successfully triggers', function () {
            const psbt = src_1.testutil.constructPsbt([
                { scriptType: 'taprootKeyPathSpend', value: BigInt(1e8) },
                { scriptType: 'p2sh', value: BigInt(1e8) },
            ], [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }], network, rootWalletKeys, 'fullsigned');
            assert(psbt.validateSignaturesOfAllInputs());
            psbt.finalizeAllInputs();
            const tx = psbt.extractTransaction();
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx), true);
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx.ins), true);
        });
        it('no taprootKeyPath inputs successfully does not trigger', function () {
            const psbt = src_1.testutil.constructPsbt([
                { scriptType: 'p2trMusig2', value: BigInt(1e8) },
                { scriptType: 'p2sh', value: BigInt(1e8) },
            ], [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }], network, rootWalletKeys, 'fullsigned');
            assert(psbt.validateSignaturesOfAllInputs());
            psbt.finalizeAllInputs();
            const tx = psbt.extractTransaction();
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx), false);
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx.ins), false);
        });
        it('unsigned inputs successfully fail', function () {
            const psbt = src_1.testutil.constructPsbt([
                { scriptType: 'p2wsh', value: BigInt(1e8) },
                { scriptType: 'p2sh', value: BigInt(1e8) },
            ], [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }], network, rootWalletKeys, 'unsigned');
            const tx = psbt.getUnsignedTx();
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx), false);
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(tx.ins), false);
        });
    });
    describe('psbt input', function () {
        it('empty inputs', function () {
            const psbt = src_1.testutil.constructPsbt([], [], network, rootWalletKeys, 'unsigned');
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(psbt), false);
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(psbt.data.inputs), false);
        });
        it('psbt with taprootKeyPathInputs successfully triggers', function () {
            const psbt = src_1.testutil.constructPsbt([
                { scriptType: 'taprootKeyPathSpend', value: BigInt(1e8) },
                { scriptType: 'p2sh', value: BigInt(1e8) },
            ], [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }], network, rootWalletKeys, 'unsigned');
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(psbt), true);
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(psbt.data.inputs), true);
        });
        it('psbt without taprootKeyPathInputs successfully does not trigger', function () {
            const psbt = src_1.testutil.constructPsbt([
                { scriptType: 'p2wsh', value: BigInt(1e8) },
                { scriptType: 'p2sh', value: BigInt(1e8) },
            ], [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }], network, rootWalletKeys, 'halfsigned');
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(psbt), false);
            assert.strictEqual((0, bitgo_1.isTransactionWithKeyPathSpendInput)(psbt.data.inputs), false);
        });
    });
});
describe('Parse PSBT', function () {
    it('p2shP2pk parsing', function () {
        const signer = rootWalletKeys['user'];
        const psbt = (0, bitgo_1.createPsbtForNetwork)({ network: src_1.networks.bitcoincash });
        const unspent = (0, testutil_1.mockReplayProtectionUnspent)(src_1.networks.bitcoincash, BigInt(1e8), { key: signer });
        const { redeemScript } = (0, outputScripts_1.createOutputScriptP2shP2pk)(signer.publicKey);
        assert(redeemScript);
        (0, bitgo_1.addReplayProtectionUnspentToPsbt)(psbt, unspent, redeemScript);
        (0, bitgo_1.addWalletOutputToPsbt)(psbt, rootWalletKeys, (0, bitgo_1.getInternalChainCode)('p2sh'), 0, BigInt(1e8 - 10000));
        const input = psbt.data.inputs[0];
        let parsed = (0, bitgo_1.parsePsbtInput)(input);
        assert.strictEqual(parsed.scriptType, 'p2shP2pk');
        assert.strictEqual(parsed.signatures, undefined);
        assert.strictEqual(parsed.publicKeys.length, 1);
        assert.ok(parsed.publicKeys[0].length === 33);
        assert.ok(parsed.pubScript.equals(redeemScript));
        psbt.signAllInputs(signer);
        assert.ok(psbt.validateSignaturesOfAllInputs());
        parsed = (0, bitgo_1.parsePsbtInput)(input);
        assert.strictEqual(parsed.scriptType, 'p2shP2pk');
        assert.strictEqual(parsed.signatures?.length, 1);
        assert.strictEqual(parsed.publicKeys.length, 1);
        assert.ok(parsed.publicKeys[0].length === 33);
        assert.ok(parsed.pubScript.equals(redeemScript));
        const sighash = parsed.signatures[0][parsed.signatures[0].length - 1];
        assert.strictEqual(sighash, (0, bitgo_1.getDefaultSigHash)(psbt.network));
    });
    it('fail to parse finalized psbt', function () {
        const unspents = (0, testutil_2.mockUnspents)(rootWalletKeys, getScriptTypes2Of3().map((inputType) => inputType), BigInt('10000000000000000'), network);
        const txBuilderParams = {
            signer: 'user',
            cosigner: 'bitgo',
            amountType: 'bigint',
            outputType: 'p2sh',
            signatureTarget: 'fullsigned',
            network,
            changeIndex: CHANGE_INDEX,
            fee: FEE,
        };
        const tx = (0, psbtUtil_1.constructTransactionUsingTxBuilder)(unspents, rootWalletKeys, txBuilderParams);
        const psbt = (0, bitgo_1.toWalletPsbt)(tx, (0, psbtUtil_1.toBigInt)(unspents), rootWalletKeys);
        psbt.validateSignaturesOfAllInputs();
        psbt.finalizeAllInputs();
        psbt.data.inputs.forEach((input, i) => {
            assert.throws(() => (0, bitgo_1.parsePsbtInput)(input), (e) => e.message === 'Finalized PSBT parsing is not supported');
        });
    });
    it('fail to parse input with more than one script type metadata', function () {
        const unspents = (0, testutil_2.mockUnspents)(rootWalletKeys, ['p2tr'], BigInt('10000000000000000'), network);
        const txBuilderParams = {
            signer: 'user',
            cosigner: 'bitgo',
            amountType: 'bigint',
            outputType: 'p2sh',
            signatureTarget: 'halfsigned',
            network,
            changeIndex: CHANGE_INDEX,
            fee: FEE,
        };
        const txP2tr = (0, psbtUtil_1.constructTransactionUsingTxBuilder)([unspents[0]], rootWalletKeys, txBuilderParams);
        const psbtP2tr = (0, bitgo_1.toWalletPsbt)(txP2tr, (0, psbtUtil_1.toBigInt)([unspents[0]]), rootWalletKeys);
        const walletKeys = rootWalletKeys.deriveForChainAndIndex((0, bitgo_1.getExternalChainCode)('p2sh'), 0);
        const { redeemScript } = (0, outputScripts_1.createOutputScript2of3)(walletKeys.publicKeys, 'p2sh');
        psbtP2tr.updateInput(0, { redeemScript });
        assert.throws(() => (0, bitgo_1.parsePsbtInput)(psbtP2tr.data.inputs[0]), (e) => e.message === 'Found both p2sh and taprootScriptPath PSBT metadata.');
    });
    it('fail to parse more than one tap leaf script per input', function () {
        const unspents = (0, testutil_2.mockUnspents)(rootWalletKeys, ['p2tr'], BigInt('10000000000000000'), network);
        const txBuilderParams = {
            signer: 'user',
            cosigner: 'bitgo',
            amountType: 'bigint',
            outputType: 'p2sh',
            signatureTarget: 'halfsigned',
            network,
            changeIndex: CHANGE_INDEX,
            fee: FEE,
        };
        const txP2tr1 = (0, psbtUtil_1.constructTransactionUsingTxBuilder)([unspents[0]], rootWalletKeys, txBuilderParams);
        const psbtP2tr1 = (0, bitgo_1.toWalletPsbt)(txP2tr1, (0, psbtUtil_1.toBigInt)([unspents[0]]), rootWalletKeys);
        const txBuilderParams2 = {
            signer: 'user',
            cosigner: 'backup',
            amountType: 'bigint',
            outputType: 'p2sh',
            signatureTarget: 'halfsigned',
            network,
            changeIndex: CHANGE_INDEX,
            fee: FEE,
        };
        const txP2tr2 = (0, psbtUtil_1.constructTransactionUsingTxBuilder)([unspents[0]], rootWalletKeys, txBuilderParams2);
        const psbtP2tr2 = (0, bitgo_1.toWalletPsbt)(txP2tr2, (0, psbtUtil_1.toBigInt)([unspents[0]]), rootWalletKeys);
        const txBuilderParams3 = {
            signer: 'user',
            cosigner: 'bitgo',
            amountType: 'bigint',
            outputType: 'p2sh',
            signatureTarget: 'unsigned',
            network,
            changeIndex: CHANGE_INDEX,
            fee: FEE,
        };
        const txP2tr3 = (0, psbtUtil_1.constructTransactionUsingTxBuilder)([unspents[0]], rootWalletKeys, txBuilderParams3);
        const psbtP2tr3 = (0, bitgo_1.toWalletPsbt)(txP2tr3, (0, psbtUtil_1.toBigInt)([unspents[0]]), rootWalletKeys);
        if (psbtP2tr1.data.inputs[0].tapLeafScript && psbtP2tr2.data.inputs[0].tapLeafScript) {
            const tapLeafScripts = [psbtP2tr1.data.inputs[0].tapLeafScript[0], psbtP2tr2.data.inputs[0].tapLeafScript[0]];
            psbtP2tr3.updateInput(0, { tapLeafScript: tapLeafScripts });
            assert.throws(() => (0, bitgo_1.parsePsbtInput)(psbtP2tr3.data.inputs[0]), (e) => e.message === 'Bitgo only supports a single tap leaf script per input.');
        }
    });
});
describe('isPsbt', function () {
    function isPsbtForNetwork(n) {
        describe(`network: ${(0, src_1.getNetworkName)(n)}`, function () {
            const psbt = (0, bitgo_1.createPsbtForNetwork)({ network: n });
            it('should return true for a valid PSBT', function () {
                const psbtBuff = psbt.toBuffer();
                assert.strictEqual((0, bitgo_1.isPsbt)(psbtBuff), true);
                assert.strictEqual((0, bitgo_1.isPsbt)(psbtBuff.toString('hex')), true);
            });
            it('should return false for a transaction', function () {
                assert.strictEqual((0, bitgo_1.isPsbt)(psbt.getUnsignedTx().toBuffer()), false);
            });
            it('should return false for a truncated magic word', function () {
                const hex = psbt.toBuffer().slice(0, 3);
                assert.strictEqual((0, bitgo_1.isPsbt)(hex), false);
                assert.strictEqual((0, bitgo_1.isPsbt)(Buffer.from(hex)), false);
            });
            it('should return false for a valid PSBT with an invalid magic', function () {
                const buffer = psbt.toBuffer();
                buffer.writeUInt8(0x00, 1);
                assert.strictEqual((0, bitgo_1.isPsbt)(psbt.getUnsignedTx().toBuffer()), false);
            });
            it('should return false for a valid PSBT with an invalid separator', function () {
                const buffer = psbt.toBuffer();
                buffer.writeUInt8(0xfe, 4);
                assert.strictEqual((0, bitgo_1.isPsbt)(psbt.getUnsignedTx().toBuffer()), false);
            });
            it('should return false for a random buffer', function () {
                const random = 'deadbeaf';
                const buffer = Buffer.from(random, 'hex');
                assert.strictEqual((0, bitgo_1.isPsbt)(random), false);
                assert.strictEqual((0, bitgo_1.isPsbt)(buffer), false);
            });
            it('should return true if buffer is changed after the separator', function () {
                const buffer = psbt.toBuffer();
                buffer.writeUInt8(0x00, 5);
                assert.strictEqual((0, bitgo_1.isPsbt)(buffer), true);
            });
        });
    }
    (0, src_1.getNetworkList)().forEach((n) => isPsbtForNetwork(n));
});
describe('Update incomplete psbt', function () {
    function removeFromPsbt(psbtHex, network, remove) {
        const utxoPsbt = (0, bitgo_1.createPsbtFromHex)(psbtHex, network);
        const psbt = (0, bitgo_1.createPsbtForNetwork)({ network: utxoPsbt.network });
        const txInputs = utxoPsbt.txInputs;
        utxoPsbt.data.inputs.map((input, ii) => {
            const { hash, index } = txInputs[ii];
            if (remove.input && ii === remove.input.index) {
                delete input[remove.input.fieldToRemove];
            }
            psbt.addInput({ ...input, hash, index });
        });
        const txOutputs = utxoPsbt.txOutputs;
        utxoPsbt.data.outputs.map((output, ii) => {
            if (remove.output && remove.output.index === ii) {
                delete output[remove.output.fieldToRemove];
            }
            psbt.addOutput({ ...output, script: txOutputs[ii].script, value: txOutputs[ii].value });
        });
        return psbt;
    }
    function signAllInputs(psbt, { assertValidSignaturesAndExtractable = true } = {}) {
        psbt.data.inputs.forEach((input, inputIndex) => {
            const parsedInput = (0, bitgo_1.parsePsbtInput)(input);
            if (parsedInput.scriptType === 'taprootKeyPathSpend') {
                psbt.setInputMusig2NonceHD(inputIndex, rootWalletKeys[signer]);
                psbt.setInputMusig2NonceHD(inputIndex, rootWalletKeys[cosigner]);
            }
            if (parsedInput.scriptType === 'p2shP2pk') {
                psbt.signInput(inputIndex, testutil_1.replayProtectionKeyPair);
            }
            else {
                psbt.signInputHD(inputIndex, rootWalletKeys[signer]);
                psbt.signInputHD(inputIndex, rootWalletKeys[cosigner]);
            }
        });
        if (assertValidSignaturesAndExtractable) {
            assert.ok(psbt.validateSignaturesOfAllInputs());
            psbt.finalizeAllInputs();
            const txExtracted = psbt.extractTransaction();
            assert.ok(txExtracted);
        }
    }
    let psbtHex;
    let unspents;
    const signer = 'user';
    const cosigner = 'bitgo';
    const scriptTypes = [...outputScripts_1.scriptTypes2Of3, 'p2shP2pk'];
    const outputValue = BigInt((2e8 * scriptTypes.length - 100) / 5);
    const outputs = [
        { chain: (0, bitgo_1.getExternalChainCode)('p2sh'), index: 88, value: outputValue },
        { chain: (0, bitgo_1.getExternalChainCode)('p2shP2wsh'), index: 89, value: outputValue },
        { chain: (0, bitgo_1.getExternalChainCode)('p2wsh'), index: 90, value: outputValue },
        { chain: (0, bitgo_1.getExternalChainCode)('p2tr'), index: 91, value: outputValue },
        { chain: (0, bitgo_1.getExternalChainCode)('p2trMusig2'), index: 92, value: outputValue },
    ];
    before(function () {
        unspents = (0, testutil_2.mockUnspents)(rootWalletKeys, scriptTypes, BigInt(2e8), network);
        const psbt = (0, Musig2Util_1.constructPsbt)(unspents, rootWalletKeys, signer, cosigner, outputs);
        psbtHex = psbt.toHex();
    });
    it('can create a sign-able psbt from an unsigned transaction extracted from the psbt', function () {
        if (true) {
            return;
        }
        const psbtOrig = (0, bitgo_1.createPsbtFromHex)(psbtHex, network);
        const tx = psbtOrig.getUnsignedTx();
        const psbt = (0, bitgo_1.createPsbtFromTransaction)(tx, unspents.map((u) => (0, bitgo_1.toPrevOutput)(u, network)));
        unspents.forEach((u, inputIndex) => {
            if ((0, bitgo_1.isWalletUnspent)(u)) {
                (0, bitgo_1.updateWalletUnspentForPsbt)(psbt, inputIndex, u, rootWalletKeys, signer, cosigner);
            }
            else {
                const { redeemScript } = (0, outputScripts_1.createOutputScriptP2shP2pk)(testutil_1.replayProtectionKeyPair.publicKey);
                (0, bitgo_1.updateReplayProtectionUnspentToPsbt)(psbt, inputIndex, u, redeemScript);
            }
        });
        signAllInputs(psbt);
    });
    const componentsOnEachInputScriptType = {
        p2sh: ['nonWitnessUtxo', 'redeemScript', 'bip32Derivation'],
        p2shP2wsh: ['witnessUtxo', 'bip32Derivation', 'redeemScript', 'witnessScript'],
        p2wsh: ['witnessUtxo', 'witnessScript', 'bip32Derivation'],
        p2tr: ['witnessUtxo', 'tapLeafScript', 'tapBip32Derivation'],
        p2trMusig2: ['witnessUtxo', 'tapBip32Derivation', 'tapInternalKey', 'tapMerkleRoot', 'unknownKeyVals'],
        p2shP2pk: ['redeemScript', 'nonWitnessUtxo'],
    };
    const p2trComponents = ['tapTree', 'tapInternalKey', 'tapBip32Derivation'];
    const componentsOnEachOutputScriptType = {
        p2sh: ['bip32Derivation', 'redeemScript'],
        p2shP2wsh: ['bip32Derivation', 'witnessScript', 'redeemScript'],
        p2wsh: ['bip32Derivation', 'witnessScript'],
        p2tr: p2trComponents,
        p2trMusig2: p2trComponents,
        p2shP2pk: [],
    };
    scriptTypes.forEach((scriptType, i) => {
        componentsOnEachInputScriptType[scriptType].forEach((inputComponent) => {
            it(`[${scriptType}] missing ${inputComponent} on input should succeed in fully signing unsigned psbt after update`, function () {
                const psbt = removeFromPsbt(psbtHex, network, { input: { index: i, fieldToRemove: inputComponent } });
                const unspent = unspents[i];
                if ((0, bitgo_1.isWalletUnspent)(unspent)) {
                    (0, bitgo_1.updateWalletUnspentForPsbt)(psbt, i, unspent, rootWalletKeys, signer, cosigner);
                }
                else {
                    const { redeemScript } = (0, outputScripts_1.createOutputScriptP2shP2pk)(testutil_1.replayProtectionKeyPair.publicKey);
                    assert.ok(redeemScript);
                    (0, bitgo_1.updateReplayProtectionUnspentToPsbt)(psbt, i, unspent, redeemScript);
                }
                signAllInputs(psbt);
            });
        });
        componentsOnEachOutputScriptType[scriptType].forEach((outputComponent) => {
            it(`[${scriptType}] missing ${outputComponent} on output should produce same hex as fully hydrated after update`, function () {
                const psbt = removeFromPsbt(psbtHex, network, { output: { index: i, fieldToRemove: outputComponent } });
                (0, bitgo_1.updateWalletOutputForPsbt)(psbt, rootWalletKeys, i, outputs[i].chain, outputs[i].index);
                assert.strictEqual(psbt.toHex(), psbtHex);
            });
        });
    });
});
describe('Psbt from transaction using wallet unspents', function () {
    function runTestSignUnspents({ inputScriptTypes, outputScriptType, signer, cosigner, amountType, testOutputAmount, signatureTarget, }) {
        it(`can be signed [inputs=${inputScriptTypes} signer=${signer} cosigner=${cosigner} amountType=${amountType} signatureTarget=${signatureTarget}]`, function () {
            const unspents = (0, testutil_2.mockUnspents)(rootWalletKeys, inputScriptTypes, testOutputAmount, network);
            // const txBuilderParams = { network, changeIndex: CHANGE_INDEX, fee: FEE };
            const txBuilderParams = {
                signer,
                cosigner,
                amountType,
                outputType: outputScriptType,
                signatureTarget: signatureTarget,
                network,
                changeIndex: CHANGE_INDEX,
                fee: FEE,
            };
            const tx = (0, psbtUtil_1.constructTransactionUsingTxBuilder)(unspents, rootWalletKeys, txBuilderParams);
            const unspentBigInt = (0, psbtUtil_1.toBigInt)(unspents);
            const psbt = (0, bitgo_1.toWalletPsbt)(tx, unspentBigInt, rootWalletKeys);
            (0, psbtUtil_1.validatePsbtParsing)(tx, psbt, unspentBigInt, signatureTarget);
            // Check that the correct unspent corresponds to the input
            unspentBigInt.forEach((unspent, inputIndex) => {
                const otherUnspent = inputIndex === 0 ? unspentBigInt[1] : unspentBigInt[0];
                assert.strictEqual((0, bitgo_1.psbtIncludesUnspentAtIndex)(psbt, inputIndex, unspent.id), true);
                assert.strictEqual((0, bitgo_1.psbtIncludesUnspentAtIndex)(psbt, inputIndex, otherUnspent.id), false);
                (0, bitgo_1.updateWalletUnspentForPsbt)(psbt, inputIndex, unspent, rootWalletKeys, signer, cosigner);
            });
            if (signatureTarget !== 'fullsigned') {
                // Now signing to make it fully signed psbt.
                // So it will be easy to verify its validity with another similar tx to be built with tx builder.
                (0, psbtUtil_1.signPsbt)(psbt, unspentBigInt, rootWalletKeys, signer, cosigner, signatureTarget);
            }
            assert.deepStrictEqual(psbt.validateSignaturesOfAllInputs(), true);
            psbt.finalizeAllInputs();
            const txFromPsbt = psbt.extractTransaction();
            const txBuilderParams2 = {
                signer,
                cosigner,
                amountType,
                outputType: outputScriptType,
                signatureTarget: 'fullsigned',
                network,
                changeIndex: CHANGE_INDEX,
                fee: FEE,
            };
            // New legacy tx resembles the signed psbt.
            const txFromTxBuilder = (0, psbtUtil_1.constructTransactionUsingTxBuilder)(unspents, rootWalletKeys, txBuilderParams2);
            assert.deepStrictEqual(txFromPsbt.getHash(), txFromTxBuilder.getHash());
        });
    }
    function getInputScripts() {
        return getScriptTypes2Of3().flatMap((t) => {
            return getScriptTypes2Of3().flatMap((lastType) => {
                return [[t, t, lastType]];
            });
        });
    }
    function getSignerPairs(containsTaprootInput) {
        const signaturePairs = [['user', 'bitgo']];
        if (containsTaprootInput) {
            signaturePairs.push(['user', 'backup']);
        }
        return signaturePairs;
    }
    ['unsigned', 'halfsigned', 'fullsigned'].forEach((signatureTarget) => {
        getInputScripts().forEach((inputScriptTypes) => {
            getSignerPairs(inputScriptTypes.includes('p2tr')).forEach(([signer, cosigner]) => {
                runTestSignUnspents({
                    inputScriptTypes,
                    outputScriptType: 'p2sh',
                    signer,
                    cosigner,
                    amountType: 'number',
                    testOutputAmount: transaction_util_1.defaultTestOutputAmount,
                    signatureTarget,
                });
                runTestSignUnspents({
                    inputScriptTypes,
                    outputScriptType: 'p2sh',
                    signer,
                    cosigner,
                    amountType: 'bigint',
                    testOutputAmount: BigInt('10000000000000000'),
                    signatureTarget,
                });
            });
        });
    });
});
function testUtxoPsbt(coinNetwork) {
    describe(`Testing UtxoPsbt (de)serialization for ${(0, src_1.getNetworkName)(coinNetwork)} network`, function () {
        let psbt;
        let psbtHex;
        let unspents;
        before(async function () {
            unspents = (0, testutil_2.mockUnspents)(rootWalletKeys, ['p2sh'], BigInt('10000000000000'), coinNetwork);
            const txBuilderParams = {
                signer: 'user',
                cosigner: 'bitgo',
                amountType: 'bigint',
                outputType: 'p2sh',
                signatureTarget: 'fullsigned',
                network: coinNetwork,
                changeIndex: CHANGE_INDEX,
                fee: FEE,
            };
            const tx = (0, psbtUtil_1.constructTransactionUsingTxBuilder)(unspents, rootWalletKeys, txBuilderParams);
            psbt = (0, bitgo_1.toWalletPsbt)(tx, (0, psbtUtil_1.toBigInt)(unspents), rootWalletKeys);
            if (coinNetwork === src_1.networks.zcash) {
                psbt.setDefaultsForVersion(network, 450);
            }
            psbtHex = psbt.toHex();
        });
        it('should be able to clone psbt', async function () {
            const clone = psbt.clone();
            assert(clone instanceof psbt.constructor, `Expected clone to be instance of ${psbt.constructor.name}`);
            assert.deepStrictEqual(clone.toBuffer(), psbt.toBuffer());
            assert.deepStrictEqual(clone.clone().toBuffer(), psbt.toBuffer());
            assert.strictEqual(clone.network, psbt.network);
            assert.strictEqual(clone.clone().network, psbt.network);
        });
        it('should be able to round-trip', async function () {
            assert.deepStrictEqual((0, bitgo_1.createPsbtFromHex)(psbtHex, coinNetwork, false).toBuffer(), psbt.toBuffer());
        });
        it('should be able to get transaction info from psbt', function () {
            const txInfo = (0, bitgo_1.getTransactionAmountsFromPsbt)(psbt);
            assert.strictEqual(txInfo.fee, FEE);
            assert.strictEqual(txInfo.inputCount, unspents.length);
            assert.strictEqual(txInfo.inputAmount, BigInt('10000000000000') * BigInt(unspents.length));
            assert.strictEqual(txInfo.outputAmount, BigInt('10000000000000') * BigInt(unspents.length) - FEE);
            assert.strictEqual(txInfo.outputCount, psbt.data.outputs.length);
        });
        function deserializeBip32PathsCorrectly(bip32PathsAbsolute) {
            function checkDerivationPrefix(bip32Derivation) {
                const path = bip32Derivation.path.split('/');
                const prefix = bip32PathsAbsolute ? 'm' : '0';
                assert(path[0] === prefix);
            }
            it(`should deserialize PSBT bip32Derivations with paths ${bip32PathsAbsolute ? '' : 'not '} absolute`, async function () {
                const deserializedPsbt = (0, bitgo_1.createPsbtFromHex)(psbtHex, coinNetwork, bip32PathsAbsolute);
                assert(deserializedPsbt);
                deserializedPsbt.data.inputs.forEach((input) => {
                    input?.bip32Derivation?.forEach((derivation) => checkDerivationPrefix(derivation));
                    input?.tapBip32Derivation?.forEach((derivation) => checkDerivationPrefix(derivation));
                });
            });
        }
        [true, false].forEach((bip32PathsAbsolute) => deserializeBip32PathsCorrectly(bip32PathsAbsolute));
    });
}
[src_1.networks.bitcoin, src_1.networks.zcash, src_1.networks.dash, src_1.networks.dogecoin, src_1.networks.litecoin].forEach((coinNetwork) => testUtxoPsbt(coinNetwork));
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Psbt.js","sourceRoot":"","sources":["../../../../test/bitgo/psbt/Psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport { Network, getNetworkName, networks, getNetworkList, testutil, isMainnet, Transaction } from '../../../src';\nimport {\n  getExternalChainCode,\n  outputScripts,\n  KeyName,\n  UtxoPsbt,\n  ZcashPsbt,\n  createPsbtFromHex,\n  parsePsbtInput,\n  toWalletPsbt,\n  createPsbtForNetwork,\n  addReplayProtectionUnspentToPsbt,\n  addWalletOutputToPsbt,\n  getInternalChainCode,\n  UtxoTransaction,\n  isTransactionWithKeyPathSpendInput,\n  isPsbt,\n  psbtIncludesUnspentAtIndex,\n  updateWalletUnspentForPsbt,\n  createPsbtFromTransaction,\n  toPrevOutput,\n  updateReplayProtectionUnspentToPsbt,\n  Unspent,\n  isWalletUnspent,\n  updateWalletOutputForPsbt,\n  extractP2msOnlyHalfSignedTx,\n  toOutput,\n  createTransactionBuilderFromTransaction,\n  addXpubsToPsbt,\n  clonePsbtWithoutNonWitnessUtxo,\n  deleteWitnessUtxoForNonSegwitInputs,\n  getPsbtInputScriptType,\n  withUnsafeNonSegwit,\n  getTransactionAmountsFromPsbt,\n  WalletUnspent,\n  getDefaultSigHash,\n  isPsbtLite,\n} from '../../../src/bitgo';\nimport {\n  createOutputScript2of3,\n  createOutputScriptP2shP2pk,\n  isSupportedScriptType,\n  ScriptType2Of3,\n  ScriptTypeP2shP2pk,\n  scriptTypes2Of3,\n} from '../../../src/bitgo/outputScripts';\n\nimport {\n  getDefaultWalletKeys,\n  Input,\n  inputScriptTypes,\n  mockReplayProtectionUnspent,\n  Output,\n  outputScriptTypes,\n  replayProtectionKeyPair,\n  signAllTxnInputs,\n} from '../../../src/testutil';\n\nimport { defaultTestOutputAmount } from '../../transaction_util';\nimport {\n  assertEqualTransactions,\n  constructTransactionUsingTxBuilder,\n  signPsbt,\n  toBigInt,\n  validatePsbtParsing,\n} from './psbtUtil';\n\nimport { mockUnspents } from '../../../src/testutil';\nimport { constructPsbt } from './Musig2Util';\n\nconst CHANGE_INDEX = 100;\nconst FEE = BigInt(100);\n\nexport type AmountType = 'number' | 'bigint';\nexport type InputType = outputScripts.ScriptType2Of3;\nexport type SignatureTargetType = 'unsigned' | 'halfsigned' | 'fullsigned';\n\nconst network = networks.bitcoin;\nconst rootWalletKeys = getDefaultWalletKeys();\n\nfunction getScriptTypes2Of3() {\n  // FIXME(BG-66941): p2trMusig2 signing does not work in this test suite yet\n  //  because the test suite is written with TransactionBuilder\n  return outputScripts.scriptTypes2Of3.filter((scriptType) => scriptType !== 'p2trMusig2');\n}\n\nconst halfSignedInputs = (['p2sh', 'p2wsh', 'p2shP2wsh'] as const).map((scriptType) => ({\n  scriptType,\n  value: BigInt(1000),\n}));\nconst halfSignedOutputs = outputScriptTypes.map((scriptType) => ({ scriptType, value: BigInt(500) }));\n\nconst psbtInputs = inputScriptTypes.map((scriptType) => ({ scriptType, value: BigInt(1000) }));\nconst psbtOutputs = outputScriptTypes.map((scriptType) => ({ scriptType, value: BigInt(900) }));\n\ndescribe('Psbt Misc', function () {\n  function getTestPsbt() {\n    return testutil.constructPsbt(\n      [{ scriptType: 'p2tr', value: BigInt(1000) }],\n      [{ scriptType: 'p2sh', value: BigInt(900) }],\n      network,\n      rootWalletKeys,\n      'fullsigned'\n    );\n  }\n  it('fail to finalise p2tr sighash mismatch', function () {\n    const psbt = getTestPsbt();\n    assert(psbt.validateSignaturesOfAllInputs());\n    const tapScriptSig = psbt.data.inputs[0].tapScriptSig;\n    assert(tapScriptSig);\n    tapScriptSig[0].signature = Buffer.concat([tapScriptSig[0].signature, Buffer.of(Transaction.SIGHASH_ALL)]);\n    assert.throws(\n      () => psbt.finalizeAllInputs(),\n      (e: any) => e.message === 'signature sighash does not match input sighash type'\n    );\n  });\n\n  describe('isPsbtLite', function () {\n    it('no inputs', function () {\n      const psbt = testutil.constructPsbt([], [], network, rootWalletKeys, 'unsigned');\n      assert.strictEqual(isPsbtLite(psbt), false);\n    });\n\n    it('all inputs are segwit', function () {\n      const psbt = testutil.constructPsbt(\n        psbtInputs.filter((s) => s.scriptType !== 'p2sh' && s.scriptType !== 'p2shP2pk'),\n        psbtOutputs,\n        network,\n        rootWalletKeys,\n        'unsigned'\n      );\n      assert.strictEqual(isPsbtLite(psbt), false);\n    });\n\n    it('some inputs are non-segwit', function () {\n      const psbt = testutil.constructPsbt(psbtInputs, psbtOutputs, network, rootWalletKeys, 'unsigned');\n      assert.strictEqual(isPsbtLite(psbt), false);\n    });\n\n    it('should be true if after clonePsbtWithoutNonWitnessUtxo', function () {\n      const psbt = testutil.constructPsbt(psbtInputs, psbtOutputs, network, rootWalletKeys, 'unsigned');\n      const clonedPsbt = clonePsbtWithoutNonWitnessUtxo(psbt);\n      assert.strictEqual(isPsbtLite(clonedPsbt), true);\n    });\n  });\n});\n\ndescribe('extractP2msOnlyHalfSignedTx failure', function () {\n  it('invalid signature count', function () {\n    const psbt = testutil.constructPsbt(halfSignedInputs, halfSignedOutputs, network, rootWalletKeys, 'unsigned');\n    assert.throws(\n      () => extractP2msOnlyHalfSignedTx(psbt),\n      (e: any) => e.message === 'unexpected signature count undefined'\n    );\n  });\n\n  it('empty inputs', function () {\n    const psbt = testutil.constructPsbt([], [], network, rootWalletKeys, 'unsigned');\n    assert.throws(\n      () => extractP2msOnlyHalfSignedTx(psbt),\n      (e: any) => e.message === 'empty inputs or outputs'\n    );\n  });\n\n  it('unsupported script type', function () {\n    const psbt = testutil.constructPsbt(\n      [{ scriptType: 'p2tr', value: BigInt(1000) }],\n      [{ scriptType: 'p2sh', value: BigInt(900) }],\n      network,\n      rootWalletKeys,\n      'halfsigned'\n    );\n    assert.throws(\n      () => extractP2msOnlyHalfSignedTx(psbt),\n      (e: any) => e.message === 'unsupported script type taprootScriptPathSpend'\n    );\n  });\n});\n\nfunction runExtractP2msOnlyHalfSignedTxTest(network: Network, inputs: Input[], outputs: Output[]) {\n  const coin = getNetworkName(network);\n\n  describe(`extractP2msOnlyHalfSignedTx success for ${coin}`, function () {\n    it(`success for ${coin}`, function () {\n      const signers: { signerName: KeyName; cosignerName: KeyName } = { signerName: 'user', cosignerName: 'backup' };\n      const txnOutputs = outputs;\n      const txnInputs = inputs\n        .map((v) =>\n          v.scriptType === 'p2sh' || v.scriptType === 'p2shP2wsh' || v.scriptType === 'p2wsh'\n            ? {\n                scriptType: v.scriptType,\n                value: v.value,\n              }\n            : undefined\n        )\n        .filter((v) => !!v) as testutil.TxnInput<bigint>[];\n\n      const psbt = testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'halfsigned', { signers });\n      const halfSignedPsbtTx = extractP2msOnlyHalfSignedTx(psbt);\n\n      let txb = testutil.constructTxnBuilder(txnInputs, txnOutputs, network, rootWalletKeys, 'halfsigned', signers);\n      const halfSignedTxbTx = txb.buildIncomplete();\n\n      const unspents = toBigInt(inputs.map((input, i) => testutil.toUnspent(input, i, network, rootWalletKeys)));\n\n      assertEqualTransactions(halfSignedPsbtTx, halfSignedTxbTx);\n      validatePsbtParsing(halfSignedPsbtTx, psbt, unspents, 'halfsigned');\n      validatePsbtParsing(halfSignedTxbTx, psbt, unspents, 'halfsigned');\n\n      testutil.signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'fullsigned', { signers });\n      const fullySignedPsbt = psbt.clone();\n      const psbtTx = psbt.finalizeAllInputs().extractTransaction();\n\n      const txnUnspents = txnInputs.map((v, i) => testutil.toTxnUnspent(v, i, network, rootWalletKeys));\n      const prevOutputs = txnUnspents.map((u) => toOutput(u, network));\n      txb = createTransactionBuilderFromTransaction<bigint>(halfSignedTxbTx, prevOutputs);\n      signAllTxnInputs(txb, txnInputs, rootWalletKeys, 'fullsigned', signers);\n      const txbTx = txb.build();\n\n      assertEqualTransactions(psbtTx, txbTx);\n      validatePsbtParsing(psbtTx, fullySignedPsbt, unspents, 'fullsigned');\n      validatePsbtParsing(txbTx, fullySignedPsbt, unspents, 'fullsigned');\n    });\n  });\n}\n\nfunction runBuildSignSendFlowTest(\n  network: Network,\n  inputs: Input[],\n  outputs: Output[],\n  { skipNonWitnessUtxo = false } = {}\n) {\n  const coin = getNetworkName(network);\n\n  function assertValidate(psbt: UtxoPsbt) {\n    psbt.data.inputs.forEach((input, i) => {\n      assert.ok(psbt.validateSignaturesOfInputHD(i, rootWalletKeys['user']));\n      if (getPsbtInputScriptType(input) !== 'p2shP2pk') {\n        assert.ok(psbt.validateSignaturesOfInputHD(i, rootWalletKeys['bitgo']));\n      }\n    });\n    assert.ok(psbt.validateSignaturesOfAllInputs());\n  }\n\n  describe(`Build, sign & send flow for ${coin}`, function () {\n    /**\n     * Skip adding nonWitnessUtxos to psbts\n     * ------------------------------------\n     * In the instance that we want to doing a bulk sweep, for network and client performance reasons we are substituting\n     * the nonWitnessUtxo for p2sh and p2shP2pk inputs with a witnessUtxo. We need the witnessUtxo so that we can half\n     * sign the transaction locally with the user key. When we send the half signed to BitGo, the PSBT will be properly\n     * populated such that the non-segwit inputs have the nonWitnessUtxo. This means when we send it to BitGo we should\n     * remove the witnessUtxo so that it just has the partialSig and redeemScript.\n     */\n    it(`success for ${coin}${skipNonWitnessUtxo ? ' without nonWitnessUtxo for p2sh' : ''}`, function () {\n      const parentPsbt = testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'unsigned', {\n        signers: {\n          signerName: 'user',\n          cosignerName: 'bitgo',\n        },\n      });\n\n      let psbt = skipNonWitnessUtxo ? clonePsbtWithoutNonWitnessUtxo(parentPsbt) : parentPsbt;\n      addXpubsToPsbt(psbt, rootWalletKeys);\n      psbt.setAllInputsMusig2NonceHD(rootWalletKeys['user']);\n\n      let psbtWithoutPrevTx = clonePsbtWithoutNonWitnessUtxo(psbt);\n      let hex = psbtWithoutPrevTx.toHex();\n\n      let psbtAtHsm = createPsbtFromHex(hex, network);\n      psbtAtHsm.setAllInputsMusig2NonceHD(rootWalletKeys['bitgo'], { deterministic: true });\n      let hexAtHsm = psbtAtHsm.toHex();\n\n      let psbtFromHsm = createPsbtFromHex(hexAtHsm, network);\n      deleteWitnessUtxoForNonSegwitInputs(psbtFromHsm);\n      psbt.combine(psbtFromHsm);\n\n      testutil.signAllPsbtInputs(psbt, inputs, rootWalletKeys, 'halfsigned', {\n        signers: {\n          signerName: 'user',\n          cosignerName: 'bitgo',\n        },\n        skipNonWitnessUtxo,\n      });\n\n      psbtWithoutPrevTx = clonePsbtWithoutNonWitnessUtxo(psbt);\n      hex = psbtWithoutPrevTx.toHex();\n\n      psbtAtHsm = createPsbtFromHex(hex, network);\n      withUnsafeNonSegwit(psbtAtHsm, () => {\n        testutil.signAllPsbtInputs(psbtAtHsm, inputs, rootWalletKeys, 'fullsigned', {\n          signers: {\n            signerName: 'user',\n            cosignerName: 'bitgo',\n          },\n          deterministic: true,\n        });\n      });\n      withUnsafeNonSegwit(psbtAtHsm, () => {\n        assertValidate(psbtAtHsm);\n      });\n      hexAtHsm = psbtAtHsm.toHex();\n\n      psbtFromHsm = createPsbtFromHex(hexAtHsm, network);\n      deleteWitnessUtxoForNonSegwitInputs(psbtFromHsm);\n\n      if (skipNonWitnessUtxo) {\n        psbt = parentPsbt;\n      }\n      psbt.combine(psbtFromHsm);\n\n      assertValidate(psbt);\n      assert.doesNotThrow(() => psbt.finalizeAllInputs().extractTransaction());\n    });\n  });\n}\n\nfunction runBuildPsbtWithSDK(network: Network, inputs: Input[], outputs: Output[]) {\n  const coin = getNetworkName(network);\n  it(`check that building a PSBT while skipping nonWitnessUtxo works - ${coin}`, async function () {\n    const psbtWithNonWitness = testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'unsigned', {\n      signers: {\n        signerName: 'user',\n        cosignerName: 'bitgo',\n      },\n    });\n    const psbtWithoutNonWitness = testutil.constructPsbt(inputs, outputs, network, rootWalletKeys, 'unsigned', {\n      signers: {\n        signerName: 'user',\n        cosignerName: 'bitgo',\n      },\n      skipNonWitnessUtxo: true,\n    });\n\n    const clonedPsbt = clonePsbtWithoutNonWitnessUtxo(psbtWithNonWitness);\n    assert.deepStrictEqual(psbtWithoutNonWitness.toHex(), clonedPsbt.toHex());\n  });\n}\n\ngetNetworkList()\n  .filter((v) => isMainnet(v) && v !== networks.bitcoinsv)\n  .forEach((network) => {\n    runExtractP2msOnlyHalfSignedTxTest(\n      network,\n      halfSignedInputs.filter((input) => isSupportedScriptType(network, input.scriptType)),\n      halfSignedOutputs.filter((output) => isSupportedScriptType(network, output.scriptType))\n    );\n\n    const supportedPsbtInputs = psbtInputs.filter((input) =>\n      isSupportedScriptType(network, input.scriptType === 'taprootKeyPathSpend' ? 'p2trMusig2' : input.scriptType)\n    );\n    const supportedPsbtOutputs = psbtOutputs.filter((output) => isSupportedScriptType(network, output.scriptType));\n    [false, true].forEach((skipNonWitnessUtxo) =>\n      runBuildSignSendFlowTest(network, supportedPsbtInputs, supportedPsbtOutputs, { skipNonWitnessUtxo })\n    );\n\n    runBuildPsbtWithSDK(network, supportedPsbtInputs, supportedPsbtOutputs);\n  });\n\ndescribe('isTransactionWithKeyPathSpendInput', function () {\n  describe('transaction input', function () {\n    it('empty inputs', function () {\n      const tx = testutil.constructTxnBuilder([], [], network, rootWalletKeys, 'unsigned').buildIncomplete();\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx), false);\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx.ins), false);\n    });\n\n    it('taprootKeyPath inputs successfully triggers', function () {\n      const psbt = testutil.constructPsbt(\n        [\n          { scriptType: 'taprootKeyPathSpend', value: BigInt(1e8) },\n          { scriptType: 'p2sh', value: BigInt(1e8) },\n        ],\n        [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }],\n        network,\n        rootWalletKeys,\n        'fullsigned'\n      );\n      assert(psbt.validateSignaturesOfAllInputs());\n      psbt.finalizeAllInputs();\n      const tx = psbt.extractTransaction() as UtxoTransaction<bigint>;\n\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx), true);\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx.ins), true);\n    });\n\n    it('no taprootKeyPath inputs successfully does not trigger', function () {\n      const psbt = testutil.constructPsbt(\n        [\n          { scriptType: 'p2trMusig2', value: BigInt(1e8) },\n          { scriptType: 'p2sh', value: BigInt(1e8) },\n        ],\n        [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }],\n        network,\n        rootWalletKeys,\n        'fullsigned'\n      );\n      assert(psbt.validateSignaturesOfAllInputs());\n      psbt.finalizeAllInputs();\n      const tx = psbt.extractTransaction();\n\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx), false);\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx.ins), false);\n    });\n\n    it('unsigned inputs successfully fail', function () {\n      const psbt = testutil.constructPsbt(\n        [\n          { scriptType: 'p2wsh', value: BigInt(1e8) },\n          { scriptType: 'p2sh', value: BigInt(1e8) },\n        ],\n        [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }],\n        network,\n        rootWalletKeys,\n        'unsigned'\n      );\n      const tx = psbt.getUnsignedTx();\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx), false);\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(tx.ins), false);\n    });\n  });\n\n  describe('psbt input', function () {\n    it('empty inputs', function () {\n      const psbt = testutil.constructPsbt([], [], network, rootWalletKeys, 'unsigned');\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(psbt), false);\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(psbt.data.inputs), false);\n    });\n\n    it('psbt with taprootKeyPathInputs successfully triggers', function () {\n      const psbt = testutil.constructPsbt(\n        [\n          { scriptType: 'taprootKeyPathSpend', value: BigInt(1e8) },\n          { scriptType: 'p2sh', value: BigInt(1e8) },\n        ],\n        [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }],\n        network,\n        rootWalletKeys,\n        'unsigned'\n      );\n\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(psbt), true);\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(psbt.data.inputs), true);\n    });\n\n    it('psbt without taprootKeyPathInputs successfully does not trigger', function () {\n      const psbt = testutil.constructPsbt(\n        [\n          { scriptType: 'p2wsh', value: BigInt(1e8) },\n          { scriptType: 'p2sh', value: BigInt(1e8) },\n        ],\n        [{ scriptType: 'p2sh', value: BigInt(2e8 - 10000) }],\n        network,\n        rootWalletKeys,\n        'halfsigned'\n      );\n\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(psbt), false);\n      assert.strictEqual(isTransactionWithKeyPathSpendInput(psbt.data.inputs), false);\n    });\n  });\n});\n\ndescribe('Parse PSBT', function () {\n  it('p2shP2pk parsing', function () {\n    const signer = rootWalletKeys['user'];\n    const psbt = createPsbtForNetwork({ network: networks.bitcoincash });\n    const unspent = mockReplayProtectionUnspent(networks.bitcoincash, BigInt(1e8), { key: signer });\n    const { redeemScript } = createOutputScriptP2shP2pk(signer.publicKey);\n    assert(redeemScript);\n    addReplayProtectionUnspentToPsbt(psbt, unspent, redeemScript);\n    addWalletOutputToPsbt(psbt, rootWalletKeys, getInternalChainCode('p2sh'), 0, BigInt(1e8 - 10000));\n    const input = psbt.data.inputs[0];\n    let parsed = parsePsbtInput(input);\n\n    assert.strictEqual(parsed.scriptType, 'p2shP2pk');\n    assert.strictEqual(parsed.signatures, undefined);\n    assert.strictEqual(parsed.publicKeys.length, 1);\n    assert.ok(parsed.publicKeys[0].length === 33);\n    assert.ok(parsed.pubScript.equals(redeemScript));\n\n    psbt.signAllInputs(signer);\n    assert.ok(psbt.validateSignaturesOfAllInputs());\n\n    parsed = parsePsbtInput(input);\n\n    assert.strictEqual(parsed.scriptType, 'p2shP2pk');\n    assert.strictEqual(parsed.signatures?.length, 1);\n    assert.strictEqual(parsed.publicKeys.length, 1);\n    assert.ok(parsed.publicKeys[0].length === 33);\n    assert.ok(parsed.pubScript.equals(redeemScript));\n\n    const sighash: number = parsed.signatures[0][parsed.signatures[0].length - 1];\n    assert.strictEqual(sighash, getDefaultSigHash(psbt.network));\n  });\n\n  it('fail to parse finalized psbt', function () {\n    const unspents = mockUnspents(\n      rootWalletKeys,\n      getScriptTypes2Of3().map((inputType) => inputType),\n      BigInt('10000000000000000'),\n      network\n    );\n    const txBuilderParams = {\n      signer: 'user',\n      cosigner: 'bitgo',\n      amountType: 'bigint',\n      outputType: 'p2sh',\n      signatureTarget: 'fullsigned',\n      network,\n      changeIndex: CHANGE_INDEX,\n      fee: FEE,\n    } as const;\n    const tx = constructTransactionUsingTxBuilder(unspents, rootWalletKeys, txBuilderParams);\n    const psbt = toWalletPsbt(tx, toBigInt(unspents), rootWalletKeys);\n    psbt.validateSignaturesOfAllInputs();\n    psbt.finalizeAllInputs();\n    psbt.data.inputs.forEach((input, i) => {\n      assert.throws(\n        () => parsePsbtInput(input),\n        (e: any) => e.message === 'Finalized PSBT parsing is not supported'\n      );\n    });\n  });\n\n  it('fail to parse input with more than one script type metadata', function () {\n    const unspents = mockUnspents(rootWalletKeys, ['p2tr'], BigInt('10000000000000000'), network);\n\n    const txBuilderParams = {\n      signer: 'user',\n      cosigner: 'bitgo',\n      amountType: 'bigint',\n      outputType: 'p2sh',\n      signatureTarget: 'halfsigned',\n      network,\n      changeIndex: CHANGE_INDEX,\n      fee: FEE,\n    } as const;\n\n    const txP2tr = constructTransactionUsingTxBuilder([unspents[0]], rootWalletKeys, txBuilderParams);\n    const psbtP2tr = toWalletPsbt(txP2tr, toBigInt([unspents[0]]), rootWalletKeys);\n\n    const walletKeys = rootWalletKeys.deriveForChainAndIndex(getExternalChainCode('p2sh'), 0);\n    const { redeemScript } = createOutputScript2of3(walletKeys.publicKeys, 'p2sh');\n    psbtP2tr.updateInput(0, { redeemScript });\n\n    assert.throws(\n      () => parsePsbtInput(psbtP2tr.data.inputs[0]),\n      (e: any) => e.message === 'Found both p2sh and taprootScriptPath PSBT metadata.'\n    );\n  });\n\n  it('fail to parse more than one tap leaf script per input', function () {\n    const unspents = mockUnspents(rootWalletKeys, ['p2tr'], BigInt('10000000000000000'), network);\n\n    const txBuilderParams = {\n      signer: 'user',\n      cosigner: 'bitgo',\n      amountType: 'bigint',\n      outputType: 'p2sh',\n      signatureTarget: 'halfsigned',\n      network,\n      changeIndex: CHANGE_INDEX,\n      fee: FEE,\n    } as const;\n\n    const txP2tr1 = constructTransactionUsingTxBuilder([unspents[0]], rootWalletKeys, txBuilderParams);\n    const psbtP2tr1 = toWalletPsbt(txP2tr1, toBigInt([unspents[0]]), rootWalletKeys);\n\n    const txBuilderParams2 = {\n      signer: 'user' as KeyName,\n      cosigner: 'backup' as KeyName,\n      amountType: 'bigint' as AmountType,\n      outputType: 'p2sh' as InputType,\n      signatureTarget: 'halfsigned' as SignatureTargetType,\n      network,\n      changeIndex: CHANGE_INDEX,\n      fee: FEE,\n    };\n\n    const txP2tr2 = constructTransactionUsingTxBuilder([unspents[0]], rootWalletKeys, txBuilderParams2);\n    const psbtP2tr2 = toWalletPsbt(txP2tr2, toBigInt([unspents[0]]), rootWalletKeys);\n\n    const txBuilderParams3 = {\n      signer: 'user',\n      cosigner: 'bitgo',\n      amountType: 'bigint',\n      outputType: 'p2sh',\n      signatureTarget: 'unsigned',\n      network,\n      changeIndex: CHANGE_INDEX,\n      fee: FEE,\n    } as const;\n    const txP2tr3 = constructTransactionUsingTxBuilder([unspents[0]], rootWalletKeys, txBuilderParams3);\n    const psbtP2tr3 = toWalletPsbt(txP2tr3, toBigInt([unspents[0]]), rootWalletKeys);\n    if (psbtP2tr1.data.inputs[0].tapLeafScript && psbtP2tr2.data.inputs[0].tapLeafScript) {\n      const tapLeafScripts = [psbtP2tr1.data.inputs[0].tapLeafScript[0], psbtP2tr2.data.inputs[0].tapLeafScript[0]];\n      psbtP2tr3.updateInput(0, { tapLeafScript: tapLeafScripts });\n\n      assert.throws(\n        () => parsePsbtInput(psbtP2tr3.data.inputs[0]),\n        (e: any) => e.message === 'Bitgo only supports a single tap leaf script per input.'\n      );\n    }\n  });\n});\n\ndescribe('isPsbt', function () {\n  function isPsbtForNetwork(n: Network) {\n    describe(`network: ${getNetworkName(n)}`, function () {\n      const psbt = createPsbtForNetwork({ network: n });\n\n      it('should return true for a valid PSBT', function () {\n        const psbtBuff = psbt.toBuffer();\n        assert.strictEqual(isPsbt(psbtBuff), true);\n        assert.strictEqual(isPsbt(psbtBuff.toString('hex')), true);\n      });\n\n      it('should return false for a transaction', function () {\n        assert.strictEqual(isPsbt(psbt.getUnsignedTx().toBuffer()), false);\n      });\n\n      it('should return false for a truncated magic word', function () {\n        const hex = psbt.toBuffer().slice(0, 3);\n        assert.strictEqual(isPsbt(hex), false);\n        assert.strictEqual(isPsbt(Buffer.from(hex)), false);\n      });\n\n      it('should return false for a valid PSBT with an invalid magic', function () {\n        const buffer = psbt.toBuffer();\n        buffer.writeUInt8(0x00, 1);\n        assert.strictEqual(isPsbt(psbt.getUnsignedTx().toBuffer()), false);\n      });\n\n      it('should return false for a valid PSBT with an invalid separator', function () {\n        const buffer = psbt.toBuffer();\n        buffer.writeUInt8(0xfe, 4);\n        assert.strictEqual(isPsbt(psbt.getUnsignedTx().toBuffer()), false);\n      });\n\n      it('should return false for a random buffer', function () {\n        const random = 'deadbeaf';\n        const buffer = Buffer.from(random, 'hex');\n        assert.strictEqual(isPsbt(random), false);\n        assert.strictEqual(isPsbt(buffer), false);\n      });\n\n      it('should return true if buffer is changed after the separator', function () {\n        const buffer = psbt.toBuffer();\n        buffer.writeUInt8(0x00, 5);\n        assert.strictEqual(isPsbt(buffer), true);\n      });\n    });\n  }\n\n  getNetworkList().forEach((n) => isPsbtForNetwork(n));\n});\n\ndescribe('Update incomplete psbt', function () {\n  function removeFromPsbt(\n    psbtHex: string,\n    network: Network,\n    remove: { input?: { index: number; fieldToRemove: string }; output?: { index: number; fieldToRemove: string } }\n  ): UtxoPsbt {\n    const utxoPsbt = createPsbtFromHex(psbtHex, network);\n    const psbt = createPsbtForNetwork({ network: utxoPsbt.network });\n    const txInputs = utxoPsbt.txInputs;\n    utxoPsbt.data.inputs.map((input, ii) => {\n      const { hash, index } = txInputs[ii];\n      if (remove.input && ii === remove.input.index) {\n        delete input[remove.input.fieldToRemove];\n      }\n      psbt.addInput({ ...input, hash, index });\n    });\n\n    const txOutputs = utxoPsbt.txOutputs;\n    utxoPsbt.data.outputs.map((output, ii) => {\n      if (remove.output && remove.output.index === ii) {\n        delete output[remove.output.fieldToRemove];\n      }\n      psbt.addOutput({ ...output, script: txOutputs[ii].script, value: txOutputs[ii].value });\n    });\n    return psbt;\n  }\n\n  function signAllInputs(psbt: UtxoPsbt, { assertValidSignaturesAndExtractable = true } = {}) {\n    psbt.data.inputs.forEach((input, inputIndex) => {\n      const parsedInput = parsePsbtInput(input);\n      if (parsedInput.scriptType === 'taprootKeyPathSpend') {\n        psbt.setInputMusig2NonceHD(inputIndex, rootWalletKeys[signer]);\n        psbt.setInputMusig2NonceHD(inputIndex, rootWalletKeys[cosigner]);\n      }\n\n      if (parsedInput.scriptType === 'p2shP2pk') {\n        psbt.signInput(inputIndex, replayProtectionKeyPair);\n      } else {\n        psbt.signInputHD(inputIndex, rootWalletKeys[signer]);\n        psbt.signInputHD(inputIndex, rootWalletKeys[cosigner]);\n      }\n    });\n\n    if (assertValidSignaturesAndExtractable) {\n      assert.ok(psbt.validateSignaturesOfAllInputs());\n      psbt.finalizeAllInputs();\n      const txExtracted = psbt.extractTransaction();\n      assert.ok(txExtracted);\n    }\n  }\n\n  let psbtHex: string;\n  let unspents: Unspent<bigint>[];\n  const signer = 'user';\n  const cosigner = 'bitgo';\n  const scriptTypes = [...scriptTypes2Of3, 'p2shP2pk'] as (ScriptType2Of3 | ScriptTypeP2shP2pk)[];\n  const outputValue = BigInt((2e8 * scriptTypes.length - 100) / 5);\n  const outputs = [\n    { chain: getExternalChainCode('p2sh'), index: 88, value: outputValue },\n    { chain: getExternalChainCode('p2shP2wsh'), index: 89, value: outputValue },\n    { chain: getExternalChainCode('p2wsh'), index: 90, value: outputValue },\n    { chain: getExternalChainCode('p2tr'), index: 91, value: outputValue },\n    { chain: getExternalChainCode('p2trMusig2'), index: 92, value: outputValue },\n  ];\n  before(function () {\n    unspents = mockUnspents(rootWalletKeys, scriptTypes, BigInt(2e8), network);\n    const psbt = constructPsbt(unspents, rootWalletKeys, signer, cosigner, outputs);\n    psbtHex = psbt.toHex();\n  });\n\n  it('can create a sign-able psbt from an unsigned transaction extracted from the psbt', function () {\n    if (true) {\n      return;\n    }\n    const psbtOrig = createPsbtFromHex(psbtHex, network);\n    const tx = psbtOrig.getUnsignedTx();\n    const psbt = createPsbtFromTransaction(\n      tx,\n      unspents.map((u) => toPrevOutput(u, network))\n    );\n    unspents.forEach((u, inputIndex) => {\n      if (isWalletUnspent(u)) {\n        updateWalletUnspentForPsbt(psbt, inputIndex, u, rootWalletKeys, signer, cosigner);\n      } else {\n        const { redeemScript } = createOutputScriptP2shP2pk(replayProtectionKeyPair.publicKey);\n        updateReplayProtectionUnspentToPsbt(psbt, inputIndex, u, redeemScript);\n      }\n    });\n\n    signAllInputs(psbt);\n  });\n\n  const componentsOnEachInputScriptType = {\n    p2sh: ['nonWitnessUtxo', 'redeemScript', 'bip32Derivation'],\n    p2shP2wsh: ['witnessUtxo', 'bip32Derivation', 'redeemScript', 'witnessScript'],\n    p2wsh: ['witnessUtxo', 'witnessScript', 'bip32Derivation'],\n    p2tr: ['witnessUtxo', 'tapLeafScript', 'tapBip32Derivation'],\n    p2trMusig2: ['witnessUtxo', 'tapBip32Derivation', 'tapInternalKey', 'tapMerkleRoot', 'unknownKeyVals'],\n    p2shP2pk: ['redeemScript', 'nonWitnessUtxo'],\n  };\n\n  const p2trComponents = ['tapTree', 'tapInternalKey', 'tapBip32Derivation'];\n  const componentsOnEachOutputScriptType = {\n    p2sh: ['bip32Derivation', 'redeemScript'],\n    p2shP2wsh: ['bip32Derivation', 'witnessScript', 'redeemScript'],\n    p2wsh: ['bip32Derivation', 'witnessScript'],\n    p2tr: p2trComponents,\n    p2trMusig2: p2trComponents,\n    p2shP2pk: [],\n  };\n  scriptTypes.forEach((scriptType, i) => {\n    componentsOnEachInputScriptType[scriptType].forEach((inputComponent) => {\n      it(`[${scriptType}] missing ${inputComponent} on input should succeed in fully signing unsigned psbt after update`, function () {\n        const psbt = removeFromPsbt(psbtHex, network, { input: { index: i, fieldToRemove: inputComponent } });\n        const unspent = unspents[i];\n        if (isWalletUnspent(unspent)) {\n          updateWalletUnspentForPsbt(psbt, i, unspent, rootWalletKeys, signer, cosigner);\n        } else {\n          const { redeemScript } = createOutputScriptP2shP2pk(replayProtectionKeyPair.publicKey);\n          assert.ok(redeemScript);\n          updateReplayProtectionUnspentToPsbt(psbt, i, unspent, redeemScript);\n        }\n        signAllInputs(psbt);\n      });\n    });\n\n    componentsOnEachOutputScriptType[scriptType].forEach((outputComponent) => {\n      it(`[${scriptType}] missing ${outputComponent} on output should produce same hex as fully hydrated after update`, function () {\n        const psbt = removeFromPsbt(psbtHex, network, { output: { index: i, fieldToRemove: outputComponent } });\n        updateWalletOutputForPsbt(psbt, rootWalletKeys, i, outputs[i].chain, outputs[i].index);\n        assert.strictEqual(psbt.toHex(), psbtHex);\n      });\n    });\n  });\n});\n\ndescribe('Psbt from transaction using wallet unspents', function () {\n  function runTestSignUnspents<TNumber extends number | bigint>({\n    inputScriptTypes,\n    outputScriptType,\n    signer,\n    cosigner,\n    amountType,\n    testOutputAmount,\n    signatureTarget,\n  }: {\n    inputScriptTypes: InputType[];\n    outputScriptType: outputScripts.ScriptType2Of3;\n    signer: KeyName;\n    cosigner: KeyName;\n    amountType: 'number' | 'bigint';\n    testOutputAmount: TNumber;\n    signatureTarget: SignatureTargetType;\n  }) {\n    it(`can be signed [inputs=${inputScriptTypes} signer=${signer} cosigner=${cosigner} amountType=${amountType} signatureTarget=${signatureTarget}]`, function () {\n      const unspents = mockUnspents(rootWalletKeys, inputScriptTypes, testOutputAmount, network);\n      // const txBuilderParams = { network, changeIndex: CHANGE_INDEX, fee: FEE };\n      const txBuilderParams = {\n        signer,\n        cosigner,\n        amountType,\n        outputType: outputScriptType,\n        signatureTarget: signatureTarget,\n        network,\n        changeIndex: CHANGE_INDEX,\n        fee: FEE,\n      };\n      const tx = constructTransactionUsingTxBuilder(unspents, rootWalletKeys, txBuilderParams);\n\n      const unspentBigInt = toBigInt(unspents);\n\n      const psbt = toWalletPsbt(tx, unspentBigInt, rootWalletKeys);\n\n      validatePsbtParsing(tx, psbt, unspentBigInt, signatureTarget);\n\n      // Check that the correct unspent corresponds to the input\n      unspentBigInt.forEach((unspent, inputIndex) => {\n        const otherUnspent = inputIndex === 0 ? unspentBigInt[1] : unspentBigInt[0];\n        assert.strictEqual(psbtIncludesUnspentAtIndex(psbt, inputIndex, unspent.id), true);\n        assert.strictEqual(psbtIncludesUnspentAtIndex(psbt, inputIndex, otherUnspent.id), false);\n        updateWalletUnspentForPsbt(psbt, inputIndex, unspent, rootWalletKeys, signer, cosigner);\n      });\n\n      if (signatureTarget !== 'fullsigned') {\n        // Now signing to make it fully signed psbt.\n        // So it will be easy to verify its validity with another similar tx to be built with tx builder.\n        signPsbt(psbt, unspentBigInt, rootWalletKeys, signer, cosigner, signatureTarget);\n      }\n      assert.deepStrictEqual(psbt.validateSignaturesOfAllInputs(), true);\n      psbt.finalizeAllInputs();\n      const txFromPsbt = psbt.extractTransaction();\n\n      const txBuilderParams2 = {\n        signer,\n        cosigner,\n        amountType,\n        outputType: outputScriptType,\n        signatureTarget: 'fullsigned' as SignatureTargetType,\n        network,\n        changeIndex: CHANGE_INDEX,\n        fee: FEE,\n      };\n\n      // New legacy tx resembles the signed psbt.\n      const txFromTxBuilder = constructTransactionUsingTxBuilder(unspents, rootWalletKeys, txBuilderParams2);\n\n      assert.deepStrictEqual(txFromPsbt.getHash(), txFromTxBuilder.getHash());\n    });\n  }\n\n  function getInputScripts(): InputType[][] {\n    return getScriptTypes2Of3().flatMap((t) => {\n      return getScriptTypes2Of3().flatMap((lastType) => {\n        return [[t, t, lastType]];\n      });\n    });\n  }\n\n  function getSignerPairs(containsTaprootInput: boolean): [signer: KeyName, cosigner: KeyName][] {\n    const signaturePairs = [['user', 'bitgo'] as [signer: KeyName, cosigner: KeyName]];\n    if (containsTaprootInput) {\n      signaturePairs.push(['user', 'backup'] as [signer: KeyName, cosigner: KeyName]);\n    }\n    return signaturePairs;\n  }\n\n  (['unsigned', 'halfsigned', 'fullsigned'] as SignatureTargetType[]).forEach((signatureTarget) => {\n    getInputScripts().forEach((inputScriptTypes) => {\n      getSignerPairs(inputScriptTypes.includes('p2tr')).forEach(([signer, cosigner]) => {\n        runTestSignUnspents({\n          inputScriptTypes,\n          outputScriptType: 'p2sh',\n          signer,\n          cosigner,\n          amountType: 'number',\n          testOutputAmount: defaultTestOutputAmount,\n          signatureTarget,\n        });\n        runTestSignUnspents<bigint>({\n          inputScriptTypes,\n          outputScriptType: 'p2sh',\n          signer,\n          cosigner,\n          amountType: 'bigint',\n          testOutputAmount: BigInt('10000000000000000'),\n          signatureTarget,\n        });\n      });\n    });\n  });\n});\n\nfunction testUtxoPsbt(coinNetwork: Network) {\n  describe(`Testing UtxoPsbt (de)serialization for ${getNetworkName(coinNetwork)} network`, function () {\n    let psbt: UtxoPsbt;\n    let psbtHex: string;\n    let unspents: (WalletUnspent<bigint> | Unspent<bigint>)[];\n    before(async function () {\n      unspents = mockUnspents(rootWalletKeys, ['p2sh'], BigInt('10000000000000'), coinNetwork);\n      const txBuilderParams = {\n        signer: 'user',\n        cosigner: 'bitgo',\n        amountType: 'bigint',\n        outputType: 'p2sh',\n        signatureTarget: 'fullsigned',\n        network: coinNetwork,\n        changeIndex: CHANGE_INDEX,\n        fee: FEE,\n      } as const;\n      const tx = constructTransactionUsingTxBuilder(unspents, rootWalletKeys, txBuilderParams);\n      psbt = toWalletPsbt(tx, toBigInt(unspents), rootWalletKeys);\n      if (coinNetwork === networks.zcash) {\n        (psbt as ZcashPsbt).setDefaultsForVersion(network, 450);\n      }\n      psbtHex = psbt.toHex();\n    });\n\n    it('should be able to clone psbt', async function () {\n      const clone = psbt.clone();\n      assert(clone instanceof psbt.constructor, `Expected clone to be instance of ${psbt.constructor.name}`);\n      assert.deepStrictEqual(clone.toBuffer(), psbt.toBuffer());\n      assert.deepStrictEqual(clone.clone().toBuffer(), psbt.toBuffer());\n      assert.strictEqual(clone.network, psbt.network);\n      assert.strictEqual(clone.clone().network, psbt.network);\n    });\n\n    it('should be able to round-trip', async function () {\n      assert.deepStrictEqual(createPsbtFromHex(psbtHex, coinNetwork, false).toBuffer(), psbt.toBuffer());\n    });\n\n    it('should be able to get transaction info from psbt', function () {\n      const txInfo = getTransactionAmountsFromPsbt(psbt);\n      assert.strictEqual(txInfo.fee, FEE);\n      assert.strictEqual(txInfo.inputCount, unspents.length);\n      assert.strictEqual(txInfo.inputAmount, BigInt('10000000000000') * BigInt(unspents.length));\n      assert.strictEqual(txInfo.outputAmount, BigInt('10000000000000') * BigInt(unspents.length) - FEE);\n      assert.strictEqual(txInfo.outputCount, psbt.data.outputs.length);\n    });\n\n    function deserializeBip32PathsCorrectly(bip32PathsAbsolute: boolean): void {\n      function checkDerivationPrefix(bip32Derivation: { path: string }): void {\n        const path = bip32Derivation.path.split('/');\n        const prefix = bip32PathsAbsolute ? 'm' : '0';\n        assert(path[0] === prefix);\n      }\n      it(`should deserialize PSBT bip32Derivations with paths ${\n        bip32PathsAbsolute ? '' : 'not '\n      } absolute`, async function () {\n        const deserializedPsbt = createPsbtFromHex(psbtHex, coinNetwork, bip32PathsAbsolute);\n        assert(deserializedPsbt);\n        deserializedPsbt.data.inputs.forEach((input) => {\n          input?.bip32Derivation?.forEach((derivation) => checkDerivationPrefix(derivation));\n          input?.tapBip32Derivation?.forEach((derivation) => checkDerivationPrefix(derivation));\n        });\n      });\n    }\n\n    [true, false].forEach((bip32PathsAbsolute) => deserializeBip32PathsCorrectly(bip32PathsAbsolute));\n  });\n}\n\n[networks.bitcoin, networks.zcash, networks.dash, networks.dogecoin, networks.litecoin].forEach((coinNetwork) =>\n  testUtxoPsbt(coinNetwork)\n);\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!