wrangler 2.0.12 → 2.0.16

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 (149) hide show
  1. package/README.md +7 -1
  2. package/bin/wrangler.js +111 -57
  3. package/miniflare-dist/index.mjs +9 -2
  4. package/package.json +156 -154
  5. package/src/__tests__/config-cache-without-cache-dir.test.ts +38 -0
  6. package/src/__tests__/config-cache.test.ts +30 -24
  7. package/src/__tests__/configuration.test.ts +3935 -3476
  8. package/src/__tests__/dev.test.tsx +1128 -979
  9. package/src/__tests__/guess-worker-format.test.ts +68 -68
  10. package/src/__tests__/helpers/cmd-shim.d.ts +6 -6
  11. package/src/__tests__/helpers/faye-websocket.d.ts +4 -4
  12. package/src/__tests__/helpers/mock-account-id.ts +24 -24
  13. package/src/__tests__/helpers/mock-bin.ts +20 -20
  14. package/src/__tests__/helpers/mock-cfetch.ts +92 -92
  15. package/src/__tests__/helpers/mock-console.ts +49 -39
  16. package/src/__tests__/helpers/mock-dialogs.ts +94 -71
  17. package/src/__tests__/helpers/mock-http-server.ts +30 -30
  18. package/src/__tests__/helpers/mock-istty.ts +65 -18
  19. package/src/__tests__/helpers/mock-kv.ts +26 -26
  20. package/src/__tests__/helpers/mock-oauth-flow.ts +223 -228
  21. package/src/__tests__/helpers/mock-process.ts +39 -0
  22. package/src/__tests__/helpers/mock-stdin.ts +82 -77
  23. package/src/__tests__/helpers/mock-web-socket.ts +21 -21
  24. package/src/__tests__/helpers/run-in-tmp.ts +27 -27
  25. package/src/__tests__/helpers/run-wrangler.ts +8 -8
  26. package/src/__tests__/helpers/write-worker-source.ts +16 -16
  27. package/src/__tests__/helpers/write-wrangler-toml.ts +9 -9
  28. package/src/__tests__/https-options.test.ts +104 -104
  29. package/src/__tests__/index.test.ts +239 -234
  30. package/src/__tests__/init.test.ts +1605 -1250
  31. package/src/__tests__/jest.setup.ts +63 -33
  32. package/src/__tests__/kv.test.ts +1128 -1011
  33. package/src/__tests__/logger.test.ts +100 -74
  34. package/src/__tests__/package-manager.test.ts +303 -303
  35. package/src/__tests__/pages.test.ts +1152 -652
  36. package/src/__tests__/parse.test.ts +252 -252
  37. package/src/__tests__/publish.test.ts +6371 -5622
  38. package/src/__tests__/pubsub.test.ts +367 -0
  39. package/src/__tests__/r2.test.ts +133 -133
  40. package/src/__tests__/route.test.ts +18 -18
  41. package/src/__tests__/secret.test.ts +382 -377
  42. package/src/__tests__/tail.test.ts +530 -530
  43. package/src/__tests__/user.test.ts +123 -111
  44. package/src/__tests__/whoami.test.tsx +198 -117
  45. package/src/__tests__/worker-namespace.test.ts +327 -0
  46. package/src/abort.d.ts +1 -1
  47. package/src/api/dev.ts +49 -0
  48. package/src/api/index.ts +1 -0
  49. package/src/bundle-reporter.tsx +29 -0
  50. package/src/bundle.ts +157 -149
  51. package/src/cfetch/index.ts +80 -80
  52. package/src/cfetch/internal.ts +90 -83
  53. package/src/cli.ts +21 -7
  54. package/src/config/config.ts +204 -195
  55. package/src/config/diagnostics.ts +61 -61
  56. package/src/config/environment.ts +390 -357
  57. package/src/config/index.ts +206 -193
  58. package/src/config/validation-helpers.ts +366 -366
  59. package/src/config/validation.ts +1573 -1376
  60. package/src/config-cache.ts +79 -41
  61. package/src/create-worker-preview.ts +206 -136
  62. package/src/create-worker-upload-form.ts +247 -238
  63. package/src/dev/dev-vars.ts +13 -13
  64. package/src/dev/dev.tsx +329 -307
  65. package/src/dev/local.tsx +304 -275
  66. package/src/dev/remote.tsx +366 -224
  67. package/src/dev/use-esbuild.ts +126 -91
  68. package/src/dev.tsx +538 -0
  69. package/src/dialogs.tsx +97 -97
  70. package/src/durable.ts +87 -87
  71. package/src/entry.ts +234 -228
  72. package/src/environment-variables.ts +23 -23
  73. package/src/errors.ts +6 -6
  74. package/src/generate.ts +33 -0
  75. package/src/git-client.ts +42 -0
  76. package/src/https-options.ts +79 -79
  77. package/src/index.tsx +1775 -2763
  78. package/src/init.ts +549 -0
  79. package/src/inspect.ts +593 -593
  80. package/src/intl-polyfill.d.ts +123 -123
  81. package/src/is-interactive.ts +12 -0
  82. package/src/kv.ts +277 -277
  83. package/src/logger.ts +46 -39
  84. package/src/miniflare-cli/enum-keys.ts +8 -8
  85. package/src/miniflare-cli/index.ts +42 -31
  86. package/src/miniflare-cli/request-context.ts +18 -18
  87. package/src/module-collection.ts +212 -212
  88. package/src/open-in-browser.ts +4 -6
  89. package/src/package-manager.ts +123 -123
  90. package/src/pages/build.tsx +202 -0
  91. package/src/pages/constants.ts +7 -0
  92. package/src/pages/deployments.tsx +101 -0
  93. package/src/pages/dev.tsx +964 -0
  94. package/src/pages/functions/buildPlugin.ts +105 -0
  95. package/src/pages/functions/buildWorker.ts +151 -0
  96. package/{pages → src/pages}/functions/filepath-routing.test.ts +113 -113
  97. package/src/pages/functions/filepath-routing.ts +189 -0
  98. package/src/pages/functions/identifiers.ts +78 -0
  99. package/src/pages/functions/routes.ts +151 -0
  100. package/src/pages/index.tsx +84 -0
  101. package/src/pages/projects.tsx +157 -0
  102. package/src/pages/publish.tsx +335 -0
  103. package/src/pages/types.ts +40 -0
  104. package/src/pages/upload.tsx +384 -0
  105. package/src/pages/utils.ts +12 -0
  106. package/src/parse.ts +202 -138
  107. package/src/paths.ts +6 -6
  108. package/src/preview.ts +31 -0
  109. package/src/proxy.ts +400 -402
  110. package/src/publish.ts +667 -621
  111. package/src/pubsub/index.ts +286 -0
  112. package/src/pubsub/pubsub-commands.tsx +577 -0
  113. package/src/r2.ts +19 -19
  114. package/src/selfsigned.d.ts +23 -23
  115. package/src/sites.tsx +271 -225
  116. package/src/tail/filters.ts +108 -108
  117. package/src/tail/index.ts +217 -217
  118. package/src/tail/printing.ts +45 -45
  119. package/src/update-check.ts +11 -11
  120. package/src/user/choose-account.tsx +60 -0
  121. package/src/user/env-vars.ts +46 -0
  122. package/src/user/generate-auth-url.ts +33 -0
  123. package/src/user/generate-random-state.ts +16 -0
  124. package/src/user/index.ts +3 -0
  125. package/src/user/user.tsx +1161 -0
  126. package/src/whoami.tsx +61 -42
  127. package/src/worker-namespace.ts +190 -0
  128. package/src/worker.ts +110 -100
  129. package/src/zones.ts +39 -36
  130. package/templates/checked-fetch.js +17 -0
  131. package/templates/new-worker-scheduled.js +3 -3
  132. package/templates/new-worker-scheduled.ts +15 -15
  133. package/templates/new-worker.js +3 -3
  134. package/templates/new-worker.ts +15 -15
  135. package/templates/no-op-worker.js +10 -0
  136. package/templates/pages-template-plugin.ts +155 -0
  137. package/templates/pages-template-worker.ts +161 -0
  138. package/templates/static-asset-facade.js +31 -31
  139. package/templates/tsconfig.json +95 -95
  140. package/wrangler-dist/cli.js +55383 -54138
  141. package/pages/functions/buildPlugin.ts +0 -105
  142. package/pages/functions/buildWorker.ts +0 -151
  143. package/pages/functions/filepath-routing.ts +0 -189
  144. package/pages/functions/identifiers.ts +0 -78
  145. package/pages/functions/routes.ts +0 -156
  146. package/pages/functions/template-plugin.ts +0 -147
  147. package/pages/functions/template-worker.ts +0 -143
  148. package/src/pages.tsx +0 -2093
  149. package/src/user.tsx +0 -1214
