@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
@@ -5,8 +5,52 @@ import { fetchKeychains, getKeySignatures, toKeychainTriple, UtxoNamedKeychains
5
5
  import { outputDifference } from '../outputDifference';
6
6
  import { fromExtendedAddressFormatToScript, toExtendedAddressFormat } from '../recipient';
7
7
  import { parseOutput } from './parseOutput';
8
+ async function parseRbfTransaction(coin, params) {
9
+ const { txParams, wallet } = params;
10
+ assert(txParams.rbfTxIds);
11
+ assert(txParams.rbfTxIds.length === 1);
12
+ const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });
13
+ const recipients = txToBeReplaced.outputs.flatMap((output) => {
14
+ // For self-sends, the walletId will be the same as the wallet's id
15
+ if (output.wallet === wallet.id()) {
16
+ return [];
17
+ }
18
+ return [coin.toCanonicalTransactionRecipient(output)];
19
+ });
20
+ // Recurse into parseTransaction with the derived recipients and without rbfTxIds
21
+ return parseTransaction(coin, {
22
+ ...params,
23
+ txParams: {
24
+ ...txParams,
25
+ recipients,
26
+ rbfTxIds: undefined,
27
+ },
28
+ });
29
+ }
30
+ function toExpectedOutputs(coin, txParams) {
31
+ // verify that each recipient from txParams has their own output
32
+ const expectedOutputs = (txParams.recipients ?? []).flatMap((output) => {
33
+ if (output.address === undefined) {
34
+ if (output.amount.toString() !== '0') {
35
+ throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${output}`);
36
+ }
37
+ return [output];
38
+ }
39
+ return [{ ...output, address: coin.canonicalAddress(output.address) }];
40
+ });
41
+ if (txParams.allowExternalChangeAddress && txParams.changeAddress) {
42
+ // when an external change address is explicitly specified, count all outputs going towards that
43
+ // address in the expected outputs (regardless of the output amount)
44
+ expectedOutputs.push({ address: coin.canonicalAddress(txParams.changeAddress), amount: 'max' });
45
+ }
46
+ return expectedOutputs;
47
+ }
8
48
  export async function parseTransaction(coin, params) {
9
49
  const { txParams, txPrebuild, wallet, verification = {}, reqId } = params;
50
+ // Branch off early for RBF transactions
51
+ if (txParams.rbfTxIds) {
52
+ return parseRbfTransaction(coin, params);
53
+ }
10
54
  if (!_.isUndefined(verification.disableNetworking) && !_.isBoolean(verification.disableNetworking)) {
11
55
  throw new Error('verification.disableNetworking must be a boolean');
12
56
  }
@@ -26,48 +70,7 @@ export async function parseTransaction(coin, params) {
26
70
  if (_.isUndefined(txPrebuild.txHex)) {
27
71
  throw new Error('missing required txPrebuild property txHex');
28
72
  }
29
- // obtain all outputs
30
- const explanation = await coin.explainTransaction({
31
- txHex: txPrebuild.txHex,
32
- txInfo: txPrebuild.txInfo,
33
- pubs: keychainArray.map((k) => k.pub),
34
- });
35
- const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];
36
- let expectedOutputs;
37
- if (txParams.rbfTxIds) {
38
- assert(txParams.rbfTxIds.length === 1);
39
- const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });
40
- expectedOutputs = txToBeReplaced.outputs.flatMap((output) => {
41
- // For self-sends, the walletId will be the same as the wallet's id
42
- if (output.wallet === wallet.id()) {
43
- return [];
44
- }
45
- return [coin.toCanonicalTransactionRecipient(output)];
46
- });
47
- }
48
- else {
49
- // verify that each recipient from txParams has their own output
50
- expectedOutputs = (txParams.recipients ?? []).flatMap((output) => {
51
- if (output.address === undefined) {
52
- if (output.amount.toString() !== '0') {
53
- throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${output}`);
54
- }
55
- return [output];
56
- }
57
- return [{ ...output, address: coin.canonicalAddress(output.address) }];
58
- });
59
- if (txParams.allowExternalChangeAddress && txParams.changeAddress) {
60
- // when an external change address is explicitly specified, count all outputs going towards that
61
- // address in the expected outputs (regardless of the output amount)
62
- expectedOutputs.push(...allOutputs.flatMap((output) => {
63
- if (output.address === undefined ||
64
- output.address !== coin.canonicalAddress(txParams.changeAddress)) {
65
- return [];
66
- }
67
- return [{ ...output, address: coin.canonicalAddress(output.address) }];
68
- }));
69
- }
70
- }
73
+ const expectedOutputs = toExpectedOutputs(coin, txParams);
71
74
  // get the keychains from the custom change wallet if needed
