@bitgo-beta/abstract-utxo 1.6.1-alpha.424 → 1.6.1-alpha.426

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 (130) hide show
  1. package/dist/cjs/src/abstractUtxoCoin.d.ts +21 -5
  2. package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
  3. package/dist/cjs/src/abstractUtxoCoin.js +52 -26
  4. package/dist/cjs/src/index.d.ts +2 -2
  5. package/dist/cjs/src/index.d.ts.map +1 -1
  6. package/dist/cjs/src/index.js +3 -3
  7. package/dist/cjs/src/recovery/backupKeyRecovery.js +4 -4
  8. package/dist/cjs/src/recovery/crossChainRecovery.d.ts +10 -0
  9. package/dist/cjs/src/recovery/crossChainRecovery.d.ts.map +1 -1
  10. package/dist/cjs/src/recovery/crossChainRecovery.js +48 -4
  11. package/dist/cjs/src/transaction/decode.d.ts +11 -0
  12. package/dist/cjs/src/transaction/decode.d.ts.map +1 -0
  13. package/dist/cjs/src/transaction/decode.js +81 -0
  14. package/dist/cjs/src/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
  15. package/dist/cjs/src/transaction/descriptor/verifyTransaction.js +3 -2
  16. package/dist/cjs/src/transaction/explainTransaction.js +3 -3
  17. package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts +24 -0
  18. package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts.map +1 -0
  19. package/dist/cjs/src/transaction/fixedScript/SigningError.js +26 -0
  20. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts +1 -1
  21. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
  22. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.js +1 -1
  23. package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  24. package/dist/cjs/src/transaction/fixedScript/explainTransaction.js +2 -2
  25. package/dist/cjs/src/transaction/fixedScript/index.d.ts +3 -0
  26. package/dist/cjs/src/transaction/fixedScript/index.d.ts.map +1 -1
  27. package/dist/cjs/src/transaction/fixedScript/index.js +18 -1
  28. package/dist/cjs/src/transaction/fixedScript/musig2.d.ts +4 -0
  29. package/dist/cjs/src/transaction/fixedScript/musig2.d.ts.map +1 -0
  30. package/dist/cjs/src/transaction/fixedScript/musig2.js +3 -0
  31. package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts +7 -0
  32. package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts.map +1 -0
  33. package/dist/cjs/src/transaction/fixedScript/replayProtection.js +78 -0
  34. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts +30 -0
  35. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -0
  36. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.js +152 -0
  37. package/dist/cjs/src/transaction/fixedScript/signPsbt.d.ts +30 -0
  38. package/dist/cjs/src/transaction/fixedScript/signPsbt.d.ts.map +1 -0
  39. package/dist/cjs/src/transaction/fixedScript/signPsbt.js +174 -0
  40. package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts +22 -0
  41. package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -0
  42. package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.js +129 -0
  43. package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts +5 -6
  44. package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts.map +1 -1
  45. package/dist/cjs/src/transaction/fixedScript/signTransaction.js +28 -79
  46. package/dist/cjs/src/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
  47. package/dist/cjs/src/transaction/fixedScript/verifyTransaction.js +3 -2
  48. package/dist/cjs/src/transaction/signTransaction.d.ts.map +1 -1
  49. package/dist/cjs/src/transaction/signTransaction.js +5 -2
  50. package/dist/cjs/src/transaction/types.d.ts +5 -0
  51. package/dist/cjs/src/transaction/types.d.ts.map +1 -1
  52. package/dist/cjs/src/transaction/types.js +5 -1
  53. package/dist/cjs/test/unit/recovery/crossChainRecovery.js +24 -1
  54. package/dist/cjs/test/unit/transaction/fixedScript/explainPsbt.js +3 -3
  55. package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +2 -2
  56. package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.d.ts +2 -0
  57. package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.d.ts.map +1 -0
  58. package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.js +59 -0
  59. package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.d.ts +2 -0
  60. package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.d.ts.map +1 -0
  61. package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.js +157 -0
  62. package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +1 -1
  63. package/dist/cjs/test/unit/transaction/fixedScript/util.js +2 -1
  64. package/dist/cjs/test/unit/transaction.js +65 -42
  65. package/dist/cjs/tsconfig.tsbuildinfo +1 -1
  66. package/dist/esm/abstractUtxoCoin.d.ts +21 -5
  67. package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
  68. package/dist/esm/abstractUtxoCoin.js +53 -27
  69. package/dist/esm/index.d.ts +2 -2
  70. package/dist/esm/index.d.ts.map +1 -1
  71. package/dist/esm/index.js +3 -3
  72. package/dist/esm/recovery/backupKeyRecovery.js +2 -2
  73. package/dist/esm/recovery/crossChainRecovery.d.ts +10 -0
  74. package/dist/esm/recovery/crossChainRecovery.d.ts.map +1 -1
  75. package/dist/esm/recovery/crossChainRecovery.js +46 -3
  76. package/dist/esm/transaction/decode.d.ts +11 -0
  77. package/dist/esm/transaction/decode.d.ts.map +1 -0
  78. package/dist/esm/transaction/decode.js +43 -0
  79. package/dist/esm/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
  80. package/dist/esm/transaction/descriptor/verifyTransaction.js +3 -2
  81. package/dist/esm/transaction/explainTransaction.js +3 -3
  82. package/dist/esm/transaction/fixedScript/SigningError.d.ts +24 -0
  83. package/dist/esm/transaction/fixedScript/SigningError.d.ts.map +1 -0
  84. package/dist/esm/transaction/fixedScript/SigningError.js +21 -0
  85. package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts +1 -1
  86. package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
  87. package/dist/esm/transaction/fixedScript/explainPsbtWasm.js +1 -1
  88. package/dist/esm/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  89. package/dist/esm/transaction/fixedScript/explainTransaction.js +2 -2
  90. package/dist/esm/transaction/fixedScript/index.d.ts +3 -0
  91. package/dist/esm/transaction/fixedScript/index.d.ts.map +1 -1
  92. package/dist/esm/transaction/fixedScript/index.js +4 -1
  93. package/dist/esm/transaction/fixedScript/musig2.d.ts +4 -0
  94. package/dist/esm/transaction/fixedScript/musig2.d.ts.map +1 -0
  95. package/dist/esm/transaction/fixedScript/musig2.js +2 -0
  96. package/dist/esm/transaction/fixedScript/replayProtection.d.ts +7 -0
  97. package/dist/esm/transaction/fixedScript/replayProtection.d.ts.map +1 -0
  98. package/dist/esm/transaction/fixedScript/replayProtection.js +39 -0
  99. package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts +30 -0
  100. package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -0
  101. package/dist/esm/transaction/fixedScript/signLegacyTransaction.js +112 -0
  102. package/dist/esm/transaction/fixedScript/signPsbt.d.ts +30 -0
  103. package/dist/esm/transaction/fixedScript/signPsbt.d.ts.map +1 -0
  104. package/dist/esm/transaction/fixedScript/signPsbt.js +134 -0
  105. package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts +22 -0
  106. package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -0
  107. package/dist/esm/transaction/fixedScript/signPsbtWasm.js +122 -0
  108. package/dist/esm/transaction/fixedScript/signTransaction.d.ts +5 -6
  109. package/dist/esm/transaction/fixedScript/signTransaction.d.ts.map +1 -1
  110. package/dist/esm/transaction/fixedScript/signTransaction.js +28 -79
  111. package/dist/esm/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
  112. package/dist/esm/transaction/fixedScript/verifyTransaction.js +3 -2
  113. package/dist/esm/transaction/signTransaction.d.ts.map +1 -1
  114. package/dist/esm/transaction/signTransaction.js +5 -2
  115. package/dist/esm/transaction/types.d.ts +5 -0
  116. package/dist/esm/transaction/types.d.ts.map +1 -1
  117. package/dist/esm/transaction/types.js +4 -2
  118. package/package.json +13 -13
  119. package/dist/cjs/src/replayProtection.d.ts +0 -5
  120. package/dist/cjs/src/replayProtection.d.ts.map +0 -1
  121. package/dist/cjs/src/replayProtection.js +0 -58
  122. package/dist/cjs/src/sign.d.ts +0 -54
  123. package/dist/cjs/src/sign.d.ts.map +0 -1
  124. package/dist/cjs/src/sign.js +0 -205
  125. package/dist/esm/replayProtection.d.ts +0 -5
  126. package/dist/esm/replayProtection.d.ts.map +0 -1
  127. package/dist/esm/replayProtection.js +0 -20
  128. package/dist/esm/sign.d.ts +0 -54
  129. package/dist/esm/sign.d.ts.map +0 -1
  130. package/dist/esm/sign.js +0 -162
