wrangler 2.0.27 → 2.1.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 (65) hide show
  1. package/bin/wrangler.js +1 -1
  2. package/miniflare-dist/index.mjs +1141 -369
  3. package/package.json +6 -4
  4. package/src/__tests__/api-dev.test.ts +19 -0
  5. package/src/__tests__/configuration.test.ts +27 -27
  6. package/src/__tests__/dev.test.tsx +8 -6
  7. package/src/__tests__/helpers/hello-world-worker.js +5 -0
  8. package/src/__tests__/helpers/mock-cfetch.ts +4 -4
  9. package/src/__tests__/helpers/mock-console.ts +11 -2
  10. package/src/__tests__/helpers/mock-get-zone-from-host.ts +8 -0
  11. package/src/__tests__/helpers/mock-known-routes.ts +7 -0
  12. package/src/__tests__/index.test.ts +37 -37
  13. package/src/__tests__/init.test.ts +356 -5
  14. package/src/__tests__/jest.setup.ts +13 -0
  15. package/src/__tests__/middleware.test.ts +768 -0
  16. package/src/__tests__/pages.test.ts +829 -104
  17. package/src/__tests__/paths.test.ts +17 -0
  18. package/src/__tests__/publish.test.ts +512 -445
  19. package/src/__tests__/tail.test.ts +79 -72
  20. package/src/__tests__/test-old-node-version.js +3 -3
  21. package/src/__tests__/worker-namespace.test.ts +37 -35
  22. package/src/api/dev.ts +93 -28
  23. package/src/bundle.ts +239 -12
  24. package/src/cfetch/internal.ts +64 -3
  25. package/src/cli.ts +1 -1
  26. package/src/config/environment.ts +1 -1
  27. package/src/config/index.ts +4 -4
  28. package/src/config/validation.ts +3 -3
  29. package/src/create-worker-upload-form.ts +29 -26
  30. package/src/dev/dev.tsx +3 -1
  31. package/src/dev/local.tsx +319 -171
  32. package/src/dev/remote.tsx +16 -4
  33. package/src/dev/start-server.ts +416 -0
  34. package/src/dev/use-esbuild.ts +4 -0
  35. package/src/dev.tsx +340 -166
  36. package/src/dialogs.tsx +12 -0
  37. package/src/{worker-namespace.ts → dispatch-namespace.ts} +18 -18
  38. package/src/entry.ts +2 -1
  39. package/src/index.tsx +59 -12
  40. package/src/init.ts +291 -16
  41. package/src/metrics/send-event.ts +6 -5
  42. package/src/miniflare-cli/assets.ts +130 -476
  43. package/src/miniflare-cli/index.ts +39 -33
  44. package/src/pages/constants.ts +3 -0
  45. package/src/pages/dev.tsx +8 -3
  46. package/src/pages/functions/buildPlugin.ts +2 -1
  47. package/src/pages/functions/buildWorker.ts +2 -1
  48. package/src/pages/functions/routes-transformation.test.ts +12 -1
  49. package/src/pages/functions/routes-transformation.ts +7 -1
  50. package/src/pages/hash.tsx +13 -0
  51. package/src/pages/publish.tsx +82 -38
  52. package/src/pages/upload.tsx +3 -18
  53. package/src/paths.ts +20 -1
  54. package/src/publish.ts +49 -8
  55. package/src/tail/filters.ts +1 -5
  56. package/src/tail/index.ts +6 -3
  57. package/src/worker.ts +10 -9
  58. package/src/zones.ts +91 -0
  59. package/templates/middleware/common.ts +62 -0
  60. package/templates/middleware/loader-modules.ts +84 -0
  61. package/templates/middleware/loader-sw.ts +213 -0
  62. package/templates/middleware/middleware-pretty-error.ts +40 -0
  63. package/templates/middleware/middleware-scheduled.ts +14 -0
  64. package/wrangler-dist/cli.d.ts +22 -8
  65. package/wrangler-dist/cli.js +71020 -65212
