@bitgo-beta/abstract-utxo 1.6.1-alpha.421 → 1.6.1-alpha.423
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.
- package/dist/cjs/src/abstractUtxoCoin.d.ts +1 -8
- package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/cjs/src/abstractUtxoCoin.js +4 -18
- package/dist/cjs/src/address/fixedScript.d.ts +2 -4
- package/dist/cjs/src/address/fixedScript.d.ts.map +1 -1
- package/dist/cjs/src/address/fixedScript.js +11 -6
- package/dist/cjs/src/keychains.d.ts +6 -2
- package/dist/cjs/src/keychains.d.ts.map +1 -1
- package/dist/cjs/src/keychains.js +33 -1
- package/dist/cjs/src/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/cjs/src/recovery/backupKeyRecovery.js +20 -13
- package/dist/cjs/src/transaction/descriptor/explainPsbt.d.ts +2 -2
- package/dist/cjs/src/transaction/descriptor/explainPsbt.d.ts.map +1 -1
- package/dist/cjs/src/transaction/descriptor/explainPsbt.js +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts +1 -0
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.js +42 -18
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts +14 -5
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/explainTransaction.js +61 -38
- package/dist/cjs/src/transaction/fixedScript/parseOutput.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/parseOutput.js +9 -6
- package/dist/cjs/src/transaction/fixedScript/parseTransaction.d.ts.map +1 -1
- package/dist/cjs/src/transaction/fixedScript/parseTransaction.js +53 -43
- package/dist/cjs/src/transaction/recipient.d.ts.map +1 -1
- package/dist/cjs/src/transaction/recipient.js +2 -2
- package/dist/cjs/test/unit/recovery/backupKeyRecovery.js +2 -1
- package/dist/cjs/test/unit/transaction/fixedScript/explainPsbt.js +40 -16
- package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +100 -28
- package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts +3 -0
- package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +1 -0
- package/dist/cjs/test/unit/transaction/fixedScript/util.js +46 -0
- package/dist/cjs/tsconfig.tsbuildinfo +1 -1
- package/dist/esm/abstractUtxoCoin.d.ts +1 -8
- package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
- package/dist/esm/abstractUtxoCoin.js +4 -18
- package/dist/esm/address/fixedScript.d.ts +2 -4
- package/dist/esm/address/fixedScript.d.ts.map +1 -1
- package/dist/esm/address/fixedScript.js +12 -7
- package/dist/esm/keychains.d.ts +6 -2
- package/dist/esm/keychains.d.ts.map +1 -1
- package/dist/esm/keychains.js +32 -1
- package/dist/esm/recovery/backupKeyRecovery.d.ts.map +1 -1
- package/dist/esm/recovery/backupKeyRecovery.js +20 -13
- package/dist/esm/transaction/descriptor/explainPsbt.d.ts +2 -2
- package/dist/esm/transaction/descriptor/explainPsbt.d.ts.map +1 -1
- package/dist/esm/transaction/descriptor/explainPsbt.js +1 -1
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts +1 -0
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/explainPsbtWasm.js +42 -18
- package/dist/esm/transaction/fixedScript/explainTransaction.d.ts +14 -5
- package/dist/esm/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/explainTransaction.js +61 -38
- package/dist/esm/transaction/fixedScript/parseOutput.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/parseOutput.js +9 -6
- package/dist/esm/transaction/fixedScript/parseTransaction.d.ts.map +1 -1
- package/dist/esm/transaction/fixedScript/parseTransaction.js +53 -43
- package/dist/esm/transaction/recipient.d.ts.map +1 -1
- package/dist/esm/transaction/recipient.js +2 -2
- package/package.json +11 -11
|
@@ -42,37 +42,49 @@ const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
|
42
42
|
const utxocore = __importStar(require("@bitgo-beta/utxo-core"));
|
|
43
43
|
const recipient_1 = require("../recipient");
|
|
44
44
|
const getPayGoVerificationPubkey_1 = require("../getPayGoVerificationPubkey");
|
|
45
|
+
const keychains_1 = require("../../keychains");
|
|
46
|
+
function toChangeOutput(txOutput, network, changeInfo) {
|
|
47
|
+
if (!changeInfo) {
|
|
48
|
+
return undefined;
|
|
49
|
+
}
|
|
50
|
+
const address = (0, recipient_1.toExtendedAddressFormat)(txOutput.script, network);
|
|
51
|
+
const change = changeInfo.find((change) => change.address === address);
|
|
52
|
+
if (!change) {
|
|
53
|
+
return undefined;
|
|
54
|
+
}
|
|
55
|
+
return {
|
|
56
|
+
address,
|
|
57
|
+
amount: txOutput.value.toString(),
|
|
58
|
+
chain: change.chain,
|
|
59
|
+
index: change.index,
|
|
60
|
+
external: false,
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
function outputSum(outputs) {
|
|
64
|
+
return outputs.reduce((sum, output) => sum + BigInt(output.amount), BigInt(0));
|
|
65
|
+
}
|
|
45
66
|
function explainCommon(tx, params, network) {
|
|
46
67
|
const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];
|
|
47
|
-
let spendAmount = BigInt(0);
|
|
48
|
-
let changeAmount = BigInt(0);
|
|
49
68
|
const changeOutputs = [];
|
|
50
|
-
const
|
|
51
|
-
const
|
|
52
|
-
const
|
|
69
|
+
const customChangeOutputs = [];
|
|
70
|
+
const externalOutputs = [];
|
|
71
|
+
const { changeInfo, customChangeInfo } = params;
|
|
53
72
|
tx.outs.forEach((currentOutput) => {
|
|
54
73
|
// Try to encode the script pubkey with an address. If it fails, try to parse it as an OP_RETURN output with the prefix.
|
|
55
74
|
// If that fails, then it is an unrecognized scriptPubkey and should fail
|
|
56
75
|
const currentAddress = (0, recipient_1.toExtendedAddressFormat)(currentOutput.script, network);
|
|
57
76
|
const currentAmount = BigInt(currentOutput.value);
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
address: currentAddress,
|
|
67
|
-
amount: currentAmount.toString(),
|
|
68
|
-
chain: change.chain,
|
|
69
|
-
index: change.index,
|
|
70
|
-
external: false,
|
|
71
|
-
});
|
|
77
|
+
const changeOutput = toChangeOutput(currentOutput, network, changeInfo);
|
|
78
|
+
if (changeOutput) {
|
|
79
|
+
changeOutputs.push(changeOutput);
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
const customChangeOutput = toChangeOutput(currentOutput, network, customChangeInfo);
|
|
83
|
+
if (customChangeOutput) {
|
|
84
|
+
customChangeOutputs.push(customChangeOutput);
|
|
72
85
|
return;
|
|
73
86
|
}
|
|
74
|
-
|
|
75
|
-
outputs.push({
|
|
87
|
+
externalOutputs.push({
|
|
76
88
|
address: currentAddress,
|
|
77
89
|
amount: currentAmount.toString(),
|
|
78
90
|
// If changeInfo has a length greater than or equal to zero, it means that the change information
|
|
@@ -84,10 +96,12 @@ function explainCommon(tx, params, network) {
|
|
|
84
96
|
});
|
|
85
97
|
});
|
|
86
98
|
const outputDetails = {
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
outputs,
|
|
99
|
+
outputs: externalOutputs,
|
|
100
|
+
outputAmount: outputSum(externalOutputs).toString(),
|
|
90
101
|
changeOutputs,
|
|
102
|
+
changeAmount: outputSum(changeOutputs).toString(),
|
|
103
|
+
customChangeAmount: outputSum(customChangeOutputs).toString(),
|
|
104
|
+
customChangeOutputs,
|
|
91
105
|
};
|
|
92
106
|
let fee;
|
|
93
107
|
let locktime;
|
|
@@ -158,19 +172,9 @@ function getChainAndIndexFromBip32Derivations(output) {
|
|
|
158
172
|
});
|
|
159
173
|
return utxolib.bitgo.getChainAndIndexFromPath(paths[0]);
|
|
160
174
|
}
|
|
161
|
-
function getChangeInfo(psbt) {
|
|
175
|
+
function getChangeInfo(psbt, walletKeys) {
|
|
162
176
|
try {
|
|
163
|
-
|
|
164
|
-
const derivationInformation = getChainAndIndexFromBip32Derivations(psbt.data.outputs[i]);
|
|
165
|
-
if (!derivationInformation) {
|
|
166
|
-
throw new Error('could not find derivation information on bip32Derivation or tapBip32Derivation');
|
|
167
|
-
}
|
|
168
|
-
return {
|
|
169
|
-
address: utxolib.address.fromOutputScript(psbt.txOutputs[i].script, psbt.network),
|
|
170
|
-
external: false,
|
|
171
|
-
...derivationInformation,
|
|
172
|
-
};
|
|
173
|
-
});
|
|
177
|
+
walletKeys = walletKeys ?? utxolib.bitgo.getSortedRootNodes(psbt);
|
|
174
178
|
}
|
|
175
179
|
catch (e) {
|
|
176
180
|
if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {
|
|
@@ -178,6 +182,17 @@ function getChangeInfo(psbt) {
|
|
|
178
182
|
}
|
|
179
183
|
throw e;
|
|
180
184
|
}
|
|
185
|
+
return utxolib.bitgo.findWalletOutputIndices(psbt, walletKeys).map((i) => {
|
|
186
|
+
const derivationInformation = getChainAndIndexFromBip32Derivations(psbt.data.outputs[i]);
|
|
187
|
+
if (!derivationInformation) {
|
|
188
|
+
throw new Error('could not find derivation information on bip32Derivation or tapBip32Derivation');
|
|
189
|
+
}
|
|
190
|
+
return {
|
|
191
|
+
address: utxolib.address.fromOutputScript(psbt.txOutputs[i].script, psbt.network),
|
|
192
|
+
external: false,
|
|
193
|
+
...derivationInformation,
|
|
194
|
+
};
|
|
195
|
+
});
|
|
181
196
|
}
|
|
182
197
|
/**
|
|
183
198
|
* Extract PayGo address proof information from the PSBT if present
|
|
@@ -267,6 +282,11 @@ function getBip322MessageInfoAndVerify(psbt, network) {
|
|
|
267
282
|
/**
|
|
268
283
|
* Decompose a raw psbt into useful information, such as the total amounts,
|
|
269
284
|
* change amounts, and transaction outputs.
|
|
285
|
+
*
|
|
286
|
+
* @param psbt {bitgo.UtxoPsbt} The PSBT to explain
|
|
287
|
+
* @param pubs {bitgo.RootWalletKeys | string[]} The public keys to use for the explanation
|
|
288
|
+
* @param network {utxolib.Network} The network to use for the explanation
|
|
289
|
+
* @param strict {boolean} Whether to throw an error if the PayGo address proof is invalid
|
|
270
290
|
*/
|
|
271
291
|
function explainPsbt(psbt, params, network, { strict = false } = {}) {
|
|
272
292
|
const payGoVerificationInfo = getPayGoVerificationInfo(psbt, network);
|
|
@@ -283,8 +303,11 @@ function explainPsbt(psbt, params, network, { strict = false } = {}) {
|
|
|
283
303
|
}
|
|
284
304
|
const messages = getBip322MessageInfoAndVerify(psbt, network);
|
|
285
305
|
const changeInfo = getChangeInfo(psbt);
|
|
306
|
+
const customChangeInfo = params.customChangePubs
|
|
307
|
+
? getChangeInfo(psbt, (0, keychains_1.toBip32Triple)(params.customChangePubs))
|
|
308
|
+
: undefined;
|
|
286
309
|
const tx = psbt.getUnsignedTx();
|
|
287
|
-
const common = explainCommon(tx, { ...params, changeInfo }, network);
|
|
310
|
+
const common = explainCommon(tx, { ...params, changeInfo, customChangeInfo }, network);
|
|
288
311
|
const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);
|
|
289
312
|
// Set fee from subtracting inputs from outputs
|
|
290
313
|
const outputAmount = psbt.txOutputs.reduce((cumulative, curr) => cumulative + BigInt(curr.value), BigInt(0));
|
|
@@ -318,4 +341,4 @@ function explainLegacyTx(tx, params, network) {
|
|
|
318
341
|
signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),
|
|
319
342
|
};
|
|
320
343
|
}
|
|
321
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"explainTransaction.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/explainTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { bip322 } from '@bitgo-beta/utxo-core';\nimport { BIP32Interface, bip32 } from '@bitgo-beta/secp256k1';\nimport { bitgo } from '@bitgo-beta/utxo-lib';\nimport { ITransactionExplanation as BaseTransactionExplanation, Triple } from '@bitgo-beta/sdk-core';\nimport * as utxocore from '@bitgo-beta/utxo-core';\n\nimport type { Bip322Message } from '../../abstractUtxoCoin';\nimport type { Output, FixedScriptWalletOutput } from '../types';\nimport { toExtendedAddressFormat } from '../recipient';\nimport { getPayGoVerificationPubkey } from '../getPayGoVerificationPubkey';\n\n// ===== Transaction Explanation Type Definitions =====\n\nexport interface AbstractUtxoTransactionExplanation<TFee = string> extends BaseTransactionExplanation<TFee, string> {\n  /** NOTE: this actually only captures external outputs */\n  outputs: Output[];\n  changeOutputs: Output[];\n\n  /**\n   * BIP322 messages extracted from the transaction inputs.\n   * These messages are used for verifying the transaction against the BIP322 standard.\n   */\n  messages?: Bip322Message[];\n}\n\n/** @deprecated - the signature fields are not very useful */\ninterface TransactionExplanationWithSignatures<TFee = string> extends AbstractUtxoTransactionExplanation<TFee> {\n  /** @deprecated - unused outside of tests */\n  locktime?: number;\n\n  /**\n   * Number of input signatures per input.\n   * @deprecated - this is not very useful without knowing who signed each input.\n   */\n  inputSignatures: number[];\n\n  /**\n   * Highest input signature count for the transaction\n   * @deprecated - this is not very useful without knowing who signed each input.\n   */\n  signatures: number;\n}\n\n/** For our wasm backend, we do not return the deprecated fields. We set TFee to string for backwards compatibility. */\nexport type TransactionExplanationWasm = AbstractUtxoTransactionExplanation<string>;\n\n/** When parsing the legacy transaction format, we cannot always infer the fee so we set it to string | undefined */\nexport type TransactionExplanationUtxolibLegacy = TransactionExplanationWithSignatures<string | undefined>;\n\n/** When parsing a PSBT, we can infer the fee so we set TFee to string. */\nexport type TransactionExplanationUtxolibPsbt = TransactionExplanationWithSignatures<string>;\n\nexport type TransactionExplanation =\n  | TransactionExplanationUtxolibLegacy\n  | TransactionExplanationUtxolibPsbt\n  | TransactionExplanationWasm;\n\nexport type ChangeAddressInfo = {\n  address: string;\n  chain: number;\n  index: number;\n};\n\nfunction explainCommon<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: {\n    changeInfo?: ChangeAddressInfo[];\n    feeInfo?: string;\n  },\n  network: utxolib.Network\n) {\n  const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];\n  let spendAmount = BigInt(0);\n  let changeAmount = BigInt(0);\n  const changeOutputs: FixedScriptWalletOutput[] = [];\n  const outputs: Output[] = [];\n\n  const { changeInfo } = params;\n  const changeAddresses = changeInfo?.map((info) => info.address) ?? [];\n\n  tx.outs.forEach((currentOutput) => {\n    // Try to encode the script pubkey with an address. If it fails, try to parse it as an OP_RETURN output with the prefix.\n    // If that fails, then it is an unrecognized scriptPubkey and should fail\n    const currentAddress = toExtendedAddressFormat(currentOutput.script, network);\n    const currentAmount = BigInt(currentOutput.value);\n\n    if (changeAddresses.includes(currentAddress)) {\n      // this is change\n      changeAmount += currentAmount;\n      const change = changeInfo?.find((change) => change.address === currentAddress);\n\n      if (!change) {\n        throw new Error('changeInfo must have change information for all change outputs');\n      }\n      changeOutputs.push({\n        address: currentAddress,\n        amount: currentAmount.toString(),\n        chain: change.chain,\n        index: change.index,\n        external: false,\n      });\n      return;\n    }\n\n    spendAmount += currentAmount;\n    outputs.push({\n      address: currentAddress,\n      amount: currentAmount.toString(),\n      // If changeInfo has a length greater than or equal to zero, it means that the change information\n      // was provided to the function but the output was not identified as change. In this case,\n      // the output is external, and we can set it as so. If changeInfo is undefined, it means we were\n      // given no information about change outputs, so we can't determine anything about the output,\n      // so we leave it undefined.\n      external: changeInfo ? true : undefined,\n    });\n  });\n\n  const outputDetails = {\n    outputAmount: spendAmount.toString(),\n    changeAmount: changeAmount.toString(),\n    outputs,\n    changeOutputs,\n  };\n\n  let fee: string | undefined;\n  let locktime: number | undefined;\n\n  if (params.feeInfo) {\n    displayOrder.push('fee');\n    fee = params.feeInfo;\n  }\n\n  if (Number.isInteger(tx.locktime) && tx.locktime > 0) {\n    displayOrder.push('locktime');\n    locktime = tx.locktime;\n  }\n\n  return { displayOrder, id: tx.getId(), ...outputDetails, fee, locktime };\n}\n\nfunction getRootWalletKeys(params: { pubs?: bitgo.RootWalletKeys | string[] }): bitgo.RootWalletKeys | undefined {\n  if (params.pubs instanceof bitgo.RootWalletKeys) {\n    return params.pubs;\n  }\n  const keys = params.pubs?.map((xpub) => bip32.fromBase58(xpub));\n  return keys && keys.length === 3 ? new bitgo.RootWalletKeys(keys as Triple<BIP32Interface>) : undefined;\n}\n\nfunction getPsbtInputSignaturesCount(\n  psbt: bitgo.UtxoPsbt,\n  params: {\n    pubs?: bitgo.RootWalletKeys | string[];\n  }\n) {\n  const rootWalletKeys = getRootWalletKeys(params);\n  return rootWalletKeys\n    ? bitgo.getSignatureValidationArrayPsbt(psbt, rootWalletKeys).map((sv) => sv[1].filter((v) => v).length)\n    : (Array(psbt.data.inputs.length) as number[]).fill(0);\n}\n\nfunction getTxInputSignaturesCount<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: {\n    txInfo?: { unspents?: bitgo.Unspent<TNumber>[] };\n    pubs?: bitgo.RootWalletKeys | string[];\n  },\n  network: utxolib.Network\n) {\n  const prevOutputs = params.txInfo?.unspents?.map((u) => bitgo.toOutput<TNumber>(u, network));\n  const rootWalletKeys = getRootWalletKeys(params);\n  const { unspents = [] } = params.txInfo ?? {};\n\n  // get the number of signatures per input\n  return tx.ins.map((input, idx): number => {\n    if (unspents.length !== tx.ins.length) {\n      return 0;\n    }\n    if (!prevOutputs) {\n      throw new Error(`invalid state`);\n    }\n    if (!rootWalletKeys) {\n      // no pub keys or incorrect number of pub keys\n      return 0;\n    }\n    try {\n      return bitgo.verifySignatureWithUnspent<TNumber>(tx, idx, unspents, rootWalletKeys).filter((v) => v).length;\n    } catch (e) {\n      // some other error occurred and we can't validate the signatures\n      return 0;\n    }\n  });\n}\n\nfunction getChainAndIndexFromBip32Derivations(output: bitgo.PsbtOutput) {\n  const derivations = output.bip32Derivation ?? output.tapBip32Derivation ?? undefined;\n  if (!derivations) {\n    return undefined;\n  }\n  const paths = derivations.map((d) => d.path);\n  if (!paths || paths.length !== 3) {\n    throw new Error('expected 3 paths in bip32Derivation or tapBip32Derivation');\n  }\n  if (!paths.every((p) => paths[0] === p)) {\n    throw new Error('expected all paths to be the same');\n  }\n\n  paths.forEach((path) => {\n    if (paths[0] !== path) {\n      throw new Error(\n        'Unable to get a single chain and index on the output because there are different paths for different keys'\n      );\n    }\n  });\n  return utxolib.bitgo.getChainAndIndexFromPath(paths[0]);\n}\n\nfunction getChangeInfo(psbt: bitgo.UtxoPsbt): ChangeAddressInfo[] | undefined {\n  try {\n    return utxolib.bitgo.findInternalOutputIndices(psbt).map((i) => {\n      const derivationInformation = getChainAndIndexFromBip32Derivations(psbt.data.outputs[i]);\n      if (!derivationInformation) {\n        throw new Error('could not find derivation information on bip32Derivation or tapBip32Derivation');\n      }\n      return {\n        address: utxolib.address.fromOutputScript(psbt.txOutputs[i].script, psbt.network),\n        external: false,\n        ...derivationInformation,\n      };\n    });\n  } catch (e) {\n    if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {\n      return undefined;\n    }\n    throw e;\n  }\n}\n\n/**\n * Extract PayGo address proof information from the PSBT if present\n * @returns Information about the PayGo proof, including the output index and address\n */\nfunction getPayGoVerificationInfo(\n  psbt: bitgo.UtxoPsbt,\n  network: utxolib.Network\n): { outputIndex: number; verificationPubkey: string } | undefined {\n  let outputIndex: number | undefined = undefined;\n  let address: string | undefined = undefined;\n  // Check if this PSBT has any PayGo address proofs\n  if (!utxocore.paygo.psbtOutputIncludesPaygoAddressProof(psbt)) {\n    return undefined;\n  }\n\n  // This pulls the pubkey depending on given network\n  const verificationPubkey = getPayGoVerificationPubkey(network);\n  // find which output index that contains the PayGo proof\n  outputIndex = utxocore.paygo.getPayGoAddressProofOutputIndex(psbt);\n  if (outputIndex === undefined || !verificationPubkey) {\n    return undefined;\n  }\n  const output = psbt.txOutputs[outputIndex];\n  address = utxolib.address.fromOutputScript(output.script, network);\n  if (!address) {\n    throw new Error(`Can not derive address ${address} Pay Go Attestation.`);\n  }\n\n  return { outputIndex, verificationPubkey };\n}\n\n/**\n * Extract the BIP322 messages and addresses from the PSBT inputs and perform\n * verification on the transaction to ensure that it meets the BIP322 requirements.\n * @returns An array of objects containing the message and address for each input,\n *          or undefined if no BIP322 messages are found.\n */\nfunction getBip322MessageInfoAndVerify(psbt: bitgo.UtxoPsbt, network: utxolib.Network): Bip322Message[] | undefined {\n  const bip322Messages: { message: string; address: string }[] = [];\n  for (let i = 0; i < psbt.data.inputs.length; i++) {\n    const message = bip322.getBip322ProofMessageAtIndex(psbt, i);\n    if (message) {\n      const input = psbt.data.inputs[i];\n      if (!input.witnessUtxo) {\n        throw new Error(`Missing witnessUtxo for input index ${i}`);\n      }\n      if (!input.nonWitnessUtxo) {\n        throw new Error(`Missing nonWitnessUtxo for input index ${i}`);\n      }\n      const scriptPubKey = input.witnessUtxo.script;\n\n      // Verify that the toSpend transaction can be recreated in the PSBT and is encoded correctly in the nonWitnessUtxo\n      const toSpend = bip322.buildToSpendTransaction(scriptPubKey, message);\n      const toSpendB64 = toSpend.toBuffer().toString('base64');\n      if (input.nonWitnessUtxo.toString('base64') !== toSpendB64) {\n        throw new Error(`Non-witness UTXO does not match the expected toSpend transaction at input index ${i}`);\n      }\n\n      // Verify that the toSpend transaction ID matches the input's referenced transaction ID\n      if (toSpend.getId() !== utxolib.bitgo.getOutputIdForInput(psbt.txInputs[i]).txid) {\n        throw new Error(`ToSpend transaction ID does not match the input at index ${i}`);\n      }\n\n      // Verify the input specifics\n      if (psbt.txInputs[i].sequence !== 0) {\n        throw new Error(`Unexpected sequence number at input index ${i}: ${psbt.txInputs[i].sequence}. Expected 0.`);\n      }\n      if (psbt.txInputs[i].index !== 0) {\n        throw new Error(`Unexpected input index at position ${i}: ${psbt.txInputs[i].index}. Expected 0.`);\n      }\n\n      bip322Messages.push({\n        message: message.toString('utf8'),\n        address: utxolib.address.fromOutputScript(scriptPubKey, network),\n      });\n    }\n  }\n\n  if (bip322Messages.length > 0) {\n    // If there is a BIP322 message in any input, all inputs must have one.\n    if (bip322Messages.length !== psbt.data.inputs.length) {\n      throw new Error('Inconsistent BIP322 messages across inputs.');\n    }\n\n    // Verify the transaction specifics for BIP322\n    if (psbt.version !== 0 && psbt.version !== 2) {\n      throw new Error(`Unsupported PSBT version for BIP322: ${psbt.version}. Expected 0 `);\n    }\n    if (\n      psbt.data.outputs.length !== 1 ||\n      psbt.txOutputs[0].script.toString('hex') !== '6a' ||\n      psbt.txOutputs[0].value !== 0n\n    ) {\n      throw new Error(`Invalid PSBT outputs for BIP322. Expected exactly one OP_RETURN output with zero value.`);\n    }\n\n    return bip322Messages;\n  }\n\n  return undefined;\n}\n\n/**\n * Decompose a raw psbt into useful information, such as the total amounts,\n * change amounts, and transaction outputs.\n */\nexport function explainPsbt(\n  psbt: bitgo.UtxoPsbt,\n  params: {\n    pubs?: bitgo.RootWalletKeys | string[];\n  },\n  network: utxolib.Network,\n  { strict = false }: { strict?: boolean } = {}\n): TransactionExplanationUtxolibPsbt {\n  const payGoVerificationInfo = getPayGoVerificationInfo(psbt, network);\n  if (payGoVerificationInfo) {\n    try {\n      utxocore.paygo.verifyPayGoAddressProof(\n        psbt,\n        payGoVerificationInfo.outputIndex,\n        bip32.fromBase58(payGoVerificationInfo.verificationPubkey, utxolib.networks.bitcoin).publicKey\n      );\n    } catch (e) {\n      if (strict) {\n        throw e;\n      }\n      console.error(e);\n    }\n  }\n\n  const messages = getBip322MessageInfoAndVerify(psbt, network);\n  const changeInfo = getChangeInfo(psbt);\n  const tx = psbt.getUnsignedTx();\n  const common = explainCommon(tx, { ...params, changeInfo }, network);\n  const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);\n\n  // Set fee from subtracting inputs from outputs\n  const outputAmount = psbt.txOutputs.reduce((cumulative, curr) => cumulative + BigInt(curr.value), BigInt(0));\n  const inputAmount = psbt.txInputs.reduce((cumulative, txInput, i) => {\n    const data = psbt.data.inputs[i];\n    if (data.witnessUtxo) {\n      return cumulative + BigInt(data.witnessUtxo.value);\n    } else if (data.nonWitnessUtxo) {\n      const tx = bitgo.createTransactionFromBuffer<bigint>(data.nonWitnessUtxo, network, { amountType: 'bigint' });\n      return cumulative + BigInt(tx.outs[txInput.index].value);\n    } else {\n      throw new Error('could not find value on input');\n    }\n  }, BigInt(0));\n\n  return {\n    ...common,\n    fee: (inputAmount - outputAmount).toString(),\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n    messages,\n  };\n}\n\nexport function explainLegacyTx<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: {\n    pubs?: string[];\n    txInfo?: { unspents?: bitgo.Unspent<TNumber>[] };\n    changeInfo?: { address: string; chain: number; index: number }[];\n  },\n  network: utxolib.Network\n): TransactionExplanationUtxolibLegacy {\n  const common = explainCommon(tx, params, network);\n  const inputSignaturesCount = getTxInputSignaturesCount(tx, params, network);\n  return {\n    ...common,\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n  };\n}\n"]}
|
|
344
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"explainTransaction.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/explainTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import * as utxolib from '@bitgo-beta/utxo-lib';\nimport { bip322 } from '@bitgo-beta/utxo-core';\nimport { BIP32Interface, bip32 } from '@bitgo-beta/secp256k1';\nimport { bitgo } from '@bitgo-beta/utxo-lib';\nimport { ITransactionExplanation as BaseTransactionExplanation, Triple } from '@bitgo-beta/sdk-core';\nimport * as utxocore from '@bitgo-beta/utxo-core';\n\nimport type { Bip322Message } from '../../abstractUtxoCoin';\nimport type { Output, FixedScriptWalletOutput } from '../types';\nimport { toExtendedAddressFormat } from '../recipient';\nimport { getPayGoVerificationPubkey } from '../getPayGoVerificationPubkey';\nimport { toBip32Triple } from '../../keychains';\n\n// ===== Transaction Explanation Type Definitions =====\n\nexport interface AbstractUtxoTransactionExplanation<TFee = string, TChangeOutput extends Output = Output>\n  extends BaseTransactionExplanation<TFee, string> {\n  /** NOTE: this actually only captures external outputs */\n  outputs: Output[];\n  changeOutputs: TChangeOutput[];\n  customChangeOutputs?: TChangeOutput[];\n  customChangeAmount?: string;\n\n  /**\n   * BIP322 messages extracted from the transaction inputs.\n   * These messages are used for verifying the transaction against the BIP322 standard.\n   */\n  messages?: Bip322Message[];\n}\n\n/** @deprecated - the signature fields are not very useful */\ninterface TransactionExplanationWithSignatures<TFee = string, TChangeOutput extends Output = Output>\n  extends AbstractUtxoTransactionExplanation<TFee, TChangeOutput> {\n  /** @deprecated - unused outside of tests */\n  locktime?: number;\n\n  /**\n   * Number of input signatures per input.\n   * @deprecated - this is not very useful without knowing who signed each input.\n   */\n  inputSignatures: number[];\n\n  /**\n   * Highest input signature count for the transaction\n   * @deprecated - this is not very useful without knowing who signed each input.\n   */\n  signatures: number;\n}\n\n/** For our wasm backend, we do not return the deprecated fields. We set TFee to string for backwards compatibility. */\nexport type TransactionExplanationWasm = AbstractUtxoTransactionExplanation<string, FixedScriptWalletOutput>;\n\n/** When parsing the legacy transaction format, we cannot always infer the fee so we set it to string | undefined */\nexport type TransactionExplanationUtxolibLegacy = TransactionExplanationWithSignatures<string | undefined>;\n\n/** When parsing a PSBT, we can infer the fee so we set TFee to string. */\nexport type TransactionExplanationUtxolibPsbt = TransactionExplanationWithSignatures<string>;\n\nexport type TransactionExplanationDescriptor = TransactionExplanationWithSignatures<string, Output>;\n\nexport type TransactionExplanation =\n  | TransactionExplanationUtxolibLegacy\n  | TransactionExplanationUtxolibPsbt\n  | TransactionExplanationWasm;\n\nexport type ChangeAddressInfo = {\n  address: string;\n  chain: number;\n  index: number;\n};\n\nfunction toChangeOutput(\n  txOutput: utxolib.TxOutput<number | bigint>,\n  network: utxolib.Network,\n  changeInfo: ChangeAddressInfo[] | undefined\n): FixedScriptWalletOutput | undefined {\n  if (!changeInfo) {\n    return undefined;\n  }\n  const address = toExtendedAddressFormat(txOutput.script, network);\n  const change = changeInfo.find((change) => change.address === address);\n  if (!change) {\n    return undefined;\n  }\n  return {\n    address,\n    amount: txOutput.value.toString(),\n    chain: change.chain,\n    index: change.index,\n    external: false,\n  };\n}\n\nfunction outputSum(outputs: { amount: string | number }[]): bigint {\n  return outputs.reduce((sum, output) => sum + BigInt(output.amount), BigInt(0));\n}\n\nfunction explainCommon<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: {\n    changeInfo?: ChangeAddressInfo[];\n    customChangeInfo?: ChangeAddressInfo[];\n    feeInfo?: string;\n  },\n  network: utxolib.Network\n) {\n  const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];\n  const changeOutputs: FixedScriptWalletOutput[] = [];\n  const customChangeOutputs: FixedScriptWalletOutput[] = [];\n  const externalOutputs: Output[] = [];\n\n  const { changeInfo, customChangeInfo } = params;\n\n  tx.outs.forEach((currentOutput) => {\n    // Try to encode the script pubkey with an address. If it fails, try to parse it as an OP_RETURN output with the prefix.\n    // If that fails, then it is an unrecognized scriptPubkey and should fail\n    const currentAddress = toExtendedAddressFormat(currentOutput.script, network);\n    const currentAmount = BigInt(currentOutput.value);\n\n    const changeOutput = toChangeOutput(currentOutput, network, changeInfo);\n    if (changeOutput) {\n      changeOutputs.push(changeOutput);\n      return;\n    }\n\n    const customChangeOutput = toChangeOutput(currentOutput, network, customChangeInfo);\n    if (customChangeOutput) {\n      customChangeOutputs.push(customChangeOutput);\n      return;\n    }\n\n    externalOutputs.push({\n      address: currentAddress,\n      amount: currentAmount.toString(),\n      // If changeInfo has a length greater than or equal to zero, it means that the change information\n      // was provided to the function but the output was not identified as change. In this case,\n      // the output is external, and we can set it as so. If changeInfo is undefined, it means we were\n      // given no information about change outputs, so we can't determine anything about the output,\n      // so we leave it undefined.\n      external: changeInfo ? true : undefined,\n    });\n  });\n\n  const outputDetails = {\n    outputs: externalOutputs,\n    outputAmount: outputSum(externalOutputs).toString(),\n\n    changeOutputs,\n    changeAmount: outputSum(changeOutputs).toString(),\n\n    customChangeAmount: outputSum(customChangeOutputs).toString(),\n    customChangeOutputs,\n  };\n\n  let fee: string | undefined;\n  let locktime: number | undefined;\n\n  if (params.feeInfo) {\n    displayOrder.push('fee');\n    fee = params.feeInfo;\n  }\n\n  if (Number.isInteger(tx.locktime) && tx.locktime > 0) {\n    displayOrder.push('locktime');\n    locktime = tx.locktime;\n  }\n\n  return { displayOrder, id: tx.getId(), ...outputDetails, fee, locktime };\n}\n\nfunction getRootWalletKeys(params: { pubs?: bitgo.RootWalletKeys | string[] }): bitgo.RootWalletKeys | undefined {\n  if (params.pubs instanceof bitgo.RootWalletKeys) {\n    return params.pubs;\n  }\n  const keys = params.pubs?.map((xpub) => bip32.fromBase58(xpub));\n  return keys && keys.length === 3 ? new bitgo.RootWalletKeys(keys as Triple<BIP32Interface>) : undefined;\n}\n\nfunction getPsbtInputSignaturesCount(\n  psbt: bitgo.UtxoPsbt,\n  params: {\n    pubs?: bitgo.RootWalletKeys | string[];\n  }\n) {\n  const rootWalletKeys = getRootWalletKeys(params);\n  return rootWalletKeys\n    ? bitgo.getSignatureValidationArrayPsbt(psbt, rootWalletKeys).map((sv) => sv[1].filter((v) => v).length)\n    : (Array(psbt.data.inputs.length) as number[]).fill(0);\n}\n\nfunction getTxInputSignaturesCount<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: {\n    txInfo?: { unspents?: bitgo.Unspent<TNumber>[] };\n    pubs?: bitgo.RootWalletKeys | string[];\n  },\n  network: utxolib.Network\n) {\n  const prevOutputs = params.txInfo?.unspents?.map((u) => bitgo.toOutput<TNumber>(u, network));\n  const rootWalletKeys = getRootWalletKeys(params);\n  const { unspents = [] } = params.txInfo ?? {};\n\n  // get the number of signatures per input\n  return tx.ins.map((input, idx): number => {\n    if (unspents.length !== tx.ins.length) {\n      return 0;\n    }\n    if (!prevOutputs) {\n      throw new Error(`invalid state`);\n    }\n    if (!rootWalletKeys) {\n      // no pub keys or incorrect number of pub keys\n      return 0;\n    }\n    try {\n      return bitgo.verifySignatureWithUnspent<TNumber>(tx, idx, unspents, rootWalletKeys).filter((v) => v).length;\n    } catch (e) {\n      // some other error occurred and we can't validate the signatures\n      return 0;\n    }\n  });\n}\n\nfunction getChainAndIndexFromBip32Derivations(output: bitgo.PsbtOutput) {\n  const derivations = output.bip32Derivation ?? output.tapBip32Derivation ?? undefined;\n  if (!derivations) {\n    return undefined;\n  }\n  const paths = derivations.map((d) => d.path);\n  if (!paths || paths.length !== 3) {\n    throw new Error('expected 3 paths in bip32Derivation or tapBip32Derivation');\n  }\n  if (!paths.every((p) => paths[0] === p)) {\n    throw new Error('expected all paths to be the same');\n  }\n\n  paths.forEach((path) => {\n    if (paths[0] !== path) {\n      throw new Error(\n        'Unable to get a single chain and index on the output because there are different paths for different keys'\n      );\n    }\n  });\n  return utxolib.bitgo.getChainAndIndexFromPath(paths[0]);\n}\n\nfunction getChangeInfo(psbt: bitgo.UtxoPsbt, walletKeys?: Triple<BIP32Interface>): ChangeAddressInfo[] | undefined {\n  try {\n    walletKeys = walletKeys ?? utxolib.bitgo.getSortedRootNodes(psbt);\n  } catch (e) {\n    if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {\n      return undefined;\n    }\n    throw e;\n  }\n\n  return utxolib.bitgo.findWalletOutputIndices(psbt, walletKeys).map((i) => {\n    const derivationInformation = getChainAndIndexFromBip32Derivations(psbt.data.outputs[i]);\n    if (!derivationInformation) {\n      throw new Error('could not find derivation information on bip32Derivation or tapBip32Derivation');\n    }\n    return {\n      address: utxolib.address.fromOutputScript(psbt.txOutputs[i].script, psbt.network),\n      external: false,\n      ...derivationInformation,\n    };\n  });\n}\n\n/**\n * Extract PayGo address proof information from the PSBT if present\n * @returns Information about the PayGo proof, including the output index and address\n */\nfunction getPayGoVerificationInfo(\n  psbt: bitgo.UtxoPsbt,\n  network: utxolib.Network\n): { outputIndex: number; verificationPubkey: string } | undefined {\n  let outputIndex: number | undefined = undefined;\n  let address: string | undefined = undefined;\n  // Check if this PSBT has any PayGo address proofs\n  if (!utxocore.paygo.psbtOutputIncludesPaygoAddressProof(psbt)) {\n    return undefined;\n  }\n\n  // This pulls the pubkey depending on given network\n  const verificationPubkey = getPayGoVerificationPubkey(network);\n  // find which output index that contains the PayGo proof\n  outputIndex = utxocore.paygo.getPayGoAddressProofOutputIndex(psbt);\n  if (outputIndex === undefined || !verificationPubkey) {\n    return undefined;\n  }\n  const output = psbt.txOutputs[outputIndex];\n  address = utxolib.address.fromOutputScript(output.script, network);\n  if (!address) {\n    throw new Error(`Can not derive address ${address} Pay Go Attestation.`);\n  }\n\n  return { outputIndex, verificationPubkey };\n}\n\n/**\n * Extract the BIP322 messages and addresses from the PSBT inputs and perform\n * verification on the transaction to ensure that it meets the BIP322 requirements.\n * @returns An array of objects containing the message and address for each input,\n *          or undefined if no BIP322 messages are found.\n */\nfunction getBip322MessageInfoAndVerify(psbt: bitgo.UtxoPsbt, network: utxolib.Network): Bip322Message[] | undefined {\n  const bip322Messages: { message: string; address: string }[] = [];\n  for (let i = 0; i < psbt.data.inputs.length; i++) {\n    const message = bip322.getBip322ProofMessageAtIndex(psbt, i);\n    if (message) {\n      const input = psbt.data.inputs[i];\n      if (!input.witnessUtxo) {\n        throw new Error(`Missing witnessUtxo for input index ${i}`);\n      }\n      if (!input.nonWitnessUtxo) {\n        throw new Error(`Missing nonWitnessUtxo for input index ${i}`);\n      }\n      const scriptPubKey = input.witnessUtxo.script;\n\n      // Verify that the toSpend transaction can be recreated in the PSBT and is encoded correctly in the nonWitnessUtxo\n      const toSpend = bip322.buildToSpendTransaction(scriptPubKey, message);\n      const toSpendB64 = toSpend.toBuffer().toString('base64');\n      if (input.nonWitnessUtxo.toString('base64') !== toSpendB64) {\n        throw new Error(`Non-witness UTXO does not match the expected toSpend transaction at input index ${i}`);\n      }\n\n      // Verify that the toSpend transaction ID matches the input's referenced transaction ID\n      if (toSpend.getId() !== utxolib.bitgo.getOutputIdForInput(psbt.txInputs[i]).txid) {\n        throw new Error(`ToSpend transaction ID does not match the input at index ${i}`);\n      }\n\n      // Verify the input specifics\n      if (psbt.txInputs[i].sequence !== 0) {\n        throw new Error(`Unexpected sequence number at input index ${i}: ${psbt.txInputs[i].sequence}. Expected 0.`);\n      }\n      if (psbt.txInputs[i].index !== 0) {\n        throw new Error(`Unexpected input index at position ${i}: ${psbt.txInputs[i].index}. Expected 0.`);\n      }\n\n      bip322Messages.push({\n        message: message.toString('utf8'),\n        address: utxolib.address.fromOutputScript(scriptPubKey, network),\n      });\n    }\n  }\n\n  if (bip322Messages.length > 0) {\n    // If there is a BIP322 message in any input, all inputs must have one.\n    if (bip322Messages.length !== psbt.data.inputs.length) {\n      throw new Error('Inconsistent BIP322 messages across inputs.');\n    }\n\n    // Verify the transaction specifics for BIP322\n    if (psbt.version !== 0 && psbt.version !== 2) {\n      throw new Error(`Unsupported PSBT version for BIP322: ${psbt.version}. Expected 0 `);\n    }\n    if (\n      psbt.data.outputs.length !== 1 ||\n      psbt.txOutputs[0].script.toString('hex') !== '6a' ||\n      psbt.txOutputs[0].value !== 0n\n    ) {\n      throw new Error(`Invalid PSBT outputs for BIP322. Expected exactly one OP_RETURN output with zero value.`);\n    }\n\n    return bip322Messages;\n  }\n\n  return undefined;\n}\n\n/**\n * Decompose a raw psbt into useful information, such as the total amounts,\n * change amounts, and transaction outputs.\n *\n * @param psbt {bitgo.UtxoPsbt} The PSBT to explain\n * @param pubs {bitgo.RootWalletKeys | string[]} The public keys to use for the explanation\n * @param network {utxolib.Network} The network to use for the explanation\n * @param strict {boolean} Whether to throw an error if the PayGo address proof is invalid\n */\nexport function explainPsbt(\n  psbt: bitgo.UtxoPsbt,\n  params: {\n    pubs?: bitgo.RootWalletKeys | string[];\n    customChangePubs?: bitgo.RootWalletKeys | string[];\n  },\n  network: utxolib.Network,\n  { strict = false }: { strict?: boolean } = {}\n): TransactionExplanationUtxolibPsbt {\n  const payGoVerificationInfo = getPayGoVerificationInfo(psbt, network);\n  if (payGoVerificationInfo) {\n    try {\n      utxocore.paygo.verifyPayGoAddressProof(\n        psbt,\n        payGoVerificationInfo.outputIndex,\n        bip32.fromBase58(payGoVerificationInfo.verificationPubkey, utxolib.networks.bitcoin).publicKey\n      );\n    } catch (e) {\n      if (strict) {\n        throw e;\n      }\n      console.error(e);\n    }\n  }\n\n  const messages = getBip322MessageInfoAndVerify(psbt, network);\n  const changeInfo = getChangeInfo(psbt);\n  const customChangeInfo = params.customChangePubs\n    ? getChangeInfo(psbt, toBip32Triple(params.customChangePubs))\n    : undefined;\n  const tx = psbt.getUnsignedTx();\n  const common = explainCommon(tx, { ...params, changeInfo, customChangeInfo }, network);\n  const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);\n\n  // Set fee from subtracting inputs from outputs\n  const outputAmount = psbt.txOutputs.reduce((cumulative, curr) => cumulative + BigInt(curr.value), BigInt(0));\n  const inputAmount = psbt.txInputs.reduce((cumulative, txInput, i) => {\n    const data = psbt.data.inputs[i];\n    if (data.witnessUtxo) {\n      return cumulative + BigInt(data.witnessUtxo.value);\n    } else if (data.nonWitnessUtxo) {\n      const tx = bitgo.createTransactionFromBuffer<bigint>(data.nonWitnessUtxo, network, { amountType: 'bigint' });\n      return cumulative + BigInt(tx.outs[txInput.index].value);\n    } else {\n      throw new Error('could not find value on input');\n    }\n  }, BigInt(0));\n\n  return {\n    ...common,\n    fee: (inputAmount - outputAmount).toString(),\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n    messages,\n  };\n}\n\nexport function explainLegacyTx<TNumber extends number | bigint>(\n  tx: bitgo.UtxoTransaction<TNumber>,\n  params: {\n    pubs?: string[];\n    txInfo?: { unspents?: bitgo.Unspent<TNumber>[] };\n    changeInfo?: { address: string; chain: number; index: number }[];\n  },\n  network: utxolib.Network\n): TransactionExplanationUtxolibLegacy {\n  const common = explainCommon(tx, params, network);\n  const inputSignaturesCount = getTxInputSignaturesCount(tx, params, network);\n  return {\n    ...common,\n    inputSignatures: inputSignaturesCount,\n    signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),\n  };\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"parseOutput.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseOutput.ts"],"names":[],"mappings":"AAEA,OAAO,EAEL,cAAc,EAEd,OAAO,EACP,mBAAmB,EAEnB,mBAAmB,EACnB,qBAAqB,EACrB,MAAM,EACP,MAAM,sBAAsB,CAAC;AAE9B,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,OAAO,EAAE,MAAM,EAAE,uBAAuB,EAAE,MAAM,UAAU,CAAC;AAI3D,wBAAgB,cAAc,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,IAAI,uBAAuB,CAIhF;AAwKD,MAAM,WAAW,mBAAmB;IAClC,IAAI,EAAE,MAAM,CAAC;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAC9B,UAAU,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;CAC5B;AAED,MAAM,WAAW,kBAAkB;IACjC,aAAa,EAAE,MAAM,CAAC;IACtB,IAAI,EAAE,gBAAgB,CAAC;IACvB,UAAU,EAAE,mBAAmB,CAAC;IAChC,YAAY,EAAE,mBAAmB,CAAC;IAClC,aAAa,EAAE,MAAM,CAAC;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IACvC,MAAM,EAAE,OAAO,CAAC;IAChB,QAAQ,EAAE;QACR,UAAU,EAAE,qBAAqB,EAAE,CAAC;QACpC,aAAa,CAAC,EAAE,MAAM,CAAC;KACxB,CAAC;IACF,YAAY,CAAC,EAAE,mBAAmB,CAAC;IACnC,KAAK,CAAC,EAAE,cAAc,CAAC;CACxB;AAED,wBAAsB,WAAW,CAAC,EAChC,aAAa,EACb,IAAI,EACJ,UAAU,EACV,YAAY,EACZ,aAAa,EACb,MAAM,EACN,QAAQ,EACR,YAAY,EACZ,KAAK,GACN,EAAE,kBAAkB,GAAG,OAAO,CAAC,MAAM,CAAC,
|
|
1
|
+
{"version":3,"file":"parseOutput.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseOutput.ts"],"names":[],"mappings":"AAEA,OAAO,EAEL,cAAc,EAEd,OAAO,EACP,mBAAmB,EAEnB,mBAAmB,EACnB,qBAAqB,EACrB,MAAM,EACP,MAAM,sBAAsB,CAAC;AAE9B,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;AAC1D,OAAO,EAAE,MAAM,EAAE,uBAAuB,EAAE,MAAM,UAAU,CAAC;AAI3D,wBAAgB,cAAc,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM,IAAI,uBAAuB,CAIhF;AAwKD,MAAM,WAAW,mBAAmB;IAClC,IAAI,EAAE,MAAM,CAAC;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IAC9B,UAAU,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;CAC5B;AAED,MAAM,WAAW,kBAAkB;IACjC,aAAa,EAAE,MAAM,CAAC;IACtB,IAAI,EAAE,gBAAgB,CAAC;IACvB,UAAU,EAAE,mBAAmB,CAAC;IAChC,YAAY,EAAE,mBAAmB,CAAC;IAClC,aAAa,EAAE,MAAM,CAAC;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;IACvC,MAAM,EAAE,OAAO,CAAC;IAChB,QAAQ,EAAE;QACR,UAAU,EAAE,qBAAqB,EAAE,CAAC;QACpC,aAAa,CAAC,EAAE,MAAM,CAAC;KACxB,CAAC;IACF,YAAY,CAAC,EAAE,mBAAmB,CAAC;IACnC,KAAK,CAAC,EAAE,cAAc,CAAC;CACxB;AAED,wBAAsB,WAAW,CAAC,EAChC,aAAa,EACb,IAAI,EACJ,UAAU,EACV,YAAY,EACZ,aAAa,EACb,MAAM,EACN,QAAQ,EACR,YAAY,EACZ,KAAK,GACN,EAAE,kBAAkB,GAAG,OAAO,CAAC,MAAM,CAAC,CAkHtC"}
|
|
@@ -162,11 +162,14 @@ async function parseOutput({ currentOutput, coin, txPrebuild, verification, keyc
|
|
|
162
162
|
return currentOutput;
|
|
163
163
|
}
|
|
164
164
|
/**
|
|
165
|
-
*
|
|
166
|
-
*
|
|
167
|
-
*
|
|
168
|
-
*
|
|
169
|
-
*
|
|
165
|
+
* For transaction with the legacy transaction format, the only way to
|
|
166
|
+
* determine whether an address is known on the wallet is to initiate a
|
|
167
|
+
* network request and fetch it. Should the request fail and return a 404,
|
|
168
|
+
* it will throw and therefore has to be caught. For that reason, address
|
|
169
|
+
* wallet ownership detection is wrapped in a try/catch. Additionally, once
|
|
170
|
+
* the address details are fetched on the wallet, a local address validation
|
|
171
|
+
* is run, whose errors however are generated client-side and can therefore
|
|
172
|
+
* be analyzed with more granularity and type checking.
|
|
170
173
|
*/
|
|
171
174
|
/**
|
|
172
175
|
* In order to minimize API requests, we assume that explicit recipients are always external when the
|
|
@@ -214,4 +217,4 @@ async function parseOutput({ currentOutput, coin, txPrebuild, verification, keyc
|
|
|
214
217
|
}));
|
|
215
218
|
}
|
|
216
219
|
}
|
|
217
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseOutput.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseOutput.ts"],"names":[],"mappings":"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","sourcesContent":["import debugLib from 'debug';\nimport _ from 'lodash';\nimport {\n  AddressVerificationData,\n  IRequestTracer,\n  InvalidAddressDerivationPropertyError,\n  IWallet,\n  TransactionPrebuild,\n  UnexpectedAddressError,\n  VerificationOptions,\n  ITransactionRecipient,\n  Triple,\n} from '@bitgo-beta/sdk-core';\n\nimport { AbstractUtxoCoin } from '../../abstractUtxoCoin';\nimport { Output, FixedScriptWalletOutput } from '../types';\n\nconst debug = debugLib('bitgo:v2:parseoutput');\n\nexport function isWalletOutput(output: Output): output is FixedScriptWalletOutput {\n  return (\n    (output as FixedScriptWalletOutput).chain !== undefined && (output as FixedScriptWalletOutput).index !== undefined\n  );\n}\n\ninterface HandleVerifyAddressErrorResponse {\n  external: boolean;\n  needsCustomChangeKeySignatureVerification?: boolean;\n}\n\n/**\n * Check an address which failed initial validation to see if it's the base address of a migrated v1 bch wallet.\n *\n * The wallet in question could be a migrated SafeHD BCH wallet, and the transaction we\n * are currently parsing is trying to spend change back to the v1 wallet base address.\n *\n * It does this since we don't allow new address creation for these wallets,\n * and instead return the base address from the v1 wallet when a new address is requested.\n * If this new address is requested for the purposes of spending change back to the wallet,\n * the change will go to the v1 wallet base address. This address *is* on the wallet,\n * but it will still cause an error to be thrown by verifyAddress, since the derivation path\n * used for this address is non-standard. (I have seen these addresses derived using paths m/0/0 and m/101,\n * whereas the v2 addresses are derived using path  m/0/0/${chain}/${index}).\n *\n * This means we need to check for this case explicitly in this catch block, and classify\n * these types of outputs as internal instead of external. Failing to do so would cause the\n * transaction's implicit external outputs (ie, outputs which go to addresses not specified in\n * the recipients array) to add up to more than the 150 basis point limit which we enforce on\n * pay-as-you-go outputs (which should be the only implicit external outputs on our transactions).\n *\n * The 150 basis point limit for implicit external sends is enforced in verifyTransaction,\n * which calls this function to get information on the total external/internal spend amounts\n * for a transaction. The idea here is to protect from the transaction being maliciously modified\n * to add more implicit external spends (eg, to an attacker-controlled wallet).\n *\n * See verifyTransaction for more information on how transaction prebuilds are verified before signing.\n *\n * @param wallet {Wallet} wallet which is making the transaction\n * @param currentAddress {string} address to check for externality relative to v1 wallet base address\n */\nfunction isMigratedAddress(wallet: IWallet, currentAddress: string): boolean {\n  if (_.isString(wallet.migratedFrom()) && wallet.migratedFrom() === currentAddress) {\n    debug('found address %s which was migrated from v1 wallet, address is not external', currentAddress);\n    return true;\n  }\n\n  return false;\n}\n\ninterface VerifyCustomChangeAddressOptions {\n  coin: AbstractUtxoCoin;\n  customChangeKeys: HandleVerifyAddressErrorOptions['customChangeKeys'];\n  addressType: HandleVerifyAddressErrorOptions['addressType'];\n  addressDetails: HandleVerifyAddressErrorOptions['addressDetails'];\n  currentAddress: HandleVerifyAddressErrorOptions['currentAddress'];\n}\n\n/**\n * Check to see if an address is derived from the given custom change keys\n * @param {VerifyCustomChangeAddressOptions} params\n * @return {boolean}\n */\nasync function verifyCustomChangeAddress(params: VerifyCustomChangeAddressOptions): Promise<boolean> {\n  const { coin, customChangeKeys, addressType, addressDetails, currentAddress } = params;\n  try {\n    return await coin.verifyAddress(\n      _.extend({ addressType }, addressDetails, {\n        keychains: customChangeKeys,\n        address: currentAddress,\n      })\n    );\n  } catch (e) {\n    debug('failed to verify custom change address %s', currentAddress);\n    return false;\n  }\n}\n\ninterface HandleVerifyAddressErrorOptions {\n  e: Error;\n  currentAddress: string;\n  wallet: IWallet;\n  txParams: {\n    changeAddress?: string;\n  };\n  customChangeKeys?: CustomChangeOptions['keys'];\n  coin: AbstractUtxoCoin;\n  addressDetails?: any;\n  addressType?: string;\n  considerMigratedFromAddressInternal?: boolean;\n}\n\nasync function handleVerifyAddressError({\n  e,\n  currentAddress,\n  wallet,\n  txParams,\n  customChangeKeys,\n  coin,\n  addressDetails,\n  addressType,\n  considerMigratedFromAddressInternal,\n}: HandleVerifyAddressErrorOptions): Promise<HandleVerifyAddressErrorResponse> {\n  // Todo: name server-side errors to avoid message-based checking [BG-5124]\n  const walletAddressNotFound = e.message.includes('wallet address not found');\n  const unexpectedAddress = e instanceof UnexpectedAddressError;\n  if (walletAddressNotFound || unexpectedAddress) {\n    if (unexpectedAddress && !walletAddressNotFound) {\n      // check to see if this is a migrated v1 bch address - it could be internal\n      const isMigrated = isMigratedAddress(wallet, currentAddress);\n      if (isMigrated) {\n        return { external: considerMigratedFromAddressInternal === false };\n      }\n\n      debug('Address %s was found on wallet but could not be reconstructed', currentAddress);\n\n      // attempt to verify address using custom change address keys if the wallet has that feature enabled\n      if (\n        customChangeKeys &&\n        (await verifyCustomChangeAddress({ coin, addressDetails, addressType, currentAddress, customChangeKeys }))\n      ) {\n        // address is valid against the custom change keys. Mark address as not external\n        // and request signature verification for the custom change keys\n        debug('Address %s verified as derived from the custom change keys', currentAddress);\n        return { external: false, needsCustomChangeKeySignatureVerification: true };\n      }\n    }\n\n    // the address was found, but not on the wallet, which simply means it's external\n    debug('Address %s presumed external', currentAddress);\n    return { external: true };\n  } else if (e instanceof InvalidAddressDerivationPropertyError && currentAddress === txParams.changeAddress) {\n    // expect to see this error when passing in a custom changeAddress with no chain or index\n    return { external: false };\n  }\n\n  console.error('Address classification failed for address', currentAddress);\n  console.trace(e);\n  /**\n   * It might be a completely invalid address or a bad validation attempt or something else completely, in\n   * which case we do not proceed and rather rethrow the error, which is safer than assuming that the address\n   * validation failed simply because it's external to the wallet.\n   */\n  throw e;\n}\n\ninterface FetchAddressDetailsOptions {\n  reqId?: IRequestTracer;\n  disableNetworking: boolean;\n  addressDetailsPrebuild: any;\n  addressDetailsVerification: any;\n  currentAddress: string;\n  wallet: IWallet;\n}\n\nasync function fetchAddressDetails({\n  reqId,\n  disableNetworking,\n  addressDetailsPrebuild,\n  addressDetailsVerification,\n  currentAddress,\n  wallet,\n}: FetchAddressDetailsOptions) {\n  let addressDetails = _.extend({}, addressDetailsPrebuild, addressDetailsVerification);\n  debug('Locally available address %s details: %O', currentAddress, addressDetails);\n  if (_.isEmpty(addressDetails) && !disableNetworking) {\n    addressDetails = await wallet.getAddress({ address: currentAddress, reqId });\n    debug('Downloaded address %s details: %O', currentAddress, addressDetails);\n  }\n  return addressDetails;\n}\n\nexport interface CustomChangeOptions {\n  keys: Triple<{ pub: string }>;\n  signatures: Triple<string>;\n}\n\nexport interface ParseOutputOptions {\n  currentOutput: Output;\n  coin: AbstractUtxoCoin;\n  txPrebuild: TransactionPrebuild;\n  verification: VerificationOptions;\n  keychainArray: Triple<{ pub: string }>;\n  wallet: IWallet;\n  txParams: {\n    recipients: ITransactionRecipient[];\n    changeAddress?: string;\n  };\n  customChange?: CustomChangeOptions;\n  reqId?: IRequestTracer;\n}\n\nexport async function parseOutput({\n  currentOutput,\n  coin,\n  txPrebuild,\n  verification,\n  keychainArray,\n  wallet,\n  txParams,\n  customChange,\n  reqId,\n}: ParseOutputOptions): Promise<Output> {\n  const disableNetworking = !!verification.disableNetworking;\n  const currentAddress = currentOutput.address;\n\n  if (currentAddress === undefined) {\n    // In the case that the address is undefined, it means that the output has a non-encodeable scriptPubkey\n    // If this is the case, then we need to check that the amount is 0 and we can skip the rest.\n    if (currentOutput.amount.toString() !== '0') {\n      throw new Error('output with undefined address must have amount of 0');\n    }\n    return currentOutput;\n  }\n\n  // attempt to grab the address details from either the prebuilt tx, or the verification params.\n  // If both of these are empty, then we will try to get the address details from bitgo instead\n  const addressDetailsPrebuild = _.get(txPrebuild, `txInfo.walletAddressDetails.${currentAddress}`, {});\n  const addressDetailsVerification: AddressVerificationData = verification?.addresses?.[currentAddress] ?? {};\n  debug('Parsing address details for %s', currentAddress);\n  let currentAddressDetails = undefined;\n  let currentAddressType: string | undefined = undefined;\n  const RECIPIENT_THRESHOLD = 1000;\n  try {\n    // In the case of PSBTs, we can already determine the internal/external status of the output addresses\n    // based on the derivation information being included in the PSBT. We can short circuit GET v2.wallet.address\n    // and save on network requests. Since we have the derivation information already, we can still verify the address\n    if (currentOutput.external !== undefined) {\n      // In the case that we have a custom change wallet, we need to verify the address against the custom change keys\n      // and not the wallet keys. This check is done in the handleVerifyAddressError function if this error is thrown.\n      if (customChange !== undefined) {\n        throw new UnexpectedAddressError('`address validation failure');\n      }\n      // If it is an internal address, we can skip the network request and just verify the address locally with the\n      // derivation information we have. Otherwise, if the address is external, which is the only remaining case, we\n      // can just return the current output as is without contacting the server.\n      if (isWalletOutput(currentOutput)) {\n        const res = await coin.isWalletAddress({\n          addressType: AbstractUtxoCoin.inferAddressType({ chain: currentOutput.chain }) || undefined,\n          keychains: keychainArray,\n          address: currentAddress,\n          chain: currentOutput.chain,\n          index: currentOutput.index,\n        });\n        if (!res) {\n          throw new UnexpectedAddressError();\n        }\n      }\n      return currentOutput;\n    }\n    /**\n     * The only way to determine whether an address is known on the wallet is to initiate a network request and\n     * fetch it. Should the request fail and return a 404, it will throw and therefore has to be caught. For that\n     * reason, address wallet ownership detection is wrapped in a try/catch. Additionally, once the address\n     * details are fetched on the wallet, a local address validation is run, whose errors however are generated\n     * client-side and can therefore be analyzed with more granularity and type checking.\n     */\n\n    /**\n     * In order to minimize API requests, we assume that explicit recipients are always external when the\n     * recipient list is > 1000 This is not always a valid assumption and could lead greater apparent spend (but never lower)\n     */\n    if (txParams.recipients !== undefined && txParams.recipients.length > RECIPIENT_THRESHOLD) {\n      const isCurrentAddressInRecipients = txParams.recipients.some((recipient) =>\n        recipient.address.includes(currentAddress)\n      );\n\n      if (isCurrentAddressInRecipients) {\n        return { ...currentOutput };\n      }\n    }\n\n    const addressDetails = await fetchAddressDetails({\n      reqId,\n      addressDetailsVerification,\n      addressDetailsPrebuild,\n      currentAddress,\n      disableNetworking,\n      wallet,\n    });\n    // verify that the address is on the wallet. verifyAddress throws if\n    // it fails to correctly rederive the address, meaning it's external\n    currentAddressType = AbstractUtxoCoin.inferAddressType(addressDetails) || undefined;\n    currentAddressDetails = addressDetails;\n    await coin.verifyAddress(\n      _.extend({ addressType: currentAddressType }, addressDetails, {\n        keychains: keychainArray,\n        address: currentAddress,\n      })\n    );\n    debug('Address %s verification passed', currentAddress);\n\n    // verify address succeeded without throwing, so the address was\n    // correctly rederived from the wallet keychains, making it not external\n    return _.extend({}, currentOutput, addressDetails, { external: false });\n  } catch (e) {\n    debug('Address %s verification threw an error:', currentAddress, e);\n    return _.extend(\n      {},\n      currentOutput,\n      await handleVerifyAddressError({\n        e,\n        coin,\n        currentAddress,\n        wallet,\n        txParams,\n        customChangeKeys: customChange && customChange.keys,\n        addressDetails: currentAddressDetails,\n        addressType: currentAddressType,\n        considerMigratedFromAddressInternal: verification.considerMigratedFromAddressInternal,\n      })\n    );\n  }\n}\n"]}
|
|
220
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseOutput.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseOutput.ts"],"names":[],"mappings":"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","sourcesContent":["import debugLib from 'debug';\nimport _ from 'lodash';\nimport {\n  AddressVerificationData,\n  IRequestTracer,\n  InvalidAddressDerivationPropertyError,\n  IWallet,\n  TransactionPrebuild,\n  UnexpectedAddressError,\n  VerificationOptions,\n  ITransactionRecipient,\n  Triple,\n} from '@bitgo-beta/sdk-core';\n\nimport { AbstractUtxoCoin } from '../../abstractUtxoCoin';\nimport { Output, FixedScriptWalletOutput } from '../types';\n\nconst debug = debugLib('bitgo:v2:parseoutput');\n\nexport function isWalletOutput(output: Output): output is FixedScriptWalletOutput {\n  return (\n    (output as FixedScriptWalletOutput).chain !== undefined && (output as FixedScriptWalletOutput).index !== undefined\n  );\n}\n\ninterface HandleVerifyAddressErrorResponse {\n  external: boolean;\n  needsCustomChangeKeySignatureVerification?: boolean;\n}\n\n/**\n * Check an address which failed initial validation to see if it's the base address of a migrated v1 bch wallet.\n *\n * The wallet in question could be a migrated SafeHD BCH wallet, and the transaction we\n * are currently parsing is trying to spend change back to the v1 wallet base address.\n *\n * It does this since we don't allow new address creation for these wallets,\n * and instead return the base address from the v1 wallet when a new address is requested.\n * If this new address is requested for the purposes of spending change back to the wallet,\n * the change will go to the v1 wallet base address. This address *is* on the wallet,\n * but it will still cause an error to be thrown by verifyAddress, since the derivation path\n * used for this address is non-standard. (I have seen these addresses derived using paths m/0/0 and m/101,\n * whereas the v2 addresses are derived using path  m/0/0/${chain}/${index}).\n *\n * This means we need to check for this case explicitly in this catch block, and classify\n * these types of outputs as internal instead of external. Failing to do so would cause the\n * transaction's implicit external outputs (ie, outputs which go to addresses not specified in\n * the recipients array) to add up to more than the 150 basis point limit which we enforce on\n * pay-as-you-go outputs (which should be the only implicit external outputs on our transactions).\n *\n * The 150 basis point limit for implicit external sends is enforced in verifyTransaction,\n * which calls this function to get information on the total external/internal spend amounts\n * for a transaction. The idea here is to protect from the transaction being maliciously modified\n * to add more implicit external spends (eg, to an attacker-controlled wallet).\n *\n * See verifyTransaction for more information on how transaction prebuilds are verified before signing.\n *\n * @param wallet {Wallet} wallet which is making the transaction\n * @param currentAddress {string} address to check for externality relative to v1 wallet base address\n */\nfunction isMigratedAddress(wallet: IWallet, currentAddress: string): boolean {\n  if (_.isString(wallet.migratedFrom()) && wallet.migratedFrom() === currentAddress) {\n    debug('found address %s which was migrated from v1 wallet, address is not external', currentAddress);\n    return true;\n  }\n\n  return false;\n}\n\ninterface VerifyCustomChangeAddressOptions {\n  coin: AbstractUtxoCoin;\n  customChangeKeys: HandleVerifyAddressErrorOptions['customChangeKeys'];\n  addressType: HandleVerifyAddressErrorOptions['addressType'];\n  addressDetails: HandleVerifyAddressErrorOptions['addressDetails'];\n  currentAddress: HandleVerifyAddressErrorOptions['currentAddress'];\n}\n\n/**\n * Check to see if an address is derived from the given custom change keys\n * @param {VerifyCustomChangeAddressOptions} params\n * @return {boolean}\n */\nasync function verifyCustomChangeAddress(params: VerifyCustomChangeAddressOptions): Promise<boolean> {\n  const { coin, customChangeKeys, addressType, addressDetails, currentAddress } = params;\n  try {\n    return await coin.verifyAddress(\n      _.extend({ addressType }, addressDetails, {\n        keychains: customChangeKeys,\n        address: currentAddress,\n      })\n    );\n  } catch (e) {\n    debug('failed to verify custom change address %s', currentAddress);\n    return false;\n  }\n}\n\ninterface HandleVerifyAddressErrorOptions {\n  e: Error;\n  currentAddress: string;\n  wallet: IWallet;\n  txParams: {\n    changeAddress?: string;\n  };\n  customChangeKeys?: CustomChangeOptions['keys'];\n  coin: AbstractUtxoCoin;\n  addressDetails?: any;\n  addressType?: string;\n  considerMigratedFromAddressInternal?: boolean;\n}\n\nasync function handleVerifyAddressError({\n  e,\n  currentAddress,\n  wallet,\n  txParams,\n  customChangeKeys,\n  coin,\n  addressDetails,\n  addressType,\n  considerMigratedFromAddressInternal,\n}: HandleVerifyAddressErrorOptions): Promise<HandleVerifyAddressErrorResponse> {\n  // Todo: name server-side errors to avoid message-based checking [BG-5124]\n  const walletAddressNotFound = e.message.includes('wallet address not found');\n  const unexpectedAddress = e instanceof UnexpectedAddressError;\n  if (walletAddressNotFound || unexpectedAddress) {\n    if (unexpectedAddress && !walletAddressNotFound) {\n      // check to see if this is a migrated v1 bch address - it could be internal\n      const isMigrated = isMigratedAddress(wallet, currentAddress);\n      if (isMigrated) {\n        return { external: considerMigratedFromAddressInternal === false };\n      }\n\n      debug('Address %s was found on wallet but could not be reconstructed', currentAddress);\n\n      // attempt to verify address using custom change address keys if the wallet has that feature enabled\n      if (\n        customChangeKeys &&\n        (await verifyCustomChangeAddress({ coin, addressDetails, addressType, currentAddress, customChangeKeys }))\n      ) {\n        // address is valid against the custom change keys. Mark address as not external\n        // and request signature verification for the custom change keys\n        debug('Address %s verified as derived from the custom change keys', currentAddress);\n        return { external: false, needsCustomChangeKeySignatureVerification: true };\n      }\n    }\n\n    // the address was found, but not on the wallet, which simply means it's external\n    debug('Address %s presumed external', currentAddress);\n    return { external: true };\n  } else if (e instanceof InvalidAddressDerivationPropertyError && currentAddress === txParams.changeAddress) {\n    // expect to see this error when passing in a custom changeAddress with no chain or index\n    return { external: false };\n  }\n\n  console.error('Address classification failed for address', currentAddress);\n  console.trace(e);\n  /**\n   * It might be a completely invalid address or a bad validation attempt or something else completely, in\n   * which case we do not proceed and rather rethrow the error, which is safer than assuming that the address\n   * validation failed simply because it's external to the wallet.\n   */\n  throw e;\n}\n\ninterface FetchAddressDetailsOptions {\n  reqId?: IRequestTracer;\n  disableNetworking: boolean;\n  addressDetailsPrebuild: any;\n  addressDetailsVerification: any;\n  currentAddress: string;\n  wallet: IWallet;\n}\n\nasync function fetchAddressDetails({\n  reqId,\n  disableNetworking,\n  addressDetailsPrebuild,\n  addressDetailsVerification,\n  currentAddress,\n  wallet,\n}: FetchAddressDetailsOptions) {\n  let addressDetails = _.extend({}, addressDetailsPrebuild, addressDetailsVerification);\n  debug('Locally available address %s details: %O', currentAddress, addressDetails);\n  if (_.isEmpty(addressDetails) && !disableNetworking) {\n    addressDetails = await wallet.getAddress({ address: currentAddress, reqId });\n    debug('Downloaded address %s details: %O', currentAddress, addressDetails);\n  }\n  return addressDetails;\n}\n\nexport interface CustomChangeOptions {\n  keys: Triple<{ pub: string }>;\n  signatures: Triple<string>;\n}\n\nexport interface ParseOutputOptions {\n  currentOutput: Output;\n  coin: AbstractUtxoCoin;\n  txPrebuild: TransactionPrebuild;\n  verification: VerificationOptions;\n  keychainArray: Triple<{ pub: string }>;\n  wallet: IWallet;\n  txParams: {\n    recipients: ITransactionRecipient[];\n    changeAddress?: string;\n  };\n  customChange?: CustomChangeOptions;\n  reqId?: IRequestTracer;\n}\n\nexport async function parseOutput({\n  currentOutput,\n  coin,\n  txPrebuild,\n  verification,\n  keychainArray,\n  wallet,\n  txParams,\n  customChange,\n  reqId,\n}: ParseOutputOptions): Promise<Output> {\n  const disableNetworking = !!verification.disableNetworking;\n  const currentAddress = currentOutput.address;\n\n  if (currentAddress === undefined) {\n    // In the case that the address is undefined, it means that the output has a non-encodeable scriptPubkey\n    // If this is the case, then we need to check that the amount is 0 and we can skip the rest.\n    if (currentOutput.amount.toString() !== '0') {\n      throw new Error('output with undefined address must have amount of 0');\n    }\n    return currentOutput;\n  }\n\n  // attempt to grab the address details from either the prebuilt tx, or the verification params.\n  // If both of these are empty, then we will try to get the address details from bitgo instead\n  const addressDetailsPrebuild = _.get(txPrebuild, `txInfo.walletAddressDetails.${currentAddress}`, {});\n  const addressDetailsVerification: AddressVerificationData = verification?.addresses?.[currentAddress] ?? {};\n  debug('Parsing address details for %s', currentAddress);\n  let currentAddressDetails = undefined;\n  let currentAddressType: string | undefined = undefined;\n  const RECIPIENT_THRESHOLD = 1000;\n  try {\n    // In the case of PSBTs, we can already determine the internal/external status of the output addresses\n    // based on the derivation information being included in the PSBT. We can short circuit GET v2.wallet.address\n    // and save on network requests. Since we have the derivation information already, we can still verify the address\n    if (currentOutput.external !== undefined) {\n      // In the case that we have a custom change wallet, we need to verify the address against the custom change keys\n      // and not the wallet keys. This check is done in the handleVerifyAddressError function if this error is thrown.\n      if (customChange !== undefined) {\n        throw new UnexpectedAddressError('`address validation failure');\n      }\n      // If it is an internal address, we can skip the network request and just verify the address locally with the\n      // derivation information we have. Otherwise, if the address is external, which is the only remaining case, we\n      // can just return the current output as is without contacting the server.\n      if (isWalletOutput(currentOutput)) {\n        const res = await coin.isWalletAddress({\n          addressType: AbstractUtxoCoin.inferAddressType({ chain: currentOutput.chain }) || undefined,\n          keychains: keychainArray,\n          address: currentAddress,\n          chain: currentOutput.chain,\n          index: currentOutput.index,\n        });\n        if (!res) {\n          throw new UnexpectedAddressError();\n        }\n      }\n      return currentOutput;\n    }\n    /**\n     * For transaction with the legacy transaction format, the only way to\n     * determine whether an address is known on the wallet is to initiate a\n     * network request and fetch it. Should the request fail and return a 404,\n     * it will throw and therefore has to be caught. For that reason, address\n     * wallet ownership detection is wrapped in a try/catch. Additionally, once\n     * the address details are fetched on the wallet, a local address validation\n     * is run, whose errors however are generated client-side and can therefore\n     * be analyzed with more granularity and type checking.\n     */\n\n    /**\n     * In order to minimize API requests, we assume that explicit recipients are always external when the\n     * recipient list is > 1000 This is not always a valid assumption and could lead greater apparent spend (but never lower)\n     */\n    if (txParams.recipients !== undefined && txParams.recipients.length > RECIPIENT_THRESHOLD) {\n      const isCurrentAddressInRecipients = txParams.recipients.some((recipient) =>\n        recipient.address.includes(currentAddress)\n      );\n\n      if (isCurrentAddressInRecipients) {\n        return { ...currentOutput };\n      }\n    }\n\n    const addressDetails = await fetchAddressDetails({\n      reqId,\n      addressDetailsVerification,\n      addressDetailsPrebuild,\n      currentAddress,\n      disableNetworking,\n      wallet,\n    });\n    // verify that the address is on the wallet. verifyAddress throws if\n    // it fails to correctly rederive the address, meaning it's external\n    currentAddressType = AbstractUtxoCoin.inferAddressType(addressDetails) || undefined;\n    currentAddressDetails = addressDetails;\n    await coin.verifyAddress(\n      _.extend({ addressType: currentAddressType }, addressDetails, {\n        keychains: keychainArray,\n        address: currentAddress,\n      })\n    );\n    debug('Address %s verification passed', currentAddress);\n\n    // verify address succeeded without throwing, so the address was\n    // correctly rederived from the wallet keychains, making it not external\n    return _.extend({}, currentOutput, addressDetails, { external: false });\n  } catch (e) {\n    debug('Address %s verification threw an error:', currentAddress, e);\n    return _.extend(\n      {},\n      currentOutput,\n      await handleVerifyAddressError({\n        e,\n        coin,\n        currentAddress,\n        wallet,\n        txParams,\n        customChangeKeys: customChange && customChange.keys,\n        addressDetails: currentAddressDetails,\n        addressType: currentAddressType,\n        considerMigratedFromAddressInternal: verification.considerMigratedFromAddressInternal,\n      })\n    );\n  }\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"parseTransaction.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseTransaction.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,gBAAgB,EAAE,uBAAuB,EAAE,MAAM,wBAAwB,CAAC;AACxF,OAAO,KAAK,EAAmC,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAEnF,OAAO,EAAE,gBAAgB,EAAoB,MAAM,qBAAqB,CAAC;AAMzE,MAAM,MAAM,4BAA4B,CAAC,MAAM,IAAI,gBAAgB,CAAC,MAAM,CAAC,GAAG;IAC5E,QAAQ,EAAE,OAAO,GAAG,SAAS,CAAC;CAC/B,CAAC;
|
|
1
|
+
{"version":3,"file":"parseTransaction.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/parseTransaction.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,gBAAgB,EAAE,uBAAuB,EAAE,MAAM,wBAAwB,CAAC;AACxF,OAAO,KAAK,EAAmC,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAEnF,OAAO,EAAE,gBAAgB,EAAoB,MAAM,qBAAqB,CAAC;AAMzE,MAAM,MAAM,4BAA4B,CAAC,MAAM,IAAI,gBAAgB,CAAC,MAAM,CAAC,GAAG;IAC5E,QAAQ,EAAE,OAAO,GAAG,SAAS,CAAC;CAC/B,CAAC;AA2DF,wBAAsB,gBAAgB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACpE,IAAI,EAAE,gBAAgB,EACtB,MAAM,EAAE,uBAAuB,CAAC,OAAO,CAAC,GACvC,OAAO,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC,CAuKrC"}
|