create-cloudflare 0.0.0-facb3ffc9 → 0.0.0-fbba583df

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 (106) hide show
  1. package/dist/cli.js +28282 -26413
  2. package/dist/tsconfig.tsbuildinfo +1 -1
  3. package/package.json +13 -8
  4. package/templates/analog/templates/worker-configuration.d.ts +1 -1
  5. package/templates/angular/c3.ts +24 -5
  6. package/templates/astro/templates/ts/worker-configuration.d.ts +1 -1
  7. package/templates/common/ts/src/index.ts +1 -1
  8. package/templates/common/ts/worker-configuration.d.ts +1 -1
  9. package/templates/docusaurus/c3.ts +1 -0
  10. package/templates/hello-world/c3.ts +17 -1
  11. package/templates/hello-world/js/package.json +2 -2
  12. package/templates/hello-world/js/test/index.spec.js +1 -1
  13. package/templates/hello-world/js/vitest.config.js +1 -1
  14. package/templates/hello-world/js/{wrangler.json → wrangler.jsonc} +2 -2
  15. package/templates/hello-world/py/.python-version +1 -0
  16. package/templates/hello-world/py/README.md +23 -0
  17. package/templates/hello-world/py/pyproject.toml +9 -0
  18. package/templates/hello-world/py/uv.lock +22 -0
  19. package/templates/hello-world/py/{wrangler.json → wrangler.jsonc} +2 -2
  20. package/templates/hello-world/ts/package.json +2 -2
  21. package/templates/hello-world/ts/src/index.ts +1 -1
  22. package/templates/hello-world/ts/vitest.config.mts +1 -1
  23. package/templates/hello-world/ts/worker-configuration.d.ts +1 -1
  24. package/templates/hello-world/ts/{wrangler.json → wrangler.jsonc} +2 -2
  25. package/templates/hello-world-durable-object/js/src/index.js +3 -3
  26. package/templates/hello-world-durable-object/ts/src/index.ts +3 -3
  27. package/templates/hono/c3.ts +1 -0
  28. package/templates/hono/templates/worker-configuration.d.ts +1 -1
  29. package/templates/next/README.md +1 -1
  30. package/templates/next/c3.ts +12 -9
  31. package/templates/nuxt/templates/worker-configuration.d.ts +1 -1
  32. package/templates/openapi/ts/worker-configuration.d.ts +1 -1
  33. package/templates/pre-existing/c3.ts +1 -1
  34. package/templates/queues/ts/src/index.ts +1 -1
  35. package/templates/queues/ts/worker-configuration.d.ts +1 -1
  36. package/templates/qwik/c3.ts +3 -2
  37. package/templates/qwik/templates/worker-configuration.d.ts +1 -1
  38. package/templates/remix/templates/worker-configuration.d.ts +1 -1
  39. package/templates/scheduled/js/src/index.js +1 -1
  40. package/templates/scheduled/ts/src/index.ts +2 -2
  41. package/templates/scheduled/ts/worker-configuration.d.ts +1 -1
  42. package/templates-experimental/angular/c3.ts +23 -5
  43. package/templates-experimental/astro/templates/ts/worker-configuration.d.ts +1 -1
  44. package/templates-experimental/docusaurus/c3.ts +1 -0
  45. package/templates-experimental/hello-world-durable-object-with-assets/js/src/index.js +3 -3
  46. package/templates-experimental/hello-world-durable-object-with-assets/ts/src/index.ts +3 -3
  47. package/templates-experimental/hello-world-with-assets/js/package.json +2 -2
  48. package/templates-experimental/hello-world-with-assets/js/vitest.config.js +1 -1
  49. package/templates-experimental/hello-world-with-assets/ts/package.json +2 -2
  50. package/templates-experimental/hello-world-with-assets/ts/src/index.ts +1 -1
  51. package/templates-experimental/hello-world-with-assets/ts/vitest.config.mts +1 -1
  52. package/templates-experimental/hello-world-with-assets/ts/worker-configuration.d.ts +1 -1
  53. package/templates-experimental/hono/c3.ts +1 -0
  54. package/templates-experimental/hono/templates/worker-configuration.d.ts +1 -1
  55. package/templates-experimental/next/c3.ts +27 -11
  56. package/templates-experimental/nuxt/templates/worker-configuration.d.ts +1 -1
  57. package/templates-experimental/qwik/c3.ts +3 -2
  58. package/templates-experimental/qwik/templates/worker-configuration.d.ts +1 -1
  59. package/templates-experimental/react/c3.ts +182 -0
  60. package/templates-experimental/react/js/api/index.js +13 -0
  61. package/templates-experimental/react/js/src/App.jsx +58 -0
  62. package/templates-experimental/react/js/src/assets/Cloudflare_Logo.svg +26 -0
  63. package/templates-experimental/react/js/wrangler.json +9 -0
  64. package/templates-experimental/react/ts/api/index.ts +17 -0
  65. package/templates-experimental/react/ts/src/App.tsx +58 -0
  66. package/templates-experimental/react/ts/src/assets/Cloudflare_Logo.svg +26 -0
  67. package/templates-experimental/react/ts/tsconfig.worker.json +8 -0
  68. package/templates-experimental/react/ts/wrangler.json +9 -0
  69. package/templates/pre-existing/js/wrangler.json +0 -5
  70. /package/templates/analog/templates/{wrangler.json → wrangler.jsonc} +0 -0
  71. /package/templates/astro/templates/js/{wrangler.json → wrangler.jsonc} +0 -0
  72. /package/templates/astro/templates/ts/{wrangler.json → wrangler.jsonc} +0 -0
  73. /package/templates/common/js/{wrangler.json → wrangler.jsonc} +0 -0
  74. /package/templates/common/ts/{wrangler.json → wrangler.jsonc} +0 -0
  75. /package/templates/hello-world-durable-object/js/{wrangler.json → wrangler.jsonc} +0 -0
  76. /package/templates/hello-world-durable-object/ts/{wrangler.json → wrangler.jsonc} +0 -0
  77. /package/templates/hono/templates/{wrangler.json → wrangler.jsonc} +0 -0
  78. /package/templates/next/{wrangler.json → wrangler.jsonc} +0 -0
  79. /package/templates/nuxt/templates/{wrangler.json → wrangler.jsonc} +0 -0
  80. /package/templates/openapi/ts/{wrangler.json → wrangler.jsonc} +0 -0
  81. /package/templates/queues/js/{wrangler.json → wrangler.jsonc} +0 -0
  82. /package/templates/queues/ts/{wrangler.json → wrangler.jsonc} +0 -0
  83. /package/templates/qwik/templates/{wrangler.json → wrangler.jsonc} +0 -0
  84. /package/templates/remix/templates/{wrangler.json → wrangler.jsonc} +0 -0
  85. /package/templates/scheduled/js/{wrangler.json → wrangler.jsonc} +0 -0
  86. /package/templates/scheduled/ts/{wrangler.json → wrangler.jsonc} +0 -0
  87. /package/templates/solid/templates/{wrangler.json → wrangler.jsonc} +0 -0
  88. /package/templates/svelte/templates/{wrangler.json → wrangler.jsonc} +0 -0
  89. /package/templates-experimental/angular/templates/{wrangler.json → wrangler.jsonc} +0 -0
  90. /package/templates-experimental/astro/templates/js/{wrangler.json → wrangler.jsonc} +0 -0
  91. /package/templates-experimental/astro/templates/ts/{wrangler.json → wrangler.jsonc} +0 -0
  92. /package/templates-experimental/docusaurus/templates/{wrangler.json → wrangler.jsonc} +0 -0
  93. /package/templates-experimental/gatsby/templates/{wrangler.json → wrangler.jsonc} +0 -0
  94. /package/templates-experimental/hello-world-assets-only/templates/{wrangler.json → wrangler.jsonc} +0 -0
  95. /package/templates-experimental/hello-world-durable-object-with-assets/js/{wrangler.json → wrangler.jsonc} +0 -0
  96. /package/templates-experimental/hello-world-durable-object-with-assets/ts/{wrangler.json → wrangler.jsonc} +0 -0
  97. /package/templates-experimental/hello-world-with-assets/js/{wrangler.json → wrangler.jsonc} +0 -0
  98. /package/templates-experimental/hello-world-with-assets/py/{wrangler.json → wrangler.jsonc} +0 -0
  99. /package/templates-experimental/hello-world-with-assets/ts/{wrangler.json → wrangler.jsonc} +0 -0
  100. /package/templates-experimental/hono/templates/{wrangler.json → wrangler.jsonc} +0 -0
  101. /package/templates-experimental/next/templates/{wrangler.json → wrangler.jsonc} +0 -0
  102. /package/templates-experimental/nuxt/templates/{wrangler.json → wrangler.jsonc} +0 -0
  103. /package/templates-experimental/qwik/templates/{wrangler.json → wrangler.jsonc} +0 -0
  104. /package/templates-experimental/solid/templates/{wrangler.json → wrangler.jsonc} +0 -0
  105. /package/templates-experimental/svelte/templates/{wrangler.json → wrangler.jsonc} +0 -0
  106. /package/templates-experimental/vue/templates/{wrangler.json → wrangler.jsonc} +0 -0
