@bitgo-beta/abstract-utxo 1.6.1-alpha.14 → 1.6.1-alpha.141

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (32) hide show
  1. package/CHANGELOG.md +753 -0
  2. package/dist/src/abstractUtxoCoin.d.ts +110 -21
  3. package/dist/src/abstractUtxoCoin.d.ts.map +1 -1
  4. package/dist/src/abstractUtxoCoin.js +284 -174
  5. package/dist/src/index.js +6 -2
  6. package/dist/src/parseOutput.d.ts.map +1 -1
  7. package/dist/src/parseOutput.js +12 -1
  8. package/dist/src/recovery/RecoveryProvider.d.ts +6 -5
  9. package/dist/src/recovery/RecoveryProvider.d.ts.map +1 -1
  10. package/dist/src/recovery/RecoveryProvider.js +1 -2
  11. package/dist/src/recovery/backupKeyRecovery.d.ts +30 -9
  12. package/dist/src/recovery/backupKeyRecovery.d.ts.map +1 -1
  13. package/dist/src/recovery/backupKeyRecovery.js +108 -83
  14. package/dist/src/recovery/crossChainRecovery.d.ts +12 -23
  15. package/dist/src/recovery/crossChainRecovery.d.ts.map +1 -1
  16. package/dist/src/recovery/crossChainRecovery.js +59 -69
  17. package/dist/src/recovery/index.d.ts +0 -1
  18. package/dist/src/recovery/index.d.ts.map +1 -1
  19. package/dist/src/recovery/index.js +6 -3
  20. package/dist/src/recovery/mempoolApi.d.ts.map +1 -1
  21. package/dist/src/recovery/mempoolApi.js +6 -3
  22. package/dist/src/sign.d.ts +27 -3
  23. package/dist/src/sign.d.ts.map +1 -1
  24. package/dist/src/sign.js +70 -4
  25. package/dist/src/transaction.d.ts +36 -0
  26. package/dist/src/transaction.d.ts.map +1 -0
  27. package/dist/src/transaction.js +236 -0
  28. package/dist/tsconfig.tsbuildinfo +1 -7736
  29. package/package.json +10 -10
  30. package/dist/src/recovery/smartbitApi.d.ts +0 -11
  31. package/dist/src/recovery/smartbitApi.d.ts.map +0 -1
  32. package/dist/src/recovery/smartbitApi.js +0 -36
package/dist/src/sign.js CHANGED
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.signAndVerifyWalletTransaction = exports.TransactionSigningError = exports.InputSigningError = void 0;
3
+ exports.signAndVerifyWalletTransaction = exports.signAndVerifyPsbt = exports.TransactionSigningError = exports.InputSigningError = void 0;
4
4
  /**
5
5
  * @prettier
6
6
  */
@@ -28,6 +28,72 @@ class TransactionSigningError extends Error {
28
28
  }
29
29
  }
30
30
  exports.TransactionSigningError = TransactionSigningError;
