@metamask/snaps-execution-environments 1.0.1 → 2.0.0

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 (234) hide show
  1. package/CHANGELOG.md +37 -273
  2. package/dist/browserify/iframe/bundle.js +6 -4
  3. package/dist/browserify/iframe/index.html +2 -0
  4. package/dist/browserify/node-process/bundle.js +8 -2
  5. package/dist/browserify/node-thread/bundle.js +8 -2
  6. package/dist/browserify/offscreen/bundle.js +6 -9
  7. package/dist/browserify/offscreen/index.html +2 -0
  8. package/dist/browserify/worker-executor/bundle.js +12157 -0
  9. package/dist/browserify/worker-pool/bundle.js +7 -0
  10. package/dist/browserify/worker-pool/index.html +12152 -0
  11. package/dist/cjs/common/BaseSnapExecutor.js +423 -0
  12. package/dist/cjs/common/BaseSnapExecutor.js.map +1 -0
  13. package/dist/cjs/common/commands.js +81 -0
  14. package/dist/cjs/common/commands.js.map +1 -0
  15. package/dist/cjs/common/endowments/commonEndowmentFactory.js +147 -0
  16. package/dist/cjs/common/endowments/commonEndowmentFactory.js.map +1 -0
  17. package/dist/cjs/common/endowments/console.js +134 -0
  18. package/dist/cjs/common/endowments/console.js.map +1 -0
  19. package/dist/cjs/common/endowments/crypto.js +44 -0
  20. package/dist/cjs/common/endowments/crypto.js.map +1 -0
  21. package/dist/cjs/common/endowments/date.js +53 -0
  22. package/dist/cjs/common/endowments/date.js.map +1 -0
  23. package/dist/cjs/common/endowments/index.js +91 -0
  24. package/dist/cjs/common/endowments/index.js.map +1 -0
  25. package/dist/cjs/common/endowments/interval.js +61 -0
  26. package/dist/cjs/common/endowments/interval.js.map +1 -0
  27. package/dist/cjs/common/endowments/math.js +67 -0
  28. package/dist/cjs/common/endowments/math.js.map +1 -0
  29. package/dist/cjs/common/endowments/network.js +216 -0
  30. package/dist/cjs/common/endowments/network.js.map +1 -0
  31. package/dist/cjs/common/endowments/textDecoder.js +28 -0
  32. package/dist/cjs/common/endowments/textDecoder.js.map +1 -0
  33. package/dist/cjs/common/endowments/textEncoder.js +28 -0
  34. package/dist/cjs/common/endowments/textEncoder.js.map +1 -0
  35. package/dist/cjs/common/endowments/timeout.js +63 -0
  36. package/dist/cjs/common/endowments/timeout.js.map +1 -0
  37. package/dist/cjs/common/globalEvents.js +39 -0
  38. package/dist/cjs/common/globalEvents.js.map +1 -0
  39. package/dist/cjs/common/globalObject.js +53 -0
  40. package/dist/{common → cjs/common}/globalObject.js.map +1 -1
  41. package/dist/cjs/common/lockdown/lockdown-events.js +74 -0
  42. package/dist/cjs/common/lockdown/lockdown-events.js.map +1 -0
  43. package/dist/cjs/common/lockdown/lockdown-more.js +73 -0
  44. package/dist/cjs/common/lockdown/lockdown-more.js.map +1 -0
  45. package/dist/cjs/common/lockdown/lockdown.js +30 -0
  46. package/dist/cjs/common/lockdown/lockdown.js.map +1 -0
  47. package/dist/cjs/common/sortParams.js +26 -0
  48. package/dist/cjs/common/sortParams.js.map +1 -0
  49. package/dist/cjs/common/utils.js +132 -0
  50. package/dist/cjs/common/utils.js.map +1 -0
  51. package/dist/cjs/common/validation.js +118 -0
  52. package/dist/cjs/common/validation.js.map +1 -0
  53. package/dist/cjs/iframe/IFrameSnapExecutor.js +50 -0
  54. package/dist/cjs/iframe/IFrameSnapExecutor.js.map +1 -0
  55. package/dist/cjs/iframe/index.js +13 -0
  56. package/dist/cjs/iframe/index.js.map +1 -0
  57. package/dist/cjs/logging.js +15 -0
  58. package/dist/cjs/logging.js.map +1 -0
  59. package/dist/cjs/node-process/ChildProcessSnapExecutor.js +39 -0
  60. package/dist/cjs/node-process/ChildProcessSnapExecutor.js.map +1 -0
  61. package/dist/cjs/node-process/index.js +11 -0
  62. package/dist/cjs/node-process/index.js.map +1 -0
  63. package/dist/cjs/node-thread/ThreadSnapExecutor.js +39 -0
  64. package/dist/cjs/node-thread/ThreadSnapExecutor.js.map +1 -0
  65. package/dist/cjs/node-thread/index.js +11 -0
  66. package/dist/cjs/node-thread/index.js.map +1 -0
  67. package/dist/cjs/offscreen/OffscreenSnapExecutor.js +173 -0
  68. package/dist/cjs/offscreen/OffscreenSnapExecutor.js.map +1 -0
  69. package/dist/cjs/offscreen/index.js +19 -0
  70. package/dist/cjs/offscreen/index.js.map +1 -0
  71. package/dist/cjs/webworker/executor/WebWorkerSnapExecutor.js +45 -0
  72. package/dist/cjs/webworker/executor/WebWorkerSnapExecutor.js.map +1 -0
  73. package/dist/cjs/webworker/executor/index.js +13 -0
  74. package/dist/cjs/webworker/executor/index.js.map +1 -0
  75. package/dist/cjs/webworker/pool/WebWorkerPool.js +248 -0
  76. package/dist/cjs/webworker/pool/WebWorkerPool.js.map +1 -0
  77. package/dist/cjs/webworker/pool/index.js +13 -0
  78. package/dist/cjs/webworker/pool/index.js.map +1 -0
  79. package/dist/esm/common/BaseSnapExecutor.js +408 -0
  80. package/dist/esm/common/BaseSnapExecutor.js.map +1 -0
  81. package/dist/esm/common/commands.js +80 -0
  82. package/dist/esm/common/commands.js.map +1 -0
  83. package/dist/esm/common/endowments/commonEndowmentFactory.js +132 -0
  84. package/dist/esm/common/endowments/commonEndowmentFactory.js.map +1 -0
  85. package/dist/esm/common/endowments/console.js +116 -0
  86. package/dist/esm/common/endowments/console.js.map +1 -0
  87. package/dist/esm/common/endowments/crypto.js +26 -0
  88. package/dist/esm/common/endowments/crypto.js.map +1 -0
  89. package/dist/{common → esm/common}/endowments/date.js +20 -16
  90. package/dist/esm/common/endowments/date.js.map +1 -0
  91. package/dist/{common → esm/common}/endowments/index.js +34 -36
  92. package/dist/esm/common/endowments/index.js.map +1 -0
  93. package/dist/{common → esm/common}/endowments/interval.js +13 -12
  94. package/dist/esm/common/endowments/interval.js.map +1 -0
  95. package/dist/{common → esm/common}/endowments/math.js +21 -15
  96. package/dist/esm/common/endowments/math.js.map +1 -0
  97. package/dist/esm/common/endowments/network.js +206 -0
  98. package/dist/esm/common/endowments/network.js.map +1 -0
  99. package/dist/esm/common/endowments/textDecoder.js +18 -0
  100. package/dist/esm/common/endowments/textDecoder.js.map +1 -0
  101. package/dist/esm/common/endowments/textEncoder.js +18 -0
  102. package/dist/esm/common/endowments/textEncoder.js.map +1 -0
  103. package/dist/{common → esm/common}/endowments/timeout.js +14 -13
  104. package/dist/esm/common/endowments/timeout.js.map +1 -0
  105. package/dist/esm/common/globalEvents.js +35 -0
  106. package/dist/esm/common/globalEvents.js.map +1 -0
  107. package/dist/esm/common/globalObject.js +36 -0
  108. package/dist/esm/common/globalObject.js.map +1 -0
  109. package/dist/esm/common/lockdown/lockdown-events.js +66 -0
  110. package/dist/esm/common/lockdown/lockdown-events.js.map +1 -0
  111. package/dist/{common → esm/common}/lockdown/lockdown-more.js +15 -22
  112. package/dist/esm/common/lockdown/lockdown-more.js.map +1 -0
  113. package/dist/{common → esm/common}/lockdown/lockdown.js +6 -11
  114. package/dist/esm/common/lockdown/lockdown.js.map +1 -0
  115. package/dist/{common → esm/common}/sortParams.js +7 -13
  116. package/dist/esm/common/sortParams.js.map +1 -0
  117. package/dist/{common → esm/common}/utils.js +43 -56
  118. package/dist/esm/common/utils.js.map +1 -0
  119. package/dist/esm/common/validation.js +94 -0
  120. package/dist/esm/common/validation.js.map +1 -0
  121. package/dist/esm/iframe/IFrameSnapExecutor.js +35 -0
  122. package/dist/esm/iframe/IFrameSnapExecutor.js.map +1 -0
  123. package/dist/esm/iframe/index.js +9 -0
  124. package/dist/esm/iframe/index.js.map +1 -0
  125. package/dist/esm/logging.js +10 -0
  126. package/dist/esm/logging.js.map +1 -0
  127. package/dist/esm/node-process/ChildProcessSnapExecutor.js +24 -0
  128. package/dist/esm/node-process/ChildProcessSnapExecutor.js.map +1 -0
  129. package/dist/esm/node-process/index.js +7 -0
  130. package/dist/esm/node-process/index.js.map +1 -0
  131. package/dist/esm/node-thread/ThreadSnapExecutor.js +24 -0
  132. package/dist/esm/node-thread/ThreadSnapExecutor.js.map +1 -0
  133. package/dist/esm/node-thread/index.js +7 -0
  134. package/dist/esm/node-thread/index.js.map +1 -0
  135. package/dist/esm/offscreen/OffscreenSnapExecutor.js +176 -0
  136. package/dist/esm/offscreen/OffscreenSnapExecutor.js.map +1 -0
  137. package/dist/esm/offscreen/index.js +15 -0
  138. package/dist/esm/offscreen/index.js.map +1 -0
  139. package/dist/esm/webworker/executor/WebWorkerSnapExecutor.js +30 -0
  140. package/dist/esm/webworker/executor/WebWorkerSnapExecutor.js.map +1 -0
  141. package/dist/esm/webworker/executor/index.js +9 -0
  142. package/dist/esm/webworker/executor/index.js.map +1 -0
  143. package/dist/esm/webworker/pool/WebWorkerPool.js +245 -0
  144. package/dist/esm/webworker/pool/WebWorkerPool.js.map +1 -0
  145. package/dist/esm/webworker/pool/index.js +9 -0
  146. package/dist/esm/webworker/pool/index.js.map +1 -0
  147. package/dist/{common → types/common}/BaseSnapExecutor.d.ts +7 -7
  148. package/dist/{common → types/common}/commands.d.ts +2 -2
  149. package/dist/{common → types/common}/endowments/commonEndowmentFactory.d.ts +5 -1
  150. package/dist/types/common/endowments/console.d.ts +45 -0
  151. package/dist/{common → types/common}/endowments/crypto.d.ts +4 -0
  152. package/dist/{common → types/common}/endowments/index.d.ts +5 -3
  153. package/dist/types/common/endowments/network.d.ts +22 -0
  154. package/dist/{common → types/common}/globalEvents.d.ts +1 -0
  155. package/dist/{common → types/common}/sortParams.d.ts +1 -1
  156. package/dist/{common → types/common}/utils.d.ts +5 -4
  157. package/dist/{common → types/common}/validation.d.ts +26 -10
  158. package/dist/{iframe → types/iframe}/IFrameSnapExecutor.d.ts +1 -1
  159. package/dist/{offscreen → types/offscreen}/OffscreenSnapExecutor.d.ts +2 -1
  160. package/dist/types/webworker/executor/WebWorkerSnapExecutor.d.ts +13 -0
  161. package/dist/types/webworker/executor/index.d.ts +1 -0
  162. package/dist/types/webworker/pool/WebWorkerPool.d.ts +23 -0
  163. package/dist/types/webworker/pool/index.d.ts +1 -0
  164. package/package.json +52 -41
  165. package/dist/common/BaseSnapExecutor.js +0 -370
  166. package/dist/common/BaseSnapExecutor.js.map +0 -1
  167. package/dist/common/commands.js +0 -65
  168. package/dist/common/commands.js.map +0 -1
  169. package/dist/common/endowments/commonEndowmentFactory.js +0 -75
  170. package/dist/common/endowments/commonEndowmentFactory.js.map +0 -1
  171. package/dist/common/endowments/crypto.js +0 -28
  172. package/dist/common/endowments/crypto.js.map +0 -1
  173. package/dist/common/endowments/date.js.map +0 -1
  174. package/dist/common/endowments/index.js.map +0 -1
  175. package/dist/common/endowments/interval.js.map +0 -1
  176. package/dist/common/endowments/math.js.map +0 -1
  177. package/dist/common/endowments/network.d.ts +0 -8
  178. package/dist/common/endowments/network.js +0 -170
  179. package/dist/common/endowments/network.js.map +0 -1
  180. package/dist/common/endowments/textDecoder.js +0 -18
  181. package/dist/common/endowments/textDecoder.js.map +0 -1
  182. package/dist/common/endowments/textEncoder.js +0 -18
  183. package/dist/common/endowments/textEncoder.js.map +0 -1
  184. package/dist/common/endowments/timeout.js.map +0 -1
  185. package/dist/common/globalEvents.js +0 -47
  186. package/dist/common/globalEvents.js.map +0 -1
  187. package/dist/common/globalObject.js +0 -50
  188. package/dist/common/keyring.d.ts +0 -12
  189. package/dist/common/keyring.js +0 -42
  190. package/dist/common/keyring.js.map +0 -1
  191. package/dist/common/lockdown/lockdown-events.js +0 -60
  192. package/dist/common/lockdown/lockdown-events.js.map +0 -1
  193. package/dist/common/lockdown/lockdown-more.js.map +0 -1
  194. package/dist/common/lockdown/lockdown.js.map +0 -1
  195. package/dist/common/sortParams.js.map +0 -1
  196. package/dist/common/utils.js.map +0 -1
  197. package/dist/common/validation.js +0 -109
  198. package/dist/common/validation.js.map +0 -1
  199. package/dist/iframe/IFrameSnapExecutor.js +0 -42
  200. package/dist/iframe/IFrameSnapExecutor.js.map +0 -1
  201. package/dist/iframe/index.js +0 -10
  202. package/dist/iframe/index.js.map +0 -1
  203. package/dist/logging.js +0 -13
  204. package/dist/logging.js.map +0 -1
  205. package/dist/node-process/ChildProcessSnapExecutor.js +0 -30
  206. package/dist/node-process/ChildProcessSnapExecutor.js.map +0 -1
  207. package/dist/node-process/index.js +0 -8
  208. package/dist/node-process/index.js.map +0 -1
  209. package/dist/node-thread/ThreadSnapExecutor.js +0 -30
  210. package/dist/node-thread/ThreadSnapExecutor.js.map +0 -1
  211. package/dist/node-thread/index.js +0 -8
  212. package/dist/node-thread/index.js.map +0 -1
  213. package/dist/offscreen/OffscreenSnapExecutor.js +0 -104
  214. package/dist/offscreen/OffscreenSnapExecutor.js.map +0 -1
  215. package/dist/offscreen/index.js +0 -16
  216. package/dist/offscreen/index.js.map +0 -1
  217. package/dist/openrpc.json +0 -210
  218. /package/dist/{common → types/common}/endowments/date.d.ts +0 -0
  219. /package/dist/{common → types/common}/endowments/interval.d.ts +0 -0
  220. /package/dist/{common → types/common}/endowments/math.d.ts +0 -0
  221. /package/dist/{common → types/common}/endowments/textDecoder.d.ts +0 -0
  222. /package/dist/{common → types/common}/endowments/textEncoder.d.ts +0 -0
  223. /package/dist/{common → types/common}/endowments/timeout.d.ts +0 -0
  224. /package/dist/{common → types/common}/globalObject.d.ts +0 -0
  225. /package/dist/{common → types/common}/lockdown/lockdown-events.d.ts +0 -0
  226. /package/dist/{common → types/common}/lockdown/lockdown-more.d.ts +0 -0
  227. /package/dist/{common → types/common}/lockdown/lockdown.d.ts +0 -0
  228. /package/dist/{iframe → types/iframe}/index.d.ts +0 -0
  229. /package/dist/{logging.d.ts → types/logging.d.ts} +0 -0
  230. /package/dist/{node-process → types/node-process}/ChildProcessSnapExecutor.d.ts +0 -0
  231. /package/dist/{node-process → types/node-process}/index.d.ts +0 -0
  232. /package/dist/{node-thread → types/node-thread}/ThreadSnapExecutor.d.ts +0 -0
  233. /package/dist/{node-thread → types/node-thread}/index.d.ts +0 -0
  234. /package/dist/{offscreen → types/offscreen}/index.d.ts +0 -0
