utxo-lib 1.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (249) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +59 -0
  3. package/dist/src/address.d.ts +11 -0
  4. package/dist/src/address.d.ts.map +1 -0
  5. package/dist/src/address.js +37 -0
  6. package/dist/src/addressFormat.d.ts +53 -0
  7. package/dist/src/addressFormat.d.ts.map +1 -0
  8. package/dist/src/addressFormat.js +110 -0
  9. package/dist/src/bitgo/Musig2.d.ts +51 -0
  10. package/dist/src/bitgo/Musig2.d.ts.map +1 -0
  11. package/dist/src/bitgo/Musig2.js +175 -0
  12. package/dist/src/bitgo/Unspent.d.ts +95 -0
  13. package/dist/src/bitgo/Unspent.d.ts.map +1 -0
  14. package/dist/src/bitgo/Unspent.js +138 -0
  15. package/dist/src/bitgo/UtxoPsbt.d.ts +132 -0
  16. package/dist/src/bitgo/UtxoPsbt.d.ts.map +1 -0
  17. package/dist/src/bitgo/UtxoPsbt.js +469 -0
  18. package/dist/src/bitgo/UtxoTransaction.d.ts +22 -0
  19. package/dist/src/bitgo/UtxoTransaction.d.ts.map +1 -0
  20. package/dist/src/bitgo/UtxoTransaction.js +97 -0
  21. package/dist/src/bitgo/UtxoTransactionBuilder.d.ts +26 -0
  22. package/dist/src/bitgo/UtxoTransactionBuilder.d.ts.map +1 -0
  23. package/dist/src/bitgo/UtxoTransactionBuilder.js +67 -0
  24. package/dist/src/bitgo/bbc/DashPsbt.d.ts +12 -0
  25. package/dist/src/bitgo/bbc/DashPsbt.d.ts.map +1 -0
  26. package/dist/src/bitgo/bbc/DashPsbt.js +27 -0
  27. package/dist/src/bitgo/bbc/DashTransaction.d.ts +39 -0
  28. package/dist/src/bitgo/bbc/DashTransaction.d.ts.map +1 -0
  29. package/dist/src/bitgo/bbc/DashTransaction.js +109 -0
  30. package/dist/src/bitgo/bbc/DashTransactionBuilder.d.ts +14 -0
  31. package/dist/src/bitgo/bbc/DashTransactionBuilder.d.ts.map +1 -0
  32. package/dist/src/bitgo/bbc/DashTransactionBuilder.js +28 -0
  33. package/dist/src/bitgo/bbc/index.d.ts +4 -0
  34. package/dist/src/bitgo/bbc/index.d.ts.map +1 -0
  35. package/dist/src/bitgo/bbc/index.js +16 -0
  36. package/dist/src/bitgo/bitcoincash/address.d.ts +35 -0
  37. package/dist/src/bitgo/bitcoincash/address.d.ts.map +1 -0
  38. package/dist/src/bitgo/bitcoincash/address.js +151 -0
  39. package/dist/src/bitgo/bitcoincash/index.d.ts +2 -0
  40. package/dist/src/bitgo/bitcoincash/index.d.ts.map +1 -0
  41. package/dist/src/bitgo/bitcoincash/index.js +14 -0
  42. package/dist/src/bitgo/dash/DashPsbt.d.ts +12 -0
  43. package/dist/src/bitgo/dash/DashPsbt.d.ts.map +1 -0
  44. package/dist/src/bitgo/dash/DashPsbt.js +27 -0
  45. package/dist/src/bitgo/dash/DashTransaction.d.ts +39 -0
  46. package/dist/src/bitgo/dash/DashTransaction.d.ts.map +1 -0
  47. package/dist/src/bitgo/dash/DashTransaction.js +109 -0
  48. package/dist/src/bitgo/dash/DashTransactionBuilder.d.ts +14 -0
  49. package/dist/src/bitgo/dash/DashTransactionBuilder.d.ts.map +1 -0
  50. package/dist/src/bitgo/dash/DashTransactionBuilder.js +28 -0
  51. package/dist/src/bitgo/dash/index.d.ts +4 -0
  52. package/dist/src/bitgo/dash/index.d.ts.map +1 -0
  53. package/dist/src/bitgo/dash/index.js +16 -0
  54. package/dist/src/bitgo/index.d.ts +17 -0
  55. package/dist/src/bitgo/index.d.ts.map +1 -0
  56. package/dist/src/bitgo/index.js +30 -0
  57. package/dist/src/bitgo/keyutil.d.ts +17 -0
  58. package/dist/src/bitgo/keyutil.d.ts.map +1 -0
  59. package/dist/src/bitgo/keyutil.js +37 -0
  60. package/dist/src/bitgo/nonStandardHalfSigned.d.ts +7 -0
  61. package/dist/src/bitgo/nonStandardHalfSigned.d.ts.map +1 -0
  62. package/dist/src/bitgo/nonStandardHalfSigned.js +56 -0
  63. package/dist/src/bitgo/outputScripts.d.ts +84 -0
  64. package/dist/src/bitgo/outputScripts.d.ts.map +1 -0
  65. package/dist/src/bitgo/outputScripts.js +297 -0
  66. package/dist/src/bitgo/parseInput.d.ts +92 -0
  67. package/dist/src/bitgo/parseInput.d.ts.map +1 -0
  68. package/dist/src/bitgo/parseInput.js +344 -0
  69. package/dist/src/bitgo/psbt/fromHalfSigned.d.ts +24 -0
  70. package/dist/src/bitgo/psbt/fromHalfSigned.d.ts.map +1 -0
  71. package/dist/src/bitgo/psbt/fromHalfSigned.js +91 -0
  72. package/dist/src/bitgo/psbt/scriptTypes.d.ts +6 -0
  73. package/dist/src/bitgo/psbt/scriptTypes.d.ts.map +1 -0
  74. package/dist/src/bitgo/psbt/scriptTypes.js +23 -0
  75. package/dist/src/bitgo/signature.d.ts +84 -0
  76. package/dist/src/bitgo/signature.d.ts.map +1 -0
  77. package/dist/src/bitgo/signature.js +222 -0
  78. package/dist/src/bitgo/tnumber.d.ts +9 -0
  79. package/dist/src/bitgo/tnumber.d.ts.map +1 -0
  80. package/dist/src/bitgo/tnumber.js +31 -0
  81. package/dist/src/bitgo/transaction.d.ts +29 -0
  82. package/dist/src/bitgo/transaction.d.ts.map +1 -0
  83. package/dist/src/bitgo/transaction.js +247 -0
  84. package/dist/src/bitgo/types.d.ts +5 -0
  85. package/dist/src/bitgo/types.d.ts.map +1 -0
  86. package/dist/src/bitgo/types.js +12 -0
  87. package/dist/src/bitgo/wallet/Psbt.d.ts +47 -0
  88. package/dist/src/bitgo/wallet/Psbt.d.ts.map +1 -0
  89. package/dist/src/bitgo/wallet/Psbt.js +232 -0
  90. package/dist/src/bitgo/wallet/Unspent.d.ts +47 -0
  91. package/dist/src/bitgo/wallet/Unspent.d.ts.map +1 -0
  92. package/dist/src/bitgo/wallet/Unspent.js +154 -0
  93. package/dist/src/bitgo/wallet/WalletKeys.d.ts +72 -0
  94. package/dist/src/bitgo/wallet/WalletKeys.d.ts.map +1 -0
  95. package/dist/src/bitgo/wallet/WalletKeys.js +104 -0
  96. package/dist/src/bitgo/wallet/WalletOutput.d.ts +18 -0
  97. package/dist/src/bitgo/wallet/WalletOutput.d.ts.map +1 -0
  98. package/dist/src/bitgo/wallet/WalletOutput.js +69 -0
  99. package/dist/src/bitgo/wallet/WalletScripts.d.ts +7 -0
  100. package/dist/src/bitgo/wallet/WalletScripts.d.ts.map +1 -0
  101. package/dist/src/bitgo/wallet/WalletScripts.js +15 -0
  102. package/dist/src/bitgo/wallet/WalletUnspentSigner.d.ts +19 -0
  103. package/dist/src/bitgo/wallet/WalletUnspentSigner.d.ts.map +1 -0
  104. package/dist/src/bitgo/wallet/WalletUnspentSigner.js +47 -0
  105. package/dist/src/bitgo/wallet/chains.d.ts +57 -0
  106. package/dist/src/bitgo/wallet/chains.d.ts.map +1 -0
  107. package/dist/src/bitgo/wallet/chains.js +106 -0
  108. package/dist/src/bitgo/wallet/index.d.ts +8 -0
  109. package/dist/src/bitgo/wallet/index.d.ts.map +1 -0
  110. package/dist/src/bitgo/wallet/index.js +20 -0
  111. package/dist/src/bitgo/zcash/ZcashBufferutils.d.ts +26 -0
  112. package/dist/src/bitgo/zcash/ZcashBufferutils.d.ts.map +1 -0
  113. package/dist/src/bitgo/zcash/ZcashBufferutils.js +157 -0
  114. package/dist/src/bitgo/zcash/ZcashPsbt.d.ts +36 -0
  115. package/dist/src/bitgo/zcash/ZcashPsbt.d.ts.map +1 -0
  116. package/dist/src/bitgo/zcash/ZcashPsbt.js +146 -0
  117. package/dist/src/bitgo/zcash/ZcashTransaction.d.ts +61 -0
  118. package/dist/src/bitgo/zcash/ZcashTransaction.d.ts.map +1 -0
  119. package/dist/src/bitgo/zcash/ZcashTransaction.js +341 -0
  120. package/dist/src/bitgo/zcash/ZcashTransactionBuilder.d.ts +21 -0
  121. package/dist/src/bitgo/zcash/ZcashTransactionBuilder.d.ts.map +1 -0
  122. package/dist/src/bitgo/zcash/ZcashTransactionBuilder.js +105 -0
  123. package/dist/src/bitgo/zcash/address.d.ts +8 -0
  124. package/dist/src/bitgo/zcash/address.d.ts.map +1 -0
  125. package/dist/src/bitgo/zcash/address.js +57 -0
  126. package/dist/src/bitgo/zcash/hashZip0244.d.ts +27 -0
  127. package/dist/src/bitgo/zcash/hashZip0244.d.ts.map +1 -0
  128. package/dist/src/bitgo/zcash/hashZip0244.js +184 -0
  129. package/dist/src/bitgo/zcash/index.d.ts +4 -0
  130. package/dist/src/bitgo/zcash/index.d.ts.map +1 -0
  131. package/dist/src/bitgo/zcash/index.js +16 -0
  132. package/dist/src/classify.d.ts +19 -0
  133. package/dist/src/classify.d.ts.map +1 -0
  134. package/dist/src/classify.js +89 -0
  135. package/dist/src/index.d.ts +13 -0
  136. package/dist/src/index.d.ts.map +1 -0
  137. package/dist/src/index.js +36 -0
  138. package/dist/src/networks.d.ts +119 -0
  139. package/dist/src/networks.d.ts.map +1 -0
  140. package/dist/src/networks.js +477 -0
  141. package/dist/src/noble_ecc.d.ts +28 -0
  142. package/dist/src/noble_ecc.d.ts.map +1 -0
  143. package/dist/src/noble_ecc.js +168 -0
  144. package/dist/src/payments/index.d.ts +4 -0
  145. package/dist/src/payments/index.d.ts.map +1 -0
  146. package/dist/src/payments/index.js +8 -0
  147. package/dist/src/payments/p2tr.d.ts +3 -0
  148. package/dist/src/payments/p2tr.d.ts.map +1 -0
  149. package/dist/src/payments/p2tr.js +348 -0
  150. package/dist/src/payments/p2tr_ns.d.ts +3 -0
  151. package/dist/src/payments/p2tr_ns.d.ts.map +1 -0
  152. package/dist/src/payments/p2tr_ns.js +134 -0
  153. package/dist/src/taproot.d.ts +141 -0
  154. package/dist/src/taproot.d.ts.map +1 -0
  155. package/dist/src/taproot.js +384 -0
  156. package/dist/src/templates/multisig/index.d.ts +4 -0
  157. package/dist/src/templates/multisig/index.d.ts.map +1 -0
  158. package/dist/src/templates/multisig/index.js +8 -0
  159. package/dist/src/templates/multisig/input.d.ts +7 -0
  160. package/dist/src/templates/multisig/input.d.ts.map +1 -0
  161. package/dist/src/templates/multisig/input.js +25 -0
  162. package/dist/src/templates/multisig/output.d.ts +7 -0
  163. package/dist/src/templates/multisig/output.d.ts.map +1 -0
  164. package/dist/src/templates/multisig/output.js +38 -0
  165. package/dist/src/templates/nulldata.d.ts +10 -0
  166. package/dist/src/templates/nulldata.d.ts.map +1 -0
  167. package/dist/src/templates/nulldata.js +17 -0
  168. package/dist/src/templates/pubkey/index.d.ts +4 -0
  169. package/dist/src/templates/pubkey/index.d.ts.map +1 -0
  170. package/dist/src/templates/pubkey/index.js +8 -0
  171. package/dist/src/templates/pubkey/input.d.ts +7 -0
  172. package/dist/src/templates/pubkey/input.d.ts.map +1 -0
  173. package/dist/src/templates/pubkey/input.js +14 -0
  174. package/dist/src/templates/pubkey/output.d.ts +7 -0
  175. package/dist/src/templates/pubkey/output.d.ts.map +1 -0
  176. package/dist/src/templates/pubkey/output.js +15 -0
  177. package/dist/src/templates/pubkeyhash/index.d.ts +4 -0
  178. package/dist/src/templates/pubkeyhash/index.d.ts.map +1 -0
  179. package/dist/src/templates/pubkeyhash/index.js +8 -0
  180. package/dist/src/templates/pubkeyhash/input.d.ts +7 -0
  181. package/dist/src/templates/pubkeyhash/input.d.ts.map +1 -0
  182. package/dist/src/templates/pubkeyhash/input.js +16 -0
  183. package/dist/src/templates/pubkeyhash/output.d.ts +6 -0
  184. package/dist/src/templates/pubkeyhash/output.d.ts.map +1 -0
  185. package/dist/src/templates/pubkeyhash/output.js +20 -0
  186. package/dist/src/templates/scripthash/index.d.ts +4 -0
  187. package/dist/src/templates/scripthash/index.d.ts.map +1 -0
  188. package/dist/src/templates/scripthash/index.js +8 -0
  189. package/dist/src/templates/scripthash/input.d.ts +6 -0
  190. package/dist/src/templates/scripthash/input.d.ts.map +1 -0
  191. package/dist/src/templates/scripthash/input.js +43 -0
  192. package/dist/src/templates/scripthash/output.d.ts +6 -0
  193. package/dist/src/templates/scripthash/output.d.ts.map +1 -0
  194. package/dist/src/templates/scripthash/output.js +15 -0
  195. package/dist/src/templates/taproot/index.d.ts +4 -0
  196. package/dist/src/templates/taproot/index.d.ts.map +1 -0
  197. package/dist/src/templates/taproot/index.js +8 -0
  198. package/dist/src/templates/taproot/input.d.ts +6 -0
  199. package/dist/src/templates/taproot/input.d.ts.map +1 -0
  200. package/dist/src/templates/taproot/input.js +23 -0
  201. package/dist/src/templates/taproot/output.d.ts +6 -0
  202. package/dist/src/templates/taproot/output.d.ts.map +1 -0
  203. package/dist/src/templates/taproot/output.js +15 -0
  204. package/dist/src/templates/taprootnofn/index.d.ts +4 -0
  205. package/dist/src/templates/taprootnofn/index.d.ts.map +1 -0
  206. package/dist/src/templates/taprootnofn/index.js +8 -0
  207. package/dist/src/templates/taprootnofn/input.d.ts +7 -0
  208. package/dist/src/templates/taprootnofn/input.d.ts.map +1 -0
  209. package/dist/src/templates/taprootnofn/input.js +24 -0
  210. package/dist/src/templates/taprootnofn/output.d.ts +7 -0
  211. package/dist/src/templates/taprootnofn/output.d.ts.map +1 -0
  212. package/dist/src/templates/taprootnofn/output.js +28 -0
  213. package/dist/src/templates/witnesscommitment/index.d.ts +3 -0
  214. package/dist/src/templates/witnesscommitment/index.d.ts.map +1 -0
  215. package/dist/src/templates/witnesscommitment/index.js +6 -0
  216. package/dist/src/templates/witnesscommitment/output.d.ts +8 -0
  217. package/dist/src/templates/witnesscommitment/output.d.ts.map +1 -0
  218. package/dist/src/templates/witnesscommitment/output.js +31 -0
  219. package/dist/src/templates/witnesspubkeyhash/index.d.ts +4 -0
  220. package/dist/src/templates/witnesspubkeyhash/index.d.ts.map +1 -0
  221. package/dist/src/templates/witnesspubkeyhash/index.js +8 -0
  222. package/dist/src/templates/witnesspubkeyhash/input.d.ts +7 -0
  223. package/dist/src/templates/witnesspubkeyhash/input.d.ts.map +1 -0
  224. package/dist/src/templates/witnesspubkeyhash/input.js +19 -0
  225. package/dist/src/templates/witnesspubkeyhash/output.d.ts +6 -0
  226. package/dist/src/templates/witnesspubkeyhash/output.d.ts.map +1 -0
  227. package/dist/src/templates/witnesspubkeyhash/output.js +15 -0
  228. package/dist/src/templates/witnessscripthash/index.d.ts +4 -0
  229. package/dist/src/templates/witnessscripthash/index.d.ts.map +1 -0
  230. package/dist/src/templates/witnessscripthash/index.js +8 -0
  231. package/dist/src/templates/witnessscripthash/input.d.ts +6 -0
  232. package/dist/src/templates/witnessscripthash/input.d.ts.map +1 -0
  233. package/dist/src/templates/witnessscripthash/input.js +35 -0
  234. package/dist/src/templates/witnessscripthash/output.d.ts +6 -0
  235. package/dist/src/templates/witnessscripthash/output.d.ts.map +1 -0
  236. package/dist/src/templates/witnessscripthash/output.js +15 -0
  237. package/dist/src/testutil/index.d.ts +3 -0
  238. package/dist/src/testutil/index.d.ts.map +1 -0
  239. package/dist/src/testutil/index.js +15 -0
  240. package/dist/src/testutil/keys.d.ts +10 -0
  241. package/dist/src/testutil/keys.d.ts.map +1 -0
  242. package/dist/src/testutil/keys.js +40 -0
  243. package/dist/src/testutil/mock.d.ts +21 -0
  244. package/dist/src/testutil/mock.d.ts.map +1 -0
  245. package/dist/src/testutil/mock.js +86 -0
  246. package/dist/src/transaction_builder.d.ts +47 -0
  247. package/dist/src/transaction_builder.d.ts.map +1 -0
  248. package/dist/src/transaction_builder.js +1084 -0
  249. package/package.json +87 -0
