PHP WebShell

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

Просмотр файла: WalletUnspent.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 testutil_1 = require("../../../src/testutil");
const transaction_util_1 = require("../../transaction_util");
const mock_1 = require("../../../src/testutil/mock");
const CHANGE_INDEX = 100;
const FEE = BigInt(100);
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');
}
describe('WalletUnspent', function () {
    const network = src_1.networks.bitcoin;
    const walletKeys = (0, testutil_1.getDefaultWalletKeys)();
    const hash = Buffer.alloc(32).fill(0xff);
    hash[0] = 0; // show endianness
    const input = { hash, index: 0 };
    const expectedOutPoint = {
        txid: 'ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00',
        vout: 0,
    };
    it('parses and formats txid', function () {
        assert.deepStrictEqual((0, bitgo_1.getOutputIdForInput)(input), expectedOutPoint);
        assert.deepStrictEqual((0, bitgo_1.formatOutputId)(expectedOutPoint), 'ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00:0');
        assert.deepStrictEqual((0, bitgo_1.parseOutputId)((0, bitgo_1.formatOutputId)(expectedOutPoint)), expectedOutPoint);
    });
    it('identifies wallet unspents', function () {
        const unspent = {
            id: (0, bitgo_1.formatOutputId)(expectedOutPoint),
            address: (0, bitgo_1.getWalletAddress)(walletKeys, 0, 0, network),
            value: 1e8,
        };
        assert.strictEqual((0, bitgo_1.isWalletUnspent)(unspent), false);
        assert.strictEqual((0, bitgo_1.isWalletUnspent)({ ...unspent, chain: 0, index: 0 }), true);
    });
    function constructAndSignTransactionUsingPsbt(unspents, signer, cosigner, outputType) {
        const psbt = (0, bitgo_1.createPsbtForNetwork)({ network });
        const total = BigInt((0, bitgo_1.unspentSum)(unspents, 'bigint'));
        (0, bitgo_1.addWalletOutputToPsbt)(psbt, walletKeys, (0, bitgo_1.getInternalChainCode)(outputType), CHANGE_INDEX, total - FEE);
        unspents.forEach((u) => {
            if ((0, bitgo_1.isWalletUnspent)(u)) {
                (0, bitgo_1.addWalletUnspentToPsbt)(psbt, u, walletKeys, signer, cosigner);
            }
            else {
                throw new Error(`invalid unspent`);
            }
        });
        // TODO: Test rederiving scripts from PSBT and keys only
        psbt.signAllInputsHD(walletKeys[signer]);
        psbt.signAllInputsHD(walletKeys[cosigner]);
        assert(psbt.validateSignaturesOfAllInputs());
        psbt.finalizeAllInputs();
        // extract transaction has a return type of Transaction instead of UtxoTransaction
        const tx = psbt.extractTransaction();
        const psbt2 = (0, bitgo_1.createPsbtFromTransaction)(tx, unspents.map((u) => ({ ...(0, bitgo_1.toPrevOutput)(u, network), prevTx: u.prevTx })));
        assert(psbt2.validateSignaturesOfAllInputs());
        return tx;
    }
    function constructAndSignTransactionUsingTransactionBuilder(unspents, signer, cosigner, amountType = 'number', outputType) {
        const txb = (0, bitgo_1.createTransactionBuilderForNetwork)(network);
        const total = BigInt((0, bitgo_1.unspentSum)(unspents, amountType));
        // Kinda weird, treating entire value as change, but tests the relevant paths
        txb.addOutput((0, bitgo_1.getWalletAddress)(walletKeys, (0, bitgo_1.getInternalChainCode)(outputType), CHANGE_INDEX, network), (0, bitgo_1.toTNumber)(total - FEE, amountType));
        unspents.forEach((u) => {
            (0, bitgo_1.addToTransactionBuilder)(txb, u);
        });
        unspents.forEach((u, i) => {
            if ((0, mock_1.isReplayProtectionUnspent)(u, network)) {
                (0, bitgo_1.signInputP2shP2pk)(txb, i, mock_1.replayProtectionKeyPair);
            }
        });
        [
            bitgo_1.WalletUnspentSigner.from(walletKeys, walletKeys[signer], walletKeys[cosigner]),
            bitgo_1.WalletUnspentSigner.from(walletKeys, walletKeys[cosigner], walletKeys[signer]),
        ].forEach((walletSigner, nSignature) => {
            unspents.forEach((u, i) => {
                if ((0, bitgo_1.isWalletUnspent)(u)) {
                    (0, bitgo_1.signInputWithUnspent)(txb, i, u, walletSigner);
                }
                else if ((0, mock_1.isReplayProtectionUnspent)(u, network)) {
                    return;
                }
                else {
                    throw new Error(`unexpected unspent ${u.id}`);
                }
            });
            const tx = nSignature === 0 ? txb.buildIncomplete() : txb.build();
            // Verify each signature for the unspent
            unspents.forEach((u, i) => {
                if ((0, mock_1.isReplayProtectionUnspent)(u, network)) {
                    // signature verification not implemented for replay protection unspents
                    return;
                }
                assert.deepStrictEqual((0, bitgo_1.verifySignatureWithUnspent)(tx, i, unspents, walletKeys), walletKeys.triple.map((k) => k === walletKeys[signer] || (nSignature === 1 && k === walletKeys[cosigner])));
            });
        });
        return txb.build();
    }
    function validateLockTimeAndSequence(transaction) {
        // locktime should default to 0 and sequence to 0xffffffff for all inputs
        assert.deepStrictEqual(transaction.locktime, 0);
        const inputs = transaction.ins;
        for (const input of inputs) {
            assert.deepStrictEqual(input.sequence, 0xffffffff);
        }
    }
    function runTestSignUnspents({ inputScriptTypes, outputScriptType, signer, cosigner, amountType, testOutputAmount, }) {
        it(`can be signed [inputs=${inputScriptTypes} signer=${signer} cosigner=${cosigner} amountType=${amountType}]`, function () {
            const unspents = inputScriptTypes.map((t, i) => {
                if (bitgo_1.outputScripts.isScriptType2Of3(t)) {
                    return (0, mock_1.mockWalletUnspent)(network, testOutputAmount, {
                        keys: walletKeys,
                        chain: (0, bitgo_1.getExternalChainCode)(t),
                        vout: i,
                    });
                }
                if (t === 'p2shP2pk') {
                    return (0, mock_1.mockReplayProtectionUnspent)(network, (0, bitgo_1.toTNumber)(1000, amountType));
                }
                throw new Error(`invalid input type ${t}`);
            });
            const txbTransaction = constructAndSignTransactionUsingTransactionBuilder(unspents, signer, cosigner, amountType, outputScriptType);
            validateLockTimeAndSequence(txbTransaction);
            if (amountType === 'bigint') {
                if (inputScriptTypes.includes('p2shP2pk')) {
                    // FIMXE(BG-47824): add p2shP2pk support for Psbt
                    return;
                }
                const psbtTransaction = constructAndSignTransactionUsingPsbt(unspents, signer, cosigner, outputScriptType);
                assert.deepStrictEqual(txbTransaction.toBuffer(), psbtTransaction.toBuffer());
                validateLockTimeAndSequence(psbtTransaction);
            }
        });
    }
    function getInputScripts() {
        return getScriptTypes2Of3().flatMap((t) => [
            [t, t],
            [t, t, 'p2shP2pk'],
        ]);
    }
    function getSignerPairs() {
        const keyNames = ['user', 'backup', 'bitgo'];
        return keyNames.flatMap((signer) => keyNames.flatMap((cosigner) => (signer === cosigner ? [] : [[signer, cosigner]])));
    }
    getInputScripts().forEach((inputScriptTypes) => {
        getSignerPairs().forEach(([signer, cosigner]) => {
            runTestSignUnspents({
                inputScriptTypes,
                outputScriptType: 'p2sh',
                signer,
                cosigner,
                amountType: 'number',
                testOutputAmount: transaction_util_1.defaultTestOutputAmount,
            });
            runTestSignUnspents({
                inputScriptTypes,
                outputScriptType: 'p2sh',
                signer,
                cosigner,
                amountType: 'bigint',
                testOutputAmount: BigInt('10000000000000000'),
            });
        });
    });
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"WalletUnspent.js","sourceRoot":"","sources":["../../../../test/bitgo/wallet/WalletUnspent.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport { Transaction, networks } from '../../../src';\nimport {\n  isWalletUnspent,\n  formatOutputId,\n  getOutputIdForInput,\n  parseOutputId,\n  TxOutPoint,\n  Unspent,\n  createTransactionBuilderForNetwork,\n  getInternalChainCode,\n  getExternalChainCode,\n  addToTransactionBuilder,\n  signInputWithUnspent,\n  WalletUnspentSigner,\n  outputScripts,\n  unspentSum,\n  getWalletAddress,\n  verifySignatureWithUnspent,\n  toTNumber,\n  UtxoTransaction,\n  createPsbtForNetwork,\n  createPsbtFromTransaction,\n  addWalletUnspentToPsbt,\n  addWalletOutputToPsbt,\n  toPrevOutput,\n  KeyName,\n  signInputP2shP2pk,\n} from '../../../src/bitgo';\n\nimport { getDefaultWalletKeys } from '../../../src/testutil';\nimport { defaultTestOutputAmount } from '../../transaction_util';\nimport {\n  mockWalletUnspent,\n  isReplayProtectionUnspent,\n  mockReplayProtectionUnspent,\n  replayProtectionKeyPair,\n} from '../../../src/testutil/mock';\n\nconst CHANGE_INDEX = 100;\nconst FEE = BigInt(100);\n\ntype InputType = outputScripts.ScriptType2Of3 | 'p2shP2pk';\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\ndescribe('WalletUnspent', function () {\n  const network = networks.bitcoin;\n  const walletKeys = getDefaultWalletKeys();\n  const hash = Buffer.alloc(32).fill(0xff);\n  hash[0] = 0; // show endianness\n  const input = { hash, index: 0 };\n  const expectedOutPoint: TxOutPoint = {\n    txid: 'ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00',\n    vout: 0,\n  };\n\n  it('parses and formats txid', function () {\n    assert.deepStrictEqual(getOutputIdForInput(input), expectedOutPoint);\n    assert.deepStrictEqual(\n      formatOutputId(expectedOutPoint),\n      'ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00:0'\n    );\n    assert.deepStrictEqual(parseOutputId(formatOutputId(expectedOutPoint)), expectedOutPoint);\n  });\n\n  it('identifies wallet unspents', function () {\n    const unspent: Unspent = {\n      id: formatOutputId(expectedOutPoint),\n      address: getWalletAddress(walletKeys, 0, 0, network),\n      value: 1e8,\n    };\n    assert.strictEqual(isWalletUnspent(unspent), false);\n    assert.strictEqual(isWalletUnspent({ ...unspent, chain: 0, index: 0 } as Unspent), true);\n  });\n\n  function constructAndSignTransactionUsingPsbt(\n    unspents: (Unspent<bigint> & { prevTx?: Buffer })[],\n    signer: KeyName,\n    cosigner: KeyName,\n    outputType: outputScripts.ScriptType2Of3\n  ): Transaction<bigint> {\n    const psbt = createPsbtForNetwork({ network });\n    const total = BigInt(unspentSum<bigint>(unspents, 'bigint'));\n    addWalletOutputToPsbt(psbt, walletKeys, getInternalChainCode(outputType), CHANGE_INDEX, total - FEE);\n\n    unspents.forEach((u) => {\n      if (isWalletUnspent(u)) {\n        addWalletUnspentToPsbt(psbt, u, walletKeys, signer, cosigner);\n      } else {\n        throw new Error(`invalid unspent`);\n      }\n    });\n\n    // TODO: Test rederiving scripts from PSBT and keys only\n    psbt.signAllInputsHD(walletKeys[signer]);\n    psbt.signAllInputsHD(walletKeys[cosigner]);\n    assert(psbt.validateSignaturesOfAllInputs());\n    psbt.finalizeAllInputs();\n    // extract transaction has a return type of Transaction instead of UtxoTransaction\n    const tx = psbt.extractTransaction() as UtxoTransaction<bigint>;\n\n    const psbt2 = createPsbtFromTransaction(\n      tx,\n      unspents.map((u) => ({ ...toPrevOutput<bigint>(u, network), prevTx: u.prevTx }))\n    );\n    assert(psbt2.validateSignaturesOfAllInputs());\n    return tx;\n  }\n\n  function constructAndSignTransactionUsingTransactionBuilder<TNumber extends number | bigint>(\n    unspents: Unspent<TNumber>[],\n    signer: string,\n    cosigner: string,\n    amountType: 'number' | 'bigint' = 'number',\n    outputType: outputScripts.ScriptType2Of3\n  ): UtxoTransaction<TNumber> {\n    const txb = createTransactionBuilderForNetwork<TNumber>(network);\n    const total = BigInt(unspentSum<TNumber>(unspents, amountType));\n    // Kinda weird, treating entire value as change, but tests the relevant paths\n    txb.addOutput(\n      getWalletAddress(walletKeys, getInternalChainCode(outputType), CHANGE_INDEX, network),\n      toTNumber<TNumber>(total - FEE, amountType)\n    );\n    unspents.forEach((u) => {\n      addToTransactionBuilder(txb, u);\n    });\n    unspents.forEach((u, i) => {\n      if (isReplayProtectionUnspent(u, network)) {\n        signInputP2shP2pk(txb, i, replayProtectionKeyPair);\n      }\n    });\n\n    [\n      WalletUnspentSigner.from(walletKeys, walletKeys[signer], walletKeys[cosigner]),\n      WalletUnspentSigner.from(walletKeys, walletKeys[cosigner], walletKeys[signer]),\n    ].forEach((walletSigner, nSignature) => {\n      unspents.forEach((u, i) => {\n        if (isWalletUnspent(u)) {\n          signInputWithUnspent(txb, i, u, walletSigner);\n        } else if (isReplayProtectionUnspent(u, network)) {\n          return;\n        } else {\n          throw new Error(`unexpected unspent ${u.id}`);\n        }\n      });\n\n      const tx = nSignature === 0 ? txb.buildIncomplete() : txb.build();\n      // Verify each signature for the unspent\n      unspents.forEach((u, i) => {\n        if (isReplayProtectionUnspent(u, network)) {\n          // signature verification not implemented for replay protection unspents\n          return;\n        }\n        assert.deepStrictEqual(\n          verifySignatureWithUnspent(tx, i, unspents, walletKeys),\n          walletKeys.triple.map((k) => k === walletKeys[signer] || (nSignature === 1 && k === walletKeys[cosigner]))\n        );\n      });\n    });\n\n    return txb.build();\n  }\n\n  function validateLockTimeAndSequence<TNumber extends number | bigint>(\n    transaction: UtxoTransaction<TNumber> | Transaction<bigint>\n  ) {\n    // locktime should default to 0 and sequence to 0xffffffff for all inputs\n    assert.deepStrictEqual(transaction.locktime, 0);\n    const inputs = transaction.ins;\n    for (const input of inputs) {\n      assert.deepStrictEqual(input.sequence, 0xffffffff);\n    }\n  }\n\n  function runTestSignUnspents<TNumber extends number | bigint>({\n    inputScriptTypes,\n    outputScriptType,\n    signer,\n    cosigner,\n    amountType,\n    testOutputAmount,\n  }: {\n    inputScriptTypes: InputType[];\n    outputScriptType: outputScripts.ScriptType2Of3;\n    signer: KeyName;\n    cosigner: KeyName;\n    amountType: 'number' | 'bigint';\n    testOutputAmount: TNumber;\n  }) {\n    it(`can be signed [inputs=${inputScriptTypes} signer=${signer} cosigner=${cosigner} amountType=${amountType}]`, function () {\n      const unspents = inputScriptTypes.map((t, i): Unspent<TNumber> => {\n        if (outputScripts.isScriptType2Of3(t)) {\n          return mockWalletUnspent(network, testOutputAmount, {\n            keys: walletKeys,\n            chain: getExternalChainCode(t),\n            vout: i,\n          });\n        }\n\n        if (t === 'p2shP2pk') {\n          return mockReplayProtectionUnspent(network, toTNumber(1_000, amountType));\n        }\n\n        throw new Error(`invalid input type ${t}`);\n      });\n\n      const txbTransaction = constructAndSignTransactionUsingTransactionBuilder(\n        unspents,\n        signer,\n        cosigner,\n        amountType,\n        outputScriptType\n      );\n      validateLockTimeAndSequence(txbTransaction);\n      if (amountType === 'bigint') {\n        if (inputScriptTypes.includes('p2shP2pk')) {\n          // FIMXE(BG-47824): add p2shP2pk support for Psbt\n          return;\n        }\n        const psbtTransaction = constructAndSignTransactionUsingPsbt(\n          unspents as Unspent<bigint>[],\n          signer,\n          cosigner,\n          outputScriptType\n        );\n        assert.deepStrictEqual(txbTransaction.toBuffer(), psbtTransaction.toBuffer());\n        validateLockTimeAndSequence(psbtTransaction);\n      }\n    });\n  }\n\n  function getInputScripts(): InputType[][] {\n    return getScriptTypes2Of3().flatMap((t) => [\n      [t, t],\n      [t, t, 'p2shP2pk'],\n    ]);\n  }\n\n  function getSignerPairs(): [signer: KeyName, cosigner: KeyName][] {\n    const keyNames: KeyName[] = ['user', 'backup', 'bitgo'];\n    return keyNames.flatMap((signer) =>\n      keyNames.flatMap((cosigner): [KeyName, KeyName][] => (signer === cosigner ? [] : [[signer, cosigner]]))\n    );\n  }\n\n  getInputScripts().forEach((inputScriptTypes) => {\n    getSignerPairs().forEach(([signer, cosigner]) => {\n      runTestSignUnspents({\n        inputScriptTypes,\n        outputScriptType: 'p2sh',\n        signer,\n        cosigner,\n        amountType: 'number',\n        testOutputAmount: defaultTestOutputAmount,\n      });\n      runTestSignUnspents<bigint>({\n        inputScriptTypes,\n        outputScriptType: 'p2sh',\n        signer,\n        cosigner,\n        amountType: 'bigint',\n        testOutputAmount: BigInt('10000000000000000'),\n      });\n    });\n  });\n});\n"]}

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


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