PHP WebShell

Текущая директория: /opt/BitGoJS/modules/abstract-utxo/dist/src/transaction/fixedScript

Просмотр файла: explainTransaction.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;
    };
})();
Object.defineProperty(exports, "__esModule", { value: true });
exports.explainPsbt = explainPsbt;
exports.explainLegacyTx = explainLegacyTx;
const utxolib = __importStar(require("@bitgo/utxo-lib"));
const utxo_lib_1 = require("@bitgo/utxo-lib");
const recipient_1 = require("../recipient");
function explainCommon(tx, params, network) {
    const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs'];
    let spendAmount = BigInt(0);
    let changeAmount = BigInt(0);
    const changeOutputs = [];
    const outputs = [];
    const { changeInfo } = params;
    const changeAddresses = changeInfo?.map((info) => info.address) ?? [];
    tx.outs.forEach((currentOutput) => {
        // Try to encode the script pubkey with an address. If it fails, try to parse it as an OP_RETURN output with the prefix.
        // If that fails, then it is an unrecognized scriptPubkey and should fail
        const currentAddress = (0, recipient_1.toExtendedAddressFormat)(currentOutput.script, network);
        const currentAmount = BigInt(currentOutput.value);
        if (changeAddresses.includes(currentAddress)) {
            // this is change
            changeAmount += currentAmount;
            const change = changeInfo?.find((change) => change.address === currentAddress);
            if (!change) {
                throw new Error('changeInfo must have change information for all change outputs');
            }
            changeOutputs.push({
                address: currentAddress,
                amount: currentAmount.toString(),
                chain: change.chain,
                index: change.index,
                external: false,
            });
            return;
        }
        spendAmount += currentAmount;
        outputs.push({
            address: currentAddress,
            amount: currentAmount.toString(),
            // If changeInfo has a length greater than or equal to zero, it means that the change information
            // was provided to the function but the output was not identified as change. In this case,
            // the output is external, and we can set it as so. If changeInfo is undefined, it means we were
            // given no information about change outputs, so we can't determine anything about the output,
            // so we leave it undefined.
            external: changeInfo ? true : undefined,
        });
    });
    const outputDetails = {
        outputAmount: spendAmount.toString(),
        changeAmount: changeAmount.toString(),
        outputs,
        changeOutputs,
    };
    let fee;
    let locktime;
    if (params.feeInfo) {
        displayOrder.push('fee');
        fee = params.feeInfo;
    }
    if (Number.isInteger(tx.locktime) && tx.locktime > 0) {
        displayOrder.push('locktime');
        locktime = tx.locktime;
    }
    return { displayOrder, id: tx.getId(), ...outputDetails, fee, locktime };
}
function getRootWalletKeys(params) {
    const keys = params.pubs?.map((xpub) => utxo_lib_1.bip32.fromBase58(xpub));
    return keys && keys.length === 3 ? new utxo_lib_1.bitgo.RootWalletKeys(keys) : undefined;
}
function getPsbtInputSignaturesCount(psbt, params) {
    const rootWalletKeys = getRootWalletKeys(params);
    return rootWalletKeys
        ? utxo_lib_1.bitgo.getSignatureValidationArrayPsbt(psbt, rootWalletKeys).map((sv) => sv[1].filter((v) => v).length)
        : Array(psbt.data.inputs.length).fill(0);
}
function getTxInputSignaturesCount(tx, params, network) {
    const prevOutputs = params.txInfo?.unspents?.map((u) => utxo_lib_1.bitgo.toOutput(u, network));
    const rootWalletKeys = getRootWalletKeys(params);
    const { unspents = [] } = params.txInfo ?? {};
    // get the number of signatures per input
    return tx.ins.map((input, idx) => {
        if (unspents.length !== tx.ins.length) {
            return 0;
        }
        if (!prevOutputs) {
            throw new Error(`invalid state`);
        }
        if (!rootWalletKeys) {
            // no pub keys or incorrect number of pub keys
            return 0;
        }
        try {
            return utxo_lib_1.bitgo.verifySignatureWithUnspent(tx, idx, unspents, rootWalletKeys).filter((v) => v).length;
        }
        catch (e) {
            // some other error occurred and we can't validate the signatures
            return 0;
        }
    });
}
/**
 * Decompose a raw psbt into useful information, such as the total amounts,
 * change amounts, and transaction outputs.
 */
