@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.
Files changed (57) hide show
  1. package/dist/cjs/src/OrdOutput.d.ts.map +1 -0
  2. package/dist/cjs/src/OrdOutput.js +197 -0
  3. package/dist/cjs/src/OutputLayout.d.ts.map +1 -0
  4. package/dist/cjs/src/OutputLayout.js +246 -0
  5. package/dist/cjs/src/SatPoint.d.ts.map +1 -0
  6. package/dist/cjs/src/SatPoint.js +48 -0
  7. package/dist/cjs/src/SatRange.d.ts.map +1 -0
  8. package/dist/cjs/src/SatRange.js +58 -0
  9. package/dist/cjs/src/combinations.d.ts.map +1 -0
  10. package/dist/cjs/src/combinations.js +13 -0
  11. package/dist/cjs/src/index.d.ts.map +1 -0
  12. package/dist/{src → cjs/src}/index.js +1 -1
  13. package/dist/cjs/src/inscriptions.d.ts.map +1 -0
  14. package/dist/cjs/src/inscriptions.js +202 -0
  15. package/dist/cjs/src/psbt.d.ts.map +1 -0
  16. package/dist/cjs/src/psbt.js +154 -0
  17. package/dist/cjs/tsconfig.tsbuildinfo +1 -0
  18. package/dist/esm/OrdOutput.d.ts +65 -0
  19. package/dist/esm/OrdOutput.js +192 -0
  20. package/dist/esm/OutputLayout.d.ts +47 -0
  21. package/dist/esm/OutputLayout.js +240 -0
  22. package/dist/esm/SatPoint.d.ts +13 -0
  23. package/dist/esm/SatPoint.js +43 -0
  24. package/dist/esm/SatRange.d.ts +26 -0
  25. package/dist/esm/SatRange.js +53 -0
  26. package/dist/esm/combinations.d.ts +2 -0
  27. package/dist/esm/combinations.js +10 -0
  28. package/dist/esm/index.d.ts +10 -0
  29. package/dist/esm/index.js +10 -0
  30. package/dist/esm/inscriptions.d.ts +31 -0
  31. package/dist/esm/inscriptions.js +164 -0
  32. package/dist/esm/psbt.d.ts +67 -0
  33. package/dist/{src → esm}/psbt.js +31 -38
  34. package/package.json +26 -9
  35. package/dist/src/OrdOutput.js +0 -197
  36. package/dist/src/OutputLayout.js +0 -246
  37. package/dist/src/SatPoint.js +0 -48
  38. package/dist/src/SatRange.js +0 -58
  39. package/dist/src/combinations.js +0 -13
  40. package/dist/src/inscriptions.js +0 -202
  41. package/dist/tsconfig.tsbuildinfo +0 -1
  42. /package/dist/{src → cjs/src}/OrdOutput.d.ts +0 -0
  43. /package/dist/{src → cjs/src}/OutputLayout.d.ts +0 -0
  44. /package/dist/{src → cjs/src}/SatPoint.d.ts +0 -0
  45. /package/dist/{src → cjs/src}/SatRange.d.ts +0 -0
  46. /package/dist/{src → cjs/src}/combinations.d.ts +0 -0
  47. /package/dist/{src → cjs/src}/index.d.ts +0 -0
  48. /package/dist/{src → cjs/src}/inscriptions.d.ts +0 -0
  49. /package/dist/{src → cjs/src}/psbt.d.ts +0 -0
  50. /package/dist/{src → esm}/OrdOutput.d.ts.map +0 -0
  51. /package/dist/{src → esm}/OutputLayout.d.ts.map +0 -0
  52. /package/dist/{src → esm}/SatPoint.d.ts.map +0 -0
  53. /package/dist/{src → esm}/SatRange.d.ts.map +0 -0
  54. /package/dist/{src → esm}/combinations.d.ts.map +0 -0
  55. /package/dist/{src → esm}/index.d.ts.map +0 -0
  56. /package/dist/{src → esm}/inscriptions.d.ts.map +0 -0
  57. /package/dist/{src → esm}/psbt.d.ts.map +0 -0
