create-cloudflare 0.0.0-e7ccc859 → 0.0.0-e7ea6005c

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 (221) hide show
  1. package/README.md +1 -1
  2. package/dist/cli.js +59166 -56638
  3. package/dist/tsconfig.tsbuildinfo +1 -0
  4. package/package.json +29 -24
  5. package/templates/analog/c3.ts +135 -0
  6. package/templates/analog/snippets/devBindingsModule.ts +7 -0
  7. package/templates/analog/templates/env.d.ts +13 -0
  8. package/templates/analog/templates/src/dev-bindings.ts +18 -0
  9. package/templates/analog/templates/worker-configuration.d.ts +4 -0
  10. package/templates/analog/templates/wrangler.toml +85 -0
  11. package/templates/angular/c3.ts +14 -11
  12. package/templates/astro/c3.ts +83 -20
  13. package/templates/astro/snippets/runtimeDeclaration.ts +5 -0
  14. package/templates/astro/templates/wrangler.toml +85 -0
  15. package/templates/common/c3.ts +10 -5
  16. package/templates/common/js/.editorconfig +0 -1
  17. package/templates/common/js/package.json +1 -1
  18. package/templates/common/js/wrangler.toml +84 -22
  19. package/templates/common/ts/.editorconfig +0 -1
  20. package/templates/common/ts/package.json +4 -3
  21. package/templates/common/ts/src/ab-test.ts +2 -2
  22. package/templates/common/ts/src/index.ts +5 -2
  23. package/templates/common/ts/src/proxy.ts +2 -2
  24. package/templates/common/ts/src/redirect.ts +2 -2
  25. package/templates/common/ts/tsconfig.json +37 -95
  26. package/templates/common/ts/worker-configuration.d.ts +2 -14
  27. package/templates/common/ts/wrangler.toml +84 -22
  28. package/templates/docusaurus/c3.ts +7 -6
  29. package/templates/gatsby/c3.ts +8 -5
  30. package/templates/hello-world/c3.ts +12 -6
  31. package/templates/hello-world/js/.editorconfig +0 -1
  32. package/templates/hello-world/js/package.json +5 -2
  33. package/templates/hello-world/js/test/index.spec.js +20 -0
  34. package/templates/hello-world/js/vitest.config.js +11 -0
  35. package/templates/hello-world/js/wrangler.toml +85 -22
  36. package/templates/hello-world/py/__dot__gitignore +68 -0
  37. package/templates/hello-world/py/package.json +13 -0
  38. package/templates/hello-world/py/src/entry.py +4 -0
  39. package/templates/hello-world/py/wrangler.toml +114 -0
  40. package/templates/hello-world/ts/.editorconfig +0 -1
  41. package/templates/hello-world/ts/package.json +7 -3
  42. package/templates/hello-world/ts/src/index.ts +5 -19
  43. package/templates/hello-world/ts/test/index.spec.ts +25 -0
  44. package/templates/hello-world/ts/test/tsconfig.json +8 -0
  45. package/templates/hello-world/ts/tsconfig.json +39 -96
  46. package/templates/hello-world/ts/vitest.config.mts +11 -0
  47. package/templates/hello-world/ts/worker-configuration.d.ts +4 -0
  48. package/templates/hello-world/ts/wrangler.toml +85 -22
  49. package/templates/hello-world-durable-object/c3.ts +10 -6
  50. package/templates/hello-world-durable-object/js/.editorconfig +0 -1
  51. package/templates/hello-world-durable-object/js/package.json +1 -1
  52. package/templates/hello-world-durable-object/js/src/index.js +18 -14
  53. package/templates/hello-world-durable-object/js/wrangler.toml +84 -22
  54. package/templates/hello-world-durable-object/ts/.editorconfig +0 -1
  55. package/templates/hello-world-durable-object/ts/package.json +4 -4
  56. package/templates/hello-world-durable-object/ts/src/index.ts +24 -37
  57. package/templates/hello-world-durable-object/ts/tsconfig.json +36 -95
  58. package/templates/hello-world-durable-object/ts/worker-configuration.d.ts +5 -0
  59. package/templates/hello-world-durable-object/ts/wrangler.toml +84 -22
  60. package/templates/hono/c3.ts +45 -1
  61. package/templates/hono/snippets/appDeclaration.ts +1 -0
  62. package/templates/hono/templates/worker-configuration.d.ts +4 -0
  63. package/templates/hono/templates/wrangler.toml +113 -0
  64. package/templates/next/README.md +68 -0
  65. package/templates/next/app/js/app/api/hello/route.js +21 -0
  66. package/templates/next/app/js/app/not-found.js +58 -0
  67. package/templates/next/app/ts/app/api/hello/route.ts +22 -0
  68. package/templates/next/app/ts/app/not-found.tsx +58 -0
  69. package/templates/next/c3.ts +109 -84
  70. package/templates/next/env.d.ts +5 -0
  71. package/templates/next/pages/js/pages/api/hello.js +23 -0
  72. package/templates/next/pages/ts/pages/api/hello.ts +24 -0
  73. package/templates/next/wrangler.toml +86 -0
  74. package/templates/nuxt/c3.ts +94 -19
  75. package/templates/nuxt/templates/env.d.ts +14 -0
  76. package/templates/nuxt/templates/worker-configuration.d.ts +4 -0
  77. package/templates/nuxt/templates/wrangler.toml +86 -0
  78. package/templates/openapi/c3.ts +1 -0
  79. package/templates/openapi/ts/README.md +3 -3
  80. package/templates/openapi/ts/package.json +7 -4
  81. package/templates/openapi/ts/src/endpoints/taskCreate.ts +26 -16
  82. package/templates/openapi/ts/src/endpoints/taskDelete.ts +20 -19
  83. package/templates/openapi/ts/src/endpoints/taskFetch.ts +30 -23
  84. package/templates/openapi/ts/src/endpoints/taskList.ts +27 -24
  85. package/templates/openapi/ts/src/index.ts +14 -20
  86. package/templates/openapi/ts/src/types.ts +9 -8
  87. package/templates/openapi/ts/worker-configuration.d.ts +4 -0
  88. package/templates/openapi/ts/wrangler.toml +110 -0
  89. package/templates/pre-existing/c3.ts +34 -15
  90. package/templates/pre-existing/js/.editorconfig +0 -1
  91. package/templates/pre-existing/js/package.json +1 -1
  92. package/templates/pre-existing/js/wrangler.toml +1 -0
  93. package/templates/queues/c3.ts +9 -5
  94. package/templates/queues/js/.editorconfig +0 -1
  95. package/templates/queues/js/package.json +1 -1
  96. package/templates/queues/js/wrangler.toml +101 -2
  97. package/templates/queues/ts/.editorconfig +0 -1
  98. package/templates/queues/ts/package.json +4 -3
  99. package/templates/queues/ts/src/index.ts +6 -8
  100. package/templates/queues/ts/tsconfig.json +36 -95
  101. package/templates/queues/ts/worker-configuration.d.ts +5 -0
  102. package/templates/queues/ts/wrangler.toml +101 -2
  103. package/templates/qwik/c3.ts +85 -30
  104. package/templates/qwik/snippets/getPlatformProxy.ts +6 -0
  105. package/templates/qwik/templates/worker-configuration.d.ts +4 -0
  106. package/templates/qwik/templates/wrangler.toml +86 -0
  107. package/templates/react/c3.ts +38 -6
  108. package/templates/remix/c3.ts +40 -6
  109. package/templates/remix/templates/worker-configuration.d.ts +4 -0
  110. package/templates/remix/templates/wrangler.toml +85 -0
  111. package/templates/scheduled/c3.ts +9 -5
  112. package/templates/scheduled/js/.editorconfig +0 -1
  113. package/templates/scheduled/js/package.json +3 -3
  114. package/templates/scheduled/js/src/index.js +8 -1
  115. package/templates/scheduled/js/wrangler.toml +110 -0
  116. package/templates/scheduled/ts/.editorconfig +0 -1
  117. package/templates/scheduled/ts/package.json +5 -4
  118. package/templates/scheduled/ts/src/index.ts +6 -23
  119. package/templates/scheduled/ts/tsconfig.json +36 -95
  120. package/templates/scheduled/ts/worker-configuration.d.ts +4 -0
  121. package/templates/scheduled/ts/wrangler.toml +110 -0
  122. package/templates/solid/c3.ts +61 -11
  123. package/templates/solid/templates/wrangler.toml +84 -0
  124. package/templates/svelte/c3.ts +84 -30
  125. package/templates/svelte/js/wrangler.toml +85 -0
  126. package/templates/svelte/ts/wrangler.toml +85 -0
  127. package/templates/vue/c3.ts +8 -6
  128. package/templates-experimental/angular/c3.ts +98 -0
  129. package/templates-experimental/angular/templates/server.ts +34 -0
  130. package/templates-experimental/angular/templates/tools/alter-polyfills.mjs +32 -0
  131. package/templates-experimental/angular/templates/wrangler.toml +11 -0
  132. package/templates-experimental/astro/c3.ts +114 -0
  133. package/templates-experimental/astro/snippets/runtimeDeclaration.ts +5 -0
  134. package/templates-experimental/astro/templates/public/.assetsignore +4 -0
  135. package/templates-experimental/astro/templates/wrangler.toml +12 -0
  136. package/templates-experimental/docusaurus/c3.ts +32 -0
  137. package/templates-experimental/docusaurus/templates/wrangler.toml +10 -0
  138. package/templates-experimental/gatsby/c3.ts +53 -0
  139. package/templates-experimental/gatsby/templates/wrangler.toml +10 -0
  140. package/templates-experimental/hello-world-assets-only/c3.ts +15 -0
  141. package/templates-experimental/hello-world-assets-only/templates/package.json +13 -0
  142. package/templates-experimental/hello-world-assets-only/templates/public/index.html +11 -0
  143. package/templates-experimental/hello-world-assets-only/templates/wrangler.toml +10 -0
  144. package/templates-experimental/hello-world-durable-object-with-assets/c3.ts +23 -0
  145. package/templates-experimental/hello-world-durable-object-with-assets/js/.editorconfig +12 -0
  146. package/templates-experimental/hello-world-durable-object-with-assets/js/.prettierrc +6 -0
  147. package/templates-experimental/hello-world-durable-object-with-assets/js/__dot__gitignore +172 -0
  148. package/templates-experimental/hello-world-durable-object-with-assets/js/package.json +13 -0
  149. package/templates-experimental/hello-world-durable-object-with-assets/js/public/index.html +19 -0
  150. package/templates-experimental/hello-world-durable-object-with-assets/js/src/index.js +69 -0
  151. package/templates-experimental/hello-world-durable-object-with-assets/js/wrangler.toml +24 -0
  152. package/templates-experimental/hello-world-durable-object-with-assets/ts/.editorconfig +12 -0
  153. package/templates-experimental/hello-world-durable-object-with-assets/ts/.prettierrc +6 -0
  154. package/templates-experimental/hello-world-durable-object-with-assets/ts/__dot__gitignore +172 -0
  155. package/templates-experimental/hello-world-durable-object-with-assets/ts/package.json +15 -0
  156. package/templates-experimental/hello-world-durable-object-with-assets/ts/public/index.html +19 -0
  157. package/templates-experimental/hello-world-durable-object-with-assets/ts/src/index.ts +66 -0
  158. package/templates-experimental/hello-world-durable-object-with-assets/ts/tsconfig.json +42 -0
  159. package/templates-experimental/hello-world-durable-object-with-assets/ts/worker-configuration.d.ts +5 -0
  160. package/templates-experimental/hello-world-durable-object-with-assets/ts/wrangler.toml +24 -0
  161. package/templates-experimental/hello-world-with-assets/c3.ts +22 -0
  162. package/templates-experimental/hello-world-with-assets/js/.editorconfig +12 -0
  163. package/templates-experimental/hello-world-with-assets/js/.prettierrc +6 -0
  164. package/templates-experimental/hello-world-with-assets/js/__dot__gitignore +172 -0
  165. package/templates-experimental/hello-world-with-assets/js/package.json +16 -0
  166. package/templates-experimental/hello-world-with-assets/js/public/index.html +31 -0
  167. package/templates-experimental/hello-world-with-assets/js/src/index.js +23 -0
  168. package/templates-experimental/hello-world-with-assets/js/test/index.spec.js +41 -0
  169. package/templates-experimental/hello-world-with-assets/js/vitest.config.js +11 -0
  170. package/templates-experimental/hello-world-with-assets/js/wrangler.toml +12 -0
  171. package/templates-experimental/hello-world-with-assets/py/__dot__gitignore +68 -0
  172. package/templates-experimental/hello-world-with-assets/py/package.json +13 -0
  173. package/templates-experimental/hello-world-with-assets/py/public/index.html +31 -0
  174. package/templates-experimental/hello-world-with-assets/py/src/entry.py +9 -0
  175. package/templates-experimental/hello-world-with-assets/py/wrangler.toml +12 -0
  176. package/templates-experimental/hello-world-with-assets/ts/.editorconfig +12 -0
  177. package/templates-experimental/hello-world-with-assets/ts/.prettierrc +6 -0
  178. package/templates-experimental/hello-world-with-assets/ts/__dot__gitignore +172 -0
  179. package/templates-experimental/hello-world-with-assets/ts/package.json +18 -0
  180. package/templates-experimental/hello-world-with-assets/ts/public/index.html +31 -0
  181. package/templates-experimental/hello-world-with-assets/ts/src/index.ts +26 -0
  182. package/templates-experimental/hello-world-with-assets/ts/test/index.spec.ts +41 -0
  183. package/templates-experimental/hello-world-with-assets/ts/test/tsconfig.json +8 -0
  184. package/templates-experimental/hello-world-with-assets/ts/tsconfig.json +44 -0
  185. package/templates-experimental/hello-world-with-assets/ts/vitest.config.mts +11 -0
  186. package/templates-experimental/hello-world-with-assets/ts/worker-configuration.d.ts +4 -0
  187. package/templates-experimental/hello-world-with-assets/ts/wrangler.toml +12 -0
  188. package/templates-experimental/hono/c3.ts +43 -0
  189. package/templates-experimental/hono/templates/public/index.html +19 -0
  190. package/templates-experimental/hono/templates/src/index.ts +9 -0
  191. package/templates-experimental/hono/templates/worker-configuration.d.ts +4 -0
  192. package/templates-experimental/hono/templates/wrangler.toml +5 -0
  193. package/templates-experimental/next/c3.ts +52 -0
  194. package/templates-experimental/next/templates/__dot__gitignore +42 -0
  195. package/templates-experimental/next/templates/env.d.ts +5 -0
  196. package/templates-experimental/next/templates/wrangler.toml +12 -0
  197. package/templates-experimental/nuxt/c3.ts +135 -0
  198. package/templates-experimental/nuxt/templates/cloudflare-preset/nitro.config.ts +27 -0
  199. package/templates-experimental/nuxt/templates/env.d.ts +14 -0
  200. package/templates-experimental/nuxt/templates/worker-configuration.d.ts +4 -0
  201. package/templates-experimental/nuxt/templates/wrangler.toml +11 -0
  202. package/templates-experimental/qwik/c3.ts +153 -0
  203. package/templates-experimental/qwik/snippets/getPlatformProxy.ts +6 -0
  204. package/templates-experimental/qwik/templates/public/.assetsignore +4 -0
  205. package/templates-experimental/qwik/templates/worker-configuration.d.ts +4 -0
  206. package/templates-experimental/qwik/templates/wrangler.toml +12 -0
  207. package/templates-experimental/remix/c3.ts +77 -0
  208. package/templates-experimental/remix/templates/public/.assetsignore +4 -0
  209. package/templates-experimental/remix/templates/worker-configuration.d.ts +4 -0
  210. package/templates-experimental/remix/templates/wrangler.toml +11 -0
  211. package/templates-experimental/solid/c3.ts +137 -0
  212. package/templates-experimental/solid/templates/wrangler.toml +12 -0
  213. package/templates-experimental/svelte/c3.ts +132 -0
  214. package/templates-experimental/svelte/js/static/.assetsignore +4 -0
  215. package/templates-experimental/svelte/js/wrangler.toml +11 -0
  216. package/templates-experimental/svelte/ts/static/.assetsignore +4 -0
  217. package/templates-experimental/svelte/ts/wrangler.toml +11 -0
  218. package/templates/next/templates.ts +0 -281
  219. package/templates/solid/js/vite.config.js +0 -12
  220. package/templates/solid/ts/vite.config.ts +0 -12
  221. package/templates/svelte/templates.ts +0 -13
