@bitgo-beta/abstract-utxo 1.6.1-alpha.425 → 1.6.1-alpha.426

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (130) hide show
  1. package/dist/cjs/src/abstractUtxoCoin.d.ts +21 -5
  2. package/dist/cjs/src/abstractUtxoCoin.d.ts.map +1 -1
  3. package/dist/cjs/src/abstractUtxoCoin.js +52 -26
  4. package/dist/cjs/src/index.d.ts +2 -2
  5. package/dist/cjs/src/index.d.ts.map +1 -1
  6. package/dist/cjs/src/index.js +3 -3
  7. package/dist/cjs/src/recovery/backupKeyRecovery.js +4 -4
  8. package/dist/cjs/src/recovery/crossChainRecovery.d.ts +10 -0
  9. package/dist/cjs/src/recovery/crossChainRecovery.d.ts.map +1 -1
  10. package/dist/cjs/src/recovery/crossChainRecovery.js +48 -4
  11. package/dist/cjs/src/transaction/decode.d.ts +11 -0
  12. package/dist/cjs/src/transaction/decode.d.ts.map +1 -0
  13. package/dist/cjs/src/transaction/decode.js +81 -0
  14. package/dist/cjs/src/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
  15. package/dist/cjs/src/transaction/descriptor/verifyTransaction.js +3 -2
  16. package/dist/cjs/src/transaction/explainTransaction.js +3 -3
  17. package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts +24 -0
  18. package/dist/cjs/src/transaction/fixedScript/SigningError.d.ts.map +1 -0
  19. package/dist/cjs/src/transaction/fixedScript/SigningError.js +26 -0
  20. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts +1 -1
  21. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
  22. package/dist/cjs/src/transaction/fixedScript/explainPsbtWasm.js +1 -1
  23. package/dist/cjs/src/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  24. package/dist/cjs/src/transaction/fixedScript/explainTransaction.js +2 -2
  25. package/dist/cjs/src/transaction/fixedScript/index.d.ts +3 -0
  26. package/dist/cjs/src/transaction/fixedScript/index.d.ts.map +1 -1
  27. package/dist/cjs/src/transaction/fixedScript/index.js +18 -1
  28. package/dist/cjs/src/transaction/fixedScript/musig2.d.ts +4 -0
  29. package/dist/cjs/src/transaction/fixedScript/musig2.d.ts.map +1 -0
  30. package/dist/cjs/src/transaction/fixedScript/musig2.js +3 -0
  31. package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts +7 -0
  32. package/dist/cjs/src/transaction/fixedScript/replayProtection.d.ts.map +1 -0
  33. package/dist/cjs/src/transaction/fixedScript/replayProtection.js +78 -0
  34. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts +30 -0
  35. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -0
  36. package/dist/cjs/src/transaction/fixedScript/signLegacyTransaction.js +152 -0
  37. package/dist/cjs/src/transaction/fixedScript/signPsbt.d.ts +30 -0
  38. package/dist/cjs/src/transaction/fixedScript/signPsbt.d.ts.map +1 -0
  39. package/dist/cjs/src/transaction/fixedScript/signPsbt.js +174 -0
  40. package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts +22 -0
  41. package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -0
  42. package/dist/cjs/src/transaction/fixedScript/signPsbtWasm.js +129 -0
  43. package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts +5 -6
  44. package/dist/cjs/src/transaction/fixedScript/signTransaction.d.ts.map +1 -1
  45. package/dist/cjs/src/transaction/fixedScript/signTransaction.js +28 -79
  46. package/dist/cjs/src/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
  47. package/dist/cjs/src/transaction/fixedScript/verifyTransaction.js +3 -2
  48. package/dist/cjs/src/transaction/signTransaction.d.ts.map +1 -1
  49. package/dist/cjs/src/transaction/signTransaction.js +5 -2
  50. package/dist/cjs/src/transaction/types.d.ts +5 -0
  51. package/dist/cjs/src/transaction/types.d.ts.map +1 -1
  52. package/dist/cjs/src/transaction/types.js +5 -1
  53. package/dist/cjs/test/unit/recovery/crossChainRecovery.js +24 -1
  54. package/dist/cjs/test/unit/transaction/fixedScript/explainPsbt.js +3 -3
  55. package/dist/cjs/test/unit/transaction/fixedScript/parsePsbt.js +2 -2
  56. package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.d.ts +2 -0
  57. package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.d.ts.map +1 -0
  58. package/dist/cjs/test/unit/transaction/fixedScript/replayProtection.js +59 -0
  59. package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.d.ts +2 -0
  60. package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.d.ts.map +1 -0
  61. package/dist/cjs/test/unit/transaction/fixedScript/signPsbt.js +157 -0
  62. package/dist/cjs/test/unit/transaction/fixedScript/util.d.ts.map +1 -1
  63. package/dist/cjs/test/unit/transaction/fixedScript/util.js +2 -1
  64. package/dist/cjs/test/unit/transaction.js +65 -42
  65. package/dist/cjs/tsconfig.tsbuildinfo +1 -1
  66. package/dist/esm/abstractUtxoCoin.d.ts +21 -5
  67. package/dist/esm/abstractUtxoCoin.d.ts.map +1 -1
  68. package/dist/esm/abstractUtxoCoin.js +53 -27
  69. package/dist/esm/index.d.ts +2 -2
  70. package/dist/esm/index.d.ts.map +1 -1
  71. package/dist/esm/index.js +3 -3
  72. package/dist/esm/recovery/backupKeyRecovery.js +2 -2
  73. package/dist/esm/recovery/crossChainRecovery.d.ts +10 -0
  74. package/dist/esm/recovery/crossChainRecovery.d.ts.map +1 -1
  75. package/dist/esm/recovery/crossChainRecovery.js +46 -3
  76. package/dist/esm/transaction/decode.d.ts +11 -0
  77. package/dist/esm/transaction/decode.d.ts.map +1 -0
  78. package/dist/esm/transaction/decode.js +43 -0
  79. package/dist/esm/transaction/descriptor/verifyTransaction.d.ts.map +1 -1
  80. package/dist/esm/transaction/descriptor/verifyTransaction.js +3 -2
  81. package/dist/esm/transaction/explainTransaction.js +3 -3
  82. package/dist/esm/transaction/fixedScript/SigningError.d.ts +24 -0
  83. package/dist/esm/transaction/fixedScript/SigningError.d.ts.map +1 -0
  84. package/dist/esm/transaction/fixedScript/SigningError.js +21 -0
  85. package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts +1 -1
  86. package/dist/esm/transaction/fixedScript/explainPsbtWasm.d.ts.map +1 -1
  87. package/dist/esm/transaction/fixedScript/explainPsbtWasm.js +1 -1
  88. package/dist/esm/transaction/fixedScript/explainTransaction.d.ts.map +1 -1
  89. package/dist/esm/transaction/fixedScript/explainTransaction.js +2 -2
  90. package/dist/esm/transaction/fixedScript/index.d.ts +3 -0
  91. package/dist/esm/transaction/fixedScript/index.d.ts.map +1 -1
  92. package/dist/esm/transaction/fixedScript/index.js +4 -1
  93. package/dist/esm/transaction/fixedScript/musig2.d.ts +4 -0
  94. package/dist/esm/transaction/fixedScript/musig2.d.ts.map +1 -0
  95. package/dist/esm/transaction/fixedScript/musig2.js +2 -0
  96. package/dist/esm/transaction/fixedScript/replayProtection.d.ts +7 -0
  97. package/dist/esm/transaction/fixedScript/replayProtection.d.ts.map +1 -0
  98. package/dist/esm/transaction/fixedScript/replayProtection.js +39 -0
  99. package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts +30 -0
  100. package/dist/esm/transaction/fixedScript/signLegacyTransaction.d.ts.map +1 -0
  101. package/dist/esm/transaction/fixedScript/signLegacyTransaction.js +112 -0
  102. package/dist/esm/transaction/fixedScript/signPsbt.d.ts +30 -0
  103. package/dist/esm/transaction/fixedScript/signPsbt.d.ts.map +1 -0
  104. package/dist/esm/transaction/fixedScript/signPsbt.js +134 -0
  105. package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts +22 -0
  106. package/dist/esm/transaction/fixedScript/signPsbtWasm.d.ts.map +1 -0
  107. package/dist/esm/transaction/fixedScript/signPsbtWasm.js +122 -0
  108. package/dist/esm/transaction/fixedScript/signTransaction.d.ts +5 -6
  109. package/dist/esm/transaction/fixedScript/signTransaction.d.ts.map +1 -1
  110. package/dist/esm/transaction/fixedScript/signTransaction.js +28 -79
  111. package/dist/esm/transaction/fixedScript/verifyTransaction.d.ts.map +1 -1
  112. package/dist/esm/transaction/fixedScript/verifyTransaction.js +3 -2
  113. package/dist/esm/transaction/signTransaction.d.ts.map +1 -1
  114. package/dist/esm/transaction/signTransaction.js +5 -2
  115. package/dist/esm/transaction/types.d.ts +5 -0
  116. package/dist/esm/transaction/types.d.ts.map +1 -1
  117. package/dist/esm/transaction/types.js +4 -2
  118. package/package.json +13 -13
  119. package/dist/cjs/src/replayProtection.d.ts +0 -5
  120. package/dist/cjs/src/replayProtection.d.ts.map +0 -1
  121. package/dist/cjs/src/replayProtection.js +0 -58
  122. package/dist/cjs/src/sign.d.ts +0 -54
  123. package/dist/cjs/src/sign.d.ts.map +0 -1
  124. package/dist/cjs/src/sign.js +0 -205
  125. package/dist/esm/replayProtection.d.ts +0 -5
  126. package/dist/esm/replayProtection.d.ts.map +0 -1
  127. package/dist/esm/replayProtection.js +0 -20
  128. package/dist/esm/sign.d.ts +0 -54
  129. package/dist/esm/sign.d.ts.map +0 -1
  130. package/dist/esm/sign.js +0 -162
