PHP WebShell

Текущая директория: /opt/BitGoJS/modules/utxo-lib/dist/src/bitgo/wallet

Просмотр файла: WalletOutput.js

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.getPsbtBip32DerivationOutputUpdate = getPsbtBip32DerivationOutputUpdate;
exports.getPsbtOutputUpdateFromPsbtOutput = getPsbtOutputUpdateFromPsbtOutput;
exports.getPsbtOutputUpdate = getPsbtOutputUpdate;
exports.updateWalletOutputForPsbt = updateWalletOutputForPsbt;
exports.addWalletOutputToPsbt = addWalletOutputToPsbt;
exports.getScriptIdFromOutput = getScriptIdFromOutput;
const assert = require("assert");
const bitcoinjs_lib_1 = require("bitcoinjs-lib");
const chains_1 = require("./chains");
const ScriptId_1 = require("./ScriptId");
const outputScripts_1 = require("../outputScripts");
/**
 * Get the BIP32 derivation data for a PSBT output.
 *
 * @param rootWalletKeys root wallet keys used for master fingerprints
 * @param walletKeys derived wallet keys for the specific chain and index
 * @param scriptType the script type to determine whether to use regular or taproot derivation
 * @param payment optional payment object for taproot scripts to calculate leaf hashes
 * @returns Object containing BIP32 derivation data
 */
function getPsbtBip32DerivationOutputUpdate(rootWalletKeys, walletKeys, scriptType, payment) {
    const update = {};
    if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {
        if (!payment || !payment.redeems) {
            throw new Error('Payment object with redeems is required for taproot derivation');
        }
        const allLeafHashes = payment.redeems.map((r) => bitcoinjs_lib_1.taproot.hashTapLeaf(r.output));
        update.tapBip32Derivation = [0, 1, 2].map((idx) => {
            const pubkey = (0, outputScripts_1.toXOnlyPublicKey)(walletKeys.triple[idx].publicKey);
            const leafHashes = [];
            assert(payment.redeems);
            payment.redeems.forEach((r, redeemIdx) => {
                if (r.pubkeys.find((pk) => pk.equals(pubkey))) {
                    leafHashes.push(allLeafHashes[redeemIdx]);
                }
            });
            return {
                leafHashes,
                pubkey,
                path: walletKeys.paths[idx],
                masterFingerprint: rootWalletKeys.triple[idx].fingerprint,
            };
        });
    }
    else {
        update.bip32Derivation = [0, 1, 2].map((idx) => ({
            pubkey: walletKeys.triple[idx].publicKey,
            path: walletKeys.paths[idx],
            masterFingerprint: rootWalletKeys.triple[idx].fingerprint,
        }));
    }
    return update;
}
/**
 * Get the PSBT output update object from a PSBT output and output script.
 *
 * @param output the PSBT output to get update for
 * @param outputScript the output script
 * @param rootWalletKeys keys that will be able to spend the output
 * @param chain chain code to use for deriving scripts (and to determine script type)
 * @param index derivation index for the change address
 * @returns PsbtOutputUpdate object with the required information
 */
