@depup/wrangler 4.75.0-depup.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 (44) hide show
  1. package/README.md +33 -0
  2. package/bin/wrangler.js +93 -0
  3. package/changes.json +18 -0
  4. package/config-schema.json +3222 -0
  5. package/kv-asset-handler.js +1 -0
  6. package/package.json +221 -0
  7. package/templates/__tests__/pages-dev-util.test.ts +128 -0
  8. package/templates/__tests__/tsconfig-sanity.ts +12 -0
  9. package/templates/__tests__/tsconfig.json +8 -0
  10. package/templates/checked-fetch.js +28 -0
  11. package/templates/facade.d.ts +19 -0
  12. package/templates/middleware/common.ts +67 -0
  13. package/templates/middleware/loader-modules.ts +134 -0
  14. package/templates/middleware/loader-sw.ts +229 -0
  15. package/templates/middleware/middleware-ensure-req-body-drained.ts +18 -0
  16. package/templates/middleware/middleware-miniflare3-json-error.ts +32 -0
  17. package/templates/middleware/middleware-patch-console-prefix.d.ts +3 -0
  18. package/templates/middleware/middleware-patch-console-prefix.ts +21 -0
  19. package/templates/middleware/middleware-pretty-error.ts +40 -0
  20. package/templates/middleware/middleware-scheduled.ts +29 -0
  21. package/templates/modules-watch-stub.js +4 -0
  22. package/templates/new-worker-scheduled.js +17 -0
  23. package/templates/new-worker-scheduled.ts +32 -0
  24. package/templates/new-worker.js +15 -0
  25. package/templates/new-worker.ts +33 -0
  26. package/templates/no-op-worker.js +10 -0
  27. package/templates/pages-dev-pipeline.ts +33 -0
  28. package/templates/pages-dev-util.ts +55 -0
  29. package/templates/pages-shim.ts +9 -0
  30. package/templates/pages-template-plugin.ts +190 -0
  31. package/templates/pages-template-worker.ts +198 -0
  32. package/templates/remoteBindings/ProxyServerWorker.ts +143 -0
  33. package/templates/remoteBindings/wrangler.jsonc +4 -0
  34. package/templates/startDevWorker/InspectorProxyWorker.ts +699 -0
  35. package/templates/startDevWorker/ProxyWorker.ts +340 -0
  36. package/templates/tsconfig-sanity.ts +11 -0
  37. package/templates/tsconfig.init.json +22 -0
  38. package/templates/tsconfig.json +14 -0
  39. package/wrangler-dist/InspectorProxyWorker.js +486 -0
  40. package/wrangler-dist/ProxyServerWorker.js +3314 -0
  41. package/wrangler-dist/ProxyWorker.js +238 -0
  42. package/wrangler-dist/cli.d.ts +3154 -0
  43. package/wrangler-dist/cli.js +303399 -0
  44. package/wrangler-dist/metafile-cjs.json +1 -0