@@ -0,0 +1,384 @@
1
+ "use strict";
2
+ // Taproot-specific key aggregation and taptree logic as defined in:
3
+ // https://github.com/bitcoin/bips/blob/master/bip-0340.mediawiki
4
+ // https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.getTweakedOutputKey = exports.getTaptreeRoot = exports.getTapleafHash = exports.parseControlBlock = exports.parseTaprootWitness = exports.getControlBlock = exports.getHuffmanTaptree = exports.getDepthFirstTaptree = exports.tapTweakPubkey = exports.tapTweakPrivkey = exports.hashTapBranch = exports.hashTapLeaf = exports.serializeScriptSize = exports.aggregateMuSigPubkeys = exports.INITIAL_TAPSCRIPT_VERSION = exports.EVEN_Y_COORD_PREFIX = void 0;
7
+ const assert = require("assert");
8
+ const FastPriorityQueue = require("fastpriorityqueue");
9
+ const bitcoinjs_lib_1 = require("bitcoinjs-lib");
10
+ const varuint = require('varuint-bitcoin');
11
+ /**
12
+ * The 0x02 prefix indicating an even Y coordinate which is implicitly assumed
13
+ * on all 32 byte x-only pub keys as defined in BIP340.
14
+ */
15
+ exports.EVEN_Y_COORD_PREFIX = Buffer.of(0x02);
16
+ exports.INITIAL_TAPSCRIPT_VERSION = 0xc0;
17
+ /**
18
+ * Aggregates a list of public keys into a single MuSig2* public key
19
+ * according to the MuSig2 paper.
20
+ * @param ecc Elliptic curve implementation
21
+ * @param pubkeys The list of pub keys to aggregate
22
+ * @returns a 32 byte Buffer representing the aggregate key
23
+ */
24
+ function aggregateMuSigPubkeys(ecc, pubkeys) {
25
+ // TODO: Consider enforcing key uniqueness.
26
+ assert(pubkeys.length > 1, 'at least two pubkeys are required for musig key aggregation');
27
+ // Sort the keys in ascending order
28
+ pubkeys.sort(Buffer.compare);
29
+ // In MuSig all signers contribute key material to a single signing key,
30
+ // using the equation
31
+ //
32
+ // P = sum_i µ_i * P_i
33
+ //
34
+ // where `P_i` is the public key of the `i`th signer and `µ_i` is a so-called
35
+ // _MuSig coefficient_ computed according to the following equation
36
+ //
37
+ // L = H(P_1 || P_2 || ... || P_n)
38
+ // µ_i = H(L || P_i)
39
+ const L = bitcoinjs_lib_1.crypto.taggedHash('KeyAgg list', Buffer.concat(pubkeys));
40
+ const secondUniquePubkey = pubkeys.find((pubkey) => !pubkeys[0].equals(pubkey));
41
+ const tweakedPubkeys = pubkeys.map((pubkey) => {
42
+ const xyPubkey = Buffer.concat([exports.EVEN_Y_COORD_PREFIX, pubkey]);
43
+ if (secondUniquePubkey !== undefined && secondUniquePubkey.equals(pubkey)) {
44
+ // The second unique key in the pubkey list given to ''KeyAgg'' (as well
45
+ // as any keys identical to this key) gets the constant KeyAgg
46
+ // coefficient 1 which saves an exponentiation (see the MuSig2* appendix
47
+ // in the MuSig2 paper).
48
+ return xyPubkey;
49
+ }
50
+ const c = bitcoinjs_lib_1.crypto.taggedHash('KeyAgg coefficient', Buffer.concat([L, pubkey]));
51
+ const tweakedPubkey = ecc.pointMultiply(xyPubkey, c);
52
+ if (!tweakedPubkey) {
53
+ throw new Error('Failed to multiply pubkey by coefficient');
54
+ }
55
+ return tweakedPubkey;
56
+ });
57
+ const aggregatePubkey = tweakedPubkeys.reduce((prev, curr) => {
58
+ const next = ecc.pointAdd(prev, curr);
59
+ if (!next)
60
+ throw new Error('Failed to sum pubkeys');
61
+ return next;
62
+ });
63
+ return aggregatePubkey.slice(1);
64
+ }
65
+ exports.aggregateMuSigPubkeys = aggregateMuSigPubkeys;
66
+ /**
67
+ * Encodes the length of a script as a bitcoin variable length integer.
68
+ * @param script
69
+ * @returns
70
+ */
71
+ function serializeScriptSize(script) {
72
+ return varuint.encode(script.length);
73
+ }
74
+ exports.serializeScriptSize = serializeScriptSize;
75
+ /**
76
+ * Gets a tapleaf tagged hash from a script.
77
+ * @param script
78
+ * @returns
79
+ */
80
+ function hashTapLeaf(script, leafVersion = exports.INITIAL_TAPSCRIPT_VERSION) {
81
+ const size = serializeScriptSize(script);
82
+ return bitcoinjs_lib_1.crypto.taggedHash('TapLeaf', Buffer.concat([Buffer.of(leafVersion), size, script]));
83
+ }
84
+ exports.hashTapLeaf = hashTapLeaf;
85
+ /**
86
+ * Creates a lexicographically sorted tapbranch from two child taptree nodes
87
+ * and returns its tagged hash.
88
+ * @param child1
89
+ * @param child2
90
+ * @returns the tagged tapbranch hash
91
+ */
92
+ function hashTapBranch(child1, child2) {
93
+ // sort the children lexicographically
94
+ const sortedChildren = [child1, child2].sort(Buffer.compare);
95
+ return bitcoinjs_lib_1.crypto.taggedHash('TapBranch', Buffer.concat(sortedChildren));
96
+ }
97
+ exports.hashTapBranch = hashTapBranch;
98
+ function calculateTapTweak(pubkey, taptreeRoot) {
99
+ if (taptreeRoot) {
100
+ return bitcoinjs_lib_1.crypto.taggedHash('TapTweak', Buffer.concat([pubkey, taptreeRoot]));
101
+ }
102
+ // If the spending conditions do not require a script path, the output key should commit to an
103
+ // unspendable script path instead of having no script path.
104
+ // https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki#cite_note-22
105
+ return bitcoinjs_lib_1.crypto.taggedHash('TapTweak', Buffer.from(pubkey));
106
+ }
107
+ /**
108
+ * Tweaks a privkey, using the tagged hash of its pubkey, and (optionally) a taptree root
109
+ * @param ecc Elliptic curve implementation
110
+ * @param pubkey public key, used to calculate the tweak
111
+ * @param privkey the privkey to tweak
112
+ * @param taptreeRoot the taptree root tagged hash
113
+ * @returns {Buffer} the tweaked privkey
114
+ */
115
+ function tapTweakPrivkey(ecc, pubkey, privkey, taptreeRoot) {
116
+ const tapTweak = calculateTapTweak(pubkey, taptreeRoot);
117
+ const point = ecc.pointFromScalar(privkey);
118
+ if (!point)
119
+ throw new Error('Invalid private key');
120
+ if (point[0] % 2 === 1)
121
+ privkey = ecc.privateNegate(privkey);
122
+ const result = ecc.privateAdd(privkey, tapTweak);
123
+ if (!result)
124
+ throw new Error('Invalid private key');
125
+ return result;
126
+ }
127
+ exports.tapTweakPrivkey = tapTweakPrivkey;
128
+ /**
129
+ * Tweaks an internal pubkey, using the tagged hash of itself, and (optionally) a taptree root
130
+ * @param ecc Elliptic curve implementation
131
+ * @param pubkey the internal pubkey to tweak
132
+ * @param taptreeRoot the taptree root tagged hash
133
+ * @returns {TweakedPubkey} the tweaked pubkey
134
+ */
135
+ function tapTweakPubkey(ecc, pubkey, taptreeRoot) {
136
+ const tapTweak = calculateTapTweak(pubkey, taptreeRoot);
137
+ const result = ecc.xOnlyPointAddTweak(pubkey, tapTweak);
138
+ if (!result)
139
+ throw new Error('Invalid pubkey');
140
+ return result;
141
+ }
142
+ exports.tapTweakPubkey = tapTweakPubkey;
143
+ function recurseTaptree(leaves, targetDepth = 0) {
144
+ const { value, done } = leaves.next();
145
+ assert(!done, 'insufficient leaves to reconstruct tap tree');
146
+ const [index, leaf] = value;
147
+ const tree = {
148
+ root: hashTapLeaf(leaf.script, leaf.leafVersion),
149
+ paths: [],
150
+ };
151
+ tree.paths[index] = [];
152
+ for (let depth = leaf.depth; depth > targetDepth; depth--) {
153
+ const sibling = recurseTaptree(leaves, depth);
154
+ tree.paths.forEach((path) => path.push(sibling.root));
155
+ sibling.paths.forEach((path) => path.push(tree.root));
156
+ tree.root = hashTapBranch(tree.root, sibling.root);
157
+ // Merge disjoint sparse arrays of paths into tree.paths
158
+ Object.assign(tree.paths, sibling.paths);
159
+ }
160
+ return tree;
161
+ }
162
+ /**
163
+ * Gets the root hash and hash-paths of a taptree from the depth-first
164
+ * construction used in BIP-0371 PSBTs
165
+ * @param tree
166
+ * @returns {Taptree} the tree, represented by its root hash, and the paths to
167
+ * that root from each of the input scripts
168
+ */
169
+ function getDepthFirstTaptree(tree) {
170
+ const iter = tree.leaves.entries();
171
+ const ret = recurseTaptree(iter);
172
+ assert(iter.next().done, 'invalid tap tree, no path to some leaves');
173
+ return ret;
174
+ }
175
+ exports.getDepthFirstTaptree = getDepthFirstTaptree;
176
+ /**
177
+ * Gets the root hash of a taptree using a weighted Huffman construction from a
178
+ * list of scripts and corresponding weights.
179
+ * @param scripts
180
+ * @param weights
181
+ * @returns {Taptree} the tree, represented by its root hash, and the paths to that root from each of the input scripts
182
+ */
183
+ function getHuffmanTaptree(scripts, weights) {
184
+ assert(scripts.length > 0, 'at least one script is required to construct a tap tree');
185
+ // Create a queue/heap of the provided scripts prioritized according to their
186
+ // corresponding weights.
187
+ const queue = new FastPriorityQueue((a, b) => {
188
+ return a.weight < b.weight;
189
+ });
190
+ scripts.forEach((script, index) => {
191
+ const weight = weights[index] || 1;
192
+ assert(weight > 0, 'script weight must be a positive value');
193
+ queue.add({
194
+ weight,
195
+ taggedHash: hashTapLeaf(script),
196
+ paths: { [index]: [] },
197
+ });
198
+ });
199
+ // Now that we have a queue of weighted scripts, we begin a loop whereby we
200
+ // remove the two lowest weighted items from the queue. We create a tap branch
201
+ // node from the two items, and add the branch back to the queue with the
202
+ // combined weight of both its children. Each loop reduces the number of items
203
+ // in the queue by one, and we repeat until we are left with only one item -
204
+ // this becomes the tap tree root.
205
+ //
206
+ // For example, if we begin with scripts A, B, C, D with weights 6, 3, 1, 1
207
+ // After first loop: A(6), B(3), CD(1 + 1)
208
+ // After second loop: A(6), B[CD](3 + 2)
209
+ // Final loop: A[B[CD]](6+5)
210
+ // The final tree will look like:
211
+ //
212
+ // A[B[CD]]
213
+ // / \
214
+ // A B[CD]
215
+ // / \
216
+ // B [CD]
217
+ // / \
218
+ // C D
219
+ //
220
+ // This ensures that the spending conditions we believe to have the highest
221
+ // probability of being used are further up the tree than less likely scripts,
222
+ // thereby reducing the size of the merkle proofs for the more likely scripts.
223
+ while (queue.size > 1) {
224
+ // We can safely expect two polls to return non-null elements since we've
225
+ // checked that the queue has at least two elements before looping.
226
+ const child1 = queue.poll();
227
+ const child2 = queue.poll();
228
+ Object.values(child1.paths).forEach((path) => path.push(child2.taggedHash));
229
+ Object.values(child2.paths).forEach((path) => path.push(child1.taggedHash));
230
+ queue.add({
231
+ taggedHash: hashTapBranch(child1.taggedHash, child2.taggedHash),
232
+ weight: child1.weight + child2.weight,
233
+ paths: { ...child1.paths, ...child2.paths },
234
+ });
235
+ }
236
+ // After the while loop above completes we should have exactly one element
237
+ // remaining in the queue, which we can safely extract below.
238
+ const rootNode = queue.poll();
239
+ const paths = Object.entries(rootNode.paths).reduce((acc, [index, path]) => {
240
+ acc[Number(index)] = path; // TODO: Why doesn't TS know it's a number?
241
+ return acc;
242
+ }, Array(scripts.length));
243
+ return { root: rootNode.taggedHash, paths };
244
+ }
245
+ exports.getHuffmanTaptree = getHuffmanTaptree;
246
+ function getControlBlock(parity, pubkey, path, leafVersion = exports.INITIAL_TAPSCRIPT_VERSION) {
247
+ const parityVersion = leafVersion + parity;
248
+ return Buffer.concat([Buffer.of(parityVersion), pubkey, ...path]);
249
+ }
250
+ exports.getControlBlock = getControlBlock;
251
+ /**
252
+ * Parses a taproot witness stack and extracts key data elements.
253
+ * @param witnessStack
254
+ * @returns {ScriptPathWitness|KeyPathWitness} an object representing the
255
+ * parsed witness for a script path or key path spend.
256
+ * @throws {Error} if the witness stack does not conform to the BIP 341 script validation rules
257
+ */
258
+ function parseTaprootWitness(witnessStack) {
259
+ let annex;
260
+ if (witnessStack.length >= 2 && witnessStack[witnessStack.length - 1][0] === 0x50) {
261
+ // If there are at least two witness elements, and the first byte of the last element is
262
+ // 0x50, this last element is called annex a and is removed from the witness stack
263
+ annex = witnessStack[witnessStack.length - 1];
264
+ witnessStack = witnessStack.slice(0, -1);
265
+ }
266
+ if (witnessStack.length < 1) {
267
+ throw new Error('witness stack must have at least one element');
268
+ }
269
+ else if (witnessStack.length === 1) {
270
+ // key path spend
271
+ const signature = witnessStack[0];
272
+ if (!bitcoinjs_lib_1.script.isCanonicalSchnorrSignature(signature)) {
273
+ throw new Error('invalid signature');
274
+ }
275
+ return { spendType: 'Key', signature, annex };
276
+ }
277
+ // script path spend
278
+ // second to last element is the tapscript
279
+ const tapscript = witnessStack[witnessStack.length - 2];
280
+ const tapscriptChunks = bitcoinjs_lib_1.script.decompile(tapscript);
281
+ if (!tapscriptChunks || tapscriptChunks.length === 0) {
282
+ throw new Error('tapscript is not a valid script');
283
+ }
284
+ // The last stack element is called the control block c, and must have length 33 + 32m,
285
+ // for a value of m that is an integer between 0 and 128, inclusive
286
+ const controlBlock = witnessStack[witnessStack.length - 1];
287
+ if (controlBlock.length < 33 || controlBlock.length > 33 + 32 * 128 || controlBlock.length % 32 !== 1) {
288
+ throw new Error('invalid control block length');
289
+ }
290
+ return {
291
+ spendType: 'Script',
292
+ scriptSig: witnessStack.slice(0, -2),
293
+ tapscript,
294
+ controlBlock,
295
+ annex,
296
+ };
297
+ }
298
+ exports.parseTaprootWitness = parseTaprootWitness;
299
+ /**
300
+ * Parses a taproot control block.
301
+ * @param ecc Elliptic curve implementation
302
+ * @param controlBlock the control block to parse
303
+ * @returns {ControlBlock} the parsed control block
304
+ * @throws {Error} if the witness stack does not conform to the BIP 341 script validation rules
305
+ */
306
+ function parseControlBlock(ecc, controlBlock) {
307
+ if ((controlBlock.length - 1) % 32 !== 0) {
308
+ throw new TypeError('Invalid control block length');
309
+ }
310
+ const parity = controlBlock[0] & 0x01;
311
+ // Let p = c[1:33] and let P = lift_x(int(p)) where lift_x and [:] are defined as in BIP340.
312
+ // Fail if this point is not on the curve
313
+ const internalPubkey = controlBlock.slice(1, 33);
314
+ if (!ecc.isXOnlyPoint(internalPubkey)) {
315
+ throw new Error('internal pubkey is not an EC point');
316
+ }
317
+ // The leaf version cannot be 0x50 as that would result in ambiguity with the annex.
318
+ const leafVersion = controlBlock[0] & 0xfe;
319
+ if (leafVersion === 0x50) {
320
+ throw new Error('invalid leaf version');
321
+ }
322
+ const path = [];
323
+ for (let j = 33; j < controlBlock.length; j += 32) {
324
+ path.push(controlBlock.slice(j, j + 32));
325
+ }
326
+ return {
327
+ parity,
328
+ internalPubkey,
329
+ leafVersion,
330
+ path,
331
+ };
332
+ }
333
+ exports.parseControlBlock = parseControlBlock;
334
+ /**
335
+ * Calculates the tapleaf hash from a control block and script.
336
+ * @param ecc Elliptic curve implementation
337
+ * @param controlBlock the control block, either raw or parsed
338
+ * @param tapscript the leaf script corresdponding to the control block
339
+ * @returns {Buffer} the tapleaf hash
340
+ */
341
+ function getTapleafHash(ecc, controlBlock, tapscript) {
342
+ if (Buffer.isBuffer(controlBlock)) {
343
+ controlBlock = parseControlBlock(ecc, controlBlock);
344
+ }
345
+ const { leafVersion } = controlBlock;
346
+ return bitcoinjs_lib_1.crypto.taggedHash('TapLeaf', Buffer.concat([Buffer.of(leafVersion), serializeScriptSize(tapscript), tapscript]));
347
+ }
348
+ exports.getTapleafHash = getTapleafHash;
349
+ /**
350
+ * Calculates the taptree root hash from a control block and script.
351
+ * @param ecc Elliptic curve implementation
352
+ * @param controlBlock the control block, either raw or parsed
353
+ * @param tapscript the leaf script corresdponding to the control block
354
+ * @param tapleafHash the leaf hash if already calculated
355
+ * @returns {Buffer} the taptree root hash
356
+ */
357
+ function getTaptreeRoot(ecc, controlBlock, tapscript, tapleafHash) {
358
+ if (Buffer.isBuffer(controlBlock)) {
359
+ controlBlock = parseControlBlock(ecc, controlBlock);
360
+ }
361
+ const { path } = controlBlock;
362
+ tapleafHash = tapleafHash || getTapleafHash(ecc, controlBlock, tapscript);
363
+ // `taptreeMerkleHash` begins as our tapscript tapleaf hash and its value iterates
364
+ // through its parent tapbranch hashes until it ends up as the taptree root hash
365
+ let taptreeMerkleHash = tapleafHash;
366
+ for (const taptreeSiblingHash of path) {
367
+ taptreeMerkleHash =
368
+ Buffer.compare(taptreeMerkleHash, taptreeSiblingHash) === -1
369
+ ? bitcoinjs_lib_1.crypto.taggedHash('TapBranch', Buffer.concat([taptreeMerkleHash, taptreeSiblingHash]))
370
+ : bitcoinjs_lib_1.crypto.taggedHash('TapBranch', Buffer.concat([taptreeSiblingHash, taptreeMerkleHash]));
371
+ }
372
+ return taptreeMerkleHash;
373
+ }
374
+ exports.getTaptreeRoot = getTaptreeRoot;
375
+ function getTweakedOutputKey(payment) {
376
+ var _a;
377
+ assert(payment.output);
378
+ if (payment.output.length === 34) {
379
+ return (_a = payment.output) === null || _a === void 0 ? void 0 : _a.subarray(2);
380
+ }
381
+ throw new Error(`invalid p2tr tweaked output key size ${payment.output.length}`);
382
+ }
383
+ exports.getTweakedOutputKey = getTweakedOutputKey;
384
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"taproot.js","sourceRoot":"","sources":["../../src/taproot.ts"],"names":[],"mappings":"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","sourcesContent":["// Taproot-specific key aggregation and taptree logic as defined in:\n// https://github.com/bitcoin/bips/blob/master/bip-0340.mediawiki\n// https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki\n\nimport { TapTree as PsbtTapTree, TapLeaf as PsbtTapLeaf } from 'bip174/src/lib/interfaces';\nimport assert = require('assert');\nimport FastPriorityQueue = require('fastpriorityqueue');\nimport { script as bscript, crypto as bcrypto, payments as bpayments } from 'bitcoinjs-lib';\nconst varuint = require('varuint-bitcoin');\n\n/**\n * The 0x02 prefix indicating an even Y coordinate which is implicitly assumed\n * on all 32 byte x-only pub keys as defined in BIP340.\n */\nexport const EVEN_Y_COORD_PREFIX = Buffer.of(0x02);\nexport const INITIAL_TAPSCRIPT_VERSION = 0xc0;\n\nexport interface TinySecp256k1Interface {\n  isXOnlyPoint(p: Uint8Array): boolean;\n  xOnlyPointAddTweak(p: Uint8Array, tweak: Uint8Array): XOnlyPointAddTweakResult | null;\n  pointFromScalar(sk: Uint8Array, compressed?: boolean): Uint8Array | null;\n  pointMultiply(a: Uint8Array, b: Uint8Array): Uint8Array | null;\n  pointAdd(a: Uint8Array, b: Uint8Array): Uint8Array | null;\n  privateAdd(d: Uint8Array, tweak: Uint8Array): Uint8Array | null;\n  privateNegate(d: Uint8Array): Uint8Array;\n}\n\n/**\n * Aggregates a list of public keys into a single MuSig2* public key\n * according to the MuSig2 paper.\n * @param ecc Elliptic curve implementation\n * @param pubkeys The list of pub keys to aggregate\n * @returns a 32 byte Buffer representing the aggregate key\n */\nexport function aggregateMuSigPubkeys(ecc: TinySecp256k1Interface, pubkeys: Buffer[]): Uint8Array {\n  // TODO: Consider enforcing key uniqueness.\n  assert(pubkeys.length > 1, 'at least two pubkeys are required for musig key aggregation');\n\n  // Sort the keys in ascending order\n  pubkeys.sort(Buffer.compare);\n\n  // In MuSig all signers contribute key material to a single signing key,\n  // using the equation\n  //\n  //     P = sum_i µ_i * P_i\n  //\n  // where `P_i` is the public key of the `i`th signer and `µ_i` is a so-called\n  // _MuSig coefficient_ computed according to the following equation\n  //\n  // L = H(P_1 || P_2 || ... || P_n)\n  // µ_i = H(L || P_i)\n\n  const L = bcrypto.taggedHash('KeyAgg list', Buffer.concat(pubkeys));\n\n  const secondUniquePubkey = pubkeys.find((pubkey) => !pubkeys[0].equals(pubkey));\n\n  const tweakedPubkeys: Uint8Array[] = pubkeys.map((pubkey) => {\n    const xyPubkey = Buffer.concat([EVEN_Y_COORD_PREFIX, pubkey]);\n\n    if (secondUniquePubkey !== undefined && secondUniquePubkey.equals(pubkey)) {\n      // The second unique key in the pubkey list given to ''KeyAgg'' (as well\n      // as any keys identical to this key) gets the constant KeyAgg\n      // coefficient 1 which saves an exponentiation (see the MuSig2* appendix\n      // in the MuSig2 paper).\n      return xyPubkey;\n    }\n\n    const c = bcrypto.taggedHash('KeyAgg coefficient', Buffer.concat([L, pubkey]));\n\n    const tweakedPubkey = ecc.pointMultiply(xyPubkey, c);\n    if (!tweakedPubkey) {\n      throw new Error('Failed to multiply pubkey by coefficient');\n    }\n    return tweakedPubkey;\n  });\n  const aggregatePubkey = tweakedPubkeys.reduce((prev, curr) => {\n    const next = ecc.pointAdd(prev, curr);\n    if (!next) throw new Error('Failed to sum pubkeys');\n    return next;\n  });\n\n  return aggregatePubkey.slice(1);\n}\n\n/**\n * Encodes the length of a script as a bitcoin variable length integer.\n * @param script\n * @returns\n */\nexport function serializeScriptSize(script: Buffer): Buffer {\n  return varuint.encode(script.length);\n}\n\n/**\n * Gets a tapleaf tagged hash from a script.\n * @param script\n * @returns\n */\nexport function hashTapLeaf(script: Buffer, leafVersion = INITIAL_TAPSCRIPT_VERSION): Buffer {\n  const size = serializeScriptSize(script);\n  return bcrypto.taggedHash('TapLeaf', Buffer.concat([Buffer.of(leafVersion), size, script]));\n}\n\n/**\n * Creates a lexicographically sorted tapbranch from two child taptree nodes\n * and returns its tagged hash.\n * @param child1\n * @param child2\n * @returns the tagged tapbranch hash\n */\nexport function hashTapBranch(child1: Buffer, child2: Buffer): Buffer {\n  // sort the children lexicographically\n  const sortedChildren = [child1, child2].sort(Buffer.compare);\n\n  return bcrypto.taggedHash('TapBranch', Buffer.concat(sortedChildren));\n}\n\nfunction calculateTapTweak(pubkey: Uint8Array, taptreeRoot?: Uint8Array): Uint8Array {\n  if (taptreeRoot) {\n    return bcrypto.taggedHash('TapTweak', Buffer.concat([pubkey, taptreeRoot]));\n  }\n  // If the spending conditions do not require a script path, the output key should commit to an\n  // unspendable script path instead of having no script path.\n  // https://github.com/bitcoin/bips/blob/master/bip-0341.mediawiki#cite_note-22\n  return bcrypto.taggedHash('TapTweak', Buffer.from(pubkey));\n}\n\n/**\n * Tweaks a privkey, using the tagged hash of its pubkey, and (optionally) a taptree root\n * @param ecc Elliptic curve implementation\n * @param pubkey public key, used to calculate the tweak\n * @param privkey the privkey to tweak\n * @param taptreeRoot the taptree root tagged hash\n * @returns {Buffer} the tweaked privkey\n */\nexport function tapTweakPrivkey(\n  ecc: TinySecp256k1Interface,\n  pubkey: Uint8Array,\n  privkey: Uint8Array,\n  taptreeRoot?: Uint8Array\n): Uint8Array {\n  const tapTweak = calculateTapTweak(pubkey, taptreeRoot);\n\n  const point = ecc.pointFromScalar(privkey);\n  if (!point) throw new Error('Invalid private key');\n  if (point[0] % 2 === 1) privkey = ecc.privateNegate(privkey);\n  const result = ecc.privateAdd(privkey, tapTweak);\n  if (!result) throw new Error('Invalid private key');\n  return result;\n}\n\nexport interface XOnlyPointAddTweakResult {\n  parity: 1 | 0;\n  xOnlyPubkey: Uint8Array;\n}\n\n/**\n * Tweaks an internal pubkey, using the tagged hash of itself, and (optionally) a taptree root\n * @param ecc Elliptic curve implementation\n * @param pubkey the internal pubkey to tweak\n * @param taptreeRoot the taptree root tagged hash\n * @returns {TweakedPubkey} the tweaked pubkey\n */\nexport function tapTweakPubkey(\n  ecc: TinySecp256k1Interface,\n  pubkey: Uint8Array,\n  taptreeRoot?: Buffer\n): XOnlyPointAddTweakResult {\n  const tapTweak = calculateTapTweak(pubkey, taptreeRoot);\n  const result = ecc.xOnlyPointAddTweak(pubkey, tapTweak);\n  if (!result) throw new Error('Invalid pubkey');\n  return result;\n}\n\nexport interface Taptree {\n  root: Buffer;\n  paths: Buffer[][];\n}\n\ninterface WeightedTapScript {\n  /** A TapLeaf or TapBranch tagged hash */\n  taggedHash: Buffer;\n  weight: number;\n  paths: {\n    [index: number]: Buffer[];\n  };\n}\n\nfunction recurseTaptree(leaves: Iterator<[number, PsbtTapLeaf]>, targetDepth = 0): Taptree {\n  const { value, done } = leaves.next();\n  assert(!done, 'insufficient leaves to reconstruct tap tree');\n  const [index, leaf] = value;\n  const tree: Taptree = {\n    root: hashTapLeaf(leaf.script, leaf.leafVersion),\n    paths: [],\n  };\n  tree.paths[index] = [];\n  for (let depth = leaf.depth; depth > targetDepth; depth--) {\n    const sibling = recurseTaptree(leaves, depth);\n    tree.paths.forEach((path) => path.push(sibling.root));\n    sibling.paths.forEach((path) => path.push(tree.root));\n    tree.root = hashTapBranch(tree.root, sibling.root);\n    // Merge disjoint sparse arrays of paths into tree.paths\n    Object.assign(tree.paths, sibling.paths);\n  }\n  return tree;\n}\n\n/**\n * Gets the root hash and hash-paths of a taptree from the depth-first\n * construction used in BIP-0371 PSBTs\n * @param tree\n * @returns {Taptree} the tree, represented by its root hash, and the paths to\n * that root from each of the input scripts\n */\nexport function getDepthFirstTaptree(tree: PsbtTapTree): Taptree {\n  const iter = tree.leaves.entries();\n  const ret = recurseTaptree(iter);\n  assert(iter.next().done, 'invalid tap tree, no path to some leaves');\n  return ret;\n}\n\n/**\n * Gets the root hash of a taptree using a weighted Huffman construction from a\n * list of scripts and corresponding weights.\n * @param scripts\n * @param weights\n * @returns {Taptree} the tree, represented by its root hash, and the paths to that root from each of the input scripts\n */\nexport function getHuffmanTaptree(scripts: Buffer[], weights: Array<number | undefined>): Taptree {\n  assert(scripts.length > 0, 'at least one script is required to construct a tap tree');\n\n  // Create a queue/heap of the provided scripts prioritized according to their\n  // corresponding weights.\n  const queue = new FastPriorityQueue<WeightedTapScript>((a, b): boolean => {\n    return a.weight < b.weight;\n  });\n  scripts.forEach((script, index) => {\n    const weight = weights[index] || 1;\n    assert(weight > 0, 'script weight must be a positive value');\n\n    queue.add({\n      weight,\n      taggedHash: hashTapLeaf(script),\n      paths: { [index]: [] },\n    });\n  });\n\n  // Now that we have a queue of weighted scripts, we begin a loop whereby we\n  // remove the two lowest weighted items from the queue. We create a tap branch\n  // node from the two items, and add the branch back to the queue with the\n  // combined weight of both its children. Each loop reduces the number of items\n  // in the queue by one, and we repeat until we are left with only one item -\n  // this becomes the tap tree root.\n  //\n  // For example, if we begin with scripts A, B, C, D with weights 6, 3, 1, 1\n  // After first loop: A(6), B(3), CD(1 + 1)\n  // After second loop: A(6), B[CD](3 + 2)\n  // Final loop: A[B[CD]](6+5)\n  // The final tree will look like:\n  //\n  //        A[B[CD]]\n  //       /        \\\n  //      A         B[CD]\n  //               /     \\\n  //              B      [CD]\n  //                    /    \\\n  //                   C      D\n  //\n  // This ensures that the spending conditions we believe to have the highest\n  // probability of being used are further up the tree than less likely scripts,\n  // thereby reducing the size of the merkle proofs for the more likely scripts.\n  while (queue.size > 1) {\n    // We can safely expect two polls to return non-null elements since we've\n    // checked that the queue has at least two elements before looping.\n    const child1 = queue.poll() as WeightedTapScript;\n    const child2 = queue.poll() as WeightedTapScript;\n\n    Object.values(child1.paths).forEach((path) => path.push(child2.taggedHash));\n    Object.values(child2.paths).forEach((path) => path.push(child1.taggedHash));\n\n    queue.add({\n      taggedHash: hashTapBranch(child1.taggedHash, child2.taggedHash),\n      weight: child1.weight + child2.weight,\n      paths: { ...child1.paths, ...child2.paths },\n    });\n  }\n\n  // After the while loop above completes we should have exactly one element\n  // remaining in the queue, which we can safely extract below.\n  const rootNode = queue.poll() as WeightedTapScript;\n\n  const paths = Object.entries(rootNode.paths).reduce((acc, [index, path]) => {\n    acc[Number(index)] = path; // TODO: Why doesn't TS know it's a number?\n    return acc;\n  }, Array(scripts.length));\n  return { root: rootNode.taggedHash, paths };\n}\n\nexport function getControlBlock(\n  parity: 0 | 1,\n  pubkey: Uint8Array,\n  path: Buffer[],\n  leafVersion = INITIAL_TAPSCRIPT_VERSION\n): Buffer {\n  const parityVersion = leafVersion + parity;\n\n  return Buffer.concat([Buffer.of(parityVersion), pubkey, ...path]);\n}\n\nexport interface KeyPathWitness {\n  spendType: 'Key';\n  signature: Buffer;\n  annex?: Buffer;\n}\n\nexport interface ScriptPathWitness {\n  spendType: 'Script';\n  scriptSig: Buffer[];\n  tapscript: Buffer;\n  controlBlock: Buffer;\n  annex?: Buffer;\n}\n\nexport interface ControlBlock {\n  parity: number;\n  internalPubkey: Buffer;\n  leafVersion: number;\n  path: Buffer[];\n}\n\n/**\n * Parses a taproot witness stack and extracts key data elements.\n * @param witnessStack\n * @returns {ScriptPathWitness|KeyPathWitness} an object representing the\n * parsed witness for a script path or key path spend.\n * @throws {Error} if the witness stack does not conform to the BIP 341 script validation rules\n */\nexport function parseTaprootWitness(witnessStack: Buffer[]): ScriptPathWitness | KeyPathWitness {\n  let annex;\n  if (witnessStack.length >= 2 && witnessStack[witnessStack.length - 1][0] === 0x50) {\n    // If there are at least two witness elements, and the first byte of the last element is\n    // 0x50, this last element is called annex a and is removed from the witness stack\n    annex = witnessStack[witnessStack.length - 1];\n    witnessStack = witnessStack.slice(0, -1);\n  }\n\n  if (witnessStack.length < 1) {\n    throw new Error('witness stack must have at least one element');\n  } else if (witnessStack.length === 1) {\n    // key path spend\n    const signature = witnessStack[0];\n    if (!bscript.isCanonicalSchnorrSignature(signature)) {\n      throw new Error('invalid signature');\n    }\n    return { spendType: 'Key', signature, annex };\n  }\n\n  // script path spend\n  // second to last element is the tapscript\n  const tapscript = witnessStack[witnessStack.length - 2];\n  const tapscriptChunks = bscript.decompile(tapscript);\n\n  if (!tapscriptChunks || tapscriptChunks.length === 0) {\n    throw new Error('tapscript is not a valid script');\n  }\n\n  // The last stack element is called the control block c, and must have length 33 + 32m,\n  // for a value of m that is an integer between 0 and 128, inclusive\n  const controlBlock = witnessStack[witnessStack.length - 1];\n  if (controlBlock.length < 33 || controlBlock.length > 33 + 32 * 128 || controlBlock.length % 32 !== 1) {\n    throw new Error('invalid control block length');\n  }\n\n  return {\n    spendType: 'Script',\n    scriptSig: witnessStack.slice(0, -2),\n    tapscript,\n    controlBlock,\n    annex,\n  };\n}\n\n/**\n * Parses a taproot control block.\n * @param ecc Elliptic curve implementation\n * @param controlBlock the control block to parse\n * @returns {ControlBlock} the parsed control block\n * @throws {Error} if the witness stack does not conform to the BIP 341 script validation rules\n */\nexport function parseControlBlock(ecc: TinySecp256k1Interface, controlBlock: Buffer): ControlBlock {\n  if ((controlBlock.length - 1) % 32 !== 0) {\n    throw new TypeError('Invalid control block length');\n  }\n\n  const parity = controlBlock[0] & 0x01;\n\n  // Let p = c[1:33] and let P = lift_x(int(p)) where lift_x and [:] are defined as in BIP340.\n  // Fail if this point is not on the curve\n  const internalPubkey = controlBlock.slice(1, 33);\n  if (!ecc.isXOnlyPoint(internalPubkey)) {\n    throw new Error('internal pubkey is not an EC point');\n  }\n\n  // The leaf version cannot be 0x50 as that would result in ambiguity with the annex.\n  const leafVersion = controlBlock[0] & 0xfe;\n  if (leafVersion === 0x50) {\n    throw new Error('invalid leaf version');\n  }\n\n  const path: Buffer[] = [];\n  for (let j = 33; j < controlBlock.length; j += 32) {\n    path.push(controlBlock.slice(j, j + 32));\n  }\n\n  return {\n    parity,\n    internalPubkey,\n    leafVersion,\n    path,\n  };\n}\n\n/**\n * Calculates the tapleaf hash from a control block and script.\n * @param ecc Elliptic curve implementation\n * @param controlBlock the control block, either raw or parsed\n * @param tapscript the leaf script corresdponding to the control block\n * @returns {Buffer} the tapleaf hash\n */\nexport function getTapleafHash(\n  ecc: TinySecp256k1Interface,\n  controlBlock: Buffer | ControlBlock,\n  tapscript: Buffer\n): Buffer {\n  if (Buffer.isBuffer(controlBlock)) {\n    controlBlock = parseControlBlock(ecc, controlBlock);\n  }\n  const { leafVersion } = controlBlock;\n\n  return bcrypto.taggedHash(\n    'TapLeaf',\n    Buffer.concat([Buffer.of(leafVersion), serializeScriptSize(tapscript), tapscript])\n  );\n}\n\n/**\n * Calculates the taptree root hash from a control block and script.\n * @param ecc Elliptic curve implementation\n * @param controlBlock the control block, either raw or parsed\n * @param tapscript the leaf script corresdponding to the control block\n * @param tapleafHash the leaf hash if already calculated\n * @returns {Buffer} the taptree root hash\n */\nexport function getTaptreeRoot(\n  ecc: TinySecp256k1Interface,\n  controlBlock: Buffer | ControlBlock,\n  tapscript: Buffer,\n  tapleafHash?: Buffer\n): Buffer {\n  if (Buffer.isBuffer(controlBlock)) {\n    controlBlock = parseControlBlock(ecc, controlBlock);\n  }\n  const { path } = controlBlock;\n\n  tapleafHash = tapleafHash || getTapleafHash(ecc, controlBlock, tapscript);\n\n  // `taptreeMerkleHash` begins as our tapscript tapleaf hash and its value iterates\n  // through its parent tapbranch hashes until it ends up as the taptree root hash\n  let taptreeMerkleHash = tapleafHash;\n  for (const taptreeSiblingHash of path) {\n    taptreeMerkleHash =\n      Buffer.compare(taptreeMerkleHash, taptreeSiblingHash) === -1\n        ? bcrypto.taggedHash('TapBranch', Buffer.concat([taptreeMerkleHash, taptreeSiblingHash]))\n        : bcrypto.taggedHash('TapBranch', Buffer.concat([taptreeSiblingHash, taptreeMerkleHash]));\n  }\n\n  return taptreeMerkleHash;\n}\n\nexport function getTweakedOutputKey(payment: bpayments.Payment): Buffer {\n  assert(payment.output);\n  if (payment.output.length === 34) {\n    return payment.output?.subarray(2);\n  }\n  throw new Error(`invalid p2tr tweaked output key size ${payment.output.length}`);\n}\n"]}
@@ -0,0 +1,4 @@
1
+ import * as input from './input';
2
+ import * as output from './output';
3
+ export { input, output };
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/templates/multisig/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,SAAS,CAAC;AACjC,OAAO,KAAK,MAAM,MAAM,UAAU,CAAC;AAEnC,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC"}
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.output = exports.input = void 0;
4
+ const input = require("./input");
5
+ exports.input = input;
6
+ const output = require("./output");
7
+ exports.output = output;
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvdGVtcGxhdGVzL211bHRpc2lnL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLGlDQUFpQztBQUd4QixzQkFBSztBQUZkLG1DQUFtQztBQUVuQix3QkFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIGlucHV0IGZyb20gJy4vaW5wdXQnO1xuaW1wb3J0ICogYXMgb3V0cHV0IGZyb20gJy4vb3V0cHV0JztcblxuZXhwb3J0IHsgaW5wdXQsIG91dHB1dCB9O1xuIl19
@@ -0,0 +1,7 @@
1
+ /// <reference types="node" />
2
+ import { Stack } from '../../';
3
+ export declare function check(script: Buffer | Stack, allowIncomplete?: boolean): boolean;
4
+ export declare namespace check {
5
+ var toJSON: () => string;
6
+ }
7
+ //# sourceMappingURL=input.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"input.d.ts","sourceRoot":"","sources":["../../../../src/templates/multisig/input.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,KAAK,EAAE,MAAM,QAAQ,CAAC;AAQ/B,wBAAgB,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,KAAK,EAAE,eAAe,CAAC,EAAE,OAAO,GAAG,OAAO,CAUhF;yBAVe,KAAK"}
@@ -0,0 +1,25 @@
1
+ "use strict";
2
+ // OP_0 [signatures ...]
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.check = void 0;
5
+ const __1 = require("../../");
6
+ const __2 = require("../../");
7
+ function partialSignature(value) {
8
+ return value === __2.opcodes.OP_0 || __1.script.isCanonicalScriptSignature(value);
9
+ }
10
+ function check(script, allowIncomplete) {
11
+ const chunks = __1.script.decompile(script);
12
+ if (chunks.length < 2)
13
+ return false;
14
+ if (chunks[0] !== __2.opcodes.OP_0)
15
+ return false;
16
+ if (allowIncomplete) {
17
+ return chunks.slice(1).every(partialSignature);
18
+ }
19
+ return chunks.slice(1).every(__1.script.isCanonicalScriptSignature);
20
+ }
21
+ exports.check = check;
22
+ check.toJSON = () => {
23
+ return 'multisig input';
24
+ };
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5wdXQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvdGVtcGxhdGVzL211bHRpc2lnL2lucHV0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSx3QkFBd0I7OztBQUd4Qiw4QkFBMkM7QUFDM0MsOEJBQWlDO0FBRWpDLFNBQVMsZ0JBQWdCLENBQUMsS0FBc0I7SUFDOUMsT0FBTyxLQUFLLEtBQUssV0FBTyxDQUFDLElBQUksSUFBSSxVQUFPLENBQUMsMEJBQTBCLENBQUMsS0FBZSxDQUFDLENBQUM7QUFDdkYsQ0FBQztBQUVELFNBQWdCLEtBQUssQ0FBQyxNQUFzQixFQUFFLGVBQXlCO0lBQ3JFLE1BQU0sTUFBTSxHQUFHLFVBQU8sQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUFVLENBQUM7SUFDbEQsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUM7UUFBRSxPQUFPLEtBQUssQ0FBQztJQUNwQyxJQUFJLE1BQU0sQ0FBQyxDQUFDLENBQUMsS0FBSyxXQUFPLENBQUMsSUFBSTtRQUFFLE9BQU8sS0FBSyxDQUFDO0lBRTdDLElBQUksZUFBZSxFQUFFO1FBQ25CLE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztLQUNoRDtJQUVELE9BQVEsTUFBTSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQWMsQ0FBQyxLQUFLLENBQUMsVUFBTyxDQUFDLDBCQUEwQixDQUFDLENBQUM7QUFDakYsQ0FBQztBQVZELHNCQVVDO0FBQ0QsS0FBSyxDQUFDLE1BQU0sR0FBRyxHQUFXLEVBQUU7SUFDMUIsT0FBTyxnQkFBZ0IsQ0FBQztBQUMxQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBPUF8wIFtzaWduYXR1cmVzIC4uLl1cblxuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi8uLi8nO1xuaW1wb3J0IHsgc2NyaXB0IGFzIGJzY3JpcHQgfSBmcm9tICcuLi8uLi8nO1xuaW1wb3J0IHsgb3Bjb2RlcyB9IGZyb20gJy4uLy4uLyc7XG5cbmZ1bmN0aW9uIHBhcnRpYWxTaWduYXR1cmUodmFsdWU6IG51bWJlciB8IEJ1ZmZlcik6IGJvb2xlYW4ge1xuICByZXR1cm4gdmFsdWUgPT09IG9wY29kZXMuT1BfMCB8fCBic2NyaXB0LmlzQ2Fub25pY2FsU2NyaXB0U2lnbmF0dXJlKHZhbHVlIGFzIEJ1ZmZlcik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjaGVjayhzY3JpcHQ6IEJ1ZmZlciB8IFN0YWNrLCBhbGxvd0luY29tcGxldGU/OiBib29sZWFuKTogYm9vbGVhbiB7XG4gIGNvbnN0IGNodW5rcyA9IGJzY3JpcHQuZGVjb21waWxlKHNjcmlwdCkgYXMgU3RhY2s7XG4gIGlmIChjaHVua3MubGVuZ3RoIDwgMikgcmV0dXJuIGZhbHNlO1xuICBpZiAoY2h1bmtzWzBdICE9PSBvcGNvZGVzLk9QXzApIHJldHVybiBmYWxzZTtcblxuICBpZiAoYWxsb3dJbmNvbXBsZXRlKSB7XG4gICAgcmV0dXJuIGNodW5rcy5zbGljZSgxKS5ldmVyeShwYXJ0aWFsU2lnbmF0dXJlKTtcbiAgfVxuXG4gIHJldHVybiAoY2h1bmtzLnNsaWNlKDEpIGFzIEJ1ZmZlcltdKS5ldmVyeShic2NyaXB0LmlzQ2Fub25pY2FsU2NyaXB0U2lnbmF0dXJlKTtcbn1cbmNoZWNrLnRvSlNPTiA9ICgpOiBzdHJpbmcgPT4ge1xuICByZXR1cm4gJ211bHRpc2lnIGlucHV0Jztcbn07XG4iXX0=
@@ -0,0 +1,7 @@
1
+ /// <reference types="node" />
2
+ import { Stack } from '../../';
3
+ export declare function check(script: Buffer | Stack, allowIncomplete?: boolean): boolean;
4
+ export declare namespace check {
5
+ var toJSON: () => string;
6
+ }
7
+ //# sourceMappingURL=output.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"output.d.ts","sourceRoot":"","sources":["../../../../src/templates/multisig/output.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,KAAK,EAAE,MAAM,QAAQ,CAAC;AAM/B,wBAAgB,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,KAAK,EAAE,eAAe,CAAC,EAAE,OAAO,GAAG,OAAO,CAkBhF;yBAlBe,KAAK"}
@@ -0,0 +1,38 @@
1
+ "use strict";
2
+ // m [pubKeys ...] n OP_CHECKMULTISIG
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.check = void 0;
5
+ const __1 = require("../../");
6
+ const __2 = require("../../");
7
+ const types = require("bitcoinjs-lib/src/types");
8
+ const OP_INT_BASE = __2.opcodes.OP_RESERVED; // OP_1 - 1
9
+ function check(script, allowIncomplete) {
10
+ const chunks = __1.script.decompile(script);
11
+ if (chunks.length < 4)
12
+ return false;
13
+ if (chunks[chunks.length - 1] !== __2.opcodes.OP_CHECKMULTISIG)
14
+ return false;
15
+ if (!types.Number(chunks[0]))
16
+ return false;
17
+ if (!types.Number(chunks[chunks.length - 2]))
18
+ return false;
19
+ const m = chunks[0] - OP_INT_BASE;
20
+ const n = chunks[chunks.length - 2] - OP_INT_BASE;
21
+ if (m <= 0)
22
+ return false;
23
+ if (n > 16)
24
+ return false;
25
+ if (m > n)
26
+ return false;
27
+ if (n !== chunks.length - 3)
28
+ return false;
29
+ if (allowIncomplete)
30
+ return true;
31
+ const keys = chunks.slice(1, -2);
32
+ return keys.every(__1.script.isCanonicalPubKey);
33
+ }
34
+ exports.check = check;
35
+ check.toJSON = () => {
36
+ return 'multi-sig output';
37
+ };
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3V0cHV0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL3RlbXBsYXRlcy9tdWx0aXNpZy9vdXRwdXQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLHFDQUFxQzs7O0FBR3JDLDhCQUEyQztBQUMzQyw4QkFBaUM7QUFDakMsaURBQWlEO0FBQ2pELE1BQU0sV0FBVyxHQUFHLFdBQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQyxXQUFXO0FBRXBELFNBQWdCLEtBQUssQ0FBQyxNQUFzQixFQUFFLGVBQXlCO0lBQ3JFLE1BQU0sTUFBTSxHQUFHLFVBQU8sQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUFVLENBQUM7SUFFbEQsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUM7UUFBRSxPQUFPLEtBQUssQ0FBQztJQUNwQyxJQUFJLE1BQU0sQ0FBQyxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQyxLQUFLLFdBQU8sQ0FBQyxnQkFBZ0I7UUFBRSxPQUFPLEtBQUssQ0FBQztJQUN6RSxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFBRSxPQUFPLEtBQUssQ0FBQztJQUMzQyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLE1BQU0sR0FBRyxDQUFDLENBQUMsQ0FBQztRQUFFLE9BQU8sS0FBSyxDQUFDO0lBQzNELE1BQU0sQ0FBQyxHQUFJLE1BQU0sQ0FBQyxDQUFDLENBQVksR0FBRyxXQUFXLENBQUM7SUFDOUMsTUFBTSxDQUFDLEdBQUksTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFZLEdBQUcsV0FBVyxDQUFDO0lBRTlELElBQUksQ0FBQyxJQUFJLENBQUM7UUFBRSxPQUFPLEtBQUssQ0FBQztJQUN6QixJQUFJLENBQUMsR0FBRyxFQUFFO1FBQUUsT0FBTyxLQUFLLENBQUM7SUFDekIsSUFBSSxDQUFDLEdBQUcsQ0FBQztRQUFFLE9BQU8sS0FBSyxDQUFDO0lBQ3hCLElBQUksQ0FBQyxLQUFLLE1BQU0sQ0FBQyxNQUFNLEdBQUcsQ0FBQztRQUFFLE9BQU8sS0FBSyxDQUFDO0lBQzFDLElBQUksZUFBZTtRQUFFLE9BQU8sSUFBSSxDQUFDO0lBRWpDLE1BQU0sSUFBSSxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFhLENBQUM7SUFDN0MsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLFVBQU8sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO0FBQy9DLENBQUM7QUFsQkQsc0JBa0JDO0FBQ0QsS0FBSyxDQUFDLE1BQU0sR0FBRyxHQUFXLEVBQUU7SUFDMUIsT0FBTyxrQkFBa0IsQ0FBQztBQUM1QixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBtIFtwdWJLZXlzIC4uLl0gbiBPUF9DSEVDS01VTFRJU0lHXG5cbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vLi4vJztcbmltcG9ydCB7IHNjcmlwdCBhcyBic2NyaXB0IH0gZnJvbSAnLi4vLi4vJztcbmltcG9ydCB7IG9wY29kZXMgfSBmcm9tICcuLi8uLi8nO1xuaW1wb3J0ICogYXMgdHlwZXMgZnJvbSAnYml0Y29pbmpzLWxpYi9zcmMvdHlwZXMnO1xuY29uc3QgT1BfSU5UX0JBU0UgPSBvcGNvZGVzLk9QX1JFU0VSVkVEOyAvLyBPUF8xIC0gMVxuXG5leHBvcnQgZnVuY3Rpb24gY2hlY2soc2NyaXB0OiBCdWZmZXIgfCBTdGFjaywgYWxsb3dJbmNvbXBsZXRlPzogYm9vbGVhbik6IGJvb2xlYW4ge1xuICBjb25zdCBjaHVua3MgPSBic2NyaXB0LmRlY29tcGlsZShzY3JpcHQpIGFzIFN0YWNrO1xuXG4gIGlmIChjaHVua3MubGVuZ3RoIDwgNCkgcmV0dXJuIGZhbHNlO1xuICBpZiAoY2h1bmtzW2NodW5rcy5sZW5ndGggLSAxXSAhPT0gb3Bjb2Rlcy5PUF9DSEVDS01VTFRJU0lHKSByZXR1cm4gZmFsc2U7XG4gIGlmICghdHlwZXMuTnVtYmVyKGNodW5rc1swXSkpIHJldHVybiBmYWxzZTtcbiAgaWYgKCF0eXBlcy5OdW1iZXIoY2h1bmtzW2NodW5rcy5sZW5ndGggLSAyXSkpIHJldHVybiBmYWxzZTtcbiAgY29uc3QgbSA9IChjaHVua3NbMF0gYXMgbnVtYmVyKSAtIE9QX0lOVF9CQVNFO1xuICBjb25zdCBuID0gKGNodW5rc1tjaHVua3MubGVuZ3RoIC0gMl0gYXMgbnVtYmVyKSAtIE9QX0lOVF9CQVNFO1xuXG4gIGlmIChtIDw9IDApIHJldHVybiBmYWxzZTtcbiAgaWYgKG4gPiAxNikgcmV0dXJuIGZhbHNlO1xuICBpZiAobSA+IG4pIHJldHVybiBmYWxzZTtcbiAgaWYgKG4gIT09IGNodW5rcy5sZW5ndGggLSAzKSByZXR1cm4gZmFsc2U7XG4gIGlmIChhbGxvd0luY29tcGxldGUpIHJldHVybiB0cnVlO1xuXG4gIGNvbnN0IGtleXMgPSBjaHVua3Muc2xpY2UoMSwgLTIpIGFzIEJ1ZmZlcltdO1xuICByZXR1cm4ga2V5cy5ldmVyeShic2NyaXB0LmlzQ2Fub25pY2FsUHViS2V5KTtcbn1cbmNoZWNrLnRvSlNPTiA9ICgpOiBzdHJpbmcgPT4ge1xuICByZXR1cm4gJ211bHRpLXNpZyBvdXRwdXQnO1xufTtcbiJdfQ==
@@ -0,0 +1,10 @@
1
+ /// <reference types="node" />
2
+ export declare function check(script: Buffer | Array<number | Buffer>): boolean;
3
+ export declare namespace check {
4
+ var toJSON: () => string;
5
+ }
6
+ declare const output: {
7
+ check: typeof check;
8
+ };
9
+ export { output };
10
+ //# sourceMappingURL=nulldata.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"nulldata.d.ts","sourceRoot":"","sources":["../../../src/templates/nulldata.ts"],"names":[],"mappings":";AAIA,wBAAgB,KAAK,CAAC,MAAM,EAAE,MAAM,GAAG,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC,GAAG,OAAO,CAItE;yBAJe,KAAK;;;AASrB,QAAA,MAAM,MAAM;;CAAY,CAAC;AAEzB,OAAO,EAAE,MAAM,EAAE,CAAC"}
@@ -0,0 +1,17 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.output = exports.check = void 0;
4
+ // OP_RETURN {data}
5
+ const __1 = require("../");
6
+ const __2 = require("../");
7
+ function check(script) {
8
+ const buffer = __1.script.compile(script);
9
+ return buffer.length > 1 && buffer[0] === __2.opcodes.OP_RETURN;
10
+ }
11
+ exports.check = check;
12
+ check.toJSON = () => {
13
+ return 'null data output';
14
+ };
15
+ const output = { check };
16
+ exports.output = output;
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnVsbGRhdGEuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvdGVtcGxhdGVzL251bGxkYXRhLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1CQUFtQjtBQUNuQiwyQkFBd0M7QUFDeEMsMkJBQThCO0FBRTlCLFNBQWdCLEtBQUssQ0FBQyxNQUF1QztJQUMzRCxNQUFNLE1BQU0sR0FBRyxVQUFPLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBRXZDLE9BQU8sTUFBTSxDQUFDLE1BQU0sR0FBRyxDQUFDLElBQUksTUFBTSxDQUFDLENBQUMsQ0FBQyxLQUFLLFdBQU8sQ0FBQyxTQUFTLENBQUM7QUFDOUQsQ0FBQztBQUpELHNCQUlDO0FBQ0QsS0FBSyxDQUFDLE1BQU0sR0FBRyxHQUFXLEVBQUU7SUFDMUIsT0FBTyxrQkFBa0IsQ0FBQztBQUM1QixDQUFDLENBQUM7QUFFRixNQUFNLE1BQU0sR0FBRyxFQUFFLEtBQUssRUFBRSxDQUFDO0FBRWhCLHdCQUFNIiwic291cmNlc0NvbnRlbnQiOlsiLy8gT1BfUkVUVVJOIHtkYXRhfVxuaW1wb3J0IHsgc2NyaXB0IGFzIGJzY3JpcHQgfSBmcm9tICcuLi8nO1xuaW1wb3J0IHsgb3Bjb2RlcyB9IGZyb20gJy4uLyc7XG5cbmV4cG9ydCBmdW5jdGlvbiBjaGVjayhzY3JpcHQ6IEJ1ZmZlciB8IEFycmF5PG51bWJlciB8IEJ1ZmZlcj4pOiBib29sZWFuIHtcbiAgY29uc3QgYnVmZmVyID0gYnNjcmlwdC5jb21waWxlKHNjcmlwdCk7XG5cbiAgcmV0dXJuIGJ1ZmZlci5sZW5ndGggPiAxICYmIGJ1ZmZlclswXSA9PT0gb3Bjb2Rlcy5PUF9SRVRVUk47XG59XG5jaGVjay50b0pTT04gPSAoKTogc3RyaW5nID0+IHtcbiAgcmV0dXJuICdudWxsIGRhdGEgb3V0cHV0Jztcbn07XG5cbmNvbnN0IG91dHB1dCA9IHsgY2hlY2sgfTtcblxuZXhwb3J0IHsgb3V0cHV0IH07XG4iXX0=
@@ -0,0 +1,4 @@
1
+ import * as input from './input';
2
+ import * as output from './output';
3
+ export { input, output };
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../src/templates/pubkey/index.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,KAAK,MAAM,SAAS,CAAC;AACjC,OAAO,KAAK,MAAM,MAAM,UAAU,CAAC;AAEnC,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC"}
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.output = exports.input = void 0;
4
+ const input = require("./input");
5
+ exports.input = input;
6
+ const output = require("./output");
7
+ exports.output = output;
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvdGVtcGxhdGVzL3B1YmtleS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxpQ0FBaUM7QUFHeEIsc0JBQUs7QUFGZCxtQ0FBbUM7QUFFbkIsd0JBQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBpbnB1dCBmcm9tICcuL2lucHV0JztcbmltcG9ydCAqIGFzIG91dHB1dCBmcm9tICcuL291dHB1dCc7XG5cbmV4cG9ydCB7IGlucHV0LCBvdXRwdXQgfTtcbiJdfQ==
@@ -0,0 +1,7 @@
1
+ /// <reference types="node" />
2
+ import { Stack } from '../../';
3
+ export declare function check(script: Buffer | Stack): boolean;
4
+ export declare namespace check {
5
+ var toJSON: () => string;
6
+ }
7
+ //# sourceMappingURL=input.d.ts.map