package/src/inspect.ts CHANGED
@@ -38,416 +38,416 @@ import type { MessageEvent } from "ws";
38
38
  */
39
39
 
40
40
  interface InspectorProps {
41
- /**
42
- * The port that the local proxy server should listen on.
43
- */
44
- port: number;
45
- /**
46
- * The websocket URL exposed by Workers that the inspector should connect to.
47
- */
48
- inspectorUrl: string | undefined;
49
- /**
50
- * Whether console statements and exceptions should be logged to the terminal.
51
- * (We don't log them in local mode because they're already getting
52
- * logged to the terminal by nature of them actually running in node locally.)
53
- */
54
- logToTerminal: boolean;
41
+ /**
42
+ * The port that the local proxy server should listen on.
43
+ */
44
+ port: number;
45
+ /**
46
+ * The websocket URL exposed by Workers that the inspector should connect to.
47
+ */
48
+ inspectorUrl: string | undefined;
49
+ /**
50
+ * Whether console statements and exceptions should be logged to the terminal.
51
+ * (We don't log them in local mode because they're already getting
52
+ * logged to the terminal by nature of them actually running in node locally.)
53
+ */
54
+ logToTerminal: boolean;
55
55
  }
56
56
 
57
57
  export default function useInspector(props: InspectorProps) {
58
- /** A unique ID for this session. */
59
- const inspectorIdRef = useRef(randomId());
60
-
61
- /** The websocket from the devtools instance. */
62
- const [localWebSocket, setLocalWebSocket] = useState<WebSocket>();
63
- /** The websocket from the edge */
64
- const [remoteWebSocket, setRemoteWebSocket] = useState<WebSocket>();
65
-
66
- /**
67
- * The local proxy server that acts as the bridge between
68
- * the remote websocket and the local DevTools instance.
69
- */
70
- const serverRef = useRef<Server>();
71
- if (serverRef.current === undefined) {
72
- serverRef.current = createServer(
73
- (req: IncomingMessage, res: ServerResponse) => {
74
- switch (req.url) {
75
- // We implement a couple of well known end points
76
- // that are queried for metadata by chrome://inspect
77
- case "/json/version":
78
- res.setHeader("Content-Type", "application/json");
79
- res.end(
80
- JSON.stringify({
81
- Browser: `wrangler/v${version}`,
82
- // TODO: (someday): The DevTools protocol should match that of Edge Worker.
83
- // This could be exposed by the preview API.
84
- "Protocol-Version": "1.3",
85
- })
86
- );
87
- return;
88
- case "/json":
89
- case "/json/list":
90
- {
91
- res.setHeader("Content-Type", "application/json");
92
- const localHost = `localhost:${props.port}/ws`;
93
- const devtoolsFrontendUrl = `devtools://devtools/bundled/js_app.html?experiments=true&v8only=true&ws=${localHost}`;
94
- const devtoolsFrontendUrlCompat = `devtools://devtools/bundled/inspector.html?experiments=true&v8only=true&ws=${localHost}`;
95
- res.end(
96
- JSON.stringify([
97
- {
98
- id: inspectorIdRef.current,
99
- type: "node",
100
- description: "workers",
101
- webSocketDebuggerUrl: `ws://${localHost}`,
102
- devtoolsFrontendUrl,
103
- devtoolsFrontendUrlCompat,
104
- // Below are fields that are visible in the DevTools UI.
105
- title: "Cloudflare Worker",
106
- faviconUrl: "https://workers.cloudflare.com/favicon.ico",
107
- url:
108
- "https://" +
109
- (remoteWebSocket
110
- ? new URL(remoteWebSocket.url).host
111
- : "workers.dev"),
112
- },
113
- ])
114
- );
115
- }
116
- return;
117
- default:
118
- break;
119
- }
120
- }
121
- );
122
- }
123
- const server = serverRef.current;
124
-
125
- /**
126
- * The websocket server that runs on top of the proxy server.
127
- */
128
- const wsServerRef = useRef<WebSocketServer>();
129
- if (wsServerRef.current === undefined) {
130
- wsServerRef.current = new WebSocketServer({
131
- server,
132
- clientTracking: true,
133
- });
134
- }
135
- const wsServer = wsServerRef.current;
136
-
137
- wsServer.on("connection", (ws: WebSocket) => {
138
- if (wsServer.clients.size > 1) {
139
- /** We only want to have one active Devtools instance at a time. */
140
- logger.error(
141
- "Tried to open a new devtools window when a previous one was already open."
142
- );
143
- ws.close(1013, "Too many clients; only one can be connected at a time");
144
- } else {
145
- // As promised, save the created websocket in a state hook
146
- setLocalWebSocket(ws);
147
-
148
- ws.addEventListener("close", () => {
149
- // And and cleanup when devtools closes
150
- setLocalWebSocket(undefined);
151
- });
152
- }
153
- });
154
-
155
- /**
156
- * We start and stop the server in an effect to take advantage
157
- * of the component lifecycle. Convenient.
158
- */
159
- useEffect(() => {
160
- const abortController = new AbortController();
161
- async function startInspectorProxy() {
162
- await waitForPortToBeAvailable(props.port, {
163
- retryPeriod: 200,
164
- timeout: 2000,
165
- abortSignal: abortController.signal,
166
- });
167
- server.listen(props.port);
168
- }
169
- startInspectorProxy().catch((err) => {
170
- if ((err as { code: string }).code !== "ABORT_ERR") {
171
- logger.error("Failed to start inspector:", err);
172
- }
173
- });
174
- return () => {
175
- server.close();
176
- // Also disconnect any open websockets/devtools connections
177
- wsServer.clients.forEach((ws) => ws.close());
178
- wsServer.close();
179
- abortController.abort();
180
- };
181
- }, [props.port, server, wsServer]);
182
-
183
- /**
184
- * When connecting to the remote websocket, if we don't start either
185
- * the devtools instance or make an actual request to the worker in time,
186
- * then the connecting process can error out. When this happens, we
187
- * want to simply retry the connection. We use a state hook to trigger retries
188
- * of the effect that connects to the remote websocket.
189
- */
190
- const [
191
- retryRemoteWebSocketConnectionSigil,
192
- setRetryRemoteWebSocketConnectionSigil,
193
- ] = useState<number>(0);
194
- function retryRemoteWebSocketConnection() {
195
- setRetryRemoteWebSocketConnectionSigil((x) => x + 1);
196
- }
197
-
198
- /** A simple incrementing id to attach to messages we send to devtools */
199
- const messageCounterRef = useRef(1);
200
-
201
- // This effect tracks the connection to the remote websocket
202
- // (stored in, no surprises here, `remoteWebSocket`)
203
- useEffect(() => {
204
- if (!props.inspectorUrl) {
205
- return;
206
- }
207
- // The actual websocket instance
208
- const ws = new WebSocket(props.inspectorUrl);
209
- setRemoteWebSocket(ws);
210
-
211
- /**
212
- * A handle to the interval we run to keep the websocket alive
213
- */
214
- let keepAliveInterval: NodeJS.Timer;
215
-
216
- /**
217
- * Test if the websocket is closed
218
- */
219
- function isClosed() {
220
- return (
221
- ws.readyState === WebSocket.CLOSED ||
222
- ws.readyState === WebSocket.CLOSING
223
- );
224
- }
225
-
226
- /**
227
- * Send a message to the remote websocket
228
- */
229
- function send(event: Record<string, unknown>): void {
230
- if (!isClosed()) {
231
- ws.send(JSON.stringify(event));
232
- }
233
- }
234
-
235
- /**
236
- * Closes the inspector.
237
- */
238
- function close(): void {
239
- if (!isClosed()) {
240
- try {
241
- ws.close();
242
- } catch (err) {
243
- // Closing before the websocket is ready will throw an error.
244
- }
245
- }
246
- }
247
-
248
- /**
249
- * Since we have a handle on the remote websocket, we can tap
250
- * into its events, and log any pertinent ones directly to
251
- * the terminal (which means you have insight into your worker
252
- * without having to open the devtools).
253
- */
254
- if (props.logToTerminal) {
255
- ws.addEventListener("message", (event: MessageEvent) => {
256
- if (typeof event.data === "string") {
257
- const evt = JSON.parse(event.data);
258
- if (evt.method === "Runtime.exceptionThrown") {
259
- const params = evt.params as Protocol.Runtime.ExceptionThrownEvent;
260
- logger.error(
261
- params.exceptionDetails.text,
262
- params.exceptionDetails.exception?.description ?? ""
263
- );
264
- }
265
- if (evt.method === "Runtime.consoleAPICalled") {
266
- logConsoleMessage(
267
- evt.params as Protocol.Runtime.ConsoleAPICalledEvent
268
- );
269
- }
270
- } else {
271
- // We should never get here, but who know is 2022...
272
- logger.error("Unrecognised devtools event:", event);
273
- }
274
- });
275
- }
276
-
277
- ws.addEventListener("open", () => {
278
- send({ method: "Runtime.enable", id: messageCounterRef.current });
279
- // TODO: This doesn't actually work. Must fix.
280
- send({ method: "Network.enable", id: messageCounterRef.current++ });
281
-
282
- keepAliveInterval = setInterval(() => {
283
- send({
284
- method: "Runtime.getIsolateId",
285
- id: messageCounterRef.current++,
286
- });
287
- }, 10_000);
288
- });
289
-
290
- ws.on("unexpected-response", () => {
291
- logger.log("Waiting for connection...");
292
- /**
293
- * This usually means the worker is not "ready" yet
294
- * so we'll just retry the connection process
295
- */
296
- retryRemoteWebSocketConnection();
297
- });
298
-
299
- ws.addEventListener("close", () => {
300
- clearInterval(keepAliveInterval);
301
- });
302
-
303
- return () => {
304
- // clean up! Let's first stop the heartbeat interval
305
- clearInterval(keepAliveInterval);
306
- // Then we'll send a message to the devtools instance to
307
- // tell it to clear the console.
308
- wsServer.clients.forEach((client) => {
309
- // We could've used `localSocket` here, but
310
- // then we would have had to add it to the effect
311
- // change detection array, which would have made a
312
- // bunch of other stuff complicated. So we'll just
313
- // cycle through all of the server's connected clients
314
- // (in practice, there should only be one or zero) and send
315
- // the Log.clear message.
316
- client.send(
317
- JSON.stringify({
318
- // TODO: This doesn't actually work. Must fix.
319
- method: "Log.clear",
320
- // we can disable the next eslint warning since
321
- // we're referencing a ref that stays alive
322
- // eslint-disable-next-line react-hooks/exhaustive-deps
323
- id: messageCounterRef.current++,
324
- params: {},
325
- })
326
- );
327
- });
328
- // Finally, we'll close the websocket
329
- close();
330
- // And we'll clear `remoteWebsocket`
331
- setRemoteWebSocket(undefined);
332
- };
333
- }, [
334
- props.inspectorUrl,
335
- props.logToTerminal,
336
- wsServer,
337
- // We use a state value as a sigil to trigger a retry of the
338
- // remote websocket connection. It's not used inside the effect,
339
- // so react-hooks/exhaustive-deps doesn't complain if it's not
340
- // included in the dependency array. But its presence is critical,
341
- // so do NOT remove it from the dependency list.
342
- retryRemoteWebSocketConnectionSigil,
343
- ]);
344
-
345
- /**
346
- * We want to make sure we don't lose any messages we receive from the
347
- * remote websocket before devtools connects. So we use a ref to buffer
348
- * messages, and flush them whenever devtools connects.
349
- */
350
- const messageBufferRef = useRef<MessageEvent[]>([]);
351
-
352
- // This effect tracks the state changes _between_ the local
353
- // and remote websockets, and handles how messages flow between them.
354
- useEffect(() => {
355
- /**
356
- * This event listener is used for buffering messages from
357
- * the remote websocket, and flushing them
358
- * when the local websocket connects.
359
- */
360
- function bufferMessageFromRemoteSocket(event: MessageEvent) {
361
- messageBufferRef.current.push(event);
362
- // TODO: maybe we should have a max limit on this?
363
- // if so, we should be careful when removing messages
364
- // from the front, because they could be critical for
365
- // devtools (like execution context creation, etc)
366
- }
367
-
368
- if (remoteWebSocket && !localWebSocket) {
369
- // The local websocket hasn't connected yet, so we'll
370
- // buffer messages until it does.
371
- remoteWebSocket.addEventListener(
372
- "message",
373
- bufferMessageFromRemoteSocket
374
- );
375
- }
376
-
377
- /** Send a message from the local websocket to the remote websocket */
378
- function sendMessageToRemoteWebSocket(event: MessageEvent) {
379
- try {
380
- assert(
381
- remoteWebSocket,
382
- "Trying to send a message to an undefined `remoteWebSocket`"
383
- );
384
- remoteWebSocket.send(event.data);
385
- } catch (e) {
386
- if (
387
- (e as Error).message !==
388
- "WebSocket is not open: readyState 0 (CONNECTING)"
389
- ) {
390
- /**
391
- * ^ this just means we haven't opened a websocket yet
392
- * usually happens until there's at least one request
393
- * which is weird, because we may miss something that
394
- * happens on the first request. Maybe we should buffer
395
- * these messages too?
396
- */
397
- logger.error(e);
398
- }
399
- }
400
- }
401
-
402
- /** Send a message from the local websocket to the remote websocket */
403
- function sendMessageToLocalWebSocket(event: MessageEvent) {
404
- assert(
405
- localWebSocket,
406
- "Trying to send a message to an undefined `localWebSocket`"
407
- );
408
- localWebSocket.send(event.data);
409
- }
410
-
411
- if (localWebSocket && remoteWebSocket) {
412
- // Both the remote and local websockets are connected, so let's
413
- // start sending messages between them.
414
- localWebSocket.addEventListener("message", sendMessageToRemoteWebSocket);
415
- remoteWebSocket.addEventListener("message", sendMessageToLocalWebSocket);
416
-
417
- // Also, let's flush any buffered messages
418
- messageBufferRef.current.forEach(sendMessageToLocalWebSocket);
419
- messageBufferRef.current = [];
420
- }
421
-
422
- return () => {
423
- // Cleanup like good citizens
424
- if (remoteWebSocket) {
425
- remoteWebSocket.removeEventListener(
426
- "message",
427
- bufferMessageFromRemoteSocket
428
- );
429
- remoteWebSocket.removeEventListener(
430
- "message",
431
- sendMessageToLocalWebSocket
432
- );
433
- }
434
- if (localWebSocket) {
435
- localWebSocket.removeEventListener(
436
- "message",
437
- sendMessageToRemoteWebSocket
438
- );
439
- }
440
- };
441
- }, [localWebSocket, remoteWebSocket]);
58
+ /** A unique ID for this session. */
59
+ const inspectorIdRef = useRef(randomId());
60
+
61
+ /** The websocket from the devtools instance. */
62
+ const [localWebSocket, setLocalWebSocket] = useState<WebSocket>();
63
+ /** The websocket from the edge */
64
+ const [remoteWebSocket, setRemoteWebSocket] = useState<WebSocket>();
65
+
66
+ /**
67
+ * The local proxy server that acts as the bridge between
68
+ * the remote websocket and the local DevTools instance.
69
+ */
70
+ const serverRef = useRef<Server>();
71
+ if (serverRef.current === undefined) {
72
+ serverRef.current = createServer(
73
+ (req: IncomingMessage, res: ServerResponse) => {
74
+ switch (req.url) {
75
+ // We implement a couple of well known end points
76
+ // that are queried for metadata by chrome://inspect
77
+ case "/json/version":
78
+ res.setHeader("Content-Type", "application/json");
79
+ res.end(
80
+ JSON.stringify({
81
+ Browser: `wrangler/v${version}`,
82
+ // TODO: (someday): The DevTools protocol should match that of Edge Worker.
83
+ // This could be exposed by the preview API.
84
+ "Protocol-Version": "1.3",
85
+ })
86
+ );
87
+ return;
88
+ case "/json":
89
+ case "/json/list":
90
+ {
91
+ res.setHeader("Content-Type", "application/json");
92
+ const localHost = `localhost:${props.port}/ws`;
93
+ const devtoolsFrontendUrl = `devtools://devtools/bundled/js_app.html?experiments=true&v8only=true&ws=${localHost}`;
94
+ const devtoolsFrontendUrlCompat = `devtools://devtools/bundled/inspector.html?experiments=true&v8only=true&ws=${localHost}`;
95
+ res.end(
96
+ JSON.stringify([
97
+ {
98
+ id: inspectorIdRef.current,
99
+ type: "node",
100
+ description: "workers",
101
+ webSocketDebuggerUrl: `ws://${localHost}`,
102
+ devtoolsFrontendUrl,
103
+ devtoolsFrontendUrlCompat,
104
+ // Below are fields that are visible in the DevTools UI.
105
+ title: "Cloudflare Worker",
106
+ faviconUrl: "https://workers.cloudflare.com/favicon.ico",
107
+ url:
108
+ "https://" +
109
+ (remoteWebSocket
110
+ ? new URL(remoteWebSocket.url).host
111
+ : "workers.dev"),
112
+ },
113
+ ])
114
+ );
115
+ }
116
+ return;
117
+ default:
118
+ break;
119
+ }
120
+ }
121
+ );
122
+ }
123
+ const server = serverRef.current;
124
+
125
+ /**
126
+ * The websocket server that runs on top of the proxy server.
127
+ */
128
+ const wsServerRef = useRef<WebSocketServer>();
129
+ if (wsServerRef.current === undefined) {
130
+ wsServerRef.current = new WebSocketServer({
131
+ server,
132
+ clientTracking: true,
133
+ });
134
+ }
135
+ const wsServer = wsServerRef.current;
136
+
137
+ wsServer.on("connection", (ws: WebSocket) => {
138
+ if (wsServer.clients.size > 1) {
139
+ /** We only want to have one active Devtools instance at a time. */
140
+ logger.error(
141
+ "Tried to open a new devtools window when a previous one was already open."
142
+ );
143
+ ws.close(1013, "Too many clients; only one can be connected at a time");
144
+ } else {
145
+ // As promised, save the created websocket in a state hook
146
+ setLocalWebSocket(ws);
147
+
148
+ ws.addEventListener("close", () => {
149
+ // And and cleanup when devtools closes
150
+ setLocalWebSocket(undefined);
151
+ });
152
+ }
153
+ });
154
+
155
+ /**
156
+ * We start and stop the server in an effect to take advantage
157
+ * of the component lifecycle. Convenient.
158
+ */
159
+ useEffect(() => {
160
+ const abortController = new AbortController();
161
+ async function startInspectorProxy() {
162
+ await waitForPortToBeAvailable(props.port, {
163
+ retryPeriod: 200,
164
+ timeout: 2000,
165
+ abortSignal: abortController.signal,
166
+ });
167
+ server.listen(props.port);
168
+ }
169
+ startInspectorProxy().catch((err) => {
170
+ if ((err as { code: string }).code !== "ABORT_ERR") {
171
+ logger.error("Failed to start inspector:", err);
172
+ }
173
+ });
174
+ return () => {
175
+ server.close();
176
+ // Also disconnect any open websockets/devtools connections
177
+ wsServer.clients.forEach((ws) => ws.close());
178
+ wsServer.close();
179
+ abortController.abort();
180
+ };
181
+ }, [props.port, server, wsServer]);
182
+
183
+ /**
184
+ * When connecting to the remote websocket, if we don't start either
185
+ * the devtools instance or make an actual request to the worker in time,
186
+ * then the connecting process can error out. When this happens, we
187
+ * want to simply retry the connection. We use a state hook to trigger retries
188
+ * of the effect that connects to the remote websocket.
189
+ */
190
+ const [
191
+ retryRemoteWebSocketConnectionSigil,
192
+ setRetryRemoteWebSocketConnectionSigil,
193
+ ] = useState<number>(0);
194
+ function retryRemoteWebSocketConnection() {
195
+ setRetryRemoteWebSocketConnectionSigil((x) => x + 1);
196
+ }
197
+
198
+ /** A simple incrementing id to attach to messages we send to devtools */
199
+ const messageCounterRef = useRef(1);
200
+
201
+ // This effect tracks the connection to the remote websocket
202
+ // (stored in, no surprises here, `remoteWebSocket`)
203
+ useEffect(() => {
204
+ if (!props.inspectorUrl) {
205
+ return;
206
+ }
207
+ // The actual websocket instance
208
+ const ws = new WebSocket(props.inspectorUrl);
209
+ setRemoteWebSocket(ws);
210
+
211
+ /**
212
+ * A handle to the interval we run to keep the websocket alive
213
+ */
214
+ let keepAliveInterval: NodeJS.Timer;
215
+
216
+ /**
217
+ * Test if the websocket is closed
218
+ */
219
+ function isClosed() {
220
+ return (
221
+ ws.readyState === WebSocket.CLOSED ||
222
+ ws.readyState === WebSocket.CLOSING
223
+ );
224
+ }
225
+
226
+ /**
227
+ * Send a message to the remote websocket
228
+ */
229
+ function send(event: Record<string, unknown>): void {
230
+ if (!isClosed()) {
231
+ ws.send(JSON.stringify(event));
232
+ }
233
+ }
234
+
235
+ /**
236
+ * Closes the inspector.
237
+ */
238
+ function close(): void {
239
+ if (!isClosed()) {
240
+ try {
241
+ ws.close();
242
+ } catch (err) {
243
+ // Closing before the websocket is ready will throw an error.
244
+ }
245
+ }
246
+ }
247
+
248
+ /**
249
+ * Since we have a handle on the remote websocket, we can tap
250
+ * into its events, and log any pertinent ones directly to
251
+ * the terminal (which means you have insight into your worker
252
+ * without having to open the devtools).
253
+ */
254
+ if (props.logToTerminal) {
255
+ ws.addEventListener("message", (event: MessageEvent) => {
256
+ if (typeof event.data === "string") {
257
+ const evt = JSON.parse(event.data);
258
+ if (evt.method === "Runtime.exceptionThrown") {
259
+ const params = evt.params as Protocol.Runtime.ExceptionThrownEvent;
260
+ logger.error(
261
+ params.exceptionDetails.text,
262
+ params.exceptionDetails.exception?.description ?? ""
263
+ );
264
+ }
265
+ if (evt.method === "Runtime.consoleAPICalled") {
266
+ logConsoleMessage(
267
+ evt.params as Protocol.Runtime.ConsoleAPICalledEvent
268
+ );
269
+ }
270
+ } else {
271
+ // We should never get here, but who know is 2022...
272
+ logger.error("Unrecognised devtools event:", event);
273
+ }
274
+ });
275
+ }
276
+
277
+ ws.addEventListener("open", () => {
278
+ send({ method: "Runtime.enable", id: messageCounterRef.current });
279
+ // TODO: This doesn't actually work. Must fix.
280
+ send({ method: "Network.enable", id: messageCounterRef.current++ });
281
+
282
+ keepAliveInterval = setInterval(() => {
283
+ send({
284
+ method: "Runtime.getIsolateId",
285
+ id: messageCounterRef.current++,
286
+ });
287
+ }, 10_000);
288
+ });
289
+
290
+ ws.on("unexpected-response", () => {
291
+ logger.log("Waiting for connection...");
292
+ /**
293
+ * This usually means the worker is not "ready" yet
294
+ * so we'll just retry the connection process
295
+ */
296
+ retryRemoteWebSocketConnection();
297
+ });
298
+
299
+ ws.addEventListener("close", () => {
300
+ clearInterval(keepAliveInterval);
301
+ });
302
+
303
+ return () => {
304
+ // clean up! Let's first stop the heartbeat interval
305
+ clearInterval(keepAliveInterval);
306
+ // Then we'll send a message to the devtools instance to
307
+ // tell it to clear the console.
308
+ wsServer.clients.forEach((client) => {
309
+ // We could've used `localSocket` here, but
310
+ // then we would have had to add it to the effect
311
+ // change detection array, which would have made a
312
+ // bunch of other stuff complicated. So we'll just
313
+ // cycle through all of the server's connected clients
314
+ // (in practice, there should only be one or zero) and send
315
+ // the Log.clear message.
316
+ client.send(
317
+ JSON.stringify({
318
+ // TODO: This doesn't actually work. Must fix.
319
+ method: "Log.clear",
320
+ // we can disable the next eslint warning since
321
+ // we're referencing a ref that stays alive
322
+ // eslint-disable-next-line react-hooks/exhaustive-deps
323
+ id: messageCounterRef.current++,
324
+ params: {},
325
+ })
326
+ );
327
+ });
328
+ // Finally, we'll close the websocket
329
+ close();
330
+ // And we'll clear `remoteWebsocket`
331
+ setRemoteWebSocket(undefined);
332
+ };
333
+ }, [
334
+ props.inspectorUrl,
335
+ props.logToTerminal,
336
+ wsServer,
337
+ // We use a state value as a sigil to trigger a retry of the
338
+ // remote websocket connection. It's not used inside the effect,
339
+ // so react-hooks/exhaustive-deps doesn't complain if it's not
340
+ // included in the dependency array. But its presence is critical,
341
+ // so do NOT remove it from the dependency list.
342
+ retryRemoteWebSocketConnectionSigil,
343
+ ]);
344
+
345
+ /**
346
+ * We want to make sure we don't lose any messages we receive from the
347
+ * remote websocket before devtools connects. So we use a ref to buffer
348
+ * messages, and flush them whenever devtools connects.
349
+ */
350
+ const messageBufferRef = useRef<MessageEvent[]>([]);
351
+
352
+ // This effect tracks the state changes _between_ the local
353
+ // and remote websockets, and handles how messages flow between them.
354
+ useEffect(() => {
355
+ /**
356
+ * This event listener is used for buffering messages from
357
+ * the remote websocket, and flushing them
358
+ * when the local websocket connects.
359
+ */
360
+ function bufferMessageFromRemoteSocket(event: MessageEvent) {
361
+ messageBufferRef.current.push(event);
362
+ // TODO: maybe we should have a max limit on this?
363
+ // if so, we should be careful when removing messages
364
+ // from the front, because they could be critical for
365
+ // devtools (like execution context creation, etc)
366
+ }
367
+
368
+ if (remoteWebSocket && !localWebSocket) {
369
+ // The local websocket hasn't connected yet, so we'll
370
+ // buffer messages until it does.
371
+ remoteWebSocket.addEventListener(
372
+ "message",
373
+ bufferMessageFromRemoteSocket
374
+ );
375
+ }
376
+
377
+ /** Send a message from the local websocket to the remote websocket */
378
+ function sendMessageToRemoteWebSocket(event: MessageEvent) {
379
+ try {
380
+ assert(
381
+ remoteWebSocket,
382
+ "Trying to send a message to an undefined `remoteWebSocket`"
383
+ );
384
+ remoteWebSocket.send(event.data);
385
+ } catch (e) {
386
+ if (
387
+ (e as Error).message !==
388
+ "WebSocket is not open: readyState 0 (CONNECTING)"
389
+ ) {
390
+ /**
391
+ * ^ this just means we haven't opened a websocket yet
392
+ * usually happens until there's at least one request
393
+ * which is weird, because we may miss something that
394
+ * happens on the first request. Maybe we should buffer
395
+ * these messages too?
396
+ */
397
+ logger.error(e);
398
+ }
399
+ }
400
+ }
401
+
402
+ /** Send a message from the local websocket to the remote websocket */
403
+ function sendMessageToLocalWebSocket(event: MessageEvent) {
404
+ assert(
405
+ localWebSocket,
406
+ "Trying to send a message to an undefined `localWebSocket`"
407
+ );
408
+ localWebSocket.send(event.data);
409
+ }
410
+
411
+ if (localWebSocket && remoteWebSocket) {
412
+ // Both the remote and local websockets are connected, so let's
413
+ // start sending messages between them.
414
+ localWebSocket.addEventListener("message", sendMessageToRemoteWebSocket);
415
+ remoteWebSocket.addEventListener("message", sendMessageToLocalWebSocket);
416
+
417
+ // Also, let's flush any buffered messages
418
+ messageBufferRef.current.forEach(sendMessageToLocalWebSocket);
419
+ messageBufferRef.current = [];
420
+ }
421
+
422
+ return () => {
423
+ // Cleanup like good citizens
424
+ if (remoteWebSocket) {
425
+ remoteWebSocket.removeEventListener(
426
+ "message",
427
+ bufferMessageFromRemoteSocket
428
+ );
429
+ remoteWebSocket.removeEventListener(
430
+ "message",
431
+ sendMessageToLocalWebSocket
432
+ );
433
+ }
434
+ if (localWebSocket) {
435
+ localWebSocket.removeEventListener(
436
+ "message",
437
+ sendMessageToRemoteWebSocket
438
+ );
439
+ }
440
+ };
441
+ }, [localWebSocket, remoteWebSocket]);
442
442
  }
