PHP WebShell

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

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

"use strict";
/*

This file contains code for creating an output layouts for transactions that pass on inscriptions.

When passing on an inscription, we want to satisfy a few constraints:

* All outputs should be larger than a minimal value (dust limit)
* The sum of all output values needs to be less than the input to cover the transaction fee.
* The output containing the inscription should be as small as possible, but large enough to
  contain the inscription.

To keep the inscription output small, we can pad the satoshi range preceding and following the range
with change outputs, which have a minimal size and incur a fee cost.


Broadly speaking, there are four scenarios:

(1) Small inscription input that has just enough value to pay for fee and a single inscription
    output (u0). No padding outputs.
    ┌────────┬────────┐
    │        │ u0     │
    │      r ┼        │
    │        ├────────┘
    │        │ fee
    └────────┘


(2) Large inscription input with inscription close to start of input.
    Inscription output followed by change output (u1) padding the remaining value.

    ┌────────┬────────┐
    │        │ u0     │
    │      r ┼        │
    │        ├────────┤
    │        │ u1     │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        ├────────┘
    │        │
    │        │ fee
    │        │
    └────────┘


(3) Large inscription input with inscription close to end of input.
    Change output padding start followed by inscription output.

    ┌────────┬────────┐
    │        │ u0     │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        ├────────┤
    │      r ┼        │
    │        │ u1     │
    │        ├────────┘
    │        │
    │        │ fee
    │        │
    └────────┘


(4) Large inscription input with inscription in the middle.
    Inscription input (u1) with padding on both sides (u0 and u2)

    ┌────────┬────────┐
    │        │ u0     │
    │        │        │
    │        │        │
    │        │        │
    │        │        │
    │        ├────────┤
    │        │ u1     │
    │      r ┼        │
    │        ├────────┤
    │        │ u2     │
    │        │        │
    │        │        │
    │        │        │
    │        ├────────┘
    │        │
    │        │ fee
    │        │
    └────────┘
 */
Object.defineProperty(exports, "__esModule", { value: true });
exports.toArray = toArray;
exports.toParameters = toParameters;
exports.getOrdOutputsForLayout = getOrdOutputsForLayout;
exports.findOutputLayout = findOutputLayout;
const ZERO = BigInt(0);
const ONE = BigInt(1);
function max(a, b) {
    return a < b ? b : a;
}
function min(a, b) {
    return a < b ? a : b;
}
function sum(arr) {
    return arr.reduce((a, b) => a + b, ZERO);
}
/** @return canonical sequence of parameters */
function toArray(p) {
    return [p.firstChangeOutput, p.inscriptionOutput, p.secondChangeOutput, p.feeOutput];
}
function toParameters(firstChangeOutput, inscriptionOutput, secondChangeOutput, feeOutput) {
    return {
        firstChangeOutput,
        inscriptionOutput,
        secondChangeOutput,
        feeOutput,
    };
}
/**
 * Translates a layout into OrdOutputs. Absent outputs are set to `null`.
 *
 * @param inscriptionInput
 * @param layout
 * @return OrdOutputs for layout
 */
function getOrdOutputsForLayout(inscriptionInput, layout) {
    const outputs = inscriptionInput.splitAllWithParams(toArray(layout), { exact: true, allowZero: true });
    if (outputs.length !== 4) {
        throw new Error(`unexpected result`);
    }
    return toParameters(...outputs);
}
/**
 * @param constraints
 * @param inscriptionInput
 * @param layout
 * @return true iff layout satisfies constraints
 */