72
75
  let customChange;
73
76
  const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};
@@ -92,6 +95,13 @@ export async function parseTransaction(coin, params) {
92
95
  };
93
96
  }
94
97
  }
98
+ // obtain all outputs
99
+ const explanation = await coin.explainTransaction({
100
+ txHex: txPrebuild.txHex,
101
+ txInfo: txPrebuild.txInfo,
102
+ pubs: keychainArray.map((k) => k.pub),
103
+ });
104
+ const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];
95
105
  /**
96
106
  * Loop through all the outputs and classify each of them as either internal spends
97
107
  * or external spends by setting the "external" property to true or false on the output object.
@@ -163,4 +173,4 @@ export async function parseTransaction(coin, params) {
163
173
  customChange,
164
174
  };
165
175
  }
166
- //# 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"]}
176
+ //# 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"}
@@ -41,8 +41,8 @@ export function assertValidTransactionRecipient(output) {
41
41
  // We will verify that the amount is zero, and if it isnt then we will throw an error.
42
42
  if (!output.address || isScriptRecipient(output.address)) {
43
43
  if (output.amount.toString() !== '0') {
44
- throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${JSON.stringify(output)}`);
44
+ throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: amount: ${output.amount}, address: ${output.address}`);
45
45
  }
46
46
  }
47
47
  }
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjaXBpZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3RyYW5zYWN0aW9uL3JlY2lwaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLHNCQUFzQixDQUFDO0FBRWhELE1BQU0scUJBQXFCLEdBQUcsZUFBZSxDQUFDO0FBRTlDOzs7R0FHRztBQUNILE1BQU0sVUFBVSxpQkFBaUIsQ0FBQyxPQUFlO0lBQy9DLE9BQU8sT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDLFVBQVUsQ0FBQyxxQkFBcUIsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDO0FBQy9FLENBQUM7QUFFRDs7OztHQUlHO0FBQ0gsTUFBTSxVQUFVLHlCQUF5QixDQUFDLGVBQXVCO0lBQy9ELElBQUksaUJBQWlCLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQztRQUN2QyxPQUFPLEVBQUUsTUFBTSxFQUFFLGVBQWUsQ0FBQyxLQUFLLENBQUMscUJBQXFCLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztJQUN6RSxDQUFDO0lBQ0QsT0FBTyxFQUFFLE9BQU8sRUFBRSxlQUFlLEVBQUUsQ0FBQztBQUN0QyxDQUFDO0FBRUQsTUFBTSxVQUFVLGlDQUFpQyxDQUFDLGVBQXVCLEVBQUUsT0FBd0I7SUFDakcsTUFBTSxNQUFNLEdBQUcseUJBQXlCLENBQUMsZUFBZSxDQUFDLENBQUM7SUFDMUQsSUFBSSxRQUFRLElBQUksTUFBTSxFQUFFLENBQUM7UUFDdkIsT0FBTyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDM0MsQ0FBQztJQUNELE9BQU8sT0FBTyxDQUFDLGFBQWEsQ0FBQyx3QkFBd0IsQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBQ2pGLENBQUM7QUFFRDs7Ozs7R0FLRztBQUNILE1BQU0sVUFBVSx1QkFBdUIsQ0FBQyxNQUFjLEVBQUUsT0FBd0I7SUFDOUUsT0FBTyxNQUFNLENBQUMsQ0FBQyxDQUFDLEtBQUssT0FBTyxDQUFDLE9BQU8sQ0FBQyxTQUFTO1FBQzVDLENBQUMsQ0FBQyxHQUFHLHFCQUFxQixHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLEVBQUU7UUFDckQsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsZ0JBQWdCLENBQUMsTUFBTSxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBQ3hELENBQUM7QUFFRCxNQUFNLFVBQVUsK0JBQStCLENBQUMsTUFBOEQ7SUFDNUcsK0dBQStHO0lBQy9HLHNGQUFzRjtJQUN0RixJQUFJLENBQUMsTUFBTSxDQUFDLE9BQU8sSUFBSSxpQkFBaUIsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztRQUN6RCxJQUFJLE1BQU0sQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLEtBQUssR0FBRyxFQUFFLENBQUM7WUFDckMsTUFBTSxJQUFJLEtBQUssQ0FBQywrREFBK0QsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLENBQUM7UUFDM0csQ0FBQztJQUNILENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgdXR4b2xpYiBmcm9tICdAYml0Z28tYmV0YS91dHhvLWxpYic7XG5cbmNvbnN0IFNjcmlwdFJlY2lwaWVudFByZWZpeCA9ICdzY3JpcHRQdWJLZXk6JztcblxuLyoqXG4gKiBDaGVjayBpZiB0aGUgYWRkcmVzcyBpcyBhIHNjcmlwdCByZWNpcGllbnQgKHN0YXJ0cyB3aXRoIGBzY3JpcHRQdWJLZXk6YCkuXG4gKiBAcGFyYW0gYWRkcmVzc1xuICovXG5leHBvcnQgZnVuY3Rpb24gaXNTY3JpcHRSZWNpcGllbnQoYWRkcmVzczogc3RyaW5nKTogYm9vbGVhbiB7XG4gIHJldHVybiBhZGRyZXNzLnRvTG93ZXJDYXNlKCkuc3RhcnRzV2l0aChTY3JpcHRSZWNpcGllbnRQcmVmaXgudG9Mb3dlckNhc2UoKSk7XG59XG5cbi8qKlxuICogQW4gZXh0ZW5kZWQgYWRkcmVzcyBpcyBvbmUgdGhhdCBlbmNvZGVzIGVpdGhlciBhIHJlZ3VsYXIgYWRkcmVzcyBvciBhIGhleCBlbmNvZGVkIHNjcmlwdCB3aXRoIHRoZSBwcmVmaXggYHNjcmlwdFB1YktleTpgLlxuICogVGhpcyBmdW5jdGlvbiBjb252ZXJ0cyB0aGUgZXh0ZW5kZWQgYWRkcmVzcyBmb3JtYXQgdG8gZWl0aGVyIGEgc2NyaXB0IG9yIGFuIGFkZHJlc3MuXG4gKiBAcGFyYW0gZXh0ZW5kZWRBZGRyZXNzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmcm9tRXh0ZW5kZWRBZGRyZXNzRm9ybWF0KGV4dGVuZGVkQWRkcmVzczogc3RyaW5nKTogeyBhZGRyZXNzOiBzdHJpbmcgfSB8IHsgc2NyaXB0OiBzdHJpbmcgfSB7XG4gIGlmIChpc1NjcmlwdFJlY2lwaWVudChleHRlbmRlZEFkZHJlc3MpKSB7XG4gICAgcmV0dXJuIHsgc2NyaXB0OiBleHRlbmRlZEFkZHJlc3Muc2xpY2UoU2NyaXB0UmVjaXBpZW50UHJlZml4Lmxlbmd0aCkgfTtcbiAgfVxuICByZXR1cm4geyBhZGRyZXNzOiBleHRlbmRlZEFkZHJlc3MgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGZyb21FeHRlbmRlZEFkZHJlc3NGb3JtYXRUb1NjcmlwdChleHRlbmRlZEFkZHJlc3M6IHN0cmluZywgbmV0d29yazogdXR4b2xpYi5OZXR3b3JrKTogQnVmZmVyIHtcbiAgY29uc3QgcmVzdWx0ID0gZnJvbUV4dGVuZGVkQWRkcmVzc0Zvcm1hdChleHRlbmRlZEFkZHJlc3MpO1xuICBpZiAoJ3NjcmlwdCcgaW4gcmVzdWx0KSB7XG4gICAgcmV0dXJuIEJ1ZmZlci5mcm9tKHJlc3VsdC5zY3JpcHQsICdoZXgnKTtcbiAgfVxuICByZXR1cm4gdXR4b2xpYi5hZGRyZXNzRm9ybWF0LnRvT3V0cHV0U2NyaXB0VHJ5Rm9ybWF0cyhyZXN1bHQuYWRkcmVzcywgbmV0d29yayk7XG59XG5cbi8qKlxuICogQ29udmVydCBhIHNjcmlwdCBvciBhZGRyZXNzIHRvIHRoZSBleHRlbmRlZCBhZGRyZXNzIGZvcm1hdC5cbiAqIEBwYXJhbSBzY3JpcHRcbiAqIEBwYXJhbSBuZXR3b3JrXG4gKiBAcmV0dXJucyBpZiB0aGUgc2NyaXB0IGlzIGFuIE9QX1JFVFVSTiBzY3JpcHQsIHRoZW4gaXQgd2lsbCBiZSBwcmVmaXhlZCB3aXRoIGBzY3JpcHRQdWJLZXk6YCwgb3RoZXJ3aXNlIGl0IHdpbGwgYmUgY29udmVydGVkIHRvIGFuIGFkZHJlc3MuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB0b0V4dGVuZGVkQWRkcmVzc0Zvcm1hdChzY3JpcHQ6IEJ1ZmZlciwgbmV0d29yazogdXR4b2xpYi5OZXR3b3JrKTogc3RyaW5nIHtcbiAgcmV0dXJuIHNjcmlwdFswXSA9PT0gdXR4b2xpYi5vcGNvZGVzLk9QX1JFVFVSTlxuICAgID8gYCR7U2NyaXB0UmVjaXBpZW50UHJlZml4fSR7c2NyaXB0LnRvU3RyaW5nKCdoZXgnKX1gXG4gICAgOiB1dHhvbGliLmFkZHJlc3MuZnJvbU91dHB1dFNjcmlwdChzY3JpcHQsIG5ldHdvcmspO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gYXNzZXJ0VmFsaWRUcmFuc2FjdGlvblJlY2lwaWVudChvdXRwdXQ6IHsgYW1vdW50OiBiaWdpbnQgfCBudW1iZXIgfCBzdHJpbmc7IGFkZHJlc3M/OiBzdHJpbmcgfSk6IHZvaWQge1xuICAvLyBJbiB0aGUgY2FzZSB0aGF0IHRoaXMgaXMgYW4gT1BfUkVUVVJOIG91dHB1dCBvciBhbm90aGVyIG5vbi1lbmNvZGFibGUgc2NyaXB0UHVia2V5LCB3ZSBkb250IGhhdmUgYW4gYWRkcmVzcy5cbiAgLy8gV2Ugd2lsbCB2ZXJpZnkgdGhhdCB0aGUgYW1vdW50IGlzIHplcm8sIGFuZCBpZiBpdCBpc250IHRoZW4gd2Ugd2lsbCB0aHJvdyBhbiBlcnJvci5cbiAgaWYgKCFvdXRwdXQuYWRkcmVzcyB8fCBpc1NjcmlwdFJlY2lwaWVudChvdXRwdXQuYWRkcmVzcykpIHtcbiAgICBpZiAob3V0cHV0LmFtb3VudC50b1N0cmluZygpICE9PSAnMCcpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgT25seSB6ZXJvIGFtb3VudHMgYWxsb3dlZCBmb3Igbm9uLWVuY29kZWFibGUgc2NyaXB0UHVia2V5czogJHtKU09OLnN0cmluZ2lmeShvdXRwdXQpfWApO1xuICAgIH1cbiAgfVxufVxuIl19
48
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjaXBpZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3RyYW5zYWN0aW9uL3JlY2lwaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssT0FBTyxNQUFNLHNCQUFzQixDQUFDO0FBRWhELE1BQU0scUJBQXFCLEdBQUcsZUFBZSxDQUFDO0FBRTlDOzs7R0FHRztBQUNILE1BQU0sVUFBVSxpQkFBaUIsQ0FBQyxPQUFlO0lBQy9DLE9BQU8sT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDLFVBQVUsQ0FBQyxxQkFBcUIsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDO0FBQy9FLENBQUM7QUFFRDs7OztHQUlHO0FBQ0gsTUFBTSxVQUFVLHlCQUF5QixDQUFDLGVBQXVCO0lBQy9ELElBQUksaUJBQWlCLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQztRQUN2QyxPQUFPLEVBQUUsTUFBTSxFQUFFLGVBQWUsQ0FBQyxLQUFLLENBQUMscUJBQXFCLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztJQUN6RSxDQUFDO0lBQ0QsT0FBTyxFQUFFLE9BQU8sRUFBRSxlQUFlLEVBQUUsQ0FBQztBQUN0QyxDQUFDO0FBRUQsTUFBTSxVQUFVLGlDQUFpQyxDQUFDLGVBQXVCLEVBQUUsT0FBd0I7SUFDakcsTUFBTSxNQUFNLEdBQUcseUJBQXlCLENBQUMsZUFBZSxDQUFDLENBQUM7SUFDMUQsSUFBSSxRQUFRLElBQUksTUFBTSxFQUFFLENBQUM7UUFDdkIsT0FBTyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDM0MsQ0FBQztJQUNELE9BQU8sT0FBTyxDQUFDLGFBQWEsQ0FBQyx3QkFBd0IsQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBQ2pGLENBQUM7QUFFRDs7Ozs7R0FLRztBQUNILE1BQU0sVUFBVSx1QkFBdUIsQ0FBQyxNQUFjLEVBQUUsT0FBd0I7SUFDOUUsT0FBTyxNQUFNLENBQUMsQ0FBQyxDQUFDLEtBQUssT0FBTyxDQUFDLE9BQU8sQ0FBQyxTQUFTO1FBQzVDLENBQUMsQ0FBQyxHQUFHLHFCQUFxQixHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLEVBQUU7UUFDckQsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsZ0JBQWdCLENBQUMsTUFBTSxFQUFFLE9BQU8sQ0FBQyxDQUFDO0FBQ3hELENBQUM7QUFFRCxNQUFNLFVBQVUsK0JBQStCLENBQUMsTUFBOEQ7SUFDNUcsK0dBQStHO0lBQy9HLHNGQUFzRjtJQUN0RixJQUFJLENBQUMsTUFBTSxDQUFDLE9BQU8sSUFBSSxpQkFBaUIsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztRQUN6RCxJQUFJLE1BQU0sQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLEtBQUssR0FBRyxFQUFFLENBQUM7WUFDckMsTUFBTSxJQUFJLEtBQUssQ0FDYix1RUFBdUUsTUFBTSxDQUFDLE1BQU0sY0FBYyxNQUFNLENBQUMsT0FBTyxFQUFFLENBQ25ILENBQUM7UUFDSixDQUFDO0lBQ0gsQ0FBQztBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyB1dHhvbGliIGZyb20gJ0BiaXRnby1iZXRhL3V0eG8tbGliJztcblxuY29uc3QgU2NyaXB0UmVjaXBpZW50UHJlZml4ID0gJ3NjcmlwdFB1YktleTonO1xuXG4vKipcbiAqIENoZWNrIGlmIHRoZSBhZGRyZXNzIGlzIGEgc2NyaXB0IHJlY2lwaWVudCAoc3RhcnRzIHdpdGggYHNjcmlwdFB1YktleTpgKS5cbiAqIEBwYXJhbSBhZGRyZXNzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpc1NjcmlwdFJlY2lwaWVudChhZGRyZXNzOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgcmV0dXJuIGFkZHJlc3MudG9Mb3dlckNhc2UoKS5zdGFydHNXaXRoKFNjcmlwdFJlY2lwaWVudFByZWZpeC50b0xvd2VyQ2FzZSgpKTtcbn1cblxuLyoqXG4gKiBBbiBleHRlbmRlZCBhZGRyZXNzIGlzIG9uZSB0aGF0IGVuY29kZXMgZWl0aGVyIGEgcmVndWxhciBhZGRyZXNzIG9yIGEgaGV4IGVuY29kZWQgc2NyaXB0IHdpdGggdGhlIHByZWZpeCBgc2NyaXB0UHViS2V5OmAuXG4gKiBUaGlzIGZ1bmN0aW9uIGNvbnZlcnRzIHRoZSBleHRlbmRlZCBhZGRyZXNzIGZvcm1hdCB0byBlaXRoZXIgYSBzY3JpcHQgb3IgYW4gYWRkcmVzcy5cbiAqIEBwYXJhbSBleHRlbmRlZEFkZHJlc3NcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZyb21FeHRlbmRlZEFkZHJlc3NGb3JtYXQoZXh0ZW5kZWRBZGRyZXNzOiBzdHJpbmcpOiB7IGFkZHJlc3M6IHN0cmluZyB9IHwgeyBzY3JpcHQ6IHN0cmluZyB9IHtcbiAgaWYgKGlzU2NyaXB0UmVjaXBpZW50KGV4dGVuZGVkQWRkcmVzcykpIHtcbiAgICByZXR1cm4geyBzY3JpcHQ6IGV4dGVuZGVkQWRkcmVzcy5zbGljZShTY3JpcHRSZWNpcGllbnRQcmVmaXgubGVuZ3RoKSB9O1xuICB9XG4gIHJldHVybiB7IGFkZHJlc3M6IGV4dGVuZGVkQWRkcmVzcyB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZnJvbUV4dGVuZGVkQWRkcmVzc0Zvcm1hdFRvU2NyaXB0KGV4dGVuZGVkQWRkcmVzczogc3RyaW5nLCBuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmspOiBCdWZmZXIge1xuICBjb25zdCByZXN1bHQgPSBmcm9tRXh0ZW5kZWRBZGRyZXNzRm9ybWF0KGV4dGVuZGVkQWRkcmVzcyk7XG4gIGlmICgnc2NyaXB0JyBpbiByZXN1bHQpIHtcbiAgICByZXR1cm4gQnVmZmVyLmZyb20ocmVzdWx0LnNjcmlwdCwgJ2hleCcpO1xuICB9XG4gIHJldHVybiB1dHhvbGliLmFkZHJlc3NGb3JtYXQudG9PdXRwdXRTY3JpcHRUcnlGb3JtYXRzKHJlc3VsdC5hZGRyZXNzLCBuZXR3b3JrKTtcbn1cblxuLyoqXG4gKiBDb252ZXJ0IGEgc2NyaXB0IG9yIGFkZHJlc3MgdG8gdGhlIGV4dGVuZGVkIGFkZHJlc3MgZm9ybWF0LlxuICogQHBhcmFtIHNjcmlwdFxuICogQHBhcmFtIG5ldHdvcmtcbiAqIEByZXR1cm5zIGlmIHRoZSBzY3JpcHQgaXMgYW4gT1BfUkVUVVJOIHNjcmlwdCwgdGhlbiBpdCB3aWxsIGJlIHByZWZpeGVkIHdpdGggYHNjcmlwdFB1YktleTpgLCBvdGhlcndpc2UgaXQgd2lsbCBiZSBjb252ZXJ0ZWQgdG8gYW4gYWRkcmVzcy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHRvRXh0ZW5kZWRBZGRyZXNzRm9ybWF0KHNjcmlwdDogQnVmZmVyLCBuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmspOiBzdHJpbmcge1xuICByZXR1cm4gc2NyaXB0WzBdID09PSB1dHhvbGliLm9wY29kZXMuT1BfUkVUVVJOXG4gICAgPyBgJHtTY3JpcHRSZWNpcGllbnRQcmVmaXh9JHtzY3JpcHQudG9TdHJpbmcoJ2hleCcpfWBcbiAgICA6IHV0eG9saWIuYWRkcmVzcy5mcm9tT3V0cHV0U2NyaXB0KHNjcmlwdCwgbmV0d29yayk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBhc3NlcnRWYWxpZFRyYW5zYWN0aW9uUmVjaXBpZW50KG91dHB1dDogeyBhbW91bnQ6IGJpZ2ludCB8IG51bWJlciB8IHN0cmluZzsgYWRkcmVzcz86IHN0cmluZyB9KTogdm9pZCB7XG4gIC8vIEluIHRoZSBjYXNlIHRoYXQgdGhpcyBpcyBhbiBPUF9SRVRVUk4gb3V0cHV0IG9yIGFub3RoZXIgbm9uLWVuY29kYWJsZSBzY3JpcHRQdWJrZXksIHdlIGRvbnQgaGF2ZSBhbiBhZGRyZXNzLlxuICAvLyBXZSB3aWxsIHZlcmlmeSB0aGF0IHRoZSBhbW91bnQgaXMgemVybywgYW5kIGlmIGl0IGlzbnQgdGhlbiB3ZSB3aWxsIHRocm93IGFuIGVycm9yLlxuICBpZiAoIW91dHB1dC5hZGRyZXNzIHx8IGlzU2NyaXB0UmVjaXBpZW50KG91dHB1dC5hZGRyZXNzKSkge1xuICAgIGlmIChvdXRwdXQuYW1vdW50LnRvU3RyaW5nKCkgIT09ICcwJykge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgT25seSB6ZXJvIGFtb3VudHMgYWxsb3dlZCBmb3Igbm9uLWVuY29kZWFibGUgc2NyaXB0UHVia2V5czogYW1vdW50OiAke291dHB1dC5hbW91bnR9LCBhZGRyZXNzOiAke291dHB1dC5hZGRyZXNzfWBcbiAgICAgICk7XG4gICAgfVxuICB9XG59XG4iXX0=
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@bitgo-beta/abstract-utxo",
3
- "version": "1.6.1-alpha.421",
3
+ "version": "1.6.1-alpha.422",
4
4
  "description": "BitGo SDK coin library for UTXO base implementation",
5
5
  "main": "./dist/cjs/src/index.js",
6
6
  "module": "./dist/esm/index.js",
@@ -60,14 +60,14 @@
60
60
  ]
61
61
  },
62
62
  "dependencies": {
63
- "@bitgo-beta/blockapis": "1.3.3-alpha.422",
64
- "@bitgo-beta/sdk-api": "1.6.1-alpha.421",
65
- "@bitgo-beta/sdk-core": "2.4.1-alpha.421",
66
- "@bitgo-beta/secp256k1": "1.0.1-alpha.379",
67
- "@bitgo-beta/unspents": "0.11.3-alpha.422",
68
- "@bitgo-beta/utxo-core": "1.0.1-alpha.152",
69
- "@bitgo-beta/utxo-lib": "4.0.1-alpha.422",
70
- "@bitgo-beta/utxo-ord": "1.1.3-alpha.376",
63
+ "@bitgo-beta/blockapis": "1.3.3-alpha.423",
64
+ "@bitgo-beta/sdk-api": "1.6.1-alpha.422",
65
+ "@bitgo-beta/sdk-core": "2.4.1-alpha.422",
66
+ "@bitgo-beta/secp256k1": "1.0.1-alpha.380",
67
+ "@bitgo-beta/unspents": "0.11.3-alpha.423",
68
+ "@bitgo-beta/utxo-core": "1.0.1-alpha.153",
69
+ "@bitgo-beta/utxo-lib": "4.0.1-alpha.423",
70
+ "@bitgo-beta/utxo-ord": "1.1.3-alpha.377",
71
71
  "@bitgo/wasm-utxo": "1.3.0",
72
72
  "@types/lodash": "^4.14.121",
73
73
  "@types/superagent": "4.1.15",
@@ -79,8 +79,8 @@
79
79
  "superagent": "^9.0.1"
80
80
  },
81
81
  "devDependencies": {
82
- "@bitgo-beta/sdk-test": "^9.1.13",
82
+ "@bitgo-beta/sdk-test": "^9.1.15",
83
83
  "mocha": "^10.2.0"
84
84
  },
85
- "gitHead": "827938bc6035dc4925d9b1d38fcf6cc367a05348"
85
+ "gitHead": "2aaf7336e7740019d9582799c9006a90e1974b77"
86
86
  }