@@ -7,7 +7,7 @@ import { readFile, readJSON, writeFile } from "helpers/files";
7
7
  import { detectPackageManager } from "helpers/packageManagers";
8
8
  import { installPackages } from "helpers/packages";
9
9
  import type { TemplateConfig } from "../../src/templates";
10
- import type { C3Context } from "types";
10
+ import type { C3Context, PackageJson } from "types";
11
11
 
12
12
  const { npm } = detectPackageManager();
13
13
 
@@ -63,10 +63,10 @@ async function updateAppCode() {
63
63
  // Remove unwanted dependencies
64
64
  s.start(`Updating package.json`);
65
65
  const packageJsonPath = resolve("package.json");
66
- const packageManifest = readJSON(packageJsonPath);
66
+ const packageManifest = readJSON(packageJsonPath) as PackageJson;
67
67
 
68
- delete packageManifest["dependencies"]["express"];
69
- delete packageManifest["devDependencies"]["@types/express"];
68
+ delete packageManifest["dependencies"]?.["express"];
69
+ delete packageManifest["devDependencies"]?.["@types/express"];
70
70
 
71
71
  writeFile(packageJsonPath, JSON.stringify(packageManifest, null, 2));
72
72
  s.stop(`${brandColor(`updated`)} ${dim(`\`package.json\``)}`);
@@ -75,7 +75,7 @@ async function updateAppCode() {
75
75
  function updateAngularJson(ctx: C3Context) {
76
76
  const s = spinner();
77
77
  s.start(`Updating angular.json config`);
78
- const angularJson = readJSON(resolve("angular.json"));
78
+ const angularJson = readJSON(resolve("angular.json")) as AngularJson;
79
79
  // Update builder
80
80
  const architectSection = angularJson.projects[ctx.project.name].architect;
81
81
  architectSection.build.options.outputPath = "dist";
@@ -110,3 +110,21 @@ const config: TemplateConfig = {
110
110
  }),
111
111
  };
112
112
  export default config;
113
+
114
+ type AngularJson = {
115
+ projects: Record<
116
+ string,
117
+ {
118
+ architect: {
119
+ build: {
120
+ options: {
121
+ outputPath: string;
122
+ outputMode: string;
123
+ ssr: Record<string, unknown>;
124
+ assets: string[];
125
+ };
126
+ };
127
+ };
128
+ }
129
+ >;
130
+ };
@@ -1,4 +1,4 @@
1
1
  // Generated by Wrangler
2
- // After adding bindings to `wrangler.json`, regenerate this interface via `npm run cf-typegen`
2
+ // After adding bindings to `wrangler.jsonc`, regenerate this interface via `npm run cf-typegen`
3
3
  interface Env {
4
4
  }
@@ -28,5 +28,6 @@ const config: TemplateConfig = {
28
28
  }),
29
29
  devScript: "start",
30
30
  deployScript: "deploy",
31
+ previewScript: "preview",
31
32
  };
32
33
  export default config;
@@ -11,7 +11,7 @@ import { DurableObject } from "cloudflare:workers";
11
11
  */
12
12
 
13
13
  /**
14
- * Env provides a mechanism to reference bindings declared in wrangler.json within JavaScript
14
+ * Env provides a mechanism to reference bindings declared in wrangler.jsonc within JavaScript
15
15
  *
16
16
  * @typedef {Object} Env
17
17
  * @property {DurableObjectNamespace} MY_DURABLE_OBJECT - The Durable Object namespace binding
@@ -24,7 +24,7 @@ export class MyDurableObject extends DurableObject {
24
24
  * `DurableObjectStub::get` for a given identifier (no-op constructors can be omitted)
25
25
  *
26
26
  * @param {DurableObjectState} ctx - The interface for interacting with Durable Object state
27
- * @param {Env} env - The interface to reference bindings declared in wrangler.json
27
+ * @param {Env} env - The interface to reference bindings declared in wrangler.jsonc
28
28
  */
29
29
  constructor(ctx, env) {
30
30
  super(ctx, env);
@@ -47,7 +47,7 @@ export default {
47
47
  * This is the standard fetch handler for a Cloudflare Worker
48
48
  *
49
49
  * @param {Request} request - The request submitted to the Worker from the client
50
- * @param {Env} env - The interface to reference bindings declared in wrangler.json
50
+ * @param {Env} env - The interface to reference bindings declared in wrangler.jsonc
51
51
  * @param {ExecutionContext} ctx - The execution context of the Worker
52
52
  * @returns {Promise<Response>} The response to be sent back to the client
53
53
  */
@@ -7,7 +7,7 @@ import { DurableObject } from "cloudflare:workers";
7
7
  * - Open a browser tab at http://localhost:8787/ to see your Durable Object in action
8
8
  * - Run `npm run deploy` to publish your application
9
9
  *
10
- * Bind resources to your worker in `wrangler.json`. After adding bindings, a type definition for the
10
+ * Bind resources to your worker in `wrangler.jsonc`. After adding bindings, a type definition for the
11
11
  * `Env` object can be regenerated with `npm run cf-typegen`.
12
12
  *
13
13
  * Learn more at https://developers.cloudflare.com/durable-objects
@@ -21,7 +21,7 @@ export class MyDurableObject extends DurableObject {
21
21
  * `DurableObjectStub::get` for a given identifier (no-op constructors can be omitted)
22
22
  *
23
23
  * @param ctx - The interface for interacting with Durable Object state
24
- * @param env - The interface to reference bindings declared in wrangler.json
24
+ * @param env - The interface to reference bindings declared in wrangler.jsonc
25
25
  */
26
26
  constructor(ctx: DurableObjectState, env: Env) {
27
27
  super(ctx, env);
@@ -44,7 +44,7 @@ export default {
44
44
  * This is the standard fetch handler for a Cloudflare Worker
45
45
  *
46
46
  * @param request - The request submitted to the Worker from the client
47
- * @param env - The interface to reference bindings declared in wrangler.json
47
+ * @param env - The interface to reference bindings declared in wrangler.jsonc
48
48
  * @param ctx - The execution context of the Worker
49
49
  * @returns The response to be sent back to the client
50
50
  */
@@ -9,8 +9,8 @@
9
9
  "test": "vitest"
10
10
  },
11
11
  "devDependencies": {
12
- "@cloudflare/vitest-pool-workers": "^0.5.2",
12
+ "@cloudflare/vitest-pool-workers": "^0.6.4",
13
13
  "wrangler": "^3.101.0",
14
- "vitest": "2.1.8"
14
+ "vitest": "~2.1.9"
15
15
  }
16
16
  }
@@ -4,7 +4,7 @@ export default defineWorkersConfig({
4
4
  test: {
5
5
  poolOptions: {
6
6
  workers: {
7
- wrangler: { configPath: './wrangler.json' },
7
+ wrangler: { configPath: './wrangler.jsonc' },
8
8
  },
9
9
  },
10
10
  },
@@ -10,9 +10,9 @@
10
10
  "cf-typegen": "wrangler types"
11
11
  },
12
12
  "devDependencies": {
13
- "@cloudflare/vitest-pool-workers": "^0.5.2",
13
+ "@cloudflare/vitest-pool-workers": "^0.6.4",
14
14
  "typescript": "^5.5.2",
15
- "vitest": "2.1.8",
15
+ "vitest": "~2.1.9",
16
16
  "wrangler": "^3.101.0"
17
17
  }
18
18
  }
@@ -5,7 +5,7 @@
5
5
  * - Open a browser tab at http://localhost:8787/ to see your worker in action
6
6
  * - Run `npm run deploy` to publish your worker
7
7
  *
8
- * Bind resources to your worker in `wrangler.json`. After adding bindings, a type definition for the
8
+ * Bind resources to your worker in `wrangler.jsonc`. After adding bindings, a type definition for the
9
9
  * `Env` object can be regenerated with `npm run cf-typegen`.
10
10
  *
11
11
  * Learn more at https://developers.cloudflare.com/workers/
@@ -4,7 +4,7 @@ export default defineWorkersConfig({
4
4
  test: {
5
5
  poolOptions: {
6
6
  workers: {
7
- wrangler: { configPath: './wrangler.json' },
7
+ wrangler: { configPath: './wrangler.jsonc' },
8
8
  },
9
9
  },
10
10
  },
@@ -1,4 +1,4 @@
1
1
  // Generated by Wrangler
2
- // After adding bindings to `wrangler.json`, regenerate this interface via `npm run cf-typegen`
2
+ // After adding bindings to `wrangler.jsonc`, regenerate this interface via `npm run cf-typegen`
3
3
  interface Env {
4
4
  }
@@ -39,5 +39,6 @@ const config: TemplateConfig = {
39
39
  }),
40
40
  devScript: "dev",
41
41
  deployScript: "deploy",
42
+ previewScript: "dev",
42
43
  };
43
44
  export default config;
@@ -1,4 +1,4 @@
1
1
  // Generated by Wrangler
2
- // After adding bindings to `wrangler.json`, regenerate this interface via `npm run cf-typegen`
2
+ // After adding bindings to `wrangler.jsonc`, regenerate this interface via `npm run cf-typegen`
3
3
  interface CloudflareBindings {
4
4
  }
@@ -1,25 +1,18 @@
1
1
  import { brandColor, dim } from "@cloudflare/cli/colors";
2
+ import { spinner } from "@cloudflare/cli/interactive";
2
3
  import { runFrameworkGenerator } from "frameworks/index";
4
+ import { readFile, writeFile } from "helpers/files";
3
5
  import { installPackages } from "helpers/packages";
4
6
  import type { TemplateConfig } from "../../src/templates";
5
7
  import type { C3Context } from "types";
6
8
 
7
9
  const generate = async (ctx: C3Context) => {
8
- await runFrameworkGenerator(ctx, [
9
- ctx.project.name,
10
- "--ts",
11
- "--tailwind",
12
- "--eslint",
13
- "--app",
14
- "--import-alias",
15
- "@/*",
16
- "--src-dir",
17
- ]);
10
+ await runFrameworkGenerator(ctx, [ctx.project.name]);
18
11
  };
19
12
 
20
13
  const configure = async () => {
21
14
  const packages = [
22
- "@opennextjs/cloudflare@0.3.x",
15
+ "@opennextjs/cloudflare@0.5.x",
23
16
  "@cloudflare/workers-types",
24
17
  ];
25
18
  await installPackages(packages, {
@@ -27,6 +20,29 @@ const configure = async () => {
27
20
  startText: "Adding the Cloudflare adapter",
28
21
  doneText: `${brandColor(`installed`)} ${dim(packages.join(", "))}`,
29
22
  });
23
+
24
+ updateNextConfig();
25
+ };
26
+
27
+ const updateNextConfig = () => {
28
+ const s = spinner();
29
+
30
+ const configFile = "next.config.mjs";
31
+ s.start(`Updating \`${configFile}\``);
32
+
33
+ const configContent = readFile(configFile);
34
+
35
+ const updatedConfigFile =
36
+ configContent +
37
+ `
38
+ // added by create cloudflare to enable calling \`getCloudflareContext()\` in \`next dev\`
39
+ import { initOpenNextCloudflareForDev } from '@opennextjs/cloudflare';
40
+ initOpenNextCloudflareForDev();
41
+ `.replace(/\n\t*/g, "\n");
42
+
43
+ writeFile(configFile, updatedConfigFile);
44
+
45
+ s.stop(`${brandColor(`updated`)} ${dim(`\`${configFile}\``)}`);
30
46
  };
31
47
 
32
48
  export default {
@@ -1,4 +1,4 @@
1
1
  // Generated by Wrangler
2
- // After adding bindings to `wrangler.json`, regenerate this interface via `npm run cf-typegen`
2
+ // After adding bindings to `wrangler.jsonc`, regenerate this interface via `npm run cf-typegen`
3
3
  interface Env {
4
4
  }
@@ -10,7 +10,7 @@ import * as recast from "recast";
10
10
  import type { TemplateConfig } from "../../src/templates";
11
11
  import type { C3Context } from "types";
12
12
 
13
- const { npm, npx } = detectPackageManager();
13
+ const { npm, npx, name } = detectPackageManager();
14
14
 
15
15
  const generate = async (ctx: C3Context) => {
16
16
  await runFrameworkGenerator(ctx, ["playground", ctx.project.name]);
@@ -18,7 +18,8 @@ const generate = async (ctx: C3Context) => {
18
18
 
19
19
  const configure = async (ctx: C3Context) => {
20
20
  // Add the pages integration
21
- const cmd = [npx, "qwik", "add", "cloudflare-pages"];
21
+ // For some reason `pnpx qwik add` fails for qwik so we use `pnpm qwik add` instead.
22
+ const cmd = [name === "pnpm" ? npm : npx, "qwik", "add", "cloudflare-pages"];
22
23
  endSection(`Running ${quoteShellArgs(cmd)}`);
23
24
  await runCommand(cmd);
24
25
 
@@ -1,4 +1,4 @@
1
1
  // Generated by Wrangler
2
- // After adding bindings to `wrangler.json`, regenerate this interface via `npm run cf-typegen`
2
+ // After adding bindings to `wrangler.jsonc`, regenerate this interface via `npm run cf-typegen`
3
3
  interface Env {
4
4
  }
@@ -0,0 +1,182 @@
1
+ import assert from "assert";
2
+ import { logRaw } from "@cloudflare/cli";
3
+ import { brandColor, dim } from "@cloudflare/cli/colors";
4
+ import { inputPrompt, spinner } from "@cloudflare/cli/interactive";
5
+ import { runFrameworkGenerator } from "frameworks/index";
6
+ import { transformFile } from "helpers/codemod";
7
+ import { readJSON, usesTypescript, writeJSON } from "helpers/files";
8
+ import { detectPackageManager } from "helpers/packageManagers";
9
+ import { installPackages } from "helpers/packages";
10
+ import * as recast from "recast";
11
+ import type { TemplateConfig } from "../../src/templates";
12
+ import type { types } from "recast";
13
+ import type { C3Context } from "types";
14
+
15
+ const b = recast.types.builders;
16
+ const t = recast.types.namedTypes;
17
+ const { npm } = detectPackageManager();
18
+
19
+ const generate = async (ctx: C3Context) => {
20
+ const variant = await getVariant();
21
+ ctx.args.lang = variant.lang;
22
+
23
+ await runFrameworkGenerator(ctx, [
24
+ ctx.project.name,
25
+ "--template",
26
+ variant.value,
27
+ ]);
28
+
29
+ logRaw("");
30
+ };
31
+
32
+ const configure = async (ctx: C3Context) => {
33
+ await installPackages(["@cloudflare/vite-plugin"], {
34
+ dev: true,
35
+ startText: "Installing the Cloudflare Vite plugin",
36
+ doneText: `${brandColor(`installed`)} ${dim("@cloudflare/vite-plugin")}`,
37
+ });
38
+
39
+ await transformViteConfig(ctx);
40
+
41
+ if (usesTypescript(ctx)) {
42
+ updateTsconfigJson();
43
+ }
44
+ };
45
+
46
+ function transformViteConfig(ctx: C3Context) {
47
+ const filePath = `vite.config.${usesTypescript(ctx) ? "ts" : "js"}`;
48
+
49
+ transformFile(filePath, {
50
+ visitProgram(n) {
51
+ // Add an import of the @cloudflare/vite-plugin
52
+ // ```
53
+ // import {cloudflare} from "@cloudflare/vite-plugin";
54
+ // ```
55
+ const lastImportIndex = n.node.body.findLastIndex(
56
+ (statement) => statement.type === "ImportDeclaration",
57
+ );
58
+ const lastImport = n.get("body", lastImportIndex);
59
+ const importAst = b.importDeclaration(
60
+ [b.importSpecifier(b.identifier("cloudflare"))],
61
+ b.stringLiteral("@cloudflare/vite-plugin"),
62
+ );
63
+ lastImport.insertAfter(importAst);
64
+
65
+ return this.traverse(n);
66
+ },
67
+ visitCallExpression: function (n) {
68
+ // Add the imported plugin to the config
69
+ // ```
70
+ // defineConfig({
71
+ // plugins: [react(), cloudflare()],
72
+ // });
73
+ const callee = n.node.callee as types.namedTypes.Identifier;
74
+ if (callee.name !== "defineConfig") {
75
+ return this.traverse(n);
76
+ }
77
+
78
+ const config = n.node.arguments[0];
79
+ assert(t.ObjectExpression.check(config));
80
+ const pluginsProp = config.properties.find((prop) => isPluginsProp(prop));
81
+ assert(pluginsProp && t.ArrayExpression.check(pluginsProp.value));
82
+ pluginsProp.value.elements.push(
83
+ b.callExpression(b.identifier("cloudflare"), []),
84
+ );
85
+
86
+ return false;
87
+ },
88
+ });
89
+ }
90
+
91
+ function isPluginsProp(
92
+ prop: unknown,
93
+ ): prop is types.namedTypes.ObjectProperty | types.namedTypes.Property {
94
+ return (
95
+ (t.Property.check(prop) || t.ObjectProperty.check(prop)) &&
96
+ t.Identifier.check(prop.key) &&
97
+ prop.key.name === "plugins"
98
+ );
99
+ }
100
+
101
+ function updateTsconfigJson() {
102
+ const s = spinner();
103
+ s.start(`Updating tsconfig.json config`);
104
+ // Add a reference to the extra tsconfig.worker.json file.
105
+ // ```
106
+ // "references": [ ..., { path: "./tsconfig.worker.json" } ]
107
+ // ```
108
+ const tsconfig = readJSON("tsconfig.json") as { references: object[] };
109
+ if (tsconfig && typeof tsconfig === "object") {
110
+ tsconfig.references ??= [];
111
+ tsconfig.references.push({ path: "./tsconfig.worker.json" });
112
+ }
113
+ writeJSON("tsconfig.json", tsconfig);
114
+ s.stop(`${brandColor(`updated`)} ${dim(`\`tsconfig.json\``)}`);
115
+ }
116
+
117
+ async function getVariant() {
118
+ const variantsOptions = [
119
+ {
120
+ value: "react-ts",
121
+ lang: "ts",
122
+ label: "TypeScript",
123
+ },
124
+ {
125
+ value: "react-swc-ts",
126
+ lang: "ts",
127
+ label: "TypeScript + SWC",
128
+ },
129
+ {
130
+ value: "react",
131
+ lang: "js",
132
+ label: "JavaScript",
133
+ },
134
+ {
135
+ value: "react-swc",
136
+ lang: "js",
137
+ label: "JavaScript + SWC",
138
+ },
139
+ ];
140
+ const value = await inputPrompt({
141
+ type: "select",
142
+ question: "Select a variant:",
143
+ label: "variant",
144
+ options: variantsOptions,
145
+ defaultValue: variantsOptions[0].value,
146
+ });
147
+
148
+ const selected = variantsOptions.find((variant) => variant.value === value);
149
+ assert(selected, "Expected a variant to be selected");
150
+ return selected;
151
+ }
152
+
153
+ const config: TemplateConfig = {
154
+ configVersion: 1,
155
+ id: "react",
156
+ frameworkCli: "create-vite",
157
+ displayName: "React",
158
+ platform: "workers",
159
+ path: "templates-experimental/react",
160
+ copyFiles: {
161
+ variants: {
162
+ ts: {
163
+ path: "./ts",
164
+ },
165
+ js: {
166
+ path: "./js",
167
+ },
168
+ },
169
+ },
170
+ generate,
171
+ configure,
172
+ transformPackageJson: async () => ({
173
+ scripts: {
174
+ deploy: `${npm} run build && wrangler deploy`,
175
+ preview: `${npm} run build && vite preview`,
176
+ },
177
+ }),
178
+ devScript: "dev",
179
+ deployScript: "deploy",
180
+ previewScript: "preview",
181
+ };
182
+ export default config;
@@ -0,0 +1,13 @@
1
+ export default {
2
+ fetch(request, env) {
3
+ const url = new URL(request.url);
4
+
5
+ if (url.pathname.startsWith("/api/")) {
6
+ return Response.json({
7
+ name: "Cloudflare",
8
+ });
9
+ }
10
+
11
+ return env.ASSETS.fetch(request);
12
+ },
13
+ }
@@ -0,0 +1,58 @@
1
+ import { useState } from 'react'
2
+ import reactLogo from './assets/react.svg'
3
+ import viteLogo from '/vite.svg'
4
+ import cloudflareLogo from './assets/Cloudflare_Logo.svg'
5
+ import './App.css'
6
+
7
+ function App() {
8
+ const [count, setCount] = useState(0)
9
+ const [name, setName] = useState('unknown')
10
+
11
+ return (
12
+ <>
13
+ <div>
14
+ <a href='https://vite.dev' target='_blank'>
15
+ <img src={viteLogo} className='logo' alt='Vite logo' />
16
+ </a>
17
+ <a href='https://react.dev' target='_blank'>
18
+ <img src={reactLogo} className='logo react' alt='React logo' />
19
+ </a>
20
+ <a href='https://workers.cloudflare.com/' target='_blank'>
21
+ <img src={cloudflareLogo} className='logo cloudflare' alt='Cloudflare logo' />
22
+ </a>
23
+ </div>
24
+ <h1>Vite + React + Cloudflare</h1>
25
+ <div className='card'>
26
+ <button
27
+ onClick={() => setCount((count) => count + 1)}
28
+ aria-label='increment'
29
+ >
30
+ count is {count}
31
+ </button>
32
+ <p>
33
+ Edit <code>src/App.tsx</code> and save to test HMR
34
+ </p>
35
+ </div>
36
+ <div className='card'>
37
+ <button
38
+ onClick={() => {
39
+ fetch('/api/')
40
+ .then((res) => res.json())
41
+ .then((data) => setName(data.name))
42
+ }}
43
+ aria-label='get name'
44
+ >
45
+ Name from API is: {name}
46
+ </button>
47
+ <p>
48
+ Edit <code>api/index.js</code> to change the name
49
+ </p>
50
+ </div>
51
+ <p className='read-the-docs'>
52
+ Click on the Vite and React logos to learn more
53
+ </p>
54
+ </>
55
+ )
56
+ }
57
+
58
+ export default App
@@ -0,0 +1,26 @@
1
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 101.4 33.5">
2
+ <defs>
3
+ <style>
4
+ .a {
5
+ fill: #fff;
6
+ }
7
+
8
+ .b {
9
+ fill: #f48120;
10
+ }
11
+
12
+ .c {
13
+ fill: #faad3f;
14
+ }
15
+
16
+ .d {
17
+ fill: #404041;
18
+ }
19
+ </style>
20
+ </defs>
21
+ <title>Cloudflare logo</title>
22
+ <path class="a" d="M94.7,10.6,89.1,9.3l-1-.4-25.7.2V21.5l32.3.1Z"/>
23
+ <path class="b" d="M84.2,20.4a2.85546,2.85546,0,0,0-.3-2.6,3.09428,3.09428,0,0,0-2.1-1.1l-17.4-.2c-.1,0-.2-.1-.3-.1a.1875.1875,0,0,1,0-.3c.1-.2.2-.3.4-.3L82,15.6a6.29223,6.29223,0,0,0,5.1-3.8l1-2.6c0-.1.1-.2,0-.3A11.39646,11.39646,0,0,0,66.2,7.7a5.45941,5.45941,0,0,0-3.6-1A5.20936,5.20936,0,0,0,58,11.3a5.46262,5.46262,0,0,0,.1,1.8A7.30177,7.30177,0,0,0,51,20.4a4.102,4.102,0,0,0,.1,1.1.3193.3193,0,0,0,.3.3H83.5c.2,0,.4-.1.4-.3Z"/>
24
+ <path class="c" d="M89.7,9.2h-.5c-.1,0-.2.1-.3.2l-.7,2.4a2.85546,2.85546,0,0,0,.3,2.6,3.09428,3.09428,0,0,0,2.1,1.1l3.7.2c.1,0,.2.1.3.1a.1875.1875,0,0,1,0,.3c-.1.2-.2.3-.4.3l-3.8.2a6.29223,6.29223,0,0,0-5.1,3.8l-.2.9c-.1.1,0,.3.2.3H98.5a.26517.26517,0,0,0,.3-.3,10.87184,10.87184,0,0,0,.4-2.6,9.56045,9.56045,0,0,0-9.5-9.5"/>
25
+ <path class="d" d="M100.5,27.2a.9.9,0,1,1,.9-.9.89626.89626,0,0,1-.9.9m0-1.6a.7.7,0,1,0,.7.7.68354.68354,0,0,0-.7-.7m.4,1.2h-.2l-.2-.3h-.2v.3h-.2v-.9h.5a.26517.26517,0,0,1,.3.3c0,.1-.1.2-.2.3l.2.3Zm-.3-.5c.1,0,.1,0,.1-.1a.09794.09794,0,0,0-.1-.1h-.3v.3h.3Zm-89.7-.9h2.2v6h3.8v1.9h-6Zm8.3,3.9a4.10491,4.10491,0,0,1,4.3-4.1,4.02,4.02,0,0,1,4.2,4.1,4.10491,4.10491,0,0,1-4.3,4.1,4.07888,4.07888,0,0,1-4.2-4.1m6.3,0a2.05565,2.05565,0,0,0-2-2.2,2.1025,2.1025,0,0,0,0,4.2c1.2.2,2-.8,2-2m4.9.5V25.4h2.2v4.4c0,1.1.6,1.7,1.5,1.7a1.39926,1.39926,0,0,0,1.5-1.6V25.4h2.2v4.4c0,2.6-1.5,3.7-3.7,3.7-2.3-.1-3.7-1.2-3.7-3.7m10.7-4.4h3.1c2.8,0,4.5,1.6,4.5,3.9s-1.7,4-4.5,4h-3V25.4Zm3.1,5.9a2.00909,2.00909,0,1,0,0-4h-.9v4Zm7.6-5.9h6.3v1.9H54v1.3h3.7v1.8H54v2.9H51.8Zm9.4,0h2.2v6h3.8v1.9h-6Zm11.7-.1h2.2l3.4,8H76.1l-.6-1.4H72.4l-.6,1.4H69.5Zm2,4.9L74,28l-.9,2.2Zm6.4-4.8H85a3.41818,3.41818,0,0,1,2.6.9,2.62373,2.62373,0,0,1-.9,4.2l1.9,2.8H86.1l-1.6-2.4h-1v2.4H81.3Zm3.6,3.8c.7,0,1.2-.4,1.2-.9,0-.6-.5-.9-1.2-.9H83.5v1.9h1.4Zm6.5-3.8h6.4v1.8H93.6v1.2h3.8v1.8H93.6v1.2h4.3v1.9H91.4ZM6.1,30.3a1.97548,1.97548,0,0,1-1.8,1.2,2.1025,2.1025,0,0,1,0-4.2,2.0977,2.0977,0,0,1,1.9,1.3H8.5a4.13459,4.13459,0,0,0-4.2-3.3A4.1651,4.1651,0,0,0,0,29.4a4.07888,4.07888,0,0,0,4.2,4.1,4.31812,4.31812,0,0,0,4.2-3.2Z"/>
26
+ </svg>
@@ -0,0 +1,9 @@
1
+ {
2
+ "name": "<TBD>",
3
+ "main": "api/index.js",
4
+ "compatibility_date": "<TBD>",
5
+ "assets": { "not_found_handling": "single-page-application", "binding": "ASSETS" },
6
+ "observability": {
7
+ "enabled": true
8
+ }
9
+ }
@@ -0,0 +1,17 @@
1
+ interface Env {
2
+ ASSETS: Fetcher;
3
+ }
4
+
5
+ export default {
6
+ fetch(request, env) {
7
+ const url = new URL(request.url);
8
+
9
+ if (url.pathname.startsWith("/api/")) {
10
+ return Response.json({
11
+ name: "Cloudflare",
12
+ });
13
+ }
14
+
15
+ return env.ASSETS.fetch(request);
16
+ },
17
+ } satisfies ExportedHandler<Env>;
@@ -0,0 +1,58 @@
1
+ import { useState } from 'react'
2
+ import reactLogo from './assets/react.svg'
3
+ import viteLogo from '/vite.svg'
4
+ import cloudflareLogo from './assets/Cloudflare_Logo.svg'
5
+ import './App.css'
6
+
7
+ function App() {
8
+ const [count, setCount] = useState(0)
9
+ const [name, setName] = useState('unknown')
10
+
11
+ return (
12
+ <>
13
+ <div>
14
+ <a href='https://vite.dev' target='_blank'>
15
+ <img src={viteLogo} className='logo' alt='Vite logo' />
16
+ </a>
17
+ <a href='https://react.dev' target='_blank'>
18
+ <img src={reactLogo} className='logo react' alt='React logo' />
19
+ </a>
20
+ <a href='https://workers.cloudflare.com/' target='_blank'>
21
+ <img src={cloudflareLogo} className='logo cloudflare' alt='Cloudflare logo' />
22
+ </a>
23
+ </div>
24
+ <h1>Vite + React + Cloudflare</h1>
25
+ <div className='card'>
26
+ <button
27
+ onClick={() => setCount((count) => count + 1)}
28
+ aria-label='increment'
29
+ >
30
+ count is {count}
31
+ </button>
32
+ <p>
33
+ Edit <code>src/App.tsx</code> and save to test HMR
34
+ </p>
35
+ </div>
36
+ <div className='card'>
37
+ <button
38
+ onClick={() => {
39
+ fetch('/api/')
40
+ .then((res) => res.json() as Promise<{ name: string }>)
41
+ .then((data) => setName(data.name))
42
+ }}
43
+ aria-label='get name'
44
+ >
45
+ Name from API is: {name}
46
+ </button>
47
+ <p>
48
+ Edit <code>api/index.ts</code> to change the name
49
+ </p>
50
+ </div>
51
+ <p className='read-the-docs'>
52
+ Click on the Vite and React logos to learn more
53
+ </p>
54
+ </>
55
+ )
56
+ }
57
+
58
+ export default App