@bitgo-beta/abstract-utxo 1.6.1-alpha.421 → 1.6.1-alpha.422

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 (60) hide show
  1. package/dist/cjs/src/abstractUtxoCoin.d.ts +1 -8
  2. package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
  3. package/dist/cjs/src/abstractUtxoCoin.js +4 -18
  4. package/dist/cjs/src/address/fixedScript.d.ts +2 -4
  5. package/dist/cjs/src/address/fixedScript.d.ts.map +1 -1
  6. package/dist/cjs/src/address/fixedScript.js +11 -6
  7. package/dist/cjs/src/keychains.d.ts +6 -2
  8. package/dist/cjs/src/keychains.d.ts.map +1 -1
  9. package/dist/cjs/src/keychains.js +33 -1
  10. package/dist/cjs/src/recovery/backupKeyRecovery.d.ts.map +1 -1
  11. package/dist/cjs/src/recovery/backupKeyRecovery.js +20 -13
  12. package/dist/cjs/src/transaction/descriptor/explainPsbt.d.ts +2 -2
  13. package/dist/cjs/src/transaction/descriptor/explainPsbt.d.ts.map +1 -1
  14. package/dist/cjs/src/transaction/descriptor/explainPsbt.js +1 -1
  15. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts +1 -0
  16. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
  17. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.js +42 -18
  18. package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts +14 -5
  19. package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  20. package/dist/cjs/src/transaction/fixedScript/explainTransaction.js +61 -38
  21. package/dist/cjs/src/transaction/fixedScript/parseOutput.d.ts.map +1 -1
  22. package/dist/cjs/src/transaction/fixedScript/parseOutput.js +9 -6
  23. package/dist/cjs/src/transaction/fixedScript/parseTransaction.d.ts.map +1 -1
  24. package/dist/cjs/src/transaction/fixedScript/parseTransaction.js +53 -43
  25. package/dist/cjs/src/transaction/recipient.d.ts.map +1 -1
  26. package/dist/cjs/src/transaction/recipient.js +2 -2
  27. package/dist/cjs/test/unit/recovery/backupKeyRecovery.js +2 -1
  28. package/dist/cjs/test/unit/transaction/fixedScript/explainPsbt.js +40 -16
  29. package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +100 -28
  30. package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts +3 -0
  31. package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +1 -0
  32. package/dist/cjs/test/unit/transaction/fixedScript/util.js +46 -0
  33. package/dist/cjs/tsconfig.tsbuildinfo +1 -1
  34. package/dist/esm/abstractUtxoCoin.d.ts +1 -8
  35. package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
  36. package/dist/esm/abstractUtxoCoin.js +4 -18
  37. package/dist/esm/address/fixedScript.d.ts +2 -4
  38. package/dist/esm/address/fixedScript.d.ts.map +1 -1
  39. package/dist/esm/address/fixedScript.js +12 -7
  40. package/dist/esm/keychains.d.ts +6 -2
  41. package/dist/esm/keychains.d.ts.map +1 -1
  42. package/dist/esm/keychains.js +32 -1
  43. package/dist/esm/recovery/backupKeyRecovery.d.ts.map +1 -1
  44. package/dist/esm/recovery/backupKeyRecovery.js +20 -13
  45. package/dist/esm/transaction/descriptor/explainPsbt.d.ts +2 -2
  46. package/dist/esm/transaction/descriptor/explainPsbt.d.ts.map +1 -1
  47. package/dist/esm/transaction/descriptor/explainPsbt.js +1 -1
  48. package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts +1 -0
  49. package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
  50. package/dist/esm/transaction/fixedScript/explainPsbtWasm.js +42 -18
  51. package/dist/esm/transaction/fixedScript/explainTransaction.d.ts +14 -5
  52. package/dist/esm/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  53. package/dist/esm/transaction/fixedScript/explainTransaction.js +61 -38
  54. package/dist/esm/transaction/fixedScript/parseOutput.d.ts.map +1 -1
  55. package/dist/esm/transaction/fixedScript/parseOutput.js +9 -6
  56. package/dist/esm/transaction/fixedScript/parseTransaction.d.ts.map +1 -1
  57. package/dist/esm/transaction/fixedScript/parseTransaction.js +53 -43
  58. package/dist/esm/transaction/recipient.d.ts.map +1 -1
  59. package/dist/esm/transaction/recipient.js +2 -2
  60. package/package.json +11 -11
@@ -44,8 +44,52 @@ const keychains_1 = require("../../keychains");
44
44
  const outputDifference_1 = require("../outputDifference");
45
45
  const recipient_1 = require("../recipient");
46
46
  const parseOutput_1 = require("./parseOutput");
47
+ async function parseRbfTransaction(coin, params) {
48
+ const { txParams, wallet } = params;
49
+ (0, assert_1.default)(txParams.rbfTxIds);
50
+ (0, assert_1.default)(txParams.rbfTxIds.length === 1);
51
+ const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });
52
+ const recipients = txToBeReplaced.outputs.flatMap((output) => {
53
+ // For self-sends, the walletId will be the same as the wallet's id
54
+ if (output.wallet === wallet.id()) {
55
+ return [];
56
+ }
57
+ return [coin.toCanonicalTransactionRecipient(output)];
58
+ });
59
+ // Recurse into parseTransaction with the derived recipients and without rbfTxIds
60
+ return parseTransaction(coin, {
61
+ ...params,
62
+ txParams: {
63
+ ...txParams,
64
+ recipients,
65
+ rbfTxIds: undefined,
66
+ },
67
+ });
68
+ }
69
+ function toExpectedOutputs(coin, txParams) {
70
+ // verify that each recipient from txParams has their own output
71
+ const expectedOutputs = (txParams.recipients ?? []).flatMap((output) => {
72
+ if (output.address === undefined) {
73
+ if (output.amount.toString() !== '0') {
74
+ throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${output}`);
75
+ }
76
+ return [output];
77
+ }
78
+ return [{ ...output, address: coin.canonicalAddress(output.address) }];
79
+ });
80
+ if (txParams.allowExternalChangeAddress && txParams.changeAddress) {
81
+ // when an external change address is explicitly specified, count all outputs going towards that
82
+ // address in the expected outputs (regardless of the output amount)
83
+ expectedOutputs.push({ address: coin.canonicalAddress(txParams.changeAddress), amount: 'max' });
84
+ }
85
+ return expectedOutputs;
86
+ }
47
87
  async function parseTransaction(coin, params) {
48
88
  const { txParams, txPrebuild, wallet, verification = {}, reqId } = params;
89
+ // Branch off early for RBF transactions
90
+ if (txParams.rbfTxIds) {
91
+ return parseRbfTransaction(coin, params);
92
+ }
49
93
  if (!lodash_1.default.isUndefined(verification.disableNetworking) && !lodash_1.default.isBoolean(verification.disableNetworking)) {
50
94
  throw new Error('verification.disableNetworking must be a boolean');
51
95
  }
@@ -65,48 +109,7 @@ async function parseTransaction(coin, params) {
65
109
  if (lodash_1.default.isUndefined(txPrebuild.txHex)) {
66
110
  throw new Error('missing required txPrebuild property txHex');
67
111
  }
68
- // obtain all outputs
69
- const explanation = await coin.explainTransaction({
70
- txHex: txPrebuild.txHex,
71
- txInfo: txPrebuild.txInfo,
72
- pubs: keychainArray.map((k) => k.pub),
73
- });
74
- const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];
75
- let expectedOutputs;
76
- if (txParams.rbfTxIds) {
77
- (0, assert_1.default)(txParams.rbfTxIds.length === 1);
78
- const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });
79
- expectedOutputs = txToBeReplaced.outputs.flatMap((output) => {
80
- // For self-sends, the walletId will be the same as the wallet's id
81
- if (output.wallet === wallet.id()) {
82
- return [];
83
- }
84
- return [coin.toCanonicalTransactionRecipient(output)];
85
- });
86
- }
87
- else {
88
- // verify that each recipient from txParams has their own output
89
- expectedOutputs = (txParams.recipients ?? []).flatMap((output) => {
90
- if (output.address === undefined) {
91
- if (output.amount.toString() !== '0') {
92
- throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${output}`);
93
- }
94
- return [output];
95
- }
96
- return [{ ...output, address: coin.canonicalAddress(output.address) }];
97
- });
98
- if (txParams.allowExternalChangeAddress && txParams.changeAddress) {
99
- // when an external change address is explicitly specified, count all outputs going towards that
100
- // address in the expected outputs (regardless of the output amount)
101
- expectedOutputs.push(...allOutputs.flatMap((output) => {
102
- if (output.address === undefined ||
103
- output.address !== coin.canonicalAddress(txParams.changeAddress)) {
104
- return [];
105
- }
106
- return [{ ...output, address: coin.canonicalAddress(output.address) }];
107
- }));
108
- }
109
- }
112
+ const expectedOutputs = toExpectedOutputs(coin, txParams);
110
113
  // get the keychains from the custom change wallet if needed
111
114
  let customChange;
112
115
  const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};
@@ -131,6 +134,13 @@ async function parseTransaction(coin, params) {
131
134
  };
132
135
  }
133
136
  }
