wrangler 2.9.0 → 2.10.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 (82) hide show
  1. package/README.md +3 -3
  2. package/miniflare-dist/index.mjs +2 -15
  3. package/package.json +9 -9
  4. package/src/__tests__/configuration.test.ts +70 -0
  5. package/src/__tests__/d1/d1.test.ts +3 -6
  6. package/src/__tests__/d1/execute.test.ts +64 -0
  7. package/src/__tests__/d1/migrate.test.ts +107 -0
  8. package/src/__tests__/deployments.test.ts +40 -16
  9. package/src/__tests__/dev.test.tsx +3 -3
  10. package/src/__tests__/generate.test.ts +1 -1
  11. package/src/__tests__/helpers/end-event-loop.ts +6 -0
  12. package/src/__tests__/helpers/mock-get-pages-upload-token.ts +25 -0
  13. package/src/__tests__/helpers/mock-set-timeout.ts +16 -0
  14. package/src/__tests__/helpers/msw/handlers/deployments.ts +40 -16
  15. package/src/__tests__/helpers/string-dynamic-values-matcher.ts +28 -0
  16. package/src/__tests__/index.test.ts +3 -4
  17. package/src/__tests__/init.test.ts +1 -1
  18. package/src/__tests__/kv.test.ts +8 -8
  19. package/src/__tests__/middleware.test.ts +65 -0
  20. package/src/__tests__/mtls-certificates.test.ts +585 -0
  21. package/src/__tests__/pages/deployment-list.test.ts +78 -0
  22. package/src/__tests__/pages/functions-build.test.ts +402 -0
  23. package/src/__tests__/pages/pages.test.ts +81 -0
  24. package/src/__tests__/pages/project-create.test.ts +63 -0
  25. package/src/__tests__/pages/project-list.test.ts +108 -0
  26. package/src/__tests__/pages/project-upload.test.ts +481 -0
  27. package/src/__tests__/pages/publish.test.ts +2745 -0
  28. package/src/__tests__/publish.test.ts +58 -27
  29. package/src/__tests__/queues.test.ts +2 -2
  30. package/src/__tests__/secret.test.ts +4 -4
  31. package/src/__tests__/tsconfig.tsbuildinfo +1 -1
  32. package/src/__tests__/user.test.ts +1 -1
  33. package/src/__tests__/whoami.test.tsx +1 -1
  34. package/src/__tests__/worker-namespace.test.ts +1 -1
  35. package/src/api/index.ts +8 -0
  36. package/src/api/mtls-certificate.ts +148 -0
  37. package/src/api/pages/create-worker-bundle-contents.ts +75 -0
  38. package/src/api/pages/publish.tsx +52 -8
  39. package/src/bundle.ts +6 -5
  40. package/src/config/config.ts +7 -7
  41. package/src/config/environment.ts +9 -2
  42. package/src/config/index.ts +13 -0
  43. package/src/config/validation.ts +50 -3
  44. package/src/create-worker-upload-form.ts +9 -0
  45. package/src/d1/execute.tsx +124 -91
  46. package/src/d1/migrations/apply.tsx +36 -29
  47. package/src/d1/migrations/create.tsx +10 -8
  48. package/src/d1/migrations/helpers.ts +63 -38
  49. package/src/d1/migrations/list.tsx +31 -20
  50. package/src/d1/migrations/options.ts +6 -1
  51. package/src/d1/types.ts +1 -0
  52. package/src/d1/utils.ts +2 -1
  53. package/src/deployments.ts +62 -39
  54. package/src/dev/dev.tsx +1 -15
  55. package/src/dev/remote.tsx +2 -2
  56. package/src/dev.tsx +9 -6
  57. package/src/generate/index.ts +1 -1
  58. package/src/index.ts +15 -5
  59. package/src/miniflare-cli/assets.ts +1 -1
  60. package/src/miniflare-cli/tsconfig.tsbuildinfo +1 -1
  61. package/src/mtls-certificate/cli.ts +155 -0
  62. package/src/pages/build.ts +103 -23
  63. package/src/pages/buildFunctions.ts +32 -31
  64. package/src/pages/dev.ts +4 -2
  65. package/src/pages/functions/tsconfig.tsbuildinfo +1 -1
  66. package/src/pages/publish.tsx +12 -1
  67. package/src/pages/utils.ts +1 -1
  68. package/src/publish/publish.ts +3 -2
  69. package/src/secret/index.ts +1 -0
  70. package/src/sites.ts +1 -1
  71. package/src/tail/filters.ts +1 -1
  72. package/src/user/user.ts +4 -3
  73. package/src/worker.ts +6 -0
  74. package/templates/format-dev-errors.ts +1 -0
  75. package/templates/new-worker.ts +3 -0
  76. package/templates/serve-static-assets.ts +1 -0
  77. package/templates/service-bindings-module-facade.js +1 -0
  78. package/templates/tsconfig.init.json +1 -1
  79. package/templates/tsconfig.tsbuildinfo +1 -1
  80. package/wrangler-dist/cli.d.ts +82 -2
  81. package/wrangler-dist/cli.js +1726 -1616
  82. package/src/__tests__/pages.test.ts +0 -2905