@@ -1,246 +0,0 @@
1
- "use strict";
2
- /*
3
-
4
- This file contains code for creating an output layouts for transactions that pass on inscriptions.
5
-
6
- When passing on an inscription, we want to satisfy a few constraints:
7
-
8
- * All outputs should be larger than a minimal value (dust limit)
9
- * The sum of all output values needs to be less than the input to cover the transaction fee.
10
- * The output containing the inscription should be as small as possible, but large enough to
11
- contain the inscription.
12
-
13
- To keep the inscription output small, we can pad the satoshi range preceding and following the range
14
- with change outputs, which have a minimal size and incur a fee cost.
15
-
16
-
17
- Broadly speaking, there are four scenarios:
18
-
19
- (1) Small inscription input that has just enough value to pay for fee and a single inscription
20
- output (u0). No padding outputs.
21
- ┌────────┬────────┐
22
- │ │ u0 │
23
- │ r ┼ │
24
- │ ├────────┘
25
- │ │ fee
26
- └────────┘
27
-
28
-
29
- (2) Large inscription input with inscription close to start of input.
30
- Inscription output followed by change output (u1) padding the remaining value.
31
-
32
- ┌────────┬────────┐
33
- │ │ u0 │
34
- │ r ┼ │
35
- │ ├────────┤
36
- │ │ u1 │
37
- │ │ │
38
- │ │ │
39
- │ │ │
40
- │ │ │
41
- │ │ │
42
- │ │ │
43
- │ │ │
44
- │ │ │
45
- │ │ │
46
- │ ├────────┘
47
- │ │
48
- │ │ fee
49
- │ │
50
- └────────┘
51
-
52
-
53
- (3) Large inscription input with inscription close to end of input.
54
- Change output padding start followed by inscription output.
55
-
56
- ┌────────┬────────┐
57
- │ │ u0 │
58
- │ │ │
59
- │ │ │
60
- │ │ │
61
- │ │ │
62
- │ │ │
63
- │ │ │
64
- │ │ │
65
- │ │ │
66
- │ ├────────┤
67
- │ r ┼ │
68
- │ │ u1 │
69
- │ ├────────┘
70
- │ │
71
- │ │ fee
72
- │ │
73
- └────────┘
74
-
75
-
76
- (4) Large inscription input with inscription in the middle.
77
- Inscription input (u1) with padding on both sides (u0 and u2)
78
-
79
- ┌────────┬────────┐
80
- │ │ u0 │
81
- │ │ │
82
- │ │ │
83
- │ │ │
84
- │ │ │
85
- │ ├────────┤
86
- │ │ u1 │
87
- │ r ┼ │
88
- │ ├────────┤
89
- │ │ u2 │
90
- │ │ │
91
- │ │ │
92
- │ │ │
93
- │ ├────────┘
94
- │ │
95
- │ │ fee
96
- │ │
97
- └────────┘
98
- */
99
- Object.defineProperty(exports, "__esModule", { value: true });
100
- exports.toArray = toArray;
101
- exports.toParameters = toParameters;
102
- exports.getOrdOutputsForLayout = getOrdOutputsForLayout;
103
- exports.findOutputLayout = findOutputLayout;
104
- const ZERO = BigInt(0);
105
- const ONE = BigInt(1);
106
- function max(a, b) {
107
- return a < b ? b : a;
108
- }
109
- function min(a, b) {
110
- return a < b ? a : b;
111
- }
112
- function sum(arr) {
113
- return arr.reduce((a, b) => a + b, ZERO);
114
- }
115
- /** @return canonical sequence of parameters */
116
- function toArray(p) {
117
- return [p.firstChangeOutput, p.inscriptionOutput, p.secondChangeOutput, p.feeOutput];
118
- }
119
- function toParameters(firstChangeOutput, inscriptionOutput, secondChangeOutput, feeOutput) {
120
- return {
121
- firstChangeOutput,
122
- inscriptionOutput,
123
- secondChangeOutput,
124
- feeOutput,
125
- };
126
- }
127
- /**
128
- * Translates a layout into OrdOutputs. Absent outputs are set to `null`.
129
- *
130
- * @param inscriptionInput
131
- * @param layout
132
- * @return OrdOutputs for layout
133
- */
134
- function getOrdOutputsForLayout(inscriptionInput, layout) {
135
- const outputs = inscriptionInput.splitAllWithParams(toArray(layout), { exact: true, allowZero: true });
136
- if (outputs.length !== 4) {
137
- throw new Error(`unexpected result`);
138
- }
139
- return toParameters(...outputs);
140
- }
141
- /**
142
- * @param constraints
143
- * @param inscriptionInput
144
- * @param layout
145
- * @return true iff layout satisfies constraints
146
- */
147
- function check(constraints, inscriptionInput, layout) {
148
- if ((layout.firstChangeOutput === ZERO || constraints.minChangeOutput <= layout.firstChangeOutput) &&
149
- (layout.secondChangeOutput === ZERO || constraints.minChangeOutput <= layout.secondChangeOutput) &&
150
- constraints.minInscriptionOutput <= layout.inscriptionOutput &&
151
- layout.inscriptionOutput <= constraints.maxInscriptionOutput &&
152
- getFeeForOutputs(constraints, [layout.firstChangeOutput, layout.inscriptionOutput, layout.secondChangeOutput]) <=
153
- layout.feeOutput &&
154
- sum(toArray(layout)) === inscriptionInput.value) {
155
- /* make sure inscription actually lies on the inscriptionOutput */
156
- const outputs = getOrdOutputsForLayout(inscriptionInput, layout);
157
- return outputs.inscriptionOutput?.ordinals.length === 1;
158
- }
159
- return false;
160
- }
161
- function getFeeForOutputs(p, outputs) {
162
- return outputs.reduce((sum, oValue) => sum + (oValue === ZERO ? ZERO : p.feePerOutput), p.feeFixed);
163
- }
164
- function getStartChangeOutput(c) {
165
- // we don't need a change padding output
166
- if (c.satPos < c.maxInscriptionOutput) {
167
- return ZERO;
168
- }
169
- if (c.minChangeOutput <= c.satPos) {
170
- return c.satPos;
171
- }
172
- return null;
173
- }
174
- function getInscriptionOutput(c, startChangeOutput) {
175
- const result = min(c.maxInscriptionOutput, max(c.minInscriptionOutput, c.satPos - startChangeOutput + ONE));
176
- if (c.satPos < startChangeOutput || startChangeOutput + result < c.satPos) {
177
- return null;
178
- }
179
- // if is not worth creating an end change output, let's maximize the inscription output
180
- if (getEndChangeOutput(c, startChangeOutput, result) === ZERO) {
181
- const remainder = c.total - startChangeOutput - getFeeForOutputs(c, [startChangeOutput, result]);
182
- return min(remainder, c.maxInscriptionOutput);
183
- }
184
- return result;
185
- }
186
- function getEndChangeOutput(c, startChangeOutput, inscriptionOutput) {
187
- const remainder = c.total - sum([startChangeOutput, inscriptionOutput]);
188
- const minFeeWithoutSecondOutput = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput]);
189
- const minFeeWithSecondOutput = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput, c.minChangeOutput]);
190
- if (remainder < minFeeWithoutSecondOutput) {
191
- // We cannot even pay the fee for the output(s) we have so far.
192
- return null;
193
- }
194
- if (remainder - minFeeWithSecondOutput < c.minChangeOutput) {
195
- // The remainder is too small to pay for the end change output. Let's skip it and pay a higher fee.
196
- return ZERO;
197
- }
198
- // let's use as much as we can for fee while leaving enough for fee
199
- return remainder - minFeeWithSecondOutput;
200
- }
201
- function getFeeOutput(c, startChangeOutput, inscriptionOutput, endChangeOutput) {
202
- const minFee = getFeeForOutputs(c, [startChangeOutput, inscriptionOutput, endChangeOutput]);
203
- const remainder = c.total - sum([startChangeOutput, inscriptionOutput, endChangeOutput]);
204
- if (remainder < minFee) {
205
- return null;
206
- }
207
- return remainder;
208
- }
209
- /**
210
- * @param inscriptionInput
211
- * @param search
212
- * @return a solution that satisfies constraints. If no solution can be found, return `undefined`.
213
- */
214
- function findOutputLayout(inscriptionInput, search) {
215
- if (inscriptionInput.ordinals.length !== 1) {
216
- throw new Error(`unexpected ordinal count ${inscriptionInput.ordinals.length}`);
217
- }
218
- if (inscriptionInput.ordinals[0].size() !== ONE) {
219
- throw new Error(`only single-satoshi inscriptions are supported`);
220
- }
221
- const satPos = inscriptionInput.ordinals[0].start;
222
- const total = inscriptionInput.value;
223
- const constraints = { ...search, satPos, total };
224
- const startChangeOutput = getStartChangeOutput(constraints);
225
- if (startChangeOutput === null) {
226
- return;
227
- }
228
- const inscriptionOutput = getInscriptionOutput(constraints, startChangeOutput);
229
- if (inscriptionOutput === null) {
230
- return;
231
- }
232
- const endChangeOutput = getEndChangeOutput(constraints, startChangeOutput, inscriptionOutput);
233
- if (endChangeOutput === null) {
234
- return;
235
- }
236
- const feeOutput = getFeeOutput(constraints, startChangeOutput, inscriptionOutput, endChangeOutput);
237
- if (feeOutput === null) {
238
- return;
239
- }
240
- const result = toParameters(startChangeOutput, inscriptionOutput, endChangeOutput, feeOutput);
241
- if (!check(constraints, inscriptionInput, result)) {
242
- throw new Error(`invalid result`);
243
- }
244
- return result;
245
- }
246
- //# 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"]}
@@ -1,48 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.parseSatPoint = parseSatPoint;
4
- exports.formatSatPoint = formatSatPoint;
5
- exports.isSatPoint = isSatPoint;
6
- /*
7
-
8
- https://github.com/casey/ord/blob/master/bip.mediawiki#terminology-and-notation
9
-
10
- > A satpoint may be used to indicate the location of a sat within an output.
11
- > A satpoint consists of an outpoint, i.e., a transaction ID and output index, with the addition of
12
- > the offset of the ordinal within that output. For example, if the sat in question is at offset 6
13
- > in the first output of a transaction, its satpoint is:
14
- > `680df1e4d43016571e504b0b142ee43c5c0b83398a97bdcfd94ea6f287322d22:0:6`
15
-
16
- */
17
- const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
18
- function parseSatPoint(p) {
19
- const parts = p.split(':');
20
- if (parts.length !== 3) {
21
- throw new Error(`expected format txid:vout:sat`);
22
- }
23
- const [txid, vout, offsetStr] = parts;
24
- const offset = BigInt(offsetStr);
25
- if (offset.toString() !== offsetStr) {
26
- throw new Error(`SatPoint offset must be base-10`);
27
- }
28
- if (offset < 0) {
29
- throw new Error(`SatPoint offset must be positive`);
30
- }
31
- return {
32
- ...utxo_lib_1.bitgo.parseOutputId([txid, vout].join(':')),
33
- offset,
34
- };
35
- }
36
- function formatSatPoint(p) {
37
- return `${p.txid}:${p.vout}:${p.offset}`;
38
- }
39
- function isSatPoint(v) {
40
- try {
41
- parseSatPoint(v);
42
- return true;
43
- }
44
- catch {
45
- return false;
46
- }
47
- }
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2F0UG9pbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvU2F0UG9pbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFlQSxzQ0FpQkM7QUFFRCx3Q0FFQztBQUVELGdDQU9DO0FBN0NEOzs7Ozs7Ozs7O0VBVUU7QUFDRixtREFBNkM7QUFJN0MsU0FBZ0IsYUFBYSxDQUFDLENBQVc7SUFDdkMsTUFBTSxLQUFLLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUMzQixJQUFJLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7UUFDdkIsTUFBTSxJQUFJLEtBQUssQ0FBQywrQkFBK0IsQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFDRCxNQUFNLENBQUMsSUFBSSxFQUFFLElBQUksRUFBRSxTQUFTLENBQUMsR0FBRyxLQUFLLENBQUM7SUFDdEMsTUFBTSxNQUFNLEdBQUcsTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ2pDLElBQUksTUFBTSxDQUFDLFFBQVEsRUFBRSxLQUFLLFNBQVMsRUFBRSxDQUFDO1FBQ3BDLE1BQU0sSUFBSSxLQUFLLENBQUMsaUNBQWlDLENBQUMsQ0FBQztJQUNyRCxDQUFDO0lBQ0QsSUFBSSxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDZixNQUFNLElBQUksS0FBSyxDQUFDLGtDQUFrQyxDQUFDLENBQUM7SUFDdEQsQ0FBQztJQUNELE9BQU87UUFDTCxHQUFHLGdCQUFLLENBQUMsYUFBYSxDQUFDLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM5QyxNQUFNO0tBQ1AsQ0FBQztBQUNKLENBQUM7QUFFRCxTQUFnQixjQUFjLENBQUMsQ0FBaUQ7SUFDOUUsT0FBTyxHQUFHLENBQUMsQ0FBQyxJQUFJLElBQUksQ0FBQyxDQUFDLElBQUksSUFBSSxDQUFDLENBQUMsTUFBTSxFQUFXLENBQUM7QUFDcEQsQ0FBQztBQUVELFNBQWdCLFVBQVUsQ0FBQyxDQUFTO0lBQ2xDLElBQUksQ0FBQztRQUNILGFBQWEsQ0FBQyxDQUFhLENBQUMsQ0FBQztRQUM3QixPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7SUFBQyxNQUFNLENBQUM7UUFDUCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLypcblxuaHR0cHM6Ly9naXRodWIuY29tL2Nhc2V5L29yZC9ibG9iL21hc3Rlci9iaXAubWVkaWF3aWtpI3Rlcm1pbm9sb2d5LWFuZC1ub3RhdGlvblxuXG4+IEEgc2F0cG9pbnQgbWF5IGJlIHVzZWQgdG8gaW5kaWNhdGUgdGhlIGxvY2F0aW9uIG9mIGEgc2F0IHdpdGhpbiBhbiBvdXRwdXQuXG4+IEEgc2F0cG9pbnQgY29uc2lzdHMgb2YgYW4gb3V0cG9pbnQsIGkuZS4sIGEgdHJhbnNhY3Rpb24gSUQgYW5kIG91dHB1dCBpbmRleCwgd2l0aCB0aGUgYWRkaXRpb24gb2Zcbj4gdGhlIG9mZnNldCBvZiB0aGUgb3JkaW5hbCB3aXRoaW4gdGhhdCBvdXRwdXQuIEZvciBleGFtcGxlLCBpZiB0aGUgc2F0IGluIHF1ZXN0aW9uIGlzIGF0IG9mZnNldCA2XG4+IGluIHRoZSBmaXJzdCBvdXRwdXQgb2YgYSB0cmFuc2FjdGlvbiwgaXRzIHNhdHBvaW50IGlzOlxuPiBgNjgwZGYxZTRkNDMwMTY1NzFlNTA0YjBiMTQyZWU0M2M1YzBiODMzOThhOTdiZGNmZDk0ZWE2ZjI4NzMyMmQyMjowOjZgXG5cbiovXG5pbXBvcnQgeyBiaXRnbyB9IGZyb20gJ0BiaXRnby1iZXRhL3V0eG8tbGliJztcblxuZXhwb3J0IHR5cGUgU2F0UG9pbnQgPSBgJHtzdHJpbmd9OiR7bnVtYmVyfToke2JpZ2ludH1gO1xuXG5leHBvcnQgZnVuY3Rpb24gcGFyc2VTYXRQb2ludChwOiBTYXRQb2ludCk6IHsgdHhpZDogc3RyaW5nOyB2b3V0OiBudW1iZXI7IG9mZnNldDogYmlnaW50IH0ge1xuICBjb25zdCBwYXJ0cyA9IHAuc3BsaXQoJzonKTtcbiAgaWYgKHBhcnRzLmxlbmd0aCAhPT0gMykge1xuICAgIHRocm93IG5ldyBFcnJvcihgZXhwZWN0ZWQgZm9ybWF0IHR4aWQ6dm91dDpzYXRgKTtcbiAgfVxuICBjb25zdCBbdHhpZCwgdm91dCwgb2Zmc2V0U3RyXSA9IHBhcnRzO1xuICBjb25zdCBvZmZzZXQgPSBCaWdJbnQob2Zmc2V0U3RyKTtcbiAgaWYgKG9mZnNldC50b1N0cmluZygpICE9PSBvZmZzZXRTdHIpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoYFNhdFBvaW50IG9mZnNldCBtdXN0IGJlIGJhc2UtMTBgKTtcbiAgfVxuICBpZiAob2Zmc2V0IDwgMCkge1xuICAgIHRocm93IG5ldyBFcnJvcihgU2F0UG9pbnQgb2Zmc2V0IG11c3QgYmUgcG9zaXRpdmVgKTtcbiAgfVxuICByZXR1cm4ge1xuICAgIC4uLmJpdGdvLnBhcnNlT3V0cHV0SWQoW3R4aWQsIHZvdXRdLmpvaW4oJzonKSksXG4gICAgb2Zmc2V0LFxuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZm9ybWF0U2F0UG9pbnQocDogeyB0eGlkOiBzdHJpbmc7IHZvdXQ6IG51bWJlcjsgb2Zmc2V0OiBiaWdpbnQgfSk6IFNhdFBvaW50IHtcbiAgcmV0dXJuIGAke3AudHhpZH06JHtwLnZvdXR9OiR7cC5vZmZzZXR9YCBhcyBjb25zdDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGlzU2F0UG9pbnQodjogc3RyaW5nKTogdiBpcyBTYXRQb2ludCB7XG4gIHRyeSB7XG4gICAgcGFyc2VTYXRQb2ludCh2IGFzIFNhdFBvaW50KTtcbiAgICByZXR1cm4gdHJ1ZTtcbiAgfSBjYXRjaCB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG59XG4iXX0=
@@ -1,58 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.SatRange = exports.InvalidSatRange = void 0;
4
- class InvalidSatRange extends Error {
5
- constructor(message, start, end) {
6
- super(message);
7
- this.start = start;
8
- this.end = end;
9
- }
10
- }
11
- exports.InvalidSatRange = InvalidSatRange;
12
- function toRange(v) {
13
- return typeof v === 'bigint' ? new SatRange(v, v) : v;
14
- }
15
- /**
16
- * Range of satoshi, inclusive.
17
- * Inscriptions have start === end.
18
- */
19
- class SatRange {
20
- constructor(start, end) {
21
- this.start = start;
22
- this.end = end;
23
- if (start < 0 || end < 0 || end < start) {
24
- throw new InvalidSatRange(`Invalid SatRange [${start}, ${end}]`, start, end);
25
- }
26
- }
27
- /**
28
- * @param offset
29
- * @return SatRange with start and end shifted by offset
30
- */
31
- shiftedBy(offset) {
32
- return new SatRange(this.start + offset, this.end + offset);
33
- }
34
- /** @return true iff this intersects with _other_ */
35
- intersectsWith(other) {
36
- other = toRange(other);
37
- if (this.start <= other.start) {
38
- return other.start <= this.end;
39
- }
40
- return other.intersectsWith(this);
41
- }
42
- /** @return true iff this is superset of _other_. */
43
- isSupersetOf(other) {
44
- other = toRange(other);
45
- return this.start <= other.start && other.end <= this.end;
46
- }
47
- toString() {
48
- if (this.start === this.end) {
49
- return `[${this.start}]`;
50
- }
51
- return `[${this.start}..${this.end}]`;
52
- }
53
- size() {
54
- return this.end - this.start + BigInt(1);
55
- }
56
- }
57
- exports.SatRange = SatRange;
58
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2F0UmFuZ2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvU2F0UmFuZ2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsTUFBYSxlQUFnQixTQUFRLEtBQUs7SUFDeEMsWUFBWSxPQUFlLEVBQVMsS0FBYSxFQUFTLEdBQVc7UUFDbkUsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1FBRG1CLFVBQUssR0FBTCxLQUFLLENBQVE7UUFBUyxRQUFHLEdBQUgsR0FBRyxDQUFRO0lBRXJFLENBQUM7Q0FDRjtBQUpELDBDQUlDO0FBRUQsU0FBUyxPQUFPLENBQUMsQ0FBb0I7SUFDbkMsT0FBTyxPQUFPLENBQUMsS0FBSyxRQUFRLENBQUMsQ0FBQyxDQUFDLElBQUksUUFBUSxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0FBQ3hELENBQUM7QUFFRDs7O0dBR0c7QUFDSCxNQUFhLFFBQVE7SUFDbkIsWUFBbUIsS0FBYSxFQUFTLEdBQVc7UUFBakMsVUFBSyxHQUFMLEtBQUssQ0FBUTtRQUFTLFFBQUcsR0FBSCxHQUFHLENBQVE7UUFDbEQsSUFBSSxLQUFLLEdBQUcsQ0FBQyxJQUFJLEdBQUcsR0FBRyxDQUFDLElBQUksR0FBRyxHQUFHLEtBQUssRUFBRSxDQUFDO1lBQ3hDLE1BQU0sSUFBSSxlQUFlLENBQUMscUJBQXFCLEtBQUssS0FBSyxHQUFHLEdBQUcsRUFBRSxLQUFLLEVBQUUsR0FBRyxDQUFDLENBQUM7UUFDL0UsQ0FBQztJQUNILENBQUM7SUFFRDs7O09BR0c7SUFDSCxTQUFTLENBQUMsTUFBYztRQUN0QixPQUFPLElBQUksUUFBUSxDQUFDLElBQUksQ0FBQyxLQUFLLEdBQUcsTUFBTSxFQUFFLElBQUksQ0FBQyxHQUFHLEdBQUcsTUFBTSxDQUFDLENBQUM7SUFDOUQsQ0FBQztJQUVELG9EQUFvRDtJQUNwRCxjQUFjLENBQUMsS0FBd0I7UUFDckMsS0FBSyxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUN2QixJQUFJLElBQUksQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQzlCLE9BQU8sS0FBSyxDQUFDLEtBQUssSUFBSSxJQUFJLENBQUMsR0FBRyxDQUFDO1FBQ2pDLENBQUM7UUFDRCxPQUFPLEtBQUssQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDcEMsQ0FBQztJQUVELG9EQUFvRDtJQUNwRCxZQUFZLENBQUMsS0FBd0I7UUFDbkMsS0FBSyxHQUFHLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUN2QixPQUFPLElBQUksQ0FBQyxLQUFLLElBQUksS0FBSyxDQUFDLEtBQUssSUFBSSxLQUFLLENBQUMsR0FBRyxJQUFJLElBQUksQ0FBQyxHQUFHLENBQUM7SUFDNUQsQ0FBQztJQUVELFFBQVE7UUFDTixJQUFJLElBQUksQ0FBQyxLQUFLLEtBQUssSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDO1lBQzVCLE9BQU8sSUFBSSxJQUFJLENBQUMsS0FBSyxHQUFHLENBQUM7UUFDM0IsQ0FBQztRQUNELE9BQU8sSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLElBQUksQ0FBQyxHQUFHLEdBQUcsQ0FBQztJQUN4QyxDQUFDO0lBRUQsSUFBSTtRQUNGLE9BQU8sSUFBSSxDQUFDLEdBQUcsR0FBRyxJQUFJLENBQUMsS0FBSyxHQUFHLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUMzQyxDQUFDO0NBQ0Y7QUF4Q0QsNEJBd0NDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNsYXNzIEludmFsaWRTYXRSYW5nZSBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IobWVzc2FnZTogc3RyaW5nLCBwdWJsaWMgc3RhcnQ6IGJpZ2ludCwgcHVibGljIGVuZDogYmlnaW50KSB7XG4gICAgc3VwZXIobWVzc2FnZSk7XG4gIH1cbn1cblxuZnVuY3Rpb24gdG9SYW5nZSh2OiBiaWdpbnQgfCBTYXRSYW5nZSk6IFNhdFJhbmdlIHtcbiAgcmV0dXJuIHR5cGVvZiB2ID09PSAnYmlnaW50JyA/IG5ldyBTYXRSYW5nZSh2LCB2KSA6IHY7XG59XG5cbi8qKlxuICogUmFuZ2Ugb2Ygc2F0b3NoaSwgaW5jbHVzaXZlLlxuICogSW5zY3JpcHRpb25zIGhhdmUgc3RhcnQgPT09IGVuZC5cbiAqL1xuZXhwb3J0IGNsYXNzIFNhdFJhbmdlIHtcbiAgY29uc3RydWN0b3IocHVibGljIHN0YXJ0OiBiaWdpbnQsIHB1YmxpYyBlbmQ6IGJpZ2ludCkge1xuICAgIGlmIChzdGFydCA8IDAgfHwgZW5kIDwgMCB8fCBlbmQgPCBzdGFydCkge1xuICAgICAgdGhyb3cgbmV3IEludmFsaWRTYXRSYW5nZShgSW52YWxpZCBTYXRSYW5nZSBbJHtzdGFydH0sICR7ZW5kfV1gLCBzdGFydCwgZW5kKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogQHBhcmFtIG9mZnNldFxuICAgKiBAcmV0dXJuIFNhdFJhbmdlIHdpdGggc3RhcnQgYW5kIGVuZCBzaGlmdGVkIGJ5IG9mZnNldFxuICAgKi9cbiAgc2hpZnRlZEJ5KG9mZnNldDogYmlnaW50KTogU2F0UmFuZ2Uge1xuICAgIHJldHVybiBuZXcgU2F0UmFuZ2UodGhpcy5zdGFydCArIG9mZnNldCwgdGhpcy5lbmQgKyBvZmZzZXQpO1xuICB9XG5cbiAgLyoqIEByZXR1cm4gdHJ1ZSBpZmYgdGhpcyBpbnRlcnNlY3RzIHdpdGggX290aGVyXyAqL1xuICBpbnRlcnNlY3RzV2l0aChvdGhlcjogYmlnaW50IHwgU2F0UmFuZ2UpOiBib29sZWFuIHtcbiAgICBvdGhlciA9IHRvUmFuZ2Uob3RoZXIpO1xuICAgIGlmICh0aGlzLnN0YXJ0IDw9IG90aGVyLnN0YXJ0KSB7XG4gICAgICByZXR1cm4gb3RoZXIuc3RhcnQgPD0gdGhpcy5lbmQ7XG4gICAgfVxuICAgIHJldHVybiBvdGhlci5pbnRlcnNlY3RzV2l0aCh0aGlzKTtcbiAgfVxuXG4gIC8qKiBAcmV0dXJuIHRydWUgaWZmIHRoaXMgaXMgc3VwZXJzZXQgb2YgX290aGVyXy4gKi9cbiAgaXNTdXBlcnNldE9mKG90aGVyOiBiaWdpbnQgfCBTYXRSYW5nZSk6IGJvb2xlYW4ge1xuICAgIG90aGVyID0gdG9SYW5nZShvdGhlcik7XG4gICAgcmV0dXJuIHRoaXMuc3RhcnQgPD0gb3RoZXIuc3RhcnQgJiYgb3RoZXIuZW5kIDw9IHRoaXMuZW5kO1xuICB9XG5cbiAgdG9TdHJpbmcoKTogc3RyaW5nIHtcbiAgICBpZiAodGhpcy5zdGFydCA9PT0gdGhpcy5lbmQpIHtcbiAgICAgIHJldHVybiBgWyR7dGhpcy5zdGFydH1dYDtcbiAgICB9XG4gICAgcmV0dXJuIGBbJHt0aGlzLnN0YXJ0fS4uJHt0aGlzLmVuZH1dYDtcbiAgfVxuXG4gIHNpemUoKTogYmlnaW50IHtcbiAgICByZXR1cm4gdGhpcy5lbmQgLSB0aGlzLnN0YXJ0ICsgQmlnSW50KDEpO1xuICB9XG59XG4iXX0=
@@ -1,13 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.powerset = powerset;
4
- function powerset(arr) {
5
- if (arr.length === 0) {
6
- return [[]];
7
- }
8
- else {
9
- const [a, ...rest] = arr;
10
- return powerset(rest).flatMap((s) => [[a, ...s], s]);
11
- }
12
- }
13
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tYmluYXRpb25zLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbWJpbmF0aW9ucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLDRCQU9DO0FBUEQsU0FBZ0IsUUFBUSxDQUFJLEdBQVE7SUFDbEMsSUFBSSxHQUFHLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRSxDQUFDO1FBQ3JCLE9BQU8sQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUNkLENBQUM7U0FBTSxDQUFDO1FBQ04sTUFBTSxDQUFDLENBQUMsRUFBRSxHQUFHLElBQUksQ0FBQyxHQUFHLEdBQUcsQ0FBQztRQUN6QixPQUFPLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsR0FBRyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3ZELENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGZ1bmN0aW9uIHBvd2Vyc2V0PFQ+KGFycjogVFtdKTogVFtdW10ge1xuICBpZiAoYXJyLmxlbmd0aCA9PT0gMCkge1xuICAgIHJldHVybiBbW11dO1xuICB9IGVsc2Uge1xuICAgIGNvbnN0IFthLCAuLi5yZXN0XSA9IGFycjtcbiAgICByZXR1cm4gcG93ZXJzZXQocmVzdCkuZmxhdE1hcCgocykgPT4gW1thLCAuLi5zXSwgc10pO1xuICB9XG59XG4iXX0=
@@ -1,202 +0,0 @@
1
- "use strict";
2
- /*
3
- Functions for dealing with inscriptions.
4
-
5
- See https://docs.ordinals.com/inscriptions.html
6
- */
7
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
8
- if (k2 === undefined) k2 = k;
9
- var desc = Object.getOwnPropertyDescriptor(m, k);
10
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
11
- desc = { enumerable: true, get: function() { return m[k]; } };
12
- }
13
- Object.defineProperty(o, k2, desc);
14
- }) : (function(o, m, k, k2) {
15
- if (k2 === undefined) k2 = k;
16
- o[k2] = m[k];
17
- }));
18
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
19
- Object.defineProperty(o, "default", { enumerable: true, value: v });
20
- }) : function(o, v) {
21
- o["default"] = v;
22
- });
23
- var __importStar = (this && this.__importStar) || (function () {
24
- var ownKeys = function(o) {
25
- ownKeys = Object.getOwnPropertyNames || function (o) {
26
- var ar = [];
27
- for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
28
- return ar;
29
- };
30
- return ownKeys(o);
31
- };
32
- return function (mod) {
33
- if (mod && mod.__esModule) return mod;
34
- var result = {};
35
- if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
36
- __setModuleDefault(result, mod);
37
- return result;
38
- };
39
- })();
40
- Object.defineProperty(exports, "__esModule", { value: true });
41
- exports.createInscriptionRevealData = createInscriptionRevealData;
42
- exports.createOutputScriptForInscription = createOutputScriptForInscription;
43
- exports.signRevealTransaction = signRevealTransaction;
44
- const assert = __importStar(require("assert"));
45
- const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
46
- const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
47
- const OPS = utxo_lib_1.script.OPS;
48
- const MAX_LENGTH_TAP_DATA_PUSH = 520;
49
- // default "postage" amount
50
- // https://github.com/ordinals/ord/blob/0.24.2/src/lib.rs#L149
51
- const DEFAULT_POSTAGE_AMOUNT = BigInt(10000);
52
- /**
53
- * The max size of an individual OP_PUSH in a Taproot script is 520 bytes. This
54
- * function splits inscriptionData into an array buffer of 520 bytes length.
55
- * https://docs.ordinals.com/inscriptions.html
56
- * @param inscriptionData
57
- * @param chunkSize
58
- */
59
- function splitBuffer(inscriptionData, chunkSize) {
60
- const pushDataBuffers = [];
61
- for (let i = 0; i < inscriptionData.length; i += chunkSize) {
62
- pushDataBuffers.push(inscriptionData.slice(i, i + chunkSize));
63
- }
64
- return pushDataBuffers;
65
- }
66
- /**
67
- *
68
- * @returns inscription payment object
69
- * @param pubkey
70
- * @param contentType
71
- * @param inscriptionData
72
- */
73
- function createPaymentForInscription(pubkey, contentType, inscriptionData) {
74
- const dataPushBuffers = splitBuffer(inscriptionData, MAX_LENGTH_TAP_DATA_PUSH);
75
- const uncompiledScript = [
76
- pubkey,
77
- OPS.OP_CHECKSIG,
78
- OPS.OP_FALSE,
79
- OPS.OP_IF,
80
- Buffer.from('ord', 'ascii'),
81
- 1, // these two lines should be combined as a single OPS.OP_1,
82
- 1, // but `ord`'s decoder has a bug so it has to be like this
83
- Buffer.from(contentType, 'ascii'),
84
- OPS.OP_0,
85
- ...dataPushBuffers,
86
- OPS.OP_ENDIF,
87
- ];
88
- const compiledScript = utxo_lib_1.script.compile(uncompiledScript);
89
- const redeem = {
90
- output: compiledScript,
91
- depth: 0,
92
- };
93
- return utxo_lib_1.p2trPayments.p2tr({ redeems: [redeem], redeemIndex: 0 }, { eccLib: utxo_lib_1.ecc });
94
- }
95
- /**
96
- * @param payment
97
- * @param controlBlock
98
- * @param commitOutput
99
- * @param network
100
- * @return virtual size of a transaction with a single inscription reveal input and a single commitOutput
101
- */
102
- function getInscriptionRevealSize(payment, controlBlock, commitOutput, network) {
103
- const psbt = utxo_lib_1.bitgo.createPsbtForNetwork({ network });
104
- const parsedControlBlock = utxo_lib_1.taproot.parseControlBlock(utxo_lib_1.ecc, controlBlock);
105
- const leafHash = utxo_lib_1.taproot.getTapleafHash(utxo_lib_1.ecc, parsedControlBlock, payment.redeem?.output);
106
- psbt.addInput({
107
- hash: Buffer.alloc(32),
108
- index: 0,
109
- witnessUtxo: { script: commitOutput, value: BigInt(100000) },
110
- tapLeafScript: [
111
- {
112
- controlBlock,
113
- script: payment.redeem?.output,
114
- leafVersion: utxo_lib_1.taproot.INITIAL_TAPSCRIPT_VERSION,
115
- },
116
- ],
117
- });
118
- psbt.addOutput({ script: commitOutput, value: DEFAULT_POSTAGE_AMOUNT });
119
- psbt.signTaprootInput(0, {
120
- publicKey: Buffer.alloc(32),
121
- signSchnorr(hash) {
122
- // dummy schnorr-sized signature
123
- return Buffer.alloc(64);
124
- },
125
- }, [leafHash]);
126
- psbt.finalizeTapInputWithSingleLeafScriptAndSignature(0);
127
- return psbt.extractTransaction(/* disableFeeCheck */ true).virtualSize();
128
- }
129
- /**
130
- * @param pubkey
131
- * @param contentType
132
- * @param inscriptionData
133
- * @param network
134
- * @returns PreparedInscriptionRevealData
135
- */
136
- function createInscriptionRevealData(pubkey, contentType, inscriptionData, network) {
137
- const payment = createPaymentForInscription(pubkey, contentType, inscriptionData);
138
- const { output: commitOutput, controlBlock } = payment;
139
- assert.ok(commitOutput);
140
- assert.ok(controlBlock);
141
- assert.ok(payment.redeem?.output);
142
- const commitAddress = utxo_lib_1.address.fromOutputScript(commitOutput, network);
143
- const tapLeafScript = [
144
- {
145
- controlBlock,
146
- script: payment.redeem?.output,
147
- leafVersion: utxo_lib_1.taproot.INITIAL_TAPSCRIPT_VERSION,
148
- },
149
- ];
150
- const revealTransactionVSize = getInscriptionRevealSize(payment, controlBlock, commitOutput, network);
151
- return {
152
- address: commitAddress,
153
- revealTransactionVSize,
154
- tapLeafScript: tapLeafScript[0],
155
- };
156
- }
157
- /**
158
- * @param pubkey
159
- * @param contentType
160
- * @param inscriptionData
161
- * @returns inscription address
162
- */
163
- function createOutputScriptForInscription(pubkey, contentType, inscriptionData) {
164
- const payment = createPaymentForInscription(pubkey, contentType, inscriptionData);
165
- assert.ok(payment.output, 'Failed to create inscription output script');
166
- return payment.output;
167
- }
168
- /**
169
- *
170
- * @param privateKey
171
- * @param tapLeafScript
172
- * @param commitAddress
173
- * @param recipientAddress
174
- * @param unsignedCommitTx
175
- * @param network
176
- *
177
- * @return a fully signed reveal transaction
178
- */
179
- function signRevealTransaction(privateKey, tapLeafScript, commitAddress, recipientAddress, unsignedCommitTx, network) {
180
- const unserCommitTxn = utxolib.bitgo.createTransactionFromBuffer(unsignedCommitTx, network);
181
- const hash = unserCommitTxn.getHash();
182
- const commitOutput = utxolib.address.toOutputScript(commitAddress, network);
183
- const vout = unserCommitTxn.outs.findIndex((out) => out.script.equals(commitOutput));
184
- if (vout === -1) {
185
- throw new Error('Invalid commit transaction');
186
- }
187
- const psbt = utxo_lib_1.bitgo.createPsbtForNetwork({ network });
188
- psbt.addInput({
189
- hash,
190
- index: vout,
191
- witnessUtxo: { script: commitOutput, value: BigInt(unserCommitTxn.outs[vout].value) },
192
- tapLeafScript: [tapLeafScript],
193
- });
194
- const recipientOutput = utxo_lib_1.address.toOutputScript(recipientAddress, network);
195
- psbt.addOutput({ script: recipientOutput, value: BigInt(10000) });
196
- const signer = utxo_lib_1.ECPair.fromPrivateKey(privateKey);
197
- const parsedControlBlock = utxo_lib_1.taproot.parseControlBlock(utxo_lib_1.ecc, tapLeafScript.controlBlock);
198
- const leafHash = utxo_lib_1.taproot.getTapleafHash(utxo_lib_1.ecc, parsedControlBlock, tapLeafScript.script);
199
- psbt.signTaprootInput(0, signer, [leafHash]);
200
- return psbt;
201
- }
202
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"inscriptions.js","sourceRoot":"","sources":["../../src/inscriptions.ts"],"names":[],"mappings":"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","sourcesContent":["/*\nFunctions for dealing with inscriptions.\n\nSee https://docs.ordinals.com/inscriptions.html\n*/\n\nimport * as assert from 'assert';\nimport {\n  p2trPayments as payments,\n  ecc as eccLib,\n  script as bscript,\n  Payment,\n  Network,\n  bitgo,\n  address,\n  taproot,\n  ECPair,\n} from '@bitgo-beta/utxo-lib';\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { PreparedInscriptionRevealData } from '@bitgo-beta/sdk-core';\n\nconst OPS = bscript.OPS;\nconst MAX_LENGTH_TAP_DATA_PUSH = 520;\n// default \"postage\" amount\n// https://github.com/ordinals/ord/blob/0.24.2/src/lib.rs#L149\nconst DEFAULT_POSTAGE_AMOUNT = BigInt(10_000);\n\n/**\n * The max size of an individual OP_PUSH in a Taproot script is 520 bytes. This\n * function splits inscriptionData into an array buffer of 520 bytes length.\n * https://docs.ordinals.com/inscriptions.html\n * @param inscriptionData\n * @param chunkSize\n */\nfunction splitBuffer(inscriptionData: Buffer, chunkSize: number) {\n  const pushDataBuffers: Buffer[] = [];\n  for (let i = 0; i < inscriptionData.length; i += chunkSize) {\n    pushDataBuffers.push(inscriptionData.slice(i, i + chunkSize));\n  }\n\n  return pushDataBuffers;\n}\n\n/**\n *\n * @returns inscription payment object\n * @param pubkey\n * @param contentType\n * @param inscriptionData\n */\nfunction createPaymentForInscription(pubkey: Buffer, contentType: string, inscriptionData: Buffer): Payment {\n  const dataPushBuffers = splitBuffer(inscriptionData, MAX_LENGTH_TAP_DATA_PUSH);\n\n  const uncompiledScript = [\n    pubkey,\n    OPS.OP_CHECKSIG,\n    OPS.OP_FALSE,\n    OPS.OP_IF,\n    Buffer.from('ord', 'ascii'),\n    1, // these two lines should be combined as a single OPS.OP_1,\n    1, // but `ord`'s decoder has a bug so it has to be like this\n    Buffer.from(contentType, 'ascii'),\n    OPS.OP_0,\n    ...dataPushBuffers,\n    OPS.OP_ENDIF,\n  ];\n\n  const compiledScript = bscript.compile(uncompiledScript);\n  const redeem: Payment = {\n    output: compiledScript,\n    depth: 0,\n  };\n\n  return payments.p2tr({ redeems: [redeem], redeemIndex: 0 }, { eccLib });\n}\n\n/**\n * @param payment\n * @param controlBlock\n * @param commitOutput\n * @param network\n * @return virtual size of a transaction with a single inscription reveal input and a single commitOutput\n */\nfunction getInscriptionRevealSize(\n  payment: Payment,\n  controlBlock: Buffer,\n  commitOutput: Buffer,\n  network: Network\n): number {\n  const psbt = bitgo.createPsbtForNetwork({ network });\n  const parsedControlBlock = taproot.parseControlBlock(eccLib, controlBlock);\n  const leafHash = taproot.getTapleafHash(eccLib, parsedControlBlock, payment.redeem?.output as Buffer);\n\n  psbt.addInput({\n    hash: Buffer.alloc(32),\n    index: 0,\n    witnessUtxo: { script: commitOutput, value: BigInt(100_000) },\n    tapLeafScript: [\n      {\n        controlBlock,\n        script: payment.redeem?.output as Buffer,\n        leafVersion: taproot.INITIAL_TAPSCRIPT_VERSION,\n      },\n    ],\n  });\n  psbt.addOutput({ script: commitOutput, value: DEFAULT_POSTAGE_AMOUNT });\n\n  psbt.signTaprootInput(\n    0,\n    {\n      publicKey: Buffer.alloc(32),\n      signSchnorr(hash: Buffer): Buffer {\n        // dummy schnorr-sized signature\n        return Buffer.alloc(64);\n      },\n    },\n    [leafHash]\n  );\n\n  psbt.finalizeTapInputWithSingleLeafScriptAndSignature(0);\n  return psbt.extractTransaction(/* disableFeeCheck */ true).virtualSize();\n}\n\n/**\n * @param pubkey\n * @param contentType\n * @param inscriptionData\n * @param network\n * @returns PreparedInscriptionRevealData\n */\nexport function createInscriptionRevealData(\n  pubkey: Buffer,\n  contentType: string,\n  inscriptionData: Buffer,\n  network: Network\n): PreparedInscriptionRevealData {\n  const payment = createPaymentForInscription(pubkey, contentType, inscriptionData);\n\n  const { output: commitOutput, controlBlock } = payment;\n  assert.ok(commitOutput);\n  assert.ok(controlBlock);\n  assert.ok(payment.redeem?.output);\n  const commitAddress = address.fromOutputScript(commitOutput, network);\n\n  const tapLeafScript: utxolib.bitgo.TapLeafScript[] = [\n    {\n      controlBlock,\n      script: payment.redeem?.output,\n      leafVersion: taproot.INITIAL_TAPSCRIPT_VERSION,\n    },\n  ];\n  const revealTransactionVSize = getInscriptionRevealSize(payment, controlBlock, commitOutput, network);\n\n  return {\n    address: commitAddress,\n    revealTransactionVSize,\n    tapLeafScript: tapLeafScript[0],\n  };\n}\n\n/**\n * @param pubkey\n * @param contentType\n * @param inscriptionData\n * @returns inscription address\n */\nexport function createOutputScriptForInscription(pubkey: Buffer, contentType: string, inscriptionData: Buffer): Buffer {\n  const payment = createPaymentForInscription(pubkey, contentType, inscriptionData);\n\n  assert.ok(payment.output, 'Failed to create inscription output script');\n  return payment.output;\n}\n\n/**\n *\n * @param privateKey\n * @param tapLeafScript\n * @param commitAddress\n * @param recipientAddress\n * @param unsignedCommitTx\n * @param network\n *\n * @return a fully signed reveal transaction\n */\nexport function signRevealTransaction(\n  privateKey: Buffer,\n  tapLeafScript: utxolib.bitgo.TapLeafScript,\n  commitAddress: string,\n  recipientAddress: string,\n  unsignedCommitTx: Buffer,\n  network: Network\n): utxolib.bitgo.UtxoPsbt {\n  const unserCommitTxn = utxolib.bitgo.createTransactionFromBuffer(unsignedCommitTx, network);\n  const hash = unserCommitTxn.getHash();\n  const commitOutput = utxolib.address.toOutputScript(commitAddress, network);\n  const vout = unserCommitTxn.outs.findIndex((out) => out.script.equals(commitOutput));\n\n  if (vout === -1) {\n    throw new Error('Invalid commit transaction');\n  }\n\n  const psbt = bitgo.createPsbtForNetwork({ network });\n  psbt.addInput({\n    hash,\n    index: vout,\n    witnessUtxo: { script: commitOutput, value: BigInt(unserCommitTxn.outs[vout].value) },\n    tapLeafScript: [tapLeafScript],\n  });\n\n  const recipientOutput = address.toOutputScript(recipientAddress, network);\n  psbt.addOutput({ script: recipientOutput, value: BigInt(10_000) });\n\n  const signer = ECPair.fromPrivateKey(privateKey);\n  const parsedControlBlock = taproot.parseControlBlock(eccLib, tapLeafScript.controlBlock);\n  const leafHash = taproot.getTapleafHash(eccLib, parsedControlBlock, tapLeafScript.script as Buffer);\n  psbt.signTaprootInput(0, signer, [leafHash]);\n\n  return psbt;\n}\n"]}