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"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!