function check(constraints, inscriptionInput, layout) {
    if ((layout.firstChangeOutput === ZERO || constraints.minChangeOutput <= layout.firstChangeOutput) &&
        (layout.secondChangeOutput === ZERO || constraints.minChangeOutput <= layout.secondChangeOutput) &&
        constraints.minInscriptionOutput <= layout.inscriptionOutput &&
        layout.inscriptionOutput <= constraints.maxInscriptionOutput &&
        getFeeForOutputs(constraints, [layout.firstChangeOutput, layout.inscriptionOutput, layout.secondChangeOutput]) <=
            layout.feeOutput &&
        sum(toArray(layout)) === inscriptionInput.value) {
        /* make sure inscription actually lies on the inscriptionOutput */
        const outputs = getOrdOutputsForLayout(inscriptionInput, layout);
        return outputs.inscriptionOutput?.ordinals.length === 1;
    }
    return false;
}
function getFeeForOutputs(p, outputs) {
    return outputs.reduce((sum, oValue) => sum + (oValue === ZERO ? ZERO : p.feePerOutput), p.feeFixed);
}
function getStartChangeOutput(c) {
    // we don't need a change padding output
    if (c.satPos < c.maxInscriptionOutput) {
        return ZERO;
    }
    if (c.minChangeOutput <= c.satPos) {
        return c.satPos;
    }
    return null;
}
function getInscriptionOutput(c, startChangeOutput) {
    const result = min(c.maxInscriptionOutput, max(c.minInscriptionOutput, c.satPos - startChangeOutput + ONE));
    if (c.satPos < startChangeOutput || startChangeOutput + result < c.satPos) {
        return null;
    }
    // if is not worth creating an end change output, let's maximize the inscription output
    if (getEndChangeOutput(c, startChangeOutput, result) === ZERO) {
        const remainder = c.total - startChangeOutput - getFeeForOutputs(c, [startChangeOutput, result]);
        return min(remainder, c.maxInscriptionOutput);
    }
    return result;
}
function getEndChangeOutput(c, startChangeOutput, inscriptionOutput) {
    const remainder = c.total - sum([startChangeOutput, inscriptionOutput]);
    const minFeeWithoutSecondOutput = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput]);
    const minFeeWithSecondOutput = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput, c.minChangeOutput]);
    if (remainder < minFeeWithoutSecondOutput) {
        // We cannot even pay the fee for the output(s) we have so far.
        return null;
    }
    if (remainder - minFeeWithSecondOutput < c.minChangeOutput) {
        // The remainder is too small to pay for the end change output. Let's skip it and pay a higher fee.
        return ZERO;
    }
    // let's use as much as we can for fee while leaving enough for fee
    return remainder - minFeeWithSecondOutput;
}
function getFeeOutput(c, startChangeOutput, inscriptionOutput, endChangeOutput) {
    const minFee = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput, endChangeOutput]);
    const remainder = c.total - sum([startChangeOutput, inscriptionOutput, endChangeOutput]);
    if (remainder < minFee) {
        return null;
    }
    return remainder;
}
/**
 * @param inscriptionInput
 * @param search
 * @return a solution that satisfies constraints. If no solution can be found, return `undefined`.
 */
