PHP WebShell
Текущая директория: /opt/BitGoJS/modules/utxo-lib/dist/test/bitgo
Просмотр файла: signature.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const assert = require("assert");
const src_1 = require("../../src");
const testutil_1 = require("../../src/testutil");
const networks_1 = require("../../src/networks");
const outputScripts_1 = require("../../src/bitgo/outputScripts");
const bitgo_1 = require("../../src/bitgo");
const fixtureUtil = require("../fixture.util");
const fixtures_1 = require("../integration_local_rpc/generate/fixtures");
const transaction_util_1 = require("../transaction_util");
const signatureModify_1 = require("./signatureModify");
const normalize_1 = require("../testutil/normalize");
function getScriptTypes2Of3() {
// FIXME(BG-66941): p2trMusig2 signing does not work in this test suite yet
// because the test suite is written with TransactionBuilder
return outputScripts_1.scriptTypes2Of3.filter((scriptType) => scriptType !== 'p2trMusig2');
}
function keyName(k) {
return (0, testutil_1.getKeyName)(fixtures_1.fixtureKeys, k);
}
async function readFixture(network, scriptType, name, defaultValue) {
return await fixtureUtil.readFixture(`${__dirname}/fixtures/signature/${(0, networks_1.getNetworkName)(network)}/${scriptType}/${name}.json`, defaultValue);
}
function runTestCheckScriptStructure(network, scriptType, signer1, signer2, amountType = 'number') {
it(`has expected script structure [${(0, networks_1.getNetworkName)(network)} ${scriptType} ` +
`${keyName(signer1)} ${signer2 ? keyName(signer2) : ''} ${amountType}]`, async function () {
let tx;
if (scriptType === 'p2shP2pk') {
tx = (0, transaction_util_1.getFullSignedTransactionP2shP2pk)(fixtures_1.fixtureKeys, signer1, network, { amountType });
}
else {
if (!signer2) {
throw new Error(`must set cosigner`);
}
tx = (0, transaction_util_1.getFullSignedTransaction2Of3)(fixtures_1.fixtureKeys, signer1, signer2, scriptType, network, { amountType });
}
const { script, witness } = tx.ins[0];
const scriptDecompiled = src_1.script.decompile(script);
if (!scriptDecompiled) {
throw new Error();
}
const scriptASM = src_1.script.toASM(script).split(' ');
const classifyInput = src_1.classify.input(script);
const classifyWitness = src_1.classify.witness(witness);
let pubScript;
let classifyPubScript;
let pubScriptASM;
let tapscript;
let tapscriptASM;
let classifyTapscript;
if (classifyInput === 'scripthash' || classifyWitness === 'witnessscripthash') {
if (witness.length) {
pubScript = witness[witness.length - 1];
}
else {
pubScript = scriptDecompiled[scriptDecompiled.length - 1];
}
classifyPubScript = src_1.classify.output(pubScript);
pubScriptASM = src_1.script.toASM(pubScript).split(' ');
}
else if (classifyWitness === 'taproot') {
tapscript = witness[witness.length - 2];
classifyTapscript = src_1.classify.output(tapscript);
tapscriptASM = src_1.script.toASM(tapscript).split(' ');
}
const structure = {
publicKeys: fixtures_1.fixtureKeys.map((k) => k.publicKey.toString('hex')),
script: script?.toString('hex'),
witness: witness?.map((w) => w.toString('hex')),
scriptASM,
pubScriptASM,
tapscriptASM,
classifyInput,
classifyWitness,
classifyPubScript,
classifyTapscript,
};
const fixtureName = ['structure', keyName(signer1), signer2 ? keyName(signer2) : 'none'].join('-');
fixtureUtil.assertEqualJSON(structure, await readFixture(network, scriptType, fixtureName, structure));
});
}
function runTestParseScript(network, scriptType, k1, k2, amountType = 'number') {
async function testParseSignedInputs(tx, name, expectedScriptType, { expectedPlaceholderSignatures }) {
const parsed = (0, bitgo_1.parseSignatureScript)(tx.ins[0]);
assert.strictEqual(parsed.scriptType, expectedScriptType === 'p2tr' ? 'taprootScriptPathSpend' : expectedScriptType);
const parsed2Of3 = { ...parsed, scriptType: expectedScriptType };
fixtureUtil.assertEqualJSON(parsed2Of3, await readFixture(network, scriptType, ['parsed', keyName(k1), keyName(k2), name].join('-'), parsed2Of3));
if (!parsed.scriptType) {
return;
}
switch (parsed.scriptType) {
case 'p2shP2pk':
// we don't parse the signature for this script type
break;
case 'p2sh':
case 'p2shP2wsh':
case 'p2wsh':
case 'taprootScriptPathSpend':
assert.strictEqual(parsed.signatures.filter((s) => (0, bitgo_1.isPlaceholderSignature)(s)).length, expectedPlaceholderSignatures);
break;
default:
throw new Error(`unexpected scriptType ${parsed.scriptType}`);
}
}
if (scriptType !== 'p2shP2pk') {
it(`parses half-signed inputs [${(0, networks_1.getNetworkName)(network)} ${scriptType} ${amountType}]`, async function () {
await testParseSignedInputs((0, transaction_util_1.getHalfSignedTransaction2Of3)(fixtures_1.fixtureKeys, k1, k2, scriptType, network, { amountType }), 'halfSigned', scriptType, { expectedPlaceholderSignatures: scriptType === 'p2tr' ? 1 : 2 });
});
}
it(`parses full-signed inputs [${(0, networks_1.getNetworkName)(network)} ${scriptType} ${amountType}]`, async function () {
if (scriptType === 'p2shP2pk') {
await testParseSignedInputs((0, transaction_util_1.getFullSignedTransactionP2shP2pk)(fixtures_1.fixtureKeys, k1, network, { amountType }), 'fullSigned', scriptType, { expectedPlaceholderSignatures: 0 });
}
else {
await testParseSignedInputs((0, transaction_util_1.getFullSignedTransaction2Of3)(fixtures_1.fixtureKeys, k1, k2, scriptType, network, { amountType }), 'fullSigned', scriptType, { expectedPlaceholderSignatures: 0 });
}
});
}
function assertVerifySignatureEquals(tx, prevOutputs, value, testOutputAmount, verificationSettings) {
tx.ins.forEach((input, i) => {
assert.doesNotThrow(() => {
(0, bitgo_1.getSignatureVerifications)(tx, i, testOutputAmount, verificationSettings, prevOutputs);
});
assert.strictEqual((0, bitgo_1.verifySignature)(tx, i, testOutputAmount, verificationSettings, prevOutputs), value, JSON.stringify(verificationSettings));
if (verificationSettings?.signatureIndex === undefined && verificationSettings?.publicKey) {
assert.strictEqual((0, bitgo_1.verifySignatureWithPublicKey)(tx, i, prevOutputs, verificationSettings.publicKey), value);
}
});
}
function checkSignTransaction(tx, scriptType, signKeys, testOutputAmount) {
const prevOutputs = (0, transaction_util_1.getPrevOutputs)(scriptType, testOutputAmount, tx.network);
// return true iff there are any valid signatures at all
assertVerifySignatureEquals(tx, prevOutputs, signKeys.length > 0, testOutputAmount);
fixtures_1.fixtureKeys.forEach((k) => {
// if publicKey is given, return true iff it is included in signKeys
assertVerifySignatureEquals(tx, prevOutputs, signKeys.includes(k), testOutputAmount, {
publicKey: k.publicKey,
});
});
// When transactions are signed, the signatures have the same order as the public keys in the outputScript.
const orderedSigningKeys = fixtures_1.fixtureKeys.filter((fixtureKey) => signKeys.includes(fixtureKey));
[0, 1, 2].forEach((signatureIndex) => {
if (scriptType === 'p2tr') {
// signatureIndex parameter not support for p2tr verification
return;
}
fixtures_1.fixtureKeys.forEach((k) => {
// If no public key is given, return true iff any valid signature with given index exists.
assertVerifySignatureEquals(tx, prevOutputs, signatureIndex < signKeys.length, testOutputAmount, {
signatureIndex,
});
// If publicKey and signatureIndex are provided only return if both match.
assertVerifySignatureEquals(tx, prevOutputs, signatureIndex === orderedSigningKeys.indexOf(k), testOutputAmount, {
publicKey: k.publicKey,
signatureIndex,
});
});
});
tx.ins.forEach((input, i) => {
const signatureCount = (res) => res.reduce((sum, b) => sum + (b ? 1 : 0), 0);
const pubkeys = fixtures_1.fixtureKeys.map((k) => k.publicKey);
const verifyResult = (0, bitgo_1.verifySignatureWithPublicKeys)(tx, i, prevOutputs, pubkeys);
assert.deepStrictEqual(verifyResult, fixtures_1.fixtureKeys.map((k) => signKeys.includes(k)));
assert.strictEqual(signatureCount(verifyResult), signKeys.length);
if (signKeys.length > 0) {
(0, signatureModify_1.getTransactionWithHighS)(tx, i).forEach((txWithHighS) => {
assert.strictEqual(signatureCount((0, bitgo_1.verifySignatureWithPublicKeys)(txWithHighS, i, prevOutputs, pubkeys)), signKeys.length - 1);
});
if (scriptType !== 'p2tr' && scriptType !== 'p2trMusig2') {
assert.throws(() => signatureCount((0, bitgo_1.verifySignatureWithPublicKeys)(tx, i, (0, signatureModify_1.getPrevOutsWithInvalidOutputScript)(prevOutputs, i), pubkeys)), /prevout script .* does not match computed script .*/);
}
}
});
}
function runTestCheckSignatureVerify(network, scriptType, k1, k2, amountType = 'number') {
if (k1 && k2) {
describe(`verifySignature ${(0, networks_1.getNetworkName)(network)} ${scriptType} ${keyName(k1)} ${keyName(k2)} ${amountType}`, function () {
it(`verifies half-signed`, function () {
checkSignTransaction((0, transaction_util_1.getHalfSignedTransaction2Of3)(fixtures_1.fixtureKeys, k1, k2, scriptType, network, { amountType }), scriptType, [k1], (0, bitgo_1.toTNumber)(transaction_util_1.defaultTestOutputAmount, amountType));
});
it(`verifies full-signed`, function () {
checkSignTransaction((0, transaction_util_1.getFullSignedTransaction2Of3)(fixtures_1.fixtureKeys, k1, k2, scriptType, network, { amountType }), scriptType, [k1, k2], (0, bitgo_1.toTNumber)(transaction_util_1.defaultTestOutputAmount, amountType));
});
});
}
else {
describe(`verifySignature ${(0, networks_1.getNetworkName)(network)} ${scriptType} ${amountType} unsigned`, function () {
it(`verifies unsigned`, function () {
checkSignTransaction((0, transaction_util_1.getUnsignedTransaction2Of3)(fixtures_1.fixtureKeys, scriptType, network, { amountType }), scriptType, [], (0, bitgo_1.toTNumber)(transaction_util_1.defaultTestOutputAmount, amountType));
});
});
}
}
describe('Signature (scriptTypes2Of3)', function () {
(0, networks_1.getNetworkList)()
.filter(networks_1.isMainnet)
// The signing and verification methods are largely network-independent so let's focus on a
// single network to reduce test time.
// During development it might make sense to test all networks.
.filter(networks_1.isBitcoin)
.forEach((network) => {
getScriptTypes2Of3().forEach((scriptType) => {
runTestCheckSignatureVerify(network, scriptType);
(0, transaction_util_1.getSignKeyCombinations)(2).map(([k1, k2]) => {
runTestCheckSignatureVerify(network, scriptType, k1, k2);
runTestCheckScriptStructure(network, scriptType, k1, k2);
runTestParseScript(network, scriptType, k1, k2);
});
});
getScriptTypes2Of3().forEach((scriptType) => {
runTestCheckSignatureVerify(network, scriptType, undefined, undefined, 'bigint');
(0, transaction_util_1.getSignKeyCombinations)(2).map(([k1, k2]) => {
runTestCheckSignatureVerify(network, scriptType, k1, k2, 'bigint');
runTestCheckScriptStructure(network, scriptType, k1, k2, 'bigint');
runTestParseScript(network, scriptType, k1, k2, 'bigint');
});
});
});
});
describe('Signature (p2shP2pk)', function () {
it('sign and parse', function () {
const signedTransaction = (0, transaction_util_1.getFullSignedTransactionP2shP2pk)(fixtures_1.fixtureKeys, fixtures_1.fixtureKeys[0], networks_1.networks.bitcoin);
signedTransaction.ins.forEach((input) => {
assert.deepStrictEqual((0, normalize_1.normDefault)((0, bitgo_1.parseSignatureScript)(input)), (0, normalize_1.normDefault)({
scriptType: 'p2shP2pk',
publicKeys: [fixtures_1.fixtureKeys[0].publicKey],
signatures: [
'3045022100e637466be405032a633dcef0bd161305fe93d34ffe2aabc4af434d6f265912210220113d7085b1e00435a2583af82b8a4df3fb009a8d279d231351e42f31d6bac74401',
],
}));
});
});
runTestCheckScriptStructure(networks_1.networks.bitcoin, 'p2shP2pk', fixtures_1.fixtureKeys[0]);
runTestCheckScriptStructure(networks_1.networks.bitcoin, 'p2shP2pk', fixtures_1.fixtureKeys[0], undefined, 'bigint');
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signature.js","sourceRoot":"","sources":["../../../test/bitgo/signature.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport { BIP32Interface } from 'bip32';\n\nimport { script as bscript, classify, TxOutput } from '../../src';\nimport { getKeyName } from '../../src/testutil';\nimport { getNetworkList, getNetworkName, isBitcoin, isMainnet, Network, networks } from '../../src/networks';\n\nimport { ScriptType, ScriptType2Of3, scriptTypes2Of3 } from '../../src/bitgo/outputScripts';\nimport {\n  verifySignature,\n  UtxoTransaction,\n  parseSignatureScript,\n  getSignatureVerifications,\n  verifySignatureWithPublicKeys,\n  verifySignatureWithPublicKey,\n  isPlaceholderSignature,\n  toTNumber,\n} from '../../src/bitgo';\n\nimport * as fixtureUtil from '../fixture.util';\n\nimport { fixtureKeys } from '../integration_local_rpc/generate/fixtures';\nimport {\n  defaultTestOutputAmount,\n  getFullSignedTransaction2Of3,\n  getFullSignedTransactionP2shP2pk,\n  getHalfSignedTransaction2Of3,\n  getPrevOutputs,\n  getSignKeyCombinations,\n  getUnsignedTransaction2Of3,\n} from '../transaction_util';\nimport { getPrevOutsWithInvalidOutputScript, getTransactionWithHighS } from './signatureModify';\nimport { normDefault } from '../testutil/normalize';\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 scriptTypes2Of3.filter((scriptType) => scriptType !== 'p2trMusig2');\n}\n\nfunction keyName(k: BIP32Interface): string | undefined {\n  return getKeyName(fixtureKeys, k);\n}\n\nasync function readFixture<T>(\n  network: Network,\n  scriptType: ScriptType2Of3 | 'p2shP2pk',\n  name: string,\n  defaultValue: T\n): Promise<T> {\n  return await fixtureUtil.readFixture(\n    `${__dirname}/fixtures/signature/${getNetworkName(network)}/${scriptType}/${name}.json`,\n    defaultValue\n  );\n}\n\nfunction runTestCheckScriptStructure<TNumber extends number | bigint = number>(\n  network: Network,\n  scriptType: ScriptType2Of3 | 'p2shP2pk',\n  signer1: BIP32Interface,\n  signer2?: BIP32Interface,\n  amountType: 'number' | 'bigint' = 'number'\n) {\n  it(\n    `has expected script structure [${getNetworkName(network)} ${scriptType} ` +\n      `${keyName(signer1)} ${signer2 ? keyName(signer2) : ''} ${amountType}]`,\n    async function () {\n      let tx;\n\n      if (scriptType === 'p2shP2pk') {\n        tx = getFullSignedTransactionP2shP2pk<TNumber>(fixtureKeys, signer1, network, { amountType });\n      } else {\n        if (!signer2) {\n          throw new Error(`must set cosigner`);\n        }\n        tx = getFullSignedTransaction2Of3<TNumber>(fixtureKeys, signer1, signer2, scriptType, network, { amountType });\n      }\n\n      const { script, witness } = tx.ins[0];\n      const scriptDecompiled = bscript.decompile(script);\n      if (!scriptDecompiled) {\n        throw new Error();\n      }\n      const scriptASM = bscript.toASM(script).split(' ');\n      const classifyInput = classify.input(script);\n      const classifyWitness = classify.witness(witness);\n\n      let pubScript;\n      let classifyPubScript;\n      let pubScriptASM;\n\n      let tapscript;\n      let tapscriptASM;\n      let classifyTapscript;\n\n      if (classifyInput === 'scripthash' || classifyWitness === 'witnessscripthash') {\n        if (witness.length) {\n          pubScript = witness[witness.length - 1];\n        } else {\n          pubScript = scriptDecompiled[scriptDecompiled.length - 1] as Buffer;\n        }\n\n        classifyPubScript = classify.output(pubScript);\n        pubScriptASM = bscript.toASM(pubScript).split(' ');\n      } else if (classifyWitness === 'taproot') {\n        tapscript = witness[witness.length - 2];\n        classifyTapscript = classify.output(tapscript);\n        tapscriptASM = bscript.toASM(tapscript).split(' ');\n      }\n\n      const structure = {\n        publicKeys: fixtureKeys.map((k) => k.publicKey.toString('hex')),\n        script: script?.toString('hex'),\n        witness: witness?.map((w) => w.toString('hex')),\n        scriptASM,\n        pubScriptASM,\n        tapscriptASM,\n        classifyInput,\n        classifyWitness,\n        classifyPubScript,\n        classifyTapscript,\n      };\n\n      const fixtureName = ['structure', keyName(signer1), signer2 ? keyName(signer2) : 'none'].join('-');\n      fixtureUtil.assertEqualJSON(structure, await readFixture(network, scriptType, fixtureName, structure));\n    }\n  );\n}\n\nfunction runTestParseScript<TNumber extends number | bigint = number>(\n  network: Network,\n  scriptType: ScriptType,\n  k1: BIP32Interface,\n  k2: BIP32Interface,\n  amountType: 'number' | 'bigint' = 'number'\n) {\n  async function testParseSignedInputs(\n    tx: UtxoTransaction<TNumber>,\n    name: string,\n    expectedScriptType: string | undefined,\n    { expectedPlaceholderSignatures }: { expectedPlaceholderSignatures: number }\n  ) {\n    const parsed = parseSignatureScript(tx.ins[0]);\n    assert.strictEqual(\n      parsed.scriptType,\n      expectedScriptType === 'p2tr' ? 'taprootScriptPathSpend' : expectedScriptType\n    );\n    const parsed2Of3 = { ...parsed, scriptType: expectedScriptType };\n    fixtureUtil.assertEqualJSON(\n      parsed2Of3,\n      await readFixture(network, scriptType, ['parsed', keyName(k1), keyName(k2), name].join('-'), parsed2Of3)\n    );\n\n    if (!parsed.scriptType) {\n      return;\n    }\n\n    switch (parsed.scriptType) {\n      case 'p2shP2pk':\n        // we don't parse the signature for this script type\n        break;\n      case 'p2sh':\n      case 'p2shP2wsh':\n      case 'p2wsh':\n      case 'taprootScriptPathSpend':\n        assert.strictEqual(\n          parsed.signatures.filter((s) => isPlaceholderSignature(s)).length,\n          expectedPlaceholderSignatures\n        );\n        break;\n      default:\n        throw new Error(`unexpected scriptType ${(parsed as any).scriptType}`);\n    }\n  }\n\n  if (scriptType !== 'p2shP2pk') {\n    it(`parses half-signed inputs [${getNetworkName(network)} ${scriptType} ${amountType}]`, async function () {\n      await testParseSignedInputs(\n        getHalfSignedTransaction2Of3<TNumber>(fixtureKeys, k1, k2, scriptType, network, { amountType }),\n        'halfSigned',\n        scriptType,\n        { expectedPlaceholderSignatures: scriptType === 'p2tr' ? 1 : 2 }\n      );\n    });\n  }\n\n  it(`parses full-signed inputs [${getNetworkName(network)} ${scriptType} ${amountType}]`, async function () {\n    if (scriptType === 'p2shP2pk') {\n      await testParseSignedInputs(\n        getFullSignedTransactionP2shP2pk<TNumber>(fixtureKeys, k1, network, { amountType }),\n        'fullSigned',\n        scriptType,\n        { expectedPlaceholderSignatures: 0 }\n      );\n    } else {\n      await testParseSignedInputs(\n        getFullSignedTransaction2Of3<TNumber>(fixtureKeys, k1, k2, scriptType, network, { amountType }),\n        'fullSigned',\n        scriptType,\n        { expectedPlaceholderSignatures: 0 }\n      );\n    }\n  });\n}\n\nfunction assertVerifySignatureEquals<TNumber extends number | bigint>(\n  tx: UtxoTransaction<TNumber>,\n  prevOutputs: TxOutput<TNumber>[],\n  value: boolean,\n  testOutputAmount: TNumber,\n  verificationSettings?: {\n    publicKey?: Buffer;\n    signatureIndex?: number;\n  }\n) {\n  tx.ins.forEach((input, i) => {\n    assert.doesNotThrow(() => {\n      getSignatureVerifications(tx, i, testOutputAmount, verificationSettings, prevOutputs);\n    });\n    assert.strictEqual(\n      verifySignature(tx, i, testOutputAmount, verificationSettings, prevOutputs),\n      value,\n      JSON.stringify(verificationSettings)\n    );\n    if (verificationSettings?.signatureIndex === undefined && verificationSettings?.publicKey) {\n      assert.strictEqual(verifySignatureWithPublicKey(tx, i, prevOutputs, verificationSettings.publicKey), value);\n    }\n  });\n}\n\nfunction checkSignTransaction<TNumber extends number | bigint>(\n  tx: UtxoTransaction<TNumber>,\n  scriptType: ScriptType2Of3,\n  signKeys: BIP32Interface[],\n  testOutputAmount: TNumber\n) {\n  const prevOutputs = getPrevOutputs<TNumber>(scriptType, testOutputAmount, tx.network) as TxOutput<TNumber>[];\n\n  // return true iff there are any valid signatures at all\n  assertVerifySignatureEquals<TNumber>(tx, prevOutputs, signKeys.length > 0, testOutputAmount);\n\n  fixtureKeys.forEach((k) => {\n    // if publicKey is given, return true iff it is included in signKeys\n    assertVerifySignatureEquals<TNumber>(tx, prevOutputs, signKeys.includes(k), testOutputAmount, {\n      publicKey: k.publicKey,\n    });\n  });\n\n  // When transactions are signed, the signatures have the same order as the public keys in the outputScript.\n  const orderedSigningKeys = fixtureKeys.filter((fixtureKey) => signKeys.includes(fixtureKey));\n\n  [0, 1, 2].forEach((signatureIndex) => {\n    if (scriptType === 'p2tr') {\n      // signatureIndex parameter not support for p2tr verification\n      return;\n    }\n    fixtureKeys.forEach((k) => {\n      // If no public key is given, return true iff any valid signature with given index exists.\n      assertVerifySignatureEquals<TNumber>(tx, prevOutputs, signatureIndex < signKeys.length, testOutputAmount, {\n        signatureIndex,\n      });\n\n      // If publicKey and signatureIndex are provided only return if both match.\n      assertVerifySignatureEquals<TNumber>(\n        tx,\n        prevOutputs,\n        signatureIndex === orderedSigningKeys.indexOf(k),\n        testOutputAmount,\n        {\n          publicKey: k.publicKey,\n          signatureIndex,\n        }\n      );\n    });\n  });\n\n  tx.ins.forEach((input, i) => {\n    const signatureCount = (res: boolean[]) => res.reduce((sum, b) => sum + (b ? 1 : 0), 0);\n    const pubkeys = fixtureKeys.map((k) => k.publicKey);\n    const verifyResult = verifySignatureWithPublicKeys(tx, i, prevOutputs, pubkeys);\n    assert.deepStrictEqual(\n      verifyResult,\n      fixtureKeys.map((k) => signKeys.includes(k))\n    );\n    assert.strictEqual(signatureCount(verifyResult), signKeys.length);\n\n    if (signKeys.length > 0) {\n      getTransactionWithHighS(tx, i).forEach((txWithHighS) => {\n        assert.strictEqual(\n          signatureCount(verifySignatureWithPublicKeys<TNumber>(txWithHighS, i, prevOutputs, pubkeys)),\n          signKeys.length - 1\n        );\n      });\n\n      if (scriptType !== 'p2tr' && scriptType !== 'p2trMusig2') {\n        assert.throws(\n          () =>\n            signatureCount(\n              verifySignatureWithPublicKeys<TNumber>(tx, i, getPrevOutsWithInvalidOutputScript(prevOutputs, i), pubkeys)\n            ),\n          /prevout script .* does not match computed script .*/\n        );\n      }\n    }\n  });\n}\n\nfunction runTestCheckSignatureVerify<TNumber extends number | bigint = number>(\n  network: Network,\n  scriptType: ScriptType2Of3,\n  k1?: BIP32Interface,\n  k2?: BIP32Interface,\n  amountType: 'number' | 'bigint' = 'number'\n) {\n  if (k1 && k2) {\n    describe(`verifySignature ${getNetworkName(network)} ${scriptType} ${keyName(k1)} ${keyName(\n      k2\n    )} ${amountType}`, function () {\n      it(`verifies half-signed`, function () {\n        checkSignTransaction(\n          getHalfSignedTransaction2Of3<TNumber>(fixtureKeys, k1, k2, scriptType, network, { amountType }),\n          scriptType,\n          [k1],\n          toTNumber<TNumber>(defaultTestOutputAmount, amountType)\n        );\n      });\n\n      it(`verifies full-signed`, function () {\n        checkSignTransaction(\n          getFullSignedTransaction2Of3<TNumber>(fixtureKeys, k1, k2, scriptType, network, { amountType }),\n          scriptType,\n          [k1, k2],\n          toTNumber<TNumber>(defaultTestOutputAmount, amountType)\n        );\n      });\n    });\n  } else {\n    describe(`verifySignature ${getNetworkName(network)} ${scriptType} ${amountType} unsigned`, function () {\n      it(`verifies unsigned`, function () {\n        checkSignTransaction(\n          getUnsignedTransaction2Of3<TNumber>(fixtureKeys, scriptType, network, { amountType }),\n          scriptType,\n          [],\n          toTNumber<TNumber>(defaultTestOutputAmount, amountType)\n        );\n      });\n    });\n  }\n}\n\ndescribe('Signature (scriptTypes2Of3)', function () {\n  getNetworkList()\n    .filter(isMainnet)\n    // The signing and verification methods are largely network-independent so let's focus on a\n    // single network to reduce test time.\n    // During development it might make sense to test all networks.\n    .filter(isBitcoin)\n    .forEach((network) => {\n      getScriptTypes2Of3().forEach((scriptType) => {\n        runTestCheckSignatureVerify(network, scriptType);\n\n        getSignKeyCombinations(2).map(([k1, k2]) => {\n          runTestCheckSignatureVerify(network, scriptType, k1, k2);\n          runTestCheckScriptStructure(network, scriptType, k1, k2);\n          runTestParseScript(network, scriptType, k1, k2);\n        });\n      });\n      getScriptTypes2Of3().forEach((scriptType) => {\n        runTestCheckSignatureVerify<bigint>(network, scriptType, undefined, undefined, 'bigint');\n\n        getSignKeyCombinations(2).map(([k1, k2]) => {\n          runTestCheckSignatureVerify<bigint>(network, scriptType, k1, k2, 'bigint');\n          runTestCheckScriptStructure<bigint>(network, scriptType, k1, k2, 'bigint');\n          runTestParseScript<bigint>(network, scriptType, k1, k2, 'bigint');\n        });\n      });\n    });\n});\n\ndescribe('Signature (p2shP2pk)', function () {\n  it('sign and parse', function () {\n    const signedTransaction = getFullSignedTransactionP2shP2pk(fixtureKeys, fixtureKeys[0], networks.bitcoin);\n\n    signedTransaction.ins.forEach((input) => {\n      assert.deepStrictEqual(\n        normDefault(parseSignatureScript(input)),\n        normDefault({\n          scriptType: 'p2shP2pk',\n          publicKeys: [fixtureKeys[0].publicKey],\n          signatures: [\n            '3045022100e637466be405032a633dcef0bd161305fe93d34ffe2aabc4af434d6f265912210220113d7085b1e00435a2583af82b8a4df3fb009a8d279d231351e42f31d6bac74401',\n          ],\n        })\n      );\n    });\n  });\n\n  runTestCheckScriptStructure(networks.bitcoin, 'p2shP2pk', fixtureKeys[0]);\n  runTestCheckScriptStructure<bigint>(networks.bitcoin, 'p2shP2pk', fixtureKeys[0], undefined, 'bigint');\n});\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!