443
443
 
444
444
  // Credit: https://stackoverflow.com/a/2117523
445
445
  function randomId(): string {
446
- return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, function (c) {
447
- const r = (Math.random() * 16) | 0,
448
- v = c == "x" ? r : (r & 0x3) | 0x8;
449
- return v.toString(16);
450
- });
446
+ return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, function (c) {
447
+ const r = (Math.random() * 16) | 0,
448
+ v = c == "x" ? r : (r & 0x3) | 0x8;
449
+ return v.toString(16);
450
+ });
451
451
  }
452
452
 
453
453
  /**
@@ -459,204 +459,204 @@ function randomId(): string {
459
459
  */
460
460
 
461
461
  export const mapConsoleAPIMessageTypeToConsoleMethod: {
462
- [key in Protocol.Runtime.ConsoleAPICalledEvent["type"]]: Exclude<
463
- keyof Console,
464
- "Console"
465
- >;
462
+ [key in Protocol.Runtime.ConsoleAPICalledEvent["type"]]: Exclude<
463
+ keyof Console,
464
+ "Console"
465
+ >;
466
466
  } = {
467
- log: "log",
468
- debug: "debug",
469
- info: "info",
470
- warning: "warn",
471
- error: "error",
472
- dir: "dir",
473
- dirxml: "dirxml",
474
- table: "table",
475
- trace: "trace",
476
- clear: "clear",
477
- count: "count",
478
- assert: "assert",
479
- profile: "profile",
480
- profileEnd: "profileEnd",
481
- timeEnd: "timeEnd",
482
- startGroup: "group",
483
- startGroupCollapsed: "groupCollapsed",
484
- endGroup: "groupEnd",
467
+ log: "log",
468
+ debug: "debug",
469
+ info: "info",
470
+ warning: "warn",
471
+ error: "error",
472
+ dir: "dir",
473
+ dirxml: "dirxml",
474
+ table: "table",
475
+ trace: "trace",
476
+ clear: "clear",
477
+ count: "count",
478
+ assert: "assert",
479
+ profile: "profile",
480
+ profileEnd: "profileEnd",
481
+ timeEnd: "timeEnd",
482
+ startGroup: "group",
483
+ startGroupCollapsed: "groupCollapsed",
484
+ endGroup: "groupEnd",
485
485
  };