@@ -0,0 +1,39 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "ThreadSnapExecutor", {
6
+ enumerable: true,
7
+ get: function() {
8
+ return ThreadSnapExecutor;
9
+ }
10
+ });
11
+ const _objectmultiplex = /*#__PURE__*/ _interop_require_default(require("@metamask/object-multiplex"));
12
+ const _postmessagestream = require("@metamask/post-message-stream");
13
+ const _snapsutils = require("@metamask/snaps-utils");
14
+ const _stream = require("stream");
15
+ const _BaseSnapExecutor = require("../common/BaseSnapExecutor");
16
+ const _logging = require("../logging");
17
+ function _interop_require_default(obj) {
18
+ return obj && obj.__esModule ? obj : {
19
+ default: obj
20
+ };
21
+ }
22
+ class ThreadSnapExecutor extends _BaseSnapExecutor.BaseSnapExecutor {
23
+ static initialize() {
24
+ (0, _logging.log)('Worker: Connecting to parent.');
25
+ const parentStream = new _postmessagestream.ThreadMessageStream();
26
+ const mux = new _objectmultiplex.default();
27
+ (0, _stream.pipeline)(parentStream, mux, parentStream, (error)=>{
28
+ if (error) {
29
+ (0, _snapsutils.logError)(`Parent stream failure, closing worker.`, error);
30
+ }
31
+ self.close();
32
+ });
33
+ const commandStream = mux.createStream(_snapsutils.SNAP_STREAM_NAMES.COMMAND);
34
+ const rpcStream = mux.createStream(_snapsutils.SNAP_STREAM_NAMES.JSON_RPC);
35
+ return new ThreadSnapExecutor(commandStream, rpcStream);
36
+ }
37
+ }
38
+
39
+ //# sourceMappingURL=ThreadSnapExecutor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/node-thread/ThreadSnapExecutor.ts"],"sourcesContent":["import ObjectMultiplex from '@metamask/object-multiplex';\nimport { ThreadMessageStream } from '@metamask/post-message-stream';\nimport { logError, SNAP_STREAM_NAMES } from '@metamask/snaps-utils';\nimport { pipeline } from 'stream';\n\nimport { BaseSnapExecutor } from '../common/BaseSnapExecutor';\nimport { log } from '../logging';\n\nexport class ThreadSnapExecutor extends BaseSnapExecutor {\n static initialize() {\n log('Worker: Connecting to parent.');\n\n const parentStream = new ThreadMessageStream();\n const mux = new ObjectMultiplex();\n pipeline(parentStream, mux as any, parentStream, (error) => {\n if (error) {\n logError(`Parent stream failure, closing worker.`, error);\n }\n self.close();\n });\n\n const commandStream = mux.createStream(SNAP_STREAM_NAMES.COMMAND);\n const rpcStream = mux.createStream(SNAP_STREAM_NAMES.JSON_RPC) as any;\n return new ThreadSnapExecutor(commandStream, rpcStream);\n }\n}\n"],"names":["ThreadSnapExecutor","BaseSnapExecutor","initialize","log","parentStream","ThreadMessageStream","mux","ObjectMultiplex","pipeline","error","logError","self","close","commandStream","createStream","SNAP_STREAM_NAMES","COMMAND","rpcStream","JSON_RPC"],"mappings":";;;;+BAQaA;;;eAAAA;;;wEARe;mCACQ;4BACQ;wBACnB;kCAEQ;yBACb;;;;;;AAEb,MAAMA,2BAA2BC,kCAAgB;IACtD,OAAOC,aAAa;QAClBC,IAAAA,YAAG,EAAC;QAEJ,MAAMC,eAAe,IAAIC,sCAAmB;QAC5C,MAAMC,MAAM,IAAIC,wBAAe;QAC/BC,IAAAA,gBAAQ,EAACJ,cAAcE,KAAYF,cAAc,CAACK;YAChD,IAAIA,OAAO;gBACTC,IAAAA,oBAAQ,EAAC,CAAC,sCAAsC,CAAC,EAAED;YACrD;YACAE,KAAKC,KAAK;QACZ;QAEA,MAAMC,gBAAgBP,IAAIQ,YAAY,CAACC,6BAAiB,CAACC,OAAO;QAChE,MAAMC,YAAYX,IAAIQ,YAAY,CAACC,6BAAiB,CAACG,QAAQ;QAC7D,OAAO,IAAIlB,mBAAmBa,eAAeI;IAC/C;AACF"}
@@ -0,0 +1,11 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ const _lockdownmore = require("../common/lockdown/lockdown-more");
6
+ const _ThreadSnapExecutor = require("./ThreadSnapExecutor");
7
+ // Lockdown is already applied in LavaMoat
8
+ (0, _lockdownmore.executeLockdownMore)();
9
+ _ThreadSnapExecutor.ThreadSnapExecutor.initialize();
10
+
11
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/node-thread/index.ts"],"sourcesContent":["import { executeLockdownMore } from '../common/lockdown/lockdown-more';\nimport { ThreadSnapExecutor } from './ThreadSnapExecutor';\n\n// Lockdown is already applied in LavaMoat\nexecuteLockdownMore();\n\nThreadSnapExecutor.initialize();\n"],"names":["executeLockdownMore","ThreadSnapExecutor","initialize"],"mappings":";;;;8BAAoC;oCACD;AAEnC,0CAA0C;AAC1CA,IAAAA,iCAAmB;AAEnBC,sCAAkB,CAACC,UAAU"}
@@ -0,0 +1,173 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "OffscreenSnapExecutor", {
6
+ enumerable: true,
7
+ get: function() {
8
+ return OffscreenSnapExecutor;
9
+ }
10
+ });
11
+ const _postmessagestream = require("@metamask/post-message-stream");
12
+ const _snapsutils = require("@metamask/snaps-utils");
13
+ const _utils = require("@metamask/utils");
14
+ function _check_private_redeclaration(obj, privateCollection) {
15
+ if (privateCollection.has(obj)) {
16
+ throw new TypeError("Cannot initialize the same private elements twice on an object");
17
+ }
18
+ }
19
+ function _class_apply_descriptor_get(receiver, descriptor) {
20
+ if (descriptor.get) {
21
+ return descriptor.get.call(receiver);
22
+ }
23
+ return descriptor.value;
24
+ }
25
+ function _class_apply_descriptor_set(receiver, descriptor, value) {
26
+ if (descriptor.set) {
27
+ descriptor.set.call(receiver, value);
28
+ } else {
29
+ if (!descriptor.writable) {
30
+ throw new TypeError("attempted to set read only private field");
31
+ }
32
+ descriptor.value = value;
33
+ }
34
+ }
35
+ function _class_extract_field_descriptor(receiver, privateMap, action) {
36
+ if (!privateMap.has(receiver)) {
37
+ throw new TypeError("attempted to " + action + " private field on non-instance");
38
+ }
39
+ return privateMap.get(receiver);
40
+ }
41
+ function _class_private_field_get(receiver, privateMap) {
42
+ var descriptor = _class_extract_field_descriptor(receiver, privateMap, "get");
43
+ return _class_apply_descriptor_get(receiver, descriptor);
44
+ }
45
+ function _class_private_field_init(obj, privateMap, value) {
46
+ _check_private_redeclaration(obj, privateMap);
47
+ privateMap.set(obj, value);
48
+ }
49
+ function _class_private_field_set(receiver, privateMap, value) {
50
+ var descriptor = _class_extract_field_descriptor(receiver, privateMap, "set");
51
+ _class_apply_descriptor_set(receiver, descriptor, value);
52
+ return value;
53
+ }
54
+ function _class_private_method_get(receiver, privateSet, fn) {
55
+ if (!privateSet.has(receiver)) {
56
+ throw new TypeError("attempted to get private field on non-instance");
57
+ }
58
+ return fn;
59
+ }
60
+ function _class_private_method_init(obj, privateSet) {
61
+ _check_private_redeclaration(obj, privateSet);
62
+ privateSet.add(obj);
63
+ }
64
+ function _define_property(obj, key, value) {
65
+ if (key in obj) {
66
+ Object.defineProperty(obj, key, {
67
+ value: value,
68
+ enumerable: true,
69
+ configurable: true,
70
+ writable: true
71
+ });
72
+ } else {
73
+ obj[key] = value;
74
+ }
75
+ return obj;
76
+ }
77
+ var _stream = /*#__PURE__*/ new WeakMap(), /**
78
+ * Handle an incoming message from the `OffscreenExecutionService`. This
79
+ * assumes that the message contains a `jobId` property, and a JSON-RPC
80
+ * request in the `data` property.
81
+ *
82
+ * @param data - The message data.
83
+ * @param data.data - The JSON-RPC request.
84
+ * @param data.jobId - The job ID.
85
+ * @param data.extra - Extra data.
86
+ * @param data.extra.frameUrl - The URL to load in the iframe.
87
+ */ _onData = /*#__PURE__*/ new WeakSet(), _initializeJob = /*#__PURE__*/ new WeakSet(), /**
88
+ * Terminate the job with the given ID. This will close the iframe and delete
89
+ * the job from the internal job map.
90
+ *
91
+ * @param jobId - The job ID.
92
+ */ _terminateJob = /*#__PURE__*/ new WeakSet();
93
+ class OffscreenSnapExecutor {
94
+ /**
95
+ * Initialize the executor with the given stream. This is a wrapper around the
96
+ * constructor.
97
+ *
98
+ * @param stream - The stream to use for communication.
99
+ * @returns The initialized executor.
100
+ */ static initialize(stream) {
101
+ return new OffscreenSnapExecutor(stream);
102
+ }
103
+ constructor(stream){
104
+ _class_private_method_init(this, _onData);
105
+ /**
106
+ * Create a new iframe and set up a stream to communicate with it.
107
+ *
108
+ * @param jobId - The job ID.
109
+ * @param frameUrl - The URL to load in the iframe.
110
+ */ _class_private_method_init(this, _initializeJob);
111
+ _class_private_method_init(this, _terminateJob);
112
+ _class_private_field_init(this, _stream, {
113
+ writable: true,
114
+ value: void 0
115
+ });
116
+ _define_property(this, "jobs", {});
117
+ _class_private_field_set(this, _stream, stream);
118
+ _class_private_field_get(this, _stream).on('data', _class_private_method_get(this, _onData, onData).bind(this));
119
+ }
120
+ }
121
+ function onData(data) {
122
+ const { jobId, extra: { frameUrl }, data: request } = data;
123
+ if (!this.jobs[jobId]) {
124
+ // This ensures that a job is initialized before it is used. To avoid
125
+ // code duplication, we call the `#onData` method again, which will
126
+ // run the rest of the logic after initialization.
127
+ _class_private_method_get(this, _initializeJob, initializeJob).call(this, jobId, frameUrl).then(()=>{
128
+ _class_private_method_get(this, _onData, onData).call(this, data);
129
+ }).catch((error)=>{
130
+ (0, _snapsutils.logError)('[Worker] Error initializing job:', error);
131
+ });
132
+ return;
133
+ }
134
+ // This is a method specific to the `OffscreenSnapExecutor`, as the service
135
+ // itself does not have access to the iframes directly.
136
+ if (request.method === 'terminateJob') {
137
+ _class_private_method_get(this, _terminateJob, terminateJob).call(this, jobId);
138
+ return;
139
+ }
140
+ this.jobs[jobId].stream.write(request);
141
+ }
142
+ async function initializeJob(jobId, frameUrl) {
143
+ const window = await (0, _snapsutils.createWindow)(frameUrl, jobId);
144
+ const jobStream = new _postmessagestream.WindowPostMessageStream({
145
+ name: 'parent',
146
+ target: 'child',
147
+ targetWindow: window,
148
+ targetOrigin: '*'
149
+ });
150
+ // Write messages from the iframe to the parent, wrapped with the job ID.
151
+ jobStream.on('data', (data)=>{
152
+ _class_private_field_get(this, _stream).write({
153
+ data,
154
+ jobId
155
+ });
156
+ });
157
+ this.jobs[jobId] = {
158
+ id: jobId,
159
+ window,
160
+ stream: jobStream
161
+ };
162
+ return this.jobs[jobId];
163
+ }
164
+ function terminateJob(jobId) {
165
+ (0, _utils.assert)(this.jobs[jobId], `Job "${jobId}" not found.`);
166
+ const iframe = document.getElementById(jobId);
167
+ (0, _utils.assert)(iframe, `Iframe with ID "${jobId}" not found.`);
168
+ iframe.remove();
169
+ this.jobs[jobId].stream.destroy();
170
+ delete this.jobs[jobId];
171
+ }
172
+
173
+ //# sourceMappingURL=OffscreenSnapExecutor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/offscreen/OffscreenSnapExecutor.ts"],"sourcesContent":["import type { BasePostMessageStream } from '@metamask/post-message-stream';\nimport { WindowPostMessageStream } from '@metamask/post-message-stream';\nimport { createWindow, logError } from '@metamask/snaps-utils';\nimport type { JsonRpcRequest } from '@metamask/utils';\nimport { assert } from '@metamask/utils';\n\ntype ExecutorJob = {\n id: string;\n window: Window;\n stream: WindowPostMessageStream;\n};\n\n/**\n * A snap executor using the Offscreen Documents API.\n *\n * This is not a traditional snap executor, as it does not execute snaps itself.\n * Instead, it creates an iframe window for each snap execution, and sends the\n * snap execution request to the iframe window. The iframe window is responsible\n * for executing the snap.\n *\n * Extensions can only have a single offscreen document, so this executor is\n * persisted between snap executions. The offscreen snap executor essentially\n * acts as a proxy between the extension and the iframe execution environment.\n *\n * @see https://developer.chrome.com/docs/extensions/reference/offscreen/\n */\nexport class OffscreenSnapExecutor {\n readonly #stream: BasePostMessageStream;\n\n readonly jobs: Record<string, ExecutorJob> = {};\n\n /**\n * Initialize the executor with the given stream. This is a wrapper around the\n * constructor.\n *\n * @param stream - The stream to use for communication.\n * @returns The initialized executor.\n */\n static initialize(stream: BasePostMessageStream) {\n return new OffscreenSnapExecutor(stream);\n }\n\n constructor(stream: BasePostMessageStream) {\n this.#stream = stream;\n this.#stream.on('data', this.#onData.bind(this));\n }\n\n /**\n * Handle an incoming message from the `OffscreenExecutionService`. This\n * assumes that the message contains a `jobId` property, and a JSON-RPC\n * request in the `data` property.\n *\n * @param data - The message data.\n * @param data.data - The JSON-RPC request.\n * @param data.jobId - The job ID.\n * @param data.extra - Extra data.\n * @param data.extra.frameUrl - The URL to load in the iframe.\n */\n #onData(data: {\n data: JsonRpcRequest;\n jobId: string;\n extra: { frameUrl: string };\n }) {\n const {\n jobId,\n extra: { frameUrl },\n data: request,\n } = data;\n\n if (!this.jobs[jobId]) {\n // This ensures that a job is initialized before it is used. To avoid\n // code duplication, we call the `#onData` method again, which will\n // run the rest of the logic after initialization.\n this.#initializeJob(jobId, frameUrl)\n .then(() => {\n this.#onData(data);\n })\n .catch((error) => {\n logError('[Worker] Error initializing job:', error);\n });\n\n return;\n }\n\n // This is a method specific to the `OffscreenSnapExecutor`, as the service\n // itself does not have access to the iframes directly.\n if (request.method === 'terminateJob') {\n this.#terminateJob(jobId);\n return;\n }\n\n this.jobs[jobId].stream.write(request);\n }\n\n /**\n * Create a new iframe and set up a stream to communicate with it.\n *\n * @param jobId - The job ID.\n * @param frameUrl - The URL to load in the iframe.\n */\n async #initializeJob(jobId: string, frameUrl: string): Promise<ExecutorJob> {\n const window = await createWindow(frameUrl, jobId);\n const jobStream = new WindowPostMessageStream({\n name: 'parent',\n target: 'child',\n targetWindow: window,\n targetOrigin: '*',\n });\n\n // Write messages from the iframe to the parent, wrapped with the job ID.\n jobStream.on('data', (data) => {\n this.#stream.write({ data, jobId });\n });\n\n this.jobs[jobId] = { id: jobId, window, stream: jobStream };\n return this.jobs[jobId];\n }\n\n /**\n * Terminate the job with the given ID. This will close the iframe and delete\n * the job from the internal job map.\n *\n * @param jobId - The job ID.\n */\n #terminateJob(jobId: string) {\n assert(this.jobs[jobId], `Job \"${jobId}\" not found.`);\n\n const iframe = document.getElementById(jobId);\n assert(iframe, `Iframe with ID \"${jobId}\" not found.`);\n\n iframe.remove();\n this.jobs[jobId].stream.destroy();\n delete this.jobs[jobId];\n }\n}\n"],"names":["OffscreenSnapExecutor","initialize","stream","constructor","jobs","on","onData","bind","data","jobId","extra","frameUrl","request","initializeJob","then","catch","error","logError","method","terminateJob","write","window","createWindow","jobStream","WindowPostMessageStream","name","target","targetWindow","targetOrigin","id","assert","iframe","document","getElementById","remove","destroy"],"mappings":";;;;+BA0BaA;;;eAAAA;;;mCAzB2B;4BACD;uBAEhB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAuBZ,uCAoBT;;;;;;;;;;GAUC,GACD,uCA0CM,8CAkBN;;;;;GAKC,GACD;AAlGK,MAAMA;IAKX;;;;;;GAMC,GACD,OAAOC,WAAWC,MAA6B,EAAE;QAC/C,OAAO,IAAIF,sBAAsBE;IACnC;IAEAC,YAAYD,MAA6B,CAAE;QAgB3C,iCAAA;QAoCA;;;;;GAKC,GACD,iCAAM;QAwBN,iCAAA;QAjGA,gCAAS;;mBAAT,KAAA;;QAEA,uBAASE,QAAoC,CAAC;uCActCF,SAASA;QACf,yBAAA,IAAI,EAAEA,SAAOG,EAAE,CAAC,QAAQ,0BAAA,IAAI,EAAEC,SAAAA,QAAOC,IAAI,CAAC,IAAI;IAChD;AAyFF;AA5EE,SAAA,OAAQC,IAIP;IACC,MAAM,EACJC,KAAK,EACLC,OAAO,EAAEC,QAAQ,EAAE,EACnBH,MAAMI,OAAO,EACd,GAAGJ;IAEJ,IAAI,CAAC,IAAI,CAACJ,IAAI,CAACK,MAAM,EAAE;QACrB,qEAAqE;QACrE,mEAAmE;QACnE,kDAAkD;QAClD,0BAAA,IAAI,EAAEI,gBAAAA,oBAAN,IAAI,EAAgBJ,OAAOE,UACxBG,IAAI,CAAC;YACJ,0BAAA,IAAI,EAAER,SAAAA,aAAN,IAAI,EAASE;QACf,GACCO,KAAK,CAAC,CAACC;YACNC,IAAAA,oBAAQ,EAAC,oCAAoCD;QAC/C;QAEF;IACF;IAEA,2EAA2E;IAC3E,uDAAuD;IACvD,IAAIJ,QAAQM,MAAM,KAAK,gBAAgB;QACrC,0BAAA,IAAI,EAAEC,eAAAA,mBAAN,IAAI,EAAeV;QACnB;IACF;IAEA,IAAI,CAACL,IAAI,CAACK,MAAM,CAACP,MAAM,CAACkB,KAAK,CAACR;AAChC;AAQA,eAAA,cAAqBH,KAAa,EAAEE,QAAgB;IAClD,MAAMU,SAAS,MAAMC,IAAAA,wBAAY,EAACX,UAAUF;IAC5C,MAAMc,YAAY,IAAIC,0CAAuB,CAAC;QAC5CC,MAAM;QACNC,QAAQ;QACRC,cAAcN;QACdO,cAAc;IAChB;IAEA,yEAAyE;IACzEL,UAAUlB,EAAE,CAAC,QAAQ,CAACG;QACpB,yBAAA,IAAI,EAAEN,SAAOkB,KAAK,CAAC;YAAEZ;YAAMC;QAAM;IACnC;IAEA,IAAI,CAACL,IAAI,CAACK,MAAM,GAAG;QAAEoB,IAAIpB;QAAOY;QAAQnB,QAAQqB;IAAU;IAC1D,OAAO,IAAI,CAACnB,IAAI,CAACK,MAAM;AACzB;AAQA,SAAA,aAAcA,KAAa;IACzBqB,IAAAA,aAAM,EAAC,IAAI,CAAC1B,IAAI,CAACK,MAAM,EAAE,CAAC,KAAK,EAAEA,MAAM,YAAY,CAAC;IAEpD,MAAMsB,SAASC,SAASC,cAAc,CAACxB;IACvCqB,IAAAA,aAAM,EAACC,QAAQ,CAAC,gBAAgB,EAAEtB,MAAM,YAAY,CAAC;IAErDsB,OAAOG,MAAM;IACb,IAAI,CAAC9B,IAAI,CAACK,MAAM,CAACP,MAAM,CAACiC,OAAO;IAC/B,OAAO,IAAI,CAAC/B,IAAI,CAACK,MAAM;AACzB"}
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ const _postmessagestream = require("@metamask/post-message-stream");
6
+ const _lockdownevents = require("../common/lockdown/lockdown-events");
7
+ const _lockdownmore = require("../common/lockdown/lockdown-more");
8
+ const _OffscreenSnapExecutor = require("./OffscreenSnapExecutor");
9
+ // Lockdown is already applied in LavaMoat
10
+ (0, _lockdownmore.executeLockdownMore)();
11
+ (0, _lockdownevents.executeLockdownEvents)();
12
+ // The stream from the offscreen document to the execution service.
13
+ const parentStream = new _postmessagestream.BrowserRuntimePostMessageStream({
14
+ name: 'child',
15
+ target: 'parent'
16
+ });
17
+ _OffscreenSnapExecutor.OffscreenSnapExecutor.initialize(parentStream);
18
+
19
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../src/offscreen/index.ts"],"sourcesContent":["import { BrowserRuntimePostMessageStream } from '@metamask/post-message-stream';\n\nimport { executeLockdownEvents } from '../common/lockdown/lockdown-events';\nimport { executeLockdownMore } from '../common/lockdown/lockdown-more';\nimport { OffscreenSnapExecutor } from './OffscreenSnapExecutor';\n\n// Lockdown is already applied in LavaMoat\nexecuteLockdownMore();\nexecuteLockdownEvents();\n\n// The stream from the offscreen document to the execution service.\nconst parentStream = new BrowserRuntimePostMessageStream({\n name: 'child',\n target: 'parent',\n});\n\nOffscreenSnapExecutor.initialize(parentStream);\n"],"names":["executeLockdownMore","executeLockdownEvents","parentStream","BrowserRuntimePostMessageStream","name","target","OffscreenSnapExecutor","initialize"],"mappings":";;;;mCAAgD;gCAEV;8BACF;uCACE;AAEtC,0CAA0C;AAC1CA,IAAAA,iCAAmB;AACnBC,IAAAA,qCAAqB;AAErB,mEAAmE;AACnE,MAAMC,eAAe,IAAIC,kDAA+B,CAAC;IACvDC,MAAM;IACNC,QAAQ;AACV;AAEAC,4CAAqB,CAACC,UAAU,CAACL"}
@@ -0,0 +1,45 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "WebWorkerSnapExecutor", {
6
+ enumerable: true,
7
+ get: function() {
8
+ return WebWorkerSnapExecutor;
9
+ }
10
+ });
11
+ const _objectmultiplex = /*#__PURE__*/ _interop_require_default(require("@metamask/object-multiplex"));
12
+ const _postmessagestream = require("@metamask/post-message-stream");
13
+ const _snapsutils = require("@metamask/snaps-utils");
14
+ const _stream = require("stream");
15
+ const _BaseSnapExecutor = require("../../common/BaseSnapExecutor");
16
+ const _logging = require("../../logging");
17
+ function _interop_require_default(obj) {
18
+ return obj && obj.__esModule ? obj : {
19
+ default: obj
20
+ };
21
+ }
22
+ class WebWorkerSnapExecutor extends _BaseSnapExecutor.BaseSnapExecutor {
23
+ /**
24
+ * Initialize the WebWorkerSnapExecutor. This creates a post message stream
25
+ * from and to the parent window, for two-way communication with the iframe.
26
+ *
27
+ * @param stream - The stream to use for communication.
28
+ * @returns An instance of `WebWorkerSnapExecutor`, with the initialized post
29
+ * message streams.
30
+ */ static initialize(stream = new _postmessagestream.WebWorkerPostMessageStream()) {
31
+ (0, _logging.log)('Worker: Connecting to parent.');
32
+ const mux = new _objectmultiplex.default();
33
+ (0, _stream.pipeline)(stream, mux, stream, (error)=>{
34
+ if (error) {
35
+ (0, _snapsutils.logError)(`Parent stream failure, closing worker.`, error);
36
+ }
37
+ self.close();
38
+ });
39
+ const commandStream = mux.createStream(_snapsutils.SNAP_STREAM_NAMES.COMMAND);
40
+ const rpcStream = mux.createStream(_snapsutils.SNAP_STREAM_NAMES.JSON_RPC);
41
+ return new WebWorkerSnapExecutor(commandStream, rpcStream);
42
+ }
43
+ }
44
+
45
+ //# sourceMappingURL=WebWorkerSnapExecutor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../src/webworker/executor/WebWorkerSnapExecutor.ts"],"sourcesContent":["import ObjectMultiplex from '@metamask/object-multiplex';\nimport type { BasePostMessageStream } from '@metamask/post-message-stream';\nimport { WebWorkerPostMessageStream } from '@metamask/post-message-stream';\nimport { logError, SNAP_STREAM_NAMES } from '@metamask/snaps-utils';\nimport { pipeline } from 'stream';\n\nimport { BaseSnapExecutor } from '../../common/BaseSnapExecutor';\nimport { log } from '../../logging';\n\nexport class WebWorkerSnapExecutor extends BaseSnapExecutor {\n /**\n * Initialize the WebWorkerSnapExecutor. This creates a post message stream\n * from and to the parent window, for two-way communication with the iframe.\n *\n * @param stream - The stream to use for communication.\n * @returns An instance of `WebWorkerSnapExecutor`, with the initialized post\n * message streams.\n */\n static initialize(\n stream: BasePostMessageStream = new WebWorkerPostMessageStream(),\n ) {\n log('Worker: Connecting to parent.');\n\n const mux = new ObjectMultiplex();\n pipeline(stream, mux, stream, (error) => {\n if (error) {\n logError(`Parent stream failure, closing worker.`, error);\n }\n self.close();\n });\n\n const commandStream = mux.createStream(SNAP_STREAM_NAMES.COMMAND);\n const rpcStream = mux.createStream(SNAP_STREAM_NAMES.JSON_RPC);\n\n return new WebWorkerSnapExecutor(commandStream, rpcStream);\n }\n}\n"],"names":["WebWorkerSnapExecutor","BaseSnapExecutor","initialize","stream","WebWorkerPostMessageStream","log","mux","ObjectMultiplex","pipeline","error","logError","self","close","commandStream","createStream","SNAP_STREAM_NAMES","COMMAND","rpcStream","JSON_RPC"],"mappings":";;;;+BASaA;;;eAAAA;;;wEATe;mCAEe;4BACC;wBACnB;kCAEQ;yBACb;;;;;;AAEb,MAAMA,8BAA8BC,kCAAgB;IACzD;;;;;;;GAOC,GACD,OAAOC,WACLC,SAAgC,IAAIC,6CAA0B,EAAE,EAChE;QACAC,IAAAA,YAAG,EAAC;QAEJ,MAAMC,MAAM,IAAIC,wBAAe;QAC/BC,IAAAA,gBAAQ,EAACL,QAAQG,KAAKH,QAAQ,CAACM;YAC7B,IAAIA,OAAO;gBACTC,IAAAA,oBAAQ,EAAC,CAAC,sCAAsC,CAAC,EAAED;YACrD;YACAE,KAAKC,KAAK;QACZ;QAEA,MAAMC,gBAAgBP,IAAIQ,YAAY,CAACC,6BAAiB,CAACC,OAAO;QAChE,MAAMC,YAAYX,IAAIQ,YAAY,CAACC,6BAAiB,CAACG,QAAQ;QAE7D,OAAO,IAAIlB,sBAAsBa,eAAeI;IAClD;AACF"}
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ const _lockdownevents = require("../../common/lockdown/lockdown-events");
6
+ const _lockdownmore = require("../../common/lockdown/lockdown-more");
7
+ const _WebWorkerSnapExecutor = require("./WebWorkerSnapExecutor");
8
+ // Lockdown is already applied in LavaMoat
9
+ (0, _lockdownmore.executeLockdownMore)();
10
+ (0, _lockdownevents.executeLockdownEvents)();
11
+ _WebWorkerSnapExecutor.WebWorkerSnapExecutor.initialize();
12
+
13
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../src/webworker/executor/index.ts"],"sourcesContent":["import { executeLockdownEvents } from '../../common/lockdown/lockdown-events';\nimport { executeLockdownMore } from '../../common/lockdown/lockdown-more';\nimport { WebWorkerSnapExecutor } from './WebWorkerSnapExecutor';\n\n// Lockdown is already applied in LavaMoat\nexecuteLockdownMore();\nexecuteLockdownEvents();\n\nWebWorkerSnapExecutor.initialize();\n"],"names":["executeLockdownMore","executeLockdownEvents","WebWorkerSnapExecutor","initialize"],"mappings":";;;;gCAAsC;8BACF;uCACE;AAEtC,0CAA0C;AAC1CA,IAAAA,iCAAmB;AACnBC,IAAAA,qCAAqB;AAErBC,4CAAqB,CAACC,UAAU"}
@@ -0,0 +1,248 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "WebWorkerPool", {
6
+ enumerable: true,
7
+ get: function() {
8
+ return WebWorkerPool;
9
+ }
10
+ });
11
+ const _postmessagestream = require("@metamask/post-message-stream");
12
+ const _snapsutils = require("@metamask/snaps-utils");
13
+ const _utils = require("@metamask/utils");
14
+ const _nonsecure = require("nanoid/non-secure");
15
+ function _check_private_redeclaration(obj, privateCollection) {
16
+ if (privateCollection.has(obj)) {
17
+ throw new TypeError("Cannot initialize the same private elements twice on an object");
18
+ }
19
+ }
20
+ function _class_apply_descriptor_get(receiver, descriptor) {
21
+ if (descriptor.get) {
22
+ return descriptor.get.call(receiver);
23
+ }
24
+ return descriptor.value;
25
+ }
26
+ function _class_apply_descriptor_set(receiver, descriptor, value) {
27
+ if (descriptor.set) {
28
+ descriptor.set.call(receiver, value);
29
+ } else {
30
+ if (!descriptor.writable) {
31
+ throw new TypeError("attempted to set read only private field");
32
+ }
33
+ descriptor.value = value;
34
+ }
35
+ }
36
+ function _class_extract_field_descriptor(receiver, privateMap, action) {
37
+ if (!privateMap.has(receiver)) {
38
+ throw new TypeError("attempted to " + action + " private field on non-instance");
39
+ }
40
+ return privateMap.get(receiver);
41
+ }
42
+ function _class_private_field_get(receiver, privateMap) {
43
+ var descriptor = _class_extract_field_descriptor(receiver, privateMap, "get");
44
+ return _class_apply_descriptor_get(receiver, descriptor);
45
+ }
46
+ function _class_private_field_init(obj, privateMap, value) {
47
+ _check_private_redeclaration(obj, privateMap);
48
+ privateMap.set(obj, value);
49
+ }
50
+ function _class_private_field_set(receiver, privateMap, value) {
51
+ var descriptor = _class_extract_field_descriptor(receiver, privateMap, "set");
52
+ _class_apply_descriptor_set(receiver, descriptor, value);
53
+ return value;
54
+ }
55
+ function _class_private_method_get(receiver, privateSet, fn) {
56
+ if (!privateSet.has(receiver)) {
57
+ throw new TypeError("attempted to get private field on non-instance");
58
+ }
59
+ return fn;
60
+ }
61
+ function _class_private_method_init(obj, privateSet) {
62
+ _check_private_redeclaration(obj, privateSet);
63
+ privateSet.add(obj);
64
+ }
65
+ function _define_property(obj, key, value) {
66
+ if (key in obj) {
67
+ Object.defineProperty(obj, key, {
68
+ value: value,
69
+ enumerable: true,
70
+ configurable: true,
71
+ writable: true
72
+ });
73
+ } else {
74
+ obj[key] = value;
75
+ }
76
+ return obj;
77
+ }
78
+ var _poolSize = /*#__PURE__*/ new WeakMap(), _stream = /*#__PURE__*/ new WeakMap(), _url = /*#__PURE__*/ new WeakMap(), _workerSourceURL = /*#__PURE__*/ new WeakMap(), /**
79
+ * Handle an incoming message from the `WebWorkerExecutionService`. This
80
+ * assumes that the message contains a `jobId` property, and a JSON-RPC
81
+ * request in the `data` property.
82
+ *
83
+ * @param data - The message data.
84
+ * @param data.data - The JSON-RPC request.
85
+ * @param data.jobId - The job ID.
86
+ */ _onData = /*#__PURE__*/ new WeakSet(), _initializeJob = /*#__PURE__*/ new WeakSet(), /**
87
+ * Terminate the job with the given ID. This will close the worker and delete
88
+ * the job from the internal job map.
89
+ *
90
+ * @param jobId - The job ID.
91
+ */ _terminateJob = /*#__PURE__*/ new WeakSet(), _getWorker = /*#__PURE__*/ new WeakSet(), _updatePool = /*#__PURE__*/ new WeakSet(), _createWorker = /*#__PURE__*/ new WeakSet(), _getWorkerURL = /*#__PURE__*/ new WeakSet();
92
+ class WebWorkerPool {
93
+ /* istanbul ignore next - Constructor arguments. */ static initialize(stream = new _postmessagestream.WindowPostMessageStream({
94
+ name: 'child',
95
+ target: 'parent',
96
+ targetWindow: self.parent,
97
+ targetOrigin: '*'
98
+ }), url = '../executor/bundle.js', poolSize) {
99
+ return new WebWorkerPool(stream, url, poolSize);
100
+ }
101
+ constructor(stream, url, poolSize = 3){
102
+ _class_private_method_init(this, _onData);
103
+ /**
104
+ * Create a new worker and set up a stream to communicate with it.
105
+ *
106
+ * @param jobId - The job ID.
107
+ * @returns The job.
108
+ */ _class_private_method_init(this, _initializeJob);
109
+ _class_private_method_init(this, _terminateJob);
110
+ /**
111
+ * Get a worker from the pool. A new worker will be created automatically.
112
+ *
113
+ * @returns The worker.
114
+ */ _class_private_method_init(this, _getWorker);
115
+ /**
116
+ * Update the pool of workers. This will create new workers if the pool is
117
+ * below the minimum size.
118
+ */ _class_private_method_init(this, _updatePool);
119
+ /**
120
+ * Create a new worker. This will fetch the worker source if it has not
121
+ * already been fetched.
122
+ *
123
+ * @returns The worker.
124
+ */ _class_private_method_init(this, _createWorker);
125
+ /**
126
+ * Get the URL of the worker source. This will fetch the worker source if it
127
+ * has not already been fetched.
128
+ *
129
+ * @returns The worker source URL, as a `blob:` URL.
130
+ */ _class_private_method_init(this, _getWorkerURL);
131
+ _class_private_field_init(this, _poolSize, {
132
+ writable: true,
133
+ value: void 0
134
+ });
135
+ _class_private_field_init(this, _stream, {
136
+ writable: true,
137
+ value: void 0
138
+ });
139
+ _class_private_field_init(this, _url, {
140
+ writable: true,
141
+ value: void 0
142
+ });
143
+ _define_property(this, "pool", []);
144
+ _define_property(this, "jobs", new Map());
145
+ _class_private_field_init(this, _workerSourceURL, {
146
+ writable: true,
147
+ value: void 0
148
+ });
149
+ _class_private_field_set(this, _stream, stream);
150
+ _class_private_field_set(this, _url, url);
151
+ _class_private_field_set(this, _poolSize, poolSize);
152
+ _class_private_field_get(this, _stream).on('data', _class_private_method_get(this, _onData, onData).bind(this));
153
+ }
154
+ }
155
+ function onData(data) {
156
+ const { jobId, data: request } = data;
157
+ const job = this.jobs.get(jobId);
158
+ if (!job) {
159
+ // This ensures that a job is initialized before it is used. To avoid
160
+ // code duplication, we call the `#onData` method again, which will
161
+ // run the rest of the logic after initialization.
162
+ _class_private_method_get(this, _initializeJob, initializeJob).call(this, jobId).then(()=>{
163
+ _class_private_method_get(this, _onData, onData).call(this, data);
164
+ }).catch((error)=>{
165
+ (0, _snapsutils.logError)('[Worker] Error initializing job:', error.toString());
166
+ _class_private_field_get(this, _stream).write({
167
+ jobId,
168
+ data: {
169
+ name: 'command',
170
+ data: {
171
+ jsonrpc: '2.0',
172
+ id: request.id ?? null,
173
+ error: {
174
+ code: -32000,
175
+ message: 'Internal error'
176
+ }
177
+ }
178
+ }
179
+ });
180
+ });
181
+ return;
182
+ }
183
+ // This is a method specific to the `WebWorkerPool`, as the service itself
184
+ // does not have access to the workers directly.
185
+ if (request.method === 'terminateJob') {
186
+ _class_private_method_get(this, _terminateJob, terminateJob).call(this, jobId);
187
+ return;
188
+ }
189
+ job.stream.write(request);
190
+ }
191
+ async function initializeJob(jobId) {
192
+ const worker = await _class_private_method_get(this, _getWorker, getWorker).call(this);
193
+ const jobStream = new _postmessagestream.WebWorkerParentPostMessageStream({
194
+ worker
195
+ });
196
+ // Write messages from the worker to the parent, wrapped with the job ID.
197
+ jobStream.on('data', (data)=>{
198
+ _class_private_field_get(this, _stream).write({
199
+ data,
200
+ jobId
201
+ });
202
+ });
203
+ const job = {
204
+ id: jobId,
205
+ worker,
206
+ stream: jobStream
207
+ };
208
+ this.jobs.set(jobId, job);
209
+ return job;
210
+ }
211
+ function terminateJob(jobId) {
212
+ const job = this.jobs.get(jobId);
213
+ (0, _utils.assert)(job, `Job "${jobId}" not found.`);
214
+ job.stream.destroy();
215
+ job.worker.terminate();
216
+ this.jobs.delete(jobId);
217
+ }
218
+ async function getWorker() {
219
+ // Lazily create the pool of workers.
220
+ if (this.pool.length === 0) {
221
+ await _class_private_method_get(this, _updatePool, updatePool).call(this);
222
+ }
223
+ const worker = this.pool.shift();
224
+ (0, _utils.assert)(worker, 'Worker not found.');
225
+ await _class_private_method_get(this, _updatePool, updatePool).call(this);
226
+ return worker;
227
+ }
228
+ async function updatePool() {
229
+ while(this.pool.length < _class_private_field_get(this, _poolSize)){
230
+ const worker = await _class_private_method_get(this, _createWorker, createWorker).call(this);
231
+ this.pool.push(worker);
232
+ }
233
+ }
234
+ async function createWorker() {
235
+ return new Worker(await _class_private_method_get(this, _getWorkerURL, getWorkerURL).call(this), {
236
+ name: `worker-${(0, _nonsecure.nanoid)()}`
237
+ });
238
+ }
239
+ async function getWorkerURL() {
240
+ if (_class_private_field_get(this, _workerSourceURL)) {
241
+ return _class_private_field_get(this, _workerSourceURL);
242
+ }
243
+ const blob = await fetch(_class_private_field_get(this, _url)).then(async (response)=>response.blob()).then(URL.createObjectURL.bind(URL));
244
+ _class_private_field_set(this, _workerSourceURL, blob);
245
+ return blob;
246
+ }
247
+
248
+ //# sourceMappingURL=WebWorkerPool.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../src/webworker/pool/WebWorkerPool.ts"],"sourcesContent":["import type { BasePostMessageStream } from '@metamask/post-message-stream';\nimport {\n WebWorkerParentPostMessageStream,\n WindowPostMessageStream,\n} from '@metamask/post-message-stream';\nimport { logError } from '@metamask/snaps-utils';\nimport type { JsonRpcRequest } from '@metamask/utils';\nimport { assert } from '@metamask/utils';\nimport { nanoid } from 'nanoid/non-secure';\n\ntype ExecutorJob = {\n id: string;\n worker: Worker;\n stream: WebWorkerParentPostMessageStream;\n};\n\n/**\n * A snap executor using the WebWorker API.\n *\n * This is not a traditional snap executor, as it does not execute snaps itself.\n * Instead, it creates a pool of webworkers for each snap execution, and sends\n * the snap execution request to the webworker. The webworker is responsible for\n * executing the snap.\n */\nexport class WebWorkerPool {\n readonly #poolSize;\n\n readonly #stream: BasePostMessageStream;\n\n readonly #url: string;\n\n readonly pool: Worker[] = [];\n\n readonly jobs: Map<string, ExecutorJob> = new Map();\n\n #workerSourceURL?: string;\n\n /* istanbul ignore next - Constructor arguments. */\n static initialize(\n stream: BasePostMessageStream = new WindowPostMessageStream({\n name: 'child',\n target: 'parent',\n targetWindow: self.parent,\n targetOrigin: '*',\n }),\n url = '../executor/bundle.js',\n poolSize?: number,\n ) {\n return new WebWorkerPool(stream, url, poolSize);\n }\n\n constructor(stream: BasePostMessageStream, url: string, poolSize = 3) {\n this.#stream = stream;\n this.#url = url;\n this.#poolSize = poolSize;\n\n this.#stream.on('data', this.#onData.bind(this));\n }\n\n /**\n * Handle an incoming message from the `WebWorkerExecutionService`. This\n * assumes that the message contains a `jobId` property, and a JSON-RPC\n * request in the `data` property.\n *\n * @param data - The message data.\n * @param data.data - The JSON-RPC request.\n * @param data.jobId - The job ID.\n */\n #onData(data: { data: JsonRpcRequest; jobId: string }) {\n const { jobId, data: request } = data;\n\n const job = this.jobs.get(jobId);\n if (!job) {\n // This ensures that a job is initialized before it is used. To avoid\n // code duplication, we call the `#onData` method again, which will\n // run the rest of the logic after initialization.\n this.#initializeJob(jobId)\n .then(() => {\n this.#onData(data);\n })\n .catch((error) => {\n logError('[Worker] Error initializing job:', error.toString());\n\n this.#stream.write({\n jobId,\n data: {\n name: 'command',\n data: {\n jsonrpc: '2.0',\n id: request.id ?? null,\n error: {\n code: -32000,\n message: 'Internal error',\n },\n },\n },\n });\n });\n\n return;\n }\n\n // This is a method specific to the `WebWorkerPool`, as the service itself\n // does not have access to the workers directly.\n if (request.method === 'terminateJob') {\n this.#terminateJob(jobId);\n return;\n }\n\n job.stream.write(request);\n }\n\n /**\n * Create a new worker and set up a stream to communicate with it.\n *\n * @param jobId - The job ID.\n * @returns The job.\n */\n async #initializeJob(jobId: string): Promise<ExecutorJob> {\n const worker = await this.#getWorker();\n const jobStream = new WebWorkerParentPostMessageStream({\n worker,\n });\n\n // Write messages from the worker to the parent, wrapped with the job ID.\n jobStream.on('data', (data) => {\n this.#stream.write({ data, jobId });\n });\n\n const job = { id: jobId, worker, stream: jobStream };\n this.jobs.set(jobId, job);\n return job;\n }\n\n /**\n * Terminate the job with the given ID. This will close the worker and delete\n * the job from the internal job map.\n *\n * @param jobId - The job ID.\n */\n #terminateJob(jobId: string) {\n const job = this.jobs.get(jobId);\n assert(job, `Job \"${jobId}\" not found.`);\n\n job.stream.destroy();\n job.worker.terminate();\n\n this.jobs.delete(jobId);\n }\n\n /**\n * Get a worker from the pool. A new worker will be created automatically.\n *\n * @returns The worker.\n */\n async #getWorker() {\n // Lazily create the pool of workers.\n if (this.pool.length === 0) {\n await this.#updatePool();\n }\n\n const worker = this.pool.shift();\n assert(worker, 'Worker not found.');\n\n await this.#updatePool();\n\n return worker;\n }\n\n /**\n * Update the pool of workers. This will create new workers if the pool is\n * below the minimum size.\n */\n async #updatePool() {\n while (this.pool.length < this.#poolSize) {\n const worker = await this.#createWorker();\n this.pool.push(worker);\n }\n }\n\n /**\n * Create a new worker. This will fetch the worker source if it has not\n * already been fetched.\n *\n * @returns The worker.\n */\n async #createWorker() {\n return new Worker(await this.#getWorkerURL(), {\n name: `worker-${nanoid()}`,\n });\n }\n\n /**\n * Get the URL of the worker source. This will fetch the worker source if it\n * has not already been fetched.\n *\n * @returns The worker source URL, as a `blob:` URL.\n */\n async #getWorkerURL() {\n if (this.#workerSourceURL) {\n return this.#workerSourceURL;\n }\n\n const blob = await fetch(this.#url)\n .then(async (response) => response.blob())\n .then(URL.createObjectURL.bind(URL));\n\n this.#workerSourceURL = blob;\n return blob;\n }\n}\n"],"names":["WebWorkerPool","initialize","stream","WindowPostMessageStream","name","target","targetWindow","self","parent","targetOrigin","url","poolSize","constructor","pool","jobs","Map","on","onData","bind","data","jobId","request","job","get","initializeJob","then","catch","error","logError","toString","write","jsonrpc","id","code","message","method","terminateJob","worker","getWorker","jobStream","WebWorkerParentPostMessageStream","set","assert","destroy","terminate","delete","length","updatePool","shift","createWorker","push","Worker","getWorkerURL","nanoid","workerSourceURL","blob","fetch","response","URL","createObjectURL"],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ const _lockdownevents = require("../../common/lockdown/lockdown-events");
6
+ const _lockdownmore = require("../../common/lockdown/lockdown-more");
7
+ const _WebWorkerPool = require("./WebWorkerPool");
8
+ // Lockdown is already applied in LavaMoat
9
+ (0, _lockdownmore.executeLockdownMore)();
10
+ (0, _lockdownevents.executeLockdownEvents)();
11
+ _WebWorkerPool.WebWorkerPool.initialize();
12
+
13
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../src/webworker/pool/index.ts"],"sourcesContent":["import { executeLockdownEvents } from '../../common/lockdown/lockdown-events';\nimport { executeLockdownMore } from '../../common/lockdown/lockdown-more';\nimport { WebWorkerPool } from './WebWorkerPool';\n\n// Lockdown is already applied in LavaMoat\nexecuteLockdownMore();\nexecuteLockdownEvents();\n\nWebWorkerPool.initialize();\n"],"names":["executeLockdownMore","executeLockdownEvents","WebWorkerPool","initialize"],"mappings":";;;;gCAAsC;8BACF;+BACN;AAE9B,0CAA0C;AAC1CA,IAAAA,iCAAmB;AACnBC,IAAAA,qCAAqB;AAErBC,4BAAa,CAACC,UAAU"}