function getPsbtOutputUpdateFromPsbtOutput(output, outputScript, rootWalletKeys, chain, index) {
    const walletKeys = rootWalletKeys.deriveForChainAndIndex(chain, index);
    const scriptType = (0, chains_1.scriptTypeForChain)(chain);
    const update = {};
    if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {
        const payment = scriptType === 'p2tr' ? (0, outputScripts_1.createPaymentP2tr)(walletKeys.publicKeys) : (0, outputScripts_1.createPaymentP2trMusig2)(walletKeys.publicKeys);
        if (!payment.output || !payment.output.equals(outputScript)) {
            throw new Error(`cannot update a p2tr output where the scripts do not match - Failing.`);
        }
        if (!output.tapTree) {
            update.tapTree = payment.tapTree;
        }
        if (!output.tapInternalKey) {
            update.tapInternalKey = payment.internalPubkey;
        }
        if (!output.tapBip32Derivation) {
            const derivationUpdate = getPsbtBip32DerivationOutputUpdate(rootWalletKeys, walletKeys, scriptType, payment);
            update.tapBip32Derivation = derivationUpdate.tapBip32Derivation;
        }
    }
    else {
        const { scriptPubKey, witnessScript, redeemScript } = (0, outputScripts_1.createOutputScript2of3)(walletKeys.publicKeys, scriptType);
        if (!scriptPubKey.equals(outputScript)) {
            throw new Error(`cannot update an output where the scripts do not match - Failing.`);
        }
        if (!output.bip32Derivation) {
            const derivationUpdate = getPsbtBip32DerivationOutputUpdate(rootWalletKeys, walletKeys, scriptType);
            update.bip32Derivation = derivationUpdate.bip32Derivation;
        }
        if (!output.witnessScript && witnessScript) {
            update.witnessScript = witnessScript;
        }
        if (!output.redeemScript && redeemScript) {
            update.redeemScript = redeemScript;
        }
    }
    return update;
}
/**
 * Get the PSBT output update object with the required information.
 *
 * @param psbt the PSBT to get output update for
 * @param rootWalletKeys keys that will be able to spend the output
 * @param outputIndex output index where to update the output
 * @param chain chain code to use for deriving scripts (and to determine script
 *              type) chain is an API parameter in the BitGo API, and may be
 *              any valid ChainCode
 * @param index derivation index for the change address
 * @returns PsbtOutputUpdate object with the required information
 */
function getPsbtOutputUpdate(psbt, rootWalletKeys, outputIndex, chain, index) {
    if (psbt.data.outputs.length <= outputIndex) {
        throw new Error(`outputIndex (${outputIndex}) is too large for the number of outputs (${psbt.data.outputs.length})`);
    }
    const outputScript = psbt.getOutputScript(outputIndex);
    const output = psbt.data.outputs[outputIndex];
    return getPsbtOutputUpdateFromPsbtOutput(output, outputScript, rootWalletKeys, chain, index);
}
/**
 * Update the wallet output with the required information when necessary. If the
 * information is there already, it will skip over it.
 *
 * This function assumes that the output script and value have already been set.
 *
 * @param psbt the PSBT to update change output at
 * @param rootWalletKeys keys that will be able to spend the output
 * @param outputIndex output index where to update the output
 * @param chain chain code to use for deriving scripts (and to determine script
 *              type) chain is an API parameter in the BitGo API, and may be
 *              any valid ChainCode
 * @param index derivation index for the change address
 */
function updateWalletOutputForPsbt(psbt, rootWalletKeys, outputIndex, chain, index) {
    psbt.updateOutput(outputIndex, getPsbtOutputUpdate(psbt, rootWalletKeys, outputIndex, chain, index));
}
/**
 * Add a verifiable wallet output to the PSBT. The output and all data
 * needed to verify it from public keys only are added to the PSBT.
 * Typically these are change outputs.
 *
 * @param psbt the PSBT to add change output to
 * @param rootWalletKeys keys that will be able to spend the output
 * @param chain chain code to use for deriving scripts (and to determine script
 *              type) chain is an API parameter in the BitGo API, and may be
 *              any valid ChainCode
 * @param index derivation index for the change address
 * @param value value of the change output
 */
function addWalletOutputToPsbt(psbt, rootWalletKeys, chain, index, value) {
    const walletKeys = rootWalletKeys.deriveForChainAndIndex(chain, index);
    const scriptType = (0, chains_1.scriptTypeForChain)(chain);
    if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {
        const payment = scriptType === 'p2tr' ? (0, outputScripts_1.createPaymentP2tr)(walletKeys.publicKeys) : (0, outputScripts_1.createPaymentP2trMusig2)(walletKeys.publicKeys);
        psbt.addOutput({ script: payment.output, value });
    }
    else {
        const { scriptPubKey: script } = (0, outputScripts_1.createOutputScript2of3)(walletKeys.publicKeys, scriptType);
        psbt.addOutput({ script, value });
    }
    updateWalletOutputForPsbt(psbt, rootWalletKeys, psbt.data.outputs.length - 1, chain, index);
}
/**
 * Fold the script ids into a single script id, if they are all the same.
 * @param scriptIds
 */
