PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/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_core_1 = require("@bitgo/utxo-core");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const utxocore = __importStar(require("@bitgo/utxo-core"));
const recipient_1 = require("../recipient");
const getPayGoVerificationPubkey_1 = require("../getPayGoVerificationPubkey");
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, { strict = false } = {}) {
    const txOutputs = psbt.txOutputs;
    const txInputs = psbt.txInputs;
    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;
        }
    }
    /**
     * Extract PayGo address proof information from the PSBT if present
     * @returns Information about the PayGo proof, including the output index and address
     */
    function getPayGoVerificationInfo() {
        let outputIndex = undefined;
        let address = undefined;
        // Check if this PSBT has any PayGo address proofs
        if (!utxocore.paygo.psbtOutputIncludesPaygoAddressProof(psbt)) {
            return undefined;
        }
        // This pulls the pubkey depending on given network
        const verificationPubkey = (0, getPayGoVerificationPubkey_1.getPayGoVerificationPubkey)(network);
        // find which output index that contains the PayGo proof
        outputIndex = utxocore.paygo.getPayGoAddressProofOutputIndex(psbt);
        if (outputIndex === undefined || !verificationPubkey) {
            return undefined;
        }
        const output = txOutputs[outputIndex];
        address = utxolib.address.fromOutputScript(output.script, network);
        if (!address) {
            throw new Error(`Can not derive address ${address} Pay Go Attestation.`);
        }
        return { outputIndex, verificationPubkey };
    }
    /**
     * Extract the BIP322 messages and addresses from the PSBT inputs and perform
     * verification on the transaction to ensure that it meets the BIP322 requirements.
     * @returns An array of objects containing the message and address for each input,
     *          or undefined if no BIP322 messages are found.
     */
    function getBip322MessageInfoAndVerify() {
        const bip322Messages = [];
        for (let i = 0; i < psbt.data.inputs.length; i++) {
            const message = utxo_core_1.bip322.getBip322ProofMessageAtIndex(psbt, i);
            if (message) {
                const input = psbt.data.inputs[i];
                if (!input.witnessUtxo) {
                    throw new Error(`Missing witnessUtxo for input index ${i}`);
                }
                if (!input.nonWitnessUtxo) {
                    throw new Error(`Missing nonWitnessUtxo for input index ${i}`);
                }
                const scriptPubKey = input.witnessUtxo.script;
                // Verify that the toSpend transaction can be recreated in the PSBT and is encoded correctly in the nonWitnessUtxo
                const toSpend = utxo_core_1.bip322.buildToSpendTransaction(scriptPubKey, message);
                const toSpendB64 = toSpend.toBuffer().toString('base64');
                if (input.nonWitnessUtxo.toString('base64') !== toSpendB64) {
                    throw new Error(`Non-witness UTXO does not match the expected toSpend transaction at input index ${i}`);
                }
                // Verify that the toSpend transaction ID matches the input's referenced transaction ID
                if (toSpend.getId() !== utxolib.bitgo.getOutputIdForInput(txInputs[i]).txid) {
                    throw new Error(`ToSpend transaction ID does not match the input at index ${i}`);
                }
                // Verify the input specifics
                if (txInputs[i].sequence !== 0) {
                    throw new Error(`Unexpected sequence number at input index ${i}: ${txInputs[i].sequence}. Expected 0.`);
                }
                if (txInputs[i].index !== 0) {
                    throw new Error(`Unexpected input index at position ${i}: ${txInputs[i].index}. Expected 0.`);
                }
                bip322Messages.push({
                    message: message.toString('utf8'),
                    address: utxolib.address.fromOutputScript(scriptPubKey, network),
                });
            }
        }
        if (bip322Messages.length > 0) {
            // If there is a BIP322 message in any input, all inputs must have one.
            if (bip322Messages.length !== psbt.data.inputs.length) {
                throw new Error('Inconsistent BIP322 messages across inputs.');
            }
            // Verify the transaction specifics for BIP322
            if (psbt.version !== 0 && psbt.version !== 2) {
                throw new Error(`Unsupported PSBT version for BIP322: ${psbt.version}. Expected 0 `);
            }
            if (psbt.data.outputs.length !== 1 || txOutputs[0].script.toString('hex') !== '6a' || txOutputs[0].value !== 0n) {
                throw new Error(`Invalid PSBT outputs for BIP322. Expected exactly one OP_RETURN output with zero value.`);
            }
            return bip322Messages;
        }
        return undefined;
    }
    const payGoVerificationInfo = getPayGoVerificationInfo();
    if (payGoVerificationInfo) {
        try {
            utxocore.paygo.verifyPayGoAddressProof(psbt, payGoVerificationInfo.outputIndex, utxolib.bip32.fromBase58(payGoVerificationInfo.verificationPubkey, utxolib.networks.bitcoin).publicKey);
        }
        catch (e) {
            if (strict) {
                throw e;
            }
            console.error(e);
        }
    }
    const messages = getBip322MessageInfoAndVerify();
    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),
        messages,
    };
}
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 { bip322 } from '@bitgo/utxo-core';\nimport { bip32, BIP32Interface, bitgo } from '@bitgo/utxo-lib';\nimport { Triple } from '@bitgo/sdk-core';\nimport * as utxocore from '@bitgo/utxo-core';\n\nimport { Output, TransactionExplanation, Bip322Message, FixedScriptWalletOutput } from '../../abstractUtxoCoin';\nimport { toExtendedAddressFormat } from '../recipient';\nimport { getPayGoVerificationPubkey } from '../getPayGoVerificationPubkey';\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?: 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  { strict = false }: { strict?: boolean } = {}\n): TransactionExplanation {\n  const txOutputs = psbt.txOutputs;\n  const txInputs = psbt.txInputs;\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\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   */\n  function getPayGoVerificationInfo(): { 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 = 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   */\n  function getBip322MessageInfoAndVerify(): 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(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 (txInputs[i].sequence !== 0) {\n          throw new Error(`Unexpected sequence number at input index ${i}: ${txInputs[i].sequence}. Expected 0.`);\n        }\n        if (txInputs[i].index !== 0) {\n          throw new Error(`Unexpected input index at position ${i}: ${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 (psbt.data.outputs.length !== 1 || txOutputs[0].script.toString('hex') !== '6a' || txOutputs[0].value !== 0n) {\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  const payGoVerificationInfo = getPayGoVerificationInfo();\n  if (payGoVerificationInfo) {\n    try {\n      utxocore.paygo.verifyPayGoAddressProof(\n        psbt,\n        payGoVerificationInfo.outputIndex,\n        utxolib.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();\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    messages,\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"]}

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


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