@bitgo-beta/utxo-ord 1.1.3-alpha.412 → 1.1.3-alpha.414
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/src/OrdOutput.d.ts.map +1 -0
- package/dist/cjs/src/OrdOutput.js +197 -0
- package/dist/cjs/src/OutputLayout.d.ts.map +1 -0
- package/dist/cjs/src/OutputLayout.js +246 -0
- package/dist/cjs/src/SatPoint.d.ts.map +1 -0
- package/dist/cjs/src/SatPoint.js +48 -0
- package/dist/cjs/src/SatRange.d.ts.map +1 -0
- package/dist/cjs/src/SatRange.js +58 -0
- package/dist/cjs/src/combinations.d.ts.map +1 -0
- package/dist/cjs/src/combinations.js +13 -0
- package/dist/cjs/src/index.d.ts.map +1 -0
- package/dist/{src → cjs/src}/index.js +1 -1
- package/dist/cjs/src/inscriptions.d.ts.map +1 -0
- package/dist/cjs/src/inscriptions.js +202 -0
- package/dist/cjs/src/psbt.d.ts.map +1 -0
- package/dist/cjs/src/psbt.js +154 -0
- package/dist/cjs/tsconfig.tsbuildinfo +1 -0
- package/dist/esm/OrdOutput.d.ts +65 -0
- package/dist/esm/OrdOutput.js +192 -0
- package/dist/esm/OutputLayout.d.ts +47 -0
- package/dist/esm/OutputLayout.js +240 -0
- package/dist/esm/SatPoint.d.ts +13 -0
- package/dist/esm/SatPoint.js +43 -0
- package/dist/esm/SatRange.d.ts +26 -0
- package/dist/esm/SatRange.js +53 -0
- package/dist/esm/combinations.d.ts +2 -0
- package/dist/esm/combinations.js +10 -0
- package/dist/esm/index.d.ts +10 -0
- package/dist/esm/index.js +10 -0
- package/dist/esm/inscriptions.d.ts +31 -0
- package/dist/esm/inscriptions.js +164 -0
- package/dist/esm/psbt.d.ts +67 -0
- package/dist/{src → esm}/psbt.js +31 -38
- package/package.json +26 -9
- package/dist/src/OrdOutput.js +0 -197
- package/dist/src/OutputLayout.js +0 -246
- package/dist/src/SatPoint.js +0 -48
- package/dist/src/SatRange.js +0 -58
- package/dist/src/combinations.js +0 -13
- package/dist/src/inscriptions.js +0 -202
- package/dist/tsconfig.tsbuildinfo +0 -1
- /package/dist/{src → cjs/src}/OrdOutput.d.ts +0 -0
- /package/dist/{src → cjs/src}/OutputLayout.d.ts +0 -0
- /package/dist/{src → cjs/src}/SatPoint.d.ts +0 -0
- /package/dist/{src → cjs/src}/SatRange.d.ts +0 -0
- /package/dist/{src → cjs/src}/combinations.d.ts +0 -0
- /package/dist/{src → cjs/src}/index.d.ts +0 -0
- /package/dist/{src → cjs/src}/inscriptions.d.ts +0 -0
- /package/dist/{src → cjs/src}/psbt.d.ts +0 -0
- /package/dist/{src → esm}/OrdOutput.d.ts.map +0 -0
- /package/dist/{src → esm}/OutputLayout.d.ts.map +0 -0
- /package/dist/{src → esm}/SatPoint.d.ts.map +0 -0
- /package/dist/{src → esm}/SatRange.d.ts.map +0 -0
- /package/dist/{src → esm}/combinations.d.ts.map +0 -0
- /package/dist/{src → esm}/index.d.ts.map +0 -0
- /package/dist/{src → esm}/inscriptions.d.ts.map +0 -0
- /package/dist/{src → esm}/psbt.d.ts.map +0 -0
package/dist/{src → esm}/psbt.js
RENAMED
|
@@ -1,30 +1,24 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
const
|
|
9
|
-
const OrdOutput_1 = require("./OrdOutput");
|
|
10
|
-
const SatPoint_1 = require("./SatPoint");
|
|
11
|
-
const SatRange_1 = require("./SatRange");
|
|
12
|
-
const OutputLayout_1 = require("./OutputLayout");
|
|
13
|
-
const combinations_1 = require("./combinations");
|
|
14
|
-
exports.DefaultInscriptionConstraints = {
|
|
1
|
+
import { bitgo, address } from '@bitgo-beta/utxo-lib';
|
|
2
|
+
import { Dimensions, VirtualSizes } from '@bitgo-beta/unspents';
|
|
3
|
+
import { OrdOutput } from './OrdOutput';
|
|
4
|
+
import { parseSatPoint } from './SatPoint';
|
|
5
|
+
import { SatRange } from './SatRange';
|
|
6
|
+
import { getOrdOutputsForLayout, toArray, findOutputLayout } from './OutputLayout';
|
|
7
|
+
import { powerset } from './combinations';
|
|
8
|
+
export const DefaultInscriptionConstraints = {
|
|
15
9
|
minChangeOutput: BigInt(10000),
|
|
16
10
|
minInscriptionOutput: BigInt(10000),
|
|
17
11
|
maxInscriptionOutput: BigInt(20000),
|
|
18
12
|
};
|
|
19
|
-
function createPsbtFromOutputLayout(network, inputBuilder, unspents, outputs, outputLayout) {
|
|
20
|
-
const psbt =
|
|
13
|
+
export function createPsbtFromOutputLayout(network, inputBuilder, unspents, outputs, outputLayout) {
|
|
14
|
+
const psbt = bitgo.createPsbtForNetwork({ network: network });
|
|
21
15
|
if (unspents.length === 0) {
|
|
22
16
|
throw new Error(`must provide at least one unspent`);
|
|
23
17
|
}
|
|
24
|
-
unspents.forEach((u) =>
|
|
25
|
-
const ordInput =
|
|
26
|
-
const ordOutputs =
|
|
27
|
-
|
|
18
|
+
unspents.forEach((u) => bitgo.addWalletUnspentToPsbt(psbt, u, inputBuilder.walletKeys, inputBuilder.signer, inputBuilder.cosigner));
|
|
19
|
+
const ordInput = OrdOutput.joinAll(unspents.map((u) => new OrdOutput(u.value)));
|
|
20
|
+
const ordOutputs = getOrdOutputsForLayout(ordInput, outputLayout);
|
|
21
|
+
toArray(ordOutputs).forEach((ordOutput) => {
|
|
28
22
|
if (ordOutput === null) {
|
|
29
23
|
return;
|
|
30
24
|
}
|
|
@@ -37,13 +31,13 @@ function createPsbtFromOutputLayout(network, inputBuilder, unspents, outputs, ou
|
|
|
37
31
|
case ordOutputs.firstChangeOutput:
|
|
38
32
|
case ordOutputs.secondChangeOutput:
|
|
39
33
|
const { chain, index } = ordOutput === ordOutputs.firstChangeOutput ? outputs.changeOutputs[0] : outputs.changeOutputs[1];
|
|
40
|
-
|
|
34
|
+
bitgo.addWalletOutputToPsbt(psbt, inputBuilder.walletKeys, chain, index, ordOutput.value);
|
|
41
35
|
break;
|
|
42
36
|
// add actual inscription output
|
|
43
37
|
case ordOutputs.inscriptionOutput:
|
|
44
38
|
let { inscriptionRecipient } = outputs;
|
|
45
39
|
if (typeof inscriptionRecipient === 'string') {
|
|
46
|
-
inscriptionRecipient =
|
|
40
|
+
inscriptionRecipient = address.toOutputScript(inscriptionRecipient, network);
|
|
47
41
|
}
|
|
48
42
|
psbt.addOutput({
|
|
49
43
|
script: inscriptionRecipient,
|
|
@@ -55,8 +49,8 @@ function createPsbtFromOutputLayout(network, inputBuilder, unspents, outputs, ou
|
|
|
55
49
|
return psbt;
|
|
56
50
|
}
|
|
57
51
|
function toSatRange(p) {
|
|
58
|
-
const { offset } =
|
|
59
|
-
return new
|
|
52
|
+
const { offset } = parseSatPoint(p);
|
|
53
|
+
return new SatRange(offset, offset);
|
|
60
54
|
}
|
|
61
55
|
function getFee(vsize, rateSatPerKB) {
|
|
62
56
|
return BigInt(Math.ceil((vsize * rateSatPerKB) / 1000));
|
|
@@ -68,7 +62,7 @@ function getFee(vsize, rateSatPerKB) {
|
|
|
68
62
|
* @param constraints
|
|
69
63
|
* @param minimizeInputs
|
|
70
64
|
*/
|
|
71
|
-
function findOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints, { minimizeInputs = false } = {}) {
|
|
65
|
+
export function findOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints, { minimizeInputs = false } = {}) {
|
|
72
66
|
if (minimizeInputs) {
|
|
73
67
|
return findSmallestOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints);
|
|
74
68
|
}
|
|
@@ -79,24 +73,24 @@ function findOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraint
|
|
|
79
73
|
// otherwise our fee calc is too complicated
|
|
80
74
|
throw new Error(`wallet outputs must be on same chain`);
|
|
81
75
|
}
|
|
82
|
-
const { minChangeOutput =
|
|
76
|
+
const { minChangeOutput = DefaultInscriptionConstraints.minChangeOutput, minInscriptionOutput = DefaultInscriptionConstraints.minInscriptionOutput, maxInscriptionOutput = DefaultInscriptionConstraints.maxInscriptionOutput, } = constraints;
|
|
83
77
|
// Join all the inputs into a single inscriptionOutput.
|
|
84
78
|
// For the purposes of finding a layout there is no difference.
|
|
85
|
-
const inscriptionOutput =
|
|
86
|
-
const layout =
|
|
79
|
+
const inscriptionOutput = OrdOutput.joinAll(inputs.map((i) => new OrdOutput(i.value, i === inputs[0] ? [toSatRange(satPoint)] : [])));
|
|
80
|
+
const layout = findOutputLayout(inscriptionOutput, {
|
|
87
81
|
minChangeOutput,
|
|
88
82
|
minInscriptionOutput,
|
|
89
83
|
maxInscriptionOutput,
|
|
90
|
-
feeFixed: getFee(
|
|
91
|
-
|
|
84
|
+
feeFixed: getFee(VirtualSizes.txSegOverheadVSize +
|
|
85
|
+
Dimensions.fromUnspents(inputs, {
|
|
92
86
|
p2tr: { scriptPathLevel: 1 },
|
|
93
87
|
p2trMusig2: { scriptPathLevel: undefined },
|
|
94
88
|
}).getInputsVSize(), constraints.feeRateSatKB),
|
|
95
|
-
feePerOutput: getFee(
|
|
89
|
+
feePerOutput: getFee(Dimensions.fromOutputOnChain(outputs.changeOutputs[0].chain).getOutputsVSize(), constraints.feeRateSatKB),
|
|
96
90
|
});
|
|
97
91
|
return layout ? { inputs, layout } : undefined;
|
|
98
92
|
}
|
|
99
|
-
|
|
93
|
+
export const MAX_UNSPENTS_FOR_OUTPUT_LAYOUT = 5;
|
|
100
94
|
/**
|
|
101
95
|
* @param inputs - inscription input must come first
|
|
102
96
|
* @param satPoint - location of the inscription
|
|
@@ -104,11 +98,11 @@ exports.MAX_UNSPENTS_FOR_OUTPUT_LAYOUT = 5;
|
|
|
104
98
|
* @param constraints
|
|
105
99
|
*/
|
|
106
100
|
function findSmallestOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints) {
|
|
107
|
-
if (
|
|
101
|
+
if (MAX_UNSPENTS_FOR_OUTPUT_LAYOUT < inputs.length) {
|
|
108
102
|
throw new Error(`input array is too large`);
|
|
109
103
|
}
|
|
110
104
|
// create powerset of all supplementary inputs and find the cheapest result
|
|
111
|
-
const inputsArr = [inputs, ...
|
|
105
|
+
const inputsArr = [inputs, ...powerset(inputs.slice(1)).map((s) => [inputs[0], ...s])];
|
|
112
106
|
return inputsArr
|
|
113
107
|
.map((inputs) => findOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints))
|
|
114
108
|
.reduce((best, next) => {
|
|
@@ -121,12 +115,11 @@ function findSmallestOutputLayoutForWalletUnspents(inputs, satPoint, outputs, co
|
|
|
121
115
|
return best.layout.feeOutput < next.layout.feeOutput ? best : next;
|
|
122
116
|
});
|
|
123
117
|
}
|
|
124
|
-
class ErrorNoLayout extends Error {
|
|
118
|
+
export class ErrorNoLayout extends Error {
|
|
125
119
|
constructor() {
|
|
126
120
|
super('Could not find output layout for inscription passing transaction');
|
|
127
121
|
}
|
|
128
122
|
}
|
|
129
|
-
exports.ErrorNoLayout = ErrorNoLayout;
|
|
130
123
|
/**
|
|
131
124
|
* @param network
|
|
132
125
|
* @param inputBuilder
|
|
@@ -137,7 +130,7 @@ exports.ErrorNoLayout = ErrorNoLayout;
|
|
|
137
130
|
* @param supplementaryUnspents - additional inputs to cover fee.
|
|
138
131
|
* @param [minimizeInputs=true] - try to find input combination with minimal fees. Limits supplementaryUnspents to 4.
|
|
139
132
|
*/
|
|
140
|
-
function createPsbtForSingleInscriptionPassingTransaction(network, inputBuilder, unspent, satPoint, outputs, constraints, { supplementaryUnspents = [], minimizeInputs = true, } = {}) {
|
|
133
|
+
export function createPsbtForSingleInscriptionPassingTransaction(network, inputBuilder, unspent, satPoint, outputs, constraints, { supplementaryUnspents = [], minimizeInputs = true, } = {}) {
|
|
141
134
|
// support for legacy call style
|
|
142
135
|
if (Array.isArray(unspent)) {
|
|
143
136
|
if (unspent.length !== 1) {
|
|
@@ -151,4 +144,4 @@ function createPsbtForSingleInscriptionPassingTransaction(network, inputBuilder,
|
|
|
151
144
|
}
|
|
152
145
|
return createPsbtFromOutputLayout(network, inputBuilder, result.inputs, outputs, result.layout);
|
|
153
146
|
}
|
|
154
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../src/psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import { Network, bitgo, address } from '@bitgo-beta/utxo-lib';\nimport { Dimensions, VirtualSizes } from '@bitgo-beta/unspents';\n\nimport { OrdOutput } from './OrdOutput';\nimport { parseSatPoint, SatPoint } from './SatPoint';\nimport { SatRange } from './SatRange';\nimport { getOrdOutputsForLayout, OutputLayout, toArray, findOutputLayout } from './OutputLayout';\nimport { powerset } from './combinations';\n\ntype WalletUnspent = bitgo.WalletUnspent<bigint>;\n\nexport type WalletOutputPath = {\n  chain: bitgo.ChainCode;\n  index: number;\n};\n\nexport type WalletInputBuilder = {\n  walletKeys: bitgo.RootWalletKeys;\n  signer: bitgo.KeyName;\n  cosigner: bitgo.KeyName;\n};\n\n/**\n * Describes all outputs of an inscription transaction\n */\nexport type InscriptionTransactionOutputs = {\n  inscriptionRecipient: string | Buffer;\n  changeOutputs: [WalletOutputPath, WalletOutputPath];\n};\n\n/** @deprecated */\nexport type InscriptionOutputs = InscriptionTransactionOutputs;\n\nexport type InscriptionTransactionConstraints = {\n  feeRateSatKB: number;\n  minChangeOutput?: bigint;\n  minInscriptionOutput?: bigint;\n  maxInscriptionOutput?: bigint;\n};\n\nexport const DefaultInscriptionConstraints = {\n  minChangeOutput: BigInt(10_000),\n  minInscriptionOutput: BigInt(10_000),\n  maxInscriptionOutput: BigInt(20_000),\n};\n\nexport function createPsbtFromOutputLayout(\n  network: Network,\n  inputBuilder: WalletInputBuilder,\n  unspents: WalletUnspent[],\n  outputs: InscriptionTransactionOutputs,\n  outputLayout: OutputLayout\n): bitgo.UtxoPsbt {\n  const psbt = bitgo.createPsbtForNetwork({ network: network });\n  if (unspents.length === 0) {\n    throw new Error(`must provide at least one unspent`);\n  }\n  unspents.forEach((u) =>\n    bitgo.addWalletUnspentToPsbt(psbt, u, inputBuilder.walletKeys, inputBuilder.signer, inputBuilder.cosigner)\n  );\n  const ordInput = OrdOutput.joinAll(unspents.map((u) => new OrdOutput(u.value)));\n  const ordOutputs = getOrdOutputsForLayout(ordInput, outputLayout);\n  toArray(ordOutputs).forEach((ordOutput) => {\n    if (ordOutput === null) {\n      return;\n    }\n    switch (ordOutput) {\n      // skip padding outputs and fee output (virtual)\n      case null:\n      case ordOutputs.feeOutput:\n        return;\n      // add padding outputs\n      case ordOutputs.firstChangeOutput:\n      case ordOutputs.secondChangeOutput:\n        const { chain, index } =\n          ordOutput === ordOutputs.firstChangeOutput ? outputs.changeOutputs[0] : outputs.changeOutputs[1];\n        bitgo.addWalletOutputToPsbt(psbt, inputBuilder.walletKeys, chain, index, ordOutput.value);\n        break;\n      // add actual inscription output\n      case ordOutputs.inscriptionOutput:\n        let { inscriptionRecipient } = outputs;\n        if (typeof inscriptionRecipient === 'string') {\n          inscriptionRecipient = address.toOutputScript(inscriptionRecipient, network);\n        }\n        psbt.addOutput({\n          script: inscriptionRecipient,\n          value: ordOutput.value,\n        });\n        break;\n    }\n  });\n  return psbt;\n}\n\nfunction toSatRange(p: SatPoint) {\n  const { offset } = parseSatPoint(p);\n  return new SatRange(offset, offset);\n}\n\nfunction getFee(vsize: number, rateSatPerKB: number): bigint {\n  return BigInt(Math.ceil((vsize * rateSatPerKB) / 1000));\n}\n\n/**\n * @param inputs - inscription input must come first\n * @param satPoint - location of the inscription\n * @param outputs\n * @param constraints\n * @param minimizeInputs\n */\nexport function findOutputLayoutForWalletUnspents(\n  inputs: WalletUnspent[],\n  satPoint: SatPoint,\n  outputs: InscriptionTransactionOutputs,\n  constraints: InscriptionTransactionConstraints,\n  { minimizeInputs = false } = {}\n): { inputs: WalletUnspent[]; layout: OutputLayout } | undefined {\n  if (minimizeInputs) {\n    return findSmallestOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints);\n  }\n\n  if (inputs.length === 0) {\n    throw new Error(`must provide at least one input`);\n  }\n\n  if (outputs.changeOutputs[0].chain !== outputs.changeOutputs[1].chain) {\n    // otherwise our fee calc is too complicated\n    throw new Error(`wallet outputs must be on same chain`);\n  }\n\n  const {\n    minChangeOutput = DefaultInscriptionConstraints.minChangeOutput,\n    minInscriptionOutput = DefaultInscriptionConstraints.minInscriptionOutput,\n    maxInscriptionOutput = DefaultInscriptionConstraints.maxInscriptionOutput,\n  } = constraints;\n\n  // Join all the inputs into a single inscriptionOutput.\n  // For the purposes of finding a layout there is no difference.\n  const inscriptionOutput = OrdOutput.joinAll(\n    inputs.map((i) => new OrdOutput(i.value, i === inputs[0] ? [toSatRange(satPoint)] : []))\n  );\n  const layout = findOutputLayout(inscriptionOutput, {\n    minChangeOutput,\n    minInscriptionOutput,\n    maxInscriptionOutput,\n    feeFixed: getFee(\n      VirtualSizes.txSegOverheadVSize +\n        Dimensions.fromUnspents(inputs, {\n          p2tr: { scriptPathLevel: 1 },\n          p2trMusig2: { scriptPathLevel: undefined },\n        }).getInputsVSize(),\n      constraints.feeRateSatKB\n    ),\n    feePerOutput: getFee(\n      Dimensions.fromOutputOnChain(outputs.changeOutputs[0].chain).getOutputsVSize(),\n      constraints.feeRateSatKB\n    ),\n  });\n\n  return layout ? { inputs, layout } : undefined;\n}\n\nexport const MAX_UNSPENTS_FOR_OUTPUT_LAYOUT = 5;\n\n/**\n * @param inputs - inscription input must come first\n * @param satPoint - location of the inscription\n * @param outputs\n * @param constraints\n */\nfunction findSmallestOutputLayoutForWalletUnspents(\n  inputs: WalletUnspent[],\n  satPoint: SatPoint,\n  outputs: InscriptionTransactionOutputs,\n  constraints: InscriptionTransactionConstraints\n): { inputs: WalletUnspent[]; layout: OutputLayout } | undefined {\n  if (MAX_UNSPENTS_FOR_OUTPUT_LAYOUT < inputs.length) {\n    throw new Error(`input array is too large`);\n  }\n  // create powerset of all supplementary inputs and find the cheapest result\n  const inputsArr = [inputs, ...powerset(inputs.slice(1)).map((s) => [inputs[0], ...s])];\n  return inputsArr\n    .map((inputs) => findOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints))\n    .reduce((best, next) => {\n      if (best === undefined) {\n        return next;\n      }\n      if (next === undefined) {\n        return best;\n      }\n      return best.layout.feeOutput < next.layout.feeOutput ? best : next;\n    });\n}\n\nexport class ErrorNoLayout extends Error {\n  constructor() {\n    super('Could not find output layout for inscription passing transaction');\n  }\n}\n\n/**\n * @param network\n * @param inputBuilder\n * @param unspent\n * @param satPoint\n * @param outputs\n * @param constraints\n * @param supplementaryUnspents - additional inputs to cover fee.\n * @param [minimizeInputs=true] - try to find input combination with minimal fees. Limits supplementaryUnspents to 4.\n */\nexport function createPsbtForSingleInscriptionPassingTransaction(\n  network: Network,\n  inputBuilder: WalletInputBuilder,\n  unspent: WalletUnspent | WalletUnspent[],\n  satPoint: SatPoint,\n  outputs: InscriptionTransactionOutputs,\n  constraints: InscriptionTransactionConstraints,\n  {\n    supplementaryUnspents = [],\n    minimizeInputs = true,\n  }: {\n    supplementaryUnspents?: WalletUnspent[];\n    minimizeInputs?: boolean;\n  } = {}\n): bitgo.UtxoPsbt {\n  // support for legacy call style\n  if (Array.isArray(unspent)) {\n    if (unspent.length !== 1) {\n      throw new Error(`can only pass single unspent`);\n    }\n    unspent = unspent[0];\n  }\n\n  const result = findOutputLayoutForWalletUnspents(\n    [unspent, ...supplementaryUnspents],\n    satPoint,\n    outputs,\n    constraints,\n    { minimizeInputs }\n  );\n\n  if (!result) {\n    throw new ErrorNoLayout();\n  }\n\n  return createPsbtFromOutputLayout(network, inputBuilder, result.inputs, outputs, result.layout);\n}\n"]}
|
|
147
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"psbt.js","sourceRoot":"","sources":["../../src/psbt.ts"],"names":[],"mappings":"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","sourcesContent":["import { Network, bitgo, address } from '@bitgo-beta/utxo-lib';\nimport { Dimensions, VirtualSizes } from '@bitgo-beta/unspents';\n\nimport { OrdOutput } from './OrdOutput';\nimport { parseSatPoint, SatPoint } from './SatPoint';\nimport { SatRange } from './SatRange';\nimport { getOrdOutputsForLayout, OutputLayout, toArray, findOutputLayout } from './OutputLayout';\nimport { powerset } from './combinations';\n\ntype WalletUnspent = bitgo.WalletUnspent<bigint>;\n\nexport type WalletOutputPath = {\n  chain: bitgo.ChainCode;\n  index: number;\n};\n\nexport type WalletInputBuilder = {\n  walletKeys: bitgo.RootWalletKeys;\n  signer: bitgo.KeyName;\n  cosigner: bitgo.KeyName;\n};\n\n/**\n * Describes all outputs of an inscription transaction\n */\nexport type InscriptionTransactionOutputs = {\n  inscriptionRecipient: string | Buffer;\n  changeOutputs: [WalletOutputPath, WalletOutputPath];\n};\n\n/** @deprecated */\nexport type InscriptionOutputs = InscriptionTransactionOutputs;\n\nexport type InscriptionTransactionConstraints = {\n  feeRateSatKB: number;\n  minChangeOutput?: bigint;\n  minInscriptionOutput?: bigint;\n  maxInscriptionOutput?: bigint;\n};\n\nexport const DefaultInscriptionConstraints = {\n  minChangeOutput: BigInt(10_000),\n  minInscriptionOutput: BigInt(10_000),\n  maxInscriptionOutput: BigInt(20_000),\n};\n\nexport function createPsbtFromOutputLayout(\n  network: Network,\n  inputBuilder: WalletInputBuilder,\n  unspents: WalletUnspent[],\n  outputs: InscriptionTransactionOutputs,\n  outputLayout: OutputLayout\n): bitgo.UtxoPsbt {\n  const psbt = bitgo.createPsbtForNetwork({ network: network });\n  if (unspents.length === 0) {\n    throw new Error(`must provide at least one unspent`);\n  }\n  unspents.forEach((u) =>\n    bitgo.addWalletUnspentToPsbt(psbt, u, inputBuilder.walletKeys, inputBuilder.signer, inputBuilder.cosigner)\n  );\n  const ordInput = OrdOutput.joinAll(unspents.map((u) => new OrdOutput(u.value)));\n  const ordOutputs = getOrdOutputsForLayout(ordInput, outputLayout);\n  toArray(ordOutputs).forEach((ordOutput) => {\n    if (ordOutput === null) {\n      return;\n    }\n    switch (ordOutput) {\n      // skip padding outputs and fee output (virtual)\n      case null:\n      case ordOutputs.feeOutput:\n        return;\n      // add padding outputs\n      case ordOutputs.firstChangeOutput:\n      case ordOutputs.secondChangeOutput:\n        const { chain, index } =\n          ordOutput === ordOutputs.firstChangeOutput ? outputs.changeOutputs[0] : outputs.changeOutputs[1];\n        bitgo.addWalletOutputToPsbt(psbt, inputBuilder.walletKeys, chain, index, ordOutput.value);\n        break;\n      // add actual inscription output\n      case ordOutputs.inscriptionOutput:\n        let { inscriptionRecipient } = outputs;\n        if (typeof inscriptionRecipient === 'string') {\n          inscriptionRecipient = address.toOutputScript(inscriptionRecipient, network);\n        }\n        psbt.addOutput({\n          script: inscriptionRecipient,\n          value: ordOutput.value,\n        });\n        break;\n    }\n  });\n  return psbt;\n}\n\nfunction toSatRange(p: SatPoint) {\n  const { offset } = parseSatPoint(p);\n  return new SatRange(offset, offset);\n}\n\nfunction getFee(vsize: number, rateSatPerKB: number): bigint {\n  return BigInt(Math.ceil((vsize * rateSatPerKB) / 1000));\n}\n\n/**\n * @param inputs - inscription input must come first\n * @param satPoint - location of the inscription\n * @param outputs\n * @param constraints\n * @param minimizeInputs\n */\nexport function findOutputLayoutForWalletUnspents(\n  inputs: WalletUnspent[],\n  satPoint: SatPoint,\n  outputs: InscriptionTransactionOutputs,\n  constraints: InscriptionTransactionConstraints,\n  { minimizeInputs = false } = {}\n): { inputs: WalletUnspent[]; layout: OutputLayout } | undefined {\n  if (minimizeInputs) {\n    return findSmallestOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints);\n  }\n\n  if (inputs.length === 0) {\n    throw new Error(`must provide at least one input`);\n  }\n\n  if (outputs.changeOutputs[0].chain !== outputs.changeOutputs[1].chain) {\n    // otherwise our fee calc is too complicated\n    throw new Error(`wallet outputs must be on same chain`);\n  }\n\n  const {\n    minChangeOutput = DefaultInscriptionConstraints.minChangeOutput,\n    minInscriptionOutput = DefaultInscriptionConstraints.minInscriptionOutput,\n    maxInscriptionOutput = DefaultInscriptionConstraints.maxInscriptionOutput,\n  } = constraints;\n\n  // Join all the inputs into a single inscriptionOutput.\n  // For the purposes of finding a layout there is no difference.\n  const inscriptionOutput = OrdOutput.joinAll(\n    inputs.map((i) => new OrdOutput(i.value, i === inputs[0] ? [toSatRange(satPoint)] : []))\n  );\n  const layout = findOutputLayout(inscriptionOutput, {\n    minChangeOutput,\n    minInscriptionOutput,\n    maxInscriptionOutput,\n    feeFixed: getFee(\n      VirtualSizes.txSegOverheadVSize +\n        Dimensions.fromUnspents(inputs, {\n          p2tr: { scriptPathLevel: 1 },\n          p2trMusig2: { scriptPathLevel: undefined },\n        }).getInputsVSize(),\n      constraints.feeRateSatKB\n    ),\n    feePerOutput: getFee(\n      Dimensions.fromOutputOnChain(outputs.changeOutputs[0].chain).getOutputsVSize(),\n      constraints.feeRateSatKB\n    ),\n  });\n\n  return layout ? { inputs, layout } : undefined;\n}\n\nexport const MAX_UNSPENTS_FOR_OUTPUT_LAYOUT = 5;\n\n/**\n * @param inputs - inscription input must come first\n * @param satPoint - location of the inscription\n * @param outputs\n * @param constraints\n */\nfunction findSmallestOutputLayoutForWalletUnspents(\n  inputs: WalletUnspent[],\n  satPoint: SatPoint,\n  outputs: InscriptionTransactionOutputs,\n  constraints: InscriptionTransactionConstraints\n): { inputs: WalletUnspent[]; layout: OutputLayout } | undefined {\n  if (MAX_UNSPENTS_FOR_OUTPUT_LAYOUT < inputs.length) {\n    throw new Error(`input array is too large`);\n  }\n  // create powerset of all supplementary inputs and find the cheapest result\n  const inputsArr = [inputs, ...powerset(inputs.slice(1)).map((s) => [inputs[0], ...s])];\n  return inputsArr\n    .map((inputs) => findOutputLayoutForWalletUnspents(inputs, satPoint, outputs, constraints))\n    .reduce((best, next) => {\n      if (best === undefined) {\n        return next;\n      }\n      if (next === undefined) {\n        return best;\n      }\n      return best.layout.feeOutput < next.layout.feeOutput ? best : next;\n    });\n}\n\nexport class ErrorNoLayout extends Error {\n  constructor() {\n    super('Could not find output layout for inscription passing transaction');\n  }\n}\n\n/**\n * @param network\n * @param inputBuilder\n * @param unspent\n * @param satPoint\n * @param outputs\n * @param constraints\n * @param supplementaryUnspents - additional inputs to cover fee.\n * @param [minimizeInputs=true] - try to find input combination with minimal fees. Limits supplementaryUnspents to 4.\n */\nexport function createPsbtForSingleInscriptionPassingTransaction(\n  network: Network,\n  inputBuilder: WalletInputBuilder,\n  unspent: WalletUnspent | WalletUnspent[],\n  satPoint: SatPoint,\n  outputs: InscriptionTransactionOutputs,\n  constraints: InscriptionTransactionConstraints,\n  {\n    supplementaryUnspents = [],\n    minimizeInputs = true,\n  }: {\n    supplementaryUnspents?: WalletUnspent[];\n    minimizeInputs?: boolean;\n  } = {}\n): bitgo.UtxoPsbt {\n  // support for legacy call style\n  if (Array.isArray(unspent)) {\n    if (unspent.length !== 1) {\n      throw new Error(`can only pass single unspent`);\n    }\n    unspent = unspent[0];\n  }\n\n  const result = findOutputLayoutForWalletUnspents(\n    [unspent, ...supplementaryUnspents],\n    satPoint,\n    outputs,\n    constraints,\n    { minimizeInputs }\n  );\n\n  if (!result) {\n    throw new ErrorNoLayout();\n  }\n\n  return createPsbtFromOutputLayout(network, inputBuilder, result.inputs, outputs, result.layout);\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,11 +1,26 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@bitgo-beta/utxo-ord",
|
|
3
3
|
"description": "Utilities for building ordinals with BitGo utxo-lib",
|
|
4
|
-
"main": "./dist/src/index.js",
|
|
5
|
-
"
|
|
6
|
-
"
|
|
4
|
+
"main": "./dist/cjs/src/index.js",
|
|
5
|
+
"module": "./dist/esm/index.js",
|
|
6
|
+
"browser": "./dist/esm/index.js",
|
|
7
|
+
"types": "./dist/cjs/src/index.d.ts",
|
|
8
|
+
"version": "1.1.3-alpha.414",
|
|
9
|
+
"exports": {
|
|
10
|
+
".": {
|
|
11
|
+
"import": {
|
|
12
|
+
"types": "./dist/esm/index.d.ts",
|
|
13
|
+
"default": "./dist/esm/index.js"
|
|
14
|
+
},
|
|
15
|
+
"require": {
|
|
16
|
+
"types": "./dist/cjs/src/index.d.ts",
|
|
17
|
+
"default": "./dist/cjs/src/index.js"
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
},
|
|
7
21
|
"files": [
|
|
8
|
-
"dist
|
|
22
|
+
"dist/cjs",
|
|
23
|
+
"dist/esm"
|
|
9
24
|
],
|
|
10
25
|
"keywords": [
|
|
11
26
|
"bitgo",
|
|
@@ -16,7 +31,9 @@
|
|
|
16
31
|
],
|
|
17
32
|
"scripts": {
|
|
18
33
|
"prepare": "npm run build",
|
|
19
|
-
"build": "
|
|
34
|
+
"build": "npm run build:cjs && npm run build:esm",
|
|
35
|
+
"build:cjs": "yarn tsc --build --incremental --verbose .",
|
|
36
|
+
"build:esm": "yarn tsc --project tsconfig.esm.json",
|
|
20
37
|
"lint": "eslint --quiet .",
|
|
21
38
|
"unit-test": "mocha --recursive test",
|
|
22
39
|
"fmt": "prettier --write '{src,test}/**/*.{ts,js}'",
|
|
@@ -28,9 +45,9 @@
|
|
|
28
45
|
"directory": "modules/utxo-ord"
|
|
29
46
|
},
|
|
30
47
|
"dependencies": {
|
|
31
|
-
"@bitgo-beta/sdk-core": "2.4.1-alpha.
|
|
32
|
-
"@bitgo-beta/unspents": "0.11.3-alpha.
|
|
33
|
-
"@bitgo-beta/utxo-lib": "4.0.1-alpha.
|
|
48
|
+
"@bitgo-beta/sdk-core": "2.4.1-alpha.460",
|
|
49
|
+
"@bitgo-beta/unspents": "0.11.3-alpha.461",
|
|
50
|
+
"@bitgo-beta/utxo-lib": "4.0.1-alpha.461"
|
|
34
51
|
},
|
|
35
52
|
"lint-staged": {
|
|
36
53
|
"*.{js,ts}": [
|
|
@@ -42,5 +59,5 @@
|
|
|
42
59
|
"access": "public"
|
|
43
60
|
},
|
|
44
61
|
"license": "MIT",
|
|
45
|
-
"gitHead": "
|
|
62
|
+
"gitHead": "00bb3d8d26ab1e581287d5d722387f04871166d6"
|
|
46
63
|
}
|
package/dist/src/OrdOutput.js
DELETED
|
@@ -1,197 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
/*
|
|
3
|
-
|
|
4
|
-
Classes used for tracking sats across transactions.
|
|
5
|
-
|
|
6
|
-
https://github.com/casey/ord/blob/master/bip.mediawiki#design
|
|
7
|
-
|
|
8
|
-
> The ordinal numbers of sats in transaction inputs are transferred to output sats in
|
|
9
|
-
> first-in-first-out order, according to the size and order of the transactions inputs and outputs.
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
Sample scenario:
|
|
13
|
-
inputs i0, i1, i2
|
|
14
|
-
outputs u0, u1
|
|
15
|
-
inscriptions r0, r1, r2, r3, r4
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
createOutputs(
|
|
19
|
-
[i0, i1],
|
|
20
|
-
[
|
|
21
|
-
[u0, [r0, r1]],
|
|
22
|
-
[u1, [r2, r3]],
|
|
23
|
-
]
|
|
24
|
-
);
|
|
25
|
-
|
|
26
|
-
r4 is donated to the miner
|
|
27
|
-
|
|
28
|
-
┌────────┬────────┐
|
|
29
|
-
│ i0 │ u0 │
|
|
30
|
-
│ │ │
|
|
31
|
-
│ r0 ┼ │
|
|
32
|
-
│ │ │
|
|
33
|
-
├────────┤ │
|
|
34
|
-
│ i1 │ │
|
|
35
|
-
│ r1 ┼ │
|
|
36
|
-
│ │ │
|
|
37
|
-
│ ├────────┤
|
|
38
|
-
│ │ u1 │
|
|
39
|
-
│ r2 ┼ │
|
|
40
|
-
│ │ │
|
|
41
|
-
├────────┤ │
|
|
42
|
-
│ i2 │ │
|
|
43
|
-
│ r3 ┼ │
|
|
44
|
-
│ │ │
|
|
45
|
-
│ │ │
|
|
46
|
-
│ ├────────┘
|
|
47
|
-
│ │
|
|
48
|
-
│ r4 ┼
|
|
49
|
-
│ │
|
|
50
|
-
└────────┘
|
|
51
|
-
|
|
52
|
-
*/
|
|
53
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
54
|
-
exports.OrdOutput = exports.InvalidOrdOutput = void 0;
|
|
55
|
-
const SatRange_1 = require("./SatRange");
|
|
56
|
-
class InvalidOrdOutput extends Error {
|
|
57
|
-
constructor(message, value, ordinals) {
|
|
58
|
-
super(message);
|
|
59
|
-
this.value = value;
|
|
60
|
-
this.ordinals = ordinals;
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
exports.InvalidOrdOutput = InvalidOrdOutput;
|
|
64
|
-
/**
|
|
65
|
-
* The ordinal metadata for an output
|
|
66
|
-
*/
|
|
67
|
-
class OrdOutput {
|
|
68
|
-
/**
|
|
69
|
-
* @param value - the input value
|
|
70
|
-
* @param ordinals - The ordinal ranges of an output, relative to the first satoshi.
|
|
71
|
-
* Required to be ordered and non-overlapping.
|
|
72
|
-
* Not required to be exhaustive.
|
|
73
|
-
*/
|
|
74
|
-
constructor(value, ordinals = []) {
|
|
75
|
-
this.value = value;
|
|
76
|
-
this.ordinals = ordinals;
|
|
77
|
-
const maxRange = this.asSatRange();
|
|
78
|
-
ordinals.forEach((r, i) => {
|
|
79
|
-
if (!maxRange.isSupersetOf(r)) {
|
|
80
|
-
throw new InvalidOrdOutput(`range ${r} outside output maxRange ${maxRange}`, value, ordinals);
|
|
81
|
-
}
|
|
82
|
-
if (0 < i) {
|
|
83
|
-
const prevRange = ordinals[i - 1];
|
|
84
|
-
if (r.start <= prevRange.end) {
|
|
85
|
-
throw new InvalidOrdOutput(`SatRange #${i - 1} ${prevRange} overlaps SatRange #${i} ${r}`, value, ordinals);
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
});
|
|
89
|
-
}
|
|
90
|
-
/**
|
|
91
|
-
* @param other
|
|
92
|
-
* @return OrdOutput extended by other.value and SatRanges shifted by this.value
|
|
93
|
-
*/
|
|
94
|
-
joinedWith(other) {
|
|
95
|
-
return new OrdOutput(this.value + other.value, [
|
|
96
|
-
...this.ordinals,
|
|
97
|
-
...other.ordinals.map((r) => r.shiftedBy(this.value)),
|
|
98
|
-
]);
|
|
99
|
-
}
|
|
100
|
-
/**
|
|
101
|
-
* @param ords
|
|
102
|
-
* @return single OrdOutput containing all SatRanges, shifted by preceding output values
|
|
103
|
-
*/
|
|
104
|
-
static joinAll(ords) {
|
|
105
|
-
if (ords.length === 0) {
|
|
106
|
-
throw new TypeError(`empty input`);
|
|
107
|
-
}
|
|
108
|
-
return ords.reduce((a, b) => a.joinedWith(b));
|
|
109
|
-
}
|
|
110
|
-
asSatRange() {
|
|
111
|
-
return new SatRange_1.SatRange(BigInt(0), this.value - BigInt(1));
|
|
112
|
-
}
|
|
113
|
-
/**
|
|
114
|
-
* @param r
|
|
115
|
-
* @return new OrdOutput with all ranges fully contained in _r_. SatRanges are aligned to new start.
|
|
116
|
-
*/
|
|
117
|
-
fromSatRange(r) {
|
|
118
|
-
return new OrdOutput(r.size(), this.ordinals.flatMap((s) => {
|
|
119
|
-
if (r.intersectsWith(s)) {
|
|
120
|
-
if (!r.isSupersetOf(s)) {
|
|
121
|
-
throw new Error(`partial overlap in ${r} and ${s}`);
|
|
122
|
-
}
|
|
123
|
-
return s.shiftedBy(-r.start);
|
|
124
|
-
}
|
|
125
|
-
return [];
|
|
126
|
-
}));
|
|
127
|
-
}
|
|
128
|
-
/**
|
|
129
|
-
* @param value
|
|
130
|
-
* @return first OrdOutput with value `value`, second OrdOutput with remaining value.
|
|
131
|
-
* With respective SatRanges
|
|
132
|
-
*/
|
|
133
|
-
splitAt(value) {
|
|
134
|
-
if (this.value < value) {
|
|
135
|
-
throw new Error(`must split at value inside range`);
|
|
136
|
-
}
|
|
137
|
-
return [
|
|
138
|
-
this.fromSatRange(new SatRange_1.SatRange(BigInt(0), value - BigInt(1))),
|
|
139
|
-
this.fromSatRange(new SatRange_1.SatRange(value, this.value - BigInt(1))),
|
|
140
|
-
];
|
|
141
|
-
}
|
|
142
|
-
/**
|
|
143
|
-
* Like splitAt but returns _null_ where a zero-sized OrdOutput would be
|
|
144
|
-
* @param value
|
|
145
|
-
*/
|
|
146
|
-
splitAtAllowZero(value) {
|
|
147
|
-
if (value === BigInt(0)) {
|
|
148
|
-
return [null, this.fromSatRange(this.asSatRange())];
|
|
149
|
-
}
|
|
150
|
-
if (value === this.value) {
|
|
151
|
-
return [this.fromSatRange(this.asSatRange()), null];
|
|
152
|
-
}
|
|
153
|
-
return this.splitAt(value);
|
|
154
|
-
}
|
|
155
|
-
/**
|
|
156
|
-
* Split output successively at values.
|
|
157
|
-
* @param values
|
|
158
|
-
* @param exact - when set, ensure that value sum matches _this.value_
|
|
159
|
-
* @param allowZero - when set, return _null_ for zero-sized values
|
|
160
|
-
* @return (OrdOutput | null)[]. Zero-sized outputs are substituted with _null_.
|
|
161
|
-
*/
|
|
162
|
-
splitAllWithParams(values, { exact = false, allowZero = false }) {
|
|
163
|
-
if (values.length === 0) {
|
|
164
|
-
throw new Error(`invalid argument`);
|
|
165
|
-
}
|
|
166
|
-
if (exact) {
|
|
167
|
-
const valueSum = values.reduce((a, b) => a + b, BigInt(0));
|
|
168
|
-
if (this.value !== valueSum) {
|
|
169
|
-
throw new Error(`value sum ${valueSum} does not match this.value ${this.value}`);
|
|
170
|
-
}
|
|
171
|
-
return this.splitAllWithParams(values.slice(0, -1), { allowZero, exact: false });
|
|
172
|
-
}
|
|
173
|
-
const [v, ...rest] = values;
|
|
174
|
-
const [a, b] = allowZero ? this.splitAtAllowZero(v) : this.splitAt(v);
|
|
175
|
-
if (rest.length) {
|
|
176
|
-
if (b === null) {
|
|
177
|
-
throw new Error(`invalid remainder`);
|
|
178
|
-
}
|
|
179
|
-
else {
|
|
180
|
-
return [a, ...b.splitAllWithParams(rest, { exact, allowZero })];
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
else {
|
|
184
|
-
return [a, b];
|
|
185
|
-
}
|
|
186
|
-
}
|
|
187
|
-
/**
|
|
188
|
-
* Split output successively at values.
|
|
189
|
-
* @param values
|
|
190
|
-
* @return OrdOutput[] with length _values.length + 1_
|
|
191
|
-
*/
|
|
192
|
-
splitAll(values) {
|
|
193
|
-
return this.splitAllWithParams(values, { exact: false, allowZero: false });
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
exports.OrdOutput = OrdOutput;
|
|
197
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"OrdOutput.js","sourceRoot":"","sources":["../../src/OrdOutput.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n\nClasses used for tracking sats across transactions.\n\nhttps://github.com/casey/ord/blob/master/bip.mediawiki#design\n\n> The ordinal numbers of sats in transaction inputs are transferred to output sats in\n> first-in-first-out order, according to the size and order of the transactions inputs and outputs.\n\n\nSample scenario:\n   inputs         i0, i1, i2\n   outputs        u0, u1\n   inscriptions   r0, r1, r2, r3, r4\n\n\ncreateOutputs(\n  [i0, i1],\n  [\n    [u0, [r0, r1]],\n    [u1, [r2, r3]],\n  ]\n);\n\n  r4 is donated to the miner\n\n  ┌────────┬────────┐\n  │ i0     │ u0     │\n  │        │        │\n  │     r0 ┼        │\n  │        │        │\n  ├────────┤        │\n  │ i1     │        │\n  │     r1 ┼        │\n  │        │        │\n  │        ├────────┤\n  │        │ u1     │\n  │     r2 ┼        │\n  │        │        │\n  ├────────┤        │\n  │ i2     │        │\n  │     r3 ┼        │\n  │        │        │\n  │        │        │\n  │        ├────────┘\n  │        │\n  │     r4 ┼\n  │        │\n  └────────┘\n\n */\n\nimport { SatRange } from './SatRange';\n\nexport class InvalidOrdOutput extends Error {\n  constructor(message: string, public value: bigint, public ordinals: SatRange[]) {\n    super(message);\n  }\n}\n\n/**\n * The ordinal metadata for an output\n */\nexport class OrdOutput {\n  /**\n   * @param value - the input value\n   * @param ordinals - The ordinal ranges of an output, relative to the first satoshi.\n   *                   Required to be ordered and non-overlapping.\n   *                   Not required to be exhaustive.\n   */\n  constructor(public value: bigint, public ordinals: SatRange[] = []) {\n    const maxRange = this.asSatRange();\n    ordinals.forEach((r, i) => {\n      if (!maxRange.isSupersetOf(r)) {\n        throw new InvalidOrdOutput(`range ${r} outside output maxRange ${maxRange}`, value, ordinals);\n      }\n      if (0 < i) {\n        const prevRange = ordinals[i - 1];\n        if (r.start <= prevRange.end) {\n          throw new InvalidOrdOutput(`SatRange #${i - 1} ${prevRange} overlaps SatRange #${i} ${r}`, value, ordinals);\n        }\n      }\n    });\n  }\n\n  /**\n   * @param other\n   * @return OrdOutput extended by other.value and SatRanges shifted by this.value\n   */\n  joinedWith(other: OrdOutput): OrdOutput {\n    return new OrdOutput(this.value + other.value, [\n      ...this.ordinals,\n      ...other.ordinals.map((r) => r.shiftedBy(this.value)),\n    ]);\n  }\n\n  /**\n   * @param ords\n   * @return single OrdOutput containing all SatRanges, shifted by preceding output values\n   */\n  static joinAll(ords: OrdOutput[]): OrdOutput {\n    if (ords.length === 0) {\n      throw new TypeError(`empty input`);\n    }\n    return ords.reduce((a, b) => a.joinedWith(b));\n  }\n\n  asSatRange(): SatRange {\n    return new SatRange(BigInt(0), this.value - BigInt(1));\n  }\n\n  /**\n   * @param r\n   * @return new OrdOutput with all ranges fully contained in _r_. SatRanges are aligned to new start.\n   */\n  fromSatRange(r: SatRange): OrdOutput {\n    return new OrdOutput(\n      r.size(),\n      this.ordinals.flatMap((s) => {\n        if (r.intersectsWith(s)) {\n          if (!r.isSupersetOf(s)) {\n            throw new Error(`partial overlap in ${r} and ${s}`);\n          }\n          return s.shiftedBy(-r.start);\n        }\n        return [];\n      })\n    );\n  }\n\n  /**\n   * @param value\n   * @return first OrdOutput with value `value`, second OrdOutput with remaining value.\n   *         With respective SatRanges\n   */\n  splitAt(value: bigint): [OrdOutput, OrdOutput] {\n    if (this.value < value) {\n      throw new Error(`must split at value inside range`);\n    }\n    return [\n      this.fromSatRange(new SatRange(BigInt(0), value - BigInt(1))),\n      this.fromSatRange(new SatRange(value, this.value - BigInt(1))),\n    ];\n  }\n\n  /**\n   * Like splitAt but returns _null_ where a zero-sized OrdOutput would be\n   * @param value\n   */\n  splitAtAllowZero(value: bigint): [OrdOutput | null, OrdOutput | null] {\n    if (value === BigInt(0)) {\n      return [null, this.fromSatRange(this.asSatRange())];\n    }\n    if (value === this.value) {\n      return [this.fromSatRange(this.asSatRange()), null];\n    }\n    return this.splitAt(value);\n  }\n\n  /**\n   * Split output successively at values.\n   * @param values\n   * @param exact - when set, ensure that value sum matches _this.value_\n   * @param allowZero - when set, return _null_ for zero-sized values\n   * @return (OrdOutput | null)[]. Zero-sized outputs are substituted with _null_.\n   */\n  splitAllWithParams(\n    values: bigint[],\n    { exact = false, allowZero = false }: { allowZero?: boolean; exact?: boolean }\n  ): (OrdOutput | null)[] {\n    if (values.length === 0) {\n      throw new Error(`invalid argument`);\n    }\n    if (exact) {\n      const valueSum = values.reduce((a, b) => a + b, BigInt(0));\n      if (this.value !== valueSum) {\n        throw new Error(`value sum ${valueSum} does not match this.value ${this.value}`);\n      }\n      return this.splitAllWithParams(values.slice(0, -1), { allowZero, exact: false });\n    }\n    const [v, ...rest] = values;\n    const [a, b] = allowZero ? this.splitAtAllowZero(v) : this.splitAt(v);\n    if (rest.length) {\n      if (b === null) {\n        throw new Error(`invalid remainder`);\n      } else {\n        return [a, ...b.splitAllWithParams(rest, { exact, allowZero })];\n      }\n    } else {\n      return [a, b];\n    }\n  }\n\n  /**\n   * Split output successively at values.\n   * @param values\n   * @return OrdOutput[] with length _values.length + 1_\n   */\n  splitAll(values: bigint[]): OrdOutput[] {\n    return this.splitAllWithParams(values, { exact: false, allowZero: false }) as OrdOutput[];\n  }\n}\n"]}
|