@@ -0,0 +1 @@
1
+ {"version":3,"file":"replayProtection.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/replayProtection.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAGhD,eAAO,MAAM,UAAU,qBAA2F,CAAC;AAEnH,eAAO,MAAM,aAAa,qBAA2F,CAAC;AAEtH,wBAAgB,0BAA0B,CAAC,OAAO,EAAE,OAAO,CAAC,OAAO,GAAG,MAAM,EAAE,CAU7E;AAWD,wBAAgB,4BAA4B,CAC1C,OAAO,EAAE,OAAO,CAAC,OAAO,EACxB,MAAM,GAAE,SAAS,GAAG,UAAsB,GACzC,MAAM,EAAE,CAWV;AAED,wBAAgB,yBAAyB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACvE,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,EACjC,OAAO,EAAE,OAAO,CAAC,OAAO,GACvB,OAAO,CAET"}
@@ -0,0 +1,78 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.pubkeyTestnet = exports.pubkeyProd = void 0;
37
+ exports.getReplayProtectionPubkeys = getReplayProtectionPubkeys;
38
+ exports.getReplayProtectionAddresses = getReplayProtectionAddresses;
39
+ exports.isReplayProtectionUnspent = isReplayProtectionUnspent;
40
+ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
41
+ const wasm_utxo_1 = require("@bitgo/wasm-utxo");
42
+ exports.pubkeyProd = Buffer.from('0255b9f71ac2c78fffd83e3e37b9e17ae70d5437b7f56d0ed2e93b7de08015aa59', 'hex');
43
+ exports.pubkeyTestnet = Buffer.from('0219da48412c2268865fe8c126327d1b12eee350a3b69eb09e3323cc9a11828945', 'hex');
44
+ function getReplayProtectionPubkeys(network) {
45
+ switch (network) {
46
+ case utxolib.networks.bitcoincash:
47
+ case utxolib.networks.bitcoinsv:
48
+ return [exports.pubkeyProd];
49
+ case utxolib.networks.bitcoinsvTestnet:
50
+ case utxolib.networks.bitcoincashTestnet:
51
+ return [exports.pubkeyTestnet];
52
+ }
53
+ return [];
54
+ }
55
+ // sh(pk(pubkeyProd))
56
+ // 33p1q7mTGyeM5UnZERGiMcVUkY12SCsatA
57
+ // bitcoincash:pqt5x9w0m6z0f3znjkkx79wl3l7ywrszesemp8xgpf
58
+ const replayProtectionScriptsProd = [Buffer.from('a914174315cfde84f4c45395ac6f15df8ffc470e02cc87', 'hex')];
59
+ // sh(pk(pubkeyTestnet))
60
+ // 2MuMnPoSDgWEpNWH28X2nLtYMXQJCyT61eY
61
+ // bchtest:pqtjmnzwqffkrk2349g3cecfwwjwxusvnq87n07cal
62
+ const replayProtectionScriptsTestnet = [Buffer.from('a914172dcc4e025361d951a9511c670973a4e3720c9887', 'hex')];
63
+ function getReplayProtectionAddresses(network, format = 'default') {
64
+ switch (network) {
65
+ case utxolib.networks.bitcoincash:
66
+ case utxolib.networks.bitcoinsv:
67
+ return replayProtectionScriptsProd.map((script) => wasm_utxo_1.utxolibCompat.fromOutputScript(script, network, format));
68
+ case utxolib.networks.bitcoinsvTestnet:
69
+ case utxolib.networks.bitcoincashTestnet:
70
+ return replayProtectionScriptsTestnet.map((script) => wasm_utxo_1.utxolibCompat.fromOutputScript(script, network, format));
71
+ default:
72
+ return [];
73
+ }
74
+ }
75
+ function isReplayProtectionUnspent(u, network) {
76
+ return getReplayProtectionAddresses(network).includes(u.address);
77
+ }
78
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVwbGF5UHJvdGVjdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy90cmFuc2FjdGlvbi9maXhlZFNjcmlwdC9yZXBsYXlQcm90ZWN0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQU9BLGdFQVVDO0FBV0Qsb0VBY0M7QUFFRCw4REFLQztBQWpERCw4REFBZ0Q7QUFDaEQsZ0RBQWlEO0FBRXBDLFFBQUEsVUFBVSxHQUFHLE1BQU0sQ0FBQyxJQUFJLENBQUMsb0VBQW9FLEVBQUUsS0FBSyxDQUFDLENBQUM7QUFFdEcsUUFBQSxhQUFhLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxvRUFBb0UsRUFBRSxLQUFLLENBQUMsQ0FBQztBQUV0SCxTQUFnQiwwQkFBMEIsQ0FBQyxPQUF3QjtJQUNqRSxRQUFRLE9BQU8sRUFBRSxDQUFDO1FBQ2hCLEtBQUssT0FBTyxDQUFDLFFBQVEsQ0FBQyxXQUFXLENBQUM7UUFDbEMsS0FBSyxPQUFPLENBQUMsUUFBUSxDQUFDLFNBQVM7WUFDN0IsT0FBTyxDQUFDLGtCQUFVLENBQUMsQ0FBQztRQUN0QixLQUFLLE9BQU8sQ0FBQyxRQUFRLENBQUMsZ0JBQWdCLENBQUM7UUFDdkMsS0FBSyxPQUFPLENBQUMsUUFBUSxDQUFDLGtCQUFrQjtZQUN0QyxPQUFPLENBQUMscUJBQWEsQ0FBQyxDQUFDO0lBQzNCLENBQUM7SUFDRCxPQUFPLEVBQUUsQ0FBQztBQUNaLENBQUM7QUFFRCxxQkFBcUI7QUFDckIscUNBQXFDO0FBQ3JDLHlEQUF5RDtBQUN6RCxNQUFNLDJCQUEyQixHQUFHLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxnREFBZ0QsRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDO0FBQzNHLHdCQUF3QjtBQUN4QixzQ0FBc0M7QUFDdEMscURBQXFEO0FBQ3JELE1BQU0sOEJBQThCLEdBQUcsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLGdEQUFnRCxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUM7QUFFOUcsU0FBZ0IsNEJBQTRCLENBQzFDLE9BQXdCLEVBQ3hCLFNBQWlDLFNBQVM7SUFFMUMsUUFBUSxPQUFPLEVBQUUsQ0FBQztRQUNoQixLQUFLLE9BQU8sQ0FBQyxRQUFRLENBQUMsV0FBVyxDQUFDO1FBQ2xDLEtBQUssT0FBTyxDQUFDLFFBQVEsQ0FBQyxTQUFTO1lBQzdCLE9BQU8sMkJBQTJCLENBQUMsR0FBRyxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyx5QkFBYSxDQUFDLGdCQUFnQixDQUFDLE1BQU0sRUFBRSxPQUFPLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBQztRQUM5RyxLQUFLLE9BQU8sQ0FBQyxRQUFRLENBQUMsZ0JBQWdCLENBQUM7UUFDdkMsS0FBSyxPQUFPLENBQUMsUUFBUSxDQUFDLGtCQUFrQjtZQUN0QyxPQUFPLDhCQUE4QixDQUFDLEdBQUcsQ0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMseUJBQWEsQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNLEVBQUUsT0FBTyxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQUM7UUFDakg7WUFDRSxPQUFPLEVBQUUsQ0FBQztJQUNkLENBQUM7QUFDSCxDQUFDO0FBRUQsU0FBZ0IseUJBQXlCLENBQ3ZDLENBQWlDLEVBQ2pDLE9BQXdCO0lBRXhCLE9BQU8sNEJBQTRCLENBQUMsT0FBTyxDQUFDLENBQUMsUUFBUSxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQztBQUNuRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgdXR4b2xpYiBmcm9tICdAYml0Z28tYmV0YS91dHhvLWxpYic7XG5pbXBvcnQgeyB1dHhvbGliQ29tcGF0IH0gZnJvbSAnQGJpdGdvL3dhc20tdXR4byc7XG5cbmV4cG9ydCBjb25zdCBwdWJrZXlQcm9kID0gQnVmZmVyLmZyb20oJzAyNTViOWY3MWFjMmM3OGZmZmQ4M2UzZTM3YjllMTdhZTcwZDU0MzdiN2Y1NmQwZWQyZTkzYjdkZTA4MDE1YWE1OScsICdoZXgnKTtcblxuZXhwb3J0IGNvbnN0IHB1YmtleVRlc3RuZXQgPSBCdWZmZXIuZnJvbSgnMDIxOWRhNDg0MTJjMjI2ODg2NWZlOGMxMjYzMjdkMWIxMmVlZTM1MGEzYjY5ZWIwOWUzMzIzY2M5YTExODI4OTQ1JywgJ2hleCcpO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0UmVwbGF5UHJvdGVjdGlvblB1YmtleXMobmV0d29yazogdXR4b2xpYi5OZXR3b3JrKTogQnVmZmVyW10ge1xuICBzd2l0Y2ggKG5ldHdvcmspIHtcbiAgICBjYXNlIHV0eG9saWIubmV0d29ya3MuYml0Y29pbmNhc2g6XG4gICAgY2FzZSB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5zdjpcbiAgICAgIHJldHVybiBbcHVia2V5UHJvZF07XG4gICAgY2FzZSB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5zdlRlc3RuZXQ6XG4gICAgY2FzZSB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5jYXNoVGVzdG5ldDpcbiAgICAgIHJldHVybiBbcHVia2V5VGVzdG5ldF07XG4gIH1cbiAgcmV0dXJuIFtdO1xufVxuXG4vLyBzaChwayhwdWJrZXlQcm9kKSlcbi8vIDMzcDFxN21UR3llTTVVblpFUkdpTWNWVWtZMTJTQ3NhdEFcbi8vIGJpdGNvaW5jYXNoOnBxdDV4OXcwbTZ6MGYzem5qa2t4Nzl3bDNsN3l3cnN6ZXNlbXA4eGdwZlxuY29uc3QgcmVwbGF5UHJvdGVjdGlvblNjcmlwdHNQcm9kID0gW0J1ZmZlci5mcm9tKCdhOTE0MTc0MzE1Y2ZkZTg0ZjRjNDUzOTVhYzZmMTVkZjhmZmM0NzBlMDJjYzg3JywgJ2hleCcpXTtcbi8vIHNoKHBrKHB1YmtleVRlc3RuZXQpKVxuLy8gMk11TW5Qb1NEZ1dFcE5XSDI4WDJuTHRZTVhRSkN5VDYxZVlcbi8vIGJjaHRlc3Q6cHF0am1uendxZmZrcmsyMzQ5ZzNjZWNmd3dqd3h1c3ZucTg3bjA3Y2FsXG5jb25zdCByZXBsYXlQcm90ZWN0aW9uU2NyaXB0c1Rlc3RuZXQgPSBbQnVmZmVyLmZyb20oJ2E5MTQxNzJkY2M0ZTAyNTM2MWQ5NTFhOTUxMWM2NzA5NzNhNGUzNzIwYzk4ODcnLCAnaGV4JyldO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0UmVwbGF5UHJvdGVjdGlvbkFkZHJlc3NlcyhcbiAgbmV0d29yazogdXR4b2xpYi5OZXR3b3JrLFxuICBmb3JtYXQ6ICdkZWZhdWx0JyB8ICdjYXNoYWRkcicgPSAnZGVmYXVsdCdcbik6IHN0cmluZ1tdIHtcbiAgc3dpdGNoIChuZXR3b3JrKSB7XG4gICAgY2FzZSB1dHhvbGliLm5ldHdvcmtzLmJpdGNvaW5jYXNoOlxuICAgIGNhc2UgdXR4b2xpYi5uZXR3b3Jrcy5iaXRjb2luc3Y6XG4gICAgICByZXR1cm4gcmVwbGF5UHJvdGVjdGlvblNjcmlwdHNQcm9kLm1hcCgoc2NyaXB0KSA9PiB1dHhvbGliQ29tcGF0LmZyb21PdXRwdXRTY3JpcHQoc2NyaXB0LCBuZXR3b3JrLCBmb3JtYXQpKTtcbiAgICBjYXNlIHV0eG9saWIubmV0d29ya3MuYml0Y29pbnN2VGVzdG5ldDpcbiAgICBjYXNlIHV0eG9saWIubmV0d29ya3MuYml0Y29pbmNhc2hUZXN0bmV0OlxuICAgICAgcmV0dXJuIHJlcGxheVByb3RlY3Rpb25TY3JpcHRzVGVzdG5ldC5tYXAoKHNjcmlwdCkgPT4gdXR4b2xpYkNvbXBhdC5mcm9tT3V0cHV0U2NyaXB0KHNjcmlwdCwgbmV0d29yaywgZm9ybWF0KSk7XG4gICAgZGVmYXVsdDpcbiAgICAgIHJldHVybiBbXTtcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNSZXBsYXlQcm90ZWN0aW9uVW5zcGVudDxUTnVtYmVyIGV4dGVuZHMgbnVtYmVyIHwgYmlnaW50PihcbiAgdTogdXR4b2xpYi5iaXRnby5VbnNwZW50PFROdW1iZXI+LFxuICBuZXR3b3JrOiB1dHhvbGliLk5ldHdvcmtcbik6IGJvb2xlYW4ge1xuICByZXR1cm4gZ2V0UmVwbGF5UHJvdGVjdGlvbkFkZHJlc3NlcyhuZXR3b3JrKS5pbmNsdWRlcyh1LmFkZHJlc3MpO1xufVxuIl19
@@ -0,0 +1,30 @@
1
+ import * as utxolib from '@bitgo-beta/utxo-lib';
2
+ import { BIP32Interface } from '@bitgo-beta/secp256k1';
3
+ type Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;
4
+ type RootWalletKeys = utxolib.bitgo.RootWalletKeys;
5
+ /**
6
+ * Sign all inputs of a wallet transaction and verify signatures after signing.
7
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
8
+ * failed.
9
+ *
10
+ * @param transaction - wallet transaction (builder) to be signed
11
+ * @param unspents - transaction unspents
12
+ * @param walletSigner - signing parameters
13
+ * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.
14
+ * @param replayProtectionAddresses - List of replay protection addresses to skip signing
15
+ */
16
+ export declare function signAndVerifyWalletTransaction<TNumber extends number | bigint>(transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>, unspents: Unspent<TNumber>[], walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>, { isLastSignature, replayProtectionAddresses, }: {
17
+ isLastSignature: boolean;
18
+ replayProtectionAddresses?: string[];
19
+ }): utxolib.bitgo.UtxoTransaction<TNumber>;
20
+ export declare function signLegacyTransaction<TNumber extends number | bigint>(tx: utxolib.bitgo.UtxoTransaction<TNumber>, signerKeychain: BIP32Interface | undefined, params: {
21
+ isLastSignature: boolean;
22
+ signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
23
+ txInfo: {
24
+ unspents?: utxolib.bitgo.Unspent<TNumber>[];
25
+ } | undefined;
26
+ pubs: string[] | undefined;
27
+ cosignerPub: string | undefined;
28
+ }): utxolib.bitgo.UtxoTransaction<TNumber>;
29
+ export {};
30
+ //# sourceMappingURL=signLegacyTransaction.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"signLegacyTransaction.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/signLegacyTransaction.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,cAAc,EAAS,MAAM,uBAAuB,CAAC;AAY9D,KAAK,OAAO,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,GAAG,MAAM,IAAI,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;AAExF,KAAK,cAAc,GAAG,OAAO,CAAC,KAAK,CAAC,cAAc,CAAC;AAEnD;;;;;;;;;;GAUG;AACH,wBAAgB,8BAA8B,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EAC5E,WAAW,EAAE,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,GAAG,OAAO,CAAC,KAAK,CAAC,sBAAsB,CAAC,OAAO,CAAC,EACnG,QAAQ,EAAE,OAAO,CAAC,OAAO,CAAC,EAAE,EAC5B,YAAY,EAAE,OAAO,CAAC,KAAK,CAAC,mBAAmB,CAAC,cAAc,CAAC,EAC/D,EACE,eAAe,EACf,yBAAyB,GAC1B,EAAE;IACD,eAAe,EAAE,OAAO,CAAC;IACzB,yBAAyB,CAAC,EAAE,MAAM,EAAE,CAAC;CACtC,GACA,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,CAwExC;AAED,wBAAgB,qBAAqB,CAAC,OAAO,SAAS,MAAM,GAAG,MAAM,EACnE,EAAE,EAAE,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,EAC1C,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,MAAM,EAAE;IACN,eAAe,EAAE,OAAO,CAAC;IACzB,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,MAAM,EAAE;QAAE,QAAQ,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAA;KAAE,GAAG,SAAS,CAAC;IACpE,IAAI,EAAE,MAAM,EAAE,GAAG,SAAS,CAAC;IAC3B,WAAW,EAAE,MAAM,GAAG,SAAS,CAAC;CACjC,GACA,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,OAAO,CAAC,CA4BxC"}
@@ -0,0 +1,152 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ exports.signAndVerifyWalletTransaction = signAndVerifyWalletTransaction;
40
+ exports.signLegacyTransaction = signLegacyTransaction;
41
+ const assert_1 = __importDefault(require("assert"));
42
+ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
43
+ const secp256k1_1 = require("@bitgo-beta/secp256k1");
44
+ const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
45
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
46
+ const debug_1 = __importDefault(require("debug"));
47
+ const replayProtection_1 = require("./replayProtection");
48
+ const SigningError_1 = require("./SigningError");
49
+ const debug = (0, debug_1.default)('bitgo:v2:utxo');
50
+ const { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;
51
+ /**
52
+ * Sign all inputs of a wallet transaction and verify signatures after signing.
53
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
54
+ * failed.
55
+ *
56
+ * @param transaction - wallet transaction (builder) to be signed
57
+ * @param unspents - transaction unspents
58
+ * @param walletSigner - signing parameters
59
+ * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.
60
+ * @param replayProtectionAddresses - List of replay protection addresses to skip signing
61
+ */
62
+ function signAndVerifyWalletTransaction(transaction, unspents, walletSigner, { isLastSignature, replayProtectionAddresses, }) {
63
+ const network = transaction.network;
64
+ if (replayProtectionAddresses === undefined) {
65
+ replayProtectionAddresses = (0, replayProtection_1.getReplayProtectionAddresses)(network);
66
+ }
67
+ const prevOutputs = unspents.map((u) => toOutput(u, network));
68
+ let txBuilder;
69
+ if (transaction instanceof utxolib.bitgo.UtxoTransaction) {
70
+ txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction(transaction, prevOutputs);
71
+ if (transaction.ins.length !== unspents.length) {
72
+ throw new Error(`transaction inputs must match unspents`);
73
+ }
74
+ }
75
+ else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {
76
+ txBuilder = transaction;
77
+ }
78
+ else {
79
+ throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);
80
+ }
81
+ const signErrors = unspents
82
+ .map((unspent, inputIndex) => {
83
+ if (replayProtectionAddresses.includes(unspent.address)) {
84
+ debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);
85
+ return;
86
+ }
87
+ if (!isWalletUnspent(unspent)) {
88
+ return SigningError_1.InputSigningError.expectedWalletUnspent(inputIndex, null, unspent);
89
+ }
90
+ try {
91
+ signInputWithUnspent(txBuilder, inputIndex, unspent, walletSigner);
92
+ debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);
93
+ }
94
+ catch (e) {
95
+ return new SigningError_1.InputSigningError(inputIndex, null, unspent, e);
96
+ }
97
+ })
98
+ .filter((e) => e !== undefined);
99
+ const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();
100
+ const verifyErrors = signedTransaction.ins
101
+ .map((input, inputIndex) => {
102
+ const unspent = unspents[inputIndex];
103
+ if (replayProtectionAddresses.includes(unspent.address)) {
104
+ debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, unspents.length);
105
+ return;
106
+ }
107
+ if (!isWalletUnspent(unspent)) {
108
+ return SigningError_1.InputSigningError.expectedWalletUnspent(inputIndex, null, unspent);
109
+ }
110
+ try {
111
+ const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;
112
+ if (!utxolib.bitgo.verifySignatureWithPublicKey(signedTransaction, inputIndex, prevOutputs, publicKey)) {
113
+ return new SigningError_1.InputSigningError(inputIndex, null, unspent, new Error(`invalid signature`));
114
+ }
115
+ }
116
+ catch (e) {
117
+ debug('Invalid signature');
118
+ return new SigningError_1.InputSigningError(inputIndex, null, unspent, e);
119
+ }
120
+ })
121
+ .filter((e) => e !== undefined);
122
+ if (signErrors.length || verifyErrors.length) {
123
+ throw new SigningError_1.TransactionSigningError(signErrors, verifyErrors);
124
+ }
125
+ return signedTransaction;
126
+ }
127
+ function signLegacyTransaction(tx, signerKeychain, params) {
128
+ switch (params.signingStep) {
129
+ case 'signerNonce':
130
+ case 'cosignerNonce':
131
+ /**
132
+ * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
133
+ * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
134
+ */
135
+ return tx;
136
+ }
137
+ if (tx.ins.length !== params.txInfo?.unspents?.length) {
138
+ throw new Error('length of unspents array should equal to the number of transaction inputs');
139
+ }
140
+ if (!params.pubs || !(0, sdk_core_1.isTriple)(params.pubs)) {
141
+ throw new Error(`must provide xpub array`);
142
+ }
143
+ const keychains = params.pubs.map((pub) => secp256k1_1.bip32.fromBase58(pub));
144
+ const cosignerPub = params.cosignerPub ?? params.pubs[2];
145
+ const cosignerKeychain = secp256k1_1.bip32.fromBase58(cosignerPub);
146
+ (0, assert_1.default)(signerKeychain);
147
+ const walletSigner = new utxo_lib_1.bitgo.WalletUnspentSigner(keychains, signerKeychain, cosignerKeychain);
148
+ return signAndVerifyWalletTransaction(tx, params.txInfo.unspents, walletSigner, {
149
+ isLastSignature: params.isLastSignature,
150
+ });
151
+ }
152
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signLegacyTransaction.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/signLegacyTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BIP32Interface, bip32 } from '@bitgo-beta/secp256k1';\nimport { bitgo } from '@bitgo-beta/utxo-lib';\nimport { isTriple, Triple } from '@bitgo-beta/sdk-core';\nimport debugLib from 'debug';\n\nimport { getReplayProtectionAddresses } from './replayProtection';\nimport { InputSigningError, TransactionSigningError } from './SigningError';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nconst { isWalletUnspent, signInputWithUnspent, toOutput } = utxolib.bitgo;\n\ntype Unspent<TNumber extends number | bigint = number> = utxolib.bitgo.Unspent<TNumber>;\n\ntype RootWalletKeys = utxolib.bitgo.RootWalletKeys;\n\n/**\n * Sign all inputs of a wallet transaction and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * @param transaction - wallet transaction (builder) to be signed\n * @param unspents - transaction unspents\n * @param walletSigner - signing parameters\n * @param isLastSignature - Returns full-signed transaction when true. Builds half-signed when false.\n * @param replayProtectionAddresses - List of replay protection addresses to skip signing\n */\nexport function signAndVerifyWalletTransaction<TNumber extends number | bigint>(\n  transaction: utxolib.bitgo.UtxoTransaction<TNumber> | utxolib.bitgo.UtxoTransactionBuilder<TNumber>,\n  unspents: Unspent<TNumber>[],\n  walletSigner: utxolib.bitgo.WalletUnspentSigner<RootWalletKeys>,\n  {\n    isLastSignature,\n    replayProtectionAddresses,\n  }: {\n    isLastSignature: boolean;\n    replayProtectionAddresses?: string[];\n  }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  const network = transaction.network as utxolib.Network;\n  if (replayProtectionAddresses === undefined) {\n    replayProtectionAddresses = getReplayProtectionAddresses(network);\n  }\n  const prevOutputs = unspents.map((u) => toOutput(u, network));\n\n  let txBuilder: utxolib.bitgo.UtxoTransactionBuilder<TNumber>;\n  if (transaction instanceof utxolib.bitgo.UtxoTransaction) {\n    txBuilder = utxolib.bitgo.createTransactionBuilderFromTransaction<TNumber>(transaction, prevOutputs);\n    if (transaction.ins.length !== unspents.length) {\n      throw new Error(`transaction inputs must match unspents`);\n    }\n  } else if (transaction instanceof utxolib.bitgo.UtxoTransactionBuilder) {\n    txBuilder = transaction;\n  } else {\n    throw new Error(`must pass UtxoTransaction or UtxoTransactionBuilder`);\n  }\n\n  const signErrors: InputSigningError<TNumber>[] = unspents\n    .map((unspent: Unspent<TNumber>, inputIndex: number) => {\n      if (replayProtectionAddresses.includes(unspent.address)) {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, unspents.length);\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, null, unspent);\n      }\n      try {\n        signInputWithUnspent<TNumber>(txBuilder, inputIndex, unspent, walletSigner);\n        debug('Successfully signed input %d of %d', inputIndex + 1, unspents.length);\n      } catch (e) {\n        return new InputSigningError<TNumber>(inputIndex, null, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  const signedTransaction = isLastSignature ? txBuilder.build() : txBuilder.buildIncomplete();\n\n  const verifyErrors: InputSigningError<TNumber>[] = signedTransaction.ins\n    .map((input, inputIndex) => {\n      const unspent = unspents[inputIndex] as Unspent<TNumber>;\n      if (replayProtectionAddresses.includes(unspent.address)) {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          unspents.length\n        );\n        return;\n      }\n      if (!isWalletUnspent<TNumber>(unspent)) {\n        return InputSigningError.expectedWalletUnspent<TNumber>(inputIndex, null, unspent);\n      }\n      try {\n        const publicKey = walletSigner.deriveForChainAndIndex(unspent.chain, unspent.index).signer.publicKey;\n        if (\n          !utxolib.bitgo.verifySignatureWithPublicKey<TNumber>(signedTransaction, inputIndex, prevOutputs, publicKey)\n        ) {\n          return new InputSigningError(inputIndex, null, unspent, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<TNumber>(inputIndex, null, unspent, e);\n      }\n    })\n    .filter((e): e is InputSigningError<TNumber> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  return signedTransaction;\n}\n\nexport function signLegacyTransaction<TNumber extends number | bigint>(\n  tx: utxolib.bitgo.UtxoTransaction<TNumber>,\n  signerKeychain: BIP32Interface | undefined,\n  params: {\n    isLastSignature: boolean;\n    signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;\n    txInfo: { unspents?: utxolib.bitgo.Unspent<TNumber>[] } | undefined;\n    pubs: string[] | undefined;\n    cosignerPub: string | undefined;\n  }\n): utxolib.bitgo.UtxoTransaction<TNumber> {\n  switch (params.signingStep) {\n    case 'signerNonce':\n    case 'cosignerNonce':\n      /**\n       * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).\n       * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.\n       */\n      return tx;\n  }\n\n  if (tx.ins.length !== params.txInfo?.unspents?.length) {\n    throw new Error('length of unspents array should equal to the number of transaction inputs');\n  }\n\n  if (!params.pubs || !isTriple(params.pubs)) {\n    throw new Error(`must provide xpub array`);\n  }\n\n  const keychains = params.pubs.map((pub) => bip32.fromBase58(pub)) as Triple<BIP32Interface>;\n  const cosignerPub = params.cosignerPub ?? params.pubs[2];\n  const cosignerKeychain = bip32.fromBase58(cosignerPub);\n\n  assert(signerKeychain);\n  const walletSigner = new bitgo.WalletUnspentSigner<RootWalletKeys>(keychains, signerKeychain, cosignerKeychain);\n  return signAndVerifyWalletTransaction(tx, params.txInfo.unspents, walletSigner, {\n    isLastSignature: params.isLastSignature,\n  }) as utxolib.bitgo.UtxoTransaction<TNumber>;\n}\n"]}
@@ -0,0 +1,30 @@
1
+ import * as utxolib from '@bitgo-beta/utxo-lib';
2
+ import { BIP32Interface } from '@bitgo-beta/secp256k1';
3
+ import { Musig2Participant } from './musig2';
4
+ export type PsbtParsedScriptType = 'p2sh' | 'p2wsh' | 'p2shP2wsh' | 'p2shP2pk' | 'taprootKeyPathSpend' | 'taprootScriptPathSpend' | 'p2trLegacy' | 'p2trMusig2ScriptPath' | 'p2trMusig2KeyPath';
5
+ /**
6
+ * Sign all inputs of a psbt and verify signatures after signing.
7
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
8
+ * failed.
9
+ *
10
+ * If it is the last signature, finalize and extract the transaction from the psbt.
11
+ *
12
+ * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
13
+ * using TransactionBuilder
14
+ *
15
+ * @param psbt
16
+ * @param signerKeychain
17
+ * @param isLastSignature
18
+ */
19
+ export declare function signAndVerifyPsbt(psbt: utxolib.bitgo.UtxoPsbt, signerKeychain: utxolib.BIP32Interface, { isLastSignature,
20
+ /** deprecated */
21
+ allowNonSegwitSigningWithoutPrevTx, }: {
22
+ isLastSignature: boolean;
23
+ allowNonSegwitSigningWithoutPrevTx?: boolean;
24
+ }): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>;
25
+ export declare function signPsbtWithMusig2Participant(coin: Musig2Participant<utxolib.bitgo.UtxoPsbt>, tx: utxolib.bitgo.UtxoPsbt, signerKeychain: BIP32Interface | undefined, params: {
26
+ isLastSignature: boolean;
27
+ signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
28
+ walletId: string | undefined;
29
+ }): Promise<utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>>;
30
+ //# sourceMappingURL=signPsbt.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"signPsbt.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/signPsbt.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAChD,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAKvD,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAI7C,MAAM,MAAM,oBAAoB,GAC5B,MAAM,GACN,OAAO,GACP,WAAW,GACX,UAAU,GACV,qBAAqB,GACrB,wBAAwB,GAExB,YAAY,GACZ,sBAAsB,GACtB,mBAAmB,CAAC;AAExB;;;;;;;;;;;;;GAaG;AACH,wBAAgB,iBAAiB,CAC/B,IAAI,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC5B,cAAc,EAAE,OAAO,CAAC,cAAc,EACtC,EACE,eAAe;AACf,iBAAiB;AACjB,kCAAkC,GACnC,EAAE;IAAE,eAAe,EAAE,OAAO,CAAC;IAAC,kCAAkC,CAAC,EAAE,OAAO,CAAA;CAAE,GAC5E,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,CA6DhE;AAYD,wBAAsB,6BAA6B,CACjD,IAAI,EAAE,iBAAiB,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,CAAC,EAC/C,EAAE,EAAE,OAAO,CAAC,KAAK,CAAC,QAAQ,EAC1B,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,MAAM,EAAE;IACN,eAAe,EAAE,OAAO,CAAC;IACzB,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;CAC9B,GACA,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,eAAe,CAAC,MAAM,CAAC,CAAC,CAsDzE"}
@@ -0,0 +1,174 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
37
+ };
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ exports.signAndVerifyPsbt = signAndVerifyPsbt;
40
+ exports.signPsbtWithMusig2Participant = signPsbtWithMusig2Participant;
41
+ const assert_1 = __importDefault(require("assert"));
42
+ const utxolib = __importStar(require("@bitgo-beta/utxo-lib"));
43
+ const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
44
+ const debug_1 = __importDefault(require("debug"));
45
+ const SigningError_1 = require("./SigningError");
46
+ const debug = (0, debug_1.default)('bitgo:v2:utxo');
47
+ /**
48
+ * Sign all inputs of a psbt and verify signatures after signing.
49
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them
50
+ * failed.
51
+ *
52
+ * If it is the last signature, finalize and extract the transaction from the psbt.
53
+ *
54
+ * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of
55
+ * using TransactionBuilder
56
+ *
57
+ * @param psbt
58
+ * @param signerKeychain
59
+ * @param isLastSignature
60
+ */
61
+ function signAndVerifyPsbt(psbt, signerKeychain, { isLastSignature,
62
+ /** deprecated */
63
+ allowNonSegwitSigningWithoutPrevTx, }) {
64
+ const txInputs = psbt.txInputs;
65
+ const outputIds = [];
66
+ const scriptTypes = [];
67
+ const signErrors = psbt.data.inputs
68
+ .map((input, inputIndex) => {
69
+ const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));
70
+ outputIds.push(outputId);
71
+ const { scriptType } = utxolib.bitgo.parsePsbtInput(input);
72
+ scriptTypes.push(scriptType);
73
+ if (scriptType === 'p2shP2pk') {
74
+ debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);
75
+ return;
76
+ }
77
+ try {
78
+ psbt.signInputHD(inputIndex, signerKeychain);
79
+ debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);
80
+ }
81
+ catch (e) {
82
+ return new SigningError_1.InputSigningError(inputIndex, scriptType, { id: outputId }, e);
83
+ }
84
+ })
85
+ .filter((e) => e !== undefined);
86
+ const verifyErrors = psbt.data.inputs
87
+ .map((input, inputIndex) => {
88
+ const scriptType = scriptTypes[inputIndex];
89
+ if (scriptType === 'p2shP2pk') {
90
+ debug('Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)', inputIndex + 1, psbt.data.inputs.length);
91
+ return;
92
+ }
93
+ const outputId = outputIds[inputIndex];
94
+ try {
95
+ if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {
96
+ return new SigningError_1.InputSigningError(inputIndex, scriptType, { id: outputId }, new Error(`invalid signature`));
97
+ }
98
+ }
99
+ catch (e) {
100
+ debug('Invalid signature');
101
+ return new SigningError_1.InputSigningError(inputIndex, scriptType, { id: outputId }, e);
102
+ }
103
+ })
104
+ .filter((e) => e !== undefined);
105
+ if (signErrors.length || verifyErrors.length) {
106
+ throw new SigningError_1.TransactionSigningError(signErrors, verifyErrors);
107
+ }
108
+ if (isLastSignature) {
109
+ psbt.finalizeAllInputs();
110
+ return psbt.extractTransaction();
111
+ }
112
+ return psbt;
113
+ }
114
+ /**
115
+ * Key Value: Unsigned tx id => PSBT
116
+ * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.
117
+ * Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.
118
+ * For more info, check SignTransactionOptions.signingStep
119
+ *
120
+ * TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.
121
+ */
122
+ const PSBT_CACHE = new Map();
123
+ async function signPsbtWithMusig2Participant(coin, tx, signerKeychain, params) {
124
+ if (utxo_lib_1.bitgo.isTransactionWithKeyPathSpendInput(tx)) {
125
+ // We can only be the first signature on a transaction with taproot key path spend inputs because
126
+ // we require the secret nonce in the cache of the first signer, which is impossible to retrieve if
127
+ // deserialized from a hex.
128
+ if (params.isLastSignature) {
129
+ throw new Error('Cannot be last signature on a transaction with key path spend inputs');
130
+ }
131
+ switch (params.signingStep) {
132
+ case 'signerNonce':
133
+ (0, assert_1.default)(signerKeychain);
134
+ tx.setAllInputsMusig2NonceHD(signerKeychain);
135
+ PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);
136
+ return tx;
137
+ case 'cosignerNonce':
138
+ (0, assert_1.default)(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
139
+ return await coin.getMusig2Nonces(tx, params.walletId);
140
+ case 'signerSignature':
141
+ const txId = tx.getUnsignedTx().getId();
142
+ const psbt = PSBT_CACHE.get(txId);
143
+ (0, assert_1.default)(psbt, `Psbt is missing from txCache (cache size ${PSBT_CACHE.size}).
144
+ This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`);
145
+ PSBT_CACHE.delete(txId);
146
+ tx = psbt.combine(tx);
147
+ break;
148
+ default:
149
+ // this instance is not an external signer
150
+ (0, assert_1.default)(params.walletId, 'walletId is required for MuSig2 bitgo nonce');
151
+ (0, assert_1.default)(signerKeychain);
152
+ tx.setAllInputsMusig2NonceHD(signerKeychain);
153
+ const response = await coin.getMusig2Nonces(tx, params.walletId);
154
+ tx = tx.combine(response);
155
+ break;
156
+ }
157
+ }
158
+ else {
159
+ switch (params.signingStep) {
160
+ case 'signerNonce':
161
+ case 'cosignerNonce':
162
+ /**
163
+ * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).
164
+ * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.
165
+ */
166
+ return tx;
167
+ }
168
+ }
169
+ (0, assert_1.default)(signerKeychain);
170
+ return signAndVerifyPsbt(tx, signerKeychain, {
171
+ isLastSignature: params.isLastSignature,
172
+ });
173
+ }
174
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signPsbt.js","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/signPsbt.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\n\nimport * as utxolib from '@bitgo-beta/utxo-lib';\nimport { BIP32Interface } from '@bitgo-beta/secp256k1';\nimport { bitgo } from '@bitgo-beta/utxo-lib';\nimport debugLib from 'debug';\n\nimport { InputSigningError, TransactionSigningError } from './SigningError';\nimport { Musig2Participant } from './musig2';\n\nconst debug = debugLib('bitgo:v2:utxo');\n\nexport type PsbtParsedScriptType =\n  | 'p2sh'\n  | 'p2wsh'\n  | 'p2shP2wsh'\n  | 'p2shP2pk'\n  | 'taprootKeyPathSpend'\n  | 'taprootScriptPathSpend'\n  // wasm-utxo types\n  | 'p2trLegacy'\n  | 'p2trMusig2ScriptPath'\n  | 'p2trMusig2KeyPath';\n\n/**\n * Sign all inputs of a psbt and verify signatures after signing.\n * Collects and logs signing errors and verification errors, throws error in the end if any of them\n * failed.\n *\n * If it is the last signature, finalize and extract the transaction from the psbt.\n *\n * This function mirrors signAndVerifyWalletTransaction, but is used for signing PSBTs instead of\n * using TransactionBuilder\n *\n * @param psbt\n * @param signerKeychain\n * @param isLastSignature\n */\nexport function signAndVerifyPsbt(\n  psbt: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: utxolib.BIP32Interface,\n  {\n    isLastSignature,\n    /** deprecated */\n    allowNonSegwitSigningWithoutPrevTx,\n  }: { isLastSignature: boolean; allowNonSegwitSigningWithoutPrevTx?: boolean }\n): utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint> {\n  const txInputs = psbt.txInputs;\n  const outputIds: string[] = [];\n  const scriptTypes: PsbtParsedScriptType[] = [];\n\n  const signErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex: number) => {\n      const outputId = utxolib.bitgo.formatOutputId(utxolib.bitgo.getOutputIdForInput(txInputs[inputIndex]));\n      outputIds.push(outputId);\n\n      const { scriptType } = utxolib.bitgo.parsePsbtInput(input);\n      scriptTypes.push(scriptType);\n\n      if (scriptType === 'p2shP2pk') {\n        debug('Skipping signature for input %d of %d (RP input?)', inputIndex + 1, psbt.data.inputs.length);\n        return;\n      }\n\n      try {\n        psbt.signInputHD(inputIndex, signerKeychain);\n        debug('Successfully signed input %d of %d', inputIndex + 1, psbt.data.inputs.length);\n      } catch (e) {\n        return new InputSigningError<bigint>(inputIndex, scriptType, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  const verifyErrors: InputSigningError<bigint>[] = psbt.data.inputs\n    .map((input, inputIndex) => {\n      const scriptType = scriptTypes[inputIndex];\n      if (scriptType === 'p2shP2pk') {\n        debug(\n          'Skipping input signature %d of %d (unspent from replay protection address which is platform signed only)',\n          inputIndex + 1,\n          psbt.data.inputs.length\n        );\n        return;\n      }\n\n      const outputId = outputIds[inputIndex];\n      try {\n        if (!psbt.validateSignaturesOfInputHD(inputIndex, signerKeychain)) {\n          return new InputSigningError(inputIndex, scriptType, { id: outputId }, new Error(`invalid signature`));\n        }\n      } catch (e) {\n        debug('Invalid signature');\n        return new InputSigningError<bigint>(inputIndex, scriptType, { id: outputId }, e);\n      }\n    })\n    .filter((e): e is InputSigningError<bigint> => e !== undefined);\n\n  if (signErrors.length || verifyErrors.length) {\n    throw new TransactionSigningError(signErrors, verifyErrors);\n  }\n\n  if (isLastSignature) {\n    psbt.finalizeAllInputs();\n    return psbt.extractTransaction();\n  }\n\n  return psbt;\n}\n\n/**\n * Key Value: Unsigned tx id => PSBT\n * It is used to cache PSBTs with taproot key path (MuSig2) inputs during external express signer is activated.\n * Reason: MuSig2 signer secure nonce is cached in the UtxoPsbt object. It will be required during the signing step.\n * For more info, check SignTransactionOptions.signingStep\n *\n * TODO BTC-276: This cache may need to be done with LRU like memory safe caching if memory issues comes up.\n */\nconst PSBT_CACHE = new Map<string, utxolib.bitgo.UtxoPsbt>();\n\nexport async function signPsbtWithMusig2Participant(\n  coin: Musig2Participant<utxolib.bitgo.UtxoPsbt>,\n  tx: utxolib.bitgo.UtxoPsbt,\n  signerKeychain: BIP32Interface | undefined,\n  params: {\n    isLastSignature: boolean;\n    signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;\n    walletId: string | undefined;\n  }\n): Promise<utxolib.bitgo.UtxoPsbt | utxolib.bitgo.UtxoTransaction<bigint>> {\n  if (bitgo.isTransactionWithKeyPathSpendInput(tx)) {\n    // We can only be the first signature on a transaction with taproot key path spend inputs because\n    // we require the secret nonce in the cache of the first signer, which is impossible to retrieve if\n    // deserialized from a hex.\n    if (params.isLastSignature) {\n      throw new Error('Cannot be last signature on a transaction with key path spend inputs');\n    }\n\n    switch (params.signingStep) {\n      case 'signerNonce':\n        assert(signerKeychain);\n        tx.setAllInputsMusig2NonceHD(signerKeychain);\n        PSBT_CACHE.set(tx.getUnsignedTx().getId(), tx);\n        return tx;\n      case 'cosignerNonce':\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        return await coin.getMusig2Nonces(tx, params.walletId);\n      case 'signerSignature':\n        const txId = tx.getUnsignedTx().getId();\n        const psbt = PSBT_CACHE.get(txId);\n        assert(\n          psbt,\n          `Psbt is missing from txCache (cache size ${PSBT_CACHE.size}).\n            This may be due to the request being routed to a different BitGo-Express instance that for signing step 'signerNonce'.`\n        );\n        PSBT_CACHE.delete(txId);\n        tx = psbt.combine(tx);\n        break;\n      default:\n        // this instance is not an external signer\n        assert(params.walletId, 'walletId is required for MuSig2 bitgo nonce');\n        assert(signerKeychain);\n        tx.setAllInputsMusig2NonceHD(signerKeychain);\n        const response = await coin.getMusig2Nonces(tx, params.walletId);\n        tx = tx.combine(response);\n        break;\n    }\n  } else {\n    switch (params.signingStep) {\n      case 'signerNonce':\n      case 'cosignerNonce':\n        /**\n         * In certain cases, the caller of this method may not know whether the txHex contains a psbt with taproot key path spend input(s).\n         * Instead of throwing error, no-op and return the txHex. So that the caller can call this method in the same sequence.\n         */\n        return tx;\n    }\n  }\n\n  assert(signerKeychain);\n  return signAndVerifyPsbt(tx, signerKeychain, {\n    isLastSignature: params.isLastSignature,\n  });\n}\n"]}
@@ -0,0 +1,22 @@
1
+ import { BIP32Interface } from '@bitgo-beta/utxo-lib';
2
+ import { ECPair, fixedScriptWallet } from '@bitgo/wasm-utxo';
3
+ import { Musig2Participant } from './musig2';
4
+ export type ReplayProtectionKeys = {
5
+ publicKeys: (Uint8Array | ECPair)[];
6
+ };
7
+ /**
8
+ * Sign all inputs of a PSBT and verify signatures after signing.
9
+ * Collects and logs signing errors and verification errors, throws error in the end if any of them failed.
10
+ *
11
+ * If it is the last signature, finalize and extract the transaction from the psbt.
12
+ */
13
+ export declare function signAndVerifyPsbtWasm(tx: fixedScriptWallet.BitGoPsbt, signerKeychain: BIP32Interface, rootWalletKeys: fixedScriptWallet.RootWalletKeys, replayProtection: ReplayProtectionKeys, { isLastSignature }: {
14
+ isLastSignature: boolean;
15
+ }): fixedScriptWallet.BitGoPsbt | Buffer;
16
+ export declare function signPsbtWithMusig2ParticipantWasm(coin: Musig2Participant<fixedScriptWallet.BitGoPsbt>, tx: fixedScriptWallet.BitGoPsbt, signerKeychain: BIP32Interface | undefined, rootWalletKeys: fixedScriptWallet.RootWalletKeys, params: {
17
+ replayProtection: ReplayProtectionKeys;
18
+ isLastSignature: boolean;
19
+ signingStep: 'signerNonce' | 'cosignerNonce' | 'signerSignature' | undefined;
20
+ walletId: string | undefined;
21
+ }): Promise<fixedScriptWallet.BitGoPsbt | Buffer>;
22
+ //# sourceMappingURL=signPsbtWasm.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"signPsbtWasm.d.ts","sourceRoot":"","sources":["../../../../../src/transaction/fixedScript/signPsbtWasm.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,cAAc,EAAE,MAAM,sBAAsB,CAAC;AACtD,OAAO,EAAS,MAAM,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAC;AAGpE,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;AAE7C,MAAM,MAAM,oBAAoB,GAAG;IACjC,UAAU,EAAE,CAAC,UAAU,GAAG,MAAM,CAAC,EAAE,CAAC;CACrC,CAAC;AAmBF;;;;;GAKG;AACH,wBAAgB,qBAAqB,CACnC,EAAE,EAAE,iBAAiB,CAAC,SAAS,EAC/B,cAAc,EAAE,cAAc,EAC9B,cAAc,EAAE,iBAAiB,CAAC,cAAc,EAChD,gBAAgB,EAAE,oBAAoB,EACtC,EAAE,eAAe,EAAE,EAAE;IAAE,eAAe,EAAE,OAAO,CAAA;CAAE,GAChD,iBAAiB,CAAC,SAAS,GAAG,MAAM,CAkDtC;AAOD,wBAAsB,iCAAiC,CACrD,IAAI,EAAE,iBAAiB,CAAC,iBAAiB,CAAC,SAAS,CAAC,EACpD,EAAE,EAAE,iBAAiB,CAAC,SAAS,EAC/B,cAAc,EAAE,cAAc,GAAG,SAAS,EAC1C,cAAc,EAAE,iBAAiB,CAAC,cAAc,EAChD,MAAM,EAAE;IACN,gBAAgB,EAAE,oBAAoB,CAAC;IACvC,eAAe,EAAE,OAAO,CAAC;IACzB,WAAW,EAAE,aAAa,GAAG,eAAe,GAAG,iBAAiB,GAAG,SAAS,CAAC;IAC7E,QAAQ,EAAE,MAAM,GAAG,SAAS,CAAC;CAC9B,GACA,OAAO,CAAC,iBAAiB,CAAC,SAAS,GAAG,MAAM,CAAC,CA0D/C"}