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.
- package/README.md +7 -1
- package/bin/wrangler.js +111 -57
- package/miniflare-dist/index.mjs +9 -2
- package/package.json +156 -154
- package/src/__tests__/config-cache-without-cache-dir.test.ts +38 -0
- package/src/__tests__/config-cache.test.ts +30 -24
- package/src/__tests__/configuration.test.ts +3935 -3476
- package/src/__tests__/dev.test.tsx +1128 -979
- package/src/__tests__/guess-worker-format.test.ts +68 -68
- package/src/__tests__/helpers/cmd-shim.d.ts +6 -6
- package/src/__tests__/helpers/faye-websocket.d.ts +4 -4
- package/src/__tests__/helpers/mock-account-id.ts +24 -24
- package/src/__tests__/helpers/mock-bin.ts +20 -20
- package/src/__tests__/helpers/mock-cfetch.ts +92 -92
- package/src/__tests__/helpers/mock-console.ts +49 -39
- package/src/__tests__/helpers/mock-dialogs.ts +94 -71
- package/src/__tests__/helpers/mock-http-server.ts +30 -30
- package/src/__tests__/helpers/mock-istty.ts +65 -18
- package/src/__tests__/helpers/mock-kv.ts +26 -26
- package/src/__tests__/helpers/mock-oauth-flow.ts +223 -228
- package/src/__tests__/helpers/mock-process.ts +39 -0
- package/src/__tests__/helpers/mock-stdin.ts +82 -77
- package/src/__tests__/helpers/mock-web-socket.ts +21 -21
- package/src/__tests__/helpers/run-in-tmp.ts +27 -27
- package/src/__tests__/helpers/run-wrangler.ts +8 -8
- package/src/__tests__/helpers/write-worker-source.ts +16 -16
- package/src/__tests__/helpers/write-wrangler-toml.ts +9 -9
- package/src/__tests__/https-options.test.ts +104 -104
- package/src/__tests__/index.test.ts +239 -234
- package/src/__tests__/init.test.ts +1605 -1250
- package/src/__tests__/jest.setup.ts +63 -33
- package/src/__tests__/kv.test.ts +1128 -1011
- package/src/__tests__/logger.test.ts +100 -74
- package/src/__tests__/package-manager.test.ts +303 -303
- package/src/__tests__/pages.test.ts +1152 -652
- package/src/__tests__/parse.test.ts +252 -252
- package/src/__tests__/publish.test.ts +6371 -5622
- package/src/__tests__/pubsub.test.ts +367 -0
- package/src/__tests__/r2.test.ts +133 -133
- package/src/__tests__/route.test.ts +18 -18
- package/src/__tests__/secret.test.ts +382 -377
- package/src/__tests__/tail.test.ts +530 -530
- package/src/__tests__/user.test.ts +123 -111
- package/src/__tests__/whoami.test.tsx +198 -117
- package/src/__tests__/worker-namespace.test.ts +327 -0
- package/src/abort.d.ts +1 -1
- package/src/api/dev.ts +49 -0
- package/src/api/index.ts +1 -0
- package/src/bundle-reporter.tsx +29 -0
- package/src/bundle.ts +157 -149
- package/src/cfetch/index.ts +80 -80
- package/src/cfetch/internal.ts +90 -83
- package/src/cli.ts +21 -7
- package/src/config/config.ts +204 -195
- package/src/config/diagnostics.ts +61 -61
- package/src/config/environment.ts +390 -357
- package/src/config/index.ts +206 -193
- package/src/config/validation-helpers.ts +366 -366
- package/src/config/validation.ts +1573 -1376
- package/src/config-cache.ts +79 -41
- package/src/create-worker-preview.ts +206 -136
- package/src/create-worker-upload-form.ts +247 -238
- package/src/dev/dev-vars.ts +13 -13
- package/src/dev/dev.tsx +329 -307
- package/src/dev/local.tsx +304 -275
- package/src/dev/remote.tsx +366 -224
- package/src/dev/use-esbuild.ts +126 -91
- package/src/dev.tsx +538 -0
- package/src/dialogs.tsx +97 -97
- package/src/durable.ts +87 -87
- package/src/entry.ts +234 -228
- package/src/environment-variables.ts +23 -23
- package/src/errors.ts +6 -6
- package/src/generate.ts +33 -0
- package/src/git-client.ts +42 -0
- package/src/https-options.ts +79 -79
- package/src/index.tsx +1775 -2763
- package/src/init.ts +549 -0
- package/src/inspect.ts +593 -593
- package/src/intl-polyfill.d.ts +123 -123
- package/src/is-interactive.ts +12 -0
- package/src/kv.ts +277 -277
- package/src/logger.ts +46 -39
- package/src/miniflare-cli/enum-keys.ts +8 -8
- package/src/miniflare-cli/index.ts +42 -31
- package/src/miniflare-cli/request-context.ts +18 -18
- package/src/module-collection.ts +212 -212
- package/src/open-in-browser.ts +4 -6
- package/src/package-manager.ts +123 -123
- package/src/pages/build.tsx +202 -0
- package/src/pages/constants.ts +7 -0
- package/src/pages/deployments.tsx +101 -0
- package/src/pages/dev.tsx +964 -0
- package/src/pages/functions/buildPlugin.ts +105 -0
- package/src/pages/functions/buildWorker.ts +151 -0
- package/{pages → src/pages}/functions/filepath-routing.test.ts +113 -113
- package/src/pages/functions/filepath-routing.ts +189 -0
- package/src/pages/functions/identifiers.ts +78 -0
- package/src/pages/functions/routes.ts +151 -0
- package/src/pages/index.tsx +84 -0
- package/src/pages/projects.tsx +157 -0
- package/src/pages/publish.tsx +335 -0
- package/src/pages/types.ts +40 -0
- package/src/pages/upload.tsx +384 -0
- package/src/pages/utils.ts +12 -0
- package/src/parse.ts +202 -138
- package/src/paths.ts +6 -6
- package/src/preview.ts +31 -0
- package/src/proxy.ts +400 -402
- package/src/publish.ts +667 -621
- package/src/pubsub/index.ts +286 -0
- package/src/pubsub/pubsub-commands.tsx +577 -0
- package/src/r2.ts +19 -19
- package/src/selfsigned.d.ts +23 -23
- package/src/sites.tsx +271 -225
- package/src/tail/filters.ts +108 -108
- package/src/tail/index.ts +217 -217
- package/src/tail/printing.ts +45 -45
- package/src/update-check.ts +11 -11
- package/src/user/choose-account.tsx +60 -0
- package/src/user/env-vars.ts +46 -0
- package/src/user/generate-auth-url.ts +33 -0
- package/src/user/generate-random-state.ts +16 -0
- package/src/user/index.ts +3 -0
- package/src/user/user.tsx +1161 -0
- package/src/whoami.tsx +61 -42
- package/src/worker-namespace.ts +190 -0
- package/src/worker.ts +110 -100
- package/src/zones.ts +39 -36
- package/templates/checked-fetch.js +17 -0
- package/templates/new-worker-scheduled.js +3 -3
- package/templates/new-worker-scheduled.ts +15 -15
- package/templates/new-worker.js +3 -3
- package/templates/new-worker.ts +15 -15
- package/templates/no-op-worker.js +10 -0
- package/templates/pages-template-plugin.ts +155 -0
- package/templates/pages-template-worker.ts +161 -0
- package/templates/static-asset-facade.js +31 -31
- package/templates/tsconfig.json +95 -95
- package/wrangler-dist/cli.js +55383 -54138
- package/pages/functions/buildPlugin.ts +0 -105
- package/pages/functions/buildWorker.ts +0 -151
- package/pages/functions/filepath-routing.ts +0 -189
- package/pages/functions/identifiers.ts +0 -78
- package/pages/functions/routes.ts +0 -156
- package/pages/functions/template-plugin.ts +0 -147
- package/pages/functions/template-worker.ts +0 -143
- package/src/pages.tsx +0 -2093
- 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
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
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
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
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
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
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
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
462
|
+
[key in Protocol.Runtime.ConsoleAPICalledEvent["type"]]: Exclude<
|
|
463
|
+
keyof Console,
|
|
464
|
+
"Console"
|
|
465
|
+
>;
|
|
466
466
|
} = {
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
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
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
591
|
-
|
|
592
|
-
|
|
593
|
-
|
|
594
|
-
|
|
595
|
-
|
|
596
|
-
|
|
597
|
-
|
|
598
|
-
|
|
599
|
-
|
|
600
|
-
|
|
601
|
-
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
609
|
-
|
|
610
|
-
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
|
|
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
|
-
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
|
|
637
|
-
|
|
638
|
-
|
|
639
|
-
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
654
|
-
|
|
655
|
-
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
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
|
};
|