Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Socket
Sign inDemoInstall

@bitgo/utxo-lib

Package Overview
Dependencies
Maintainers
4
Versions
171
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@bitgo/utxo-lib - npm Package Compare versions

Comparing version 2.2.0-rc.2 to 2.2.0-rc.3

5

dist/src/bitgo/outputScripts.d.ts

@@ -18,2 +18,7 @@ /// <reference types="node" />

/**
* Return scripts for p2sh-p2pk (used for BCH/BSV replay protection)
* @param pubkey
*/
export declare function createOutputScriptP2shP2pk(pubkey: Buffer): SpendableScript;
/**
* Return scripts for 2-of-3 multisig output

@@ -20,0 +25,0 @@ * @param pubkeys - the key triple for multisig

20

dist/src/bitgo/outputScripts.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.createOutputScript2of3 = exports.scriptType2Of3AsPrevOutType = exports.isScriptType2Of3 = exports.scriptTypes2Of3 = void 0;
exports.createOutputScript2of3 = exports.createOutputScriptP2shP2pk = exports.scriptType2Of3AsPrevOutType = exports.isScriptType2Of3 = exports.scriptTypes2Of3 = void 0;
const assert = require("assert");

@@ -31,2 +31,18 @@ const bitcoinjs = require("bitcoinjs-lib");

/**
* Return scripts for p2sh-p2pk (used for BCH/BSV replay protection)
* @param pubkey
*/
function createOutputScriptP2shP2pk(pubkey) {
const p2pk = bitcoinjs.payments.p2pk({ pubkey });
const p2sh = bitcoinjs.payments.p2sh({ redeem: p2pk });
if (!p2sh.output || !p2pk.output) {
throw new Error(`invalid state`);
}
return {
scriptPubKey: p2sh.output,
redeemScript: p2pk.output,
};
}
exports.createOutputScriptP2shP2pk = createOutputScriptP2shP2pk;
/**
* Return scripts for 2-of-3 multisig output

@@ -107,2 +123,2 @@ * @param pubkeys - the key triple for multisig

}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"outputScripts.js","sourceRoot":"","sources":["../../../src/bitgo/outputScripts.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport * as bitcoinjs from 'bitcoinjs-lib';\nimport { OP_CHECKSIG, OP_CHECKSIGVERIFY } from 'bitcoin-ops';\nimport { isTriple, Triple } from './types';\n\nexport const scriptTypes2Of3 = ['p2sh', 'p2shP2wsh', 'p2wsh', 'p2tr'] as const;\nexport type ScriptType2Of3 = typeof scriptTypes2Of3[number];\n\nexport function isScriptType2Of3(t: string): t is ScriptType2Of3 {\n  return scriptTypes2Of3.includes(t as ScriptType2Of3);\n}\n\n/**\n * @param t\n * @return string prevOut as defined in PREVOUT_TYPES (bitcoinjs-lib/.../transaction_builder.js)\n */\nexport function scriptType2Of3AsPrevOutType(t: ScriptType2Of3): string {\n  switch (t) {\n    case 'p2sh':\n      return 'p2sh-p2ms';\n    case 'p2shP2wsh':\n      return 'p2sh-p2wsh-p2ms';\n    case 'p2wsh':\n      return 'p2wsh-p2ms';\n    default:\n      throw new Error(`unsupported script type ${t}`);\n  }\n}\n\nexport type SpendableScript = {\n  scriptPubKey: Buffer;\n  redeemScript?: Buffer;\n  witnessScript?: Buffer;\n  /** A triple of control blocks for the user+bitgo, user+backup, and backup+bitgo scripts in that order. */\n  controlBlocks?: [userBitGoScript: Buffer, userBackupScript: Buffer, backupBitGoScript: Buffer];\n};\n\n/**\n * Return scripts for 2-of-3 multisig output\n * @param pubkeys - the key triple for multisig\n * @param scriptType\n * @returns {{redeemScript, witnessScript, scriptPubKey}}\n */\nexport function createOutputScript2of3(pubkeys: Buffer[], scriptType: ScriptType2Of3): SpendableScript {\n  if (!isTriple(pubkeys)) {\n    throw new Error(`must provide pubkey triple`);\n  }\n\n  pubkeys.forEach((key) => {\n    if (key.length !== 33) {\n      throw new Error(`Unexpected key length ${key.length}. Must use compressed keys.`);\n    }\n  });\n\n  if (scriptType === 'p2tr') {\n    // p2tr addresses use a combination of 2 of 2 multisig scripts distinct from\n    // the 2 of 3 multisig used for other script types\n    return createTaprootScript2of3(pubkeys);\n  }\n\n  const script2of3 = bitcoinjs.payments.p2ms({ m: 2, pubkeys });\n  assert(script2of3.output);\n\n  let scriptPubKey: bitcoinjs.Payment;\n  let redeemScript: bitcoinjs.Payment | undefined;\n  let witnessScript: bitcoinjs.Payment | undefined;\n  switch (scriptType) {\n    case 'p2sh':\n      redeemScript = script2of3;\n      scriptPubKey = bitcoinjs.payments.p2sh({ redeem: script2of3 });\n      break;\n    case 'p2shP2wsh':\n      witnessScript = script2of3;\n      redeemScript = bitcoinjs.payments.p2wsh({ redeem: script2of3 });\n      scriptPubKey = bitcoinjs.payments.p2sh({ redeem: redeemScript });\n      break;\n    case 'p2wsh':\n      witnessScript = script2of3;\n      scriptPubKey = bitcoinjs.payments.p2wsh({ redeem: witnessScript });\n      break;\n    default:\n      throw new Error(`unknown multisig script type ${scriptType}`);\n  }\n\n  assert(scriptPubKey);\n  assert(scriptPubKey.output);\n\n  return {\n    scriptPubKey: scriptPubKey.output,\n    redeemScript: redeemScript?.output,\n    witnessScript: witnessScript?.output,\n  };\n}\n\n/**\n * Creates and returns a taproot output script using the user and bitgo keys for the aggregate\n * public key and a taptree containing a user+bitgo 2-of-2 script at the first depth level of the\n * tree and user+backup and bitgo+backup 2-of-2 scripts one level deeper.\n * @param pubkeys - a pubkey array containing the user key, backup key, and bitgo key in that order\n * @returns {{scriptPubKey}}\n */\nfunction createTaprootScript2of3([userKey, backupKey, bitGoKey]: Triple<Buffer>): SpendableScript {\n  const userBitGoScript = bitcoinjs.script.compile([userKey, OP_CHECKSIGVERIFY, bitGoKey, OP_CHECKSIG]);\n  const userBackupScript = bitcoinjs.script.compile([userKey, OP_CHECKSIGVERIFY, backupKey, OP_CHECKSIG]);\n  const backupBitGoScript = bitcoinjs.script.compile([backupKey, OP_CHECKSIGVERIFY, bitGoKey, OP_CHECKSIG]);\n\n  assert(userBitGoScript);\n  assert(userBackupScript);\n  assert(backupBitGoScript);\n\n  const { output } = bitcoinjs.payments.p2tr({\n    pubkeys: [userKey, bitGoKey],\n    scripts: [userBitGoScript, userBackupScript, backupBitGoScript],\n    weights: [2, 1, 1],\n  });\n\n  assert(output);\n\n  // TODO: return control blocks once they are returned from payments.p2tr()\n  return {\n    scriptPubKey: output,\n  };\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"outputScripts.js","sourceRoot":"","sources":["../../../src/bitgo/outputScripts.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport * as bitcoinjs from 'bitcoinjs-lib';\nimport { OP_CHECKSIG, OP_CHECKSIGVERIFY } from 'bitcoin-ops';\nimport { isTriple, Triple } from './types';\n\nexport const scriptTypes2Of3 = ['p2sh', 'p2shP2wsh', 'p2wsh', 'p2tr'] as const;\nexport type ScriptType2Of3 = typeof scriptTypes2Of3[number];\n\nexport function isScriptType2Of3(t: string): t is ScriptType2Of3 {\n  return scriptTypes2Of3.includes(t as ScriptType2Of3);\n}\n\n/**\n * @param t\n * @return string prevOut as defined in PREVOUT_TYPES (bitcoinjs-lib/.../transaction_builder.js)\n */\nexport function scriptType2Of3AsPrevOutType(t: ScriptType2Of3): string {\n  switch (t) {\n    case 'p2sh':\n      return 'p2sh-p2ms';\n    case 'p2shP2wsh':\n      return 'p2sh-p2wsh-p2ms';\n    case 'p2wsh':\n      return 'p2wsh-p2ms';\n    default:\n      throw new Error(`unsupported script type ${t}`);\n  }\n}\n\nexport type SpendableScript = {\n  scriptPubKey: Buffer;\n  redeemScript?: Buffer;\n  witnessScript?: Buffer;\n  /** A triple of control blocks for the user+bitgo, user+backup, and backup+bitgo scripts in that order. */\n  controlBlocks?: [userBitGoScript: Buffer, userBackupScript: Buffer, backupBitGoScript: Buffer];\n};\n\n/**\n * Return scripts for p2sh-p2pk (used for BCH/BSV replay protection)\n * @param pubkey\n */\nexport function createOutputScriptP2shP2pk(pubkey: Buffer): SpendableScript {\n  const p2pk = bitcoinjs.payments.p2pk({ pubkey });\n  const p2sh = bitcoinjs.payments.p2sh({ redeem: p2pk });\n  if (!p2sh.output || !p2pk.output) {\n    throw new Error(`invalid state`);\n  }\n  return {\n    scriptPubKey: p2sh.output,\n    redeemScript: p2pk.output,\n  };\n}\n\n/**\n * Return scripts for 2-of-3 multisig output\n * @param pubkeys - the key triple for multisig\n * @param scriptType\n * @returns {{redeemScript, witnessScript, scriptPubKey}}\n */\nexport function createOutputScript2of3(pubkeys: Buffer[], scriptType: ScriptType2Of3): SpendableScript {\n  if (!isTriple(pubkeys)) {\n    throw new Error(`must provide pubkey triple`);\n  }\n\n  pubkeys.forEach((key) => {\n    if (key.length !== 33) {\n      throw new Error(`Unexpected key length ${key.length}. Must use compressed keys.`);\n    }\n  });\n\n  if (scriptType === 'p2tr') {\n    // p2tr addresses use a combination of 2 of 2 multisig scripts distinct from\n    // the 2 of 3 multisig used for other script types\n    return createTaprootScript2of3(pubkeys);\n  }\n\n  const script2of3 = bitcoinjs.payments.p2ms({ m: 2, pubkeys });\n  assert(script2of3.output);\n\n  let scriptPubKey: bitcoinjs.Payment;\n  let redeemScript: bitcoinjs.Payment | undefined;\n  let witnessScript: bitcoinjs.Payment | undefined;\n  switch (scriptType) {\n    case 'p2sh':\n      redeemScript = script2of3;\n      scriptPubKey = bitcoinjs.payments.p2sh({ redeem: script2of3 });\n      break;\n    case 'p2shP2wsh':\n      witnessScript = script2of3;\n      redeemScript = bitcoinjs.payments.p2wsh({ redeem: script2of3 });\n      scriptPubKey = bitcoinjs.payments.p2sh({ redeem: redeemScript });\n      break;\n    case 'p2wsh':\n      witnessScript = script2of3;\n      scriptPubKey = bitcoinjs.payments.p2wsh({ redeem: witnessScript });\n      break;\n    default:\n      throw new Error(`unknown multisig script type ${scriptType}`);\n  }\n\n  assert(scriptPubKey);\n  assert(scriptPubKey.output);\n\n  return {\n    scriptPubKey: scriptPubKey.output,\n    redeemScript: redeemScript?.output,\n    witnessScript: witnessScript?.output,\n  };\n}\n\n/**\n * Creates and returns a taproot output script using the user and bitgo keys for the aggregate\n * public key and a taptree containing a user+bitgo 2-of-2 script at the first depth level of the\n * tree and user+backup and bitgo+backup 2-of-2 scripts one level deeper.\n * @param pubkeys - a pubkey array containing the user key, backup key, and bitgo key in that order\n * @returns {{scriptPubKey}}\n */\nfunction createTaprootScript2of3([userKey, backupKey, bitGoKey]: Triple<Buffer>): SpendableScript {\n  const userBitGoScript = bitcoinjs.script.compile([userKey, OP_CHECKSIGVERIFY, bitGoKey, OP_CHECKSIG]);\n  const userBackupScript = bitcoinjs.script.compile([userKey, OP_CHECKSIGVERIFY, backupKey, OP_CHECKSIG]);\n  const backupBitGoScript = bitcoinjs.script.compile([backupKey, OP_CHECKSIGVERIFY, bitGoKey, OP_CHECKSIG]);\n\n  assert(userBitGoScript);\n  assert(userBackupScript);\n  assert(backupBitGoScript);\n\n  const { output } = bitcoinjs.payments.p2tr({\n    pubkeys: [userKey, bitGoKey],\n    scripts: [userBitGoScript, userBackupScript, backupBitGoScript],\n    weights: [2, 1, 1],\n  });\n\n  assert(output);\n\n  // TODO: return control blocks once they are returned from payments.p2tr()\n  return {\n    scriptPubKey: output,\n  };\n}\n"]}

@@ -10,2 +10,3 @@ /// <reference types="node" />

inputClassification: InputType;
p2shOutputClassification?: string;
publicKeys?: Buffer[];

@@ -12,0 +13,0 @@ }

48

dist/src/bitgo/signature.js

@@ -45,3 +45,4 @@ "use strict";

const isNativeSegwitInput = input.script.length === 0;
let decompiledSigScript, inputClassification;
let decompiledSigScript;
let inputClassification;
if (isSegwitInput) {

@@ -68,2 +69,3 @@ // The decompiledSigScript is the script containing the signatures, public keys, and the script that was committed

if (inputClassification === classify.types.P2PKH) {
/* istanbul ignore next */
if (!decompiledSigScript || decompiledSigScript.length !== 2) {

@@ -73,2 +75,3 @@ throw new Error('unexpected signature for p2pkh');

const [signature, publicKey] = decompiledSigScript;
/* istanbul ignore next */
if (!Buffer.isBuffer(signature) || !Buffer.isBuffer(publicKey)) {

@@ -96,2 +99,23 @@ throw new Error('unexpected signature for p2pkh');

const expectedScriptType = inputClassification === classify.types.P2SH || inputClassification === classify.types.P2WSH;
if (!expectedScriptType) {
return { isSegwitInput, inputClassification };
}
const pubScript = decompiledSigScript[decompiledSigScript.length - 1];
/* istanbul ignore next */
if (!Buffer.isBuffer(pubScript)) {
throw new Error(`invalid pubScript`);
}
const p2shOutputClassification = classify.output(pubScript);
if (p2shOutputClassification !== 'multisig') {
return {
isSegwitInput,
inputClassification,
p2shOutputClassification,
};
}
const decompiledPubScript = bitcoinjs_lib_1.script.decompile(pubScript);
if (decompiledPubScript === null) {
/* istanbul ignore next */
throw new Error(`could not decompile pubScript`);
}
const expectedScriptLength =

@@ -102,9 +126,11 @@ // complete transactions with 2 signatures

decompiledSigScript.length === 5;
if (!expectedScriptType || !expectedScriptLength) {
if (!expectedScriptLength) {
return { isSegwitInput, inputClassification };
}
if (isSegwitInput) {
/* istanbul ignore next */
if (!Buffer.isBuffer(decompiledSigScript[0])) {
throw new Error(`expected decompiledSigScript[0] to be a buffer for segwit inputs`);
}
/* istanbul ignore next */
if (decompiledSigScript[0].length !== 0) {

@@ -118,13 +144,7 @@ throw new Error(`witness stack expected to start with empty buffer`);

const signatures = decompiledSigScript.slice(1 /* ignore leading OP_0 */, -1 /* ignore trailing pubScript */);
/* istanbul ignore next */
if (signatures.length !== 2 && signatures.length !== 3) {
throw new Error(`expected 2 or 3 signatures, got ${signatures.length}`);
}
const pubScript = decompiledSigScript[decompiledSigScript.length - 1];
if (!Buffer.isBuffer(pubScript)) {
throw new Error(`invalid pubscript`);
}
const decompiledPubScript = bitcoinjs_lib_1.script.decompile(pubScript);
if (decompiledPubScript === null) {
throw new Error(`could not decompile pubScript`);
}
/* istanbul ignore next */
if (decompiledPubScript.length !== 6) {

@@ -135,2 +155,3 @@ throw new Error(`unexpected decompiledPubScript length`);

publicKeys.forEach((b) => {
/* istanbul ignore next */
if (!Buffer.isBuffer(b)) {

@@ -141,2 +162,3 @@ throw new Error();

if (publicKeys.length !== 3) {
/* istanbul ignore next */
throw new Error(`expected 3 public keys, got ${publicKeys.length}`);

@@ -148,2 +170,3 @@ }

const signatureThreshold = decompiledPubScript[0] - 80;
/* istanbul ignore next */
if (signatureThreshold !== 2) {

@@ -153,2 +176,3 @@ throw new Error(`expected signatureThreshold 2, got ${signatureThreshold}`);

const nPubKeys = decompiledPubScript[len - 2] - 80;
/* istanbul ignore next */
if (nPubKeys !== 3) {

@@ -158,2 +182,3 @@ throw new Error(`expected nPubKeys 3, got ${nPubKeys}`);

const lastOpCode = decompiledPubScript[len - 1];
/* istanbul ignore next */
if (lastOpCode !== opcodes.OP_CHECKMULTISIG) {

@@ -165,2 +190,3 @@ throw new Error(`expected opcode #${opcodes.OP_CHECKMULTISIG}, got opcode #${lastOpCode}`);

inputClassification,
p2shOutputClassification,
signatures: signatures.map((b) => {

@@ -252,2 +278,2 @@ if (Buffer.isBuffer(b) || b === 0) {

exports.verifySignature = verifySignature;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signature.js","sourceRoot":"","sources":["../../../src/bitgo/signature.ts"],"names":[],"mappings":"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","sourcesContent":["import * as opcodes from 'bitcoin-ops';\n\nimport { ECPair, payments, script, Transaction, TxInput } from 'bitcoinjs-lib';\nimport * as classify from 'bitcoinjs-lib/src/classify';\nimport * as ScriptSignature from 'bitcoinjs-lib/src/script_signature';\n\nimport { Network } from '../networkTypes';\nimport * as networks from '../networks';\nimport { getMainnet } from '../coins';\nimport { UtxoTransaction } from './UtxoTransaction';\n\nconst inputTypes = [\n  'multisig',\n  'nonstandard',\n  'nulldata',\n  'pubkey',\n  'pubkeyhash',\n  'scripthash',\n  'witnesspubkeyhash',\n  'witnessscripthash',\n  'witnesscommitment',\n] as const;\n\ntype InputType = typeof inputTypes[number];\n\nexport interface ParsedSignatureScript {\n  isSegwitInput: boolean;\n  inputClassification: InputType;\n  publicKeys?: Buffer[];\n}\n\nexport interface ParsedSignatureP2PKH extends ParsedSignatureScript {\n  signatures: [Buffer];\n  publicKeys: [Buffer];\n  pubScript: Buffer;\n}\n\nexport interface ParsedSignatureScript2Of3 extends ParsedSignatureScript {\n  signatures:\n    | [Buffer, Buffer] // fully-signed transactions with signatures\n    /* Partially signed transactions with placeholder signatures.\n       For p2sh, the placeholder is OP_0 (number 0) */\n    | [Buffer | 0, Buffer | 0, Buffer | 0];\n  publicKeys: [Buffer, Buffer, Buffer];\n  pubScript: Buffer;\n}\n\nexport function getDefaultSigHash(network: Network): number {\n  switch (getMainnet(network)) {\n    case networks.bitcoincash:\n    case networks.bitcoinsv:\n    case networks.bitcoingold:\n      return Transaction.SIGHASH_ALL | UtxoTransaction.SIGHASH_FORKID;\n    default:\n      return Transaction.SIGHASH_ALL;\n  }\n}\n\n/**\n * Parse a transaction's signature script to obtain public keys, signatures, the sig script,\n * and other properties.\n *\n * Only supports script types used in BitGo transactions.\n *\n * @param input\n * @returns ParsedSignatureScript\n */\nexport function parseSignatureScript(\n  input: TxInput\n): ParsedSignatureScript | ParsedSignatureP2PKH | ParsedSignatureScript2Of3 {\n  const isSegwitInput = input.witness.length > 0;\n  const isNativeSegwitInput = input.script.length === 0;\n  let decompiledSigScript, inputClassification: InputType;\n  if (isSegwitInput) {\n    // The decompiledSigScript is the script containing the signatures, public keys, and the script that was committed\n    // to (pubScript). If this is a segwit input the decompiledSigScript is in the witness, regardless of whether it\n    // is native or not. The inputClassification is determined based on whether or not the input is native to give an\n    // accurate classification. Note that p2shP2wsh inputs will be classified as p2sh and not p2wsh.\n    decompiledSigScript = input.witness;\n    if (isNativeSegwitInput) {\n      inputClassification = classify.witness(decompiledSigScript, true) as InputType;\n    } else {\n      inputClassification = classify.input(input.script, true) as InputType;\n    }\n  } else {\n    inputClassification = classify.input(input.script, true) as InputType;\n    decompiledSigScript = script.decompile(input.script);\n  }\n\n  if (!decompiledSigScript) {\n    return { isSegwitInput, inputClassification };\n  }\n\n  if (inputClassification === classify.types.P2PKH) {\n    if (!decompiledSigScript || decompiledSigScript.length !== 2) {\n      throw new Error('unexpected signature for p2pkh');\n    }\n    const [signature, publicKey] = decompiledSigScript;\n    if (!Buffer.isBuffer(signature) || !Buffer.isBuffer(publicKey)) {\n      throw new Error('unexpected signature for p2pkh');\n    }\n    const publicKeys = [publicKey];\n    const signatures: [Buffer] = [signature];\n    const pubScript = payments.p2pkh({ pubkey: publicKey }).output;\n\n    return { isSegwitInput, inputClassification, signatures, publicKeys, pubScript };\n  }\n\n  // Note the assumption here that if we have a p2sh or p2wsh input it will be multisig (appropriate because the\n  // BitGo platform only supports multisig within these types of inputs, with the exception of replay protection inputs,\n  // which are single signature p2sh). Signatures are all but the last entry in the decompiledSigScript.\n  // The redeemScript/witnessScript (depending on which type of input this is) is the last entry in\n  // the decompiledSigScript (denoted here as the pubScript). The public keys are the second through\n  // antepenultimate entries in the decompiledPubScript. See below for a visual representation of the typical 2-of-3\n  // multisig setup:\n  //\n  //   decompiledSigScript = 0 <sig1> <sig2> [<sig3>] <pubScript>\n  //   decompiledPubScript = 2 <pub1> <pub2> <pub3> 3 OP_CHECKMULTISIG\n  //\n  // Transactions built with `.build()` only have two signatures `<sig1>` and `<sig2>` in _decompiledSigScript_.\n  // Transactions built with `.buildIncomplete()` have three signatures, where missing signatures are substituted with `OP_0`.\n  const expectedScriptType =\n    inputClassification === classify.types.P2SH || inputClassification === classify.types.P2WSH;\n  const expectedScriptLength =\n    // complete transactions with 2 signatures\n    decompiledSigScript.length === 4 ||\n    // incomplete transaction with 3 signatures or signature placeholders\n    decompiledSigScript.length === 5;\n\n  if (!expectedScriptType || !expectedScriptLength) {\n    return { isSegwitInput, inputClassification };\n  }\n\n  if (isSegwitInput) {\n    if (!Buffer.isBuffer(decompiledSigScript[0])) {\n      throw new Error(`expected decompiledSigScript[0] to be a buffer for segwit inputs`);\n    }\n    if (decompiledSigScript[0].length !== 0) {\n      throw new Error(`witness stack expected to start with empty buffer`);\n    }\n  } else if (decompiledSigScript[0] !== opcodes.OP_0) {\n    throw new Error(`sigScript expected to start with OP_0`);\n  }\n\n  const signatures = decompiledSigScript.slice(1 /* ignore leading OP_0 */, -1 /* ignore trailing pubScript */);\n  if (signatures.length !== 2 && signatures.length !== 3) {\n    throw new Error(`expected 2 or 3 signatures, got ${signatures.length}`);\n  }\n\n  const pubScript = decompiledSigScript[decompiledSigScript.length - 1];\n  if (!Buffer.isBuffer(pubScript)) {\n    throw new Error(`invalid pubscript`);\n  }\n  const decompiledPubScript = script.decompile(pubScript);\n  if (decompiledPubScript === null) {\n    throw new Error(`could not decompile pubScript`);\n  }\n  if (decompiledPubScript.length !== 6) {\n    throw new Error(`unexpected decompiledPubScript length`);\n  }\n  const publicKeys = decompiledPubScript.slice(1, -2) as Buffer[];\n  publicKeys.forEach((b) => {\n    if (!Buffer.isBuffer(b)) {\n      throw new Error();\n    }\n  });\n  if (publicKeys.length !== 3) {\n    throw new Error(`expected 3 public keys, got ${publicKeys.length}`);\n  }\n\n  // Op codes 81 through 96 represent numbers 1 through 16 (see https://en.bitcoin.it/wiki/Script#Opcodes), which is\n  // why we subtract by 80 to get the number of signatures (n) and the number of public keys (m) in an n-of-m setup.\n  const len = decompiledPubScript.length;\n  const signatureThreshold = (decompiledPubScript[0] as number) - 80;\n  if (signatureThreshold !== 2) {\n    throw new Error(`expected signatureThreshold 2, got ${signatureThreshold}`);\n  }\n  const nPubKeys = (decompiledPubScript[len - 2] as number) - 80;\n  if (nPubKeys !== 3) {\n    throw new Error(`expected nPubKeys 3, got ${nPubKeys}`);\n  }\n\n  const lastOpCode = decompiledPubScript[len - 1];\n  if (lastOpCode !== opcodes.OP_CHECKMULTISIG) {\n    throw new Error(`expected opcode #${opcodes.OP_CHECKMULTISIG}, got opcode #${lastOpCode}`);\n  }\n\n  return {\n    isSegwitInput,\n    inputClassification,\n    signatures: signatures.map((b) => {\n      if (Buffer.isBuffer(b) || b === 0) {\n        return b;\n      }\n      throw new Error(`unexpected signature element ${b}`);\n    }) as [Buffer, Buffer] | [Buffer, Buffer, Buffer],\n    publicKeys,\n    pubScript,\n  };\n}\n\nexport function parseSignatureScript2Of3(input: TxInput): ParsedSignatureScript2Of3 {\n  const result = parseSignatureScript(input) as ParsedSignatureScript2Of3;\n\n  if (![classify.types.P2WSH, classify.types.P2SH, classify.types.P2PKH].includes(result.inputClassification)) {\n    throw new Error(`unexpected inputClassification ${result.inputClassification}`);\n  }\n  if (!result.signatures) {\n    throw new Error(`missing signatures`);\n  }\n  if (result.publicKeys.length !== 3) {\n    throw new Error(`unexpected pubkey count`);\n  }\n  if (!result.pubScript || result.pubScript.length === 0) {\n    throw new Error(`pubScript missing or empty`);\n  }\n\n  return result;\n}\n\n/**\n * Constraints for signature verifications.\n * Parameters are conjunctive: if multiple parameters are set, a verification for an individual\n * signature must satisfy all of them.\n */\nexport type VerificationSettings = {\n  /**\n   * The index of the signature to verify. Only iterates over non-empty signatures.\n   */\n  signatureIndex?: number;\n  /**\n   * The hex of the public key to verify.\n   */\n  publicKey?: Buffer;\n};\n\n/**\n * Result for a individual signature verification\n */\nexport type SignatureVerification = {\n  /** Set to the public key that signed for the signature */\n  signedBy: Buffer | undefined;\n};\n\n/**\n * Get signature verifications for multsig transaction\n * @param transaction\n * @param inputIndex\n * @param amount - must be set for segwit transactions and BIP143 transactions\n * @param verificationSettings\n * @returns SignatureVerification[] - in order of parsed non-empty signatures\n */\nexport function getSignatureVerifications(\n  transaction: UtxoTransaction,\n  inputIndex: number,\n  amount: number,\n  verificationSettings: VerificationSettings = {}\n): SignatureVerification[] {\n  /* istanbul ignore next */\n  if (!transaction.ins) {\n    throw new Error(`invalid transaction`);\n  }\n\n  const input = transaction.ins[inputIndex];\n  /* istanbul ignore next */\n  if (!input) {\n    throw new Error(`no input at index ${inputIndex}`);\n  }\n\n  const parsedScript = parseSignatureScript2Of3(input);\n\n  const signatures = parsedScript.signatures\n    .filter((s) => s && s.length)\n    .filter((s, i) => verificationSettings.signatureIndex === undefined || verificationSettings.signatureIndex === i);\n\n  const publicKeys = parsedScript.publicKeys.filter(\n    (buf) => verificationSettings.publicKey === undefined || verificationSettings.publicKey.equals(buf)\n  );\n\n  return signatures.map((signatureBuffer) => {\n    // slice the last byte from the signature hash input because it's the hash type\n    if (signatureBuffer === 0 || signatureBuffer.length === 0) {\n      return { signedBy: undefined };\n    }\n    const { signature, hashType } = ScriptSignature.decode(signatureBuffer);\n    const transactionHash = parsedScript.isSegwitInput\n      ? transaction.hashForWitnessV0(inputIndex, parsedScript.pubScript, amount, hashType)\n      : transaction.hashForSignatureByNetwork(inputIndex, parsedScript.pubScript, amount, hashType);\n    const signedBy = publicKeys.filter((publicKey) =>\n      ECPair.fromPublicKey(publicKey).verify(transactionHash, signature)\n    );\n\n    if (signedBy.length === 0) {\n      return { signedBy: undefined };\n    }\n    if (signedBy.length === 1) {\n      return { signedBy: signedBy[0] };\n    }\n    throw new Error(`illegal state: signed by multiple public keys`);\n  });\n}\n\n/**\n * @param transaction\n * @param inputIndex\n * @param amount\n * @param verificationSettings - if publicKey is specified, returns true iff any signature is signed by publicKey.\n */\nexport function verifySignature(\n  transaction: UtxoTransaction,\n  inputIndex: number,\n  amount: number,\n  verificationSettings: VerificationSettings = {}\n): boolean {\n  const signatureVerifications = getSignatureVerifications(\n    transaction,\n    inputIndex,\n    amount,\n    verificationSettings\n  ).filter(\n    (v) =>\n      // If no publicKey is set in verificationSettings, all signatures must be valid.\n      // Otherwise, a single valid signature by the specified pubkey is sufficient.\n      verificationSettings.publicKey === undefined ||\n      (v.signedBy !== undefined && verificationSettings.publicKey.equals(v.signedBy))\n  );\n\n  return signatureVerifications.length > 0 && signatureVerifications.every((v) => v.signedBy !== undefined);\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signature.js","sourceRoot":"","sources":["../../../src/bitgo/signature.ts"],"names":[],"mappings":";;;AAAA,uCAAuC;AAEvC,iDAA+E;AAC/E,uDAAuD;AACvD,sEAAsE;AAGtE,wCAAwC;AACxC,oCAAsC;AACtC,uDAAoD;AAEpD,MAAM,UAAU,GAAG;IACjB,UAAU;IACV,aAAa;IACb,UAAU;IACV,QAAQ;IACR,YAAY;IACZ,YAAY;IACZ,mBAAmB;IACnB,mBAAmB;IACnB,mBAAmB;CACX,CAAC;AA2BX,SAAgB,iBAAiB,CAAC,OAAgB;IAChD,QAAQ,kBAAU,CAAC,OAAO,CAAC,EAAE;QAC3B,KAAK,QAAQ,CAAC,WAAW,CAAC;QAC1B,KAAK,QAAQ,CAAC,SAAS,CAAC;QACxB,KAAK,QAAQ,CAAC,WAAW;YACvB,OAAO,2BAAW,CAAC,WAAW,GAAG,iCAAe,CAAC,cAAc,CAAC;QAClE;YACE,OAAO,2BAAW,CAAC,WAAW,CAAC;KAClC;AACH,CAAC;AATD,8CASC;AAED;;;;;;;;GAQG;AACH,SAAgB,oBAAoB,CAClC,KAAc;IAEd,MAAM,aAAa,GAAG,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC;IAC/C,MAAM,mBAAmB,GAAG,KAAK,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,CAAC;IACtD,IAAI,mBAAkD,CAAC;IACvD,IAAI,mBAA8B,CAAC;IACnC,IAAI,aAAa,EAAE;QACjB,kHAAkH;QAClH,gHAAgH;QAChH,iHAAiH;QACjH,gGAAgG;QAChG,mBAAmB,GAAG,KAAK,CAAC,OAAO,CAAC;QACpC,IAAI,mBAAmB,EAAE;YACvB,mBAAmB,GAAG,QAAQ,CAAC,OAAO,CAAC,mBAA+B,EAAE,IAAI,CAAc,CAAC;SAC5F;aAAM;YACL,mBAAmB,GAAG,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,IAAI,CAAc,CAAC;SACvE;KACF;SAAM;QACL,mBAAmB,GAAG,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,IAAI,CAAc,CAAC;QACtE,mBAAmB,GAAG,sBAAM,CAAC,SAAS,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;KACtD;IAED,IAAI,CAAC,mBAAmB,EAAE;QACxB,OAAO,EAAE,aAAa,EAAE,mBAAmB,EAAE,CAAC;KAC/C;IAED,IAAI,mBAAmB,KAAK,QAAQ,CAAC,KAAK,CAAC,KAAK,EAAE;QAChD,0BAA0B;QAC1B,IAAI,CAAC,mBAAmB,IAAI,mBAAmB,CAAC,MAAM,KAAK,CAAC,EAAE;YAC5D,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;SACnD;QACD,MAAM,CAAC,SAAS,EAAE,SAAS,CAAC,GAAG,mBAAmB,CAAC;QACnD,0BAA0B;QAC1B,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE;YAC9D,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;SACnD;QACD,MAAM,UAAU,GAAG,CAAC,SAAS,CAAC,CAAC;QAC/B,MAAM,UAAU,GAAa,CAAC,SAAS,CAAC,CAAC;QACzC,MAAM,SAAS,GAAG,wBAAQ,CAAC,KAAK,CAAC,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC,MAAM,CAAC;QAE/D,OAAO,EAAE,aAAa,EAAE,mBAAmB,EAAE,UAAU,EAAE,UAAU,EAAE,SAAS,EAAE,CAAC;KAClF;IAED,8GAA8G;IAC9G,sHAAsH;IACtH,sGAAsG;IACtG,iGAAiG;IACjG,kGAAkG;IAClG,kHAAkH;IAClH,kBAAkB;IAClB,EAAE;IACF,+DAA+D;IAC/D,oEAAoE;IACpE,EAAE;IACF,8GAA8G;IAC9G,4HAA4H;IAC5H,MAAM,kBAAkB,GACtB,mBAAmB,KAAK,QAAQ,CAAC,KAAK,CAAC,IAAI,IAAI,mBAAmB,KAAK,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC;IAE9F,IAAI,CAAC,kBAAkB,EAAE;QACvB,OAAO,EAAE,aAAa,EAAE,mBAAmB,EAAE,CAAC;KAC/C;IAED,MAAM,SAAS,GAAG,mBAAmB,CAAC,mBAAmB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;IACtE,0BAA0B;IAC1B,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE;QAC/B,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;KACtC;IAED,MAAM,wBAAwB,GAAG,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IAE5D,IAAI,wBAAwB,KAAK,UAAU,EAAE;QAC3C,OAAO;YACL,aAAa;YACb,mBAAmB;YACnB,wBAAwB;SACzB,CAAC;KACH;IAED,MAAM,mBAAmB,GAAG,sBAAM,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;IACxD,IAAI,mBAAmB,KAAK,IAAI,EAAE;QAChC,0BAA0B;QAC1B,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;KAClD;IAED,MAAM,oBAAoB;IACxB,0CAA0C;IAC1C,mBAAmB,CAAC,MAAM,KAAK,CAAC;QAChC,qEAAqE;QACrE,mBAAmB,CAAC,MAAM,KAAK,CAAC,CAAC;IAEnC,IAAI,CAAC,oBAAoB,EAAE;QACzB,OAAO,EAAE,aAAa,EAAE,mBAAmB,EAAE,CAAC;KAC/C;IAED,IAAI,aAAa,EAAE;QACjB,0BAA0B;QAC1B,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAC,CAAC,CAAC,CAAC,EAAE;YAC5C,MAAM,IAAI,KAAK,CAAC,kEAAkE,CAAC,CAAC;SACrF;QACD,0BAA0B;QAC1B,IAAI,mBAAmB,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE;YACvC,MAAM,IAAI,KAAK,CAAC,mDAAmD,CAAC,CAAC;SACtE;KACF;SAAM,IAAI,mBAAmB,CAAC,CAAC,CAAC,KAAK,OAAO,CAAC,IAAI,EAAE;QAClD,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;KAC1D;IAED,MAAM,UAAU,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC,CAAC,yBAAyB,EAAE,CAAC,CAAC,CAAC,+BAA+B,CAAC,CAAC;IAC9G,0BAA0B;IAC1B,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE;QACtD,MAAM,IAAI,KAAK,CAAC,mCAAmC,UAAU,CAAC,MAAM,EAAE,CAAC,CAAC;KACzE;IAED,0BAA0B;IAC1B,IAAI,mBAAmB,CAAC,MAAM,KAAK,CAAC,EAAE;QACpC,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;KAC1D;IACD,MAAM,UAAU,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAa,CAAC;IAChE,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;QACvB,0BAA0B;QAC1B,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE;YACvB,MAAM,IAAI,KAAK,EAAE,CAAC;SACnB;IACH,CAAC,CAAC,CAAC;IACH,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE;QAC3B,0BAA0B;QAC1B,MAAM,IAAI,KAAK,CAAC,+BAA+B,UAAU,CAAC,MAAM,EAAE,CAAC,CAAC;KACrE;IAED,kHAAkH;IAClH,kHAAkH;IAClH,MAAM,GAAG,GAAG,mBAAmB,CAAC,MAAM,CAAC;IACvC,MAAM,kBAAkB,GAAI,mBAAmB,CAAC,CAAC,CAAY,GAAG,EAAE,CAAC;IACnE,0BAA0B;IAC1B,IAAI,kBAAkB,KAAK,CAAC,EAAE;QAC5B,MAAM,IAAI,KAAK,CAAC,sCAAsC,kBAAkB,EAAE,CAAC,CAAC;KAC7E;IACD,MAAM,QAAQ,GAAI,mBAAmB,CAAC,GAAG,GAAG,CAAC,CAAY,GAAG,EAAE,CAAC;IAC/D,0BAA0B;IAC1B,IAAI,QAAQ,KAAK,CAAC,EAAE;QAClB,MAAM,IAAI,KAAK,CAAC,4BAA4B,QAAQ,EAAE,CAAC,CAAC;KACzD;IAED,MAAM,UAAU,GAAG,mBAAmB,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC;IAChD,0BAA0B;IAC1B,IAAI,UAAU,KAAK,OAAO,CAAC,gBAAgB,EAAE;QAC3C,MAAM,IAAI,KAAK,CAAC,oBAAoB,OAAO,CAAC,gBAAgB,iBAAiB,UAAU,EAAE,CAAC,CAAC;KAC5F;IAED,OAAO;QACL,aAAa;QACb,mBAAmB;QACnB,wBAAwB;QACxB,UAAU,EAAE,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;YAC/B,IAAI,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE;gBACjC,OAAO,CAAC,CAAC;aACV;YACD,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,EAAE,CAAC,CAAC;QACvD,CAAC,CAAgD;QACjD,UAAU;QACV,SAAS;KACV,CAAC;AACJ,CAAC;AApKD,oDAoKC;AAED,SAAgB,wBAAwB,CAAC,KAAc;IACrD,MAAM,MAAM,GAAG,oBAAoB,CAAC,KAAK,CAA8B,CAAC;IAExE,IAAI,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,KAAK,EAAE,QAAQ,CAAC,KAAK,CAAC,IAAI,EAAE,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,mBAAmB,CAAC,EAAE;QAC3G,MAAM,IAAI,KAAK,CAAC,kCAAkC,MAAM,CAAC,mBAAmB,EAAE,CAAC,CAAC;KACjF;IACD,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE;QACtB,MAAM,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAAC;KACvC;IACD,IAAI,MAAM,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE;QAClC,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;KAC5C;IACD,IAAI,CAAC,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE;QACtD,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;KAC/C;IAED,OAAO,MAAM,CAAC;AAChB,CAAC;AAjBD,4DAiBC;AA0BD;;;;;;;GAOG;AACH,SAAgB,yBAAyB,CACvC,WAA4B,EAC5B,UAAkB,EAClB,MAAc,EACd,uBAA6C,EAAE;IAE/C,0BAA0B;IAC1B,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE;QACpB,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;KACxC;IAED,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;IAC1C,0BAA0B;IAC1B,IAAI,CAAC,KAAK,EAAE;QACV,MAAM,IAAI,KAAK,CAAC,qBAAqB,UAAU,EAAE,CAAC,CAAC;KACpD;IAED,MAAM,YAAY,GAAG,wBAAwB,CAAC,KAAK,CAAC,CAAC;IAErD,MAAM,UAAU,GAAG,YAAY,CAAC,UAAU;SACvC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC;SAC5B,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,oBAAoB,CAAC,cAAc,KAAK,SAAS,IAAI,oBAAoB,CAAC,cAAc,KAAK,CAAC,CAAC,CAAC;IAEpH,MAAM,UAAU,GAAG,YAAY,CAAC,UAAU,CAAC,MAAM,CAC/C,CAAC,GAAG,EAAE,EAAE,CAAC,oBAAoB,CAAC,SAAS,KAAK,SAAS,IAAI,oBAAoB,CAAC,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,CACpG,CAAC;IAEF,OAAO,UAAU,CAAC,GAAG,CAAC,CAAC,eAAe,EAAE,EAAE;QACxC,+EAA+E;QAC/E,IAAI,eAAe,KAAK,CAAC,IAAI,eAAe,CAAC,MAAM,KAAK,CAAC,EAAE;YACzD,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,CAAC;SAChC;QACD,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,GAAG,eAAe,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QACxE,MAAM,eAAe,GAAG,YAAY,CAAC,aAAa;YAChD,CAAC,CAAC,WAAW,CAAC,gBAAgB,CAAC,UAAU,EAAE,YAAY,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,CAAC;YACpF,CAAC,CAAC,WAAW,CAAC,yBAAyB,CAAC,UAAU,EAAE,YAAY,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;QAChG,MAAM,QAAQ,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,SAAS,EAAE,EAAE,CAC/C,sBAAM,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,eAAe,EAAE,SAAS,CAAC,CACnE,CAAC;QAEF,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE;YACzB,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,CAAC;SAChC;QACD,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE;YACzB,OAAO,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC;SAClC;QACD,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;IACnE,CAAC,CAAC,CAAC;AACL,CAAC;AAhDD,8DAgDC;AAED;;;;;GAKG;AACH,SAAgB,eAAe,CAC7B,WAA4B,EAC5B,UAAkB,EAClB,MAAc,EACd,uBAA6C,EAAE;IAE/C,MAAM,sBAAsB,GAAG,yBAAyB,CACtD,WAAW,EACX,UAAU,EACV,MAAM,EACN,oBAAoB,CACrB,CAAC,MAAM,CACN,CAAC,CAAC,EAAE,EAAE;IACJ,gFAAgF;IAChF,6EAA6E;IAC7E,oBAAoB,CAAC,SAAS,KAAK,SAAS;QAC5C,CAAC,CAAC,CAAC,QAAQ,KAAK,SAAS,IAAI,oBAAoB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAClF,CAAC;IAEF,OAAO,sBAAsB,CAAC,MAAM,GAAG,CAAC,IAAI,sBAAsB,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,SAAS,CAAC,CAAC;AAC5G,CAAC;AApBD,0CAoBC","sourcesContent":["import * as opcodes from 'bitcoin-ops';\n\nimport { ECPair, payments, script, Transaction, TxInput } from 'bitcoinjs-lib';\nimport * as classify from 'bitcoinjs-lib/src/classify';\nimport * as ScriptSignature from 'bitcoinjs-lib/src/script_signature';\n\nimport { Network } from '../networkTypes';\nimport * as networks from '../networks';\nimport { getMainnet } from '../coins';\nimport { UtxoTransaction } from './UtxoTransaction';\n\nconst inputTypes = [\n  'multisig',\n  'nonstandard',\n  'nulldata',\n  'pubkey',\n  'pubkeyhash',\n  'scripthash',\n  'witnesspubkeyhash',\n  'witnessscripthash',\n  'witnesscommitment',\n] as const;\n\ntype InputType = typeof inputTypes[number];\n\nexport interface ParsedSignatureScript {\n  isSegwitInput: boolean;\n  inputClassification: InputType;\n  p2shOutputClassification?: string;\n  publicKeys?: Buffer[];\n}\n\nexport interface ParsedSignatureP2PKH extends ParsedSignatureScript {\n  signatures: [Buffer];\n  publicKeys: [Buffer];\n  pubScript: Buffer;\n}\n\nexport interface ParsedSignatureScript2Of3 extends ParsedSignatureScript {\n  signatures:\n    | [Buffer, Buffer] // fully-signed transactions with signatures\n    /* Partially signed transactions with placeholder signatures.\n       For p2sh, the placeholder is OP_0 (number 0) */\n    | [Buffer | 0, Buffer | 0, Buffer | 0];\n  publicKeys: [Buffer, Buffer, Buffer];\n  pubScript: Buffer;\n}\n\nexport function getDefaultSigHash(network: Network): number {\n  switch (getMainnet(network)) {\n    case networks.bitcoincash:\n    case networks.bitcoinsv:\n    case networks.bitcoingold:\n      return Transaction.SIGHASH_ALL | UtxoTransaction.SIGHASH_FORKID;\n    default:\n      return Transaction.SIGHASH_ALL;\n  }\n}\n\n/**\n * Parse a transaction's signature script to obtain public keys, signatures, the sig script,\n * and other properties.\n *\n * Only supports script types used in BitGo transactions.\n *\n * @param input\n * @returns ParsedSignatureScript\n */\nexport function parseSignatureScript(\n  input: TxInput\n): ParsedSignatureScript | ParsedSignatureP2PKH | ParsedSignatureScript2Of3 {\n  const isSegwitInput = input.witness.length > 0;\n  const isNativeSegwitInput = input.script.length === 0;\n  let decompiledSigScript: Array<Buffer | number> | null;\n  let inputClassification: InputType;\n  if (isSegwitInput) {\n    // The decompiledSigScript is the script containing the signatures, public keys, and the script that was committed\n    // to (pubScript). If this is a segwit input the decompiledSigScript is in the witness, regardless of whether it\n    // is native or not. The inputClassification is determined based on whether or not the input is native to give an\n    // accurate classification. Note that p2shP2wsh inputs will be classified as p2sh and not p2wsh.\n    decompiledSigScript = input.witness;\n    if (isNativeSegwitInput) {\n      inputClassification = classify.witness(decompiledSigScript as Buffer[], true) as InputType;\n    } else {\n      inputClassification = classify.input(input.script, true) as InputType;\n    }\n  } else {\n    inputClassification = classify.input(input.script, true) as InputType;\n    decompiledSigScript = script.decompile(input.script);\n  }\n\n  if (!decompiledSigScript) {\n    return { isSegwitInput, inputClassification };\n  }\n\n  if (inputClassification === classify.types.P2PKH) {\n    /* istanbul ignore next */\n    if (!decompiledSigScript || decompiledSigScript.length !== 2) {\n      throw new Error('unexpected signature for p2pkh');\n    }\n    const [signature, publicKey] = decompiledSigScript;\n    /* istanbul ignore next */\n    if (!Buffer.isBuffer(signature) || !Buffer.isBuffer(publicKey)) {\n      throw new Error('unexpected signature for p2pkh');\n    }\n    const publicKeys = [publicKey];\n    const signatures: [Buffer] = [signature];\n    const pubScript = payments.p2pkh({ pubkey: publicKey }).output;\n\n    return { isSegwitInput, inputClassification, signatures, publicKeys, pubScript };\n  }\n\n  // Note the assumption here that if we have a p2sh or p2wsh input it will be multisig (appropriate because the\n  // BitGo platform only supports multisig within these types of inputs, with the exception of replay protection inputs,\n  // which are single signature p2sh). Signatures are all but the last entry in the decompiledSigScript.\n  // The redeemScript/witnessScript (depending on which type of input this is) is the last entry in\n  // the decompiledSigScript (denoted here as the pubScript). The public keys are the second through\n  // antepenultimate entries in the decompiledPubScript. See below for a visual representation of the typical 2-of-3\n  // multisig setup:\n  //\n  //   decompiledSigScript = 0 <sig1> <sig2> [<sig3>] <pubScript>\n  //   decompiledPubScript = 2 <pub1> <pub2> <pub3> 3 OP_CHECKMULTISIG\n  //\n  // Transactions built with `.build()` only have two signatures `<sig1>` and `<sig2>` in _decompiledSigScript_.\n  // Transactions built with `.buildIncomplete()` have three signatures, where missing signatures are substituted with `OP_0`.\n  const expectedScriptType =\n    inputClassification === classify.types.P2SH || inputClassification === classify.types.P2WSH;\n\n  if (!expectedScriptType) {\n    return { isSegwitInput, inputClassification };\n  }\n\n  const pubScript = decompiledSigScript[decompiledSigScript.length - 1];\n  /* istanbul ignore next */\n  if (!Buffer.isBuffer(pubScript)) {\n    throw new Error(`invalid pubScript`);\n  }\n\n  const p2shOutputClassification = classify.output(pubScript);\n\n  if (p2shOutputClassification !== 'multisig') {\n    return {\n      isSegwitInput,\n      inputClassification,\n      p2shOutputClassification,\n    };\n  }\n\n  const decompiledPubScript = script.decompile(pubScript);\n  if (decompiledPubScript === null) {\n    /* istanbul ignore next */\n    throw new Error(`could not decompile pubScript`);\n  }\n\n  const expectedScriptLength =\n    // complete transactions with 2 signatures\n    decompiledSigScript.length === 4 ||\n    // incomplete transaction with 3 signatures or signature placeholders\n    decompiledSigScript.length === 5;\n\n  if (!expectedScriptLength) {\n    return { isSegwitInput, inputClassification };\n  }\n\n  if (isSegwitInput) {\n    /* istanbul ignore next */\n    if (!Buffer.isBuffer(decompiledSigScript[0])) {\n      throw new Error(`expected decompiledSigScript[0] to be a buffer for segwit inputs`);\n    }\n    /* istanbul ignore next */\n    if (decompiledSigScript[0].length !== 0) {\n      throw new Error(`witness stack expected to start with empty buffer`);\n    }\n  } else if (decompiledSigScript[0] !== opcodes.OP_0) {\n    throw new Error(`sigScript expected to start with OP_0`);\n  }\n\n  const signatures = decompiledSigScript.slice(1 /* ignore leading OP_0 */, -1 /* ignore trailing pubScript */);\n  /* istanbul ignore next */\n  if (signatures.length !== 2 && signatures.length !== 3) {\n    throw new Error(`expected 2 or 3 signatures, got ${signatures.length}`);\n  }\n\n  /* istanbul ignore next */\n  if (decompiledPubScript.length !== 6) {\n    throw new Error(`unexpected decompiledPubScript length`);\n  }\n  const publicKeys = decompiledPubScript.slice(1, -2) as Buffer[];\n  publicKeys.forEach((b) => {\n    /* istanbul ignore next */\n    if (!Buffer.isBuffer(b)) {\n      throw new Error();\n    }\n  });\n  if (publicKeys.length !== 3) {\n    /* istanbul ignore next */\n    throw new Error(`expected 3 public keys, got ${publicKeys.length}`);\n  }\n\n  // Op codes 81 through 96 represent numbers 1 through 16 (see https://en.bitcoin.it/wiki/Script#Opcodes), which is\n  // why we subtract by 80 to get the number of signatures (n) and the number of public keys (m) in an n-of-m setup.\n  const len = decompiledPubScript.length;\n  const signatureThreshold = (decompiledPubScript[0] as number) - 80;\n  /* istanbul ignore next */\n  if (signatureThreshold !== 2) {\n    throw new Error(`expected signatureThreshold 2, got ${signatureThreshold}`);\n  }\n  const nPubKeys = (decompiledPubScript[len - 2] as number) - 80;\n  /* istanbul ignore next */\n  if (nPubKeys !== 3) {\n    throw new Error(`expected nPubKeys 3, got ${nPubKeys}`);\n  }\n\n  const lastOpCode = decompiledPubScript[len - 1];\n  /* istanbul ignore next */\n  if (lastOpCode !== opcodes.OP_CHECKMULTISIG) {\n    throw new Error(`expected opcode #${opcodes.OP_CHECKMULTISIG}, got opcode #${lastOpCode}`);\n  }\n\n  return {\n    isSegwitInput,\n    inputClassification,\n    p2shOutputClassification,\n    signatures: signatures.map((b) => {\n      if (Buffer.isBuffer(b) || b === 0) {\n        return b;\n      }\n      throw new Error(`unexpected signature element ${b}`);\n    }) as [Buffer, Buffer] | [Buffer, Buffer, Buffer],\n    publicKeys,\n    pubScript,\n  };\n}\n\nexport function parseSignatureScript2Of3(input: TxInput): ParsedSignatureScript2Of3 {\n  const result = parseSignatureScript(input) as ParsedSignatureScript2Of3;\n\n  if (![classify.types.P2WSH, classify.types.P2SH, classify.types.P2PKH].includes(result.inputClassification)) {\n    throw new Error(`unexpected inputClassification ${result.inputClassification}`);\n  }\n  if (!result.signatures) {\n    throw new Error(`missing signatures`);\n  }\n  if (result.publicKeys.length !== 3) {\n    throw new Error(`unexpected pubkey count`);\n  }\n  if (!result.pubScript || result.pubScript.length === 0) {\n    throw new Error(`pubScript missing or empty`);\n  }\n\n  return result;\n}\n\n/**\n * Constraints for signature verifications.\n * Parameters are conjunctive: if multiple parameters are set, a verification for an individual\n * signature must satisfy all of them.\n */\nexport type VerificationSettings = {\n  /**\n   * The index of the signature to verify. Only iterates over non-empty signatures.\n   */\n  signatureIndex?: number;\n  /**\n   * The hex of the public key to verify.\n   */\n  publicKey?: Buffer;\n};\n\n/**\n * Result for a individual signature verification\n */\nexport type SignatureVerification = {\n  /** Set to the public key that signed for the signature */\n  signedBy: Buffer | undefined;\n};\n\n/**\n * Get signature verifications for multsig transaction\n * @param transaction\n * @param inputIndex\n * @param amount - must be set for segwit transactions and BIP143 transactions\n * @param verificationSettings\n * @returns SignatureVerification[] - in order of parsed non-empty signatures\n */\nexport function getSignatureVerifications(\n  transaction: UtxoTransaction,\n  inputIndex: number,\n  amount: number,\n  verificationSettings: VerificationSettings = {}\n): SignatureVerification[] {\n  /* istanbul ignore next */\n  if (!transaction.ins) {\n    throw new Error(`invalid transaction`);\n  }\n\n  const input = transaction.ins[inputIndex];\n  /* istanbul ignore next */\n  if (!input) {\n    throw new Error(`no input at index ${inputIndex}`);\n  }\n\n  const parsedScript = parseSignatureScript2Of3(input);\n\n  const signatures = parsedScript.signatures\n    .filter((s) => s && s.length)\n    .filter((s, i) => verificationSettings.signatureIndex === undefined || verificationSettings.signatureIndex === i);\n\n  const publicKeys = parsedScript.publicKeys.filter(\n    (buf) => verificationSettings.publicKey === undefined || verificationSettings.publicKey.equals(buf)\n  );\n\n  return signatures.map((signatureBuffer) => {\n    // slice the last byte from the signature hash input because it's the hash type\n    if (signatureBuffer === 0 || signatureBuffer.length === 0) {\n      return { signedBy: undefined };\n    }\n    const { signature, hashType } = ScriptSignature.decode(signatureBuffer);\n    const transactionHash = parsedScript.isSegwitInput\n      ? transaction.hashForWitnessV0(inputIndex, parsedScript.pubScript, amount, hashType)\n      : transaction.hashForSignatureByNetwork(inputIndex, parsedScript.pubScript, amount, hashType);\n    const signedBy = publicKeys.filter((publicKey) =>\n      ECPair.fromPublicKey(publicKey).verify(transactionHash, signature)\n    );\n\n    if (signedBy.length === 0) {\n      return { signedBy: undefined };\n    }\n    if (signedBy.length === 1) {\n      return { signedBy: signedBy[0] };\n    }\n    throw new Error(`illegal state: signed by multiple public keys`);\n  });\n}\n\n/**\n * @param transaction\n * @param inputIndex\n * @param amount\n * @param verificationSettings - if publicKey is specified, returns true iff any signature is signed by publicKey.\n */\nexport function verifySignature(\n  transaction: UtxoTransaction,\n  inputIndex: number,\n  amount: number,\n  verificationSettings: VerificationSettings = {}\n): boolean {\n  const signatureVerifications = getSignatureVerifications(\n    transaction,\n    inputIndex,\n    amount,\n    verificationSettings\n  ).filter(\n    (v) =>\n      // If no publicKey is set in verificationSettings, all signatures must be valid.\n      // Otherwise, a single valid signature by the specified pubkey is sufficient.\n      verificationSettings.publicKey === undefined ||\n      (v.signedBy !== undefined && verificationSettings.publicKey.equals(v.signedBy))\n  );\n\n  return signatureVerifications.length > 0 && signatureVerifications.every((v) => v.signedBy !== undefined);\n}\n"]}
{
"name": "@bitgo/utxo-lib",
"version": "2.2.0-rc.2",
"version": "2.2.0-rc.3",
"description": "Client-side Bitcoin JavaScript library",

@@ -64,3 +64,3 @@ "main": "./dist/src/index.js",

"license": "MIT",
"gitHead": "68beff62d5b1bb3e9c3107caf7cdb380686f7b0e"
"gitHead": "6804f08c802d39615025a26c74519f3075b3de37"
}

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc