@bitgo-beta/utxo-ord 1.1.3-alpha.44 → 1.1.3-alpha.440

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 (64) 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/{src → cjs/src}/OutputLayout.d.ts +3 -3
  4. package/dist/cjs/src/OutputLayout.d.ts.map +1 -0
  5. package/dist/cjs/src/OutputLayout.js +246 -0
  6. package/dist/{src → cjs/src}/SatPoint.d.ts +8 -1
  7. package/dist/cjs/src/SatPoint.d.ts.map +1 -0
  8. package/dist/cjs/src/SatPoint.js +67 -0
  9. package/dist/cjs/src/SatRange.d.ts.map +1 -0
  10. package/dist/cjs/src/SatRange.js +58 -0
  11. package/dist/cjs/src/combinations.d.ts.map +1 -0
  12. package/dist/cjs/src/combinations.js +13 -0
  13. package/dist/{src → cjs/src}/index.d.ts +2 -0
  14. package/dist/cjs/src/index.d.ts.map +1 -0
  15. package/dist/cjs/src/index.js +49 -0
  16. package/dist/cjs/src/inscriptions.d.ts +58 -0
  17. package/dist/cjs/src/inscriptions.d.ts.map +1 -0
  18. package/dist/cjs/src/inscriptions.js +94 -0
  19. package/dist/{src → cjs/src}/psbt.d.ts +33 -17
  20. package/dist/cjs/src/psbt.d.ts.map +1 -0
  21. package/dist/cjs/src/psbt.js +193 -0
  22. package/dist/cjs/tsconfig.tsbuildinfo +1 -0
  23. package/dist/esm/OrdOutput.d.ts +65 -0
  24. package/dist/esm/OrdOutput.js +192 -0
  25. package/dist/esm/OutputLayout.d.ts +47 -0
  26. package/dist/esm/OutputLayout.d.ts.map +1 -0
  27. package/dist/esm/OutputLayout.js +240 -0
  28. package/dist/esm/SatPoint.d.ts +20 -0
  29. package/dist/esm/SatPoint.d.ts.map +1 -0
  30. package/dist/esm/SatPoint.js +61 -0
  31. package/dist/esm/SatRange.d.ts +26 -0
  32. package/dist/esm/SatRange.js +53 -0
  33. package/dist/esm/combinations.d.ts +2 -0
  34. package/dist/esm/combinations.js +10 -0
  35. package/dist/esm/index.d.ts +12 -0
  36. package/dist/{src → esm}/index.d.ts.map +1 -1
  37. package/dist/esm/index.js +10 -0
  38. package/dist/esm/inscriptions.d.ts +58 -0
  39. package/dist/esm/inscriptions.d.ts.map +1 -0
  40. package/dist/esm/inscriptions.js +89 -0
  41. package/dist/esm/psbt.d.ts +84 -0
  42. package/dist/esm/psbt.d.ts.map +1 -0
  43. package/dist/esm/psbt.js +186 -0
  44. package/package.json +27 -9
  45. package/dist/src/OrdOutput.js +0 -197
  46. package/dist/src/OutputLayout.d.ts.map +0 -1
  47. package/dist/src/OutputLayout.js +0 -248
  48. package/dist/src/SatPoint.d.ts.map +0 -1
  49. package/dist/src/SatPoint.js +0 -49
  50. package/dist/src/SatRange.js +0 -58
  51. package/dist/src/combinations.js +0 -14
  52. package/dist/src/index.js +0 -23
  53. package/dist/src/inscriptions.d.ts +0 -32
  54. package/dist/src/inscriptions.d.ts.map +0 -1
  55. package/dist/src/inscriptions.js +0 -169
  56. package/dist/src/psbt.d.ts.map +0 -1
  57. package/dist/src/psbt.js +0 -150
  58. package/dist/tsconfig.tsbuildinfo +0 -1
  59. /package/dist/{src → cjs/src}/OrdOutput.d.ts +0 -0
  60. /package/dist/{src → cjs/src}/SatRange.d.ts +0 -0
  61. /package/dist/{src → cjs/src}/combinations.d.ts +0 -0
  62. /package/dist/{src → esm}/OrdOutput.d.ts.map +0 -0
  63. /package/dist/{src → esm}/SatRange.d.ts.map +0 -0
  64. /package/dist/{src → esm}/combinations.d.ts.map +0 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"OrdOutput.d.ts","sourceRoot":"","sources":["../../../src/OrdOutput.ts"],"names":[],"mappings":"AAoDA,OAAO,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAC;AAEtC,qBAAa,gBAAiB,SAAQ,KAAK;IACL,KAAK,EAAE,MAAM;IAAS,QAAQ,EAAE,QAAQ,EAAE;gBAAlE,OAAO,EAAE,MAAM,EAAS,KAAK,EAAE,MAAM,EAAS,QAAQ,EAAE,QAAQ,EAAE;CAG/E;AAED;;GAEG;AACH,qBAAa,SAAS;IAOD,KAAK,EAAE,MAAM;IAAS,QAAQ,EAAE,QAAQ,EAAE;IAN7D;;;;;OAKG;gBACgB,KAAK,EAAE,MAAM,EAAS,QAAQ,GAAE,QAAQ,EAAO;IAelE;;;OAGG;IACH,UAAU,CAAC,KAAK,EAAE,SAAS,GAAG,SAAS;IAOvC;;;OAGG;IACH,MAAM,CAAC,OAAO,CAAC,IAAI,EAAE,SAAS,EAAE,GAAG,SAAS;IAO5C,UAAU,IAAI,QAAQ;IAItB;;;OAGG;IACH,YAAY,CAAC,CAAC,EAAE,QAAQ,GAAG,SAAS;IAepC;;;;OAIG;IACH,OAAO,CAAC,KAAK,EAAE,MAAM,GAAG,CAAC,SAAS,EAAE,SAAS,CAAC;IAU9C;;;OAGG;IACH,gBAAgB,CAAC,KAAK,EAAE,MAAM,GAAG,CAAC,SAAS,GAAG,IAAI,EAAE,SAAS,GAAG,IAAI,CAAC;IAUrE;;;;;;OAMG;IACH,kBAAkB,CAChB,MAAM,EAAE,MAAM,EAAE,EAChB,EAAE,KAAa,EAAE,SAAiB,EAAE,EAAE;QAAE,SAAS,CAAC,EAAE,OAAO,CAAC;QAAC,KAAK,CAAC,EAAE,OAAO,CAAA;KAAE,GAC7E,CAAC,SAAS,GAAG,IAAI,CAAC,EAAE;IAwBvB;;;;OAIG;IACH,QAAQ,CAAC,MAAM,EAAE,MAAM,EAAE,GAAG,SAAS,EAAE;CAGxC"}
@@ -0,0 +1,197 @@
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"]}
@@ -2,7 +2,7 @@ import { OrdOutput } from './OrdOutput';
2
2
  /**
3
3
  * A range constraint
4
4
  */
5
- declare type Parameters<T> = {
5
+ type Parameters<T> = {
6
6
  /** Padding preceding the inscription output */
7
7
  firstChangeOutput: T;
8
8
  /** The inscription output that will inherit the input inscription */
@@ -16,7 +16,7 @@ declare type Parameters<T> = {
16
16
  export declare function toArray<T>(p: Parameters<T>): [T, T, T, T];
17
17
  export declare function toParameters<T>(firstChangeOutput: T, inscriptionOutput: T, secondChangeOutput: T, feeOutput: T): Parameters<T>;
18
18
  /** A finished output layout */
19
- export declare type OutputLayout = Parameters<bigint>;
19
+ export type OutputLayout = Parameters<bigint>;
20
20
  /**
21
21
  * Translates a layout into OrdOutputs. Absent outputs are set to `null`.
22
22
  *
@@ -28,7 +28,7 @@ export declare function getOrdOutputsForLayout(inscriptionInput: OrdOutput, layo
28
28
  /**
29
29
  * High-level constraints for output layout
30
30
  */
31
- export declare type Constraints = {
31
+ export type Constraints = {
32
32
  minChangeOutput: bigint;
33
33
  minInscriptionOutput: bigint;
34
34
  maxInscriptionOutput: bigint;
@@ -0,0 +1 @@
1
+ {"version":3,"file":"OutputLayout.d.ts","sourceRoot":"","sources":["../../../src/OutputLayout.ts"],"names":[],"mappings":"AAkGA,OAAO,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAiBxC;;GAEG;AACH,KAAK,UAAU,CAAC,CAAC,IAAI;IACnB,+CAA+C;IAC/C,iBAAiB,EAAE,CAAC,CAAC;IACrB,qEAAqE;IACrE,iBAAiB,EAAE,CAAC,CAAC;IACrB,+CAA+C;IAC/C,kBAAkB,EAAE,CAAC,CAAC;IACtB,4DAA4D;IAC5D,SAAS,EAAE,CAAC,CAAC;CACd,CAAC;AAEF,+CAA+C;AAC/C,wBAAgB,OAAO,CAAC,CAAC,EAAE,CAAC,EAAE,UAAU,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAEzD;AAED,wBAAgB,YAAY,CAAC,CAAC,EAC5B,iBAAiB,EAAE,CAAC,EACpB,iBAAiB,EAAE,CAAC,EACpB,kBAAkB,EAAE,CAAC,EACrB,SAAS,EAAE,CAAC,GACX,UAAU,CAAC,CAAC,CAAC,CAOf;AAED,+BAA+B;AAC/B,MAAM,MAAM,YAAY,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC;AAE9C;;;;;;GAMG;AACH,wBAAgB,sBAAsB,CACpC,gBAAgB,EAAE,SAAS,EAC3B,MAAM,EAAE,YAAY,GACnB,UAAU,CAAC,SAAS,GAAG,IAAI,CAAC,CAO9B;AA0BD;;GAEG;AACH,MAAM,MAAM,WAAW,GAAG;IACxB,eAAe,EAAE,MAAM,CAAC;IACxB,oBAAoB,EAAE,MAAM,CAAC;IAC7B,oBAAoB,EAAE,MAAM,CAAC;IAC7B,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;CACf,CAAC;AA2DF;;;;GAIG;AACH,wBAAgB,gBAAgB,CAC9B,gBAAgB,EAAE,SAAS,EAC3B,MAAM,EAAE,IAAI,CAAC,WAAW,EAAE,QAAQ,GAAG,OAAO,CAAC,GAC5C,YAAY,GAAG,SAAS,CAkC1B"}
@@ -0,0 +1,246 @@
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,4 +1,11 @@
1
- export declare type SatPoint = `${string}:${number}:${bigint}`;
1
+ export type SatPoint = `${string}:${number}:${bigint}`;
2
+ /**
3
+ * Parse an output ID (txid:vout) into its components.
4
+ */
5
+ export declare function parseOutputId(outputId: string): {
6
+ txid: string;
7
+ vout: number;
8
+ };
2
9
  export declare function parseSatPoint(p: SatPoint): {
3
10
  txid: string;
4
11
  vout: number;
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SatPoint.d.ts","sourceRoot":"","sources":["../../../src/SatPoint.ts"],"names":[],"mappings":"AAYA,MAAM,MAAM,QAAQ,GAAG,GAAG,MAAM,IAAI,MAAM,IAAI,MAAM,EAAE,CAAC;AAEvD;;GAEG;AACH,wBAAgB,aAAa,CAAC,QAAQ,EAAE,MAAM,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAA;CAAE,CAe9E;AAED,wBAAgB,aAAa,CAAC,CAAC,EAAE,QAAQ,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,CAiBzF;AAED,wBAAgB,cAAc,CAAC,CAAC,EAAE;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAAE,GAAG,QAAQ,CAE1F;AAED,wBAAgB,UAAU,CAAC,CAAC,EAAE,MAAM,GAAG,CAAC,IAAI,QAAQ,CAOnD"}
@@ -0,0 +1,67 @@
1
+ "use strict";
2
+ /*
3
+
4
+ https://github.com/casey/ord/blob/master/bip.mediawiki#terminology-and-notation
5
+
6
+ > A satpoint may be used to indicate the location of a sat within an output.
7
+ > A satpoint consists of an outpoint, i.e., a transaction ID and output index, with the addition of
8
+ > the offset of the ordinal within that output. For example, if the sat in question is at offset 6
9
+ > in the first output of a transaction, its satpoint is:
10
+ > `680df1e4d43016571e504b0b142ee43c5c0b83398a97bdcfd94ea6f287322d22:0:6`
11
+
12
+ */
13
+ Object.defineProperty(exports, "__esModule", { value: true });
14
+ exports.parseOutputId = parseOutputId;
15
+ exports.parseSatPoint = parseSatPoint;
16
+ exports.formatSatPoint = formatSatPoint;
17
+ exports.isSatPoint = isSatPoint;
18
+ /**
19
+ * Parse an output ID (txid:vout) into its components.
20
+ */
21
+ function parseOutputId(outputId) {
22
+ const colonIndex = outputId.lastIndexOf(':');
23
+ if (colonIndex === -1) {
24
+ throw new Error(`Invalid output id format: missing colon`);
25
+ }
26
+ const txid = outputId.slice(0, colonIndex);
27
+ const voutStr = outputId.slice(colonIndex + 1);
28
+ if (txid.length !== 64 || !/^[0-9a-fA-F]+$/.test(txid)) {
29
+ throw new Error(`Invalid txid: must be 64 hex characters`);
30
+ }
31
+ const vout = parseInt(voutStr, 10);
32
+ if (isNaN(vout) || vout < 0) {
33
+ throw new Error(`Invalid vout: must be non-negative integer`);
34
+ }
35
+ return { txid, vout };
36
+ }
37
+ function parseSatPoint(p) {
38
+ const parts = p.split(':');
39
+ if (parts.length !== 3) {
40
+ throw new Error(`expected format txid:vout:sat`);
41
+ }
42
+ const [txid, vout, offsetStr] = parts;
43
+ const offset = BigInt(offsetStr);
44
+ if (offset.toString() !== offsetStr) {
45
+ throw new Error(`SatPoint offset must be base-10`);
46
+ }
47
+ if (offset < 0) {
48
+ throw new Error(`SatPoint offset must be positive`);
49
+ }
50
+ return {
51
+ ...parseOutputId([txid, vout].join(':')),
52
+ offset,
53
+ };
54
+ }
55
+ function formatSatPoint(p) {
56
+ return `${p.txid}:${p.vout}:${p.offset}`;
57
+ }
58
+ function isSatPoint(v) {
59
+ try {
60
+ parseSatPoint(v);
61
+ return true;
62
+ }
63
+ catch {
64
+ return false;
65
+ }
66
+ }
67
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2F0UG9pbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvU2F0UG9pbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7O0VBVUU7O0FBT0Ysc0NBZUM7QUFFRCxzQ0FpQkM7QUFFRCx3Q0FFQztBQUVELGdDQU9DO0FBbEREOztHQUVHO0FBQ0gsU0FBZ0IsYUFBYSxDQUFDLFFBQWdCO0lBQzVDLE1BQU0sVUFBVSxHQUFHLFFBQVEsQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDN0MsSUFBSSxVQUFVLEtBQUssQ0FBQyxDQUFDLEVBQUUsQ0FBQztRQUN0QixNQUFNLElBQUksS0FBSyxDQUFDLHlDQUF5QyxDQUFDLENBQUM7SUFDN0QsQ0FBQztJQUNELE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLFVBQVUsQ0FBQyxDQUFDO0lBQzNDLE1BQU0sT0FBTyxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUMsVUFBVSxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQy9DLElBQUksSUFBSSxDQUFDLE1BQU0sS0FBSyxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQztRQUN2RCxNQUFNLElBQUksS0FBSyxDQUFDLHlDQUF5QyxDQUFDLENBQUM7SUFDN0QsQ0FBQztJQUNELE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDbkMsSUFBSSxLQUFLLENBQUMsSUFBSSxDQUFDLElBQUksSUFBSSxHQUFHLENBQUMsRUFBRSxDQUFDO1FBQzVCLE1BQU0sSUFBSSxLQUFLLENBQUMsNENBQTRDLENBQUMsQ0FBQztJQUNoRSxDQUFDO0lBQ0QsT0FBTyxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsQ0FBQztBQUN4QixDQUFDO0FBRUQsU0FBZ0IsYUFBYSxDQUFDLENBQVc7SUFDdkMsTUFBTSxLQUFLLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUMzQixJQUFJLEtBQUssQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7UUFDdkIsTUFBTSxJQUFJLEtBQUssQ0FBQywrQkFBK0IsQ0FBQyxDQUFDO0lBQ25ELENBQUM7SUFDRCxNQUFNLENBQUMsSUFBSSxFQUFFLElBQUksRUFBRSxTQUFTLENBQUMsR0FBRyxLQUFLLENBQUM7SUFDdEMsTUFBTSxNQUFNLEdBQUcsTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0lBQ2pDLElBQUksTUFBTSxDQUFDLFFBQVEsRUFBRSxLQUFLLFNBQVMsRUFBRSxDQUFDO1FBQ3BDLE1BQU0sSUFBSSxLQUFLLENBQUMsaUNBQWlDLENBQUMsQ0FBQztJQUNyRCxDQUFDO0lBQ0QsSUFBSSxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7UUFDZixNQUFNLElBQUksS0FBSyxDQUFDLGtDQUFrQyxDQUFDLENBQUM7SUFDdEQsQ0FBQztJQUNELE9BQU87UUFDTCxHQUFHLGFBQWEsQ0FBQyxDQUFDLElBQUksRUFBRSxJQUFJLENBQUMsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDeEMsTUFBTTtLQUNQLENBQUM7QUFDSixDQUFDO0FBRUQsU0FBZ0IsY0FBYyxDQUFDLENBQWlEO0lBQzlFLE9BQU8sR0FBRyxDQUFDLENBQUMsSUFBSSxJQUFJLENBQUMsQ0FBQyxJQUFJLElBQUksQ0FBQyxDQUFDLE1BQU0sRUFBVyxDQUFDO0FBQ3BELENBQUM7QUFFRCxTQUFnQixVQUFVLENBQUMsQ0FBUztJQUNsQyxJQUFJLENBQUM7UUFDSCxhQUFhLENBQUMsQ0FBYSxDQUFDLENBQUM7UUFDN0IsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBQUMsTUFBTSxDQUFDO1FBQ1AsT0FBTyxLQUFLLENBQUM7SUFDZixDQUFDO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG5cbmh0dHBzOi8vZ2l0aHViLmNvbS9jYXNleS9vcmQvYmxvYi9tYXN0ZXIvYmlwLm1lZGlhd2lraSN0ZXJtaW5vbG9neS1hbmQtbm90YXRpb25cblxuPiBBIHNhdHBvaW50IG1heSBiZSB1c2VkIHRvIGluZGljYXRlIHRoZSBsb2NhdGlvbiBvZiBhIHNhdCB3aXRoaW4gYW4gb3V0cHV0LlxuPiBBIHNhdHBvaW50IGNvbnNpc3RzIG9mIGFuIG91dHBvaW50LCBpLmUuLCBhIHRyYW5zYWN0aW9uIElEIGFuZCBvdXRwdXQgaW5kZXgsIHdpdGggdGhlIGFkZGl0aW9uIG9mXG4+IHRoZSBvZmZzZXQgb2YgdGhlIG9yZGluYWwgd2l0aGluIHRoYXQgb3V0cHV0LiBGb3IgZXhhbXBsZSwgaWYgdGhlIHNhdCBpbiBxdWVzdGlvbiBpcyBhdCBvZmZzZXQgNlxuPiBpbiB0aGUgZmlyc3Qgb3V0cHV0IG9mIGEgdHJhbnNhY3Rpb24sIGl0cyBzYXRwb2ludCBpczpcbj4gYDY4MGRmMWU0ZDQzMDE2NTcxZTUwNGIwYjE0MmVlNDNjNWMwYjgzMzk4YTk3YmRjZmQ5NGVhNmYyODczMjJkMjI6MDo2YFxuXG4qL1xuXG5leHBvcnQgdHlwZSBTYXRQb2ludCA9IGAke3N0cmluZ306JHtudW1iZXJ9OiR7YmlnaW50fWA7XG5cbi8qKlxuICogUGFyc2UgYW4gb3V0cHV0IElEICh0eGlkOnZvdXQpIGludG8gaXRzIGNvbXBvbmVudHMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwYXJzZU91dHB1dElkKG91dHB1dElkOiBzdHJpbmcpOiB7IHR4aWQ6IHN0cmluZzsgdm91dDogbnVtYmVyIH0ge1xuICBjb25zdCBjb2xvbkluZGV4ID0gb3V0cHV0SWQubGFzdEluZGV4T2YoJzonKTtcbiAgaWYgKGNvbG9uSW5kZXggPT09IC0xKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBJbnZhbGlkIG91dHB1dCBpZCBmb3JtYXQ6IG1pc3NpbmcgY29sb25gKTtcbiAgfVxuICBjb25zdCB0eGlkID0gb3V0cHV0SWQuc2xpY2UoMCwgY29sb25JbmRleCk7XG4gIGNvbnN0IHZvdXRTdHIgPSBvdXRwdXRJZC5zbGljZShjb2xvbkluZGV4ICsgMSk7XG4gIGlmICh0eGlkLmxlbmd0aCAhPT0gNjQgfHwgIS9eWzAtOWEtZkEtRl0rJC8udGVzdCh0eGlkKSkge1xuICAgIHRocm93IG5ldyBFcnJvcihgSW52YWxpZCB0eGlkOiBtdXN0IGJlIDY0IGhleCBjaGFyYWN0ZXJzYCk7XG4gIH1cbiAgY29uc3Qgdm91dCA9IHBhcnNlSW50KHZvdXRTdHIsIDEwKTtcbiAgaWYgKGlzTmFOKHZvdXQpIHx8IHZvdXQgPCAwKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBJbnZhbGlkIHZvdXQ6IG11c3QgYmUgbm9uLW5lZ2F0aXZlIGludGVnZXJgKTtcbiAgfVxuICByZXR1cm4geyB0eGlkLCB2b3V0IH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBwYXJzZVNhdFBvaW50KHA6IFNhdFBvaW50KTogeyB0eGlkOiBzdHJpbmc7IHZvdXQ6IG51bWJlcjsgb2Zmc2V0OiBiaWdpbnQgfSB7XG4gIGNvbnN0IHBhcnRzID0gcC5zcGxpdCgnOicpO1xuICBpZiAocGFydHMubGVuZ3RoICE9PSAzKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBleHBlY3RlZCBmb3JtYXQgdHhpZDp2b3V0OnNhdGApO1xuICB9XG4gIGNvbnN0IFt0eGlkLCB2b3V0LCBvZmZzZXRTdHJdID0gcGFydHM7XG4gIGNvbnN0IG9mZnNldCA9IEJpZ0ludChvZmZzZXRTdHIpO1xuICBpZiAob2Zmc2V0LnRvU3RyaW5nKCkgIT09IG9mZnNldFN0cikge1xuICAgIHRocm93IG5ldyBFcnJvcihgU2F0UG9pbnQgb2Zmc2V0IG11c3QgYmUgYmFzZS0xMGApO1xuICB9XG4gIGlmIChvZmZzZXQgPCAwKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKGBTYXRQb2ludCBvZmZzZXQgbXVzdCBiZSBwb3NpdGl2ZWApO1xuICB9XG4gIHJldHVybiB7XG4gICAgLi4ucGFyc2VPdXRwdXRJZChbdHhpZCwgdm91dF0uam9pbignOicpKSxcbiAgICBvZmZzZXQsXG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBmb3JtYXRTYXRQb2ludChwOiB7IHR4aWQ6IHN0cmluZzsgdm91dDogbnVtYmVyOyBvZmZzZXQ6IGJpZ2ludCB9KTogU2F0UG9pbnQge1xuICByZXR1cm4gYCR7cC50eGlkfToke3Audm91dH06JHtwLm9mZnNldH1gIGFzIGNvbnN0O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNTYXRQb2ludCh2OiBzdHJpbmcpOiB2IGlzIFNhdFBvaW50IHtcbiAgdHJ5IHtcbiAgICBwYXJzZVNhdFBvaW50KHYgYXMgU2F0UG9pbnQpO1xuICAgIHJldHVybiB0cnVlO1xuICB9IGNhdGNoIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SatRange.d.ts","sourceRoot":"","sources":["../../../src/SatRange.ts"],"names":[],"mappings":"AAAA,qBAAa,eAAgB,SAAQ,KAAK;IACJ,KAAK,EAAE,MAAM;IAAS,GAAG,EAAE,MAAM;gBAAzD,OAAO,EAAE,MAAM,EAAS,KAAK,EAAE,MAAM,EAAS,GAAG,EAAE,MAAM;CAGtE;AAMD;;;GAGG;AACH,qBAAa,QAAQ;IACA,KAAK,EAAE,MAAM;IAAS,GAAG,EAAE,MAAM;gBAAjC,KAAK,EAAE,MAAM,EAAS,GAAG,EAAE,MAAM;IAMpD;;;OAGG;IACH,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,QAAQ;IAInC,oDAAoD;IACpD,cAAc,CAAC,KAAK,EAAE,MAAM,GAAG,QAAQ,GAAG,OAAO;IAQjD,oDAAoD;IACpD,YAAY,CAAC,KAAK,EAAE,MAAM,GAAG,QAAQ,GAAG,OAAO;IAK/C,QAAQ,IAAI,MAAM;IAOlB,IAAI,IAAI,MAAM;CAGf"}