@@ -0,0 +1,112 @@
1
+ import assert from 'assert';
2
+ import * as utxolib from '@bitgo-beta/utxo-lib';
3
+ import { bip32 } from '@bitgo-beta/secp256k1';
4
+ import { bitgo } from '@bitgo-beta/utxo-lib';
5
+ import { isTriple } from '@bitgo-beta/sdk-core';
6
+ import debugLib from 'debug';
7
+ import { getReplayProtectionAddresses } from './replayProtection';
8
+ import { InputSigningError, TransactionSigningError } from './SigningError';
9
+ const debug = debugLib('bitgo:v2:utxo');
10
+ const { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;
11
+ /**
12
+ * Sign all inputs of a wallet transaction and verify signatures after signing.
13
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
14
+ * failed.
15
+ *
16
+ * @param transaction - wallet transaction (builder) to be signed
17
+ * @param unspents - transaction unspents
18
+ * @param walletSigner - signing parameters
19
+ * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.
20
+ * @param replayProtectionAddresses - List of replay protection addresses to skip signing
21
+ */
22
+ export function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { isLastSignature, replayProtectionAddresses, }) {
23
+ const network = transaction.network;
24
+ if (replayProtectionAddresses === undefined) {
25
+ replayProtectionAddresses = getReplayProtectionAddresses(network);
26
+ }
27
+ const prevOutputs = unspents.map((u) => toOutput(u, network));
28
+ let txBuilder;
29
+ if (transaction instanceof utxolib.bitgo.UtxoTransaction) {
30
+ txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction(transaction, prevOutputs);
31
+ if (transaction.ins.length !== unspents.length) {
32
+ throw new Error(`transaction inputs must match unspents`);
33
+ }
34
+ }
35
+ else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {
36
+ txBuilder = transaction;
37
+ }
38
+ else {
39
+ throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);
40
+ }
41
+ const signErrors = unspents
42
+ .map((unspent, inputIndex) => {
43
+ if (replayProtectionAddresses.includes(unspent.address)) {
44
+ debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);
45
+ return;
46
+ }
47
+ if (!isWalletUnspent(unspent)) {
48
+ return InputSigningError.expectedWalletUnspent(inputIndex, null, unspent);
49
+ }
50
+ try {
51
+ signInputWithUnspent(txBuilder, inputIndex, unspent, walletSigner);
52
+ debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);
53
+ }
54
+ catch (e) {
55
+ return new InputSigningError(inputIndex, null, unspent, e);
56
+ }
57
+ })
58
+ .filter((e) => e !== undefined);
59
+ const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();
60
+ const verifyErrors = signedTransaction.ins
61
+ .map((input, inputIndex) => {
62
+ const unspent = unspents[inputIndex];
63
+ if (replayProtectionAddresses.includes(unspent.address)) {
64
+ debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, unspents.length);
65
+ return;
66
+ }
67
+ if (!isWalletUnspent(unspent)) {
68
+ return InputSigningError.expectedWalletUnspent(inputIndex, null, unspent);
69
+ }
70
+ try {
71
+ const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;
72
+ if (!utxolib.bitgo.verifySignatureWithPublicKey(signedTransaction, inputIndex, prevOutputs, publicKey)) {
73
+ return new InputSigningError(inputIndex, null, unspent, new Error(`invalid signature`));
74
+ }
75
+ }
76
+ catch (e) {
77
+ debug('Invalid signature');
78
+ return new InputSigningError(inputIndex, null, unspent, e);
79
+ }
80
+ })
81
+ .filter((e) => e !== undefined);
82
+ if (signErrors.length || verifyErrors.length) {
83
+ throw new TransactionSigningError(signErrors, verifyErrors);
84
+ }
85
+ return signedTransaction;
86
+ }
87
+ export function signLegacyTransaction(tx, signerKeychain, params) {
88
+ switch (params.signingStep) {
89
+ case 'signerNonce':
90
+ case 'cosignerNonce':
91
+ /**
92
+ * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
93
+ * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
94
+ */
95
+ return tx;
96
+ }
97
+ if (tx.ins.length !== params.txInfo?.unspents?.length) {
98
+ throw new Error('length of unspents array should equal to the number of transaction inputs');
99
+ }
100
+ if (!params.pubs || !isTriple(params.pubs)) {
101
+ throw new Error(`must provide xpub array`);
102
+ }
103
+ const keychains = params.pubs.map((pub) => bip32.fromBase58(pub));
104
+ const cosignerPub = params.cosignerPub ?? params.pubs[2];
105
+ const cosignerKeychain = bip32.fromBase58(cosignerPub);
106
+ assert(signerKeychain);
107
+ const walletSigner = new bitgo.WalletUnspentSigner(keychains, signerKeychain, cosignerKeychain);
108
+ return signAndVerifyWalletTransaction(tx, params.txInfo.unspents, walletSigner, {
109
+ isLastSignature: params.isLastSignature,
110
+ });
111
+ }
112
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signLegacyTransaction.js","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signLegacyTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BIP32Interface, bip32 } from '@bitgo-beta/secp256k1';\nimport { bitgo } from '@bitgo-beta/utxo-lib';\nimport { isTriple, Triple } from '@bitgo-beta/sdk-core';\nimport debugLib from 'debug';\n\nimport { getReplayProtectionAddresses } from './replayProtection';\nimport { InputSigningError, TransactionSigningError } from './SigningError';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\n\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\n\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\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 * @param replayProtectionAddresses - List of replay protection addresses to skip signing\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  {\n    isLastSignature,\n    replayProtectionAddresses,\n  }: {\n    isLastSignature: boolean;\n    replayProtectionAddresses?: string[];\n  }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  if (replayProtectionAddresses === undefined) {\n    replayProtectionAddresses = getReplayProtectionAddresses(network);\n  }\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 (replayProtectionAddresses.includes(unspent.address)) {\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, null, 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, null, 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 (replayProtectionAddresses.includes(unspent.address)) {\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, null, 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, null, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, null, 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\nexport function signLegacyTransaction<TNumber extends number | bigint>(\n  tx: utxolib.bitgo.UtxoTransaction<TNumber>,\n  signerKeychain: BIP32Interface | undefined,\n  params: {\n    isLastSignature: boolean;\n    signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;\n    txInfo: { unspents?: utxolib.bitgo.Unspent<TNumber>[] } | undefined;\n    pubs: string[] | undefined;\n    cosignerPub: string | undefined;\n  }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  switch (params.signingStep) {\n    case 'signerNonce':\n    case 'cosignerNonce':\n      /**\n       * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).\n       * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.\n       */\n      return tx;\n  }\n\n  if (tx.ins.length !== params.txInfo?.unspents?.length) {\n    throw new Error('length of unspents array should equal to the number of transaction inputs');\n  }\n\n  if (!params.pubs || !isTriple(params.pubs)) {\n    throw new Error(`must provide xpub array`);\n  }\n\n  const keychains = params.pubs.map((pub) => bip32.fromBase58(pub)) as Triple<BIP32Interface>;\n  const cosignerPub = params.cosignerPub ?? params.pubs[2];\n  const cosignerKeychain = bip32.fromBase58(cosignerPub);\n\n  assert(signerKeychain);\n  const walletSigner = new bitgo.WalletUnspentSigner<RootWalletKeys>(keychains, signerKeychain, cosignerKeychain);\n  return signAndVerifyWalletTransaction(tx, params.txInfo.unspents, walletSigner, {\n    isLastSignature: params.isLastSignature,\n  }) as utxolib.bitgo.UtxoTransaction<TNumber>;\n}\n"]}
@@ -0,0 +1,30 @@
1
+ import * as utxolib from '@bitgo-beta/utxo-lib';
2
+ import { BIP32Interface } from '@bitgo-beta/secp256k1';
3
+ import { Musig2Participant } from './musig2';
4
+ export type PsbtParsedScriptType = 'p2sh' | 'p2wsh' | 'p2shP2wsh' | 'p2shP2pk' | 'taprootKeyPathSpend' | 'taprootScriptPathSpend' | 'p2trLegacy' | 'p2trMusig2ScriptPath' | 'p2trMusig2KeyPath';
5
+ /**
6
+ * Sign all inputs of a psbt and verify signatures after signing.
7
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
8
+ * failed.
9
+ *
10
+ * If it is the last signature, finalize and extract the transaction from the psbt.
11
+ *
12
+ * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
13
+ * using TransactionBuilder
14
+ *
15
+ * @param psbt
16
+ * @param signerKeychain
17
+ * @param isLastSignature
18
+ */
19
+ export declare function signAndVerifyPsbt(psbt: utxolib.bitgo.UtxoPsbt, signerKeychain: utxolib.BIP32Interface, { isLastSignature,
20
+ /** deprecated */
21
+ allowNonSegwitSigningWithoutPrevTx, }: {
22
+ isLastSignature: boolean;
23
+ allowNonSegwitSigningWithoutPrevTx?: boolean;
24
+ }): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>;
25
+ export declare function signPsbtWithMusig2Participant(coin: Musig2Participant<utxolib.bitgo.UtxoPsbt>, tx: utxolib.bitgo.UtxoPsbt, signerKeychain: BIP32Interface | undefined, params: {
26
+ isLastSignature: boolean;
27
+ signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
28
+ walletId: string | undefined;
29
+ }): Promise<utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>>;
30
+ //# sourceMappingURL=signPsbt.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"signPsbt.d.ts","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signPsbt.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAKvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAI7C,MAAM,MAAM,oBAAoB,GAC5B,MAAM,GACN,OAAO,GACP,WAAW,GACX,UAAU,GACV,qBAAqB,GACrB,wBAAwB,GAExB,YAAY,GACZ,sBAAsB,GACtB,mBAAmB,CAAC;AAExB;;;;;;;;;;;;;GAaG;AACH,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC5B,cAAc,EAAE,OAAO,CAAC,cAAc,EACtC,EACE,eAAe;AACf,iBAAiB;AACjB,kCAAkC,GACnC,EAAE;IAAE,eAAe,EAAE,OAAO,CAAC;IAAC,kCAAkC,CAAC,EAAE,OAAO,CAAA;CAAE,GAC5E,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,CA6DhE;AAYD,wBAAsB,6BAA6B,CACjD,IAAI,EAAE,iBAAiB,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,EAC/C,EAAE,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC1B,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,MAAM,EAAE;IACN,eAAe,EAAE,OAAO,CAAC;IACzB,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;CAC9B,GACA,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,CAsDzE"}
@@ -0,0 +1,134 @@
1
+ import assert from 'assert';
2
+ import * as utxolib from '@bitgo-beta/utxo-lib';
3
+ import { bitgo } from '@bitgo-beta/utxo-lib';
4
+ import debugLib from 'debug';
5
+ import { InputSigningError, TransactionSigningError } from './SigningError';
6
+ const debug = debugLib('bitgo:v2:utxo');
7
+ /**
8
+ * Sign all inputs of a psbt and verify signatures after signing.
9
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
10
+ * failed.
11
+ *
12
+ * If it is the last signature, finalize and extract the transaction from the psbt.
13
+ *
14
+ * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
15
+ * using TransactionBuilder
16
+ *
17
+ * @param psbt
18
+ * @param signerKeychain
19
+ * @param isLastSignature
20
+ */
21
+ export function signAndVerifyPsbt(psbt, signerKeychain, { isLastSignature,
22
+ /** deprecated */
23
+ allowNonSegwitSigningWithoutPrevTx, }) {
24
+ const txInputs = psbt.txInputs;
25
+ const outputIds = [];
26
+ const scriptTypes = [];
27
+ const signErrors = psbt.data.inputs
28
+ .map((input, inputIndex) => {
29
+ const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
30
+ outputIds.push(outputId);
31
+ const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
32
+ scriptTypes.push(scriptType);
33
+ if (scriptType === 'p2shP2pk') {
34
+ debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
35
+ return;
36
+ }
37
+ try {
38
+ psbt.signInputHD(inputIndex, signerKeychain);
39
+ debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
40
+ }
41
+ catch (e) {
42
+ return new InputSigningError(inputIndex, scriptType, { id: outputId }, e);
43
+ }
44
+ })
45
+ .filter((e) => e !== undefined);
46
+ const verifyErrors = psbt.data.inputs
47
+ .map((input, inputIndex) => {
48
+ const scriptType = scriptTypes[inputIndex];
49
+ if (scriptType === 'p2shP2pk') {
50
+ debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
51
+ return;
52
+ }
53
+ const outputId = outputIds[inputIndex];
54
+ try {
55
+ if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {
56
+ return new InputSigningError(inputIndex, scriptType, { id: outputId }, new Error(`invalid signature`));
57
+ }
58
+ }
59
+ catch (e) {
60
+ debug('Invalid signature');
61
+ return new InputSigningError(inputIndex, scriptType, { id: outputId }, e);
62
+ }
63
+ })
64
+ .filter((e) => e !== undefined);
65
+ if (signErrors.length || verifyErrors.length) {
66
+ throw new TransactionSigningError(signErrors, verifyErrors);
67
+ }
68
+ if (isLastSignature) {
69
+ psbt.finalizeAllInputs();
70
+ return psbt.extractTransaction();
71
+ }
72
+ return psbt;
73
+ }
74
+ /**
75
+ * Key Value: Unsigned tx id => PSBT
76
+ * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.
77
+ * Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.
78
+ * For more info, check SignTransactionOptions.signingStep
79
+ *
80
+ * TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.
81
+ */
82
+ const PSBT_CACHE = new Map();
83
+ export async function signPsbtWithMusig2Participant(coin, tx, signerKeychain, params) {
84
+ if (bitgo.isTransactionWithKeyPathSpendInput(tx)) {
85
+ // We can only be the first signature on a transaction with taproot key path spend inputs because
86
+ // we require the secret nonce in the cache of the first signer, which is impossible to retrieve if
87
+ // deserialized from a hex.
88
+ if (params.isLastSignature) {
89
+ throw new Error('Cannot be last signature on a transaction with key path spend inputs');
90
+ }
91
+ switch (params.signingStep) {
92
+ case 'signerNonce':
93
+ assert(signerKeychain);
94
+ tx.setAllInputsMusig2NonceHD(signerKeychain);
95
+ PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);
96
+ return tx;
97
+ case 'cosignerNonce':
98
+ assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
99
+ return await coin.getMusig2Nonces(tx, params.walletId);
100
+ case 'signerSignature':
101
+ const txId = tx.getUnsignedTx().getId();
102
+ const psbt = PSBT_CACHE.get(txId);
103
+ assert(psbt, `Psbt is missing from txCache (cache size ${PSBT_CACHE.size}).
104
+ This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`);
105
+ PSBT_CACHE.delete(txId);
106
+ tx = psbt.combine(tx);
107
+ break;
108
+ default:
109
+ // this instance is not an external signer
110
+ assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
111
+ assert(signerKeychain);
112
+ tx.setAllInputsMusig2NonceHD(signerKeychain);
113
+ const response = await coin.getMusig2Nonces(tx, params.walletId);
114
+ tx = tx.combine(response);
115
+ break;
116
+ }
117
+ }
118
+ else {
119
+ switch (params.signingStep) {
120
+ case 'signerNonce':
121
+ case 'cosignerNonce':
122
+ /**
123
+ * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
124
+ * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
125
+ */
126
+ return tx;
127
+ }
128
+ }
129
+ assert(signerKeychain);
130
+ return signAndVerifyPsbt(tx, signerKeychain, {
131
+ isLastSignature: params.isLastSignature,
132
+ });
133
+ }
134
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signPsbt.js","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signPsbt.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,QAAQ,CAAC;AAE5B,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,EAAE,KAAK,EAAE,MAAM,sBAAsB,CAAC;AAC7C,OAAO,QAAQ,MAAM,OAAO,CAAC;AAE7B,OAAO,EAAE,iBAAiB,EAAE,uBAAuB,EAAE,MAAM,gBAAgB,CAAC;AAG5E,MAAM,KAAK,GAAG,QAAQ,CAAC,eAAe,CAAC,CAAC;AAcxC;;;;;;;;;;;;;GAaG;AACH,MAAM,UAAU,iBAAiB,CAC/B,IAA4B,EAC5B,cAAsC,EACtC,EACE,eAAe;AACf,iBAAiB;AACjB,kCAAkC,GACyC;IAE7E,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;IAC/B,MAAM,SAAS,GAAa,EAAE,CAAC;IAC/B,MAAM,WAAW,GAA2B,EAAE,CAAC;IAE/C,MAAM,UAAU,GAAgC,IAAI,CAAC,IAAI,CAAC,MAAM;SAC7D,GAAG,CAAC,CAAC,KAAK,EAAE,UAAkB,EAAE,EAAE;QACjC,MAAM,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,OAAO,CAAC,KAAK,CAAC,mBAAmB,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QACvG,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEzB,MAAM,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;QAC3D,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAE7B,IAAI,UAAU,KAAK,UAAU,EAAE,CAAC;YAC9B,KAAK,CAAC,mDAAmD,EAAE,UAAU,GAAG,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACpG,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,cAAc,CAAC,CAAC;YAC7C,KAAK,CAAC,oCAAoC,EAAE,UAAU,GAAG,CAAC,EAAE,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACvF,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,IAAI,iBAAiB,CAAS,UAAU,EAAE,UAAU,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC;QACpF,CAAC;IACH,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAkC,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC;IAElE,MAAM,YAAY,GAAgC,IAAI,CAAC,IAAI,CAAC,MAAM;SAC/D,GAAG,CAAC,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;QACzB,MAAM,UAAU,GAAG,WAAW,CAAC,UAAU,CAAC,CAAC;QAC3C,IAAI,UAAU,KAAK,UAAU,EAAE,CAAC;YAC9B,KAAK,CACH,0GAA0G,EAC1G,UAAU,GAAG,CAAC,EACd,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CACxB,CAAC;YACF,OAAO;QACT,CAAC;QAED,MAAM,QAAQ,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC;QACvC,IAAI,CAAC;YACH,IAAI,CAAC,IAAI,CAAC,2BAA2B,CAAC,UAAU,EAAE,cAAc,CAAC,EAAE,CAAC;gBAClE,OAAO,IAAI,iBAAiB,CAAC,UAAU,EAAE,UAAU,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC,CAAC;YACzG,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,KAAK,CAAC,mBAAmB,CAAC,CAAC;YAC3B,OAAO,IAAI,iBAAiB,CAAS,UAAU,EAAE,UAAU,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC;QACpF,CAAC;IACH,CAAC,CAAC;SACD,MAAM,CAAC,CAAC,CAAC,EAAkC,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC;IAElE,IAAI,UAAU,CAAC,MAAM,IAAI,YAAY,CAAC,MAAM,EAAE,CAAC;QAC7C,MAAM,IAAI,uBAAuB,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;IAC9D,CAAC;IAED,IAAI,eAAe,EAAE,CAAC;QACpB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,OAAO,IAAI,CAAC,kBAAkB,EAAE,CAAC;IACnC,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,UAAU,GAAG,IAAI,GAAG,EAAkC,CAAC;AAE7D,MAAM,CAAC,KAAK,UAAU,6BAA6B,CACjD,IAA+C,EAC/C,EAA0B,EAC1B,cAA0C,EAC1C,MAIC;IAED,IAAI,KAAK,CAAC,kCAAkC,CAAC,EAAE,CAAC,EAAE,CAAC;QACjD,iGAAiG;QACjG,mGAAmG;QACnG,2BAA2B;QAC3B,IAAI,MAAM,CAAC,eAAe,EAAE,CAAC;YAC3B,MAAM,IAAI,KAAK,CAAC,sEAAsE,CAAC,CAAC;QAC1F,CAAC;QAED,QAAQ,MAAM,CAAC,WAAW,EAAE,CAAC;YAC3B,KAAK,aAAa;gBAChB,MAAM,CAAC,cAAc,CAAC,CAAC;gBACvB,EAAE,CAAC,yBAAyB,CAAC,cAAc,CAAC,CAAC;gBAC7C,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,aAAa,EAAE,CAAC,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC;gBAC/C,OAAO,EAAE,CAAC;YACZ,KAAK,eAAe;gBAClB,MAAM,CAAC,MAAM,CAAC,QAAQ,EAAE,6CAA6C,CAAC,CAAC;gBACvE,OAAO,MAAM,IAAI,CAAC,eAAe,CAAC,EAAE,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;YACzD,KAAK,iBAAiB;gBACpB,MAAM,IAAI,GAAG,EAAE,CAAC,aAAa,EAAE,CAAC,KAAK,EAAE,CAAC;gBACxC,MAAM,IAAI,GAAG,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;gBAClC,MAAM,CACJ,IAAI,EACJ,4CAA4C,UAAU,CAAC,IAAI;mIAC8D,CAC1H,CAAC;gBACF,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;gBACxB,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACtB,MAAM;YACR;gBACE,0CAA0C;gBAC1C,MAAM,CAAC,MAAM,CAAC,QAAQ,EAAE,6CAA6C,CAAC,CAAC;gBACvE,MAAM,CAAC,cAAc,CAAC,CAAC;gBACvB,EAAE,CAAC,yBAAyB,CAAC,cAAc,CAAC,CAAC;gBAC7C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,EAAE,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;gBACjE,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;gBAC1B,MAAM;QACV,CAAC;IACH,CAAC;SAAM,CAAC;QACN,QAAQ,MAAM,CAAC,WAAW,EAAE,CAAC;YAC3B,KAAK,aAAa,CAAC;YACnB,KAAK,eAAe;gBAClB;;;mBAGG;gBACH,OAAO,EAAE,CAAC;QACd,CAAC;IACH,CAAC;IAED,MAAM,CAAC,cAAc,CAAC,CAAC;IACvB,OAAO,iBAAiB,CAAC,EAAE,EAAE,cAAc,EAAE;QAC3C,eAAe,EAAE,MAAM,CAAC,eAAe;KACxC,CAAC,CAAC;AACL,CAAC","sourcesContent":["import assert from 'assert';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BIP32Interface } from '@bitgo-beta/secp256k1';\nimport { bitgo } from '@bitgo-beta/utxo-lib';\nimport debugLib from 'debug';\n\nimport { InputSigningError, TransactionSigningError } from './SigningError';\nimport { Musig2Participant } from './musig2';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nexport type PsbtParsedScriptType =\n  | 'p2sh'\n  | 'p2wsh'\n  | 'p2shP2wsh'\n  | 'p2shP2pk'\n  | 'taprootKeyPathSpend'\n  | 'taprootScriptPathSpend'\n  // wasm-utxo types\n  | 'p2trLegacy'\n  | 'p2trMusig2ScriptPath'\n  | 'p2trMusig2KeyPath';\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    /** deprecated */\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: PsbtParsedScriptType[] = [];\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        psbt.signInputHD(inputIndex, signerKeychain);\n        debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);\n      } catch (e) {\n        return new InputSigningError<bigint>(inputIndex, scriptType, { 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 (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {\n          return new InputSigningError(inputIndex, scriptType, { id: outputId }, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<bigint>(inputIndex, scriptType, { 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 * Key Value: Unsigned tx id => PSBT\n * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.\n * Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.\n * For more info, check SignTransactionOptions.signingStep\n *\n * TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.\n */\nconst PSBT_CACHE = new Map<string, utxolib.bitgo.UtxoPsbt>();\n\nexport async function signPsbtWithMusig2Participant(\n  coin: Musig2Participant<utxolib.bitgo.UtxoPsbt>,\n  tx: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: BIP32Interface | undefined,\n  params: {\n    isLastSignature: boolean;\n    signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;\n    walletId: string | undefined;\n  }\n): Promise<utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>> {\n  if (bitgo.isTransactionWithKeyPathSpendInput(tx)) {\n    // We can only be the first signature on a transaction with taproot key path spend inputs because\n    // we require the secret nonce in the cache of the first signer, which is impossible to retrieve if\n    // deserialized from a hex.\n    if (params.isLastSignature) {\n      throw new Error('Cannot be last signature on a transaction with key path spend inputs');\n    }\n\n    switch (params.signingStep) {\n      case 'signerNonce':\n        assert(signerKeychain);\n        tx.setAllInputsMusig2NonceHD(signerKeychain);\n        PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);\n        return tx;\n      case 'cosignerNonce':\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        return await coin.getMusig2Nonces(tx, params.walletId);\n      case 'signerSignature':\n        const txId = tx.getUnsignedTx().getId();\n        const psbt = PSBT_CACHE.get(txId);\n        assert(\n          psbt,\n          `Psbt is missing from txCache (cache size ${PSBT_CACHE.size}).\n            This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`\n        );\n        PSBT_CACHE.delete(txId);\n        tx = psbt.combine(tx);\n        break;\n      default:\n        // this instance is not an external signer\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        assert(signerKeychain);\n        tx.setAllInputsMusig2NonceHD(signerKeychain);\n        const response = await coin.getMusig2Nonces(tx, params.walletId);\n        tx = tx.combine(response);\n        break;\n    }\n  } else {\n    switch (params.signingStep) {\n      case 'signerNonce':\n      case 'cosignerNonce':\n        /**\n         * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).\n         * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.\n         */\n        return tx;\n    }\n  }\n\n  assert(signerKeychain);\n  return signAndVerifyPsbt(tx, signerKeychain, {\n    isLastSignature: params.isLastSignature,\n  });\n}\n"]}
@@ -0,0 +1,22 @@
1
+ import { BIP32Interface } from '@bitgo-beta/utxo-lib';
2
+ import { ECPair, fixedScriptWallet } from '@bitgo/wasm-utxo';
3
+ import { Musig2Participant } from './musig2';
4
+ export type ReplayProtectionKeys = {
5
+ publicKeys: (Uint8Array | ECPair)[];
6
+ };
7
+ /**
8
+ * Sign all inputs of a PSBT and verify signatures after signing.
9
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them failed.
10
+ *
11
+ * If it is the last signature, finalize and extract the transaction from the psbt.
12
+ */
13
+ export declare function signAndVerifyPsbtWasm(tx: fixedScriptWallet.BitGoPsbt, signerKeychain: BIP32Interface, rootWalletKeys: fixedScriptWallet.RootWalletKeys, replayProtection: ReplayProtectionKeys, { isLastSignature }: {
14
+ isLastSignature: boolean;
15
+ }): fixedScriptWallet.BitGoPsbt | Buffer;
16
+ export declare function signPsbtWithMusig2ParticipantWasm(coin: Musig2Participant<fixedScriptWallet.BitGoPsbt>, tx: fixedScriptWallet.BitGoPsbt, signerKeychain: BIP32Interface | undefined, rootWalletKeys: fixedScriptWallet.RootWalletKeys, params: {
17
+ replayProtection: ReplayProtectionKeys;
18
+ isLastSignature: boolean;
19
+ signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
20
+ walletId: string | undefined;
21
+ }): Promise<fixedScriptWallet.BitGoPsbt | Buffer>;
22
+ //# sourceMappingURL=signPsbtWasm.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"signPsbtWasm.d.ts","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signPsbtWasm.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAS,MAAM,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AAGpE,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAE7C,MAAM,MAAM,oBAAoB,GAAG;IACjC,UAAU,EAAE,CAAC,UAAU,GAAG,MAAM,CAAC,EAAE,CAAC;CACrC,CAAC;AAmBF;;;;;GAKG;AACH,wBAAgB,qBAAqB,CACnC,EAAE,EAAE,iBAAiB,CAAC,SAAS,EAC/B,cAAc,EAAE,cAAc,EAC9B,cAAc,EAAE,iBAAiB,CAAC,cAAc,EAChD,gBAAgB,EAAE,oBAAoB,EACtC,EAAE,eAAe,EAAE,EAAE;IAAE,eAAe,EAAE,OAAO,CAAA;CAAE,GAChD,iBAAiB,CAAC,SAAS,GAAG,MAAM,CAkDtC;AAOD,wBAAsB,iCAAiC,CACrD,IAAI,EAAE,iBAAiB,CAAC,iBAAiB,CAAC,SAAS,CAAC,EACpD,EAAE,EAAE,iBAAiB,CAAC,SAAS,EAC/B,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,cAAc,EAAE,iBAAiB,CAAC,cAAc,EAChD,MAAM,EAAE;IACN,gBAAgB,EAAE,oBAAoB,CAAC;IACvC,eAAe,EAAE,OAAO,CAAC;IACzB,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;CAC9B,GACA,OAAO,CAAC,iBAAiB,CAAC,SAAS,GAAG,MAAM,CAAC,CA0D/C"}
@@ -0,0 +1,122 @@
1
+ import assert from 'assert';
2
+ import { BIP32 } from '@bitgo/wasm-utxo';
3
+ import { InputSigningError, TransactionSigningError } from './SigningError';
4
+ /**
5
+ * Key Value: Unsigned tx id => PSBT
6
+ * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.
7
+ * Reason: MuSig2 signer secure nonce is cached in the BitGoPsbt object. It will be required during the signing step.
8
+ * For more info, check SignTransactionOptions.signingStep
9
+ */
10
+ const PSBT_CACHE_WASM = new Map();
11
+ function hasKeyPathSpendInput(tx, rootWalletKeys, replayProtection) {
12
+ const parsed = tx.parseTransactionWithWalletKeys(rootWalletKeys, replayProtection);
13
+ return parsed.inputs.some((input) => input.scriptType === 'p2trMusig2KeyPath');
14
+ }
15
+ /**
16
+ * Sign all inputs of a PSBT and verify signatures after signing.
17
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them failed.
18
+ *
19
+ * If it is the last signature, finalize and extract the transaction from the psbt.
20
+ */
21
+ export function signAndVerifyPsbtWasm(tx, signerKeychain, rootWalletKeys, replayProtection, { isLastSignature }) {
22
+ const wasmSigner = toWasmBIP32(signerKeychain);
23
+ const parsed = tx.parseTransactionWithWalletKeys(rootWalletKeys, replayProtection);
24
+ const signErrors = [];
25
+ const verifyErrors = [];
26
+ // Sign all inputs (skipping replay protection inputs)
27
+ parsed.inputs.forEach((input, inputIndex) => {
28
+ if (input.scriptType === 'p2shP2pk') {
29
+ // Skip replay protection inputs - they are platform signed only
30
+ return;
31
+ }
32
+ const outputId = `${input.previousOutput.txid}:${input.previousOutput.vout}`;
33
+ try {
34
+ tx.sign(inputIndex, wasmSigner);
35
+ }
36
+ catch (e) {
37
+ signErrors.push(new InputSigningError(inputIndex, input.scriptType, { id: outputId }, e));
38
+ }
39
+ });
40
+ // Verify signatures for all signed inputs
41
+ parsed.inputs.forEach((input, inputIndex) => {
42
+ if (input.scriptType === 'p2shP2pk') {
43
+ return;
44
+ }
45
+ const outputId = `${input.previousOutput.txid}:${input.previousOutput.vout}`;
46
+ try {
47
+ if (!tx.verifySignature(inputIndex, wasmSigner)) {
48
+ verifyErrors.push(new InputSigningError(inputIndex, input.scriptType, { id: outputId }, new Error('invalid signature')));
49
+ }
50
+ }
51
+ catch (e) {
52
+ verifyErrors.push(new InputSigningError(inputIndex, input.scriptType, { id: outputId }, e));
53
+ }
54
+ });
55
+ if (signErrors.length || verifyErrors.length) {
56
+ throw new TransactionSigningError(signErrors, verifyErrors);
57
+ }
58
+ if (isLastSignature) {
59
+ tx.finalizeAllInputs();
60
+ return Buffer.from(tx.extractTransaction());
61
+ }
62
+ return tx;
63
+ }
64
+ function toWasmBIP32(key) {
65
+ // Convert using base58 string to ensure private key is properly transferred
66
+ return BIP32.fromBase58(key.toBase58());
67
+ }
68
+ export async function signPsbtWithMusig2ParticipantWasm(coin, tx, signerKeychain, rootWalletKeys, params) {
69
+ const wasmSigner = signerKeychain ? toWasmBIP32(signerKeychain) : undefined;
70
+ if (hasKeyPathSpendInput(tx, rootWalletKeys, params.replayProtection)) {
71
+ // We can only be the first signature on a transaction with taproot key path spend inputs because
72
+ // we require the secret nonce in the cache of the first signer, which is impossible to retrieve if
73
+ // deserialized from a hex.
74
+ if (params.isLastSignature) {
75
+ throw new Error('Cannot be last signature on a transaction with key path spend inputs');
76
+ }
77
+ switch (params.signingStep) {
78
+ case 'signerNonce':
79
+ assert(wasmSigner);
80
+ tx.generateMusig2Nonces(wasmSigner);
81
+ PSBT_CACHE_WASM.set(tx.unsignedTxid(), tx);
82
+ return tx;
83
+ case 'cosignerNonce':
84
+ assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
85
+ return await coin.getMusig2Nonces(tx, params.walletId);
86
+ case 'signerSignature': {
87
+ const txId = tx.unsignedTxid();
88
+ const cachedPsbt = PSBT_CACHE_WASM.get(txId);
89
+ assert(cachedPsbt, `Psbt is missing from txCache (cache size ${PSBT_CACHE_WASM.size}).
90
+ This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`);
91
+ PSBT_CACHE_WASM.delete(txId);
92
+ cachedPsbt.combineMusig2Nonces(tx);
93
+ tx = cachedPsbt;
94
+ break;
95
+ }
96
+ default:
97
+ // this instance is not an external signer
98
+ assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
99
+ assert(wasmSigner);
100
+ tx.generateMusig2Nonces(wasmSigner);
101
+ const response = await coin.getMusig2Nonces(tx, params.walletId);
102
+ tx.combineMusig2Nonces(response);
103
+ break;
104
+ }
105
+ }
106
+ else {
107
+ switch (params.signingStep) {
108
+ case 'signerNonce':
109
+ case 'cosignerNonce':
110
+ /**
111
+ * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
112
+ * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
113
+ */
114
+ return tx;
115
+ }
116
+ }
117
+ assert(signerKeychain);
118
+ return signAndVerifyPsbtWasm(tx, signerKeychain, rootWalletKeys, params.replayProtection, {
119
+ isLastSignature: params.isLastSignature,
120
+ });
121
+ }
122
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signPsbtWasm.js","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signPsbtWasm.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport { BIP32Interface } from '@bitgo-beta/utxo-lib';\nimport { BIP32, ECPair, fixedScriptWallet } from '@bitgo/wasm-utxo';\n\nimport { InputSigningError, TransactionSigningError } from './SigningError';\nimport { Musig2Participant } from './musig2';\n\nexport type ReplayProtectionKeys = {\n  publicKeys: (Uint8Array | ECPair)[];\n};\n\n/**\n * Key Value: Unsigned tx id => PSBT\n * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.\n * Reason: MuSig2 signer secure nonce is cached in the BitGoPsbt object. It will be required during the signing step.\n * For more info, check SignTransactionOptions.signingStep\n */\nconst PSBT_CACHE_WASM = new Map<string, fixedScriptWallet.BitGoPsbt>();\n\nfunction hasKeyPathSpendInput(\n  tx: fixedScriptWallet.BitGoPsbt,\n  rootWalletKeys: fixedScriptWallet.RootWalletKeys,\n  replayProtection: ReplayProtectionKeys\n): boolean {\n  const parsed = tx.parseTransactionWithWalletKeys(rootWalletKeys, replayProtection);\n  return parsed.inputs.some((input) => input.scriptType === 'p2trMusig2KeyPath');\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 failed.\n *\n * If it is the last signature, finalize and extract the transaction from the psbt.\n */\nexport function signAndVerifyPsbtWasm(\n  tx: fixedScriptWallet.BitGoPsbt,\n  signerKeychain: BIP32Interface,\n  rootWalletKeys: fixedScriptWallet.RootWalletKeys,\n  replayProtection: ReplayProtectionKeys,\n  { isLastSignature }: { isLastSignature: boolean }\n): fixedScriptWallet.BitGoPsbt | Buffer {\n  const wasmSigner = toWasmBIP32(signerKeychain);\n  const parsed = tx.parseTransactionWithWalletKeys(rootWalletKeys, replayProtection);\n\n  const signErrors: InputSigningError<bigint>[] = [];\n  const verifyErrors: InputSigningError<bigint>[] = [];\n\n  // Sign all inputs (skipping replay protection inputs)\n  parsed.inputs.forEach((input, inputIndex) => {\n    if (input.scriptType === 'p2shP2pk') {\n      // Skip replay protection inputs - they are platform signed only\n      return;\n    }\n\n    const outputId = `${input.previousOutput.txid}:${input.previousOutput.vout}`;\n    try {\n      tx.sign(inputIndex, wasmSigner);\n    } catch (e) {\n      signErrors.push(new InputSigningError<bigint>(inputIndex, input.scriptType, { id: outputId }, e));\n    }\n  });\n\n  // Verify signatures for all signed inputs\n  parsed.inputs.forEach((input, inputIndex) => {\n    if (input.scriptType === 'p2shP2pk') {\n      return;\n    }\n\n    const outputId = `${input.previousOutput.txid}:${input.previousOutput.vout}`;\n    try {\n      if (!tx.verifySignature(inputIndex, wasmSigner)) {\n        verifyErrors.push(\n          new InputSigningError(inputIndex, input.scriptType, { id: outputId }, new Error('invalid signature'))\n        );\n      }\n    } catch (e) {\n      verifyErrors.push(new InputSigningError<bigint>(inputIndex, input.scriptType, { id: outputId }, e));\n    }\n  });\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  if (isLastSignature) {\n    tx.finalizeAllInputs();\n    return Buffer.from(tx.extractTransaction());\n  }\n\n  return tx;\n}\n\nfunction toWasmBIP32(key: BIP32Interface): BIP32 {\n  // Convert using base58 string to ensure private key is properly transferred\n  return BIP32.fromBase58(key.toBase58());\n}\n\nexport async function signPsbtWithMusig2ParticipantWasm(\n  coin: Musig2Participant<fixedScriptWallet.BitGoPsbt>,\n  tx: fixedScriptWallet.BitGoPsbt,\n  signerKeychain: BIP32Interface | undefined,\n  rootWalletKeys: fixedScriptWallet.RootWalletKeys,\n  params: {\n    replayProtection: ReplayProtectionKeys;\n    isLastSignature: boolean;\n    signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;\n    walletId: string | undefined;\n  }\n): Promise<fixedScriptWallet.BitGoPsbt | Buffer> {\n  const wasmSigner = signerKeychain ? toWasmBIP32(signerKeychain) : undefined;\n\n  if (hasKeyPathSpendInput(tx, rootWalletKeys, params.replayProtection)) {\n    // We can only be the first signature on a transaction with taproot key path spend inputs because\n    // we require the secret nonce in the cache of the first signer, which is impossible to retrieve if\n    // deserialized from a hex.\n    if (params.isLastSignature) {\n      throw new Error('Cannot be last signature on a transaction with key path spend inputs');\n    }\n\n    switch (params.signingStep) {\n      case 'signerNonce':\n        assert(wasmSigner);\n        tx.generateMusig2Nonces(wasmSigner);\n        PSBT_CACHE_WASM.set(tx.unsignedTxid(), tx);\n        return tx;\n      case 'cosignerNonce':\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        return await coin.getMusig2Nonces(tx, params.walletId);\n      case 'signerSignature': {\n        const txId = tx.unsignedTxid();\n        const cachedPsbt = PSBT_CACHE_WASM.get(txId);\n        assert(\n          cachedPsbt,\n          `Psbt is missing from txCache (cache size ${PSBT_CACHE_WASM.size}).\n            This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`\n        );\n        PSBT_CACHE_WASM.delete(txId);\n        cachedPsbt.combineMusig2Nonces(tx);\n        tx = cachedPsbt;\n        break;\n      }\n      default:\n        // this instance is not an external signer\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        assert(wasmSigner);\n        tx.generateMusig2Nonces(wasmSigner);\n        const response = await coin.getMusig2Nonces(tx, params.walletId);\n        tx.combineMusig2Nonces(response);\n        break;\n    }\n  } else {\n    switch (params.signingStep) {\n      case 'signerNonce':\n      case 'cosignerNonce':\n        /**\n         * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).\n         * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.\n         */\n        return tx;\n    }\n  }\n\n  assert(signerKeychain);\n  return signAndVerifyPsbtWasm(tx, signerKeychain, rootWalletKeys, params.replayProtection, {\n    isLastSignature: params.isLastSignature,\n  });\n}\n"]}
@@ -1,10 +1,11 @@
1
1
  import { BIP32Interface } from '@bitgo-beta/secp256k1';
2
2
  import * as utxolib from '@bitgo-beta/utxo-lib';
3
- import { AbstractUtxoCoin, DecodedTransaction } from '../../abstractUtxoCoin';
4
- export declare function signTransaction<TNumber extends number | bigint>(coin: AbstractUtxoCoin, tx: DecodedTransaction<TNumber>, signerKeychain: BIP32Interface | undefined, params: {
3
+ import { fixedScriptWallet } from '@bitgo/wasm-utxo';
4
+ import { Musig2Participant } from './musig2';
5
+ export declare function signTransaction<T extends utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint | number> | fixedScriptWallet.BitGoPsbt>(coin: Musig2Participant<utxolib.bitgo.UtxoPsbt> | Musig2Participant<fixedScriptWallet.BitGoPsbt>, tx: T, signerKeychain: BIP32Interface | undefined, network: utxolib.Network, params: {
5
6
  walletId: string | undefined;
6
7
  txInfo: {
7
- unspents?: utxolib.bitgo.Unspent<TNumber>[];
8
+ unspents?: utxolib.bitgo.Unspent<bigint | number>[];
8
9
  } | undefined;
9
10
  isLastSignature: boolean;
10
11
  signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
@@ -12,7 +13,5 @@ export declare function signTransaction<TNumber extends number | bigint>(coin: A
12
13
  allowNonSegwitSigningWithoutPrevTx: boolean;
13
14
  pubs: string[] | undefined;
14
15
  cosignerPub: string | undefined;
15
- }): Promise<{
16
- txHex: string;
17
- }>;
16
+ }): Promise<utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint | number> | fixedScriptWallet.BitGoPsbt | Buffer>;
18
17
  //# sourceMappingURL=signTransaction.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"signTransaction.d.ts","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signTransaction.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAS,MAAM,uBAAuB,CAAC;AAE9D,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAIhD,OAAO,EAAE,gBAAgB,EAAE,kBAAkB,EAAkB,MAAM,wBAAwB,CAAC;AAY9F,wBAAsB,eAAe,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACnE,IAAI,EAAE,gBAAgB,EACtB,EAAE,EAAE,kBAAkB,CAAC,OAAO,CAAC,EAC/B,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,MAAM,EAAE;IACN,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;IAC7B,MAAM,EAAE;QAAE,QAAQ,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAA;KAAE,GAAG,SAAS,CAAC;IACpE,eAAe,EAAE,OAAO,CAAC;IACzB,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,iBAAiB;IACjB,kCAAkC,EAAE,OAAO,CAAC;IAC5C,IAAI,EAAE,MAAM,EAAE,GAAG,SAAS,CAAC;IAC3B,WAAW,EAAE,MAAM,GAAG,SAAS,CAAC;CACjC,GACA,OAAO,CAAC;IAAE,KAAK,EAAE,MAAM,CAAA;CAAE,CAAC,CAuF5B"}
1
+ {"version":3,"file":"signTransaction.d.ts","sourceRoot":"","sources":["../../../../src/transaction/fixedScript/signTransaction.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAEvD,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AAErD,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAM7C,wBAAsB,eAAe,CACnC,CAAC,SAAS,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,GAAG,MAAM,CAAC,GAAG,iBAAiB,CAAC,SAAS,EAE/G,IAAI,EAAE,iBAAiB,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,iBAAiB,CAAC,iBAAiB,CAAC,SAAS,CAAC,EAChG,EAAE,EAAE,CAAC,EACL,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,OAAO,EAAE,OAAO,CAAC,OAAO,EACxB,MAAM,EAAE;IACN,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;IAC7B,MAAM,EAAE;QAAE,QAAQ,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,MAAM,CAAC,EAAE,CAAA;KAAE,GAAG,SAAS,CAAC;IAC5E,eAAe,EAAE,OAAO,CAAC;IACzB,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,iBAAiB;IACjB,kCAAkC,EAAE,OAAO,CAAC;IAC5C,IAAI,EAAE,MAAM,EAAE,GAAG,SAAS,CAAC;IAC3B,WAAW,EAAE,MAAM,GAAG,SAAS,CAAC;CACjC,GACA,OAAO,CACR,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,GAAG,MAAM,CAAC,GAAG,iBAAiB,CAAC,SAAS,GAAG,MAAM,CAC/G,CAwCA"}