486
486
 
487
487
  function logConsoleMessage(evt: Protocol.Runtime.ConsoleAPICalledEvent): void {
488
- const args: string[] = [];
489
- for (const ro of evt.args) {
490
- switch (ro.type) {
491
- case "string":
492
- case "number":
493
- case "boolean":
494
- case "undefined":
495
- case "symbol":
496
- case "bigint":
497
- args.push(ro.value);
498
- break;
499
- case "function":
500
- args.push(`[Function: ${ro.description ?? "<no-description>"}]`);
501
- break;
502
- case "object":
503
- if (!ro.preview) {
504
- args.push(
505
- ro.subtype === "null"
506
- ? "null"
507
- : ro.description ?? "<no-description>"
508
- );
509
- } else {
510
- args.push(ro.preview.description ?? "<no-description>");
511
-
512
- switch (ro.preview.subtype) {
513
- case "array":
514
- args.push(
515
- "[ " +
516
- ro.preview.properties
517
- .map(({ value }) => {
518
- return value;
519
- })
520
- .join(", ") +
521
- (ro.preview.overflow ? "..." : "") +
522
- " ]"
523
- );
524
-
525
- break;
526
- case "weakmap":
527
- case "map":
528
- args.push(
529
- "{\n" +
530
- // Maps always have entries
531
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
532
- ro.preview
533
- .entries!.map(({ key, value }) => {
534
- return ` ${key?.description ?? "<unknown>"} => ${
535
- value.description
536
- }`;
537
- })
538
- .join(",\n") +
539
- (ro.preview.overflow ? "\n ..." : "") +
540
- "\n}"
541
- );
542
-
543
- break;
544
- case "weakset":
545
- case "set":
546
- args.push(
547
- "{ " +
548
- // Sets always have entries
549
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
550
- ro.preview
551
- .entries!.map(({ value }) => {
552
- return `${value.description}`;
553
- })
554
- .join(", ") +
555
- (ro.preview.overflow ? ", ..." : "") +
556
- " }"
557
- );
558
- break;
559
- case "regexp":
560
- break;
561
- case "date":
562
- break;
563
- case "generator":
564
- args.push(ro.preview.properties[0].value || "");
565
- break;
566
- case "promise":
567
- if (ro.preview.properties[0].value === "pending") {
568
- args.push(`{<${ro.preview.properties[0].value}>}`);
569
- } else {
570
- args.push(
571
- `{<${ro.preview.properties[0].value}>: ${ro.preview.properties[1].value}}`
572
- );
573
- }
574
- break;
575
- case "node":
576
- case "iterator":
577
- case "proxy":
578
- case "typedarray":
579
- case "arraybuffer":
580
- case "dataview":
581
- case "webassemblymemory":
582
- case "wasmvalue":
583
- break;
584
- case "error":
585
- default:
586
- // just a pojo
587
- args.push(
588
- "{\n" +
589
- ro.preview.properties
590
- .map(({ name, value }) => {
591
- return ` ${name}: ${value}`;
592
- })
593
- .join(",\n") +
594
- (ro.preview.overflow ? "\n ..." : "") +
595
- "\n}"
596
- );
597
- }
598
- }
599
- break;
600
- default:
601
- args.push(ro.description || ro.unserializableValue || "🦋");
602
- break;
603
- }
604
- }
605
-
606
- const method = mapConsoleAPIMessageTypeToConsoleMethod[evt.type];
607
-
608
- if (method in console) {
609
- switch (method) {
610
- case "dir":
611
- console.dir(args);
612
- break;
613
- case "table":
614
- console.table(args);
615
- break;
616
- default:
617
- // eslint-disable-next-line prefer-spread
618
- console[method].apply(console, args);
619
- break;
620
- }
621
- } else {
622
- logger.warn(`Unsupported console method: ${method}`);
623
- logger.warn("console event:", evt);
624
- }
488
+ const args: string[] = [];
489
+ for (const ro of evt.args) {
490
+ switch (ro.type) {
491
+ case "string":
492
+ case "number":
493
+ case "boolean":
494
+ case "undefined":
495
+ case "symbol":
496
+ case "bigint":
497
+ args.push(ro.value);
498
+ break;
499
+ case "function":
500
+ args.push(`[Function: ${ro.description ?? "<no-description>"}]`);
501
+ break;
502
+ case "object":
503
+ if (!ro.preview) {
504
+ args.push(
505
+ ro.subtype === "null"
506
+ ? "null"
507
+ : ro.description ?? "<no-description>"
508
+ );
509
+ } else {
510
+ args.push(ro.preview.description ?? "<no-description>");
511
+
512
+ switch (ro.preview.subtype) {
513
+ case "array":
514
+ args.push(
515
+ "[ " +
516
+ ro.preview.properties
517
+ .map(({ value }) => {
518
+ return value;
519
+ })
520
+ .join(", ") +
521
+ (ro.preview.overflow ? "..." : "") +
522
+ " ]"
523
+ );
524
+
525
+ break;
526
+ case "weakmap":
527
+ case "map":
528
+ args.push(
529
+ "{\n" +
530
+ // Maps always have entries
531
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
532
+ ro.preview
533
+ .entries!.map(({ key, value }) => {
534
+ return ` ${key?.description ?? "<unknown>"} => ${
535
+ value.description
536
+ }`;
537
+ })
538
+ .join(",\n") +
539
+ (ro.preview.overflow ? "\n ..." : "") +
540
+ "\n}"
541
+ );
542
+
543
+ break;
544
+ case "weakset":
545
+ case "set":
546
+ args.push(
547
+ "{ " +
548
+ // Sets always have entries
549
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
550
+ ro.preview
551
+ .entries!.map(({ value }) => {
552
+ return `${value.description}`;
553
+ })
554
+ .join(", ") +
555
+ (ro.preview.overflow ? ", ..." : "") +
556
+ " }"
557
+ );
558
+ break;
559
+ case "regexp":
560
+ break;
561
+ case "date":
562
+ break;
563
+ case "generator":
564
+ args.push(ro.preview.properties[0].value || "");
565
+ break;
566
+ case "promise":
567
+ if (ro.preview.properties[0].value === "pending") {
568
+ args.push(`{<${ro.preview.properties[0].value}>}`);
569
+ } else {
570
+ args.push(
571
+ `{<${ro.preview.properties[0].value}>: ${ro.preview.properties[1].value}}`
572
+ );
573
+ }
574
+ break;
575
+ case "node":
576
+ case "iterator":
577
+ case "proxy":
578
+ case "typedarray":
579
+ case "arraybuffer":
580
+ case "dataview":
581
+ case "webassemblymemory":
582
+ case "wasmvalue":
583
+ break;
584
+ case "error":
585
+ default:
586
+ // just a pojo
587
+ args.push(
588
+ "{\n" +
589
+ ro.preview.properties
590
+ .map(({ name, value }) => {
591
+ return ` ${name}: ${value}`;
592
+ })
593
+ .join(",\n") +
594
+ (ro.preview.overflow ? "\n ..." : "") +
595
+ "\n}"
596
+ );
597
+ }
598
+ }
599
+ break;
600
+ default:
601
+ args.push(ro.description || ro.unserializableValue || "🦋");
602
+ break;
603
+ }
604
+ }
605
+
606
+ const method = mapConsoleAPIMessageTypeToConsoleMethod[evt.type];
607
+
608
+ if (method in console) {
609
+ switch (method) {
610
+ case "dir":
611
+ console.dir(args);
612
+ break;
613
+ case "table":
614
+ console.table(args);
615
+ break;
616
+ default:
617
+ // eslint-disable-next-line prefer-spread
618
+ console[method].apply(console, args);
619
+ break;
620
+ }
621
+ } else {
622
+ logger.warn(`Unsupported console method: ${method}`);
623
+ logger.warn("console event:", evt);
624
+ }
625
625
  }