@@ -1,31 +1,23 @@
1
- import { existsSync, mkdirSync } from "fs";
2
- import { crash, updateStatus, warn } from "@cloudflare/cli";
3
- import { processArgument } from "@cloudflare/cli/args";
1
+ import { join } from "path";
2
+ import { updateStatus, warn } from "@cloudflare/cli";
4
3
  import { brandColor, dim } from "@cloudflare/cli/colors";
5
- import { installPackages, runFrameworkGenerator } from "helpers/command";
4
+ import { inputPrompt, spinner } from "@cloudflare/cli/interactive";
5
+ import { runFrameworkGenerator } from "frameworks/index";
6
6
  import {
7
- compatDateFlag,
7
+ copyFile,
8
8
  probePaths,
9
+ readFile,
9
10
  readJSON,
10
11
  usesEslint,
11
12
  usesTypescript,
12
13
  writeFile,
13
14
  writeJSON,
14
15
  } from "helpers/files";
15
- import { detectPackageManager } from "helpers/packages";
16
- import {
17
- apiAppDirHelloJs,
18
- apiAppDirHelloTs,
19
- apiPagesDirHelloJs,
20
- apiPagesDirHelloTs,
21
- appDirNotFoundJs,
22
- appDirNotFoundTs,
23
- envDts,
24
- nextConfig,
25
- readme,
26
- } from "./templates";
16
+ import { detectPackageManager } from "helpers/packageManagers";
17
+ import { installPackages } from "helpers/packages";
18
+ import { getTemplatePath } from "../../src/templates";
27
19
  import type { TemplateConfig } from "../../src/templates";