function explainPsbt(psbt, params, network) {
    const txOutputs = psbt.txOutputs;
    function getChainAndIndexFromBip32Derivations(output) {
        const derivations = output.bip32Derivation ?? output.tapBip32Derivation ?? undefined;
        if (!derivations) {
            return undefined;
        }
        const paths = derivations.map((d) => d.path);
        if (!paths || paths.length !== 3) {
            throw new Error('expected 3 paths in bip32Derivation or tapBip32Derivation');
        }
        if (!paths.every((p) => paths[0] === p)) {
            throw new Error('expected all paths to be the same');
        }
        paths.forEach((path) => {
            if (paths[0] !== path) {
                throw new Error('Unable to get a single chain and index on the output because there are different paths for different keys');
            }
        });
        return utxolib.bitgo.getChainAndIndexFromPath(paths[0]);
    }
    function getChangeInfo() {
        try {
            return utxolib.bitgo.findInternalOutputIndices(psbt).map((i) => {
                const derivationInformation = getChainAndIndexFromBip32Derivations(psbt.data.outputs[i]);
                if (!derivationInformation) {
                    throw new Error('could not find derivation information on bip32Derivation or tapBip32Derivation');
                }
                return {
                    address: utxolib.address.fromOutputScript(txOutputs[i].script, network),
                    external: false,
                    ...derivationInformation,
                };
            });
        }
        catch (e) {
            if (e instanceof utxolib.bitgo.ErrorNoMultiSigInputFound) {
                return undefined;
            }
            throw e;
        }
    }
    const changeInfo = getChangeInfo();
    const tx = psbt.getUnsignedTx();
    const common = explainCommon(tx, { ...params, changeInfo }, network);
    const inputSignaturesCount = getPsbtInputSignaturesCount(psbt, params);
    // Set fee from subtracting inputs from outputs
    const outputAmount = txOutputs.reduce((cumulative, curr) => cumulative + BigInt(curr.value), BigInt(0));
    const inputAmount = psbt.txInputs.reduce((cumulative, txInput, i) => {
        const data = psbt.data.inputs[i];
        if (data.witnessUtxo) {
            return cumulative + BigInt(data.witnessUtxo.value);
        }
        else if (data.nonWitnessUtxo) {
            const tx = utxo_lib_1.bitgo.createTransactionFromBuffer(data.nonWitnessUtxo, network, { amountType: 'bigint' });
            return cumulative + BigInt(tx.outs[txInput.index].value);
        }
        else {
            throw new Error('could not find value on input');
        }
    }, BigInt(0));
    return {
        ...common,
        fee: (inputAmount - outputAmount).toString(),
        inputSignatures: inputSignaturesCount,
        signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),
    };
}
function explainLegacyTx(tx, params, network) {
    const common = explainCommon(tx, params, network);
    const inputSignaturesCount = getTxInputSignaturesCount(tx, params, network);
    return {
        ...common,
        inputSignatures: inputSignaturesCount,
        signatures: inputSignaturesCount.reduce((prev, curr) => (curr > prev ? curr : prev), 0),
    };
}
//# 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/utxo-lib';\nimport { bip32, BIP32Interface, bitgo } from '@bitgo/utxo-lib';\nimport { Triple } from '@bitgo/sdk-core';\n\nimport { Output, TransactionExplanation, FixedScriptWalletOutput } from '../../abstractUtxoCoin';\nimport { toExtendedAddressFormat } from '../recipient';\n\nexport type ChangeAddressInfo = { address: string; chain: number; index: number };\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?: string[] }) {\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?: 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?: 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\n/**\n * Decompose a raw psbt into useful information, such as the total amounts,\n * change amounts, and transaction outputs.\n */\nexport function explainPsbt<TNumber extends number | bigint, Tx extends bitgo.UtxoTransaction<bigint>>(\n  psbt: bitgo.UtxoPsbt<Tx>,\n  params: {\n    pubs?: string[];\n    txInfo?: { unspents?: bitgo.Unspent<TNumber>[] };\n  },\n  network: utxolib.Network\n): TransactionExplanation {\n  const txOutputs = psbt.txOutputs;\n\n  function 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\n  function getChangeInfo() {\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(txOutputs[i].script, 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  const changeInfo = getChangeInfo();\n  const tx = psbt.getUnsignedTx() as bitgo.UtxoTransaction<TNumber>;\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 = 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  } as TransactionExplanation;\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): TransactionExplanation {\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  } as TransactionExplanation;\n}\n"]}

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


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