626
626
 
627
627
  /**
628
628
  * Opens the chrome debugger
629
629
  */
630
630
  export const openInspector = async (inspectorPort: number) => {
631
- const url = `https://built-devtools.pages.dev/js_app?experiments=true&v8only=true&ws=localhost:${inspectorPort}/ws`;
632
- const errorMessage =
633
- "Failed to open inspector.\nInspector depends on having a Chromium-based browser installed, maybe you need to install one?";
634
-
635
- // see: https://github.com/sindresorhus/open/issues/177#issue-610016699
636
- let braveBrowser: string;
637
- switch (os.platform()) {
638
- case "darwin":
639
- case "win32":
640
- braveBrowser = "Brave";
641
- break;
642
- default:
643
- braveBrowser = "brave";
644
- }
645
-
646
- const childProcess = await open(url, {
647
- app: [
648
- {
649
- name: open.apps.chrome,
650
- },
651
- {
652
- name: braveBrowser,
653
- },
654
- {
655
- name: open.apps.edge,
656
- },
657
- ],
658
- });
659
- childProcess.on("error", () => {
660
- logger.warn(errorMessage);
661
- });
631
+ const url = `https://built-devtools.pages.dev/js_app?experiments=true&v8only=true&ws=localhost:${inspectorPort}/ws`;
632
+ const errorMessage =
633
+ "Failed to open inspector.\nInspector depends on having a Chromium-based browser installed, maybe you need to install one?";
634
+
635
+ // see: https://github.com/sindresorhus/open/issues/177#issue-610016699
636
+ let braveBrowser: string;
637
+ switch (os.platform()) {
638
+ case "darwin":
639
+ case "win32":
640
+ braveBrowser = "Brave";
641
+ break;
642
+ default:
643
+ braveBrowser = "brave";
644
+ }
645
+
646
+ const childProcess = await open(url, {
647
+ app: [
648
+ {
649
+ name: open.apps.chrome,
650
+ },
651
+ {
652
+ name: braveBrowser,
653
+ },
654
+ {
655
+ name: open.apps.edge,
656
+ },
657
+ ],
658
+ });
659
+ childProcess.on("error", () => {
660
+ logger.warn(errorMessage);
661
+ });
662
662
  };