function findOutputLayout(inscriptionInput, search) {
    if (inscriptionInput.ordinals.length !== 1) {
        throw new Error(`unexpected ordinal count ${inscriptionInput.ordinals.length}`);
    }
    if (inscriptionInput.ordinals[0].size() !== ONE) {
        throw new Error(`only single-satoshi inscriptions are supported`);
    }
    const satPos = inscriptionInput.ordinals[0].start;
    const total = inscriptionInput.value;
    const constraints = { ...search, satPos, total };
    const startChangeOutput = getStartChangeOutput(constraints);
    if (startChangeOutput === null) {
        return;
    }
    const inscriptionOutput = getInscriptionOutput(constraints, startChangeOutput);
    if (inscriptionOutput === null) {
        return;
    }
    const endChangeOutput = getEndChangeOutput(constraints, startChangeOutput, inscriptionOutput);
    if (endChangeOutput === null) {
        return;
    }
    const feeOutput = getFeeOutput(constraints, startChangeOutput, inscriptionOutput, endChangeOutput);
    if (feeOutput === null) {
        return;
    }
    const result = toParameters(startChangeOutput, inscriptionOutput, endChangeOutput, feeOutput);
    if (!check(constraints, inscriptionInput, result)) {
        throw new Error(`invalid result`);
    }
    return result;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"OutputLayout.js","sourceRoot":"","sources":["../../src/OutputLayout.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n\nThis file contains code for creating an output layouts for transactions that pass on inscriptions.\n\nWhen passing on an inscription, we want to satisfy a few constraints:\n\n* All outputs should be larger than a minimal value (dust limit)\n* The sum of all output values needs to be less than the input to cover the transaction fee.\n* The output containing the inscription should be as small as possible, but large enough to\n  contain the inscription.\n\nTo keep the inscription output small, we can pad the satoshi range preceding and following the range\nwith change outputs, which have a minimal size and incur a fee cost.\n\n\nBroadly speaking, there are four scenarios:\n\n(1) Small inscription input that has just enough value to pay for fee and a single inscription\n    output (u0). No padding outputs.\n    ┌────────┬────────┐\n    │        │ u0     │\n    │      r ┼        │\n    │        ├────────┘\n    │        │ fee\n    └────────┘\n\n\n(2) Large inscription input with inscription close to start of input.\n    Inscription output followed by change output (u1) padding the remaining value.\n\n    ┌────────┬────────┐\n    │        │ u0     │\n    │      r ┼        │\n    │        ├────────┤\n    │        │ u1     │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        ├────────┘\n    │        │\n    │        │ fee\n    │        │\n    └────────┘\n\n\n(3) Large inscription input with inscription close to end of input.\n    Change output padding start followed by inscription output.\n\n    ┌────────┬────────┐\n    │        │ u0     │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        ├────────┤\n    │      r ┼        │\n    │        │ u1     │\n    │        ├────────┘\n    │        │\n    │        │ fee\n    │        │\n    └────────┘\n\n\n(4) Large inscription input with inscription in the middle.\n    Inscription input (u1) with padding on both sides (u0 and u2)\n\n    ┌────────┬────────┐\n    │        │ u0     │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        ├────────┤\n    │        │ u1     │\n    │      r ┼        │\n    │        ├────────┤\n    │        │ u2     │\n    │        │        │\n    │        │        │\n    │        │        │\n    │        ├────────┘\n    │        │\n    │        │ fee\n    │        │\n    └────────┘\n */\n\nimport { OrdOutput } from './OrdOutput';\n\nconst ZERO = BigInt(0);\nconst ONE = BigInt(1);\n\nfunction max(a: bigint, b: bigint): bigint {\n  return a < b ? b : a;\n}\n\nfunction min(a: bigint, b: bigint): bigint {\n  return a < b ? a : b;\n}\n\nfunction sum(arr: bigint[]): bigint {\n  return arr.reduce((a, b) => a + b, ZERO);\n}\n\n/**\n * A range constraint\n */\ntype Parameters<T> = {\n  /** Padding preceding the inscription output */\n  firstChangeOutput: T;\n  /** The inscription output that will inherit the input inscription */\n  inscriptionOutput: T;\n  /** Padding following the inscription output */\n  secondChangeOutput: T;\n  /** Not a real output, used only to simplify calculations */\n  feeOutput: T;\n};\n\n/** @return canonical sequence of parameters */\nexport function toArray<T>(p: Parameters<T>): [T, T, T, T] {\n  return [p.firstChangeOutput, p.inscriptionOutput, p.secondChangeOutput, p.feeOutput];\n}\n\nexport function toParameters<T>(\n  firstChangeOutput: T,\n  inscriptionOutput: T,\n  secondChangeOutput: T,\n  feeOutput: T\n): Parameters<T> {\n  return {\n    firstChangeOutput,\n    inscriptionOutput,\n    secondChangeOutput,\n    feeOutput,\n  };\n}\n\n/** A finished output layout */\nexport type OutputLayout = Parameters<bigint>;\n\n/**\n * Translates a layout into OrdOutputs. Absent outputs are set to `null`.\n *\n * @param inscriptionInput\n * @param layout\n * @return OrdOutputs for layout\n */\nexport function getOrdOutputsForLayout(\n  inscriptionInput: OrdOutput,\n  layout: OutputLayout\n): Parameters<OrdOutput | null> {\n  const outputs = inscriptionInput.splitAllWithParams(toArray(layout), { exact: true, allowZero: true });\n  if (outputs.length !== 4) {\n    throw new Error(`unexpected result`);\n  }\n  type T = OrdOutput | null;\n  return toParameters(...(outputs as [T, T, T, T]));\n}\n\n/**\n * @param constraints\n * @param inscriptionInput\n * @param layout\n * @return true iff layout satisfies constraints\n */\nfunction check(constraints: Constraints, inscriptionInput: OrdOutput, layout: OutputLayout): boolean {\n  if (\n    (layout.firstChangeOutput === ZERO || constraints.minChangeOutput <= layout.firstChangeOutput) &&\n    (layout.secondChangeOutput === ZERO || constraints.minChangeOutput <= layout.secondChangeOutput) &&\n    constraints.minInscriptionOutput <= layout.inscriptionOutput &&\n    layout.inscriptionOutput <= constraints.maxInscriptionOutput &&\n    getFeeForOutputs(constraints, [layout.firstChangeOutput, layout.inscriptionOutput, layout.secondChangeOutput]) <=\n      layout.feeOutput &&\n    sum(toArray(layout)) === inscriptionInput.value\n  ) {\n    /* make sure inscription actually lies on the inscriptionOutput */\n    const outputs = getOrdOutputsForLayout(inscriptionInput, layout);\n    return outputs.inscriptionOutput?.ordinals.length === 1;\n  }\n\n  return false;\n}\n\n/**\n * High-level constraints for output layout\n */\nexport type Constraints = {\n  minChangeOutput: bigint;\n  minInscriptionOutput: bigint;\n  maxInscriptionOutput: bigint;\n  feeFixed: bigint;\n  feePerOutput: bigint;\n  satPos: bigint;\n  total: bigint;\n};\n\nfunction getFeeForOutputs(p: { feePerOutput: bigint; feeFixed: bigint }, outputs: bigint[]): bigint {\n  return outputs.reduce((sum, oValue): bigint => sum + (oValue === ZERO ? ZERO : p.feePerOutput), p.feeFixed);\n}\nfunction getStartChangeOutput(c: Constraints): bigint | null {\n  // we don't need a change padding output\n  if (c.satPos < c.maxInscriptionOutput) {\n    return ZERO;\n  }\n  if (c.minChangeOutput <= c.satPos) {\n    return c.satPos;\n  }\n  return null;\n}\n\nfunction getInscriptionOutput(c: Constraints, startChangeOutput: bigint): bigint | null {\n  const result = min(c.maxInscriptionOutput, max(c.minInscriptionOutput, c.satPos - startChangeOutput + ONE));\n  if (c.satPos < startChangeOutput || startChangeOutput + result < c.satPos) {\n    return null;\n  }\n  // if is not worth creating an end change output, let's maximize the inscription output\n  if (getEndChangeOutput(c, startChangeOutput, result) === ZERO) {\n    const remainder = c.total - startChangeOutput - getFeeForOutputs(c, [startChangeOutput, result]);\n    return min(remainder, c.maxInscriptionOutput);\n  }\n  return result;\n}\n\nfunction getEndChangeOutput(c: Constraints, startChangeOutput: bigint, inscriptionOutput: bigint): bigint | null {\n  const remainder = c.total - sum([startChangeOutput, inscriptionOutput]);\n  const minFeeWithoutSecondOutput = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput]);\n  const minFeeWithSecondOutput = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput, c.minChangeOutput]);\n  if (remainder < minFeeWithoutSecondOutput) {\n    // We cannot even pay the fee for the output(s) we have so far.\n    return null;\n  }\n  if (remainder - minFeeWithSecondOutput < c.minChangeOutput) {\n    // The remainder is too small to pay for the end change output. Let's skip it and pay a higher fee.\n    return ZERO;\n  }\n  // let's use as much as we can for fee while leaving enough for fee\n  return remainder - minFeeWithSecondOutput;\n}\n\nfunction getFeeOutput(\n  c: Constraints,\n  startChangeOutput: bigint,\n  inscriptionOutput: bigint,\n  endChangeOutput: bigint\n): bigint | null {\n  const minFee = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput, endChangeOutput]);\n  const remainder = c.total - sum([startChangeOutput, inscriptionOutput, endChangeOutput]);\n  if (remainder < minFee) {\n    return null;\n  }\n  return remainder;\n}\n\n/**\n * @param inscriptionInput\n * @param search\n * @return a solution that satisfies constraints. If no solution can be found, return `undefined`.\n */\nexport function findOutputLayout(\n  inscriptionInput: OrdOutput,\n  search: Omit<Constraints, 'satPos' | 'total'>\n): OutputLayout | undefined {\n  if (inscriptionInput.ordinals.length !== 1) {\n    throw new Error(`unexpected ordinal count ${inscriptionInput.ordinals.length}`);\n  }\n  if (inscriptionInput.ordinals[0].size() !== ONE) {\n    throw new Error(`only single-satoshi inscriptions are supported`);\n  }\n\n  const satPos = inscriptionInput.ordinals[0].start;\n  const total = inscriptionInput.value;\n\n  const constraints: Constraints = { ...search, satPos, total };\n\n  const startChangeOutput = getStartChangeOutput(constraints);\n  if (startChangeOutput === null) {\n    return;\n  }\n  const inscriptionOutput = getInscriptionOutput(constraints, startChangeOutput);\n  if (inscriptionOutput === null) {\n    return;\n  }\n  const endChangeOutput = getEndChangeOutput(constraints, startChangeOutput, inscriptionOutput);\n  if (endChangeOutput === null) {\n    return;\n  }\n  const feeOutput = getFeeOutput(constraints, startChangeOutput, inscriptionOutput, endChangeOutput);\n  if (feeOutput === null) {\n    return;\n  }\n  const result = toParameters(startChangeOutput, inscriptionOutput, endChangeOutput, feeOutput);\n  if (!check(constraints, inscriptionInput, result)) {\n    throw new Error(`invalid result`);\n  }\n  return result;\n}\n"]}

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


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