28
- import type { C3Args, C3Context } from "types";
20
+ import type { C3Context } from "types";
29
21
 
30
22
  const { npm, npx } = detectPackageManager();
31
23
 
@@ -33,27 +25,53 @@ const generate = async (ctx: C3Context) => {
33
25
  const projectName = ctx.project.name;
34
26
 
35
27
  await runFrameworkGenerator(ctx, [projectName]);
36
- };
37
28
 
38
- const getApiTemplate = (
39
- apiPath: string,
40
- isTypescript: boolean
41
- ): [string, string] => {
42
- const isAppDir = /\/app\/api$/.test(apiPath);
29
+ const wranglerToml = readFile(join(getTemplatePath(ctx), "wrangler.toml"));
43
30
 
44
- if (isAppDir) {
45
- // App directory uses route handlers that are defined in a subdirectory (`/api/hello/route.ts`).
46
- const routeHandlerPath = `${apiPath}/hello`;
47
- mkdirSync(routeHandlerPath, { recursive: true });
31
+ // Note: here we add `# KV Example:` to the toml file for the KV example, we don't actually
32
+ // include the comment in the template wrangler.toml file just so to keep it identical
33
+ // and consistent with that of all the other frameworks
34
+ // (instead of making it a special case which needs extra care)
35
+ const newTomlContent = wranglerToml.replace(
36
+ /#\s+\[\[kv_namespaces\]\]\n#\s+binding\s+=\s+"MY_KV_NAMESPACE"\n#\s+id\s+=\s+"[a-zA-Z0-9]+?"/,
37
+ ($1) => `# KV Example:\n${$1}`,
38
+ );
48
39
 
49
- return isTypescript
50
- ? [`${routeHandlerPath}/route.ts`, apiAppDirHelloTs]
51
- : [`${routeHandlerPath}/route.js`, apiAppDirHelloJs];
40
+ if (!/# KV Example/.test(newTomlContent)) {
41
+ // This should never happen to users, it is a check mostly so that
42
+ // if the toml file is changed in a way that breaks the "KV Example" addition
43
+ // the C3 Next.js e2e runs will fail with this
44
+ throw new Error("Failed to properly generate the wrangler.toml file");
52
45
  }
53
46
 
54
- return isTypescript
55
- ? [`${apiPath}/hello.ts`, apiPagesDirHelloTs]
56
- : [`${apiPath}/hello.js`, apiPagesDirHelloJs];
47
+ writeFile(join(ctx.project.path, "wrangler.toml"), newTomlContent);
48
+
49
+ updateStatus("Created wrangler.toml file");
50
+ };
51
+
52
+ const updateNextConfig = () => {
53
+ const s = spinner();
54
+
55
+ const configFile = "next.config.mjs";
56
+ s.start(`Updating \`${configFile}\``);
57
+
58
+ const configContent = readFile(configFile);
59
+
60
+ const updatedConfigFile =
61
+ `import { setupDevPlatform } from '@cloudflare/next-on-pages/next-dev';
62
+
63
+ // Here we use the @cloudflare/next-on-pages next-dev module to allow us to use bindings during local development
64
+ // (when running the application with \`next dev\`), for more information see:
65
+ // https://github.com/cloudflare/next-on-pages/blob/main/internal-packages/next-dev/README.md
66
+ if (process.env.NODE_ENV === 'development') {
67
+ await setupDevPlatform();
68
+ }
69
+
70
+ `.replace(/\n\t*/g, "\n") + configContent;
71
+
72
+ writeFile(configFile, updatedConfigFile);
73
+
74
+ s.stop(`${brandColor(`updated`)} ${dim(`\`${configFile}\``)}`);
57
75
  };
58
76
 
59
77
  const configure = async (ctx: C3Context) => {
@@ -70,39 +88,16 @@ const configure = async (ctx: C3Context) => {
70
88
  ]);
71
89
 
72
90
  if (!path) {
73
- crash("Could not find the `/api` or `/app` directory");
91
+ throw new Error("Could not find the `/api` or `/app` directory");
74
92
  }
75
93
 
76
- // App directory template may not generate an API route handler, so we update the path to add an `api` directory.
77
- const apiPath = path.replace(/\/app$/, "/app/api");
78
-
79
94
  const usesTs = usesTypescript(ctx);
80
95
 
81
- const appDirPath = probePaths([
82
- `${projectPath}/src/app`,
83
- `${projectPath}/app`,
84
- ]);
85
-
86
- if (appDirPath) {
87
- // Add a custom app not-found edge route as recommended in next-on-pages
88
- // (see: https://github.com/cloudflare/next-on-pages/blob/2b5c8f25/packages/next-on-pages/docs/gotchas.md#not-found)
89
- const notFoundPath = `${appDirPath}/not-found.${usesTs ? "tsx" : "js"}`;
90
- if (!existsSync(notFoundPath)) {
91
- const notFoundContent = usesTs ? appDirNotFoundTs : appDirNotFoundJs;
92
- writeFile(notFoundPath, notFoundContent);
93
- updateStatus("Created a custom edge not-found route");
94
- }
95
- }
96
-
97
- const [handlerPath, handlerFile] = getApiTemplate(
98
- apiPath,
99
- usesTypescript(ctx)
100
- );
101
- writeFile(handlerPath, handlerFile);
102
- updateStatus("Created an example API route handler");
103
-
104
96
  if (usesTs) {
105
- writeFile(`${projectPath}/env.d.ts`, envDts);
97
+ copyFile(
98
+ join(getTemplatePath(ctx), "env.d.ts"),
99
+ join(projectPath, "env.d.ts"),
100
+ );
106
101
  updateStatus("Created an env.d.ts file");
107
102
  }
108
103
 
@@ -112,30 +107,34 @@ const configure = async (ctx: C3Context) => {
112
107
  await writeEslintrc(ctx);
113
108
  }
114
109
 
115
- writeFile(`${projectPath}/next.config.mjs`, nextConfig);
116
- updateStatus("Updated the next.config.js file");
110
+ updateNextConfig();
117
111
 
118
- writeFile(`${projectPath}/README.md`, readme);
112
+ copyFile(
113
+ join(getTemplatePath(ctx), "README.md"),
114
+ join(projectPath, "README.md"),
115
+ );
119
116
  updateStatus("Updated the README file");
120
117
 
121
118
  await addDevDependencies(installEslintPlugin);
122
119
  };
123
120
 
124
121
  export const shouldInstallNextOnPagesEslintPlugin = async (
125
- ctx: C3Context
122
+ ctx: C3Context,
126
123
  ): Promise<boolean> => {
127
124
  const eslintUsage = usesEslint(ctx);
128
125
 
129
- if (!eslintUsage.used) return false;
126
+ if (!eslintUsage.used) {
127
+ return false;
128
+ }
130
129
 
131
130
  if (eslintUsage.configType !== ".eslintrc.json") {
132
131
  warn(
133
- `Expected .eslintrc.json from Next.js scaffolding but found ${eslintUsage.configType} instead`
132
+ `Expected .eslintrc.json from Next.js scaffolding but found ${eslintUsage.configType} instead`,
134
133
  );
135
134
  return false;
136
135
  }
137
136
 
138
- return await processArgument(ctx.args, "eslint-plugin" as keyof C3Args, {
137
+ return await inputPrompt({
139
138
  type: "confirm",
140
139
  question: "Do you want to use the next-on-pages eslint-plugin?",
141
140
  label: "eslint-plugin",
@@ -175,13 +174,43 @@ const addDevDependencies = async (installEslintPlugin: boolean) => {
175
174
  export default {
176
175
  configVersion: 1,
177
176
  id: "next",
177
+ frameworkCli: "create-next-app",
178
178
  platform: "pages",
179
179
  displayName: "Next",
180
- devScript: "dev",
181
- previewScript: "pages:preview",
182
180
  generate,
183
181
  configure,
184
- transformPackageJson: async () => {
182
+ copyFiles: {
183
+ async selectVariant(ctx) {
184
+ const isApp = probePaths([
185
+ `${ctx.project.path}/src/app`,
186
+ `${ctx.project.path}/app`,
187
+ ]);
188
+
189
+ const isTypescript = usesTypescript(ctx);
190
+
191
+ const dir = isApp ? "app" : "pages";
192
+ return `${dir}/${isTypescript ? "ts" : "js"}`;
193
+ },
194
+ destinationDir(ctx) {
195
+ const srcPath = probePaths([`${ctx.project.path}/src`]);
196
+ return srcPath ? "./src" : "./";
197
+ },
198
+ variants: {
199
+ "app/ts": {
200
+ path: "./app/ts",
201
+ },
202
+ "app/js": {
203
+ path: "./app/js",
204
+ },
205
+ "pages/ts": {
206
+ path: "./pages/ts",
207
+ },
208
+ "pages/js": {
209
+ path: "./pages/js",
210
+ },
211
+ },
212
+ },
213
+ transformPackageJson: async (_, ctx) => {
185
214
  const isNpm = npm === "npm";
186
215
  const isBun = npm === "bun";
187
216
  const isNpmOrBun = isNpm || isBun;
@@ -194,20 +223,16 @@ export default {
194
223
  return {
195
224
  scripts: {
196
225
  "pages:build": `${pmCommand} ${nextOnPagesCommand}`,
197
- "pages:preview": `${pagesBuildRunCommand} && wrangler pages dev .vercel/output/static ${await compatDateFlag()} --compatibility-flag=nodejs_compat`,
198
- "pages:deploy": `${pagesBuildRunCommand} && wrangler pages deploy .vercel/output/static`,
226
+ preview: `${pagesBuildRunCommand} && wrangler pages dev`,
227
+ deploy: `${pagesBuildRunCommand} && wrangler pages deploy`,
228
+ ...(usesTypescript(ctx) && {
229
+ "cf-typegen": `wrangler types --env-interface CloudflareEnv env.d.ts`,
230
+ }),
199
231
  },
200
232
  };
201
233
  },
202
- testFlags: [
203
- "--typescript",
204
- "--no-install",
205
- "--eslint",
206
- "--tailwind",
207
- "--src-dir",
208
- "--app",
209
- "--import-alias",
210
- "@/*",
211
- ],
234
+ devScript: "dev",
235
+ previewScript: "preview",
236
+ deployScript: "deploy",
212
237
  compatibilityFlags: ["nodejs_compat"],
213
238
  } as TemplateConfig;
@@ -0,0 +1,5 @@
1
+ // Generated by Wrangler
2
+ // by running `wrangler types --env-interface CloudflareEnv env.d.ts`
3
+
4
+ interface CloudflareEnv {
5
+ }
@@ -0,0 +1,23 @@
1
+ import { getRequestContext } from '@cloudflare/next-on-pages'
2
+
3
+ export const config = {
4
+ runtime: 'edge',
5
+ }
6
+
7
+ export default async function handler(req) {
8
+ let responseText = 'Hello World'
9
+
10
+ // In the edge runtime you can use Bindings that are available in your application
11
+ // (for more details see:
12
+ // - https://developers.cloudflare.com/pages/framework-guides/deploy-a-nextjs-site/#use-bindings-in-your-nextjs-application
13
+ // - https://developers.cloudflare.com/pages/functions/bindings/
14
+ // )
15
+ //
16
+ // KV Example:
17
+ // const myKv = getRequestContext().env.MY_KV_NAMESPACE
18
+ // await myKv.put('suffix', ' from a KV store!')
19
+ // const suffix = await myKv.get('suffix')
20
+ // responseText += suffix
21
+
22
+ return new Response(responseText)
23
+ }
@@ -0,0 +1,24 @@
1
+ import type { NextRequest } from 'next/server'
2
+ import { getRequestContext } from '@cloudflare/next-on-pages'
3
+
4
+ export const config = {
5
+ runtime: 'edge',
6
+ }
7
+
8
+ export default async function handler(req: NextRequest) {
9
+ let responseText = 'Hello World'
10
+
11
+ // In the edge runtime you can use Bindings that are available in your application
12
+ // (for more details see:
13
+ // - https://developers.cloudflare.com/pages/framework-guides/deploy-a-nextjs-site/#use-bindings-in-your-nextjs-application
14
+ // - https://developers.cloudflare.com/pages/functions/bindings/
15
+ // )
16
+ //
17
+ // KV Example:
18
+ // const myKv = getRequestContext().env.MY_KV_NAMESPACE
19
+ // await myKv.put('suffix', ' from a KV store!')
20
+ // const suffix = await myKv.get('suffix')
21
+ // responseText += suffix
22
+
23
+ return new Response(responseText)
24
+ }
@@ -0,0 +1,86 @@
1
+ #:schema node_modules/wrangler/config-schema.json
2
+ name = "<TBD>"
3
+ compatibility_date = "<TBD>"
4
+ compatibility_flags = ["nodejs_compat"]
5
+ pages_build_output_dir = ".vercel/output/static"
6
+
7
+ # Automatically place your workloads in an optimal location to minimize latency.
8
+ # If you are running back-end logic in a Pages Function, running it closer to your back-end infrastructure
9
+ # rather than the end user may result in better performance.
10
+ # Docs: https://developers.cloudflare.com/pages/functions/smart-placement/#smart-placement
11
+ # [placement]
12
+ # mode = "smart"
13
+
14
+ # Variable bindings. These are arbitrary, plaintext strings (similar to environment variables)
15
+ # Docs:
16
+ # - https://developers.cloudflare.com/pages/functions/bindings/#environment-variables
17
+ # Note: Use secrets to store sensitive data.
18
+ # - https://developers.cloudflare.com/pages/functions/bindings/#secrets
19
+ # [vars]
20
+ # MY_VARIABLE = "production_value"
21
+
22
+ # Bind the Workers AI model catalog. Run machine learning models, powered by serverless GPUs, on Cloudflare’s global network
23
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#workers-ai
24
+ # [ai]
25
+ # binding = "AI"
26
+
27
+ # Bind a D1 database. D1 is Cloudflare’s native serverless SQL database.
28
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#d1-databases
29
+ # [[d1_databases]]
30
+ # binding = "MY_DB"
31
+ # database_name = "my-database"
32
+ # database_id = "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
33
+
34
+ # Bind a Durable Object. Durable objects are a scale-to-zero compute primitive based on the actor model.
35
+ # Durable Objects can live for as long as needed. Use these when you need a long-running "server", such as in realtime apps.
36
+ # Docs: https://developers.cloudflare.com/workers/runtime-apis/durable-objects
37
+ # [[durable_objects.bindings]]
38
+ # name = "MY_DURABLE_OBJECT"
39
+ # class_name = "MyDurableObject"
40
+ # script_name = 'my-durable-object'
41
+
42
+ # Bind a KV Namespace. Use KV as persistent storage for small key-value pairs.
43
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#kv-namespaces
44
+ # [[kv_namespaces]]
45
+ # binding = "MY_KV_NAMESPACE"
46
+ # id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
47
+
48
+ # Bind a Queue producer. Use this binding to schedule an arbitrary task that may be processed later by a Queue consumer.
49
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#queue-producers
50
+ # [[queues.producers]]
51
+ # binding = "MY_QUEUE"
52
+ # queue = "my-queue"
53
+
54
+ # Bind an R2 Bucket. Use R2 to store arbitrarily large blobs of data, such as files.
55
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#r2-buckets
56
+ # [[r2_buckets]]
57
+ # binding = "MY_BUCKET"
58
+ # bucket_name = "my-bucket"
59
+
60
+ # Bind another Worker service. Use this binding to call another Worker without network overhead.
61
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#service-bindings
62
+ # [[services]]
63
+ # binding = "MY_SERVICE"
64
+ # service = "my-service"
65
+
66
+ # To use different bindings for preview and production environments, follow the examples below.
67
+ # When using environment-specific overrides for bindings, ALL bindings must be specified on a per-environment basis.
68
+ # Docs: https://developers.cloudflare.com/pages/functions/wrangler-configuration#environment-specific-overrides
69
+
70
+ ######## PREVIEW environment config ########
71
+
72
+ # [env.preview.vars]
73
+ # API_KEY = "xyz789"
74
+
75
+ # [[env.preview.kv_namespaces]]
76
+ # binding = "MY_KV_NAMESPACE"
77
+ # id = "<PREVIEW_NAMESPACE_ID>"
78
+
79
+ ######## PRODUCTION environment config ########
80
+
81
+ # [env.production.vars]
82
+ # API_KEY = "abc123"
83
+
84
+ # [[env.production.kv_namespaces]]
85
+ # binding = "MY_KV_NAMESPACE"
86
+ # id = "<PRODUCTION_NAMESPACE_ID>"
@@ -1,15 +1,17 @@
1
- import { readFileSync } from "node:fs";
2
- import { resolve } from "node:path";
3
1
  import { logRaw } from "@cloudflare/cli";
4
2
  import { brandColor, dim } from "@cloudflare/cli/colors";
5
3
  import { spinner } from "@cloudflare/cli/interactive";
6
- import { runFrameworkGenerator } from "helpers/command";
7
- import { compatDateFlag, writeFile } from "helpers/files";
8
- import { detectPackageManager } from "helpers/packages";
4
+ import { runFrameworkGenerator } from "frameworks/index";
5
+ import { mergeObjectProperties, transformFile } from "helpers/codemod";
6
+ import { getLatestTypesEntrypoint } from "helpers/compatDate";
7
+ import { readFile, writeFile } from "helpers/files";
8
+ import { detectPackageManager } from "helpers/packageManagers";
9
+ import { installPackages } from "helpers/packages";
10
+ import * as recast from "recast";
9
11
  import type { TemplateConfig } from "../../src/templates";
10
12
  import type { C3Context } from "types";
11
13
 
12
- const { npm } = detectPackageManager();
14
+ const { npm, name: pm } = detectPackageManager();
13
15
 
14
16
  const generate = async (ctx: C3Context) => {
15
17
  const gitFlag = ctx.args.git ? `--gitInit` : `--no-gitInit`;
@@ -27,33 +29,106 @@ const generate = async (ctx: C3Context) => {
27
29
  logRaw(""); // newline
28
30
  };
29
31
 
30
- const configure = async () => {
31
- const configFileName = "nuxt.config.ts";
32
- const configFilePath = resolve(configFileName);
32
+ const configure = async (ctx: C3Context) => {
33
+ const packages = ["nitro-cloudflare-dev"];
34
+
35
+ // When using pnpm, explicitly add h3 package so the H3Event type declaration can be updated.
36
+ // Package managers other than pnpm will hoist the dependency, as will pnpm with `--shamefully-hoist`
37
+ if (pm === "pnpm") {
38
+ packages.push("h3");
39
+ }
40
+
41
+ await installPackages(packages, {
42
+ dev: true,
43
+ startText: "Installing nitro module `nitro-cloudflare-dev`",
44
+ doneText: `${brandColor("installed")} ${dim(`via \`${npm} install\``)}`,
45
+ });
46
+ updateNuxtConfig();
47
+
48
+ updateEnvTypes(ctx);
49
+ };
50
+
51
+ const updateEnvTypes = (ctx: C3Context) => {
52
+ const filepath = "env.d.ts";
53
+
54
+ const s = spinner();
55
+ s.start(`Updating ${filepath}`);
56
+
57
+ let file = readFile(filepath);
58
+
59
+ let typesEntrypoint = `@cloudflare/workers-types`;
60
+ const latestEntrypoint = getLatestTypesEntrypoint(ctx);
61
+ if (latestEntrypoint) {
62
+ typesEntrypoint += `/${latestEntrypoint}`;
63
+ }
64
+
65
+ // Replace placeholder with actual types entrypoint
66
+ file = file.replace("WORKERS_TYPES_ENTRYPOINT", typesEntrypoint);
67
+ writeFile("env.d.ts", file);
68
+
69
+ s.stop(`${brandColor(`updated`)} ${dim(`\`${filepath}\``)}`);
70
+ };
71
+
72
+ const updateNuxtConfig = () => {
33
73
  const s = spinner();
34
- s.start(`Updating \`${configFileName}\``);
35
- // Add the cloudflare preset into the configuration file.
36
- const originalConfigFile = readFileSync(configFilePath, "utf8");
37
- const updatedConfigFile = originalConfigFile.replace(
38
- "defineNuxtConfig({",
39
- "defineNuxtConfig({\n nitro: {\n preset: 'cloudflare-pages'\n },"
74
+
75
+ const configFile = "nuxt.config.ts";
76
+ s.start(`Updating \`${configFile}\``);
77
+
78
+ const b = recast.types.builders;
79
+
80
+ const presetDef = b.objectProperty(
81
+ b.identifier("nitro"),
82
+ b.objectExpression([
83
+ b.objectProperty(
84
+ b.identifier("preset"),
85
+ b.stringLiteral("cloudflare-pages"),
86
+ ),
87
+ ]),
40
88
  );
41
- writeFile(configFilePath, updatedConfigFile);
42
- s.stop(`${brandColor(`updated`)} ${dim(`\`${configFileName}\``)}`);
89
+
90
+ const moduleDef = b.objectProperty(
91
+ b.identifier("modules"),
92
+ b.arrayExpression([b.stringLiteral("nitro-cloudflare-dev")]),
93
+ );
94
+
95
+ transformFile(configFile, {
96
+ visitCallExpression: function (n) {
97
+ const callee = n.node.callee as recast.types.namedTypes.Identifier;
98
+ if (callee.name === "defineNuxtConfig") {
99
+ mergeObjectProperties(
100
+ n.node.arguments[0] as recast.types.namedTypes.ObjectExpression,
101
+ [presetDef, moduleDef],
102
+ );
103
+ }
104
+
105
+ return this.traverse(n);
106
+ },
107
+ });
108
+
109
+ s.stop(`${brandColor(`updated`)} ${dim(`\`${configFile}\``)}`);
43
110
  };
44
111
 
45
112
  const config: TemplateConfig = {
46
113
  configVersion: 1,
47
114
  id: "nuxt",
115
+ frameworkCli: "nuxi",
48
116
  platform: "pages",
49
117
  displayName: "Nuxt",
118
+ copyFiles: {
119
+ path: "./templates",
120
+ },
50
121
  generate,
51
122
  configure,
52
123
  transformPackageJson: async () => ({
53
124
  scripts: {
54
- "pages:dev": `wrangler pages dev ${await compatDateFlag()} --proxy 3000 -- ${npm} run dev`,
55
- "pages:deploy": `${npm} run build && wrangler pages deploy ./dist`,
125
+ deploy: `${npm} run build && wrangler pages deploy`,
126
+ preview: `${npm} run build && wrangler pages dev`,
127
+ "cf-typegen": `wrangler types`,
56
128
  },
57
129
  }),
130
+ devScript: "dev",
131
+ deployScript: "deploy",
132
+ previewScript: "preview",
58
133
  };
59
134
  export default config;
@@ -0,0 +1,14 @@
1
+ /// <reference types="WORKERS_TYPES_ENTRYPOINT" />
2
+
3
+ declare module "h3" {
4
+ interface H3EventContext {
5
+ cf: CfProperties;
6
+ cloudflare: {
7
+ request: Request;
8
+ env: Env;
9
+ context: ExecutionContext;
10
+ };
11
+ }
12
+ }
13
+
14
+ export {};
@@ -0,0 +1,4 @@
1
+ // Generated by Wrangler
2
+ // After adding bindings to `wrangler.toml`, regenerate this interface via `npm run cf-typegen`
3
+ interface Env {
4
+ }
@@ -0,0 +1,86 @@
1
+ #:schema node_modules/wrangler/config-schema.json
2
+ name = "<TBD>"
3
+ compatibility_date = "<TBD>"
4
+ compatibility_flags = ["nodejs_compat"]
5
+ pages_build_output_dir = "./dist"
6
+
7
+ # Automatically place your workloads in an optimal location to minimize latency.
8
+ # If you are running back-end logic in a Pages Function, running it closer to your back-end infrastructure
9
+ # rather than the end user may result in better performance.
10
+ # Docs: https://developers.cloudflare.com/pages/functions/smart-placement/#smart-placement
11
+ # [placement]
12
+ # mode = "smart"
13
+
14
+ # Variable bindings. These are arbitrary, plaintext strings (similar to environment variables)
15
+ # Docs:
16
+ # - https://developers.cloudflare.com/pages/functions/bindings/#environment-variables
17
+ # Note: Use secrets to store sensitive data.
18
+ # - https://developers.cloudflare.com/pages/functions/bindings/#secrets
19
+ # [vars]
20
+ # MY_VARIABLE = "production_value"
21
+
22
+ # Bind the Workers AI model catalog. Run machine learning models, powered by serverless GPUs, on Cloudflare’s global network
23
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#workers-ai
24
+ # [ai]
25
+ # binding = "AI"
26
+
27
+ # Bind a D1 database. D1 is Cloudflare’s native serverless SQL database.
28
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#d1-databases
29
+ # [[d1_databases]]
30
+ # binding = "MY_DB"
31
+ # database_name = "my-database"
32
+ # database_id = "xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx"
33
+
34
+ # Bind a Durable Object. Durable objects are a scale-to-zero compute primitive based on the actor model.
35
+ # Durable Objects can live for as long as needed. Use these when you need a long-running "server", such as in realtime apps.
36
+ # Docs: https://developers.cloudflare.com/workers/runtime-apis/durable-objects
37
+ # [[durable_objects.bindings]]
38
+ # name = "MY_DURABLE_OBJECT"
39
+ # class_name = "MyDurableObject"
40
+ # script_name = 'my-durable-object'
41
+
42
+ # Bind a KV Namespace. Use KV as persistent storage for small key-value pairs.
43
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#kv-namespaces
44
+ # [[kv_namespaces]]
45
+ # binding = "MY_KV_NAMESPACE"
46
+ # id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
47
+
48
+ # Bind a Queue producer. Use this binding to schedule an arbitrary task that may be processed later by a Queue consumer.
49
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#queue-producers
50
+ # [[queues.producers]]
51
+ # binding = "MY_QUEUE"
52
+ # queue = "my-queue"
53
+
54
+ # Bind an R2 Bucket. Use R2 to store arbitrarily large blobs of data, such as files.
55
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#r2-buckets
56
+ # [[r2_buckets]]
57
+ # binding = "MY_BUCKET"
58
+ # bucket_name = "my-bucket"
59
+
60
+ # Bind another Worker service. Use this binding to call another Worker without network overhead.
61
+ # Docs: https://developers.cloudflare.com/pages/functions/bindings/#service-bindings
62
+ # [[services]]
63
+ # binding = "MY_SERVICE"
64
+ # service = "my-service"
65
+
66
+ # To use different bindings for preview and production environments, follow the examples below.
67
+ # When using environment-specific overrides for bindings, ALL bindings must be specified on a per-environment basis.
68
+ # Docs: https://developers.cloudflare.com/pages/functions/wrangler-configuration#environment-specific-overrides
69
+
70
+ ######## PREVIEW environment config ########
71
+
72
+ # [env.preview.vars]
73
+ # API_KEY = "xyz789"
74
+
75
+ # [[env.preview.kv_namespaces]]
76
+ # binding = "MY_KV_NAMESPACE"
77
+ # id = "<PREVIEW_NAMESPACE_ID>"
78
+
79
+ ######## PRODUCTION environment config ########
80
+
81
+ # [env.production.vars]
82
+ # API_KEY = "abc123"
83
+
84
+ # [[env.production.kv_namespaces]]
85
+ # binding = "MY_KV_NAMESPACE"
86
+ # id = "<PRODUCTION_NAMESPACE_ID>"
@@ -2,6 +2,7 @@ export default {
2
2
  configVersion: 1,
3
3
  id: "openapi",
4
4
  displayName: "API starter (OpenAPI compliant)",
5
+ description: "Get started building a basic API on Workers",
5
6
  platform: "workers",
6
7
  copyFiles: {
7
8
  path: "./ts",