PHP WebShell
Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/abstract-utxo/dist/src/transaction/fixedScript
Просмотр файла: parseTransaction.js
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.parseTransaction = parseTransaction;
const assert_1 = __importDefault(require("assert"));
const lodash_1 = __importDefault(require("lodash"));
const utxolib = __importStar(require("@bitgo/utxo-lib"));
const keychains_1 = require("../../keychains");
const outputDifference_1 = require("../outputDifference");
const recipient_1 = require("../recipient");
const parseOutput_1 = require("./parseOutput");
async function parseTransaction(coin, params) {
const { txParams, txPrebuild, wallet, verification = {}, reqId } = params;
if (!lodash_1.default.isUndefined(verification.disableNetworking) && !lodash_1.default.isBoolean(verification.disableNetworking)) {
throw new Error('verification.disableNetworking must be a boolean');
}
const disableNetworking = verification.disableNetworking;
// obtain the keychains and key signatures
let keychains = verification.keychains;
if (!keychains) {
if (disableNetworking) {
throw new Error('cannot fetch keychains without networking');
}
keychains = await (0, keychains_1.fetchKeychains)(coin, wallet, reqId);
}
if (!keychains_1.UtxoNamedKeychains.is(keychains)) {
throw new Error('invalid keychains');
}
const keychainArray = (0, keychains_1.toKeychainTriple)(keychains);
if (lodash_1.default.isUndefined(txPrebuild.txHex)) {
throw new Error('missing required txPrebuild property txHex');
}
// obtain all outputs
const explanation = await coin.explainTransaction({
txHex: txPrebuild.txHex,
txInfo: txPrebuild.txInfo,
pubs: keychainArray.map((k) => k.pub),
});
const allOutputs = [...explanation.outputs, ...explanation.changeOutputs];
let expectedOutputs;
if (txParams.rbfTxIds) {
(0, assert_1.default)(txParams.rbfTxIds.length === 1);
const txToBeReplaced = await wallet.getTransaction({ txHash: txParams.rbfTxIds[0], includeRbf: true });
expectedOutputs = txToBeReplaced.outputs.flatMap((output) => {
// For self-sends, the walletId will be the same as the wallet's id
if (output.wallet === wallet.id()) {
return [];
}
return [coin.toCanonicalTransactionRecipient(output)];
});
}
else {
// verify that each recipient from txParams has their own output
expectedOutputs = (txParams.recipients ?? []).flatMap((output) => {
if (output.address === undefined) {
if (output.amount.toString() !== '0') {
throw new Error(`Only zero amounts allowed for non-encodeable scriptPubkeys: ${output}`);
}
return [output];
}
return [{ ...output, address: coin.canonicalAddress(output.address) }];
});
if (txParams.allowExternalChangeAddress && txParams.changeAddress) {
// when an external change address is explicitly specified, count all outputs going towards that
// address in the expected outputs (regardless of the output amount)
expectedOutputs.push(...allOutputs.flatMap((output) => {
if (output.address === undefined ||
output.address !== coin.canonicalAddress(txParams.changeAddress)) {
return [];
}
return [{ ...output, address: coin.canonicalAddress(output.address) }];
}));
}
}
// get the keychains from the custom change wallet if needed
let customChange;
const { customChangeWalletId = undefined } = wallet.coinSpecific() || {};
if (customChangeWalletId) {
// fetch keychains from custom change wallet for deriving addresses.
// These keychains should be signed and this should be verified in verifyTransaction
const customChangeKeySignatures = wallet._wallet.customChangeKeySignatures;
const customChangeWallet = await coin.wallets().get({ id: customChangeWalletId });
const customChangeKeys = await (0, keychains_1.fetchKeychains)(coin, customChangeWallet, reqId);
if (!customChangeKeys) {
throw new Error('failed to fetch keychains for custom change wallet');
}
if (customChangeKeys.user && customChangeKeys.backup && customChangeKeys.bitgo && customChangeWallet) {
const customChangeKeychains = [
customChangeKeys.user,
customChangeKeys.backup,
customChangeKeys.bitgo,
];
customChange = {
keys: customChangeKeychains,
signatures: [customChangeKeySignatures.user, customChangeKeySignatures.backup, customChangeKeySignatures.bitgo],
};
}
}
/**
* Loop through all the outputs and classify each of them as either internal spends
* or external spends by setting the "external" property to true or false on the output object.
*/
const allOutputDetails = await Promise.all(allOutputs.map((currentOutput) => {
return (0, parseOutput_1.parseOutput)({
currentOutput,
coin,
txPrebuild,
verification,
keychainArray: (0, keychains_1.toKeychainTriple)(keychains),
wallet,
txParams: {
recipients: expectedOutputs,
changeAddress: txParams.changeAddress,
},
customChange,
reqId,
});
}));
const needsCustomChangeKeySignatureVerification = allOutputDetails.some((output) => output?.needsCustomChangeKeySignatureVerification);
const changeOutputs = lodash_1.default.filter(allOutputDetails, { external: false });
function toComparableOutputsWithExternal(outputs) {
return outputs.map((output) => ({
script: (0, recipient_1.fromExtendedAddressFormatToScript)(output.address, coin.network),
value: output.amount === 'max' ? 'max' : BigInt(output.amount),
external: output.external,
}));
}
const missingOutputs = (0, outputDifference_1.outputDifference)(toComparableOutputsWithExternal(expectedOutputs), toComparableOutputsWithExternal(allOutputs));
const implicitOutputs = (0, outputDifference_1.outputDifference)(toComparableOutputsWithExternal(allOutputDetails), toComparableOutputsWithExternal(expectedOutputs));
const explicitOutputs = (0, outputDifference_1.outputDifference)(toComparableOutputsWithExternal(allOutputDetails), implicitOutputs);
// these are all the non-wallet outputs that had been originally explicitly specified in recipients
const explicitExternalOutputs = explicitOutputs.filter((output) => output.external);
// this is the sum of all the originally explicitly specified non-wallet output values
const explicitExternalSpendAmount = utxolib.bitgo.toTNumber(explicitExternalOutputs.reduce((sum, o) => sum + BigInt(o.value), BigInt(0)), coin.amountType);
/**
* The calculation of the implicit external spend amount pertains to verifying the pay-as-you-go-fee BitGo
* automatically applies to transactions sending money out of the wallet. The logic is fairly straightforward
* in that we compare the external spend amount that was specified explicitly by the user to the portion
* that was specified implicitly. To protect customers from people tampering with the transaction outputs, we
* define a threshold for the maximum percentage of the implicit external spend in relation to the explicit
* external spend.
*/
// make sure that all the extra addresses are change addresses
// get all the additional external outputs the server added and calculate their values
const implicitExternalOutputs = implicitOutputs.filter((output) => output.external);
const implicitExternalSpendAmount = utxolib.bitgo.toTNumber(implicitExternalOutputs.reduce((sum, o) => sum + BigInt(o.value), BigInt(0)), coin.amountType);
function toOutputs(outputs) {
return outputs.map((output) => ({
address: (0, recipient_1.toExtendedAddressFormat)(output.script, coin.network),
amount: output.value.toString(),
external: output.external,
}));
}
return {
keychains,
keySignatures: (0, keychains_1.getKeySignatures)(wallet) ?? {},
outputs: allOutputDetails,
missingOutputs: toOutputs(missingOutputs),
explicitExternalOutputs: toOutputs(explicitExternalOutputs),
implicitExternalOutputs: toOutputs(implicitExternalOutputs),
changeOutputs,
explicitExternalSpendAmount,
implicitExternalSpendAmount,
needsCustomChangeKeySignatureVerification,
customChange,
};
}
//# 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/sdk-core';\nimport * as utxolib from '@bitgo/utxo-lib';\n\nimport {\n  AbstractUtxoCoin,\n  FixedScriptWalletOutput,\n  Output,\n  TransactionExplanation,\n  ParsedTransaction,\n  ParseTransactionOptions,\n} from '../../abstractUtxoCoin';\nimport { fetchKeychains, getKeySignatures, toKeychainTriple, UtxoKeychain, UtxoNamedKeychains } from '../../keychains';\nimport { ComparableOutput, outputDifference } from '../outputDifference';\nimport { fromExtendedAddressFormatToScript, toExtendedAddressFormat } from '../recipient';\n\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 applies 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\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"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!