@@ -0,0 +1,416 @@
1
+ import { fork } from "node:child_process";
2
+ import { realpathSync } from "node:fs";
3
+ import { writeFile } from "node:fs/promises";
4
+ import * as path from "node:path";
5
+ import onExit from "signal-exit";
6
+ import tmp from "tmp-promise";
7
+ import { bundleWorker } from "../bundle";
8
+ import { registerWorker } from "../dev-registry";
9
+ import { runCustomBuild } from "../entry";
10
+ import { logger } from "../logger";
11
+ import { waitForPortToBeAvailable } from "../proxy";
12
+ import {
13
+ setupBindings,
14
+ setupMiniflareOptions,
15
+ setupNodeOptions,
16
+ } from "./local";
17
+ import { validateDevProps } from "./validate-dev-props";
18
+
19
+ import type { Config } from "../config";
20
+ import type { Entry } from "../entry";
21
+ import type { DevProps, DirectorySyncResult } from "./dev";
22
+ import type { LocalProps } from "./local";
23
+ import type { EsbuildBundle } from "./use-esbuild";
24
+
25
+ import type { ChildProcess } from "node:child_process";
26
+
27
+ export async function startDevServer(
28
+ props: DevProps & {
29
+ local: boolean;
30
+ }
31
+ ) {
32
+ try {
33
+ validateDevProps(props);
34
+
35
+ if (props.build.command) {
36
+ const relativeFile =
37
+ path.relative(props.entry.directory, props.entry.file) || ".";
38
+ await runCustomBuild(props.entry.file, relativeFile, props.build).catch(
39
+ (err) => {
40
+ logger.error("Custom build failed:", err);
41
+ }
42
+ );
43
+ }
44
+
45
+ //implement a react-free version of useTmpDir
46
+ const directory = setupTempDir();
47
+ if (!directory) {
48
+ throw new Error("Failed to create temporary directory.");
49
+ }
50
+
51
+ //implement a react-free version of useEsbuild
52
+ const bundle = await runEsbuild({
53
+ entry: props.entry,
54
+ destination: directory.name,
55
+ jsxFactory: props.jsxFactory,
56
+ rules: props.rules,
57
+ jsxFragment: props.jsxFragment,
58
+ serveAssetsFromWorker: Boolean(
59
+ props.assetPaths && !props.isWorkersSite && props.local
60
+ ),
61
+ tsconfig: props.tsconfig,
62
+ minify: props.minify,
63
+ nodeCompat: props.nodeCompat,
64
+ define: props.define,
65
+ noBundle: props.noBundle,
66
+ assets: props.assetsConfig,
67
+ services: props.bindings.services,
68
+ });
69
+
70
+ //run local now
71
+ const { stop, inspectorUrl } = await startLocalServer({
72
+ name: props.name,
73
+ bundle: bundle,
74
+ format: props.entry.format,
75
+ compatibilityDate: props.compatibilityDate,
76
+ compatibilityFlags: props.compatibilityFlags,
77
+ bindings: props.bindings,
78
+ assetPaths: props.assetPaths,
79
+ port: props.port,
80
+ ip: props.ip,
81
+ rules: props.rules,
82
+ inspectorPort: props.inspectorPort,
83
+ enableLocalPersistence: props.enableLocalPersistence,
84
+ liveReload: props.liveReload,
85
+ crons: props.crons,
86
+ localProtocol: props.localProtocol,
87
+ localUpstream: props.localUpstream,
88
+ logLevel: props.logLevel,
89
+ logPrefix: props.logPrefix,
90
+ inspect: props.inspect,
91
+ onReady: props.onReady,
92
+ enablePagesAssetsServiceBinding: props.enablePagesAssetsServiceBinding,
93
+ usageModel: undefined,
94
+ workerDefinitions: undefined,
95
+ });
96
+
97
+ return {
98
+ stop: async () => {
99
+ stop();
100
+ },
101
+ inspectorUrl,
102
+ };
103
+ } catch (err) {
104
+ logger.error(err);
105
+ }
106
+ }
107
+
108
+ function setupTempDir(): DirectorySyncResult | undefined {
109
+ let dir: DirectorySyncResult | undefined;
110
+ try {
111
+ dir = tmp.dirSync({ unsafeCleanup: true });
112
+
113
+ return dir;
114
+ } catch (err) {
115
+ logger.error("Failed to create temporary directory to store built files.");
116
+ }
117
+ }
118
+
119
+ async function runEsbuild({
120
+ entry,
121
+ destination,
122
+ jsxFactory,
123
+ jsxFragment,
124
+ rules,
125
+ assets,
126
+ serveAssetsFromWorker,
127
+ tsconfig,
128
+ minify,
129
+ nodeCompat,
130
+ define,
131
+ noBundle,
132
+ }: {
133
+ entry: Entry;
134
+ destination: string | undefined;
135
+ jsxFactory: string | undefined;
136
+ jsxFragment: string | undefined;
137
+ rules: Config["rules"];
138
+ assets: Config["assets"];
139
+ define: Config["define"];
140
+ services: Config["services"];
141
+ serveAssetsFromWorker: boolean;
142
+ tsconfig: string | undefined;
143
+ minify: boolean | undefined;
144
+ nodeCompat: boolean | undefined;
145
+ noBundle: boolean;
146
+ }): Promise<EsbuildBundle | undefined> {
147
+ if (!destination) return;
148
+
149
+ const {
150
+ resolvedEntryPointPath,
151
+ bundleType,
152
+ modules,
153
+ sourceMapPath,
154
+ }: Awaited<ReturnType<typeof bundleWorker>> = noBundle
155
+ ? {
156
+ modules: [],
157
+ resolvedEntryPointPath: entry.file,
158
+ bundleType: entry.format === "modules" ? "esm" : "commonjs",
159
+ stop: undefined,
160
+ sourceMapPath: undefined,
161
+ }
162
+ : await bundleWorker(entry, destination, {
163
+ serveAssetsFromWorker,
164
+ jsxFactory,
165
+ jsxFragment,
166
+ rules,
167
+ tsconfig,
168
+ minify,
169
+ nodeCompat,
170
+ define,
171
+ checkFetch: true,
172
+ assets: assets && {
173
+ ...assets,
174
+ // disable the cache in dev
175
+ bypassCache: true,
176
+ },
177
+ services: undefined,
178
+ workerDefinitions: undefined,
179
+ firstPartyWorkerDevFacade: undefined,
180
+ targetConsumer: "dev", // We are starting a dev server
181
+ });
182
+
183
+ return {
184
+ id: 0,
185
+ entry,
186
+ path: resolvedEntryPointPath,
187
+ type: bundleType,
188
+ modules,
189
+ sourceMapPath,
190
+ };
191
+ }
192
+
193
+ export async function startLocalServer({
194
+ name: workerName,
195
+ bundle,
196
+ format,
197
+ compatibilityDate,
198
+ compatibilityFlags,
199
+ usageModel,
200
+ bindings,
201
+ workerDefinitions,
202
+ assetPaths,
203
+ port,
204
+ inspectorPort,
205
+ rules,
206
+ enableLocalPersistence,
207
+ liveReload,
208
+ ip,
209
+ crons,
210
+ localProtocol,
211
+ localUpstream,
212
+ inspect,
213
+ onReady,
214
+ logLevel,
215
+ logPrefix,
216
+ enablePagesAssetsServiceBinding,
217
+ }: LocalProps) {
218
+ let local: ChildProcess | undefined;
219
+ let removeSignalExitListener: (() => void) | undefined;
220
+ let inspectorUrl: string | undefined;
221
+ const setInspectorUrl = (url: string) => {
222
+ inspectorUrl = url;
223
+ };
224
+
225
+ // if we're using local persistence for data, we should use the cwd
226
+ // as an explicit path, or else it'll use the temp dir
227
+ // which disappears when dev ends
228
+ const localPersistencePath = enableLocalPersistence
229
+ ? // Maybe we could make the path configurable as well?
230
+ path.join(process.cwd(), "wrangler-local-state")
231
+ : // We otherwise choose null, but choose true later
232
+ // so that it's persisted in the temp dir across a dev session
233
+ // even when we change source and reload
234
+ null;
235
+
236
+ const abortController = new AbortController();
237
+ async function startLocalWorker() {
238
+ if (!bundle || !format) return;
239
+
240
+ // port for the worker
241
+
242
+ if (port !== 0) {
243
+ await waitForPortToBeAvailable(port, {
244
+ retryPeriod: 200,
245
+ timeout: 2000,
246
+ abortSignal: abortController.signal,
247
+ });
248
+ }
249
+
250
+ if (bindings.services && bindings.services.length > 0) {
251
+ throw new Error(
252
+ "⎔ Service bindings are not yet supported in local mode."
253
+ );
254
+ }
255
+
256
+ // In local mode, we want to copy all referenced modules into
257
+ // the output bundle directory before starting up
258
+ for (const module of bundle.modules) {
259
+ await writeFile(
260
+ path.join(path.dirname(bundle.path), module.name),
261
+ module.content
262
+ );
263
+ }
264
+
265
+ const scriptPath = realpathSync(bundle.path);
266
+
267
+ const upstream =
268
+ typeof localUpstream === "string"
269
+ ? `${localProtocol}://${localUpstream}`
270
+ : undefined;
271
+
272
+ const {
273
+ externalDurableObjects,
274
+ internalDurableObjects,
275
+ wasmBindings,
276
+ textBlobBindings,
277
+ dataBlobBindings,
278
+ } = setupBindings({
279
+ wasm_modules: bindings.wasm_modules,
280
+ text_blobs: bindings.text_blobs,
281
+ data_blobs: bindings.data_blobs,
282
+ durable_objects: bindings.durable_objects,
283
+ format,
284
+ bundle,
285
+ });
286
+
287
+ const { forkOptions, miniflareCLIPath } = setupMiniflareOptions({
288
+ workerName,
289
+ port,
290
+ scriptPath,
291
+ localProtocol,
292
+ ip,
293
+ format,
294
+ rules,
295
+ compatibilityDate,
296
+ compatibilityFlags,
297
+ usageModel,
298
+ kv_namespaces: bindings?.kv_namespaces,
299
+ r2_buckets: bindings?.r2_buckets,
300
+ internalDurableObjects,
301
+ externalDurableObjects,
302
+ localPersistencePath,
303
+ liveReload,
304
+ assetPaths,
305
+ vars: bindings?.vars,
306
+ wasmBindings,
307
+ textBlobBindings,
308
+ dataBlobBindings,
309
+ crons,
310
+ upstream,
311
+ logLevel,
312
+ logPrefix,
313
+ workerDefinitions,
314
+ enablePagesAssetsServiceBinding,
315
+ });
316
+
317
+ const nodeOptions = setupNodeOptions({ inspect, ip, inspectorPort });
318
+ logger.log("⎔ Starting a local server...");
319
+
320
+ const child = (local = fork(miniflareCLIPath, forkOptions, {
321
+ cwd: path.dirname(scriptPath),
322
+ execArgv: nodeOptions,
323
+ stdio: "pipe",
324
+ }));
325
+
326
+ child.on("message", async (messageString) => {
327
+ const message = JSON.parse(messageString as string);
328
+ if (message.ready) {
329
+ // Let's register our presence in the dev registry
330
+ if (workerName) {
331
+ await registerWorker(workerName, {
332
+ protocol: localProtocol,
333
+ mode: "local",
334
+ port: message.mfPort,
335
+ host: ip,
336
+ durableObjects: internalDurableObjects.map((binding) => ({
337
+ name: binding.name,
338
+ className: binding.class_name,
339
+ })),
340
+ ...(message.durableObjectsPort
341
+ ? {
342
+ durableObjectsHost: ip,
343
+ durableObjectsPort: message.durableObjectsPort,
344
+ }
345
+ : {}),
346
+ });
347
+ }
348
+ onReady?.(ip, message.mfPort);
349
+ }
350
+ });
351
+
352
+ child.on("close", (code) => {
353
+ if (code) {
354
+ logger.log(`Miniflare process exited with code ${code}`);
355
+ }
356
+ });
357
+
358
+ child.stdout?.on("data", (data: Buffer) => {
359
+ process.stdout.write(data);
360
+ });
361
+
362
+ // parse the node inspector url (which may be received in chunks) from stderr
363
+ let stderrData = "";
364
+ let inspectorUrlFound = false;
365
+ child.stderr?.on("data", (data: Buffer) => {
366
+ if (!inspectorUrlFound) {
367
+ stderrData += data.toString();
368
+ const matches =
369
+ /Debugger listening on (ws:\/\/127\.0\.0\.1:\d+\/[A-Za-z0-9-]+)[\r|\n]/.exec(
370
+ stderrData
371
+ );
372
+ if (matches) {
373
+ inspectorUrlFound = true;
374
+ setInspectorUrl(matches[1]);
375
+ }
376
+ }
377
+
378
+ process.stderr.write(data);
379
+ });
380
+
381
+ child.on("exit", (code) => {
382
+ if (code) {
383
+ logger.error(`Miniflare process exited with code ${code}`);
384
+ }
385
+ });
386
+
387
+ child.on("error", (error: Error) => {
388
+ logger.error(`Miniflare process failed to spawn`);
389
+ logger.error(error);
390
+ });
391
+
392
+ removeSignalExitListener = onExit((_code, _signal) => {
393
+ logger.log("⎔ Shutting down local server.");
394
+ child.kill();
395
+ local = undefined;
396
+ });
397
+ }
398
+
399
+ startLocalWorker().catch((err) => {
400
+ logger.error("local worker:", err);
401
+ });
402
+
403
+ return {
404
+ inspectorUrl,
405
+ stop: () => {
406
+ abortController.abort();
407
+ if (local) {
408
+ logger.log("⎔ Shutting down local server.");
409
+ local.kill();
410
+ local = undefined;
411
+ removeSignalExitListener && removeSignalExitListener();
412
+ removeSignalExitListener = undefined;
413
+ }
414
+ },
415
+ };
416
+ }
@@ -36,6 +36,7 @@ export function useEsbuild({
36
36
  services,
37
37
  durableObjects,
38
38
  firstPartyWorkerDevFacade,
39
+ targetConsumer,
39
40
  }: {
40
41
  entry: Entry;
41
42
  destination: string | undefined;
@@ -53,6 +54,7 @@ export function useEsbuild({
53
54
  workerDefinitions: WorkerRegistry;
54
55
  durableObjects: Config["durable_objects"];
55
56
  firstPartyWorkerDevFacade: boolean | undefined;
57
+ targetConsumer: "dev" | "publish";
56
58
  }): EsbuildBundle | undefined {
57
59
  const [bundle, setBundle] = useState<EsbuildBundle>();
58
60
  const { exit } = useApp();
@@ -116,6 +118,7 @@ export function useEsbuild({
116
118
  workerDefinitions,
117
119
  services,
118
120
  firstPartyWorkerDevFacade,
121
+ targetConsumer,
119
122
  });
120
123
 
121
124
  // Capture the `stop()` method to use as the `useEffect()` destructor.
@@ -173,6 +176,7 @@ export function useEsbuild({
173
176
  durableObjects,
174
177
  workerDefinitions,
175
178
  firstPartyWorkerDevFacade,
179
+ targetConsumer,
176
180
  ]);
177
181
  return bundle;
178
182
  }