function foldScriptIds(scriptIds) {
    if (scriptIds.length === 0) {
        throw new Error('cannot fold empty script ids');
    }
    scriptIds.forEach((scriptId, i) => {
        if (scriptId.chain !== scriptIds[0].chain) {
            throw new Error(`chain mismatch: ${scriptId.chain} != ${scriptIds[0].chain}`);
        }
        if (scriptId.index !== scriptIds[0].index) {
            throw new Error(`index mismatch: ${scriptId.index} != ${scriptIds[0].index}`);
        }
    });
    return scriptIds[0];
}
/**
 * Get the script id from the output.
 * The output can have either bip32Derivation or tapBip32Derivation, but not both.
 * @param output
 * @throws Error if neither or both bip32Derivation and tapBip32Derivation are present
 * @throws Error if the output is empty
 * @throws Error if we cannot fold the script ids into a single script id
 */
function getScriptIdFromOutput(output) {
    if (output.bip32Derivation && output.tapBip32Derivation) {
        throw new Error('cannot get script id from output with both bip32Derivation and tapBip32Derivation');
    }
    if (output.bip32Derivation) {
        return foldScriptIds(output.bip32Derivation.map((d) => (0, ScriptId_1.getScriptIdFromPath)(d.path)));
    }
    if (output.tapBip32Derivation) {
        return foldScriptIds(output.tapBip32Derivation.map((d) => (0, ScriptId_1.getScriptIdFromPath)(d.path)));
    }
    throw new Error('cannot get script id from output without bip32Derivation or tapBip32Derivation');
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"WalletOutput.js","sourceRoot":"","sources":["../../../../src/bitgo/wallet/WalletOutput.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\n\nimport { Payment, taproot } from 'bitcoinjs-lib';\nimport { PsbtOutput, PsbtOutputUpdate } from 'bip174/src/lib/interfaces';\nimport { UtxoPsbt } from '../UtxoPsbt';\nimport { RootWalletKeys, DerivedWalletKeys } from './WalletKeys';\nimport { ChainCode, scriptTypeForChain } from './chains';\nimport { getScriptIdFromPath, ScriptId } from './ScriptId';\nimport { createOutputScript2of3, createPaymentP2tr, createPaymentP2trMusig2, toXOnlyPublicKey } from '../outputScripts';\n\n/**\n * Get the BIP32 derivation data for a PSBT output.\n *\n * @param rootWalletKeys root wallet keys used for master fingerprints\n * @param walletKeys derived wallet keys for the specific chain and index\n * @param scriptType the script type to determine whether to use regular or taproot derivation\n * @param payment optional payment object for taproot scripts to calculate leaf hashes\n * @returns Object containing BIP32 derivation data\n */\nexport function getPsbtBip32DerivationOutputUpdate(\n  rootWalletKeys: RootWalletKeys,\n  walletKeys: DerivedWalletKeys,\n  scriptType: string,\n  payment?: Payment\n): PsbtOutputUpdate {\n  const update: PsbtOutputUpdate = {};\n\n  if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {\n    if (!payment || !payment.redeems) {\n      throw new Error('Payment object with redeems is required for taproot derivation');\n    }\n\n    const allLeafHashes = payment.redeems.map((r) => taproot.hashTapLeaf(r.output!));\n\n    update.tapBip32Derivation = [0, 1, 2].map((idx) => {\n      const pubkey = toXOnlyPublicKey(walletKeys.triple[idx].publicKey);\n      const leafHashes: Buffer[] = [];\n\n      assert(payment.redeems);\n      payment.redeems.forEach((r: any, redeemIdx: number) => {\n        if (r.pubkeys!.find((pk: Buffer) => pk.equals(pubkey))) {\n          leafHashes.push(allLeafHashes[redeemIdx]);\n        }\n      });\n\n      return {\n        leafHashes,\n        pubkey,\n        path: walletKeys.paths[idx],\n        masterFingerprint: rootWalletKeys.triple[idx].fingerprint,\n      };\n    });\n  } else {\n    update.bip32Derivation = [0, 1, 2].map((idx) => ({\n      pubkey: walletKeys.triple[idx].publicKey,\n      path: walletKeys.paths[idx],\n      masterFingerprint: rootWalletKeys.triple[idx].fingerprint,\n    }));\n  }\n\n  return update;\n}\n\n/**\n * Get the PSBT output update object from a PSBT output and output script.\n *\n * @param output the PSBT output to get update for\n * @param outputScript the output script\n * @param rootWalletKeys keys that will be able to spend the output\n * @param chain chain code to use for deriving scripts (and to determine script type)\n * @param index derivation index for the change address\n * @returns PsbtOutputUpdate object with the required information\n */\nexport function getPsbtOutputUpdateFromPsbtOutput(\n  output: PsbtOutput,\n  outputScript: Buffer,\n  rootWalletKeys: RootWalletKeys,\n  chain: ChainCode,\n  index: number\n): PsbtOutputUpdate {\n  const walletKeys = rootWalletKeys.deriveForChainAndIndex(chain, index);\n  const scriptType = scriptTypeForChain(chain);\n  const update: PsbtOutputUpdate = {};\n\n  if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {\n    const payment =\n      scriptType === 'p2tr' ? createPaymentP2tr(walletKeys.publicKeys) : createPaymentP2trMusig2(walletKeys.publicKeys);\n    if (!payment.output || !payment.output.equals(outputScript)) {\n      throw new Error(`cannot update a p2tr output where the scripts do not match - Failing.`);\n    }\n\n    if (!output.tapTree) {\n      update.tapTree = payment.tapTree;\n    }\n    if (!output.tapInternalKey) {\n      update.tapInternalKey = payment.internalPubkey;\n    }\n\n    if (!output.tapBip32Derivation) {\n      const derivationUpdate = getPsbtBip32DerivationOutputUpdate(rootWalletKeys, walletKeys, scriptType, payment);\n      update.tapBip32Derivation = derivationUpdate.tapBip32Derivation;\n    }\n  } else {\n    const { scriptPubKey, witnessScript, redeemScript } = createOutputScript2of3(walletKeys.publicKeys, scriptType);\n    if (!scriptPubKey.equals(outputScript)) {\n      throw new Error(`cannot update an output where the scripts do not match - Failing.`);\n    }\n\n    if (!output.bip32Derivation) {\n      const derivationUpdate = getPsbtBip32DerivationOutputUpdate(rootWalletKeys, walletKeys, scriptType);\n      update.bip32Derivation = derivationUpdate.bip32Derivation;\n    }\n\n    if (!output.witnessScript && witnessScript) {\n      update.witnessScript = witnessScript;\n    }\n    if (!output.redeemScript && redeemScript) {\n      update.redeemScript = redeemScript;\n    }\n  }\n\n  return update;\n}\n\n/**\n * Get the PSBT output update object with the required information.\n *\n * @param psbt the PSBT to get output update for\n * @param rootWalletKeys keys that will be able to spend the output\n * @param outputIndex output index where to update the output\n * @param chain chain code to use for deriving scripts (and to determine script\n *              type) chain is an API parameter in the BitGo API, and may be\n *              any valid ChainCode\n * @param index derivation index for the change address\n * @returns PsbtOutputUpdate object with the required information\n */\nexport function getPsbtOutputUpdate(\n  psbt: UtxoPsbt,\n  rootWalletKeys: RootWalletKeys,\n  outputIndex: number,\n  chain: ChainCode,\n  index: number\n): PsbtOutputUpdate {\n  if (psbt.data.outputs.length <= outputIndex) {\n    throw new Error(\n      `outputIndex (${outputIndex}) is too large for the number of outputs (${psbt.data.outputs.length})`\n    );\n  }\n\n  const outputScript = psbt.getOutputScript(outputIndex);\n  const output = psbt.data.outputs[outputIndex];\n\n  return getPsbtOutputUpdateFromPsbtOutput(output, outputScript, rootWalletKeys, chain, index);\n}\n\n/**\n * Update the wallet output with the required information when necessary. If the\n * information is there already, it will skip over it.\n *\n * This function assumes that the output script and value have already been set.\n *\n * @param psbt the PSBT to update change output at\n * @param rootWalletKeys keys that will be able to spend the output\n * @param outputIndex output index where to update the output\n * @param chain chain code to use for deriving scripts (and to determine script\n *              type) chain is an API parameter in the BitGo API, and may be\n *              any valid ChainCode\n * @param index derivation index for the change address\n */\nexport function updateWalletOutputForPsbt(\n  psbt: UtxoPsbt,\n  rootWalletKeys: RootWalletKeys,\n  outputIndex: number,\n  chain: ChainCode,\n  index: number\n): void {\n  psbt.updateOutput(outputIndex, getPsbtOutputUpdate(psbt, rootWalletKeys, outputIndex, chain, index));\n}\n\n/**\n * Add a verifiable wallet output to the PSBT. The output and all data\n * needed to verify it from public keys only are added to the PSBT.\n * Typically these are change outputs.\n *\n * @param psbt the PSBT to add change output to\n * @param rootWalletKeys keys that will be able to spend the output\n * @param chain chain code to use for deriving scripts (and to determine script\n *              type) chain is an API parameter in the BitGo API, and may be\n *              any valid ChainCode\n * @param index derivation index for the change address\n * @param value value of the change output\n */\nexport function addWalletOutputToPsbt(\n  psbt: UtxoPsbt,\n  rootWalletKeys: RootWalletKeys,\n  chain: ChainCode,\n  index: number,\n  value: bigint\n): void {\n  const walletKeys = rootWalletKeys.deriveForChainAndIndex(chain, index);\n  const scriptType = scriptTypeForChain(chain);\n  if (scriptType === 'p2tr' || scriptType === 'p2trMusig2') {\n    const payment =\n      scriptType === 'p2tr' ? createPaymentP2tr(walletKeys.publicKeys) : createPaymentP2trMusig2(walletKeys.publicKeys);\n    psbt.addOutput({ script: payment.output!, value });\n  } else {\n    const { scriptPubKey: script } = createOutputScript2of3(walletKeys.publicKeys, scriptType);\n    psbt.addOutput({ script, value });\n  }\n  updateWalletOutputForPsbt(psbt, rootWalletKeys, psbt.data.outputs.length - 1, chain, index);\n}\n\n/**\n * Fold the script ids into a single script id, if they are all the same.\n * @param scriptIds\n */\nfunction foldScriptIds(scriptIds: ScriptId[]): ScriptId {\n  if (scriptIds.length === 0) {\n    throw new Error('cannot fold empty script ids');\n  }\n  scriptIds.forEach((scriptId, i) => {\n    if (scriptId.chain !== scriptIds[0].chain) {\n      throw new Error(`chain mismatch: ${scriptId.chain} != ${scriptIds[0].chain}`);\n    }\n    if (scriptId.index !== scriptIds[0].index) {\n      throw new Error(`index mismatch: ${scriptId.index} != ${scriptIds[0].index}`);\n    }\n  });\n  return scriptIds[0];\n}\n\n/**\n * Get the script id from the output.\n * The output can have either bip32Derivation or tapBip32Derivation, but not both.\n * @param output\n * @throws Error if neither or both bip32Derivation and tapBip32Derivation are present\n * @throws Error if the output is empty\n * @throws Error if we cannot fold the script ids into a single script id\n */\nexport function getScriptIdFromOutput(output: {\n  bip32Derivation?: { path: string }[];\n  tapBip32Derivation?: { path: string }[];\n}): ScriptId {\n  if (output.bip32Derivation && output.tapBip32Derivation) {\n    throw new Error('cannot get script id from output with both bip32Derivation and tapBip32Derivation');\n  }\n  if (output.bip32Derivation) {\n    return foldScriptIds(output.bip32Derivation.map((d) => getScriptIdFromPath(d.path)));\n  }\n  if (output.tapBip32Derivation) {\n    return foldScriptIds(output.tapBip32Derivation.map((d) => getScriptIdFromPath(d.path)));\n  }\n  throw new Error('cannot get script id from output without bip32Derivation or tapBip32Derivation');\n}\n"]}

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


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