@@ -0,0 +1,486 @@
1
+ // templates/startDevWorker/InspectorProxyWorker.ts
2
+ import assert2 from "node:assert";
3
+
4
+ // src/api/startDevWorker/events.ts
5
+ function serialiseError(e) {
6
+ if (e instanceof Error) {
7
+ return {
8
+ message: e.message,
9
+ name: e.name,
10
+ stack: e.stack,
11
+ cause: e.cause && serialiseError(e.cause)
12
+ };
13
+ } else {
14
+ return { message: String(e) };
15
+ }
16
+ }
17
+
18
+ // src/api/startDevWorker/utils.ts
19
+ import assert from "node:assert";
20
+ function createDeferred(previousDeferred) {
21
+ let resolve, reject;
22
+ const newPromise = new Promise((_resolve, _reject) => {
23
+ resolve = _resolve;
24
+ reject = _reject;
25
+ });
26
+ assert(resolve);
27
+ assert(reject);
28
+ previousDeferred?.resolve(newPromise);
29
+ return {
30
+ promise: newPromise,
31
+ resolve,
32
+ reject
33
+ };
34
+ }
35
+ function urlFromParts(parts, base = "http://localhost") {
36
+ const url = new URL(base);
37
+ Object.assign(url, parts);
38
+ return url;
39
+ }
40
+
41
+ // src/utils/assert-never.ts
42
+ function assertNever(_value) {
43
+ }
44
+
45
+ // templates/startDevWorker/InspectorProxyWorker.ts
46
+ var ALLOWED_HOST_HOSTNAMES = ["127.0.0.1", "[::1]", "localhost"];
47
+ var ALLOWED_ORIGIN_HOSTNAMES = [
48
+ "devtools.devprod.cloudflare.dev",
49
+ // Workers + Assets (current deployment)
50
+ "cloudflare-devtools.devprod.workers.dev",
51
+ /^[a-z0-9]+-cloudflare-devtools\.devprod\.workers\.dev$/,
52
+ // Cloudflare Pages (legacy deployment)
53
+ "cloudflare-devtools.pages.dev",
54
+ /^[a-z0-9]+\.cloudflare-devtools\.pages\.dev$/,
55
+ "127.0.0.1",
56
+ "[::1]",
57
+ "localhost"
58
+ ];
59
+ var InspectorProxyWorker_default = {
60
+ fetch(req, env) {
61
+ const singleton = env.DURABLE_OBJECT.idFromName("");
62
+ const inspectorProxy = env.DURABLE_OBJECT.get(singleton);
63
+ return inspectorProxy.fetch(req);
64
+ }
65
+ };
66
+ function isDevToolsEvent(event, name) {
67
+ return typeof event === "object" && event !== null && "method" in event && event.method === name;
68
+ }
69
+ var InspectorProxyWorker = class {
70
+ constructor(_state, env) {
71
+ this.env = env;
72
+ }
73
+ websockets = {
74
+ runtimeDeferred: createDeferred()
75
+ };
76
+ proxyData;
77
+ runtimeMessageBuffer = [];
78
+ // Only allow a limited number of error-based reconnections, so as not to infinite loop
79
+ reconnectionsRemaining = 3;
80
+ async fetch(req) {
81
+ if (req.headers.get("Authorization") === this.env.PROXY_CONTROLLER_AUTH_SECRET) {
82
+ return this.handleProxyControllerRequest(req);
83
+ }
84
+ if (req.headers.get("Upgrade") === "websocket") {
85
+ return this.handleDevToolsWebSocketUpgradeRequest(req);
86
+ }
87
+ return this.handleDevToolsJsonRequest(req);
88
+ }
89
+ // ************************
90
+ // ** PROXY CONTROLLER **
91
+ // ************************
92
+ handleProxyControllerRequest(req) {
93
+ assert2(
94
+ req.headers.get("Upgrade") === "websocket",
95
+ "Expected proxy controller data request to be WebSocket upgrade"
96
+ );
97
+ const { 0: response, 1: proxyController } = new WebSocketPair();
98
+ proxyController.accept();
99
+ proxyController.addEventListener("close", (event) => {
100
+ this.sendDebugLog(
101
+ "PROXY CONTROLLER WEBSOCKET CLOSED",
102
+ event.code,
103
+ event.reason
104
+ );
105
+ if (this.websockets.proxyController === proxyController) {
106
+ this.websockets.proxyController = void 0;
107
+ }
108
+ });
109
+ proxyController.addEventListener("error", (event) => {
110
+ const error = serialiseError(event.error);
111
+ this.sendDebugLog("PROXY CONTROLLER WEBSOCKET ERROR", error);
112
+ if (this.websockets.proxyController === proxyController) {
113
+ this.websockets.proxyController = void 0;
114
+ }
115
+ });
116
+ proxyController.addEventListener(
117
+ "message",
118
+ this.handleProxyControllerIncomingMessage
119
+ );
120
+ this.websockets.proxyController = proxyController;
121
+ return new Response(null, {
122
+ status: 101,
123
+ webSocket: response
124
+ });
125
+ }
126
+ handleProxyControllerIncomingMessage = (event) => {
127
+ assert2(
128
+ typeof event.data === "string",
129
+ "Expected event.data from proxy controller to be string"
130
+ );
131
+ const message = JSON.parse(
132
+ event.data
133
+ );
134
+ this.sendDebugLog("handleProxyControllerIncomingMessage", event.data);
135
+ switch (message.type) {
136
+ case "reloadStart": {
137
+ this.sendRuntimeDiscardConsoleEntries();
138
+ break;
139
+ }
140
+ case "reloadComplete": {
141
+ this.proxyData = message.proxyData;
142
+ this.reconnectRuntimeWebSocket();
143
+ break;
144
+ }
145
+ default: {
146
+ assertNever(message);
147
+ }
148
+ }
149
+ };
150
+ sendProxyControllerMessage(message) {
151
+ message = typeof message === "string" ? message : JSON.stringify(message);
152
+ this.websockets.proxyController?.send(message);
153
+ }
154
+ async sendProxyControllerRequest(message) {
155
+ try {
156
+ const res = await this.env.PROXY_CONTROLLER.fetch("http://dummy", {
157
+ method: "POST",
158
+ body: JSON.stringify(message)
159
+ });
160
+ return res.ok ? await res.text() : void 0;
161
+ } catch (e) {
162
+ this.sendDebugLog(
163
+ "FAILED TO SEND PROXY CONTROLLER REQUEST",
164
+ serialiseError(e)
165
+ );
166
+ return void 0;
167
+ }
168
+ }
169
+ sendDebugLog = (...args) => {
170
+ this.sendProxyControllerRequest({ type: "debug-log", args });
171
+ };
172
+ // ***************
173
+ // ** RUNTIME **
174
+ // ***************
175
+ handleRuntimeIncomingMessage = (event) => {
176
+ assert2(typeof event.data === "string");
177
+ const msg = JSON.parse(event.data);
178
+ this.sendDebugLog("RUNTIME INCOMING MESSAGE", msg);
179
+ if (isDevToolsEvent(msg, "Runtime.exceptionThrown")) {
180
+ this.sendProxyControllerMessage(event.data);
181
+ }
182
+ if (this.proxyData?.proxyLogsToController && isDevToolsEvent(msg, "Runtime.consoleAPICalled")) {
183
+ this.sendProxyControllerMessage(event.data);
184
+ }
185
+ this.runtimeMessageBuffer.push(msg);
186
+ this.tryDrainRuntimeMessageBuffer();
187
+ };
188
+ handleRuntimeScriptParsed(msg) {
189
+ if (!this.websockets.devtoolsHasFileSystemAccess && msg.params.sourceMapURL !== void 0 && // Don't try to find a sourcemap for e.g. node-internal: scripts
190
+ msg.params.url.startsWith("file:")) {
191
+ const url = new URL(msg.params.sourceMapURL, msg.params.url);
192
+ if (url.protocol === "file:") {
193
+ msg.params.sourceMapURL = url.href.replace("file:", "wrangler-file:");
194
+ }
195
+ }
196
+ void this.sendDevToolsMessage(msg);
197
+ }
198
+ tryDrainRuntimeMessageBuffer = () => {
199
+ if (this.websockets.devtools === void 0) return;
200
+ for (const msg of this.runtimeMessageBuffer.splice(0)) {
201
+ if (isDevToolsEvent(msg, "Debugger.scriptParsed")) {
202
+ this.handleRuntimeScriptParsed(msg);
203
+ } else {
204
+ void this.sendDevToolsMessage(msg);
205
+ }
206
+ }
207
+ };
208
+ runtimeAbortController = new AbortController();
209
+ // will abort the in-flight websocket upgrade request to the remote runtime
210
+ runtimeKeepAliveInterval = null;
211
+ async reconnectRuntimeWebSocket() {
212
+ assert2(this.proxyData, "Expected this.proxyData to be defined");
213
+ this.sendDebugLog("reconnectRuntimeWebSocket");
214
+ this.websockets.runtime?.close();
215
+ this.websockets.runtime = void 0;
216
+ this.runtimeAbortController.abort();
217
+ this.runtimeAbortController = new AbortController();
218
+ this.websockets.runtimeDeferred = createDeferred(
219
+ this.websockets.runtimeDeferred
220
+ );
221
+ const runtimeWebSocketUrl = urlFromParts(
222
+ this.proxyData.userWorkerInspectorUrl
223
+ );
224
+ runtimeWebSocketUrl.protocol = this.proxyData.userWorkerUrl.protocol;
225
+ this.sendDebugLog("NEW RUNTIME WEBSOCKET", runtimeWebSocketUrl);
226
+ this.sendDevToolsMessage({
227
+ method: "Runtime.executionContextsCleared",
228
+ params: void 0
229
+ });
230
+ const upgrade = await fetch(runtimeWebSocketUrl, {
231
+ headers: {
232
+ ...this.proxyData.headers,
233
+ "User-Agent": `wrangler/${this.env.WRANGLER_VERSION}`,
234
+ Upgrade: "websocket"
235
+ },
236
+ signal: this.runtimeAbortController.signal
237
+ });
238
+ const runtime = upgrade.webSocket;
239
+ if (!runtime) {
240
+ const error = new Error(
241
+ `Failed to establish the WebSocket connection: expected server to reply with HTTP status code 101 (switching protocols), but received ${upgrade.status} instead.`
242
+ );
243
+ if (upgrade.status === 502 && this.reconnectionsRemaining >= 0) {
244
+ await scheduler.wait((3 - this.reconnectionsRemaining) * 1e3);
245
+ this.sendDebugLog(
246
+ "RECONNECTING RUNTIME WEBSOCKET after 502. Reconnections remaining:",
247
+ this.reconnectionsRemaining
248
+ );
249
+ return this.reconnectRuntimeWebSocket();
250
+ }
251
+ this.websockets.runtimeDeferred.reject(error);
252
+ this.sendProxyControllerRequest({
253
+ type: "runtime-websocket-error",
254
+ error: serialiseError(error)
255
+ });
256
+ return;
257
+ }
258
+ this.websockets.runtime = runtime;
259
+ runtime.addEventListener("message", this.handleRuntimeIncomingMessage);
260
+ runtime.addEventListener("close", (event) => {
261
+ this.sendDebugLog("RUNTIME WEBSOCKET CLOSED", event.code, event.reason);
262
+ clearInterval(this.runtimeKeepAliveInterval);
263
+ if (this.websockets.runtime === runtime) {
264
+ this.websockets.runtime = void 0;
265
+ }
266
+ });
267
+ runtime.addEventListener("error", (event) => {
268
+ const error = serialiseError(event.error);
269
+ this.sendDebugLog("RUNTIME WEBSOCKET ERROR", error);
270
+ clearInterval(this.runtimeKeepAliveInterval);
271
+ if (this.websockets.runtime === runtime) {
272
+ this.websockets.runtime = void 0;
273
+ }
274
+ this.sendProxyControllerRequest({
275
+ type: "runtime-websocket-error",
276
+ error
277
+ });
278
+ });
279
+ runtime.accept();
280
+ this.handleRuntimeWebSocketOpen(runtime);
281
+ }
282
+ #runtimeMessageCounter = 1e8;
283
+ nextCounter() {
284
+ return ++this.#runtimeMessageCounter;
285
+ }
286
+ handleRuntimeWebSocketOpen(runtime) {
287
+ this.sendDebugLog("RUNTIME WEBSOCKET OPENED");
288
+ this.reconnectionsRemaining = 3;
289
+ this.sendRuntimeMessage(
290
+ { method: "Runtime.enable", id: this.nextCounter() },
291
+ runtime
292
+ );
293
+ if (this.websockets.devtools !== void 0) {
294
+ this.sendRuntimeMessage(
295
+ { method: "Debugger.enable", id: this.nextCounter() },
296
+ runtime
297
+ );
298
+ }
299
+ this.sendRuntimeMessage(
300
+ { method: "Network.enable", id: this.nextCounter() },
301
+ runtime
302
+ );
303
+ clearInterval(this.runtimeKeepAliveInterval);
304
+ this.runtimeKeepAliveInterval = setInterval(() => {
305
+ this.sendRuntimeMessage(
306
+ { method: "Runtime.getIsolateId", id: this.nextCounter() },
307
+ runtime
308
+ );
309
+ }, 1e4);
310
+ this.websockets.runtimeDeferred.resolve(runtime);
311
+ }
312
+ sendRuntimeDiscardConsoleEntries() {
313
+ if (this.websockets.runtime) {
314
+ this.sendRuntimeMessage(
315
+ {
316
+ method: "Runtime.discardConsoleEntries",
317
+ id: this.nextCounter()
318
+ },
319
+ this.websockets.runtime
320
+ );
321
+ }
322
+ }
323
+ async sendRuntimeMessage(message, runtime = this.websockets.runtimeDeferred.promise) {
324
+ runtime = await runtime;
325
+ message = typeof message === "string" ? message : JSON.stringify(message);
326
+ this.sendDebugLog("SEND TO RUNTIME", message);
327
+ runtime.send(message);
328
+ }
329
+ // ****************
330
+ // ** DEVTOOLS **
331
+ // ****************
332
+ #inspectorId = crypto.randomUUID();
333
+ async handleDevToolsJsonRequest(req) {
334
+ const url = new URL(req.url);
335
+ if (url.pathname === "/json/version") {
336
+ return Response.json({
337
+ Browser: `wrangler/v${this.env.WRANGLER_VERSION}`,
338
+ // TODO: (someday): The DevTools protocol should match that of workerd.
339
+ // This could be exposed by the preview API.
340
+ "Protocol-Version": "1.3"
341
+ });
342
+ }
343
+ if (url.pathname === "/json" || url.pathname === "/json/list") {
344
+ const localHost = `${url.host}/ws`;
345
+ const devtoolsFrontendUrl = `https://devtools.devprod.cloudflare.dev/js_app?theme=systemPreferred&debugger=true&ws=${localHost}`;
346
+ return Response.json([
347
+ {
348
+ id: this.#inspectorId,
349
+ type: "node",
350
+ // TODO: can we specify different type?
351
+ description: "workers",
352
+ webSocketDebuggerUrl: `ws://${localHost}`,
353
+ devtoolsFrontendUrl,
354
+ devtoolsFrontendUrlCompat: devtoolsFrontendUrl,
355
+ // Below are fields that are visible in the DevTools UI.
356
+ title: "Cloudflare Worker",
357
+ faviconUrl: "https://workers.cloudflare.com/favicon.ico"
358
+ // url: "http://" + localHost, // looks unnecessary
359
+ }
360
+ ]);
361
+ }
362
+ return new Response(null, { status: 404 });
363
+ }
364
+ async handleDevToolsWebSocketUpgradeRequest(req) {
365
+ let hostHeader = req.headers.get("Host");
366
+ if (hostHeader == null) return new Response(null, { status: 400 });
367
+ try {
368
+ const host = new URL(`http://${hostHeader}`);
369
+ if (!ALLOWED_HOST_HOSTNAMES.includes(host.hostname)) {
370
+ return new Response("Disallowed `Host` header", { status: 401 });
371
+ }
372
+ } catch {
373
+ return new Response("Expected `Host` header", { status: 400 });
374
+ }
375
+ let originHeader = req.headers.get("Origin");
376
+ if (originHeader === null && !req.headers.has("User-Agent")) {
377
+ originHeader = "http://localhost";
378
+ }
379
+ if (originHeader === null) {
380
+ return new Response("Expected `Origin` header", { status: 400 });
381
+ }
382
+ try {
383
+ const origin = new URL(originHeader);
384
+ const allowed = ALLOWED_ORIGIN_HOSTNAMES.some((rule) => {
385
+ if (typeof rule === "string") return origin.hostname === rule;
386
+ else return rule.test(origin.hostname);
387
+ });
388
+ if (!allowed) {
389
+ return new Response("Disallowed `Origin` header", { status: 401 });
390
+ }
391
+ } catch {
392
+ return new Response("Expected `Origin` header", { status: 400 });
393
+ }
394
+ this.sendDebugLog("DEVTOOLS WEBSOCKET TRYING TO CONNECT");
395
+ await this.websockets.runtimeDeferred.promise;
396
+ this.sendDebugLog("DEVTOOLS WEBSOCKET CAN NOW CONNECT");
397
+ assert2(
398
+ req.headers.get("Upgrade") === "websocket",
399
+ "Expected DevTools connection to be WebSocket upgrade"
400
+ );
401
+ const { 0: response, 1: devtools } = new WebSocketPair();
402
+ devtools.accept();
403
+ if (this.websockets.devtools !== void 0) {
404
+ devtools.close(
405
+ 1013,
406
+ "Too many clients; only one can be connected at a time"
407
+ );
408
+ } else {
409
+ devtools.addEventListener("message", this.handleDevToolsIncomingMessage);
410
+ const disconnectDevtools = () => {
411
+ if (this.websockets.devtools === devtools) {
412
+ this.websockets.devtools = void 0;
413
+ if (this.websockets.runtime) {
414
+ this.sendRuntimeMessage({
415
+ id: this.nextCounter(),
416
+ method: "Debugger.disable"
417
+ });
418
+ }
419
+ }
420
+ };
421
+ devtools.addEventListener("close", (event) => {
422
+ this.sendDebugLog(
423
+ "DEVTOOLS WEBSOCKET CLOSED",
424
+ event.code,
425
+ event.reason
426
+ );
427
+ disconnectDevtools();
428
+ });
429
+ devtools.addEventListener("error", (event) => {
430
+ const error = serialiseError(event.error);
431
+ this.sendDebugLog("DEVTOOLS WEBSOCKET ERROR", error);
432
+ disconnectDevtools();
433
+ });
434
+ this.sendRuntimeMessage({
435
+ id: this.nextCounter(),
436
+ method: "Debugger.disable"
437
+ });
438
+ this.sendDebugLog("DEVTOOLS WEBSOCKET CONNECTED");
439
+ const userAgent = req.headers.get("User-Agent") ?? "";
440
+ const hasFileSystemAccess = !/mozilla/i.test(userAgent);
441
+ this.websockets.devtools = devtools;
442
+ this.websockets.devtoolsHasFileSystemAccess = hasFileSystemAccess;
443
+ this.tryDrainRuntimeMessageBuffer();
444
+ }
445
+ return new Response(null, { status: 101, webSocket: response });
446
+ }
447
+ handleDevToolsIncomingMessage = (event) => {
448
+ assert2(
449
+ typeof event.data === "string",
450
+ "Expected devtools incoming message to be of type string"
451
+ );
452
+ const message = JSON.parse(event.data);
453
+ this.sendDebugLog("DEVTOOLS INCOMING MESSAGE", message);
454
+ if (message.method === "Network.loadNetworkResource") {
455
+ return void this.handleDevToolsLoadNetworkResource(message);
456
+ }
457
+ this.sendRuntimeMessage(JSON.stringify(message));
458
+ };
459
+ async handleDevToolsLoadNetworkResource(message) {
460
+ const response = await this.sendProxyControllerRequest({
461
+ type: "load-network-resource",
462
+ url: message.params.url
463
+ });
464
+ if (response === void 0) {
465
+ this.sendDebugLog(
466
+ `ProxyController could not resolve Network.loadNetworkResource for "${message.params.url}"`
467
+ );
468
+ this.sendRuntimeMessage(JSON.stringify(message));
469
+ } else {
470
+ this.sendDevToolsMessage({
471
+ id: message.id,
472
+ // @ts-expect-error DevTools Protocol type does not match our patched devtools -- result.resource.text was added
473
+ result: { resource: { success: true, text: response } }
474
+ });
475
+ }
476
+ }
477
+ sendDevToolsMessage(message) {
478
+ message = typeof message === "string" ? message : JSON.stringify(message);
479
+ this.sendDebugLog("SEND TO DEVTOOLS", message);
480
+ this.websockets.devtools?.send(message);
481
+ }
482
+ };
483
+ export {
484
+ InspectorProxyWorker,
485
+ InspectorProxyWorker_default as default
486
+ };