@@ -0,0 +1,402 @@
1
+ import { existsSync, mkdirSync, readFileSync, writeFileSync } from "node:fs";
2
+ import { endEventLoop } from "../helpers/end-event-loop";
3
+ import { mockConsoleMethods } from "../helpers/mock-console";
4
+ import { runInTempDir } from "../helpers/run-in-tmp";
5
+ import { runWrangler } from "../helpers/run-wrangler";
6
+ import { replaceRandomWithConstantData } from "../helpers/string-dynamic-values-matcher";
7
+
8
+ describe("functions build", () => {
9
+ const std = mockConsoleMethods();
10
+
11
+ runInTempDir();
12
+
13
+ afterEach(async () => {
14
+ // Force a tick to ensure that all promises resolve
15
+ await endEventLoop();
16
+ });
17
+
18
+ it("should throw an error if no worker script and no Functions directory was found", async () => {
19
+ await expect(runWrangler("pages functions build")).rejects.toThrowError();
20
+ expect(std.err).toContain("Could not find anything to build.");
21
+ });
22
+
23
+ it("should build functions", async () => {
24
+ /* ---------------------------- */
25
+ /* Set up Functions */
26
+ /* ---------------------------- */
27
+ mkdirSync("functions");
28
+ writeFileSync(
29
+ "functions/hello.js",
30
+ `
31
+ export async function onRequest() {
32
+ return new Response("Hello from Pages Functions");
33
+ }
34
+ `
35
+ );
36
+
37
+ /* --------------------------------- */
38
+ /* Run cmd & make assertions */
39
+ /* --------------------------------- */
40
+ await runWrangler(`pages functions build`);
41
+
42
+ expect(existsSync("_worker.js")).toBe(true);
43
+ expect(std.out).toMatchInlineSnapshot(`
44
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
45
+ ✨ Compiled Worker successfully"
46
+ `);
47
+ expect(std.err).toMatchInlineSnapshot(`""`);
48
+ });
49
+
50
+ it("should include any external modules imported by functions in the output bundle when the [--experimental-worker-bundle] flag is set", async () => {
51
+ /* ---------------------------- */
52
+ /* Set up wasm files */
53
+ /* ---------------------------- */
54
+ mkdirSync("wasm");
55
+ writeFileSync("wasm/greeting.wasm", "Hello");
56
+ writeFileSync("wasm/name.wasm", "wasm Functions");
57
+
58
+ /* ---------------------------- */
59
+ /* Set up Functions */
60
+ /* ---------------------------- */
61
+ mkdirSync("functions");
62
+ writeFileSync(
63
+ "functions/hello.js",
64
+ `
65
+ import hello from "./../wasm/greeting.wasm";
66
+ import name from "./../wasm/name.wasm";
67
+
68
+ export async function onRequest() {
69
+ const greetingModule = await WebAssembly.instantiate(greeting);
70
+ const nameModule = await WebAssembly.instantiate(name);
71
+ return new Response(greetingModule.exports.hello + " " + nameModule.exports.name);
72
+ }
73
+ `
74
+ );
75
+
76
+ /* --------------------------------- */
77
+ /* Run cmd & make assertions */
78
+ /* --------------------------------- */
79
+ await runWrangler(
80
+ `pages functions build --outfile=_worker.bundle --experimental-worker-bundle=true`
81
+ );
82
+
83
+ expect(existsSync("_worker.bundle")).toBe(true);
84
+ expect(std.out).toMatchInlineSnapshot(`
85
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
86
+ ✨ Compiled Worker successfully"
87
+ `);
88
+
89
+ // some values in workerBundleContents, such as the undici form boundary
90
+ // or the file hashes, are randomly generated. Let's replace them
91
+ // with static values so we can test the file contents
92
+ const workerBundleContents = readFileSync("_worker.bundle", "utf-8");
93
+ const workerBundleWithConstantData = replaceRandomWithConstantData(
94
+ workerBundleContents,
95
+ [
96
+ [/------formdata-undici-0.[0-9]*/g, "------formdata-undici-0.test"],
97
+ [/functionsWorker-0.[0-9]*.js/g, "functionsWorker-0.test.js"],
98
+ [/[0-9a-z]*-greeting.wasm/g, "test-greeting.wasm"],
99
+ [/[0-9a-z]*-name.wasm/g, "test-name.wasm"],
100
+ ]
101
+ );
102
+
103
+ // check we appended the metadata
104
+ expect(workerBundleWithConstantData).toContain(
105
+ `Content-Disposition: form-data; name="metadata"`
106
+ );
107
+ expect(workerBundleWithConstantData).toContain(
108
+ `{"main_module":"functionsWorker-0.test.js"}`
109
+ );
110
+
111
+ // check we appended the compiled Worker
112
+ expect(workerBundleWithConstantData).toContain(
113
+ `Content-Disposition: form-data; name="functionsWorker-0.test.js"; filename="functionsWorker-0.test.js"`
114
+ );
115
+
116
+ // check we appended the wasm modules
117
+ expect(workerBundleWithConstantData).toContain(
118
+ `Content-Disposition: form-data; name="./test-greeting.wasm"; filename="./test-greeting.wasm"`
119
+ );
120
+ expect(workerBundleWithConstantData).toContain(
121
+ `Content-Disposition: form-data; name="./test-name.wasm"; filename="./test-name.wasm"`
122
+ );
123
+ expect(workerBundleWithConstantData).toContain(`Hello`);
124
+ expect(workerBundleWithConstantData).toContain(`wasm Functions`);
125
+
126
+ expect(std.err).toMatchInlineSnapshot(`""`);
127
+ });
128
+
129
+ it("should fail building if functions use external module imports, but the [--experimental-worker-bundle] flag is not set", async () => {
130
+ /* ---------------------------- */
131
+ /* Set up wasm files */
132
+ /* ---------------------------- */
133
+ mkdirSync("wasm");
134
+ writeFileSync("wasm/greeting.wasm", "Hello wasm Functions!");
135
+
136
+ /* ---------------------------- */
137
+ /* Set up Functions */
138
+ /* ---------------------------- */
139
+ mkdirSync("functions");
140
+ writeFileSync(
141
+ "functions/hello.js",
142
+ `
143
+ import hello from "./../wasm/greeting.wasm";
144
+
145
+ export async function onRequest() {
146
+ const module = await WebAssembly.instantiate(greeting);
147
+ return new Response(module.exports.hello);
148
+ }
149
+ `
150
+ );
151
+
152
+ /* --------------------------------- */
153
+ /* Run cmd & make assertions */
154
+ /* --------------------------------- */
155
+ await expect(runWrangler(`pages functions build`)).rejects.toThrowError();
156
+ expect(std.err).toContain(
157
+ `ERROR: No loader is configured for ".wasm" files: ../wasm/greeting.wasm`
158
+ );
159
+ });
160
+
161
+ it("should build _worker.js if the [--experimental-worker-bundle] flag is set", async () => {
162
+ /* ---------------------------- */
163
+ /* Set up js files */
164
+ /* ---------------------------- */
165
+ mkdirSync("utils");
166
+ writeFileSync(
167
+ "utils/meaning-of-life.js",
168
+ `
169
+ export const MEANING_OF_LIFE = 21;
170
+ `
171
+ );
172
+
173
+ /* ---------------------------- */
174
+ /* Set up _worker.js */
175
+ /* ---------------------------- */
176
+ mkdirSync("public");
177
+ writeFileSync(
178
+ "public/_worker.js",
179
+ `
180
+ import { MEANING_OF_LIFE } from "./../utils/meaning-of-life.js";
181
+
182
+ export default {
183
+ async fetch(request, env) {
184
+ return new Response("Hello from _worker.js. The meaning of life is " + MEANING_OF_LIFE);
185
+ },
186
+ };`
187
+ );
188
+
189
+ /* --------------------------------- */
190
+ /* Run cmd & make assertions */
191
+ /* --------------------------------- */
192
+ await runWrangler(
193
+ `pages functions build --build-output-directory=public --outfile=_worker.bundle --experimental-worker-bundle=true`
194
+ );
195
+ expect(existsSync("_worker.bundle")).toBe(true);
196
+ expect(std.out).toMatchInlineSnapshot(`
197
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
198
+ ✨ Compiled Worker successfully"
199
+ `);
200
+
201
+ // some values in workerBundleContents, such as the undici form boundary
202
+ // or the file hashes, are randomly generated. Let's replace them
203
+ // with static values so we can test the file contents
204
+ const workerBundleContents = readFileSync("_worker.bundle", "utf-8");
205
+ const workerBundleWithConstantData = replaceRandomWithConstantData(
206
+ workerBundleContents,
207
+ [
208
+ [/------formdata-undici-0.[0-9]*/g, "------formdata-undici-0.test"],
209
+ [/bundledWorker-0.[0-9]*.mjs/g, "bundledWorker-0.test.mjs"],
210
+ ]
211
+ );
212
+
213
+ expect(workerBundleWithConstantData).toMatchInlineSnapshot(`
214
+ "------formdata-undici-0.test
215
+ Content-Disposition: form-data; name=\\"metadata\\"
216
+
217
+ {\\"main_module\\":\\"bundledWorker-0.test.mjs\\"}
218
+ ------formdata-undici-0.test
219
+ Content-Disposition: form-data; name=\\"bundledWorker-0.test.mjs\\"; filename=\\"bundledWorker-0.test.mjs\\"
220
+ Content-Type: application/javascript+module
221
+
222
+ // ../utils/meaning-of-life.js
223
+ var MEANING_OF_LIFE = 21;
224
+
225
+ // _worker.js
226
+ var worker_default = {
227
+ async fetch(request, env) {
228
+ return new Response(\\"Hello from _worker.js. The meaning of life is \\" + MEANING_OF_LIFE);
229
+ }
230
+ };
231
+ export {
232
+ worker_default as default
233
+ };
234
+ //# sourceMappingURL=bundledWorker-0.test.mjs.map
235
+
236
+ ------formdata-undici-0.test--"
237
+ `);
238
+ expect(std.err).toMatchInlineSnapshot(`""`);
239
+
240
+ await expect(
241
+ runWrangler(`pages functions build --build-output-directory=public`)
242
+ ).rejects.toThrowError();
243
+ expect(std.err).toMatch(
244
+ /ENOENT: no such file or directory, scandir '.*functions'/
245
+ );
246
+ });
247
+
248
+ it("should include all external modules imported by _worker.js in the output bundle, when bundling _worker.js", async () => {
249
+ /* ---------------------------- */
250
+ /* Set up wasm files */
251
+ /* ---------------------------- */
252
+ mkdirSync("wasm");
253
+ writeFileSync("wasm/greeting.wasm", "Hello");
254
+ writeFileSync("wasm/name.wasm", "wasm _worker.js");
255
+
256
+ /* ---------------------------- */
257
+ /* Set up _worker.js */
258
+ /* ---------------------------- */
259
+ mkdirSync("public");
260
+ writeFileSync(
261
+ "public/_worker.js",
262
+ `
263
+ import greeting from "./../wasm/greeting.wasm";
264
+ import name from "./../wasm/name.wasm";
265
+
266
+ export default {
267
+ async fetch(request, env) {
268
+ const greetingModule = await WebAssembly.instantiate(greeting);
269
+ const nameModule = await WebAssembly.instantiate(name);
270
+ return new Response(greetingModule.exports.hello + " " + nameModule.exports.name + "!");
271
+ },
272
+ };`
273
+ );
274
+
275
+ /* --------------------------------- */
276
+ /* Run cmd & make assertions */
277
+ /* --------------------------------- */
278
+ await runWrangler(
279
+ `pages functions build --build-output-directory=public --experimental-worker-bundle=true`
280
+ );
281
+
282
+ // built to _worker.js by default
283
+ expect(existsSync("_worker.js")).toBe(true);
284
+ expect(std.out).toMatchInlineSnapshot(`
285
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
286
+ ✨ Compiled Worker successfully"
287
+ `);
288
+
289
+ // some values in workerBundleContents, such as the undici form boundary
290
+ // or the file hashes, are randomly generated. Let's replace them
291
+ // with static values so we can test the file contents
292
+ const workerBundleContents = readFileSync("_worker.js", "utf-8");
293
+ const workerBundleWithConstantData = replaceRandomWithConstantData(
294
+ workerBundleContents,
295
+ [
296
+ [/------formdata-undici-0.[0-9]*/g, "------formdata-undici-0.test"],
297
+ [/bundledWorker-0.[0-9]*.mjs/g, "bundledWorker-0.test.mjs"],
298
+ [/[0-9a-z]*-greeting.wasm/g, "test-greeting.wasm"],
299
+ [/[0-9a-z]*-name.wasm/g, "test-name.wasm"],
300
+ ]
301
+ );
302
+
303
+ // check we appended the metadata
304
+ expect(workerBundleWithConstantData).toContain(
305
+ `Content-Disposition: form-data; name="metadata"`
306
+ );
307
+ expect(workerBundleWithConstantData).toContain(
308
+ `{"main_module":"bundledWorker-0.test.mjs"}`
309
+ );
310
+
311
+ // check we appended the wasm modules
312
+ expect(workerBundleWithConstantData).toContain(
313
+ `Content-Disposition: form-data; name="./test-greeting.wasm"; filename="./test-greeting.wasm"`
314
+ );
315
+ expect(workerBundleWithConstantData).toContain(
316
+ `Content-Disposition: form-data; name="./test-name.wasm"; filename="./test-name.wasm"`
317
+ );
318
+ expect(workerBundleWithConstantData).toContain(`Hello`);
319
+ expect(workerBundleWithConstantData).toContain(`wasm _worker.js`);
320
+
321
+ expect(std.err).toMatchInlineSnapshot(`""`);
322
+ });
323
+
324
+ it("should build _worker.js over /functions, if both are present and the [--experimental-worker-bundle] flag is set", async () => {
325
+ /* ---------------------------- */
326
+ /* Set up _worker.js */
327
+ /* ---------------------------- */
328
+ mkdirSync("public");
329
+ writeFileSync(
330
+ "public/_worker.js",
331
+ `
332
+ export default {
333
+ async fetch(request, env) {
334
+ return new Response("Hello from _worker.js");
335
+ },
336
+ };`
337
+ );
338
+
339
+ /* ---------------------------- */
340
+ /* Set up Functions */
341
+ /* ---------------------------- */
342
+ mkdirSync("functions");
343
+ writeFileSync(
344
+ "functions/hello.js",
345
+ `
346
+ export async function onRequest() {
347
+ return new Response("Hello from Pages Functions");
348
+ }
349
+ `
350
+ );
351
+
352
+ /* --------------------------------- */
353
+ /* Run cmd & make assertions */
354
+ /* --------------------------------- */
355
+ await runWrangler(
356
+ `pages functions build --outfile=public/_worker.bundle --experimental-worker-bundle=true`
357
+ );
358
+
359
+ expect(existsSync("public/_worker.bundle")).toBe(true);
360
+ expect(std.out).toMatchInlineSnapshot(`
361
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
362
+ ✨ Compiled Worker successfully"
363
+ `);
364
+
365
+ // some values in workerBundleContents, such as the undici form boundary
366
+ // or the file hashes, are randomly generated. Let's replace them
367
+ // with static values so we can test the file contents
368
+ const workerBundleContents = readFileSync("public/_worker.bundle", "utf-8");
369
+ const workerBundleWithConstantData = replaceRandomWithConstantData(
370
+ workerBundleContents,
371
+ [
372
+ [/------formdata-undici-0.[0-9]*/g, "------formdata-undici-0.test"],
373
+ [/bundledWorker-0.[0-9]*.mjs/g, "bundledWorker-0.test.mjs"],
374
+ ]
375
+ );
376
+
377
+ expect(workerBundleWithConstantData).toMatchInlineSnapshot(`
378
+ "------formdata-undici-0.test
379
+ Content-Disposition: form-data; name=\\"metadata\\"
380
+
381
+ {\\"main_module\\":\\"bundledWorker-0.test.mjs\\"}
382
+ ------formdata-undici-0.test
383
+ Content-Disposition: form-data; name=\\"bundledWorker-0.test.mjs\\"; filename=\\"bundledWorker-0.test.mjs\\"
384
+ Content-Type: application/javascript+module
385
+
386
+ // _worker.js
387
+ var worker_default = {
388
+ async fetch(request, env) {
389
+ return new Response(\\"Hello from _worker.js\\");
390
+ }
391
+ };
392
+ export {
393
+ worker_default as default
394
+ };
395
+ //# sourceMappingURL=bundledWorker-0.test.mjs.map
396
+
397
+ ------formdata-undici-0.test--"
398
+ `);
399
+
400
+ expect(std.err).toMatchInlineSnapshot(`""`);
401
+ });
402
+ });
@@ -0,0 +1,81 @@
1
+ import { endEventLoop } from "../helpers/end-event-loop";
2
+ import { mockConsoleMethods } from "../helpers/mock-console";
3
+ import { runInTempDir } from "../helpers/run-in-tmp";
4
+ import { runWrangler } from "../helpers/run-wrangler";
5
+
6
+ describe("pages", () => {
7
+ const std = mockConsoleMethods();
8
+
9
+ runInTempDir();
10
+
11
+ afterEach(async () => {
12
+ // Force a tick to ensure that all promises resolve
13
+ await endEventLoop();
14
+ });
15
+
16
+ it("should display a list of available subcommands, for pages with no subcommand", async () => {
17
+ await runWrangler("pages");
18
+ await endEventLoop();
19
+
20
+ expect(std.out).toMatchInlineSnapshot(`
21
+ "wrangler pages
22
+
23
+ ⚡️ Configure Cloudflare Pages
24
+
25
+ Commands:
26
+ wrangler pages dev [directory] [-- command..] 🧑‍💻 Develop your full-stack Pages application locally
27
+ wrangler pages project ⚡️ Interact with your Pages projects
28
+ wrangler pages deployment 🚀 Interact with the deployments of a project
29
+ wrangler pages publish [directory] 🆙 Publish a directory of static assets as a Pages deployment
30
+
31
+ Flags:
32
+ -j, --experimental-json-config Experimental: Support wrangler.json [boolean]
33
+ -c, --config Path to .toml configuration file [string]
34
+ -e, --env Environment to use for operations and .env files [string]
35
+ -h, --help Show help [boolean]
36
+ -v, --version Show version number [boolean]
37
+
38
+ 🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose"
39
+ `);
40
+ });
41
+
42
+ describe("beta message for subcommands", () => {
43
+ it("should display for pages:dev", async () => {
44
+ await expect(
45
+ runWrangler("pages dev")
46
+ ).rejects.toThrowErrorMatchingInlineSnapshot(
47
+ `"Must specify a directory of static assets to serve or a command to run or a proxy port."`
48
+ );
49
+
50
+ expect(std.out).toMatchInlineSnapshot(`
51
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
52
+
53
+ If you think this is a bug then please create an issue at https://github.com/cloudflare/workers-sdk/issues/new/choose"
54
+ `);
55
+ });
56
+
57
+ it("should display for pages:functions:build", async () => {
58
+ await expect(runWrangler("pages functions build")).rejects.toThrowError();
59
+
60
+ expect(std.out).toMatchInlineSnapshot(`
61
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
62
+
63
+ If you think this is a bug then please create an issue at https://github.com/cloudflare/workers-sdk/issues/new/choose"
64
+ `);
65
+ });
66
+
67
+ it("should display for pages:functions:optimize-routes", async () => {
68
+ await expect(
69
+ runWrangler(
70
+ 'pages functions optimize-routes --routes-path="/build/_routes.json" --output-routes-path="/build/_optimized-routes.json"'
71
+ )
72
+ ).rejects.toThrowError();
73
+
74
+ expect(std.out).toMatchInlineSnapshot(`
75
+ "🚧 'wrangler pages <command>' is a beta command. Please report any issues to https://github.com/cloudflare/workers-sdk/issues/new/choose
76
+
77
+ If you think this is a bug then please create an issue at https://github.com/cloudflare/workers-sdk/issues/new/choose"
78
+ `);
79
+ });
80
+ });
81
+ });
@@ -0,0 +1,63 @@
1
+ import { rest } from "msw";
2
+ import { endEventLoop } from "../helpers/end-event-loop";
3
+ import { mockAccountId, mockApiToken } from "./../helpers/mock-account-id";
4
+ import { mockConsoleMethods } from "./../helpers/mock-console";
5
+ import { msw } from "./../helpers/msw";
6
+ import { runInTempDir } from "./../helpers/run-in-tmp";
7
+ import { runWrangler } from "./../helpers/run-wrangler";
8
+
9
+ describe("project create", () => {
10
+ const std = mockConsoleMethods();
11
+
12
+ runInTempDir();
13
+ mockAccountId();
14
+ mockApiToken();
15
+
16
+ afterEach(async () => {
17
+ // Force a tick to ensure that all promises resolve
18
+ await endEventLoop();
19
+ // Reset MSW after tick to ensure that all requests have been handled
20
+ msw.resetHandlers();
21
+ msw.restoreHandlers();
22
+ });
23
+
24
+ it("should create a project with a production branch", async () => {
25
+ msw.use(
26
+ rest.post(
27
+ "*/accounts/:accountId/pages/projects",
28
+ async (req, res, ctx) => {
29
+ const body = await req.json();
30
+
31
+ expect(req.params.accountId).toEqual("some-account-id");
32
+ expect(body).toEqual({
33
+ name: "a-new-project",
34
+ production_branch: "main",
35
+ });
36
+
37
+ return res.once(
38
+ ctx.status(200),
39
+ ctx.json({
40
+ success: true,
41
+ errors: [],
42
+ messages: [],
43
+ result: {
44
+ name: "a-new-project",
45
+ subdomain: "a-new-project.pages.dev",
46
+ production_branch: "main",
47
+ },
48
+ })
49
+ );
50
+ }
51
+ )
52
+ );
53
+
54
+ await runWrangler(
55
+ "pages project create a-new-project --production-branch=main"
56
+ );
57
+
58
+ expect(std.out).toMatchInlineSnapshot(`
59
+ "✨ Successfully created the 'a-new-project' project. It will be available at https://a-new-project.pages.dev/ once you create your first deployment.
60
+ To deploy a folder of assets, run 'wrangler pages publish [directory]'."
61
+ `);
62
+ });
63
+ });
@@ -0,0 +1,108 @@
1
+ import { rest } from "msw";
2
+ import { endEventLoop } from "../helpers/end-event-loop";
3
+ import { mockAccountId, mockApiToken } from "./../helpers/mock-account-id";
4
+ import { msw } from "./../helpers/msw";
5
+ import { runInTempDir } from "./../helpers/run-in-tmp";
6
+ import { runWrangler } from "./../helpers/run-wrangler";
7
+ import type { Project } from "./../../pages/types";
8
+
9
+ describe("project list", () => {
10
+ runInTempDir();
11
+ mockAccountId();
12
+ mockApiToken();
13
+
14
+ afterEach(async () => {
15
+ // Force a tick to ensure that all promises resolve
16
+ await endEventLoop();
17
+ // Reset MSW after tick to ensure that all requests have been handled
18
+ msw.resetHandlers();
19
+ msw.restoreHandlers();
20
+ });
21
+
22
+ it("should make request to list projects", async () => {
23
+ const projects: Project[] = [
24
+ {
25
+ name: "dogs",
26
+ subdomain: "docs.pages.dev",
27
+ domains: ["dogs.pages.dev"],
28
+ source: {
29
+ type: "github",
30
+ },
31
+ latest_deployment: {
32
+ modified_on: "2021-11-17T14:52:26.133835Z",
33
+ },
34
+ created_on: "2021-11-17T14:52:26.133835Z",
35
+ production_branch: "main",
36
+ },
37
+ {
38
+ name: "cats",
39
+ subdomain: "cats.pages.dev",
40
+ domains: ["cats.pages.dev", "kitten.com"],
41
+ latest_deployment: {
42
+ modified_on: "2021-11-17T14:52:26.133835Z",
43
+ },
44
+ created_on: "2021-11-17T14:52:26.133835Z",
45
+ production_branch: "main",
46
+ },
47
+ ];
48
+
49
+ const requests = mockProjectListRequest(projects);
50
+ await runWrangler("pages project list");
51
+
52
+ expect(requests.count).toBe(1);
53
+ });
54
+
55
+ it("should make multiple requests for paginated results", async () => {
56
+ const projects: Project[] = [];
57
+ for (let i = 0; i < 15; i++) {
58
+ projects.push({
59
+ name: "dogs" + i,
60
+ subdomain: i + "dogs.pages.dev",
61
+ domains: [i + "dogs.pages.dev"],
62
+ source: {
63
+ type: "github",
64
+ },
65
+ latest_deployment: {
66
+ modified_on: "2021-11-17T14:52:26.133835Z",
67
+ },
68
+ created_on: "2021-11-17T14:52:26.133835Z",
69
+ production_branch: "main",
70
+ });
71
+ }
72
+ const requests = mockProjectListRequest(projects);
73
+ await runWrangler("pages project list");
74
+ expect(requests.count).toEqual(2);
75
+ });
76
+ });
77
+
78
+ /* -------------------------------------------------- */
79
+ /* Helper Functions */
80
+ /* -------------------------------------------------- */
81
+
82
+ function mockProjectListRequest(projects: unknown[]) {
83
+ const requests = { count: 0 };
84
+ msw.use(
85
+ rest.get("*/accounts/:accountId/pages/projects", async (req, res, ctx) => {
86
+ requests.count++;
87
+ const pageSize = Number(req.url.searchParams.get("per_page"));
88
+ const page = Number(req.url.searchParams.get("page"));
89
+ const expectedPageSize = 10;
90
+ const expectedPage = requests.count;
91
+ expect(req.params.accountId).toEqual("some-account-id");
92
+ expect(pageSize).toEqual(expectedPageSize);
93
+ expect(page).toEqual(expectedPage);
94
+ expect(await req.text()).toEqual("");
95
+
96
+ return res(
97
+ ctx.status(200),
98
+ ctx.json({
99
+ success: true,
100
+ errors: [],
101
+ messages: [],
102
+ result: projects.slice((page - 1) * pageSize, page * pageSize),
103
+ })
104
+ );
105
+ })
106
+ );
107
+ return requests;
108
+ }