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,344 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.parsePubScript = exports.parseSignatureScript2Of3 = exports.parseSignatureScript = exports.getScriptPathLevel = exports.calculateScriptPathLevel = exports.isValidControlBock = exports.isPlaceholderSignature = void 0;
4
+ const opcodes = require("bitcoin-ops");
5
+ const bitcoinjs_lib_1 = require("bitcoinjs-lib");
6
+ const outputScripts_1 = require("./outputScripts");
7
+ const types_1 = require("./types");
8
+ function isPlaceholderSignature(v) {
9
+ if (Buffer.isBuffer(v)) {
10
+ return v.length === 0;
11
+ }
12
+ return v === 0;
13
+ }
14
+ exports.isPlaceholderSignature = isPlaceholderSignature;
15
+ /**
16
+ * @return true iff P2TR script path's control block matches BitGo's need
17
+ */
18
+ function isValidControlBock(controlBlock) {
19
+ // The last stack element is called the control block c, and must have length 33 + 32m
20
+ return Buffer.isBuffer(controlBlock) && 33 <= controlBlock.length && controlBlock.length % 32 === 1;
21
+ }
22
+ exports.isValidControlBock = isValidControlBock;
23
+ /**
24
+ * @return script path level for P2TR control block
25
+ */
26
+ function calculateScriptPathLevel(controlBlock) {
27
+ if (!Buffer.isBuffer(controlBlock)) {
28
+ throw new Error('Invalid control block type.');
29
+ }
30
+ if (controlBlock.length === 65) {
31
+ return 1;
32
+ }
33
+ if (controlBlock.length === 97) {
34
+ return 2;
35
+ }
36
+ throw new Error('unexpected control block length.');
37
+ }
38
+ exports.calculateScriptPathLevel = calculateScriptPathLevel;
39
+ /**
40
+ * @return leaf version for P2TR control block.
41
+ */
42
+ function getScriptPathLevel(controlBlock) {
43
+ if (Buffer.isBuffer(controlBlock) && controlBlock.length > 0) {
44
+ return controlBlock[0] & 0xfe;
45
+ }
46
+ throw new Error('unexpected leafVersion.');
47
+ }
48
+ exports.getScriptPathLevel = getScriptPathLevel;
49
+ function emptyMatchResult() {
50
+ return {
51
+ ':pubkey': [],
52
+ ':pubkey-xonly': [],
53
+ ':control-block': [],
54
+ ':signature': [],
55
+ ':script': [],
56
+ };
57
+ }
58
+ class MatchError extends Error {
59
+ constructor(message) {
60
+ super(message);
61
+ // this property is required to prohibit `return new Error()` when the return type demands `MatchError`
62
+ this.__type = 'MatchError';
63
+ }
64
+ static forPatternElement(p) {
65
+ if (typeof p === 'object' && ':script' in p) {
66
+ return new MatchError(`error matching nested script`);
67
+ }
68
+ return new MatchError(`error matching ${p}`);
69
+ }
70
+ }
71
+ /**
72
+ * @param script
73
+ * @param pattern
74
+ * @return MatchResult if script matches pattern. The result will contain the matched values.
75
+ */
76
+ function matchScript(script, pattern) {
77
+ /**
78
+ * Match a single script element with a ScriptPatternElement
79
+ */
80
+ function matchElement(e, p) {
81
+ switch (p) {
82
+ case 'OP_0':
83
+ return e === opcodes.OP_0 || (Buffer.isBuffer(e) && e.length === 0);
84
+ case 'OP_1':
85
+ case 'OP_2':
86
+ case 'OP_3':
87
+ case 'OP_CHECKMULTISIG':
88
+ case 'OP_CHECKSIG':
89
+ case 'OP_CHECKSIGVERIFY':
90
+ return e === opcodes[p];
91
+ case ':pubkey':
92
+ return Buffer.isBuffer(e) && e.length === 33;
93
+ case ':pubkey-xonly':
94
+ return Buffer.isBuffer(e) && e.length === 32;
95
+ case ':signature':
96
+ return Buffer.isBuffer(e) || isPlaceholderSignature(e);
97
+ case ':control-block':
98
+ return Buffer.isBuffer(e) && isValidControlBock(e);
99
+ default:
100
+ throw new Error(`unknown pattern element ${p}`);
101
+ }
102
+ }
103
+ if (script.length !== pattern.length) {
104
+ return new MatchError(`length mismatch`);
105
+ }
106
+ // Go over each pattern element.
107
+ // Collect captures into a result object.
108
+ return pattern.reduce((obj, p, i) => {
109
+ // if we had a previous mismatch, short-circuit
110
+ if (obj instanceof MatchError) {
111
+ return obj;
112
+ }
113
+ const e = script[i];
114
+ // for ':script' pattern elements, decompile script element and recurse
115
+ if (typeof p === 'object' && ':script' in p) {
116
+ if (!Buffer.isBuffer(e)) {
117
+ return new MatchError(`expected buffer for :script`);
118
+ }
119
+ const dec = bitcoinjs_lib_1.script.decompile(e);
120
+ if (!dec) {
121
+ return new MatchError(`error decompiling nested script`);
122
+ }
123
+ const match = matchScript(dec, p[':script']);
124
+ if (match instanceof MatchError) {
125
+ return match;
126
+ }
127
+ obj[':script'].push({
128
+ buffer: e,
129
+ match,
130
+ });
131
+ return obj;
132
+ }
133
+ const match = matchElement(e, p);
134
+ if (!match) {
135
+ return MatchError.forPatternElement(p);
136
+ }
137
+ // if pattern element is a capture, add it to the result obj
138
+ if (p === ':signature' && e === 0) {
139
+ obj[p].push(e);
140
+ }
141
+ else if (p in obj) {
142
+ if (!Buffer.isBuffer(e)) {
143
+ throw new Error(`invalid capture value`);
144
+ }
145
+ obj[p].push(e);
146
+ }
147
+ return obj;
148
+ }, emptyMatchResult());
149
+ }
150
+ /**
151
+ * @param script
152
+ * @param patterns
153
+ * @return first match
154
+ */
155
+ function matchScriptSome(script, patterns) {
156
+ for (const p of patterns) {
157
+ const m = matchScript(script, p);
158
+ if (m instanceof MatchError) {
159
+ continue;
160
+ }
161
+ return m;
162
+ }
163
+ return new MatchError(`no match for script`);
164
+ }
165
+ function isLegacy(p) {
166
+ return Boolean(p.script && !p.witness);
167
+ }
168
+ function isWrappedSegwit(p) {
169
+ return Boolean(p.script && p.witness);
170
+ }
171
+ function isNativeSegwit(p) {
172
+ return Boolean(!p.script && p.witness);
173
+ }
174
+ const parseP2shP2pk = (p) => {
175
+ if (!isLegacy(p)) {
176
+ return new MatchError(`expected legacy input`);
177
+ }
178
+ const match = matchScript(p.script, [':signature', { ':script': [':pubkey', 'OP_CHECKSIG'] }]);
179
+ if (match instanceof MatchError) {
180
+ return match;
181
+ }
182
+ return {
183
+ scriptType: 'p2shP2pk',
184
+ publicKeys: match[':script'][0].match[':pubkey'],
185
+ signatures: match[':signature'],
186
+ };
187
+ };
188
+ function parseP2ms(decScript, scriptType) {
189
+ const pattern2Of3 = ['OP_2', ':pubkey', ':pubkey', ':pubkey', 'OP_3', 'OP_CHECKMULTISIG'];
190
+ const match = matchScriptSome(decScript, [
191
+ /* full-signed, no placeholder signature */
192
+ ['OP_0', ':signature', ':signature', { ':script': pattern2Of3 }],
193
+ /* half-signed, placeholder signatures */
194
+ ['OP_0', ':signature', ':signature', ':signature', { ':script': pattern2Of3 }],
195
+ ]);
196
+ if (match instanceof MatchError) {
197
+ return match;
198
+ }
199
+ const [redeemScript] = match[':script'];
200
+ if (!types_1.isTriple(redeemScript.match[':pubkey'])) {
201
+ throw new Error(`invalid pubkey count`);
202
+ }
203
+ return {
204
+ scriptType,
205
+ publicKeys: redeemScript.match[':pubkey'],
206
+ pubScript: redeemScript.buffer,
207
+ signatures: match[':signature'],
208
+ redeemScript: scriptType === 'p2sh' ? redeemScript.buffer : undefined,
209
+ witnessScript: scriptType === 'p2shP2wsh' || scriptType === 'p2wsh' ? redeemScript.buffer : undefined,
210
+ };
211
+ }
212
+ const parseP2sh2Of3 = (p) => {
213
+ if (!isLegacy(p)) {
214
+ return new MatchError(`expected legacy input`);
215
+ }
216
+ return parseP2ms(p.script, 'p2sh');
217
+ };
218
+ const parseP2shP2wsh2Of3 = (p) => {
219
+ if (!isWrappedSegwit(p)) {
220
+ return new MatchError(`expected wrapped segwit input`);
221
+ }
222
+ return { ...parseP2ms(p.witness, 'p2shP2wsh'), redeemScript: p.script[0] };
223
+ };
224
+ const parseP2wsh2Of3 = (p) => {
225
+ if (!isNativeSegwit(p)) {
226
+ return new MatchError(`expected native segwit`);
227
+ }
228
+ return parseP2ms(p.witness, 'p2wsh');
229
+ };
230
+ const parseP2tr2Of3 = (p) => {
231
+ if (!isNativeSegwit(p)) {
232
+ return new MatchError(`expected native segwit`);
233
+ }
234
+ // assumes no annex
235
+ const match = matchScript(p.witness, [
236
+ ':signature',
237
+ ':signature',
238
+ { ':script': [':pubkey-xonly', 'OP_CHECKSIGVERIFY', ':pubkey-xonly', 'OP_CHECKSIG'] },
239
+ ':control-block',
240
+ ]);
241
+ if (match instanceof MatchError) {
242
+ return match;
243
+ }
244
+ const [controlBlock] = match[':control-block'];
245
+ const scriptPathLevel = calculateScriptPathLevel(controlBlock);
246
+ const leafVersion = getScriptPathLevel(controlBlock);
247
+ return {
248
+ scriptType: 'p2tr',
249
+ pubScript: match[':script'][0].buffer,
250
+ publicKeys: match[':script'][0].match[':pubkey-xonly'],
251
+ signatures: match[':signature'],
252
+ controlBlock,
253
+ scriptPathLevel,
254
+ leafVersion,
255
+ };
256
+ };
257
+ /**
258
+ * Parse a transaction's signature script to obtain public keys, signatures, the sig script,
259
+ * and other properties.
260
+ *
261
+ * Only supports script types used in BitGo transactions.
262
+ *
263
+ * @param input
264
+ * @returns ParsedSignatureScript
265
+ */
266
+ function parseSignatureScript(input) {
267
+ const decScript = bitcoinjs_lib_1.script.decompile(input.script);
268
+ const parsers = [parseP2sh2Of3, parseP2shP2wsh2Of3, parseP2wsh2Of3, parseP2tr2Of3, parseP2shP2pk];
269
+ for (const f of parsers) {
270
+ const parsed = f({
271
+ script: (decScript === null || decScript === void 0 ? void 0 : decScript.length) === 0 ? null : decScript,
272
+ witness: input.witness.length === 0 ? null : input.witness,
273
+ });
274
+ if (parsed instanceof MatchError) {
275
+ continue;
276
+ }
277
+ return parsed;
278
+ }
279
+ throw new Error(`could not parse input`);
280
+ }
281
+ exports.parseSignatureScript = parseSignatureScript;
282
+ function parseSignatureScript2Of3(input) {
283
+ const result = parseSignatureScript(input);
284
+ if (!outputScripts_1.isScriptType2Of3(result.scriptType)) {
285
+ throw new Error(`invalid script type`);
286
+ }
287
+ if (!result.signatures) {
288
+ throw new Error(`missing signatures`);
289
+ }
290
+ if (result.publicKeys.length !== 3 && (result.publicKeys.length !== 2 || result.scriptType !== 'p2tr')) {
291
+ throw new Error(`unexpected pubkey count`);
292
+ }
293
+ return result;
294
+ }
295
+ exports.parseSignatureScript2Of3 = parseSignatureScript2Of3;
296
+ const parseP2msPubScript = (pubScript, scriptType) => {
297
+ const match = matchScript([pubScript], [{ ':script': ['OP_2', ':pubkey', ':pubkey', ':pubkey', 'OP_3', 'OP_CHECKMULTISIG'] }]);
298
+ if (match instanceof MatchError) {
299
+ return match;
300
+ }
301
+ const [redeemScript] = match[':script'];
302
+ if (!types_1.isTriple(redeemScript.match[':pubkey'])) {
303
+ throw new Error('invalid pubkey count');
304
+ }
305
+ return {
306
+ scriptType,
307
+ publicKeys: redeemScript.match[':pubkey'],
308
+ pubScript: redeemScript.buffer,
309
+ redeemScript: scriptType === 'p2sh' ? redeemScript.buffer : undefined,
310
+ witnessScript: scriptType === 'p2shP2wsh' || scriptType === 'p2wsh' ? redeemScript.buffer : undefined,
311
+ };
312
+ };
313
+ const parseP2tr2Of3PubScript = (pubScript, scriptType) => {
314
+ const match = matchScript([pubScript], [{ ':script': [':pubkey-xonly', 'OP_CHECKSIGVERIFY', ':pubkey-xonly', 'OP_CHECKSIG'] }]);
315
+ if (match instanceof MatchError) {
316
+ return match;
317
+ }
318
+ return {
319
+ scriptType: 'p2tr',
320
+ pubScript: match[':script'][0].buffer,
321
+ publicKeys: match[':script'][0].match[':pubkey-xonly'],
322
+ };
323
+ };
324
+ /**
325
+ * @return pubScript (scriptPubKey/redeemScript/witnessScript) is parsed.
326
+ * P2SH => scriptType, pubScript (redeemScript), redeemScript, public keys
327
+ * PW2SH => scriptType, pubScript (witnessScript), witnessScript, public keys.
328
+ * P2SH-PW2SH => scriptType, pubScript (witnessScript), witnessScript, public keys.
329
+ * P2TR => scriptType, pubScript, controlBlock, scriptPathLevel, leafVersion, pub keys, signatures.
330
+ */
331
+ function parsePubScript(inputPubScript, scriptType) {
332
+ const result = scriptType === 'p2tr'
333
+ ? parseP2tr2Of3PubScript(inputPubScript, scriptType)
334
+ : parseP2msPubScript(inputPubScript, scriptType);
335
+ if (result instanceof MatchError) {
336
+ throw new Error(result.message);
337
+ }
338
+ if (result.publicKeys.length !== 3 && (result.publicKeys.length !== 2 || result.scriptType !== 'p2tr')) {
339
+ throw new Error('unexpected pubkey count');
340
+ }
341
+ return result;
342
+ }
343
+ exports.parsePubScript = parsePubScript;
344
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parseInput.js","sourceRoot":"","sources":["../../../src/bitgo/parseInput.ts"],"names":[],"mappings":"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","sourcesContent":["import * as opcodes from 'bitcoin-ops';\nimport { TxInput, script as bscript } from 'bitcoinjs-lib';\n\nimport { isScriptType2Of3, ScriptType, ScriptType2Of3 } from './outputScripts';\nimport { isTriple } from './types';\n\nexport function isPlaceholderSignature(v: number | Buffer): boolean {\n  if (Buffer.isBuffer(v)) {\n    return v.length === 0;\n  }\n  return v === 0;\n}\n\n/**\n * @return true iff P2TR script path's control block matches BitGo's need\n */\nexport function isValidControlBock(controlBlock: Buffer): boolean {\n  // The last stack element is called the control block c, and must have length 33 + 32m\n  return Buffer.isBuffer(controlBlock) && 33 <= controlBlock.length && controlBlock.length % 32 === 1;\n}\n\n/**\n * @return script path level for P2TR control block\n */\nexport function calculateScriptPathLevel(controlBlock: Buffer): number {\n  if (!Buffer.isBuffer(controlBlock)) {\n    throw new Error('Invalid control block type.');\n  }\n  if (controlBlock.length === 65) {\n    return 1;\n  }\n  if (controlBlock.length === 97) {\n    return 2;\n  }\n  throw new Error('unexpected control block length.');\n}\n\n/**\n * @return leaf version for P2TR control block.\n */\nexport function getScriptPathLevel(controlBlock: Buffer): number {\n  if (Buffer.isBuffer(controlBlock) && controlBlock.length > 0) {\n    return controlBlock[0] & 0xfe;\n  }\n  throw new Error('unexpected leafVersion.');\n}\n\ninterface ParsedScript {\n  scriptType: ScriptType;\n}\n\nexport type ParsedPubScript = ParsedScript;\nexport type ParsedSignatureScript = ParsedScript;\n\nexport interface ParsedSignatureScriptP2shP2pk extends ParsedSignatureScript {\n  scriptType: 'p2shP2pk';\n  publicKeys: [Buffer];\n  signatures: [Buffer];\n}\n\nexport interface ParsedPubScriptTaprootScriptPath extends ParsedPubScript {\n  scriptType: 'p2tr';\n  publicKeys: [Buffer, Buffer];\n  pubScript: Buffer;\n}\n\nexport interface ParsedPubScript2Of3 extends ParsedPubScript {\n  scriptType: 'p2sh' | 'p2shP2wsh' | 'p2wsh';\n  publicKeys: [Buffer, Buffer, Buffer];\n  pubScript: Buffer;\n  redeemScript: Buffer | undefined;\n  witnessScript: Buffer | undefined;\n}\n\nexport interface ParsedSignatureScript2Of3 extends ParsedSignatureScript {\n  scriptType: 'p2sh' | 'p2shP2wsh' | 'p2wsh';\n  publicKeys: [Buffer, Buffer, Buffer];\n  signatures:\n    | [Buffer, Buffer] // fully-signed transactions with signatures\n    /* Partially signed transactions with placeholder signatures.\n       For p2sh, the placeholder is OP_0 (number 0) */\n    | [Buffer | 0, Buffer | 0, Buffer | 0];\n  pubScript: Buffer;\n  redeemScript: Buffer | undefined;\n  witnessScript: Buffer | undefined;\n}\n\n/**\n * Keypath spends only have a single pubkey and single signature\n */\nexport interface ParsedSignatureScriptTaprootKeyPath extends ParsedSignatureScript {\n  scriptType: 'p2tr';\n  publicKeys: [Buffer];\n  signatures: [Buffer];\n  pubScript: Buffer;\n}\n\n/**\n * Taproot Scriptpath spends are more similar to regular p2ms spends and have two public keys and\n * two signatures\n */\nexport interface ParsedSignatureScriptTaprootScriptPath extends ParsedSignatureScript {\n  scriptType: 'p2tr';\n  publicKeys: [Buffer, Buffer];\n  signatures: [Buffer, Buffer];\n  controlBlock: Buffer;\n  leafVersion: number;\n  /** Indicates the level inside the taptree. */\n  scriptPathLevel: number;\n  pubScript: Buffer;\n}\n\nexport type ParsedSignatureScriptTaproot = ParsedSignatureScriptTaprootKeyPath | ParsedSignatureScriptTaprootScriptPath;\n\ntype DecompiledScript = Array<Buffer | number>;\n\n/**\n * Static script elements\n */\ntype ScriptPatternConstant =\n  | 'OP_0'\n  | 'OP_1'\n  | 'OP_2'\n  | 'OP_3'\n  | 'OP_CHECKMULTISIG'\n  | 'OP_CHECKSIG'\n  | 'OP_CHECKSIGVERIFY';\n\n/**\n * Script elements that can be captured\n */\ntype ScriptPatternCapture =\n  | ':pubkey'\n  | ':pubkey-xonly'\n  | ':signature'\n  | ':control-block'\n  | { ':script': ScriptPatternElement[] };\n\ntype ScriptPatternElement = ScriptPatternConstant | ScriptPatternCapture;\n\n/**\n * Result for a successful script match\n */\ntype MatchResult = {\n  ':pubkey': Buffer[];\n  ':pubkey-xonly': Buffer[];\n  ':control-block': Buffer[];\n  ':signature': (Buffer | 0)[];\n  ':script': { buffer: Buffer; match: MatchResult }[];\n};\n\nfunction emptyMatchResult(): MatchResult {\n  return {\n    ':pubkey': [],\n    ':pubkey-xonly': [],\n    ':control-block': [],\n    ':signature': [],\n    ':script': [],\n  };\n}\n\nclass MatchError extends Error {\n  // this property is required to prohibit `return new Error()` when the return type demands `MatchError`\n  __type = 'MatchError';\n  constructor(message: string) {\n    super(message);\n  }\n\n  static forPatternElement(p: ScriptPatternElement): MatchError {\n    if (typeof p === 'object' && ':script' in p) {\n      return new MatchError(`error matching nested script`);\n    }\n    return new MatchError(`error matching ${p}`);\n  }\n}\n\n/**\n * @param script\n * @param pattern\n * @return MatchResult if script matches pattern. The result will contain the matched values.\n */\nfunction matchScript(script: DecompiledScript, pattern: ScriptPatternElement[]): MatchResult | MatchError {\n  /**\n   * Match a single script element with a ScriptPatternElement\n   */\n  function matchElement(e: Buffer | number, p: ScriptPatternElement): MatchResult | boolean {\n    switch (p) {\n      case 'OP_0':\n        return e === opcodes.OP_0 || (Buffer.isBuffer(e) && e.length === 0);\n      case 'OP_1':\n      case 'OP_2':\n      case 'OP_3':\n      case 'OP_CHECKMULTISIG':\n      case 'OP_CHECKSIG':\n      case 'OP_CHECKSIGVERIFY':\n        return e === opcodes[p];\n      case ':pubkey':\n        return Buffer.isBuffer(e) && e.length === 33;\n      case ':pubkey-xonly':\n        return Buffer.isBuffer(e) && e.length === 32;\n      case ':signature':\n        return Buffer.isBuffer(e) || isPlaceholderSignature(e);\n      case ':control-block':\n        return Buffer.isBuffer(e) && isValidControlBock(e);\n      default:\n        throw new Error(`unknown pattern element ${p}`);\n    }\n  }\n\n  if (script.length !== pattern.length) {\n    return new MatchError(`length mismatch`);\n  }\n\n  // Go over each pattern element.\n  // Collect captures into a result object.\n  return pattern.reduce((obj: MatchResult | MatchError, p, i): MatchResult | MatchError => {\n    // if we had a previous mismatch, short-circuit\n    if (obj instanceof MatchError) {\n      return obj;\n    }\n\n    const e = script[i];\n\n    // for ':script' pattern elements, decompile script element and recurse\n    if (typeof p === 'object' && ':script' in p) {\n      if (!Buffer.isBuffer(e)) {\n        return new MatchError(`expected buffer for :script`);\n      }\n      const dec = bscript.decompile(e);\n      if (!dec) {\n        return new MatchError(`error decompiling nested script`);\n      }\n      const match = matchScript(dec, p[':script']);\n      if (match instanceof MatchError) {\n        return match;\n      }\n      obj[':script'].push({\n        buffer: e,\n        match,\n      });\n      return obj;\n    }\n\n    const match = matchElement(e, p);\n    if (!match) {\n      return MatchError.forPatternElement(p);\n    }\n\n    // if pattern element is a capture, add it to the result obj\n    if (p === ':signature' && e === 0) {\n      obj[p].push(e);\n    } else if (p in obj) {\n      if (!Buffer.isBuffer(e)) {\n        throw new Error(`invalid capture value`);\n      }\n      obj[p].push(e);\n    }\n\n    return obj;\n  }, emptyMatchResult());\n}\n\n/**\n * @param script\n * @param patterns\n * @return first match\n */\nfunction matchScriptSome(script: DecompiledScript, patterns: ScriptPatternElement[][]): MatchResult | MatchError {\n  for (const p of patterns) {\n    const m = matchScript(script, p);\n    if (m instanceof MatchError) {\n      continue;\n    }\n    return m;\n  }\n  return new MatchError(`no match for script`);\n}\n\ntype InputScripts<TScript, TWitness> = {\n  script: TScript;\n  witness: TWitness;\n};\n\ntype InputScriptsLegacy = InputScripts<DecompiledScript, null>;\ntype InputScriptsWrappedSegwit = InputScripts<DecompiledScript, Buffer[]>;\ntype InputScriptsNativeSegwit = InputScripts<null, Buffer[]>;\n\ntype InputScriptsUnknown = InputScripts<DecompiledScript | null, Buffer[] | null>;\n\ntype InputParser<T extends ParsedSignatureScriptP2shP2pk | ParsedSignatureScript2Of3 | ParsedSignatureScriptTaproot> = (\n  p: InputScriptsUnknown\n) => T | MatchError;\n\nexport type InputPubScript = Buffer;\n\ntype PubScriptParser<T extends ParsedPubScriptTaprootScriptPath | ParsedPubScript2Of3> = (\n  p: InputPubScript,\n  ScriptType2Of3\n) => T | MatchError;\n\nfunction isLegacy(p: InputScriptsUnknown): p is InputScriptsLegacy {\n  return Boolean(p.script && !p.witness);\n}\n\nfunction isWrappedSegwit(p: InputScriptsUnknown): p is InputScriptsWrappedSegwit {\n  return Boolean(p.script && p.witness);\n}\n\nfunction isNativeSegwit(p: InputScriptsUnknown): p is InputScriptsNativeSegwit {\n  return Boolean(!p.script && p.witness);\n}\n\nconst parseP2shP2pk: InputParser<ParsedSignatureScriptP2shP2pk> = (p) => {\n  if (!isLegacy(p)) {\n    return new MatchError(`expected legacy input`);\n  }\n  const match = matchScript(p.script, [':signature', { ':script': [':pubkey', 'OP_CHECKSIG'] }]);\n  if (match instanceof MatchError) {\n    return match;\n  }\n  return {\n    scriptType: 'p2shP2pk',\n    publicKeys: match[':script'][0].match[':pubkey'] as [Buffer],\n    signatures: match[':signature'] as [Buffer],\n  };\n};\n\nfunction parseP2ms(\n  decScript: DecompiledScript,\n  scriptType: 'p2sh' | 'p2shP2wsh' | 'p2wsh'\n): ParsedSignatureScript2Of3 | MatchError {\n  const pattern2Of3: ScriptPatternElement[] = ['OP_2', ':pubkey', ':pubkey', ':pubkey', 'OP_3', 'OP_CHECKMULTISIG'];\n\n  const match = matchScriptSome(decScript, [\n    /* full-signed, no placeholder signature */\n    ['OP_0', ':signature', ':signature', { ':script': pattern2Of3 }],\n    /* half-signed, placeholder signatures */\n    ['OP_0', ':signature', ':signature', ':signature', { ':script': pattern2Of3 }],\n  ]);\n  if (match instanceof MatchError) {\n    return match;\n  }\n\n  const [redeemScript] = match[':script'];\n\n  if (!isTriple(redeemScript.match[':pubkey'])) {\n    throw new Error(`invalid pubkey count`);\n  }\n\n  return {\n    scriptType,\n    publicKeys: redeemScript.match[':pubkey'],\n    pubScript: redeemScript.buffer,\n    signatures: match[':signature'] as ParsedSignatureScript2Of3['signatures'],\n    redeemScript: scriptType === 'p2sh' ? redeemScript.buffer : undefined,\n    witnessScript: scriptType === 'p2shP2wsh' || scriptType === 'p2wsh' ? redeemScript.buffer : undefined,\n  };\n}\n\nconst parseP2sh2Of3: InputParser<ParsedSignatureScript2Of3> = (p) => {\n  if (!isLegacy(p)) {\n    return new MatchError(`expected legacy input`);\n  }\n  return parseP2ms(p.script, 'p2sh');\n};\n\nconst parseP2shP2wsh2Of3: InputParser<ParsedSignatureScript2Of3> = (p) => {\n  if (!isWrappedSegwit(p)) {\n    return new MatchError(`expected wrapped segwit input`);\n  }\n  return { ...parseP2ms(p.witness, 'p2shP2wsh'), redeemScript: p.script[0] as Buffer };\n};\n\nconst parseP2wsh2Of3: InputParser<ParsedSignatureScript2Of3> = (p) => {\n  if (!isNativeSegwit(p)) {\n    return new MatchError(`expected native segwit`);\n  }\n  return parseP2ms(p.witness, 'p2wsh');\n};\n\nconst parseP2tr2Of3: InputParser<ParsedSignatureScriptTaproot> = (p) => {\n  if (!isNativeSegwit(p)) {\n    return new MatchError(`expected native segwit`);\n  }\n  // assumes no annex\n  const match = matchScript(p.witness, [\n    ':signature',\n    ':signature',\n    { ':script': [':pubkey-xonly', 'OP_CHECKSIGVERIFY', ':pubkey-xonly', 'OP_CHECKSIG'] },\n    ':control-block',\n  ]);\n  if (match instanceof MatchError) {\n    return match;\n  }\n  const [controlBlock] = match[':control-block'];\n  const scriptPathLevel = calculateScriptPathLevel(controlBlock);\n\n  const leafVersion = getScriptPathLevel(controlBlock);\n\n  return {\n    scriptType: 'p2tr',\n    pubScript: match[':script'][0].buffer,\n    publicKeys: match[':script'][0].match[':pubkey-xonly'] as [Buffer, Buffer],\n    signatures: match[':signature'] as [Buffer, Buffer],\n    controlBlock,\n    scriptPathLevel,\n    leafVersion,\n  };\n};\n\n/**\n * Parse a transaction's signature script to obtain public keys, signatures, the sig script,\n * and other properties.\n *\n * Only supports script types used in BitGo transactions.\n *\n * @param input\n * @returns ParsedSignatureScript\n */\nexport function parseSignatureScript(\n  input: TxInput\n): ParsedSignatureScriptP2shP2pk | ParsedSignatureScript2Of3 | ParsedSignatureScriptTaproot {\n  const decScript = bscript.decompile(input.script);\n  const parsers = [parseP2sh2Of3, parseP2shP2wsh2Of3, parseP2wsh2Of3, parseP2tr2Of3, parseP2shP2pk] as const;\n  for (const f of parsers) {\n    const parsed = f({\n      script: decScript?.length === 0 ? null : decScript,\n      witness: input.witness.length === 0 ? null : input.witness,\n    });\n    if (parsed instanceof MatchError) {\n      continue;\n    }\n    return parsed;\n  }\n  throw new Error(`could not parse input`);\n}\n\nexport function parseSignatureScript2Of3(input: TxInput): ParsedSignatureScript2Of3 | ParsedSignatureScriptTaproot {\n  const result = parseSignatureScript(input);\n\n  if (!isScriptType2Of3(result.scriptType)) {\n    throw new Error(`invalid script type`);\n  }\n\n  if (!result.signatures) {\n    throw new Error(`missing signatures`);\n  }\n  if (result.publicKeys.length !== 3 && (result.publicKeys.length !== 2 || result.scriptType !== 'p2tr')) {\n    throw new Error(`unexpected pubkey count`);\n  }\n\n  return result as ParsedSignatureScript2Of3 | ParsedSignatureScriptTaproot;\n}\n\nconst parseP2msPubScript: PubScriptParser<ParsedPubScript2Of3> = (\n  pubScript,\n  scriptType: 'p2sh' | 'p2shP2wsh' | 'p2wsh'\n) => {\n  const match = matchScript(\n    [pubScript],\n    [{ ':script': ['OP_2', ':pubkey', ':pubkey', ':pubkey', 'OP_3', 'OP_CHECKMULTISIG'] }]\n  );\n  if (match instanceof MatchError) {\n    return match;\n  }\n\n  const [redeemScript] = match[':script'];\n\n  if (!isTriple(redeemScript.match[':pubkey'])) {\n    throw new Error('invalid pubkey count');\n  }\n\n  return {\n    scriptType,\n    publicKeys: redeemScript.match[':pubkey'],\n    pubScript: redeemScript.buffer,\n    redeemScript: scriptType === 'p2sh' ? redeemScript.buffer : undefined,\n    witnessScript: scriptType === 'p2shP2wsh' || scriptType === 'p2wsh' ? redeemScript.buffer : undefined,\n  };\n};\n\nconst parseP2tr2Of3PubScript: PubScriptParser<ParsedPubScriptTaprootScriptPath> = (pubScript, scriptType: 'p2tr') => {\n  const match = matchScript(\n    [pubScript],\n    [{ ':script': [':pubkey-xonly', 'OP_CHECKSIGVERIFY', ':pubkey-xonly', 'OP_CHECKSIG'] }]\n  );\n  if (match instanceof MatchError) {\n    return match;\n  }\n\n  return {\n    scriptType: 'p2tr',\n    pubScript: match[':script'][0].buffer,\n    publicKeys: match[':script'][0].match[':pubkey-xonly'] as [Buffer, Buffer],\n  };\n};\n\n/**\n * @return pubScript (scriptPubKey/redeemScript/witnessScript) is parsed.\n * P2SH => scriptType, pubScript (redeemScript), redeemScript, public keys\n * PW2SH => scriptType, pubScript (witnessScript), witnessScript, public keys.\n * P2SH-PW2SH => scriptType, pubScript (witnessScript), witnessScript, public keys.\n * P2TR => scriptType, pubScript, controlBlock, scriptPathLevel, leafVersion, pub keys, signatures.\n */\nexport function parsePubScript(\n  inputPubScript: InputPubScript,\n  scriptType: ScriptType2Of3\n): ParsedPubScript2Of3 | ParsedPubScriptTaprootScriptPath {\n  const result =\n    scriptType === 'p2tr'\n      ? parseP2tr2Of3PubScript(inputPubScript, scriptType)\n      : parseP2msPubScript(inputPubScript, scriptType);\n\n  if (result instanceof MatchError) {\n    throw new Error(result.message);\n  }\n\n  if (result.publicKeys.length !== 3 && (result.publicKeys.length !== 2 || result.scriptType !== 'p2tr')) {\n    throw new Error('unexpected pubkey count');\n  }\n\n  return result;\n}\n"]}
@@ -0,0 +1,24 @@
1
+ /// <reference types="node" />
2
+ import { PsbtInputUpdate } from 'bip174/src/lib/interfaces';
3
+ import { TxOutput } from '../..';
4
+ import { UtxoTransaction } from '../UtxoTransaction';
5
+ export declare function getInputUpdate(tx: UtxoTransaction<bigint>, vin: number, prevOuts: (TxOutput<bigint> & {
6
+ prevTx?: Buffer;
7
+ })[]): PsbtInputUpdate;
8
+ /**
9
+ * Takes a partially signed transaction and removes the scripts and signatures.
10
+ *
11
+ * Inputs must be one of:
12
+ * - p2shP2pk
13
+ * - p2sh 2-of-3
14
+ * - p2shP2wsh 2-of-3
15
+ * - p2wsh 2-of-3
16
+ * - p2tr script path 2-of-2
17
+ *
18
+ * @param tx the partially signed transaction
19
+ * @param prevOuts
20
+ *
21
+ * @return the removed scripts and signatures, ready to be added to a PSBT
22
+ */
23
+ export declare function unsign(tx: UtxoTransaction<bigint>, prevOuts: TxOutput<bigint>[]): PsbtInputUpdate[];
24
+ //# sourceMappingURL=fromHalfSigned.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fromHalfSigned.d.ts","sourceRoot":"","sources":["../../../../src/bitgo/psbt/fromHalfSigned.ts"],"names":[],"mappings":";AAAA,OAAO,EAAE,eAAe,EAAc,MAAM,2BAA2B,CAAC;AACxE,OAAO,EAAiB,QAAQ,EAAiC,MAAM,OAAO,CAAC;AAC/E,OAAO,EAAE,eAAe,EAAE,MAAM,oBAAoB,CAAC;AASrD,wBAAgB,cAAc,CAC5B,EAAE,EAAE,eAAe,CAAC,MAAM,CAAC,EAC3B,GAAG,EAAE,MAAM,EACX,QAAQ,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG;IAAE,MAAM,CAAC,EAAE,MAAM,CAAA;CAAE,CAAC,EAAE,GACnD,eAAe,CA2DjB;AAED;;;;;;;;;;;;;;GAcG;AACH,wBAAgB,MAAM,CAAC,EAAE,EAAE,eAAe,CAAC,MAAM,CAAC,EAAE,QAAQ,EAAE,QAAQ,CAAC,MAAM,CAAC,EAAE,GAAG,eAAe,EAAE,CAOnG"}
@@ -0,0 +1,91 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.unsign = exports.getInputUpdate = void 0;
4
+ const __1 = require("../..");
5
+ const parseInput_1 = require("../parseInput");
6
+ const signature_1 = require("../signature");
7
+ const outputScripts_1 = require("../outputScripts");
8
+ function omitUndefined(v) {
9
+ return Object.fromEntries(Object.entries(v).filter(([k, v]) => v !== undefined));
10
+ }
11
+ function getInputUpdate(tx, vin, prevOuts) {
12
+ const nonWitnessUtxo = prevOuts[vin].prevTx;
13
+ const { script, witness } = tx.ins[vin];
14
+ if (script.length === 0 && witness.length === 0) {
15
+ return nonWitnessUtxo ? { nonWitnessUtxo } : {};
16
+ }
17
+ const parsedInput = parseInput_1.parseSignatureScript(tx.ins[vin]);
18
+ function getPartialSigs() {
19
+ return signature_1.getSignaturesWithPublicKeys(tx, vin, prevOuts, parsedInput.publicKeys).flatMap((signature, i) => signature
20
+ ? [
21
+ {
22
+ pubkey: parsedInput.publicKeys[i],
23
+ signature,
24
+ },
25
+ ]
26
+ : []);
27
+ }
28
+ // Because Zcash directly hashes the value for non-segwit transactions, we do not need to check indirectly
29
+ // with the previous transaction. Therefore, we can treat Zcash non-segwit transactions as Bitcoin
30
+ // segwit transactions
31
+ if (!outputScripts_1.hasWitnessData(parsedInput.scriptType) && !nonWitnessUtxo && __1.getMainnet(tx.network) !== __1.networks.zcash) {
32
+ throw new Error(`scriptType ${parsedInput.scriptType} requires prevTx Buffer`);
33
+ }
34
+ switch (parsedInput.scriptType) {
35
+ case 'p2shP2pk':
36
+ return {
37
+ nonWitnessUtxo,
38
+ partialSig: [{ pubkey: parsedInput.publicKeys[0], signature: parsedInput.signatures[0] }],
39
+ };
40
+ case 'p2sh':
41
+ case 'p2wsh':
42
+ case 'p2shP2wsh':
43
+ return omitUndefined({
44
+ nonWitnessUtxo,
45
+ partialSig: getPartialSigs(),
46
+ redeemScript: parsedInput.redeemScript,
47
+ witnessScript: parsedInput.witnessScript,
48
+ });
49
+ case 'p2tr':
50
+ if (!('controlBlock' in parsedInput)) {
51
+ throw new Error(`keypath not implemented`);
52
+ }
53
+ const leafHash = __1.taproot.getTapleafHash(__1.ecc, parsedInput.controlBlock, parsedInput.pubScript);
54
+ return {
55
+ tapLeafScript: [
56
+ {
57
+ controlBlock: parsedInput.controlBlock,
58
+ script: parsedInput.pubScript,
59
+ leafVersion: parsedInput.leafVersion,
60
+ },
61
+ ],
62
+ tapScriptSig: getPartialSigs().map((obj) => ({ ...obj, leafHash })),
63
+ };
64
+ }
65
+ }
66
+ exports.getInputUpdate = getInputUpdate;
67
+ /**
68
+ * Takes a partially signed transaction and removes the scripts and signatures.
69
+ *
70
+ * Inputs must be one of:
71
+ * - p2shP2pk
72
+ * - p2sh 2-of-3
73
+ * - p2shP2wsh 2-of-3
74
+ * - p2wsh 2-of-3
75
+ * - p2tr script path 2-of-2
76
+ *
77
+ * @param tx the partially signed transaction
78
+ * @param prevOuts
79
+ *
80
+ * @return the removed scripts and signatures, ready to be added to a PSBT
81
+ */
82
+ function unsign(tx, prevOuts) {
83
+ return tx.ins.map((input, vin) => {
84
+ const update = getInputUpdate(tx, vin, prevOuts);
85
+ input.witness = [];
86
+ input.script = Buffer.alloc(0);
87
+ return update;
88
+ });
89
+ }
90
+ exports.unsign = unsign;
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fromHalfSigned.js","sourceRoot":"","sources":["../../../../src/bitgo/psbt/fromHalfSigned.ts"],"names":[],"mappings":"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","sourcesContent":["import { PsbtInputUpdate, PartialSig } from 'bip174/src/lib/interfaces';\nimport { ecc as eccLib, TxOutput, taproot, getMainnet, networks } from '../..';\nimport { UtxoTransaction } from '../UtxoTransaction';\nimport { parseSignatureScript } from '../parseInput';\nimport { getSignaturesWithPublicKeys } from '../signature';\nimport { hasWitnessData } from '../outputScripts';\n\nfunction omitUndefined<T extends Record<string, unknown>>(v: Record<string, unknown>): T {\n  return Object.fromEntries(Object.entries(v).filter(([k, v]) => v !== undefined)) as T;\n}\n\nexport function getInputUpdate(\n  tx: UtxoTransaction<bigint>,\n  vin: number,\n  prevOuts: (TxOutput<bigint> & { prevTx?: Buffer })[]\n): PsbtInputUpdate {\n  const nonWitnessUtxo = prevOuts[vin].prevTx;\n  const { script, witness } = tx.ins[vin];\n  if (script.length === 0 && witness.length === 0) {\n    return nonWitnessUtxo ? { nonWitnessUtxo } : {};\n  }\n\n  const parsedInput = parseSignatureScript(tx.ins[vin]);\n\n  function getPartialSigs(): PartialSig[] {\n    return getSignaturesWithPublicKeys(tx, vin, prevOuts, parsedInput.publicKeys).flatMap((signature, i) =>\n      signature\n        ? [\n            {\n              pubkey: parsedInput.publicKeys[i],\n              signature,\n            },\n          ]\n        : []\n    );\n  }\n  // Because Zcash directly hashes the value for non-segwit transactions, we do not need to check indirectly\n  // with the previous transaction. Therefore, we can treat Zcash non-segwit transactions as Bitcoin\n  // segwit transactions\n  if (!hasWitnessData(parsedInput.scriptType) && !nonWitnessUtxo && getMainnet(tx.network) !== networks.zcash) {\n    throw new Error(`scriptType ${parsedInput.scriptType} requires prevTx Buffer`);\n  }\n\n  switch (parsedInput.scriptType) {\n    case 'p2shP2pk':\n      return {\n        nonWitnessUtxo,\n        partialSig: [{ pubkey: parsedInput.publicKeys[0], signature: parsedInput.signatures[0] }],\n      };\n    case 'p2sh':\n    case 'p2wsh':\n    case 'p2shP2wsh':\n      return omitUndefined({\n        nonWitnessUtxo,\n        partialSig: getPartialSigs(),\n        redeemScript: parsedInput.redeemScript,\n        witnessScript: parsedInput.witnessScript,\n      });\n    case 'p2tr':\n      if (!('controlBlock' in parsedInput)) {\n        throw new Error(`keypath not implemented`);\n      }\n      const leafHash = taproot.getTapleafHash(eccLib, parsedInput.controlBlock, parsedInput.pubScript);\n      return {\n        tapLeafScript: [\n          {\n            controlBlock: parsedInput.controlBlock,\n            script: parsedInput.pubScript,\n            leafVersion: parsedInput.leafVersion,\n          },\n        ],\n        tapScriptSig: getPartialSigs().map((obj) => ({ ...obj, leafHash })),\n      };\n  }\n}\n\n/**\n * Takes a partially signed transaction and removes the scripts and signatures.\n *\n * Inputs must be one of:\n *  - p2shP2pk\n *  - p2sh 2-of-3\n *  - p2shP2wsh 2-of-3\n *  - p2wsh 2-of-3\n *  - p2tr script path 2-of-2\n *\n * @param tx the partially signed transaction\n * @param prevOuts\n *\n * @return the removed scripts and signatures, ready to be added to a PSBT\n */\nexport function unsign(tx: UtxoTransaction<bigint>, prevOuts: TxOutput<bigint>[]): PsbtInputUpdate[] {\n  return tx.ins.map((input, vin) => {\n    const update = getInputUpdate(tx, vin, prevOuts);\n    input.witness = [];\n    input.script = Buffer.alloc(0);\n    return update;\n  });\n}\n"]}
@@ -0,0 +1,6 @@
1
+ /// <reference types="node" />
2
+ export declare function isP2wsh(scriptPubkey: Buffer, redeemScript?: Buffer): boolean;
3
+ export declare function isP2wpkh(scriptPubkey: Buffer, redeemScript?: Buffer): boolean;
4
+ export declare function isTaproot(scriptPubkey: Buffer): boolean;
5
+ export declare function isSegwit(scriptPubkey: Buffer, redeemScript?: Buffer): boolean;
6
+ //# sourceMappingURL=scriptTypes.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scriptTypes.d.ts","sourceRoot":"","sources":["../../../../src/bitgo/psbt/scriptTypes.ts"],"names":[],"mappings":";AAEA,wBAAgB,OAAO,CAAC,YAAY,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,MAAM,GAAG,OAAO,CAG5E;AAED,wBAAgB,QAAQ,CAAC,YAAY,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,MAAM,GAAG,OAAO,CAG7E;AAED,wBAAgB,SAAS,CAAC,YAAY,EAAE,MAAM,GAAG,OAAO,CAEvD;AAED,wBAAgB,QAAQ,CAAC,YAAY,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,MAAM,GAAG,OAAO,CAE7E"}
@@ -0,0 +1,23 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.isSegwit = exports.isTaproot = exports.isP2wpkh = exports.isP2wsh = void 0;
4
+ const opcodes = require("bitcoin-ops");
5
+ function isP2wsh(scriptPubkey, redeemScript) {
6
+ const witnessProgramCandidate = redeemScript !== null && redeemScript !== void 0 ? redeemScript : scriptPubkey;
7
+ return witnessProgramCandidate[0] === opcodes.OP_0 && witnessProgramCandidate.length === 34;
8
+ }
9
+ exports.isP2wsh = isP2wsh;
10
+ function isP2wpkh(scriptPubkey, redeemScript) {
11
+ const witnessProgramCandidate = redeemScript !== null && redeemScript !== void 0 ? redeemScript : scriptPubkey;
12
+ return witnessProgramCandidate[0] === opcodes.OP_0 && witnessProgramCandidate.length === 22;
13
+ }
14
+ exports.isP2wpkh = isP2wpkh;
15
+ function isTaproot(scriptPubkey) {
16
+ return scriptPubkey[0] === opcodes.OP_1 && scriptPubkey.length === 34;
17
+ }
18
+ exports.isTaproot = isTaproot;
19
+ function isSegwit(scriptPubkey, redeemScript) {
20
+ return isTaproot(scriptPubkey) || isP2wsh(scriptPubkey, redeemScript) || isP2wpkh(scriptPubkey, redeemScript);
21
+ }
22
+ exports.isSegwit = isSegwit;
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NyaXB0VHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvYml0Z28vcHNidC9zY3JpcHRUeXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx1Q0FBdUM7QUFFdkMsU0FBZ0IsT0FBTyxDQUFDLFlBQW9CLEVBQUUsWUFBcUI7SUFDakUsTUFBTSx1QkFBdUIsR0FBRyxZQUFZLGFBQVosWUFBWSxjQUFaLFlBQVksR0FBSSxZQUFZLENBQUM7SUFDN0QsT0FBTyx1QkFBdUIsQ0FBQyxDQUFDLENBQUMsS0FBSyxPQUFPLENBQUMsSUFBSSxJQUFJLHVCQUF1QixDQUFDLE1BQU0sS0FBSyxFQUFFLENBQUM7QUFDOUYsQ0FBQztBQUhELDBCQUdDO0FBRUQsU0FBZ0IsUUFBUSxDQUFDLFlBQW9CLEVBQUUsWUFBcUI7SUFDbEUsTUFBTSx1QkFBdUIsR0FBRyxZQUFZLGFBQVosWUFBWSxjQUFaLFlBQVksR0FBSSxZQUFZLENBQUM7SUFDN0QsT0FBTyx1QkFBdUIsQ0FBQyxDQUFDLENBQUMsS0FBSyxPQUFPLENBQUMsSUFBSSxJQUFJLHVCQUF1QixDQUFDLE1BQU0sS0FBSyxFQUFFLENBQUM7QUFDOUYsQ0FBQztBQUhELDRCQUdDO0FBRUQsU0FBZ0IsU0FBUyxDQUFDLFlBQW9CO0lBQzVDLE9BQU8sWUFBWSxDQUFDLENBQUMsQ0FBQyxLQUFLLE9BQU8sQ0FBQyxJQUFJLElBQUksWUFBWSxDQUFDLE1BQU0sS0FBSyxFQUFFLENBQUM7QUFDeEUsQ0FBQztBQUZELDhCQUVDO0FBRUQsU0FBZ0IsUUFBUSxDQUFDLFlBQW9CLEVBQUUsWUFBcUI7SUFDbEUsT0FBTyxTQUFTLENBQUMsWUFBWSxDQUFDLElBQUksT0FBTyxDQUFDLFlBQVksRUFBRSxZQUFZLENBQUMsSUFBSSxRQUFRLENBQUMsWUFBWSxFQUFFLFlBQVksQ0FBQyxDQUFDO0FBQ2hILENBQUM7QUFGRCw0QkFFQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIG9wY29kZXMgZnJvbSAnYml0Y29pbi1vcHMnO1xuXG5leHBvcnQgZnVuY3Rpb24gaXNQMndzaChzY3JpcHRQdWJrZXk6IEJ1ZmZlciwgcmVkZWVtU2NyaXB0PzogQnVmZmVyKTogYm9vbGVhbiB7XG4gIGNvbnN0IHdpdG5lc3NQcm9ncmFtQ2FuZGlkYXRlID0gcmVkZWVtU2NyaXB0ID8/IHNjcmlwdFB1YmtleTtcbiAgcmV0dXJuIHdpdG5lc3NQcm9ncmFtQ2FuZGlkYXRlWzBdID09PSBvcGNvZGVzLk9QXzAgJiYgd2l0bmVzc1Byb2dyYW1DYW5kaWRhdGUubGVuZ3RoID09PSAzNDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGlzUDJ3cGtoKHNjcmlwdFB1YmtleTogQnVmZmVyLCByZWRlZW1TY3JpcHQ/OiBCdWZmZXIpOiBib29sZWFuIHtcbiAgY29uc3Qgd2l0bmVzc1Byb2dyYW1DYW5kaWRhdGUgPSByZWRlZW1TY3JpcHQgPz8gc2NyaXB0UHVia2V5O1xuICByZXR1cm4gd2l0bmVzc1Byb2dyYW1DYW5kaWRhdGVbMF0gPT09IG9wY29kZXMuT1BfMCAmJiB3aXRuZXNzUHJvZ3JhbUNhbmRpZGF0ZS5sZW5ndGggPT09IDIyO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNUYXByb290KHNjcmlwdFB1YmtleTogQnVmZmVyKTogYm9vbGVhbiB7XG4gIHJldHVybiBzY3JpcHRQdWJrZXlbMF0gPT09IG9wY29kZXMuT1BfMSAmJiBzY3JpcHRQdWJrZXkubGVuZ3RoID09PSAzNDtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGlzU2Vnd2l0KHNjcmlwdFB1YmtleTogQnVmZmVyLCByZWRlZW1TY3JpcHQ/OiBCdWZmZXIpOiBib29sZWFuIHtcbiAgcmV0dXJuIGlzVGFwcm9vdChzY3JpcHRQdWJrZXkpIHx8IGlzUDJ3c2goc2NyaXB0UHVia2V5LCByZWRlZW1TY3JpcHQpIHx8IGlzUDJ3cGtoKHNjcmlwdFB1YmtleSwgcmVkZWVtU2NyaXB0KTtcbn1cbiJdfQ==