137
+ // obtain all outputs
138
+ const explanation = await coin.explainTransaction({
139
+ txHex: txPrebuild.txHex,
140
+ txInfo: txPrebuild.txInfo,
141
+ pubs: keychainArray.map((k) => k.pub),
142
+ });
143
+ const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];
134
144
  /**
135
145
  * Loop through all the outputs and classify each of them as either internal spends
136
146
  * or external spends by setting the "external" property to true or false on the output object.
@@ -202,4 +212,4 @@ async function parseTransaction(coin, params) {
202
212
  customChange,
203
213
  };
204
214
  }
205
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseTransaction.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport _ from 'lodash';\nimport { Triple, VerificationOptions, Wallet } from '@bitgo-beta/sdk-core';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\n\nimport type { AbstractUtxoCoin, ParseTransactionOptions } from '../../abstractUtxoCoin';\nimport type { FixedScriptWalletOutput, Output, ParsedTransaction } from '../types';\nimport { fetchKeychains, getKeySignatures, toKeychainTriple, UtxoKeychain, UtxoNamedKeychains } from '../../keychains';\nimport { ComparableOutput, outputDifference } from '../outputDifference';\nimport { fromExtendedAddressFormatToScript, toExtendedAddressFormat } from '../recipient';\n\nimport type { TransactionExplanation } from './explainTransaction';\nimport { CustomChangeOptions, parseOutput } from './parseOutput';\n\nexport type ComparableOutputWithExternal<TValue> = ComparableOutput<TValue> & {\n  external: boolean | undefined;\n};\n\nexport async function parseTransaction<TNumber extends bigint | number>(\n  coin: AbstractUtxoCoin,\n  params: ParseTransactionOptions<TNumber>\n): Promise<ParsedTransaction<TNumber>> {\n  const { txParams, txPrebuild, wallet, verification = {}, reqId } = params;\n\n  if (!_.isUndefined(verification.disableNetworking) && !_.isBoolean(verification.disableNetworking)) {\n    throw new Error('verification.disableNetworking must be a boolean');\n  }\n  const disableNetworking = verification.disableNetworking;\n\n  // obtain the keychains and key signatures\n  let keychains: UtxoNamedKeychains | VerificationOptions['keychains'] | undefined = verification.keychains;\n  if (!keychains) {\n    if (disableNetworking) {\n      throw new Error('cannot fetch keychains without networking');\n    }\n    keychains = await fetchKeychains(coin, wallet, reqId);\n  }\n\n  if (!UtxoNamedKeychains.is(keychains)) {\n    throw new Error('invalid keychains');\n  }\n\n  const keychainArray: Triple<UtxoKeychain> = toKeychainTriple(keychains);\n\n  if (_.isUndefined(txPrebuild.txHex)) {\n    throw new Error('missing required txPrebuild property txHex');\n  }\n\n  // obtain all outputs\n  const explanation: TransactionExplanation = await coin.explainTransaction<TNumber>({\n    txHex: txPrebuild.txHex,\n    txInfo: txPrebuild.txInfo,\n    pubs: keychainArray.map((k) => k.pub) as Triple<string>,\n  });\n\n  const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];\n\n  let expectedOutputs;\n  if (txParams.rbfTxIds) {\n    assert(txParams.rbfTxIds.length === 1);\n\n    const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });\n    expectedOutputs = txToBeReplaced.outputs.flatMap(\n      (output: { valueString: string; address?: string; wallet?: string }) => {\n        // For self-sends, the walletId will be the same as the wallet's id\n        if (output.wallet === wallet.id()) {\n          return [];\n        }\n        return [coin.toCanonicalTransactionRecipient(output)];\n      }\n    );\n  } else {\n    // verify that each recipient from txParams has their own output\n    expectedOutputs = (txParams.recipients ?? []).flatMap((output) => {\n      if (output.address === undefined) {\n        if (output.amount.toString() !== '0') {\n          throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${output}`);\n        }\n        return [output];\n      }\n      return [{ ...output, address: coin.canonicalAddress(output.address) }];\n    });\n    if (txParams.allowExternalChangeAddress && txParams.changeAddress) {\n      // when an external change address is explicitly specified, count all outputs going towards that\n      // address in the expected outputs (regardless of the output amount)\n      expectedOutputs.push(\n        ...allOutputs.flatMap((output) => {\n          if (\n            output.address === undefined ||\n            output.address !== coin.canonicalAddress(txParams.changeAddress as string)\n          ) {\n            return [];\n          }\n          return [{ ...output, address: coin.canonicalAddress(output.address) }];\n        })\n      );\n    }\n  }\n\n  // get the keychains from the custom change wallet if needed\n  let customChange: CustomChangeOptions | undefined;\n  const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};\n  if (customChangeWalletId) {\n    // fetch keychains from custom change wallet for deriving addresses.\n    // These keychains should be signed and this should be verified in verifyTransaction\n    const customChangeKeySignatures = wallet._wallet.customChangeKeySignatures;\n    const customChangeWallet: Wallet = await coin.wallets().get({ id: customChangeWalletId });\n    const customChangeKeys = await fetchKeychains(coin, customChangeWallet, reqId);\n\n    if (!customChangeKeys) {\n      throw new Error('failed to fetch keychains for custom change wallet');\n    }\n\n    if (customChangeKeys.user && customChangeKeys.backup && customChangeKeys.bitgo && customChangeWallet) {\n      const customChangeKeychains: Triple<UtxoKeychain> = [\n        customChangeKeys.user,\n        customChangeKeys.backup,\n        customChangeKeys.bitgo,\n      ];\n\n      customChange = {\n        keys: customChangeKeychains,\n        signatures: [customChangeKeySignatures.user, customChangeKeySignatures.backup, customChangeKeySignatures.bitgo],\n      };\n    }\n  }\n\n  /**\n   * Loop through all the outputs and classify each of them as either internal spends\n   * or external spends by setting the \"external\" property to true or false on the output object.\n   */\n  const allOutputDetails: Output[] = await Promise.all(\n    allOutputs.map((currentOutput) => {\n      return parseOutput({\n        currentOutput,\n        coin,\n        txPrebuild,\n        verification,\n        keychainArray: toKeychainTriple(keychains),\n        wallet,\n        txParams: {\n          recipients: expectedOutputs,\n          changeAddress: txParams.changeAddress,\n        },\n        customChange,\n        reqId,\n      });\n    })\n  );\n\n  const needsCustomChangeKeySignatureVerification = allOutputDetails.some(\n    (output) => (output as FixedScriptWalletOutput)?.needsCustomChangeKeySignatureVerification\n  );\n\n  const changeOutputs = _.filter(allOutputDetails, { external: false });\n\n  function toComparableOutputsWithExternal(outputs: Output[]): ComparableOutputWithExternal<bigint | 'max'>[] {\n    return outputs.map((output) => ({\n      script: fromExtendedAddressFormatToScript(output.address, coin.network),\n      value: output.amount === 'max' ? 'max' : (BigInt(output.amount) as bigint | 'max'),\n      external: output.external,\n    }));\n  }\n\n  const missingOutputs = outputDifference(\n    toComparableOutputsWithExternal(expectedOutputs),\n    toComparableOutputsWithExternal(allOutputs)\n  );\n\n  const implicitOutputs = outputDifference(\n    toComparableOutputsWithExternal(allOutputDetails),\n    toComparableOutputsWithExternal(expectedOutputs)\n  );\n  const explicitOutputs = outputDifference(toComparableOutputsWithExternal(allOutputDetails), implicitOutputs);\n\n  // these are all the non-wallet outputs that had been originally explicitly specified in recipients\n  const explicitExternalOutputs = explicitOutputs.filter((output) => output.external);\n  // this is the sum of all the originally explicitly specified non-wallet output values\n  const explicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n    explicitExternalOutputs.reduce((sum: bigint, o) => sum + BigInt(o.value), BigInt(0)) as bigint,\n    coin.amountType\n  );\n\n  /**\n   * The calculation of the implicit external spend amount pertains to verifying the pay-as-you-go-fee BitGo\n   * automatically applied to transactions sending money out of the wallet. The logic is fairly straightforward\n   * in that we compare the external spend amount that was specified explicitly by the user to the portion\n   * that was specified implicitly. To protect customers from people tampering with the transaction outputs, we\n   * define a threshold for the maximum percentage of the implicit external spend in relation to the explicit\n   * external spend.\n   *\n   * This has become obsolete with the intoduction of `utxocore.paygo.verifyPayGoAddressProof()`.\n   */\n\n  // make sure that all the extra addresses are change addresses\n  // get all the additional external outputs the server added and calculate their values\n  const implicitExternalOutputs = implicitOutputs.filter((output) => output.external);\n  const implicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n    implicitExternalOutputs.reduce((sum: bigint, o) => sum + BigInt(o.value), BigInt(0)) as bigint,\n    coin.amountType\n  );\n\n  function toOutputs(outputs: ComparableOutputWithExternal<bigint | 'max'>[]): Output[] {\n    return outputs.map((output) => ({\n      address: toExtendedAddressFormat(output.script, coin.network),\n      amount: output.value.toString(),\n      external: output.external,\n    }));\n  }\n\n  return {\n    keychains,\n    keySignatures: getKeySignatures(wallet) ?? {},\n    outputs: allOutputDetails,\n    missingOutputs: toOutputs(missingOutputs),\n    explicitExternalOutputs: toOutputs(explicitExternalOutputs),\n    implicitExternalOutputs: toOutputs(implicitExternalOutputs),\n    changeOutputs,\n    explicitExternalSpendAmount,\n    implicitExternalSpendAmount,\n    needsCustomChangeKeySignatureVerification,\n    customChange,\n  };\n}\n"]}
215
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseTransaction.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport _ from 'lodash';\nimport { ITransactionRecipient, Triple, VerificationOptions, Wallet } from '@bitgo-beta/sdk-core';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\n\nimport type { AbstractUtxoCoin, ParseTransactionOptions } from '../../abstractUtxoCoin';\nimport type { FixedScriptWalletOutput, Output, ParsedTransaction } from '../types';\nimport { fetchKeychains, getKeySignatures, toKeychainTriple, UtxoKeychain, UtxoNamedKeychains } from '../../keychains';\nimport { ComparableOutput, outputDifference } from '../outputDifference';\nimport { fromExtendedAddressFormatToScript, toExtendedAddressFormat } from '../recipient';\n\nimport type { TransactionExplanation } from './explainTransaction';\nimport { CustomChangeOptions, parseOutput } from './parseOutput';\n\nexport type ComparableOutputWithExternal<TValue> = ComparableOutput<TValue> & {\n  external: boolean | undefined;\n};\n\nasync function parseRbfTransaction<TNumber extends bigint | number>(\n  coin: AbstractUtxoCoin,\n  params: ParseTransactionOptions<TNumber>\n): Promise<ParsedTransaction<TNumber>> {\n  const { txParams, wallet } = params;\n\n  assert(txParams.rbfTxIds);\n  assert(txParams.rbfTxIds.length === 1);\n\n  const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });\n  const recipients = txToBeReplaced.outputs.flatMap(\n    (output: { valueString: string; address?: string; wallet?: string }) => {\n      // For self-sends, the walletId will be the same as the wallet's id\n      if (output.wallet === wallet.id()) {\n        return [];\n      }\n      return [coin.toCanonicalTransactionRecipient(output)];\n    }\n  );\n\n  // Recurse into parseTransaction with the derived recipients and without rbfTxIds\n  return parseTransaction(coin, {\n    ...params,\n    txParams: {\n      ...txParams,\n      recipients,\n      rbfTxIds: undefined,\n    },\n  });\n}\n\nfunction toExpectedOutputs(\n  coin: AbstractUtxoCoin,\n  txParams: {\n    recipients?: ITransactionRecipient[];\n    allowExternalChangeAddress?: boolean;\n    changeAddress?: string;\n  }\n): Output[] {\n  // verify that each recipient from txParams has their own output\n  const expectedOutputs = (txParams.recipients ?? []).flatMap((output) => {\n    if (output.address === undefined) {\n      if (output.amount.toString() !== '0') {\n        throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${output}`);\n      }\n      return [output];\n    }\n    return [{ ...output, address: coin.canonicalAddress(output.address) }];\n  });\n  if (txParams.allowExternalChangeAddress && txParams.changeAddress) {\n    // when an external change address is explicitly specified, count all outputs going towards that\n    // address in the expected outputs (regardless of the output amount)\n    expectedOutputs.push({ address: coin.canonicalAddress(txParams.changeAddress), amount: 'max' });\n  }\n  return expectedOutputs;\n}\n\nexport async function parseTransaction<TNumber extends bigint | number>(\n  coin: AbstractUtxoCoin,\n  params: ParseTransactionOptions<TNumber>\n): Promise<ParsedTransaction<TNumber>> {\n  const { txParams, txPrebuild, wallet, verification = {}, reqId } = params;\n\n  // Branch off early for RBF transactions\n  if (txParams.rbfTxIds) {\n    return parseRbfTransaction(coin, params);\n  }\n\n  if (!_.isUndefined(verification.disableNetworking) && !_.isBoolean(verification.disableNetworking)) {\n    throw new Error('verification.disableNetworking must be a boolean');\n  }\n  const disableNetworking = verification.disableNetworking;\n\n  // obtain the keychains and key signatures\n  let keychains: UtxoNamedKeychains | VerificationOptions['keychains'] | undefined = verification.keychains;\n  if (!keychains) {\n    if (disableNetworking) {\n      throw new Error('cannot fetch keychains without networking');\n    }\n    keychains = await fetchKeychains(coin, wallet, reqId);\n  }\n\n  if (!UtxoNamedKeychains.is(keychains)) {\n    throw new Error('invalid keychains');\n  }\n\n  const keychainArray: Triple<UtxoKeychain> = toKeychainTriple(keychains);\n\n  if (_.isUndefined(txPrebuild.txHex)) {\n    throw new Error('missing required txPrebuild property txHex');\n  }\n\n  const expectedOutputs = toExpectedOutputs(coin, txParams);\n\n  // get the keychains from the custom change wallet if needed\n  let customChange: CustomChangeOptions | undefined;\n  const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};\n  if (customChangeWalletId) {\n    // fetch keychains from custom change wallet for deriving addresses.\n    // These keychains should be signed and this should be verified in verifyTransaction\n    const customChangeKeySignatures = wallet._wallet.customChangeKeySignatures;\n    const customChangeWallet: Wallet = await coin.wallets().get({ id: customChangeWalletId });\n    const customChangeKeys = await fetchKeychains(coin, customChangeWallet, reqId);\n\n    if (!customChangeKeys) {\n      throw new Error('failed to fetch keychains for custom change wallet');\n    }\n\n    if (customChangeKeys.user && customChangeKeys.backup && customChangeKeys.bitgo && customChangeWallet) {\n      const customChangeKeychains: Triple<UtxoKeychain> = [\n        customChangeKeys.user,\n        customChangeKeys.backup,\n        customChangeKeys.bitgo,\n      ];\n\n      customChange = {\n        keys: customChangeKeychains,\n        signatures: [customChangeKeySignatures.user, customChangeKeySignatures.backup, customChangeKeySignatures.bitgo],\n      };\n    }\n  }\n\n  // obtain all outputs\n  const explanation: TransactionExplanation = await coin.explainTransaction<TNumber>({\n    txHex: txPrebuild.txHex,\n    txInfo: txPrebuild.txInfo,\n    pubs: keychainArray.map((k) => k.pub) as Triple<string>,\n  });\n\n  const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];\n\n  /**\n   * Loop through all the outputs and classify each of them as either internal spends\n   * or external spends by setting the \"external\" property to true or false on the output object.\n   */\n  const allOutputDetails: Output[] = await Promise.all(\n    allOutputs.map((currentOutput) => {\n      return parseOutput({\n        currentOutput,\n        coin,\n        txPrebuild,\n        verification,\n        keychainArray: toKeychainTriple(keychains),\n        wallet,\n        txParams: {\n          recipients: expectedOutputs,\n          changeAddress: txParams.changeAddress,\n        },\n        customChange,\n        reqId,\n      });\n    })\n  );\n\n  const needsCustomChangeKeySignatureVerification = allOutputDetails.some(\n    (output) => (output as FixedScriptWalletOutput)?.needsCustomChangeKeySignatureVerification\n  );\n\n  const changeOutputs = _.filter(allOutputDetails, { external: false });\n\n  function toComparableOutputsWithExternal(outputs: Output[]): ComparableOutputWithExternal<bigint | 'max'>[] {\n    return outputs.map((output) => ({\n      script: fromExtendedAddressFormatToScript(output.address, coin.network),\n      value: output.amount === 'max' ? 'max' : (BigInt(output.amount) as bigint | 'max'),\n      external: output.external,\n    }));\n  }\n\n  const missingOutputs = outputDifference(\n    toComparableOutputsWithExternal(expectedOutputs),\n    toComparableOutputsWithExternal(allOutputs)\n  );\n\n  const implicitOutputs = outputDifference(\n    toComparableOutputsWithExternal(allOutputDetails),\n    toComparableOutputsWithExternal(expectedOutputs)\n  );\n  const explicitOutputs = outputDifference(toComparableOutputsWithExternal(allOutputDetails), implicitOutputs);\n\n  // these are all the non-wallet outputs that had been originally explicitly specified in recipients\n  const explicitExternalOutputs = explicitOutputs.filter((output) => output.external);\n  // this is the sum of all the originally explicitly specified non-wallet output values\n  const explicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n    explicitExternalOutputs.reduce((sum: bigint, o) => sum + BigInt(o.value), BigInt(0)) as bigint,\n    coin.amountType\n  );\n\n  /**\n   * The calculation of the implicit external spend amount pertains to verifying the pay-as-you-go-fee BitGo\n   * automatically applied to transactions sending money out of the wallet. The logic is fairly straightforward\n   * in that we compare the external spend amount that was specified explicitly by the user to the portion\n   * that was specified implicitly. To protect customers from people tampering with the transaction outputs, we\n   * define a threshold for the maximum percentage of the implicit external spend in relation to the explicit\n   * external spend.\n   *\n   * This has become obsolete with the intoduction of `utxocore.paygo.verifyPayGoAddressProof()`.\n   */\n\n  // make sure that all the extra addresses are change addresses\n  // get all the additional external outputs the server added and calculate their values\n  const implicitExternalOutputs = implicitOutputs.filter((output) => output.external);\n  const implicitExternalSpendAmount = utxolib.bitgo.toTNumber<TNumber>(\n    implicitExternalOutputs.reduce((sum: bigint, o) => sum + BigInt(o.value), BigInt(0)) as bigint,\n    coin.amountType\n  );\n\n  function toOutputs(outputs: ComparableOutputWithExternal<bigint | 'max'>[]): Output[] {\n    return outputs.map((output) => ({\n      address: toExtendedAddressFormat(output.script, coin.network),\n      amount: output.value.toString(),\n      external: output.external,\n    }));\n  }\n\n  return {\n    keychains,\n    keySignatures: getKeySignatures(wallet) ?? {},\n    outputs: allOutputDetails,\n    missingOutputs: toOutputs(missingOutputs),\n    explicitExternalOutputs: toOutputs(explicitExternalOutputs),\n    implicitExternalOutputs: toOutputs(implicitExternalOutputs),\n    changeOutputs,\n    explicitExternalSpendAmount,\n    implicitExternalSpendAmount,\n    needsCustomChangeKeySignatureVerification,\n    customChange,\n  };\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"recipient.d.ts","sourceRoot":"","sources":["../../../../src/transaction/recipient.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAIhD;;;GAGG;AACH,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAE1D;AAED;;;;GAIG;AACH,wBAAgB,yBAAyB,CAAC,eAAe,EAAE,MAAM,GAAG;IAAE,OAAO,EAAE,MAAM,CAAA;CAAE,GAAG;IAAE,MAAM,EAAE,MAAM,CAAA;CAAE,CAK3G;AAED,wBAAgB,iCAAiC,CAAC,eAAe,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,MAAM,CAM3G;AAED;;;;;GAKG;AACH,wBAAgB,uBAAuB,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,MAAM,CAIxF;AAED,wBAAgB,+BAA+B,CAAC,MAAM,EAAE;IAAE,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAE,GAAG,IAAI,CAQpH"}
1
+ {"version":3,"file":"recipient.d.ts","sourceRoot":"","sources":["../../../../src/transaction/recipient.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAIhD;;;GAGG;AACH,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAE1D;AAED;;;;GAIG;AACH,wBAAgB,yBAAyB,CAAC,eAAe,EAAE,MAAM,GAAG;IAAE,OAAO,EAAE,MAAM,CAAA;CAAE,GAAG;IAAE,MAAM,EAAE,MAAM,CAAA;CAAE,CAK3G;AAED,wBAAgB,iCAAiC,CAAC,eAAe,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,MAAM,CAM3G;AAED;;;;;GAKG;AACH,wBAAgB,uBAAuB,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,MAAM,CAIxF;AAED,wBAAgB,+BAA+B,CAAC,MAAM,EAAE;IAAE,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAAE,GAAG,IAAI,CAUpH"}
@@ -81,8 +81,8 @@ function assertValidTransactionRecipient(output) {
81
81
  // We will verify that the amount is zero, and if it isnt then we will throw an error.
82
82
  if (!output.address || isScriptRecipient(output.address)) {
83
83
  if (output.amount.toString() !== '0') {
84
- throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${JSON.stringify(output)}`);
84
+ throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: amount: ${output.amount}, address: ${output.address}`);
85
85
  }
86
86
  }
87
87
  }
88
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjaXBpZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3RyYW5zYWN0aW9uL3JlY2lwaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQVFBLDhDQUVDO0FBT0QsOERBS0M7QUFFRCw4RUFNQztBQVFELDBEQUlDO0FBRUQsMEVBUUM7QUFwREQsOERBQWdEO0FBRWhELE1BQU0scUJBQXFCLEdBQUcsZUFBZSxDQUFDO0FBRTlDOzs7R0FHRztBQUNILFNBQWdCLGlCQUFpQixDQUFDLE9BQWU7SUFDL0MsT0FBTyxPQUFPLENBQUMsV0FBVyxFQUFFLENBQUMsVUFBVSxDQUFDLHFCQUFxQixDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUM7QUFDL0UsQ0FBQztBQUVEOzs7O0dBSUc7QUFDSCxTQUFnQix5QkFBeUIsQ0FBQyxlQUF1QjtJQUMvRCxJQUFJLGlCQUFpQixDQUFDLGVBQWUsQ0FBQyxFQUFFLENBQUM7UUFDdkMsT0FBTyxFQUFFLE1BQU0sRUFBRSxlQUFlLENBQUMsS0FBSyxDQUFDLHFCQUFxQixDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUM7SUFDekUsQ0FBQztJQUNELE9BQU8sRUFBRSxPQUFPLEVBQUUsZUFBZSxFQUFFLENBQUM7QUFDdEMsQ0FBQztBQUVELFNBQWdCLGlDQUFpQyxDQUFDLGVBQXVCLEVBQUUsT0FBd0I7SUFDakcsTUFBTSxNQUFNLEdBQUcseUJBQXlCLENBQUMsZUFBZSxDQUFDLENBQUM7SUFDMUQsSUFBSSxRQUFRLElBQUksTUFBTSxFQUFFLENBQUM7UUFDdkIsT0FBTyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDM0MsQ0FBQztJQUNELE9BQU8sT0FBTyxDQUFDLGFBQWEsQ0FBQyx3QkFBd0IsQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBQ2pGLENBQUM7QUFFRDs7Ozs7R0FLRztBQUNILFNBQWdCLHVCQUF1QixDQUFDLE1BQWMsRUFBRSxPQUF3QjtJQUM5RSxPQUFPLE1BQU0sQ0FBQyxDQUFDLENBQUMsS0FBSyxPQUFPLENBQUMsT0FBTyxDQUFDLFNBQVM7UUFDNUMsQ0FBQyxDQUFDLEdBQUcscUJBQXFCLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsRUFBRTtRQUNyRCxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLEVBQUUsT0FBTyxDQUFDLENBQUM7QUFDeEQsQ0FBQztBQUVELFNBQWdCLCtCQUErQixDQUFDLE1BQThEO0lBQzVHLCtHQUErRztJQUMvRyxzRkFBc0Y7SUFDdEYsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLElBQUksaUJBQWlCLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDekQsSUFBSSxNQUFNLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxLQUFLLEdBQUcsRUFBRSxDQUFDO1lBQ3JDLE1BQU0sSUFBSSxLQUFLLENBQUMsK0RBQStELElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBQzNHLENBQUM7SUFDSCxDQUFDO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuXG5jb25zdCBTY3JpcHRSZWNpcGllbnRQcmVmaXggPSAnc2NyaXB0UHViS2V5Oic7XG5cbi8qKlxuICogQ2hlY2sgaWYgdGhlIGFkZHJlc3MgaXMgYSBzY3JpcHQgcmVjaXBpZW50IChzdGFydHMgd2l0aCBgc2NyaXB0UHViS2V5OmApLlxuICogQHBhcmFtIGFkZHJlc3NcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGlzU2NyaXB0UmVjaXBpZW50KGFkZHJlc3M6IHN0cmluZyk6IGJvb2xlYW4ge1xuICByZXR1cm4gYWRkcmVzcy50b0xvd2VyQ2FzZSgpLnN0YXJ0c1dpdGgoU2NyaXB0UmVjaXBpZW50UHJlZml4LnRvTG93ZXJDYXNlKCkpO1xufVxuXG4vKipcbiAqIEFuIGV4dGVuZGVkIGFkZHJlc3MgaXMgb25lIHRoYXQgZW5jb2RlcyBlaXRoZXIgYSByZWd1bGFyIGFkZHJlc3Mgb3IgYSBoZXggZW5jb2RlZCBzY3JpcHQgd2l0aCB0aGUgcHJlZml4IGBzY3JpcHRQdWJLZXk6YC5cbiAqIFRoaXMgZnVuY3Rpb24gY29udmVydHMgdGhlIGV4dGVuZGVkIGFkZHJlc3MgZm9ybWF0IHRvIGVpdGhlciBhIHNjcmlwdCBvciBhbiBhZGRyZXNzLlxuICogQHBhcmFtIGV4dGVuZGVkQWRkcmVzc1xuICovXG5leHBvcnQgZnVuY3Rpb24gZnJvbUV4dGVuZGVkQWRkcmVzc0Zvcm1hdChleHRlbmRlZEFkZHJlc3M6IHN0cmluZyk6IHsgYWRkcmVzczogc3RyaW5nIH0gfCB7IHNjcmlwdDogc3RyaW5nIH0ge1xuICBpZiAoaXNTY3JpcHRSZWNpcGllbnQoZXh0ZW5kZWRBZGRyZXNzKSkge1xuICAgIHJldHVybiB7IHNjcmlwdDogZXh0ZW5kZWRBZGRyZXNzLnNsaWNlKFNjcmlwdFJlY2lwaWVudFByZWZpeC5sZW5ndGgpIH07XG4gIH1cbiAgcmV0dXJuIHsgYWRkcmVzczogZXh0ZW5kZWRBZGRyZXNzIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBmcm9tRXh0ZW5kZWRBZGRyZXNzRm9ybWF0VG9TY3JpcHQoZXh0ZW5kZWRBZGRyZXNzOiBzdHJpbmcsIG5ldHdvcms6IHV0eG9saWIuTmV0d29yayk6IEJ1ZmZlciB7XG4gIGNvbnN0IHJlc3VsdCA9IGZyb21FeHRlbmRlZEFkZHJlc3NGb3JtYXQoZXh0ZW5kZWRBZGRyZXNzKTtcbiAgaWYgKCdzY3JpcHQnIGluIHJlc3VsdCkge1xuICAgIHJldHVybiBCdWZmZXIuZnJvbShyZXN1bHQuc2NyaXB0LCAnaGV4Jyk7XG4gIH1cbiAgcmV0dXJuIHV0eG9saWIuYWRkcmVzc0Zvcm1hdC50b091dHB1dFNjcmlwdFRyeUZvcm1hdHMocmVzdWx0LmFkZHJlc3MsIG5ldHdvcmspO1xufVxuXG4vKipcbiAqIENvbnZlcnQgYSBzY3JpcHQgb3IgYWRkcmVzcyB0byB0aGUgZXh0ZW5kZWQgYWRkcmVzcyBmb3JtYXQuXG4gKiBAcGFyYW0gc2NyaXB0XG4gKiBAcGFyYW0gbmV0d29ya1xuICogQHJldHVybnMgaWYgdGhlIHNjcmlwdCBpcyBhbiBPUF9SRVRVUk4gc2NyaXB0LCB0aGVuIGl0IHdpbGwgYmUgcHJlZml4ZWQgd2l0aCBgc2NyaXB0UHViS2V5OmAsIG90aGVyd2lzZSBpdCB3aWxsIGJlIGNvbnZlcnRlZCB0byBhbiBhZGRyZXNzLlxuICovXG5leHBvcnQgZnVuY3Rpb24gdG9FeHRlbmRlZEFkZHJlc3NGb3JtYXQoc2NyaXB0OiBCdWZmZXIsIG5ldHdvcms6IHV0eG9saWIuTmV0d29yayk6IHN0cmluZyB7XG4gIHJldHVybiBzY3JpcHRbMF0gPT09IHV0eG9saWIub3Bjb2Rlcy5PUF9SRVRVUk5cbiAgICA/IGAke1NjcmlwdFJlY2lwaWVudFByZWZpeH0ke3NjcmlwdC50b1N0cmluZygnaGV4Jyl9YFxuICAgIDogdXR4b2xpYi5hZGRyZXNzLmZyb21PdXRwdXRTY3JpcHQoc2NyaXB0LCBuZXR3b3JrKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGFzc2VydFZhbGlkVHJhbnNhY3Rpb25SZWNpcGllbnQob3V0cHV0OiB7IGFtb3VudDogYmlnaW50IHwgbnVtYmVyIHwgc3RyaW5nOyBhZGRyZXNzPzogc3RyaW5nIH0pOiB2b2lkIHtcbiAgLy8gSW4gdGhlIGNhc2UgdGhhdCB0aGlzIGlzIGFuIE9QX1JFVFVSTiBvdXRwdXQgb3IgYW5vdGhlciBub24tZW5jb2RhYmxlIHNjcmlwdFB1YmtleSwgd2UgZG9udCBoYXZlIGFuIGFkZHJlc3MuXG4gIC8vIFdlIHdpbGwgdmVyaWZ5IHRoYXQgdGhlIGFtb3VudCBpcyB6ZXJvLCBhbmQgaWYgaXQgaXNudCB0aGVuIHdlIHdpbGwgdGhyb3cgYW4gZXJyb3IuXG4gIGlmICghb3V0cHV0LmFkZHJlc3MgfHwgaXNTY3JpcHRSZWNpcGllbnQob3V0cHV0LmFkZHJlc3MpKSB7XG4gICAgaWYgKG91dHB1dC5hbW91bnQudG9TdHJpbmcoKSAhPT0gJzAnKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYE9ubHkgemVybyBhbW91bnRzIGFsbG93ZWQgZm9yIG5vbi1lbmNvZGVhYmxlIHNjcmlwdFB1YmtleXM6ICR7SlNPTi5zdHJpbmdpZnkob3V0cHV0KX1gKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
88
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjaXBpZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3RyYW5zYWN0aW9uL3JlY2lwaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQVFBLDhDQUVDO0FBT0QsOERBS0M7QUFFRCw4RUFNQztBQVFELDBEQUlDO0FBRUQsMEVBVUM7QUF0REQsOERBQWdEO0FBRWhELE1BQU0scUJBQXFCLEdBQUcsZUFBZSxDQUFDO0FBRTlDOzs7R0FHRztBQUNILFNBQWdCLGlCQUFpQixDQUFDLE9BQWU7SUFDL0MsT0FBTyxPQUFPLENBQUMsV0FBVyxFQUFFLENBQUMsVUFBVSxDQUFDLHFCQUFxQixDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUM7QUFDL0UsQ0FBQztBQUVEOzs7O0dBSUc7QUFDSCxTQUFnQix5QkFBeUIsQ0FBQyxlQUF1QjtJQUMvRCxJQUFJLGlCQUFpQixDQUFDLGVBQWUsQ0FBQyxFQUFFLENBQUM7UUFDdkMsT0FBTyxFQUFFLE1BQU0sRUFBRSxlQUFlLENBQUMsS0FBSyxDQUFDLHFCQUFxQixDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUM7SUFDekUsQ0FBQztJQUNELE9BQU8sRUFBRSxPQUFPLEVBQUUsZUFBZSxFQUFFLENBQUM7QUFDdEMsQ0FBQztBQUVELFNBQWdCLGlDQUFpQyxDQUFDLGVBQXVCLEVBQUUsT0FBd0I7SUFDakcsTUFBTSxNQUFNLEdBQUcseUJBQXlCLENBQUMsZUFBZSxDQUFDLENBQUM7SUFDMUQsSUFBSSxRQUFRLElBQUksTUFBTSxFQUFFLENBQUM7UUFDdkIsT0FBTyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDM0MsQ0FBQztJQUNELE9BQU8sT0FBTyxDQUFDLGFBQWEsQ0FBQyx3QkFBd0IsQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBQ2pGLENBQUM7QUFFRDs7Ozs7R0FLRztBQUNILFNBQWdCLHVCQUF1QixDQUFDLE1BQWMsRUFBRSxPQUF3QjtJQUM5RSxPQUFPLE1BQU0sQ0FBQyxDQUFDLENBQUMsS0FBSyxPQUFPLENBQUMsT0FBTyxDQUFDLFNBQVM7UUFDNUMsQ0FBQyxDQUFDLEdBQUcscUJBQXFCLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsRUFBRTtRQUNyRCxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLEVBQUUsT0FBTyxDQUFDLENBQUM7QUFDeEQsQ0FBQztBQUVELFNBQWdCLCtCQUErQixDQUFDLE1BQThEO0lBQzVHLCtHQUErRztJQUMvRyxzRkFBc0Y7SUFDdEYsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLElBQUksaUJBQWlCLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDekQsSUFBSSxNQUFNLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxLQUFLLEdBQUcsRUFBRSxDQUFDO1lBQ3JDLE1BQU0sSUFBSSxLQUFLLENBQ2IsdUVBQXVFLE1BQU0sQ0FBQyxNQUFNLGNBQWMsTUFBTSxDQUFDLE9BQU8sRUFBRSxDQUNuSCxDQUFDO1FBQ0osQ0FBQztJQUNILENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgdXR4b2xpYiBmcm9tICdAYml0Z28tYmV0YS91dHhvLWxpYic7XG5cbmNvbnN0IFNjcmlwdFJlY2lwaWVudFByZWZpeCA9ICdzY3JpcHRQdWJLZXk6JztcblxuLyoqXG4gKiBDaGVjayBpZiB0aGUgYWRkcmVzcyBpcyBhIHNjcmlwdCByZWNpcGllbnQgKHN0YXJ0cyB3aXRoIGBzY3JpcHRQdWJLZXk6YCkuXG4gKiBAcGFyYW0gYWRkcmVzc1xuICovXG5leHBvcnQgZnVuY3Rpb24gaXNTY3JpcHRSZWNpcGllbnQoYWRkcmVzczogc3RyaW5nKTogYm9vbGVhbiB7XG4gIHJldHVybiBhZGRyZXNzLnRvTG93ZXJDYXNlKCkuc3RhcnRzV2l0aChTY3JpcHRSZWNpcGllbnRQcmVmaXgudG9Mb3dlckNhc2UoKSk7XG59XG5cbi8qKlxuICogQW4gZXh0ZW5kZWQgYWRkcmVzcyBpcyBvbmUgdGhhdCBlbmNvZGVzIGVpdGhlciBhIHJlZ3VsYXIgYWRkcmVzcyBvciBhIGhleCBlbmNvZGVkIHNjcmlwdCB3aXRoIHRoZSBwcmVmaXggYHNjcmlwdFB1YktleTpgLlxuICogVGhpcyBmdW5jdGlvbiBjb252ZXJ0cyB0aGUgZXh0ZW5kZWQgYWRkcmVzcyBmb3JtYXQgdG8gZWl0aGVyIGEgc2NyaXB0IG9yIGFuIGFkZHJlc3MuXG4gKiBAcGFyYW0gZXh0ZW5kZWRBZGRyZXNzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmcm9tRXh0ZW5kZWRBZGRyZXNzRm9ybWF0KGV4dGVuZGVkQWRkcmVzczogc3RyaW5nKTogeyBhZGRyZXNzOiBzdHJpbmcgfSB8IHsgc2NyaXB0OiBzdHJpbmcgfSB7XG4gIGlmIChpc1NjcmlwdFJlY2lwaWVudChleHRlbmRlZEFkZHJlc3MpKSB7XG4gICAgcmV0dXJuIHsgc2NyaXB0OiBleHRlbmRlZEFkZHJlc3Muc2xpY2UoU2NyaXB0UmVjaXBpZW50UHJlZml4Lmxlbmd0aCkgfTtcbiAgfVxuICByZXR1cm4geyBhZGRyZXNzOiBleHRlbmRlZEFkZHJlc3MgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGZyb21FeHRlbmRlZEFkZHJlc3NGb3JtYXRUb1NjcmlwdChleHRlbmRlZEFkZHJlc3M6IHN0cmluZywgbmV0d29yazogdXR4b2xpYi5OZXR3b3JrKTogQnVmZmVyIHtcbiAgY29uc3QgcmVzdWx0ID0gZnJvbUV4dGVuZGVkQWRkcmVzc0Zvcm1hdChleHRlbmRlZEFkZHJlc3MpO1xuICBpZiAoJ3NjcmlwdCcgaW4gcmVzdWx0KSB7XG4gICAgcmV0dXJuIEJ1ZmZlci5mcm9tKHJlc3VsdC5zY3JpcHQsICdoZXgnKTtcbiAgfVxuICByZXR1cm4gdXR4b2xpYi5hZGRyZXNzRm9ybWF0LnRvT3V0cHV0U2NyaXB0VHJ5Rm9ybWF0cyhyZXN1bHQuYWRkcmVzcywgbmV0d29yayk7XG59XG5cbi8qKlxuICogQ29udmVydCBhIHNjcmlwdCBvciBhZGRyZXNzIHRvIHRoZSBleHRlbmRlZCBhZGRyZXNzIGZvcm1hdC5cbiAqIEBwYXJhbSBzY3JpcHRcbiAqIEBwYXJhbSBuZXR3b3JrXG4gKiBAcmV0dXJucyBpZiB0aGUgc2NyaXB0IGlzIGFuIE9QX1JFVFVSTiBzY3JpcHQsIHRoZW4gaXQgd2lsbCBiZSBwcmVmaXhlZCB3aXRoIGBzY3JpcHRQdWJLZXk6YCwgb3RoZXJ3aXNlIGl0IHdpbGwgYmUgY29udmVydGVkIHRvIGFuIGFkZHJlc3MuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB0b0V4dGVuZGVkQWRkcmVzc0Zvcm1hdChzY3JpcHQ6IEJ1ZmZlciwgbmV0d29yazogdXR4b2xpYi5OZXR3b3JrKTogc3RyaW5nIHtcbiAgcmV0dXJuIHNjcmlwdFswXSA9PT0gdXR4b2xpYi5vcGNvZGVzLk9QX1JFVFVSTlxuICAgID8gYCR7U2NyaXB0UmVjaXBpZW50UHJlZml4fSR7c2NyaXB0LnRvU3RyaW5nKCdoZXgnKX1gXG4gICAgOiB1dHhvbGliLmFkZHJlc3MuZnJvbU91dHB1dFNjcmlwdChzY3JpcHQsIG5ldHdvcmspO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gYXNzZXJ0VmFsaWRUcmFuc2FjdGlvblJlY2lwaWVudChvdXRwdXQ6IHsgYW1vdW50OiBiaWdpbnQgfCBudW1iZXIgfCBzdHJpbmc7IGFkZHJlc3M/OiBzdHJpbmcgfSk6IHZvaWQge1xuICAvLyBJbiB0aGUgY2FzZSB0aGF0IHRoaXMgaXMgYW4gT1BfUkVUVVJOIG91dHB1dCBvciBhbm90aGVyIG5vbi1lbmNvZGFibGUgc2NyaXB0UHVia2V5LCB3ZSBkb250IGhhdmUgYW4gYWRkcmVzcy5cbiAgLy8gV2Ugd2lsbCB2ZXJpZnkgdGhhdCB0aGUgYW1vdW50IGlzIHplcm8sIGFuZCBpZiBpdCBpc250IHRoZW4gd2Ugd2lsbCB0aHJvdyBhbiBlcnJvci5cbiAgaWYgKCFvdXRwdXQuYWRkcmVzcyB8fCBpc1NjcmlwdFJlY2lwaWVudChvdXRwdXQuYWRkcmVzcykpIHtcbiAgICBpZiAob3V0cHV0LmFtb3VudC50b1N0cmluZygpICE9PSAnMCcpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgYE9ubHkgemVybyBhbW91bnRzIGFsbG93ZWQgZm9yIG5vbi1lbmNvZGVhYmxlIHNjcmlwdFB1YmtleXM6IGFtb3VudDogJHtvdXRwdXQuYW1vdW50fSwgYWRkcmVzczogJHtvdXRwdXQuYWRkcmVzc31gXG4gICAgICApO1xuICAgIH1cbiAgfVxufVxuIl19
@@ -116,6 +116,7 @@ function run(coin, scriptTypes, walletKeys, params, tags = []) {
116
116
  let recoverUnspents;
117
117
  let mockedApiUnspents;
118
118
  before('create recovery data', async function () {
119
+ this.timeout(10000);
119
120
  recoverUnspents = scriptTypes.flatMap((scriptType, index) => [
120
121
  utxolib.testutil.toUnspent({ scriptType, value: BigInt(1e8) * valueMul }, index, coin.network, walletKeys),
121
122
  utxolib.testutil.toUnspent({ scriptType, value: BigInt(2e8) * valueMul }, index, coin.network, walletKeys),
@@ -282,4 +283,4 @@ describe('Backup Key Recovery', function () {
282
283
  // taproot compatible coins
283
284
  runWithScriptTypes(['p2tr', 'p2trMusig2'], { runKeyProviderTests: false });
284
285
  });
285
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backupKeyRecovery.js","sourceRoot":"","sources":["../../../../../test/unit/recovery/backupKeyRecovery.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,oDAA4B;AAE5B,kBAAgB;AAEhB,6CAA+B;AAC/B,6BAA8B;AAE9B,8DAAgD;AAChD,mDAAoE;AAEpE,sCAMsB;AACtB,kCAYiB;AAEjB,iCAA8C;AAE9C,MAAM,EAAE,QAAQ,EAAE,GAAG,OAAO,CAAC,KAAK,CAAC;AAKnC,MAAM,MAAM,GAAG,EAAE,YAAY,EAAZ,uBAAY,EAAE,CAAC;AAEhC,IAAI,CAAC,iBAAiB,EAAE,CAAC;AAEzB,SAAS,cAAc,CAAC,CAA2B;IACjD,MAAM,MAAM,GAAG,EAAE,GAAG,uBAAY,EAAE,CAAC;IACnC,MAAM,CAAC;QACL,CAAC,CAAC,MAAM,CAAC,CAAC;IACZ,CAAC,CAAC,CAAC;IACH,KAAK,CAAC;QACJ,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE;YACxC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QAChB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AAED,MAAM,gBAAgB,GAAG,KAAK,CAAC;AAQ/B,SAAS,YAAY,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EAAE,QAAgB;IAC5F,SAAS,MAAM,CAAC,CAAiB;QAC/B,OAAO,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,IAAA,sBAAe,EAAC,QAAQ,EAAE,IAAA,uBAAgB,EAAC,CAAC,CAAC,CAAC,CAAC;IACxF,CAAC;IACD,OAAO;QACL,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC;QACxB,SAAS,EAAE,MAAM,CAAC,SAAS,CAAC;QAC5B,QAAQ,EAAE,MAAM,CAAC,QAAQ,CAAC;KAC3B,CAAC;AACJ,CAAC;AAED,SAAS,uBAAuB,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EAAE,QAAgB;IACvG,OAAO,YAAY,CAAC,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,SAAS,CAAC,QAAQ,EAAE,EAAE,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,CAAC,CAAC;AACjG,CAAC;AAED,SAAS,4BAA4B,CACnC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EACtD,QAAgB;IAEhB,OAAO,YAAY,CAAC,CAAC,OAAO,EAAE,SAAS,CAAC,QAAQ,EAAE,EAAE,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,CAAC,CAAC;AACtF,CAAC;AAED,SAAS,4BAA4B,CACnC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAyB,EACtD,QAAgB;IAEhB,OAAO,YAAY,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAAE,QAAQ,CAAC,CAAC;AAC3E,CAAC;AAED,MAAM,UAAU,GAAG,IAAA,2BAAoB,GAAE,CAAC;AAC1C,MAAM,iBAAiB,GAAG,uBAAuB,CAAC,UAAU,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AACvF,MAAM,sBAAsB,GAAG,4BAA4B,CAAC,UAAU,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AACjG,MAAM,sBAAsB,GAAG,4BAA4B,CAAC,UAAU,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AAEjG,MAAM,iBAAiB,GAAG,OAAO,CAAC;AAClC,MAAM,gBAAgB,GAAG,IAAI,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,gBAAS,EAAE;IACnE,iBAAiB;IACjB,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,aAAa;IAC1C,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC,aAAa;CAC3C,CAAC,CAAC;AACH,MAAM,4BAA4B,GAAG,4BAA4B,CAAC,gBAAgB,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;AAE7G,SAAS,GAAG,CACV,IAAsB,EACtB,WAA6B,EAC7B,UAA0B,EAC1B,MAQC,EACD,OAAiB,EAAE;IAEnB,QAAQ,CAAC,wBAAwB,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,GAAG,IAAI,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE;QAC5F,MAAM,cAAc,GAAG,IAAA,oBAAa,EAAC,UAAU,CAAC,CAAC;QACjD,MAAM,mBAAmB,GAAG,IAAA,uBAAgB,EAAC,IAAI,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;QAE3E,IAAI,yBAAiC,CAAC;QACtC,IAAI,QAA6F,CAAC;QAClG,IAAI,UAAmF,CAAC;QAExF,uGAAuG;QACvG,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;QACxE,MAAM,CAAC,MAAM,EAAE;YACb,KAAK,CAAC,IAAI,CAAC,kBAAY,CAAC,SAAS,EAAE,aAAa,CAAC,CAAC,QAAQ,CAAC,KAAM,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;QAEH,cAAc,CAAC,UAAU,MAAc;YACrC,MAAM,kBAAkB,GAAG,EAAE,GAAG,MAAM,CAAC,YAAY,EAAE,CAAC;YACtD,kBAAkB,CAAC,GAAG,CAAC,cAAc,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;YAC3D,kBAAkB,CAAC,SAAS,CAAC,cAAc,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;YACjE,yBAAyB,GAAG,IAAA,uBAAgB,EAAC,IAAI,CAAC,OAAO,EAAE,cAAc,EAAE,CAAC,EAAE,GAAG,CAAC,CAAC;YACnF,kBAAkB,CAAC,SAAS,CAAC,YAAY,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,EAAE,yBAAyB,EAAE,CAAC;YAC7F,MAAM,CAAC,YAAY,GAAG,kBAAkB,CAAC;QAC3C,CAAC,CAAC,CAAC;QAEH,KAAK,CAAC;YACJ,KAAK,CAAC,OAAO,EAAE,CAAC;QAClB,CAAC,CAAC,CAAC;QAEH,IAAI,eAAgD,CAAC;QACrD,IAAI,iBAAkD,CAAC;QAEvD,MAAM,CAAC,sBAAsB,EAAE,KAAK;YAClC,eAAe,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC,UAAU,EAAE,KAAK,EAAE,EAAE,CAAC;gBAC3D,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;gBAC1G,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;gBAC1G,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;aAC3G,CAAC,CAAC;YAEH,qHAAqH;YACrH,gDAAgD;YAChD,iBAAiB;gBACf,IAAI,CAAC,QAAQ,EAAE,KAAK,KAAK,IAAI,IAAI,CAAC,QAAQ,EAAE,KAAK,MAAM;oBACrD,CAAC,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,OAAO,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;oBAC7G,CAAC,CAAC,eAAe,CAAC;YAEtB,gBAAM,CAAC,WAAW,CAAC,iBAAiB,CAAC,MAAM,EAAE,eAAe,CAAC,MAAM,CAAC,CAAC;YACrE,QAAQ,GAAG,MAAM,IAAA,uBAAiB,EAAC,IAAI,EAAE,mBAAY,EAAE;gBACrD,gBAAgB;gBAChB,mBAAmB;gBACnB,IAAI,EAAE,CAAC;gBACP,kBAAkB,EAAE,EAAE;gBACtB,WAAW,EAAE,MAAM,CAAC,WAAW;gBAC/B,WAAW,EAAE,MAAM,CAAC,WAAW;gBAC/B,GAAG,MAAM,CAAC,IAAI;gBACd,gBAAgB,EAAE,IAAI,2BAAoB,CAAC,iBAAiB,CAAC;aAC9D,CAAC,CAAC;YACH,MAAM,KAAK,GACR,QAA4C,CAAC,cAAc,IAAK,QAAwC,CAAC,KAAK,CAAC;YAClH,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YAC3C,UAAU,GAAG,MAAM;gBACjB,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,iBAAiB,CAAC,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC;gBACtD,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,wBAAwB,CAAC,KAAe,EAAE,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;YAC3F,QAAQ,CAAC,IAAI;gBACX,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;QAC3G,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,qCAAqC,EAAE,KAAK;YAC7C,MAAM,gBAAgB,GAAG,IAAI,2BAAoB,CAAC,iBAAiB,CAAC,CAAC;YACrE,CAAC,MAAM,gBAAgB,CAAC,uBAAuB,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CACzG,iBAAiB,CAAC,MAAM,CACzB,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,iBAAiB,EAAE,KAAK;YACzB,MAAM,WAAW,GAAG,IAAA,2BAAoB,EAAC,IAAI,CAAC,CAAC;YAC/C,MAAM,eAAe,GAAG,EAAE,GAAG,QAAQ,EAAE,CAAC;YACxC,IAAI,eAAe,CAAC,IAAI,EAAE,CAAC;gBACzB,eAAe,CAAC,IAAI,GAAG,WAAW,CAAC,QAAQ,EAAE,CAAC;YAChD,CAAC;YAED,IAAA,sBAAe,EACb,eAAe,EACf,MAAM,IAAA,iBAAU,EACd,WAAW,EACX,8BAA8B,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,EACzG,QAAQ,CACT,CACF,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0BAA0B,EAAE;YAC7B,CAAC,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CACxG,eAAe,CAAC,MAAM,CACvB,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,SAAS,mBAAmB,CAC1B,EAA2E,EAC3E,OAAuB,EACvB,WAAmB;YAEnB,IAAI,EAAE,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;gBACzC,SAAS,QAAQ,CAAC,EAA0B,EAAE,UAAkB;oBAC9D,IAAI,CAAC;wBACH,OAAO,EAAE,CAAC,2BAA2B,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;oBAC7D,CAAC;oBAAC,OAAO,CAAC,EAAE,CAAC;wBACX,IAAI,CAAC,CAAC,OAAO,KAAK,2BAA2B,EAAE,CAAC;4BAC9C,OAAO,KAAK,CAAC;wBACf,CAAC;wBACD,MAAM,CAAC,CAAC;oBACV,CAAC;gBACH,CAAC;gBACD,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;oBAC3C,QAAQ,CAAC,EAAE,EAAE,UAAU,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;gBACrD,CAAC,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,MAAM,WAAW,GAAG,eAAe;qBAChC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;qBACrC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC,CAAC;gBACpF,EAAE,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;oBACnC,MAAM,OAAO,GAAG,eAAe,CAAC,UAAU,CAAkB,CAAC;oBAC7D,MAAM,EAAE,SAAS,EAAE,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,iBAAiB,CAAC,OAAO,EAAE,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;oBAC9G,MAAM,UAAU,GAAG,OAAO,CAAC,KAAK;yBAC7B,yBAAyB,CACxB,EAAE,EACF,UAAU,EACV,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,EACvD,EAAE,SAAS,EAAE,EACb,WAAW,CACZ;yBACA,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,SAAS,CAAC,CAAC;oBAC3C,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;gBAC5C,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,EAAE,CAAC,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,iBAAiB,EAAE;YACnE,mBAAmB,CAAC,UAAU,EAAE,UAAU,CAAC,IAAI,EAAE,MAAM,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpF,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,mBAAmB,EAAE;YACvE,mBAAmB,CAAC,UAAU,EAAE,UAAU,CAAC,MAAM,EAAE,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACxF,CAAC,CAAC,CAAC;QAEH,IAAI,MAAM,CAAC,gBAAgB,IAAI,MAAM,CAAC,kBAAkB,EAAE,CAAC;YACzD,EAAE,CAAC,+BAA+B,EAAE;gBAClC,IAAI,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,eAAe,EAAE,CAAC;oBACxD,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;wBAC/B,MAAM,MAAM,GAAG,OAAO,CAAC,KAAK,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC;wBACzD,QAAQ,MAAM,CAAC,UAAU,EAAE,CAAC;4BAC1B,KAAK,MAAM,CAAC;4BACZ,KAAK,WAAW,CAAC;4BACjB,KAAK,OAAO,CAAC;4BACb,KAAK,wBAAwB;gCAC3B,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,CAAC,EAAE,EAAE;oCACzC,IAAI,OAAO,CAAC,KAAK,CAAC,sBAAsB,CAAC,SAAS,CAAC,EAAE,CAAC;wCACpD,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,EAAE,CAAC,CAAC;oCACzD,CAAC;gCACH,CAAC,CAAC,CAAC;gCACH,MAAM;4BACR;gCACE,MAAM,IAAI,KAAK,CAAC,yBAAyB,MAAM,CAAC,UAAU,EAAE,CAAC,CAAC;wBAClE,CAAC;oBACH,CAAC,CAAC,CAAC;gBACL,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,IAAI,EAAE,CAAC;gBACd,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC;QAED,EAAE,CAAC,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,8BAA8B,EAAE;YAC5E,MAAM,IAAI,GAAG,UAAU,YAAY,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC,aAAa,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC;YAC9G,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YAC1B,MAAM,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC,CAAC,MAAM,EAAE,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC;YACxG,eAAe;iBACZ,QAAQ,CAAC,yBAAyB,CAAC;iBACnC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,YAAY,IAAI,MAAM,CAAC,WAAW,KAAK,WAAW,CAAC,CAAC;YAC3E,eAAe,CAAC,QAAQ,CAAC,mBAAmB,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACjE,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AAED,SAAS,kBAAkB,CACzB,WAA6B,EAC7B,EAAE,mBAAmB,GAAG,IAAI,KAAwC,EAAE;IAEtE,gBAAS;SACN,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,CAAC,CAAC;SAC7E,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;QAChB,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;YACE,IAAI,EAAE,iBAAiB;YACvB,gBAAgB,EAAE,KAAK;YACvB,kBAAkB,EAAE,KAAK;SAC1B,EACD,CAAC,kBAAkB,EAAE,GAAG,WAAW,CAAC,CACrC,CAAC;QAEF,IAAI,mBAAmB,EAAE,CAAC;YACxB,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,EAAE;gBAC3C,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;oBACE,IAAI,EAAE,sBAAsB;oBAC5B,WAAW,EAAE,WAAW;oBACxB,gBAAgB,EAAE,IAAI;oBACtB,kBAAkB,EAAE,KAAK;oBACzB,YAAY,EAAE,KAAK;iBACpB,EACD,CAAC,oBAAoB,EAAE,GAAG,WAAW,CAAC,CACvC,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC;QAED,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;YACE,IAAI,EAAE,sBAAsB;YAC5B,gBAAgB,EAAE,IAAI;YACtB,kBAAkB,EAAE,IAAI;SACzB,EACD,CAAC,oBAAoB,EAAE,GAAG,WAAW,CAAC,CACvC,CAAC;QAEF,GAAG,CACD,IAAI,EACJ,WAAW,EACX,UAAU,EACV;YACE,IAAI,EAAE,sBAAsB;YAC5B,gBAAgB,EAAE,IAAI;YACtB,kBAAkB,EAAE,IAAI;YACxB,OAAO,EAAE,CAAC;SACX,EACD,CAAC,oBAAoB,EAAE,cAAc,EAAE,GAAG,WAAW,CAAC,CACvD,CAAC;QAEF,GAAG,CACD,IAAI,EACJ,WAAW,EACX,gBAAgB,EAChB;YACE,IAAI,EAAE,4BAA4B;YAClC,WAAW,EAAE,iBAAiB;YAC9B,gBAAgB,EAAE,IAAI;YACtB,kBAAkB,EAAE,IAAI;SACzB,EACD,CAAC,oBAAoB,EAAE,mBAAmB,EAAE,GAAG,WAAW,CAAC,CAC5D,CAAC;IACJ,CAAC,CAAC,CAAC;AACP,CAAC;AAED,QAAQ,CAAC,qBAAqB,EAAE;IAC9B,4BAA4B;IAC5B,kBAAkB,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;IAE7B,0BAA0B;IAC1B,kBAAkB,CAAC,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC,CAAC;IAE3C,2BAA2B;IAC3B,kBAAkB,CAAC,CAAC,MAAM,EAAE,YAAY,CAAC,EAAE,EAAE,mBAAmB,EAAE,KAAK,EAAE,CAAC,CAAC;AAC7E,CAAC,CAAC,CAAC","sourcesContent":["import assert from 'assert';\n\nimport 'should';\nimport * as mocha from 'mocha';\nimport * as sinon from 'sinon';\nimport nock = require('nock');\nimport { BIP32Interface } from '@bitgo-beta/utxo-lib';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Config, krsProviders, Triple } from '@bitgo-beta/sdk-core';\n\nimport {\n  AbstractUtxoCoin,\n  backupKeyRecovery,\n  BackupKeyRecoveryTransansaction,\n  CoingeckoApi,\n  FormattedOfflineVaultTxInfo,\n} from '../../../src';\nimport {\n  defaultBitGo,\n  encryptKeychain,\n  getDefaultWalletKeys,\n  getFixture,\n  getNormalTestnetCoin,\n  getWalletAddress,\n  getWalletKeys,\n  keychains,\n  shouldEqualJSON,\n  toKeychainBase58,\n  utxoCoins,\n} from '../util';\n\nimport { MockRecoveryProvider } from './mock';\n\nconst { toOutput } = utxolib.bitgo;\ntype WalletUnspent = utxolib.bitgo.WalletUnspent<bigint>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype ScriptType2Of3 = utxolib.bitgo.outputScripts.ScriptType2Of3;\n\nconst config = { krsProviders };\n\nnock.disableNetConnect();\n\nfunction configOverride(f: (config: Config) => void) {\n  const backup = { ...krsProviders };\n  before(function () {\n    f(config);\n  });\n  after(function () {\n    Object.entries(backup).forEach(([k, v]) => {\n      config[k] = v;\n    });\n  });\n}\n\nconst walletPassphrase = 'lol';\n\ntype NamedKeys = {\n  userKey: string;\n  backupKey: string;\n  bitgoKey: string;\n};\n\nfunction getNamedKeys([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  function encode(k: BIP32Interface): string {\n    return k.isNeutered() ? k.toBase58() : encryptKeychain(password, toKeychainBase58(k));\n  }\n  return {\n    userKey: encode(userKey),\n    backupKey: encode(backupKey),\n    bitgoKey: encode(bitgoKey),\n  };\n}\n\nfunction getKeysForUnsignedSweep([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  return getNamedKeys([userKey.neutered(), backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForKeyRecoveryService(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForFullSignedRecovery(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey, bitgoKey.neutered()], password);\n}\n\nconst walletKeys = getDefaultWalletKeys();\nconst keysUnsignedSweep = getKeysForUnsignedSweep(walletKeys.triple, walletPassphrase);\nconst keysKeyRecoveryService = getKeysForKeyRecoveryService(walletKeys.triple, walletPassphrase);\nconst keysFullSignedRecovery = getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase);\n\nconst exoticUserKeyPath = '99/99';\nconst exoticWalletKeys = new utxolib.bitgo.RootWalletKeys(keychains, [\n  exoticUserKeyPath,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n]);\nconst keysFullSignedRecoveryExotic = getKeysForFullSignedRecovery(exoticWalletKeys.triple, walletPassphrase);\n\nfunction run(\n  coin: AbstractUtxoCoin,\n  scriptTypes: ScriptType2Of3[],\n  walletKeys: RootWalletKeys,\n  params: {\n    keys: NamedKeys;\n    userKeyPath?: string;\n    krsProvider?: string;\n    hasUserSignature: boolean;\n    hasBackupSignature: boolean;\n    hasKrsOutput?: boolean;\n    feeRate?: number;\n  },\n  tags: string[] = []\n) {\n  describe(`Backup Key Recovery [${[coin.getChain(), ...tags, params.krsProvider].join(',')}]`, function () {\n    const externalWallet = getWalletKeys('external');\n    const recoveryDestination = getWalletAddress(coin.network, externalWallet);\n\n    let keyRecoveryServiceAddress: string;\n    let recovery: (BackupKeyRecoveryTransansaction | FormattedOfflineVaultTxInfo) & { txid?: string };\n    let recoveryTx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt;\n\n    // 1e8 * 9e7 < 9.007e15 but 2e8 * 9e7 > 9.007e15 to test both code paths in queryBlockchainUnspentsPath\n    const valueMul = coin.amountType === 'bigint' ? BigInt(9e7) : BigInt(1);\n    before('mock', function () {\n      sinon.stub(CoingeckoApi.prototype, 'getUSDPrice').resolves(69_420);\n    });\n\n    configOverride(function (config: Config) {\n      const configKrsProviders = { ...config.krsProviders };\n      configKrsProviders.dai.supportedCoins = [coin.getFamily()];\n      configKrsProviders.keyternal.supportedCoins = [coin.getFamily()];\n      keyRecoveryServiceAddress = getWalletAddress(coin.network, externalWallet, 0, 100);\n      configKrsProviders.keyternal.feeAddresses = { [coin.getChain()]: keyRecoveryServiceAddress };\n      config.krsProviders = configKrsProviders;\n    });\n\n    after(function () {\n      sinon.restore();\n    });\n\n    let recoverUnspents: utxolib.bitgo.Unspent<bigint>[];\n    let mockedApiUnspents: utxolib.bitgo.Unspent<bigint>[];\n\n    before('create recovery data', async function () {\n      recoverUnspents = scriptTypes.flatMap((scriptType, index) => [\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(1e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(2e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(3e8) * valueMul }, index, coin.network, walletKeys),\n      ]);\n\n      // If the coin is bch, convert the mocked unspent address to cashaddr format since that is the format that blockchair\n      // returns on the /dashboards/addresses response\n      mockedApiUnspents =\n        coin.getChain() === 'bch' || coin.getChain() === 'bcha'\n          ? recoverUnspents.map((u) => ({ ...u, address: coin.canonicalAddress(u.address, 'cashaddr').split(':')[1] }))\n          : recoverUnspents;\n\n      assert.strictEqual(mockedApiUnspents.length, recoverUnspents.length);\n      recovery = await backupKeyRecovery(coin, defaultBitGo, {\n        walletPassphrase,\n        recoveryDestination,\n        scan: 5,\n        ignoreAddressTypes: [],\n        userKeyPath: params.userKeyPath,\n        krsProvider: params.krsProvider,\n        ...params.keys,\n        recoveryProvider: new MockRecoveryProvider(mockedApiUnspents),\n      });\n      const txHex =\n        (recovery as BackupKeyRecoveryTransansaction).transactionHex ?? (recovery as FormattedOfflineVaultTxInfo).txHex;\n      const isPsbt = utxolib.bitgo.isPsbt(txHex);\n      recoveryTx = isPsbt\n        ? utxolib.bitgo.createPsbtFromHex(txHex, coin.network)\n        : utxolib.bitgo.createTransactionFromHex(txHex as string, coin.network, coin.amountType);\n      recovery.txid =\n        recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().getId() : recoveryTx.getId();\n    });\n\n    it('has correct recovery provider mocks', async function () {\n      const recoveryProvider = new MockRecoveryProvider(mockedApiUnspents);\n      (await recoveryProvider.getUnspentsForAddresses(mockedApiUnspents.map((u) => u.address))).length.should.eql(\n        mockedApiUnspents.length\n      );\n    });\n\n    it('matches fixture', async function () {\n      const fixtureCoin = getNormalTestnetCoin(coin);\n      const fixtureRecovery = { ...recovery };\n      if (fixtureRecovery.coin) {\n        fixtureRecovery.coin = fixtureCoin.getChain();\n      }\n\n      shouldEqualJSON(\n        fixtureRecovery,\n        await getFixture(\n          fixtureCoin,\n          `recovery/backupKeyRecovery-${(params.krsProvider ? tags.concat([params.krsProvider]) : tags).join('-')}`,\n          recovery\n        )\n      );\n    });\n\n    it('has expected input count', function () {\n      (recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.data.inputs : recoveryTx.ins).length.should.eql(\n        recoverUnspents.length\n      );\n    });\n\n    function checkInputsSignedBy(\n      tx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt,\n      rootKey: BIP32Interface,\n      expectCount: number\n    ) {\n      if (tx instanceof utxolib.bitgo.UtxoPsbt) {\n        function validate(tx: utxolib.bitgo.UtxoPsbt, inputIndex: number) {\n          try {\n            return tx.validateSignaturesOfInputHD(inputIndex, rootKey);\n          } catch (e) {\n            if (e.message === 'No signatures to validate') {\n              return false;\n            }\n            throw e;\n          }\n        }\n        tx.data.inputs.forEach((input, inputIndex) => {\n          validate(tx, inputIndex).should.eql(!!expectCount);\n        });\n      } else {\n        const prevOutputs = recoverUnspents\n          .map((u) => toOutput(u, coin.network))\n          .map((v) => ({ ...v, value: utxolib.bitgo.toTNumber(v.value, coin.amountType) }));\n        tx.ins.forEach((input, inputIndex) => {\n          const unspent = recoverUnspents[inputIndex] as WalletUnspent;\n          const { publicKey } = rootKey.derivePath(walletKeys.getDerivationPath(rootKey, unspent.chain, unspent.index));\n          const signatures = utxolib.bitgo\n            .getSignatureVerifications(\n              tx,\n              inputIndex,\n              utxolib.bitgo.toTNumber(unspent.value, coin.amountType),\n              { publicKey },\n              prevOutputs\n            )\n            .filter((s) => s.signedBy !== undefined);\n          signatures.length.should.eql(expectCount);\n        });\n      }\n    }\n\n    it((params.hasUserSignature ? 'has' : 'has no') + ' user signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.user, params.hasUserSignature ? 1 : 0);\n    });\n\n    it((params.hasBackupSignature ? 'has' : 'has no') + ' backup signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.backup, params.hasBackupSignature ? 1 : 0);\n    });\n\n    if (params.hasUserSignature && params.hasBackupSignature) {\n      it('has no placeholder signatures', function (this: mocha.Context) {\n        if (recoveryTx instanceof utxolib.bitgo.UtxoTransaction) {\n          recoveryTx.ins.forEach((input) => {\n            const parsed = utxolib.bitgo.parseSignatureScript(input);\n            switch (parsed.scriptType) {\n              case 'p2sh':\n              case 'p2shP2wsh':\n              case 'p2wsh':\n              case 'taprootScriptPathSpend':\n                parsed.signatures.forEach((signature, i) => {\n                  if (utxolib.bitgo.isPlaceholderSignature(signature)) {\n                    throw new Error(`placeholder signature at index ${i}`);\n                  }\n                });\n                break;\n              default:\n                throw new Error(`unexpected scriptType ${parsed.scriptType}`);\n            }\n          });\n        } else {\n          this.skip();\n        }\n      });\n    }\n\n    it((params.hasKrsOutput ? 'has' : 'has no') + ' key recovery service output', function () {\n      const outs = recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().outs : recoveryTx.outs;\n      outs.length.should.eql(1);\n      const outputAddresses = outs.map((o) => utxolib.address.fromOutputScript(o.script, recoveryTx.network));\n      outputAddresses\n        .includes(keyRecoveryServiceAddress)\n        .should.eql(!!params.hasKrsOutput && params.krsProvider === 'keyternal');\n      outputAddresses.includes(recoveryDestination).should.eql(true);\n    });\n  });\n}\n\nfunction runWithScriptTypes(\n  scriptTypes: ScriptType2Of3[],\n  { runKeyProviderTests = true }: { runKeyProviderTests?: boolean } = {}\n) {\n  utxoCoins\n    .filter((coin) => scriptTypes.every((type) => coin.supportsAddressType(type)))\n    .forEach((coin) => {\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysUnsignedSweep,\n          hasUserSignature: false,\n          hasBackupSignature: false,\n        },\n        ['unsignedRecovery', ...scriptTypes]\n      );\n\n      if (runKeyProviderTests) {\n        ['dai', 'keyternal'].forEach((krsProvider) => {\n          run(\n            coin,\n            scriptTypes,\n            walletKeys,\n            {\n              keys: keysKeyRecoveryService,\n              krsProvider: krsProvider,\n              hasUserSignature: true,\n              hasBackupSignature: false,\n              hasKrsOutput: false,\n            },\n            ['keyRecoveryService', ...scriptTypes]\n          );\n        });\n      }\n\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysFullSignedRecovery,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', ...scriptTypes]\n      );\n\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysFullSignedRecovery,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n          feeRate: 2,\n        },\n        ['fullSignedRecovery', 'fixedFeeRate', ...scriptTypes]\n      );\n\n      run(\n        coin,\n        scriptTypes,\n        exoticWalletKeys,\n        {\n          keys: keysFullSignedRecoveryExotic,\n          userKeyPath: exoticUserKeyPath,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', 'customUserKeyPath', ...scriptTypes]\n      );\n    });\n}\n\ndescribe('Backup Key Recovery', function () {\n  // compatible with all coins\n  runWithScriptTypes(['p2sh']);\n\n  // segwit compatible coins\n  runWithScriptTypes(['p2shP2wsh', 'p2wsh']);\n\n  // taproot compatible coins\n  runWithScriptTypes(['p2tr', 'p2trMusig2'], { runKeyProviderTests: false });\n});\n"]}
286
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backupKeyRecovery.js","sourceRoot":"","sources":["../../../../../test/unit/recovery/backupKeyRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport 'should';\nimport * as mocha from 'mocha';\nimport * as sinon from 'sinon';\nimport nock = require('nock');\nimport { BIP32Interface } from '@bitgo-beta/utxo-lib';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { Config, krsProviders, Triple } from '@bitgo-beta/sdk-core';\n\nimport {\n  AbstractUtxoCoin,\n  backupKeyRecovery,\n  BackupKeyRecoveryTransansaction,\n  CoingeckoApi,\n  FormattedOfflineVaultTxInfo,\n} from '../../../src';\nimport {\n  defaultBitGo,\n  encryptKeychain,\n  getDefaultWalletKeys,\n  getFixture,\n  getNormalTestnetCoin,\n  getWalletAddress,\n  getWalletKeys,\n  keychains,\n  shouldEqualJSON,\n  toKeychainBase58,\n  utxoCoins,\n} from '../util';\n\nimport { MockRecoveryProvider } from './mock';\n\nconst { toOutput } = utxolib.bitgo;\ntype WalletUnspent = utxolib.bitgo.WalletUnspent<bigint>;\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\ntype ScriptType2Of3 = utxolib.bitgo.outputScripts.ScriptType2Of3;\n\nconst config = { krsProviders };\n\nnock.disableNetConnect();\n\nfunction configOverride(f: (config: Config) => void) {\n  const backup = { ...krsProviders };\n  before(function () {\n    f(config);\n  });\n  after(function () {\n    Object.entries(backup).forEach(([k, v]) => {\n      config[k] = v;\n    });\n  });\n}\n\nconst walletPassphrase = 'lol';\n\ntype NamedKeys = {\n  userKey: string;\n  backupKey: string;\n  bitgoKey: string;\n};\n\nfunction getNamedKeys([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  function encode(k: BIP32Interface): string {\n    return k.isNeutered() ? k.toBase58() : encryptKeychain(password, toKeychainBase58(k));\n  }\n  return {\n    userKey: encode(userKey),\n    backupKey: encode(backupKey),\n    bitgoKey: encode(bitgoKey),\n  };\n}\n\nfunction getKeysForUnsignedSweep([userKey, backupKey, bitgoKey]: Triple<BIP32Interface>, password: string): NamedKeys {\n  return getNamedKeys([userKey.neutered(), backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForKeyRecoveryService(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey.neutered(), bitgoKey.neutered()], password);\n}\n\nfunction getKeysForFullSignedRecovery(\n  [userKey, backupKey, bitgoKey]: Triple<BIP32Interface>,\n  password: string\n): NamedKeys {\n  return getNamedKeys([userKey, backupKey, bitgoKey.neutered()], password);\n}\n\nconst walletKeys = getDefaultWalletKeys();\nconst keysUnsignedSweep = getKeysForUnsignedSweep(walletKeys.triple, walletPassphrase);\nconst keysKeyRecoveryService = getKeysForKeyRecoveryService(walletKeys.triple, walletPassphrase);\nconst keysFullSignedRecovery = getKeysForFullSignedRecovery(walletKeys.triple, walletPassphrase);\n\nconst exoticUserKeyPath = '99/99';\nconst exoticWalletKeys = new utxolib.bitgo.RootWalletKeys(keychains, [\n  exoticUserKeyPath,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n  utxolib.bitgo.RootWalletKeys.defaultPrefix,\n]);\nconst keysFullSignedRecoveryExotic = getKeysForFullSignedRecovery(exoticWalletKeys.triple, walletPassphrase);\n\nfunction run(\n  coin: AbstractUtxoCoin,\n  scriptTypes: ScriptType2Of3[],\n  walletKeys: RootWalletKeys,\n  params: {\n    keys: NamedKeys;\n    userKeyPath?: string;\n    krsProvider?: string;\n    hasUserSignature: boolean;\n    hasBackupSignature: boolean;\n    hasKrsOutput?: boolean;\n    feeRate?: number;\n  },\n  tags: string[] = []\n) {\n  describe(`Backup Key Recovery [${[coin.getChain(), ...tags, params.krsProvider].join(',')}]`, function () {\n    const externalWallet = getWalletKeys('external');\n    const recoveryDestination = getWalletAddress(coin.network, externalWallet);\n\n    let keyRecoveryServiceAddress: string;\n    let recovery: (BackupKeyRecoveryTransansaction | FormattedOfflineVaultTxInfo) & { txid?: string };\n    let recoveryTx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt;\n\n    // 1e8 * 9e7 < 9.007e15 but 2e8 * 9e7 > 9.007e15 to test both code paths in queryBlockchainUnspentsPath\n    const valueMul = coin.amountType === 'bigint' ? BigInt(9e7) : BigInt(1);\n    before('mock', function () {\n      sinon.stub(CoingeckoApi.prototype, 'getUSDPrice').resolves(69_420);\n    });\n\n    configOverride(function (config: Config) {\n      const configKrsProviders = { ...config.krsProviders };\n      configKrsProviders.dai.supportedCoins = [coin.getFamily()];\n      configKrsProviders.keyternal.supportedCoins = [coin.getFamily()];\n      keyRecoveryServiceAddress = getWalletAddress(coin.network, externalWallet, 0, 100);\n      configKrsProviders.keyternal.feeAddresses = { [coin.getChain()]: keyRecoveryServiceAddress };\n      config.krsProviders = configKrsProviders;\n    });\n\n    after(function () {\n      sinon.restore();\n    });\n\n    let recoverUnspents: utxolib.bitgo.Unspent<bigint>[];\n    let mockedApiUnspents: utxolib.bitgo.Unspent<bigint>[];\n\n    before('create recovery data', async function () {\n      this.timeout(10_000);\n      recoverUnspents = scriptTypes.flatMap((scriptType, index) => [\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(1e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(2e8) * valueMul }, index, coin.network, walletKeys),\n        utxolib.testutil.toUnspent({ scriptType, value: BigInt(3e8) * valueMul }, index, coin.network, walletKeys),\n      ]);\n\n      // If the coin is bch, convert the mocked unspent address to cashaddr format since that is the format that blockchair\n      // returns on the /dashboards/addresses response\n      mockedApiUnspents =\n        coin.getChain() === 'bch' || coin.getChain() === 'bcha'\n          ? recoverUnspents.map((u) => ({ ...u, address: coin.canonicalAddress(u.address, 'cashaddr').split(':')[1] }))\n          : recoverUnspents;\n\n      assert.strictEqual(mockedApiUnspents.length, recoverUnspents.length);\n      recovery = await backupKeyRecovery(coin, defaultBitGo, {\n        walletPassphrase,\n        recoveryDestination,\n        scan: 5,\n        ignoreAddressTypes: [],\n        userKeyPath: params.userKeyPath,\n        krsProvider: params.krsProvider,\n        ...params.keys,\n        recoveryProvider: new MockRecoveryProvider(mockedApiUnspents),\n      });\n      const txHex =\n        (recovery as BackupKeyRecoveryTransansaction).transactionHex ?? (recovery as FormattedOfflineVaultTxInfo).txHex;\n      const isPsbt = utxolib.bitgo.isPsbt(txHex);\n      recoveryTx = isPsbt\n        ? utxolib.bitgo.createPsbtFromHex(txHex, coin.network)\n        : utxolib.bitgo.createTransactionFromHex(txHex as string, coin.network, coin.amountType);\n      recovery.txid =\n        recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().getId() : recoveryTx.getId();\n    });\n\n    it('has correct recovery provider mocks', async function () {\n      const recoveryProvider = new MockRecoveryProvider(mockedApiUnspents);\n      (await recoveryProvider.getUnspentsForAddresses(mockedApiUnspents.map((u) => u.address))).length.should.eql(\n        mockedApiUnspents.length\n      );\n    });\n\n    it('matches fixture', async function () {\n      const fixtureCoin = getNormalTestnetCoin(coin);\n      const fixtureRecovery = { ...recovery };\n      if (fixtureRecovery.coin) {\n        fixtureRecovery.coin = fixtureCoin.getChain();\n      }\n\n      shouldEqualJSON(\n        fixtureRecovery,\n        await getFixture(\n          fixtureCoin,\n          `recovery/backupKeyRecovery-${(params.krsProvider ? tags.concat([params.krsProvider]) : tags).join('-')}`,\n          recovery\n        )\n      );\n    });\n\n    it('has expected input count', function () {\n      (recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.data.inputs : recoveryTx.ins).length.should.eql(\n        recoverUnspents.length\n      );\n    });\n\n    function checkInputsSignedBy(\n      tx: utxolib.bitgo.UtxoTransaction<number | bigint> | utxolib.bitgo.UtxoPsbt,\n      rootKey: BIP32Interface,\n      expectCount: number\n    ) {\n      if (tx instanceof utxolib.bitgo.UtxoPsbt) {\n        function validate(tx: utxolib.bitgo.UtxoPsbt, inputIndex: number) {\n          try {\n            return tx.validateSignaturesOfInputHD(inputIndex, rootKey);\n          } catch (e) {\n            if (e.message === 'No signatures to validate') {\n              return false;\n            }\n            throw e;\n          }\n        }\n        tx.data.inputs.forEach((input, inputIndex) => {\n          validate(tx, inputIndex).should.eql(!!expectCount);\n        });\n      } else {\n        const prevOutputs = recoverUnspents\n          .map((u) => toOutput(u, coin.network))\n          .map((v) => ({ ...v, value: utxolib.bitgo.toTNumber(v.value, coin.amountType) }));\n        tx.ins.forEach((input, inputIndex) => {\n          const unspent = recoverUnspents[inputIndex] as WalletUnspent;\n          const { publicKey } = rootKey.derivePath(walletKeys.getDerivationPath(rootKey, unspent.chain, unspent.index));\n          const signatures = utxolib.bitgo\n            .getSignatureVerifications(\n              tx,\n              inputIndex,\n              utxolib.bitgo.toTNumber(unspent.value, coin.amountType),\n              { publicKey },\n              prevOutputs\n            )\n            .filter((s) => s.signedBy !== undefined);\n          signatures.length.should.eql(expectCount);\n        });\n      }\n    }\n\n    it((params.hasUserSignature ? 'has' : 'has no') + ' user signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.user, params.hasUserSignature ? 1 : 0);\n    });\n\n    it((params.hasBackupSignature ? 'has' : 'has no') + ' backup signature', function () {\n      checkInputsSignedBy(recoveryTx, walletKeys.backup, params.hasBackupSignature ? 1 : 0);\n    });\n\n    if (params.hasUserSignature && params.hasBackupSignature) {\n      it('has no placeholder signatures', function (this: mocha.Context) {\n        if (recoveryTx instanceof utxolib.bitgo.UtxoTransaction) {\n          recoveryTx.ins.forEach((input) => {\n            const parsed = utxolib.bitgo.parseSignatureScript(input);\n            switch (parsed.scriptType) {\n              case 'p2sh':\n              case 'p2shP2wsh':\n              case 'p2wsh':\n              case 'taprootScriptPathSpend':\n                parsed.signatures.forEach((signature, i) => {\n                  if (utxolib.bitgo.isPlaceholderSignature(signature)) {\n                    throw new Error(`placeholder signature at index ${i}`);\n                  }\n                });\n                break;\n              default:\n                throw new Error(`unexpected scriptType ${parsed.scriptType}`);\n            }\n          });\n        } else {\n          this.skip();\n        }\n      });\n    }\n\n    it((params.hasKrsOutput ? 'has' : 'has no') + ' key recovery service output', function () {\n      const outs = recoveryTx instanceof utxolib.bitgo.UtxoPsbt ? recoveryTx.getUnsignedTx().outs : recoveryTx.outs;\n      outs.length.should.eql(1);\n      const outputAddresses = outs.map((o) => utxolib.address.fromOutputScript(o.script, recoveryTx.network));\n      outputAddresses\n        .includes(keyRecoveryServiceAddress)\n        .should.eql(!!params.hasKrsOutput && params.krsProvider === 'keyternal');\n      outputAddresses.includes(recoveryDestination).should.eql(true);\n    });\n  });\n}\n\nfunction runWithScriptTypes(\n  scriptTypes: ScriptType2Of3[],\n  { runKeyProviderTests = true }: { runKeyProviderTests?: boolean } = {}\n) {\n  utxoCoins\n    .filter((coin) => scriptTypes.every((type) => coin.supportsAddressType(type)))\n    .forEach((coin) => {\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysUnsignedSweep,\n          hasUserSignature: false,\n          hasBackupSignature: false,\n        },\n        ['unsignedRecovery', ...scriptTypes]\n      );\n\n      if (runKeyProviderTests) {\n        ['dai', 'keyternal'].forEach((krsProvider) => {\n          run(\n            coin,\n            scriptTypes,\n            walletKeys,\n            {\n              keys: keysKeyRecoveryService,\n              krsProvider: krsProvider,\n              hasUserSignature: true,\n              hasBackupSignature: false,\n              hasKrsOutput: false,\n            },\n            ['keyRecoveryService', ...scriptTypes]\n          );\n        });\n      }\n\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysFullSignedRecovery,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', ...scriptTypes]\n      );\n\n      run(\n        coin,\n        scriptTypes,\n        walletKeys,\n        {\n          keys: keysFullSignedRecovery,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n          feeRate: 2,\n        },\n        ['fullSignedRecovery', 'fixedFeeRate', ...scriptTypes]\n      );\n\n      run(\n        coin,\n        scriptTypes,\n        exoticWalletKeys,\n        {\n          keys: keysFullSignedRecoveryExotic,\n          userKeyPath: exoticUserKeyPath,\n          hasUserSignature: true,\n          hasBackupSignature: true,\n        },\n        ['fullSignedRecovery', 'customUserKeyPath', ...scriptTypes]\n      );\n    });\n}\n\ndescribe('Backup Key Recovery', function () {\n  // compatible with all coins\n  runWithScriptTypes(['p2sh']);\n\n  // segwit compatible coins\n  runWithScriptTypes(['p2shP2wsh', 'p2wsh']);\n\n  // taproot compatible coins\n  runWithScriptTypes(['p2tr', 'p2trMusig2'], { runKeyProviderTests: false });\n});\n"]}
@@ -41,44 +41,52 @@ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
41
41
  const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
42
42
  const wasm_utxo_1 = require("@bitgo/wasm-utxo");
43
43
  const fixedScript_1 = require("../../../../src/transaction/fixedScript");
44
- function hasWasmUtxoSupport(network) {
45
- return ![
46
- utxolib.networks.bitcoincash,
47
- utxolib.networks.bitcoingold,
48
- utxolib.networks.ecash,
49
- utxolib.networks.zcash,
50
- ].includes(utxolib.getMainnet(network));
51
- }
44
+ const util_1 = require("./util");
52
45
  function describeTransactionWith(acidTest) {
53
46
  describe(`${acidTest.name}`, function () {
47
+ let psbt;
54
48
  let psbtBytes;
49
+ let walletXpubs;
50
+ let customChangeWalletXpubs;
51
+ let wasmPsbt;
55
52
  let refExplanation;
56
53
  before('prepare', function () {
57
- const psbt = acidTest.createPsbt();
54
+ psbt = acidTest.createPsbt();
58
55
  refExplanation = (0, fixedScript_1.explainPsbt)(psbt, { pubs: acidTest.rootWalletKeys }, acidTest.network, {
59
56
  strict: true,
60
57
  });
61
58
  psbtBytes = psbt.toBuffer();
59
+ const networkName = utxolib.getNetworkName(acidTest.network);
60
+ (0, strict_1.default)(networkName);
61
+ walletXpubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());
62
+ customChangeWalletXpubs = acidTest.otherWalletKeys.triple.map((k) => k.neutered().toBase58());
63
+ if ((0, util_1.hasWasmUtxoSupport)(acidTest.network)) {
64
+ wasmPsbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.fromBytes(psbtBytes, networkName);
65
+ }
62
66
  });
63
67
  it('should match the expected values for explainPsbt', function () {
64
68
  // note: `outputs` means external outputs here
65
69
  strict_1.default.strictEqual(refExplanation.outputs.length, 3);
66
70
  strict_1.default.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);
67
- strict_1.default.strictEqual(refExplanation.outputAmount, '2700');
71
+ strict_1.default.strictEqual(refExplanation.outputAmount, '1800');
68
72
  strict_1.default.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);
69
73
  refExplanation.changeOutputs.forEach((change) => {
70
74
  strict_1.default.strictEqual(change.amount, '900');
71
75
  strict_1.default.strictEqual(typeof change.address, 'string');
72
76
  });
73
77
  });
78
+ it('reference implementation should support custom change outputs', function () {
79
+ const customChangeExplanation = (0, fixedScript_1.explainPsbt)(psbt, { pubs: acidTest.rootWalletKeys, customChangePubs: acidTest.otherWalletKeys }, acidTest.network, { strict: true });
80
+ strict_1.default.ok(customChangeExplanation.customChangeOutputs);
81
+ strict_1.default.strictEqual(customChangeExplanation.changeOutputs.length, refExplanation.changeOutputs.length);
82
+ strict_1.default.strictEqual(customChangeExplanation.outputs.length, refExplanation.outputs.length - 1);
83
+ strict_1.default.strictEqual(customChangeExplanation.customChangeOutputs.length, 1);
84
+ strict_1.default.strictEqual(customChangeExplanation.customChangeOutputs[0].amount, '900');
85
+ });
74
86
  it('should match explainPsbtWasm', function () {
75
- if (!hasWasmUtxoSupport(acidTest.network)) {
87
+ if (!(0, util_1.hasWasmUtxoSupport)(acidTest.network)) {
76
88
  return this.skip();
77
89
  }
78
- const networkName = utxolib.getNetworkName(acidTest.network);
79
- (0, strict_1.default)(networkName);
80
- const wasmPsbt = wasm_utxo_1.fixedScriptWallet.BitGoPsbt.fromBytes(psbtBytes, networkName);
81
- const walletXpubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58());
82
90
  const wasmExplanation = (0, fixedScript_1.explainPsbtWasm)(wasmPsbt, walletXpubs, {
83
91
  replayProtection: {
84
92
  outputScripts: [acidTest.getReplayProtectionOutputScript()],
@@ -100,9 +108,25 @@ function describeTransactionWith(acidTest) {
100
108
  }
101
109
  }
102
110
  });
111
+ if (acidTest.network !== utxolib.networks.bitcoin) {
112
+ return;
113
+ }
114
+ // extended test suite for bitcoin
115
+ it('returns custom change outputs when parameter is set', function () {
116
+ const wasmExplanation = (0, fixedScript_1.explainPsbtWasm)(wasmPsbt, walletXpubs, {
117
+ replayProtection: {
118
+ outputScripts: [acidTest.getReplayProtectionOutputScript()],
119
+ },
120
+ customChangeWalletXpubs,
121
+ });
122
+ strict_1.default.ok(wasmExplanation.customChangeOutputs);
123
+ strict_1.default.deepStrictEqual(wasmExplanation.outputs.length, 2);
124
+ strict_1.default.deepStrictEqual(wasmExplanation.customChangeOutputs.length, 1);
125
+ strict_1.default.deepStrictEqual(wasmExplanation.customChangeOutputs[0].amount, '900');
126
+ });
103
127
  });
104
128
  }
105
129
  describe('explainPsbt(Wasm)', function () {
106
130
  utxo_lib_1.testutil.AcidTest.suite().forEach((test) => describeTransactionWith(test));
107
131
  });
108
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXhwbGFpblBzYnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi90ZXN0L3VuaXQvdHJhbnNhY3Rpb24vZml4ZWRTY3JpcHQvZXhwbGFpblBzYnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxnRUFBd0M7QUFFeEMsOERBQWdEO0FBQ2hELG1EQUFnRDtBQUNoRCxnREFBNkQ7QUFHN0QseUVBQXVGO0FBRXZGLFNBQVMsa0JBQWtCLENBQUMsT0FBd0I7SUFDbEQsT0FBTyxDQUFDO1FBQ04sT0FBTyxDQUFDLFFBQVEsQ0FBQyxXQUFXO1FBQzVCLE9BQU8sQ0FBQyxRQUFRLENBQUMsV0FBVztRQUM1QixPQUFPLENBQUMsUUFBUSxDQUFDLEtBQUs7UUFDdEIsT0FBTyxDQUFDLFFBQVEsQ0FBQyxLQUFLO0tBQ3ZCLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztBQUMxQyxDQUFDO0FBRUQsU0FBUyx1QkFBdUIsQ0FBQyxRQUEyQjtJQUMxRCxRQUFRLENBQUMsR0FBRyxRQUFRLENBQUMsSUFBSSxFQUFFLEVBQUU7UUFDM0IsSUFBSSxTQUFpQixDQUFDO1FBQ3RCLElBQUksY0FBc0MsQ0FBQztRQUMzQyxNQUFNLENBQUMsU0FBUyxFQUFFO1lBQ2hCLE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxVQUFVLEVBQUUsQ0FBQztZQUNuQyxjQUFjLEdBQUcsSUFBQSx5QkFBVyxFQUFDLElBQUksRUFBRSxFQUFFLElBQUksRUFBRSxRQUFRLENBQUMsY0FBYyxFQUFFLEVBQUUsUUFBUSxDQUFDLE9BQU8sRUFBRTtnQkFDdEYsTUFBTSxFQUFFLElBQUk7YUFDYixDQUFDLENBQUM7WUFDSCxTQUFTLEdBQUcsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQzlCLENBQUMsQ0FBQyxDQUFDO1FBRUgsRUFBRSxD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QyxDQUFDLENBQUM7QUFDN0UsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgYXNzZXJ0IGZyb20gJ25vZGU6YXNzZXJ0L3N0cmljdCc7XG5cbmltcG9ydCAqIGFzIHV0eG9saWIgZnJvbSAnQGJpdGdvLWJldGEvdXR4by1saWInO1xuaW1wb3J0IHsgdGVzdHV0aWwgfSBmcm9tICdAYml0Z28tYmV0YS91dHhvLWxpYic7XG5pbXBvcnQgeyBmaXhlZFNjcmlwdFdhbGxldCwgVHJpcGxlIH0gZnJvbSAnQGJpdGdvL3dhc20tdXR4byc7XG5cbmltcG9ydCB0eXBlIHsgVHJhbnNhY3Rpb25FeHBsYW5hdGlvbiB9IGZyb20gJy4uLy4uLy4uLy4uL3NyYy90cmFuc2FjdGlvbi9maXhlZFNjcmlwdC9leHBsYWluVHJhbnNhY3Rpb24nO1xuaW1wb3J0IHsgZXhwbGFpblBzYnQsIGV4cGxhaW5Qc2J0V2FzbSB9IGZyb20gJy4uLy4uLy4uLy4uL3NyYy90cmFuc2FjdGlvbi9maXhlZFNjcmlwdCc7XG5cbmZ1bmN0aW9uIGhhc1dhc21VdHhvU3VwcG9ydChuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmspOiBib29sZWFuIHtcbiAgcmV0dXJuICFbXG4gICAgdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luY2FzaCxcbiAgICB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5nb2xkLFxuICAgIHV0eG9saWIubmV0d29ya3MuZWNhc2gsXG4gICAgdXR4b2xpYi5uZXR3b3Jrcy56Y2FzaCxcbiAgXS5pbmNsdWRlcyh1dHhvbGliLmdldE1haW5uZXQobmV0d29yaykpO1xufVxuXG5mdW5jdGlvbiBkZXNjcmliZVRyYW5zYWN0aW9uV2l0aChhY2lkVGVzdDogdGVzdHV0aWwuQWNpZFRlc3QpIHtcbiAgZGVzY3JpYmUoYCR7YWNpZFRlc3QubmFtZX1gLCBmdW5jdGlvbiAoKSB7XG4gICAgbGV0IHBzYnRCeXRlczogQnVmZmVyO1xuICAgIGxldCByZWZFeHBsYW5hdGlvbjogVHJhbnNhY3Rpb25FeHBsYW5hdGlvbjtcbiAgICBiZWZvcmUoJ3ByZXBhcmUnLCBmdW5jdGlvbiAoKSB7XG4gICAgICBjb25zdCBwc2J0ID0gYWNpZFRlc3QuY3JlYXRlUHNidCgpO1xuICAgICAgcmVmRXhwbGFuYXRpb24gPSBleHBsYWluUHNidChwc2J0LCB7IHB1YnM6IGFjaWRUZXN0LnJvb3RXYWxsZXRLZXlzIH0sIGFjaWRUZXN0Lm5ldHdvcmssIHtcbiAgICAgICAgc3RyaWN0OiB0cnVlLFxuICAgICAgfSk7XG4gICAgICBwc2J0Qnl0ZXMgPSBwc2J0LnRvQnVmZmVyKCk7XG4gICAgfSk7XG5cbiAgICBpdCgnc2hvdWxkIG1hdGNoIHRoZSBleHBlY3RlZCB2YWx1ZXMgZm9yIGV4cGxhaW5Qc2J0JywgZnVuY3Rpb24gKCkge1xuICAgICAgLy8gbm90ZTogYG91dHB1dHNgIG1lYW5zIGV4dGVybmFsIG91dHB1dHMgaGVyZVxuICAgICAgYXNzZXJ0LnN0cmljdEVxdWFsKHJlZkV4cGxhbmF0aW9uLm91dHB1dHMubGVuZ3RoLCAzKTtcbiAgICAgIGFzc2VydC5zdHJpY3RFcXVhbChyZWZFeHBsYW5hdGlvbi5jaGFuZ2VPdXRwdXRzLmxlbmd0aCwgYWNpZFRlc3Qub3V0cHV0cy5sZW5ndGggLSAzKTtcbiAgICAgIGFzc2VydC5zdHJpY3RFcXVhbChyZWZFeHBsYW5hdGlvbi5vdXRwdXRBbW91bnQsICcyNzAwJyk7XG4gICAgICBhc3NlcnQuc3RyaWN0RXF1YWwocmVmRXhwbGFuYXRpb24uY2hhbmdlT3V0cHV0cy5sZW5ndGgsIGFjaWRUZXN0Lm91dHB1dHMubGVuZ3RoIC0gMyk7XG4gICAgICByZWZFeHBsYW5hdGlvbi5jaGFuZ2VPdXRwdXRzLmZvckVhY2goKGNoYW5nZSkgPT4ge1xuICAgICAgICBhc3NlcnQuc3RyaWN0RXF1YWwoY2hhbmdlLmFtb3VudCwgJzkwMCcpO1xuICAgICAgICBhc3NlcnQuc3RyaWN0RXF1YWwodHlwZW9mIGNoYW5nZS5hZGRyZXNzLCAnc3RyaW5nJyk7XG4gICAgICB9KTtcbiAgICB9KTtcblxuICAgIGl0KCdzaG91bGQgbWF0Y2ggZXhwbGFpblBzYnRXYXNtJywgZnVuY3Rpb24gKCkge1xuICAgICAgaWYgKCFoYXNXYXNtVXR4b1N1cHBvcnQoYWNpZFRlc3QubmV0d29yaykpIHtcbiAgICAgICAgcmV0dXJuIHRoaXMuc2tpcCgpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBuZXR3b3JrTmFtZSA9IHV0eG9saWIuZ2V0TmV0d29ya05hbWUoYWNpZFRlc3QubmV0d29yayk7XG4gICAgICBhc3NlcnQobmV0d29ya05hbWUpO1xuICAgICAgY29uc3Qgd2FzbVBzYnQgPSBmaXhlZFNjcmlwdFdhbGxldC5CaXRHb1BzYnQuZnJvbUJ5dGVzKHBzYnRCeXRlcywgbmV0d29ya05hbWUpO1xuICAgICAgY29uc3Qgd2FsbGV0WHB1YnMgPSBhY2lkVGVzdC5yb290V2FsbGV0S2V5cy50cmlwbGUubWFwKChrKSA9PiBrLm5ldXRlcmVkKCkudG9CYXNlNTgoKSkgYXMgVHJpcGxlPHN0cmluZz47XG4gICAgICBjb25zdCB3YXNtRXhwbGFuYXRpb24gPSBleHBsYWluUHNidFdhc20od2FzbVBzYnQsIHdhbGxldFhwdWJzLCB7XG4gICAgICAgIHJlcGxheVByb3RlY3Rpb246IHtcbiAgICAgICAgICBvdXRwdXRTY3JpcHRzOiBbYWNpZFRlc3QuZ2V0UmVwbGF5UHJvdGVjdGlvbk91dHB1dFNjcmlwdCgpXSxcbiAgICAgICAgfSxcbiAgICAgIH0pO1xuXG4gICAgICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyhyZWZFeHBsYW5hdGlvbikpIHtcbiAgICAgICAgY29uc3QgcmVmVmFsdWUgPSByZWZFeHBsYW5hdGlvbltrZXldO1xuICAgICAgICBjb25zdCB3YXNtVmFsdWUgPSB3YXNtRXhwbGFuYXRpb25ba2V5XTtcbiAgICAgICAgc3dpdGNoIChrZXkpIHtcbiAgICAgICAgICBjYXNlICdkaXNwbGF5T3JkZXInOlxuICAgICAgICAgIGNhc2UgJ2lucHV0U2lnbmF0dXJlcyc6XG4gICAgICAgICAgY2FzZSAnc2lnbmF0dXJlcyc6XG4gICAgICAgICAgICAvLyB0aGVzZSBhcmUgZGVwcmVjYXRlZCBmaWVsZHMgdGhhdCB3ZSB3YW50IHRvIGdldCByaWQgb2ZcbiAgICAgICAgICAgIGFzc2VydC5kZWVwU3RyaWN0RXF1YWwod2FzbVZhbHVlLCB1bmRlZmluZWQpO1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICAgIGFzc2VydC5kZWVwU3RyaWN0RXF1YWwod2FzbVZhbHVlLCByZWZWYWx1ZSwgYG1pc21hdGNoIGZvciBrZXkgJHtrZXl9YCk7XG4gICAgICAgICAgICBicmVhaztcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0pO1xuICB9KTtcbn1cblxuZGVzY3JpYmUoJ2V4cGxhaW5Qc2J0KFdhc20pJywgZnVuY3Rpb24gKCkge1xuICB0ZXN0dXRpbC5BY2lkVGVzdC5zdWl0ZSgpLmZvckVhY2goKHRlc3QpID0+IGRlc2NyaWJlVHJhbnNhY3Rpb25XaXRoKHRlc3QpKTtcbn0pO1xuIl19
132
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"explainPsbt.js","sourceRoot":"","sources":["../../../../../../test/unit/transaction/fixedScript/explainPsbt.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'node:assert/strict';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { testutil } from '@bitgo-beta/utxo-lib';\nimport { fixedScriptWallet, Triple } from '@bitgo/wasm-utxo';\n\nimport type { TransactionExplanation } from '../../../../src/transaction/fixedScript/explainTransaction';\nimport { explainPsbt, explainPsbtWasm } from '../../../../src/transaction/fixedScript';\n\nimport { hasWasmUtxoSupport } from './util';\n\nfunction describeTransactionWith(acidTest: testutil.AcidTest) {\n  describe(`${acidTest.name}`, function () {\n    let psbt: utxolib.bitgo.UtxoPsbt;\n    let psbtBytes: Buffer;\n    let walletXpubs: Triple<string>;\n    let customChangeWalletXpubs: Triple<string> | undefined;\n    let wasmPsbt: fixedScriptWallet.BitGoPsbt;\n    let refExplanation: TransactionExplanation;\n    before('prepare', function () {\n      psbt = acidTest.createPsbt();\n      refExplanation = explainPsbt(psbt, { pubs: acidTest.rootWalletKeys }, acidTest.network, {\n        strict: true,\n      });\n      psbtBytes = psbt.toBuffer();\n      const networkName = utxolib.getNetworkName(acidTest.network);\n      assert(networkName);\n      walletXpubs = acidTest.rootWalletKeys.triple.map((k) => k.neutered().toBase58()) as Triple<string>;\n      customChangeWalletXpubs = acidTest.otherWalletKeys.triple.map((k) => k.neutered().toBase58()) as Triple<string>;\n      if (hasWasmUtxoSupport(acidTest.network)) {\n        wasmPsbt = fixedScriptWallet.BitGoPsbt.fromBytes(psbtBytes, networkName);\n      }\n    });\n\n    it('should match the expected values for explainPsbt', function () {\n      // note: `outputs` means external outputs here\n      assert.strictEqual(refExplanation.outputs.length, 3);\n      assert.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);\n      assert.strictEqual(refExplanation.outputAmount, '1800');\n      assert.strictEqual(refExplanation.changeOutputs.length, acidTest.outputs.length - 3);\n      refExplanation.changeOutputs.forEach((change) => {\n        assert.strictEqual(change.amount, '900');\n        assert.strictEqual(typeof change.address, 'string');\n      });\n    });\n\n    it('reference implementation should support custom change outputs', function () {\n      const customChangeExplanation = explainPsbt(\n        psbt,\n        { pubs: acidTest.rootWalletKeys, customChangePubs: acidTest.otherWalletKeys },\n        acidTest.network,\n        { strict: true }\n      );\n      assert.ok(customChangeExplanation.customChangeOutputs);\n      assert.strictEqual(customChangeExplanation.changeOutputs.length, refExplanation.changeOutputs.length);\n      assert.strictEqual(customChangeExplanation.outputs.length, refExplanation.outputs.length - 1);\n      assert.strictEqual(customChangeExplanation.customChangeOutputs.length, 1);\n      assert.strictEqual(customChangeExplanation.customChangeOutputs[0].amount, '900');\n    });\n\n    it('should match explainPsbtWasm', function () {\n      if (!hasWasmUtxoSupport(acidTest.network)) {\n        return this.skip();\n      }\n\n      const wasmExplanation = explainPsbtWasm(wasmPsbt, walletXpubs, {\n        replayProtection: {\n          outputScripts: [acidTest.getReplayProtectionOutputScript()],\n        },\n      });\n\n      for (const key of Object.keys(refExplanation)) {\n        const refValue = refExplanation[key];\n        const wasmValue = wasmExplanation[key];\n        switch (key) {\n          case 'displayOrder':\n          case 'inputSignatures':\n          case 'signatures':\n            // these are deprecated fields that we want to get rid of\n            assert.deepStrictEqual(wasmValue, undefined);\n            break;\n          default:\n            assert.deepStrictEqual(wasmValue, refValue, `mismatch for key ${key}`);\n            break;\n        }\n      }\n    });\n\n    if (acidTest.network !== utxolib.networks.bitcoin) {\n      return;\n    }\n\n    // extended test suite for bitcoin\n\n    it('returns custom change outputs when parameter is set', function () {\n      const wasmExplanation = explainPsbtWasm(wasmPsbt, walletXpubs, {\n        replayProtection: {\n          outputScripts: [acidTest.getReplayProtectionOutputScript()],\n        },\n        customChangeWalletXpubs,\n      });\n      assert.ok(wasmExplanation.customChangeOutputs);\n      assert.deepStrictEqual(wasmExplanation.outputs.length, 2);\n      assert.deepStrictEqual(wasmExplanation.customChangeOutputs.length, 1);\n      assert.deepStrictEqual(wasmExplanation.customChangeOutputs[0].amount, '900');\n    });\n  });\n}\n\ndescribe('explainPsbt(Wasm)', function () {\n  testutil.AcidTest.suite().forEach((test) => describeTransactionWith(test));\n});\n"]}