31
+ /**
32
+ * Sign all inputs of a psbt and verify signatures after signing.
33
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
34
+ * failed.
35
+ *
36
+ * If it is the last signature, finalize and extract the transaction from the psbt.
37
+ *
38
+ * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
39
+ * using TransactionBuilder
40
+ *
41
+ * @param psbt
42
+ * @param signerKeychain
43
+ * @param isLastSignature
44
+ */
45
+ function signAndVerifyPsbt(psbt, signerKeychain, { isLastSignature, allowNonSegwitSigningWithoutPrevTx, }) {
46
+ const txInputs = psbt.txInputs;
47
+ const outputIds = [];
48
+ const scriptTypes = [];
49
+ const signErrors = psbt.data.inputs
50
+ .map((input, inputIndex) => {
51
+ const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
52
+ outputIds.push(outputId);
53
+ const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
54
+ scriptTypes.push(scriptType);
55
+ if (scriptType === 'p2shP2pk') {
56
+ debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
57
+ return;
58
+ }
59
+ try {
60
+ utxolib.bitgo.withUnsafeNonSegwit(psbt, () => psbt.signInputHD(inputIndex, signerKeychain), !!allowNonSegwitSigningWithoutPrevTx);
61
+ debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
62
+ }
63
+ catch (e) {
64
+ return new InputSigningError(inputIndex, { id: outputId }, e);
65
+ }
66
+ })
67
+ .filter((e) => e !== undefined);
68
+ const verifyErrors = psbt.data.inputs
69
+ .map((input, inputIndex) => {
70
+ const scriptType = scriptTypes[inputIndex];
71
+ if (scriptType === 'p2shP2pk') {
72
+ debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
73
+ return;
74
+ }
75
+ const outputId = outputIds[inputIndex];
76
+ try {
77
+ if (!utxolib.bitgo.withUnsafeNonSegwit(psbt, () => psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain), !!allowNonSegwitSigningWithoutPrevTx)) {
78
+ return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));
79
+ }
80
+ }
81
+ catch (e) {
82
+ debug('Invalid signature');
83
+ return new InputSigningError(inputIndex, { id: outputId }, e);
84
+ }
85
+ })
86
+ .filter((e) => e !== undefined);
87
+ if (signErrors.length || verifyErrors.length) {
88
+ throw new TransactionSigningError(signErrors, verifyErrors);
89
+ }
90
+ if (isLastSignature) {
91
+ psbt.finalizeAllInputs();
92
+ return psbt.extractTransaction();
93
+ }
94
+ return psbt;
95
+ }
96
+ exports.signAndVerifyPsbt = signAndVerifyPsbt;
31
97
  /**
32
98
  * Sign all inputs of a wallet transaction and verify signatures after signing.
33
99
  * Collects and logs signing errors and verification errors, throws error in the end if any of them
@@ -56,7 +122,7 @@ function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { i
56
122
  }
57
123
  const signErrors = unspents
58
124
  .map((unspent, inputIndex) => {
59
- if (replayProtection_1.isReplayProtectionUnspent(unspent, network)) {
125
+ if ((0, replayProtection_1.isReplayProtectionUnspent)(unspent, network)) {
60
126
  debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);
61
127
  return;
62
128
  }
@@ -76,7 +142,7 @@ function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { i
76
142
  const verifyErrors = signedTransaction.ins
77
143
  .map((input, inputIndex) => {
78
144
  const unspent = unspents[inputIndex];
79
- if (replayProtection_1.isReplayProtectionUnspent(unspent, network)) {
145
+ if ((0, replayProtection_1.isReplayProtectionUnspent)(unspent, network)) {
80
146
  debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, unspents.length);
81
147
  return;
82
148
  }
@@ -101,4 +167,4 @@ function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { i
101
167
  return signedTransaction;
102
168
  }
103
169
  exports.signAndVerifyWalletTransaction = signAndVerifyWalletTransaction;
104
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign.js","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\n\nimport * as debugLib from 'debug';\n\nimport { isReplayProtectionUnspent } from './replayProtection';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nexport class InputSigningError<TNumber extends number | bigint = number> extends Error {\n  static expectedWalletUnspent<TNumber extends number | bigint>(\n    inputIndex: number,\n    unspent: Unspent<TNumber>\n  ): InputSigningError<TNumber> {\n    return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);\n  }\n\n  constructor(public inputIndex: number, public unspent: Unspent<TNumber>, public reason: Error | string) {\n    super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);\n  }\n}\n\nexport class TransactionSigningError<TNumber extends number | bigint = number> extends Error {\n  constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]) {\n    super(\n      `sign errors at inputs: [${signErrors.join(',')}], ` +\n        `verify errors at inputs: [${verifyError.join(',')}], see log for details`\n    );\n  }\n}\n\n/**\n * Sign all inputs of a wallet transaction and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * @param transaction - wallet transaction (builder) to be signed\n * @param unspents - transaction unspents\n * @param walletSigner - signing parameters\n * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.\n */\nexport function signAndVerifyWalletTransaction<TNumber extends number | bigint>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>,\n  unspents: Unspent<TNumber>[],\n  walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  { isLastSignature }: { isLastSignature: boolean }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  const prevOutputs = unspents.map((u) => toOutput(u, network));\n\n  let txBuilder: utxolib.bitgo.UtxoTransactionBuilder<TNumber>;\n  if (transaction instanceof utxolib.bitgo.UtxoTransaction) {\n    txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction<TNumber>(transaction, prevOutputs);\n    if (transaction.ins.length !== unspents.length) {\n      throw new Error(`transaction inputs must match unspents`);\n    }\n  } else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {\n    txBuilder = transaction;\n  } else {\n    throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);\n  }\n\n  const signErrors: InputSigningError<TNumber>[] = unspents\n    .map((unspent: Unspent<TNumber>, inputIndex: number) => {\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        signInputWithUnspent<TNumber>(txBuilder, inputIndex, unspent, walletSigner);\n        debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);\n      } catch (e) {\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();\n\n  const verifyErrors: InputSigningError<TNumber>[] = signedTransaction.ins\n    .map((input, inputIndex) => {\n      const unspent = unspents[inputIndex] as Unspent<TNumber>;\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          unspents.length\n        );\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;\n        if (\n          !utxolib.bitgo.verifySignatureWithPublicKey<TNumber>(signedTransaction, inputIndex, prevOutputs, publicKey)\n        ) {\n          return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return signedTransaction;\n}\n"]}
170
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sign.js","sourceRoot":"","sources":["../../src/sign.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\n\nimport * as debugLib from 'debug';\n\nimport { isReplayProtectionUnspent } from './replayProtection';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\ntype PsbtParsedScriptTypes =\n  | 'p2sh'\n  | 'p2wsh'\n  | 'p2shP2wsh'\n  | 'p2shP2pk'\n  | 'taprootKeyPathSpend'\n  | 'taprootScriptPathSpend';\n\nexport class InputSigningError<TNumber extends number | bigint = number> extends Error {\n  static expectedWalletUnspent<TNumber extends number | bigint>(\n    inputIndex: number,\n    unspent: Unspent<TNumber> | { id: string }\n  ): InputSigningError<TNumber> {\n    return new InputSigningError(inputIndex, unspent, `not a wallet unspent, not a replay protection unspent`);\n  }\n\n  constructor(\n    public inputIndex: number,\n    public unspent: Unspent<TNumber> | { id: string },\n    public reason: Error | string\n  ) {\n    super(`signing error at input ${inputIndex}: unspentId=${unspent.id}: ${reason}`);\n  }\n}\n\nexport class TransactionSigningError<TNumber extends number | bigint = number> extends Error {\n  constructor(signErrors: InputSigningError<TNumber>[], verifyError: InputSigningError<TNumber>[]) {\n    super(\n      `sign errors at inputs: [${signErrors.join(',')}], ` +\n        `verify errors at inputs: [${verifyError.join(',')}], see log for details`\n    );\n  }\n}\n\n/**\n * Sign all inputs of a psbt and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * If it is the last signature, finalize and extract the transaction from the psbt.\n *\n * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of\n * using TransactionBuilder\n *\n * @param psbt\n * @param signerKeychain\n * @param isLastSignature\n */\nexport function signAndVerifyPsbt(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: utxolib.BIP32Interface,\n  {\n    isLastSignature,\n    allowNonSegwitSigningWithoutPrevTx,\n  }: { isLastSignature: boolean; allowNonSegwitSigningWithoutPrevTx?: boolean }\n): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint> {\n  const txInputs = psbt.txInputs;\n  const outputIds: string[] = [];\n  const scriptTypes: PsbtParsedScriptTypes[] = [];\n\n  const signErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex: number) => {\n      const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));\n      outputIds.push(outputId);\n\n      const { scriptType } = utxolib.bitgo.parsePsbtInput(input);\n      scriptTypes.push(scriptType);\n\n      if (scriptType === 'p2shP2pk') {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);\n        return;\n      }\n\n      try {\n        utxolib.bitgo.withUnsafeNonSegwit(\n          psbt,\n          () => psbt.signInputHD(inputIndex, signerKeychain),\n          !!allowNonSegwitSigningWithoutPrevTx\n        );\n        debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);\n      } catch (e) {\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  const verifyErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex) => {\n      const scriptType = scriptTypes[inputIndex];\n      if (scriptType === 'p2shP2pk') {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          psbt.data.inputs.length\n        );\n        return;\n      }\n\n      const outputId = outputIds[inputIndex];\n      try {\n        if (\n          !utxolib.bitgo.withUnsafeNonSegwit(\n            psbt,\n            () => psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain),\n            !!allowNonSegwitSigningWithoutPrevTx\n          )\n        ) {\n          return new InputSigningError(inputIndex, { id: outputId }, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<bigint>(inputIndex, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  if (isLastSignature) {\n    psbt.finalizeAllInputs();\n    return psbt.extractTransaction();\n  }\n\n  return psbt;\n}\n\n/**\n * Sign all inputs of a wallet transaction and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * @param transaction - wallet transaction (builder) to be signed\n * @param unspents - transaction unspents\n * @param walletSigner - signing parameters\n * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.\n */\nexport function signAndVerifyWalletTransaction<TNumber extends number | bigint>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>,\n  unspents: Unspent<TNumber>[],\n  walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  { isLastSignature }: { isLastSignature: boolean }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  const prevOutputs = unspents.map((u) => toOutput(u, network));\n\n  let txBuilder: utxolib.bitgo.UtxoTransactionBuilder<TNumber>;\n  if (transaction instanceof utxolib.bitgo.UtxoTransaction) {\n    txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction<TNumber>(transaction, prevOutputs);\n    if (transaction.ins.length !== unspents.length) {\n      throw new Error(`transaction inputs must match unspents`);\n    }\n  } else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {\n    txBuilder = transaction;\n  } else {\n    throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);\n  }\n\n  const signErrors: InputSigningError<TNumber>[] = unspents\n    .map((unspent: Unspent<TNumber>, inputIndex: number) => {\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        signInputWithUnspent<TNumber>(txBuilder, inputIndex, unspent, walletSigner);\n        debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);\n      } catch (e) {\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();\n\n  const verifyErrors: InputSigningError<TNumber>[] = signedTransaction.ins\n    .map((input, inputIndex) => {\n      const unspent = unspents[inputIndex] as Unspent<TNumber>;\n      if (isReplayProtectionUnspent<TNumber>(unspent, network)) {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          unspents.length\n        );\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, unspent);\n      }\n      try {\n        const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;\n        if (\n          !utxolib.bitgo.verifySignatureWithPublicKey<TNumber>(signedTransaction, inputIndex, prevOutputs, publicKey)\n        ) {\n          return new InputSigningError(inputIndex, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return signedTransaction;\n}\n"]}
@@ -0,0 +1,36 @@
1
+ import * as utxolib from '@bitgo-beta/utxo-lib';
2
+ import { BitGoBase, IRequestTracer } from '@bitgo-beta/sdk-core';
3
+ import { AbstractUtxoCoin, ExplainTransactionOptions, TransactionExplanation, TransactionPrebuild } from './abstractUtxoCoin';
4
+ /**
5
+ * Get the inputs for a psbt from a prebuild.
6
+ */
7
+ export declare function getPsbtTxInputs(psbtArg: string | utxolib.bitgo.UtxoPsbt, network: utxolib.Network): {
8
+ address: string;
9
+ value: bigint;
10
+ valueString: string;
11
+ }[];
12
+ /**
13
+ * Get the inputs for a transaction from a prebuild.
14
+ */
15
+ export declare function getTxInputs<TNumber extends number | bigint>(params: {
16
+ txPrebuild: TransactionPrebuild<TNumber>;
17
+ bitgo: BitGoBase;
18
+ coin: AbstractUtxoCoin;
19
+ disableNetworking: boolean;
20
+ reqId?: IRequestTracer;
21
+ }): Promise<{
22
+ address: string;
23
+ value: TNumber;
24
+ valueString: string;
25
+ }[]>;
26
+ /**
27
+ * Decompose a raw psbt into useful information, such as the total amounts,
28
+ * change amounts, and transaction outputs.
29
+ */
30
+ export declare function explainPsbt<TNumber extends number | bigint>(params: ExplainTransactionOptions<TNumber>, network: utxolib.Network): TransactionExplanation;
31
+ /**
32
+ * Decompose a raw transaction into useful information, such as the total amounts,
33
+ * change amounts, and transaction outputs.
34
+ */
35
+ export declare function explainTx<TNumber extends number | bigint>(params: ExplainTransactionOptions<TNumber>, coin: AbstractUtxoCoin): TransactionExplanation;
36
+ //# sourceMappingURL=transaction.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transaction.d.ts","sourceRoot":"","sources":["../../src/transaction.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,SAAS,EAAE,cAAc,EAAU,MAAM,sBAAsB,CAAC;AACzE,OAAO,EACL,gBAAgB,EAChB,yBAAyB,EAEzB,sBAAsB,EACtB,mBAAmB,EACpB,MAAM,oBAAoB,CAAC;AAG5B;;GAEG;AACH,wBAAgB,eAAe,CAC7B,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,QAAQ,EACxC,OAAO,EAAE,OAAO,CAAC,OAAO,GACvB;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,WAAW,EAAE,MAAM,CAAA;CAAE,EAAE,CAyB3D;AAED;;GAEG;AACH,wBAAsB,WAAW,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAAE,MAAM,EAAE;IACzE,UAAU,EAAE,mBAAmB,CAAC,OAAO,CAAC,CAAC;IACzC,KAAK,EAAE,SAAS,CAAC;IACjB,IAAI,EAAE,gBAAgB,CAAC;IACvB,iBAAiB,EAAE,OAAO,CAAC;IAC3B,KAAK,CAAC,EAAE,cAAc,CAAC;CACxB,GAAG,OAAO,CAAC;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,OAAO,CAAC;IAAC,WAAW,EAAE,MAAM,CAAA;CAAE,EAAE,CAAC,CAoCtE;AAwGD;;;GAGG;AACH,wBAAgB,WAAW,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACzD,MAAM,EAAE,yBAAyB,CAAC,OAAO,CAAC,EAC1C,OAAO,EAAE,OAAO,CAAC,OAAO,GACvB,sBAAsB,CA8CxB;AAED;;;GAGG;AACH,wBAAgB,SAAS,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACvD,MAAM,EAAE,yBAAyB,CAAC,OAAO,CAAC,EAC1C,IAAI,EAAE,gBAAgB,GACrB,sBAAsB,CAexB"}
@@ -0,0 +1,236 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.explainTx = exports.explainPsbt = exports.getTxInputs = exports.getPsbtTxInputs = void 0;
4
+ const utxolib = require("@bitgo-beta/utxo-lib");
5
+ const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
6
+ /**
7
+ * Get the inputs for a psbt from a prebuild.
8
+ */
9
+ function getPsbtTxInputs(psbtArg, network) {
10
+ const psbt = psbtArg instanceof utxolib.bitgo.UtxoPsbt ? psbtArg : utxolib.bitgo.createPsbtFromHex(psbtArg, network);
11
+ const txInputs = psbt.txInputs;
12
+ return psbt.data.inputs.map((input, index) => {
13
+ let address;
14
+ let value;
15
+ if (input.witnessUtxo) {
16
+ address = utxolib.address.fromOutputScript(input.witnessUtxo.script, network);
17
+ value = input.witnessUtxo.value;
18
+ }
19
+ else if (input.nonWitnessUtxo) {
20
+ const tx = utxolib.bitgo.createTransactionFromBuffer(input.nonWitnessUtxo, network, {
21
+ amountType: 'bigint',
22
+ });
23
+ const txId = Buffer.from(txInputs[index].hash).reverse().toString('hex');
24
+ if (tx.getId() !== txId) {
25
+ throw new Error('input transaction hex does not match id');
26
+ }
27
+ const prevTxOutputIndex = txInputs[index].index;
28
+ address = utxolib.address.fromOutputScript(tx.outs[prevTxOutputIndex].script, network);
29
+ value = tx.outs[prevTxOutputIndex].value;
30
+ }
31
+ else {
32
+ throw new Error('psbt input is missing both witnessUtxo and nonWitnessUtxo');
33
+ }
34
+ return { address, value, valueString: value.toString() };
35
+ });
36
+ }
37
+ exports.getPsbtTxInputs = getPsbtTxInputs;
38
+ /**
39
+ * Get the inputs for a transaction from a prebuild.
40
+ */
41
+ async function getTxInputs(params) {
42
+ const { txPrebuild, bitgo, coin, disableNetworking, reqId } = params;
43
+ if (!txPrebuild.txHex) {
44
+ throw new Error(`txPrebuild.txHex not set`);
45
+ }
46
+ const transaction = coin.createTransactionFromHex(txPrebuild.txHex);
47
+ const transactionCache = {};
48
+ return await Promise.all(transaction.ins.map(async (currentInput) => {
49
+ var _a, _b;
50
+ const transactionId = Buffer.from(currentInput.hash).reverse().toString('hex');
51
+ const txHex = (_b = (_a = txPrebuild.txInfo) === null || _a === void 0 ? void 0 : _a.txHexes) === null || _b === void 0 ? void 0 : _b[transactionId];
52
+ if (txHex) {
53
+ const localTx = coin.createTransactionFromHex(txHex);
54
+ if (localTx.getId() !== transactionId) {
55
+ throw new Error('input transaction hex does not match id');
56
+ }
57
+ const currentOutput = localTx.outs[currentInput.index];
58
+ const address = utxolib.address.fromOutputScript(currentOutput.script, coin.network);
59
+ return {
60
+ address,
61
+ value: currentOutput.value,
62
+ valueString: currentOutput.value.toString(),
63
+ };
64
+ }
65
+ else if (!transactionCache[transactionId]) {
66
+ if (disableNetworking) {
67
+ throw new Error('attempting to retrieve transaction details externally with networking disabled');
68
+ }
69
+ if (reqId) {
70
+ bitgo.setRequestTracer(reqId);
71
+ }
72
+ transactionCache[transactionId] = await bitgo.get(coin.url(`/public/tx/${transactionId}`)).result();
73
+ }
74
+ const transactionDetails = transactionCache[transactionId];
75
+ return transactionDetails.outputs[currentInput.index];
76
+ }));
77
+ }
78
+ exports.getTxInputs = getTxInputs;
79
+ function explainCommon(tx, params, network) {
80
+ var _a;
81
+ const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];
82
+ let spendAmount = BigInt(0);
83
+ let changeAmount = BigInt(0);
84
+ const changeOutputs = [];
85
+ const outputs = [];
86
+ const { changeAddresses = [] } = (_a = params.txInfo) !== null && _a !== void 0 ? _a : {};
87
+ tx.outs.forEach((currentOutput) => {
88
+ const currentAddress = utxolib.address.fromOutputScript(currentOutput.script, network);
89
+ const currentAmount = BigInt(currentOutput.value);
90
+ if (changeAddresses.includes(currentAddress)) {
91
+ // this is change
92
+ changeAmount += currentAmount;
93
+ changeOutputs.push({
94
+ address: currentAddress,
95
+ amount: currentAmount.toString(),
96
+ });
97
+ return;
98
+ }
99
+ spendAmount += currentAmount;
100
+ outputs.push({
101
+ address: currentAddress,
102
+ amount: currentAmount.toString(),
103
+ });
104
+ });
105
+ const outputDetails = {
106
+ outputAmount: spendAmount.toString(),
107
+ changeAmount: changeAmount.toString(),
108
+ outputs,
109
+ changeOutputs,
110
+ };
111
+ let fee;
112
+ let locktime;
113
+ if (params.feeInfo) {
114
+ displayOrder.push('fee');
115
+ fee = params.feeInfo;
116
+ }
117
+ if (Number.isInteger(tx.locktime) && tx.locktime > 0) {
118
+ displayOrder.push('locktime');
119
+ locktime = tx.locktime;
120
+ }
121
+ return { displayOrder, id: tx.getId(), ...outputDetails, fee, locktime };
122
+ }
123
+ function getRootWalletKeys(params) {
124
+ var _a;
125
+ const keys = (_a = params.pubs) === null || _a === void 0 ? void 0 : _a.map((xpub) => utxo_lib_1.bip32.fromBase58(xpub));
126
+ return keys && keys.length === 3 ? new utxo_lib_1.bitgo.RootWalletKeys(keys) : undefined;
127
+ }
128
+ function getPsbtInputSignaturesCount(psbt, params) {
129
+ const rootWalletKeys = getRootWalletKeys(params);
130
+ return rootWalletKeys
131
+ ? utxo_lib_1.bitgo.getSignatureValidationArrayPsbt(psbt, rootWalletKeys).map((sv) => sv[1].filter((v) => v).length)
132
+ : Array(psbt.data.inputs.length).fill(0);
133
+ }
134
+ function getTxInputSignaturesCount(tx, params, network) {
135
+ var _a, _b, _c;
136
+ const prevOutputs = (_b = (_a = params.txInfo) === null || _a === void 0 ? void 0 : _a.unspents) === null || _b === void 0 ? void 0 : _b.map((u) => utxo_lib_1.bitgo.toOutput(u, network));
137
+ const rootWalletKeys = getRootWalletKeys(params);
138
+ const { unspents = [] } = (_c = params.txInfo) !== null && _c !== void 0 ? _c : {};
139
+ // get the number of signatures per input
140
+ return tx.ins.map((input, idx) => {
141
+ if (unspents.length !== tx.ins.length) {
142
+ return 0;
143
+ }
144
+ if (!prevOutputs) {
145
+ throw new Error(`invalid state`);
146
+ }
147
+ if (!rootWalletKeys) {
148
+ // no pub keys or incorrect number of pub keys
149
+ return 0;
150
+ }
151
+ try {
152
+ return utxo_lib_1.bitgo.verifySignatureWithUnspent(tx, idx, unspents, rootWalletKeys).filter((v) => v).length;
153
+ }
154
+ catch (e) {
155
+ // some other error occurred and we can't validate the signatures
156
+ return 0;
157
+ }
158
+ });
159
+ }
160
+ /**
161
+ * Decompose a raw psbt into useful information, such as the total amounts,
162
+ * change amounts, and transaction outputs.
163
+ */
164
+ function explainPsbt(params, network) {
165
+ const { txHex } = params;
166
+ let psbt;
167
+ try {
168
+ psbt = utxo_lib_1.bitgo.createPsbtFromHex(txHex, network);
169
+ }
170
+ catch (e) {
171
+ throw new Error('failed to parse psbt hex');
172
+ }
173
+ const txOutputs = psbt.txOutputs;
174
+ function getChangeAddresses() {
175
+ try {
176
+ return utxolib.bitgo
177
+ .findInternalOutputIndices(psbt)
178
+ .map((i) => utxolib.address.fromOutputScript(txOutputs[i].script, network));
179
+ }
180
+ catch (e) {
181
+ if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {
182
+ return [];
183
+ }
184
+ throw e;
185
+ }
186
+ }
187
+ const changeAddresses = getChangeAddresses();
188
+ const tx = psbt.getUnsignedTx();
189
+ const common = explainCommon(tx, { ...params, txInfo: { ...params.txInfo, changeAddresses } }, network);
190
+ const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);
191
+ // Set fee from subtracting inputs from outputs
192
+ const outputAmount = txOutputs.reduce((cumulative, curr) => cumulative + BigInt(curr.value), BigInt(0));
193
+ const inputAmount = psbt.txInputs.reduce((cumulative, txInput, i) => {
194
+ const data = psbt.data.inputs[i];
195
+ if (data.witnessUtxo) {
196
+ return cumulative + BigInt(data.witnessUtxo.value);
197
+ }
198
+ else if (data.nonWitnessUtxo) {
199
+ const tx = utxo_lib_1.bitgo.createTransactionFromBuffer(data.nonWitnessUtxo, network, { amountType: 'bigint' });
200
+ return cumulative + BigInt(tx.outs[txInput.index].value);
201
+ }
202
+ else {
203
+ throw new Error('could not find value on input');
204
+ }
205
+ }, BigInt(0));
206
+ return {
207
+ ...common,
208
+ fee: (inputAmount - outputAmount).toString(),
209
+ inputSignatures: inputSignaturesCount,
210
+ signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),
211
+ };
212
+ }
213
+ exports.explainPsbt = explainPsbt;
214
+ /**
215
+ * Decompose a raw transaction into useful information, such as the total amounts,
216
+ * change amounts, and transaction outputs.
217
+ */
218
+ function explainTx(params, coin) {
219
+ const { txHex } = params;
220
+ let tx;
221
+ try {
222
+ tx = coin.createTransactionFromHex(txHex);
223
+ }
224
+ catch (e) {
225
+ throw new Error('failed to parse transaction hex');
226
+ }
227
+ const common = explainCommon(tx, params, coin.network);
228
+ const inputSignaturesCount = getTxInputSignaturesCount(tx, params, coin.network);
229
+ return {
230
+ ...common,
231
+ inputSignatures: inputSignaturesCount,
232
+ signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),
233
+ };
234
+ }
235
+ exports.explainTx = explainTx;
236
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../src/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BitGoBase, IRequestTracer, Triple } from '@bitgo-beta/sdk-core';\nimport {\n  AbstractUtxoCoin,\n  ExplainTransactionOptions,\n  Output,\n  TransactionExplanation,\n  TransactionPrebuild,\n} from './abstractUtxoCoin';\nimport { bip32, BIP32Interface, bitgo } from '@bitgo-beta/utxo-lib';\n\n/**\n * Get the inputs for a psbt from a prebuild.\n */\nexport function getPsbtTxInputs(\n  psbtArg: string | utxolib.bitgo.UtxoPsbt,\n  network: utxolib.Network\n): { address: string; value: bigint; valueString: string }[] {\n  const psbt = psbtArg instanceof utxolib.bitgo.UtxoPsbt ? psbtArg : utxolib.bitgo.createPsbtFromHex(psbtArg, network);\n  const txInputs = psbt.txInputs;\n  return psbt.data.inputs.map((input, index) => {\n    let address: string;\n    let value: bigint;\n    if (input.witnessUtxo) {\n      address = utxolib.address.fromOutputScript(input.witnessUtxo.script, network);\n      value = input.witnessUtxo.value;\n    } else if (input.nonWitnessUtxo) {\n      const tx = utxolib.bitgo.createTransactionFromBuffer<bigint>(input.nonWitnessUtxo, network, {\n        amountType: 'bigint',\n      });\n      const txId = (Buffer.from(txInputs[index].hash).reverse() as Buffer).toString('hex');\n      if (tx.getId() !== txId) {\n        throw new Error('input transaction hex does not match id');\n      }\n      const prevTxOutputIndex = txInputs[index].index;\n      address = utxolib.address.fromOutputScript(tx.outs[prevTxOutputIndex].script, network);\n      value = tx.outs[prevTxOutputIndex].value;\n    } else {\n      throw new Error('psbt input is missing both witnessUtxo and nonWitnessUtxo');\n    }\n    return { address, value, valueString: value.toString() };\n  });\n}\n\n/**\n * Get the inputs for a transaction from a prebuild.\n */\nexport async function getTxInputs<TNumber extends number | bigint>(params: {\n  txPrebuild: TransactionPrebuild<TNumber>;\n  bitgo: BitGoBase;\n  coin: AbstractUtxoCoin;\n  disableNetworking: boolean;\n  reqId?: IRequestTracer;\n}): Promise<{ address: string; value: TNumber; valueString: string }[]> {\n  const { txPrebuild, bitgo, coin, disableNetworking, reqId } = params;\n  if (!txPrebuild.txHex) {\n    throw new Error(`txPrebuild.txHex not set`);\n  }\n  const transaction = coin.createTransactionFromHex<TNumber>(txPrebuild.txHex);\n  const transactionCache = {};\n  return await Promise.all(\n    transaction.ins.map(async (currentInput): Promise<{ address: string; value: TNumber; valueString: string }> => {\n      const transactionId = (Buffer.from(currentInput.hash).reverse() as Buffer).toString('hex');\n      const txHex = txPrebuild.txInfo?.txHexes?.[transactionId];\n      if (txHex) {\n        const localTx = coin.createTransactionFromHex<TNumber>(txHex);\n        if (localTx.getId() !== transactionId) {\n          throw new Error('input transaction hex does not match id');\n        }\n        const currentOutput = localTx.outs[currentInput.index];\n        const address = utxolib.address.fromOutputScript(currentOutput.script, coin.network);\n        return {\n          address,\n          value: currentOutput.value,\n          valueString: currentOutput.value.toString(),\n        };\n      } else if (!transactionCache[transactionId]) {\n        if (disableNetworking) {\n          throw new Error('attempting to retrieve transaction details externally with networking disabled');\n        }\n        if (reqId) {\n          bitgo.setRequestTracer(reqId);\n        }\n        transactionCache[transactionId] = await bitgo.get(coin.url(`/public/tx/${transactionId}`)).result();\n      }\n      const transactionDetails = transactionCache[transactionId];\n      return transactionDetails.outputs[currentInput.index];\n    })\n  );\n}\n\nfunction explainCommon<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: ExplainTransactionOptions<TNumber>,\n  network: utxolib.Network\n) {\n  const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];\n  let spendAmount = BigInt(0);\n  let changeAmount = BigInt(0);\n  const changeOutputs: Output[] = [];\n  const outputs: Output[] = [];\n\n  const { changeAddresses = [] } = params.txInfo ?? {};\n\n  tx.outs.forEach((currentOutput) => {\n    const currentAddress = utxolib.address.fromOutputScript(currentOutput.script, network);\n    const currentAmount = BigInt(currentOutput.value);\n\n    if (changeAddresses.includes(currentAddress)) {\n      // this is change\n      changeAmount += currentAmount;\n      changeOutputs.push({\n        address: currentAddress,\n        amount: currentAmount.toString(),\n      });\n      return;\n    }\n\n    spendAmount += currentAmount;\n    outputs.push({\n      address: currentAddress,\n      amount: currentAmount.toString(),\n    });\n  });\n\n  const outputDetails = {\n    outputAmount: spendAmount.toString(),\n    changeAmount: changeAmount.toString(),\n    outputs,\n    changeOutputs,\n  };\n\n  let fee: string | undefined;\n  let locktime: number | undefined;\n\n  if (params.feeInfo) {\n    displayOrder.push('fee');\n    fee = params.feeInfo;\n  }\n\n  if (Number.isInteger(tx.locktime) && tx.locktime > 0) {\n    displayOrder.push('locktime');\n    locktime = tx.locktime;\n  }\n\n  return { displayOrder, id: tx.getId(), ...outputDetails, fee, locktime };\n}\n\nfunction getRootWalletKeys<TNumber extends number | bigint>(params: ExplainTransactionOptions<TNumber>) {\n  const keys = params.pubs?.map((xpub) => bip32.fromBase58(xpub));\n  return keys && keys.length === 3 ? new bitgo.RootWalletKeys(keys as Triple<BIP32Interface>) : undefined;\n}\n\nfunction getPsbtInputSignaturesCount<TNumber extends number | bigint>(\n  psbt: bitgo.UtxoPsbt,\n  params: ExplainTransactionOptions<TNumber>\n) {\n  const rootWalletKeys = getRootWalletKeys(params);\n  return rootWalletKeys\n    ? bitgo.getSignatureValidationArrayPsbt(psbt, rootWalletKeys).map((sv) => sv[1].filter((v) => v).length)\n    : (Array(psbt.data.inputs.length) as number[]).fill(0);\n}\n\nfunction getTxInputSignaturesCount<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: ExplainTransactionOptions<TNumber>,\n  network: utxolib.Network\n) {\n  const prevOutputs = params.txInfo?.unspents?.map((u) => bitgo.toOutput<TNumber>(u, network));\n  const rootWalletKeys = getRootWalletKeys(params);\n  const { unspents = [] } = params.txInfo ?? {};\n\n  // get the number of signatures per input\n  return tx.ins.map((input, idx): number => {\n    if (unspents.length !== tx.ins.length) {\n      return 0;\n    }\n    if (!prevOutputs) {\n      throw new Error(`invalid state`);\n    }\n    if (!rootWalletKeys) {\n      // no pub keys or incorrect number of pub keys\n      return 0;\n    }\n    try {\n      return bitgo.verifySignatureWithUnspent<TNumber>(tx, idx, unspents, rootWalletKeys).filter((v) => v).length;\n    } catch (e) {\n      // some other error occurred and we can't validate the signatures\n      return 0;\n    }\n  });\n}\n\n/**\n * Decompose a raw psbt into useful information, such as the total amounts,\n * change amounts, and transaction outputs.\n */\nexport function explainPsbt<TNumber extends number | bigint>(\n  params: ExplainTransactionOptions<TNumber>,\n  network: utxolib.Network\n): TransactionExplanation {\n  const { txHex } = params;\n  let psbt: bitgo.UtxoPsbt;\n  try {\n    psbt = bitgo.createPsbtFromHex(txHex, network);\n  } catch (e) {\n    throw new Error('failed to parse psbt hex');\n  }\n  const txOutputs = psbt.txOutputs;\n  function getChangeAddresses() {\n    try {\n      return utxolib.bitgo\n        .findInternalOutputIndices(psbt)\n        .map((i) => utxolib.address.fromOutputScript(txOutputs[i].script, network));\n    } catch (e) {\n      if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {\n        return [];\n      }\n      throw e;\n    }\n  }\n  const changeAddresses = getChangeAddresses();\n  const tx = psbt.getUnsignedTx() as bitgo.UtxoTransaction<TNumber>;\n  const common = explainCommon(tx, { ...params, txInfo: { ...params.txInfo, changeAddresses } }, network);\n  const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);\n\n  // Set fee from subtracting inputs from outputs\n  const outputAmount = txOutputs.reduce((cumulative, curr) => cumulative + BigInt(curr.value), BigInt(0));\n  const inputAmount = psbt.txInputs.reduce((cumulative, txInput, i) => {\n    const data = psbt.data.inputs[i];\n    if (data.witnessUtxo) {\n      return cumulative + BigInt(data.witnessUtxo.value);\n    } else if (data.nonWitnessUtxo) {\n      const tx = bitgo.createTransactionFromBuffer<bigint>(data.nonWitnessUtxo, network, { amountType: 'bigint' });\n      return cumulative + BigInt(tx.outs[txInput.index].value);\n    } else {\n      throw new Error('could not find value on input');\n    }\n  }, BigInt(0));\n\n  return {\n    ...common,\n    fee: (inputAmount - outputAmount).toString(),\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n  } as TransactionExplanation;\n}\n\n/**\n * Decompose a raw transaction into useful information, such as the total amounts,\n * change amounts, and transaction outputs.\n */\nexport function explainTx<TNumber extends number | bigint>(\n  params: ExplainTransactionOptions<TNumber>,\n  coin: AbstractUtxoCoin\n): TransactionExplanation {\n  const { txHex } = params;\n  let tx;\n  try {\n    tx = coin.createTransactionFromHex(txHex);\n  } catch (e) {\n    throw new Error('failed to parse transaction hex');\n  }\n  const common = explainCommon(tx, params, coin.network);\n  const inputSignaturesCount = getTxInputSignaturesCount(tx, params, coin.network);\n  return {\n    ...common,\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n  } as TransactionExplanation;\n}\n"]}