create-cloudflare 0.0.0-e61dba50 → 0.0.0-e640ed9d

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 (113) hide show
  1. package/dist/cli.js +30511 -28086
  2. package/dist/tsconfig.tsbuildinfo +1 -0
  3. package/package.json +11 -10
  4. package/templates/analog/c3.ts +134 -0
  5. package/templates/analog/snippets/devBindingsModule.ts +7 -0
  6. package/templates/analog/templates/env.d.ts +13 -0
  7. package/templates/analog/templates/src/dev-bindings.ts +18 -0
  8. package/templates/analog/templates/worker-configuration.d.ts +4 -0
  9. package/templates/analog/templates/wrangler.toml +85 -0
  10. package/templates/angular/c3.ts +12 -11
  11. package/templates/astro/c3.ts +82 -20
  12. package/templates/astro/snippets/runtimeDeclaration.ts +5 -0
  13. package/templates/astro/templates/wrangler.toml +85 -0
  14. package/templates/common/c3.ts +7 -5
  15. package/templates/common/js/.editorconfig +0 -1
  16. package/templates/common/js/wrangler.toml +78 -22
  17. package/templates/common/ts/.editorconfig +0 -1
  18. package/templates/common/ts/package.json +2 -1
  19. package/templates/common/ts/src/index.ts +3 -0
  20. package/templates/common/ts/worker-configuration.d.ts +2 -14
  21. package/templates/common/ts/wrangler.toml +78 -22
  22. package/templates/docusaurus/c3.ts +5 -6
  23. package/templates/gatsby/c3.ts +7 -5
  24. package/templates/hello-world/c3.ts +7 -5
  25. package/templates/hello-world/js/.editorconfig +0 -1
  26. package/templates/hello-world/js/package.json +5 -2
  27. package/templates/hello-world/js/test/index.spec.js +20 -0
  28. package/templates/hello-world/js/vitest.config.js +11 -0
  29. package/templates/hello-world/js/wrangler.toml +79 -22
  30. package/templates/hello-world/ts/.editorconfig +0 -1
  31. package/templates/hello-world/ts/package.json +6 -2
  32. package/templates/hello-world/ts/src/index.ts +3 -17
  33. package/templates/hello-world/ts/test/index.spec.ts +25 -0
  34. package/templates/hello-world/ts/test/tsconfig.json +8 -0
  35. package/templates/hello-world/ts/tsconfig.json +3 -2
  36. package/templates/hello-world/ts/vitest.config.ts +11 -0
  37. package/templates/hello-world/ts/worker-configuration.d.ts +4 -0
  38. package/templates/hello-world/ts/wrangler.toml +79 -22
  39. package/templates/hello-world-durable-object/c3.ts +7 -5
  40. package/templates/hello-world-durable-object/js/.editorconfig +0 -1
  41. package/templates/hello-world-durable-object/js/src/index.js +18 -14
  42. package/templates/hello-world-durable-object/js/wrangler.toml +78 -22
  43. package/templates/hello-world-durable-object/ts/.editorconfig +0 -1
  44. package/templates/hello-world-durable-object/ts/package.json +2 -2
  45. package/templates/hello-world-durable-object/ts/src/index.ts +24 -16
  46. package/templates/hello-world-durable-object/ts/worker-configuration.d.ts +6 -0
  47. package/templates/hello-world-durable-object/ts/wrangler.toml +78 -22
  48. package/templates/hello-world-python/c3.ts +9 -0
  49. package/templates/hello-world-python/py/__dot__gitignore +68 -0
  50. package/templates/hello-world-python/py/package.json +13 -0
  51. package/templates/hello-world-python/py/src/entry.py +4 -0
  52. package/templates/hello-world-python/py/wrangler.toml +108 -0
  53. package/templates/hono/c3.ts +54 -1
  54. package/templates/hono/snippets/appDeclaration.ts +1 -0
  55. package/templates/hono/snippets/bindingsType.ts +3 -0
  56. package/templates/hono/templates/worker-configuration.d.ts +4 -0
  57. package/templates/hono/templates/wrangler.toml +107 -0
  58. package/templates/next/README.md +68 -0
  59. package/templates/next/app/js/app/api/hello/route.js +21 -0
  60. package/templates/next/app/js/app/not-found.js +58 -0
  61. package/templates/next/app/ts/app/api/hello/route.ts +22 -0
  62. package/templates/next/app/ts/app/not-found.tsx +58 -0
  63. package/templates/next/c3.ts +104 -79
  64. package/templates/next/env.d.ts +5 -0
  65. package/templates/next/pages/js/pages/api/hello.js +23 -0
  66. package/templates/next/pages/ts/pages/api/hello.ts +24 -0
  67. package/templates/next/wrangler.toml +86 -0
  68. package/templates/nuxt/c3.ts +94 -19
  69. package/templates/nuxt/templates/env.d.ts +14 -0
  70. package/templates/nuxt/templates/worker-configuration.d.ts +4 -0
  71. package/templates/nuxt/templates/wrangler.toml +85 -0
  72. package/templates/openapi/ts/package.json +2 -1
  73. package/templates/openapi/ts/worker-configuration.d.ts +4 -0
  74. package/templates/openapi/ts/wrangler.toml +104 -0
  75. package/templates/pre-existing/c3.ts +7 -7
  76. package/templates/pre-existing/js/.editorconfig +0 -1
  77. package/templates/pre-existing/js/wrangler.toml +1 -0
  78. package/templates/queues/c3.ts +7 -5
  79. package/templates/queues/js/.editorconfig +0 -1
  80. package/templates/queues/js/wrangler.toml +95 -2
  81. package/templates/queues/ts/.editorconfig +0 -1
  82. package/templates/queues/ts/package.json +2 -1
  83. package/templates/queues/ts/src/index.ts +3 -5
  84. package/templates/queues/ts/worker-configuration.d.ts +5 -0
  85. package/templates/queues/ts/wrangler.toml +95 -2
  86. package/templates/qwik/c3.ts +118 -8
  87. package/templates/qwik/snippets/getPlatformProxy.ts +6 -0
  88. package/templates/qwik/templates/worker-configuration.d.ts +4 -0
  89. package/templates/qwik/templates/wrangler.toml +85 -0
  90. package/templates/react/c3.ts +7 -5
  91. package/templates/remix/c3.ts +39 -6
  92. package/templates/remix/templates/worker-configuration.d.ts +4 -0
  93. package/templates/remix/templates/wrangler.toml +85 -0
  94. package/templates/scheduled/c3.ts +7 -5
  95. package/templates/scheduled/js/.editorconfig +0 -1
  96. package/templates/scheduled/js/package.json +2 -2
  97. package/templates/scheduled/js/src/index.js +8 -1
  98. package/templates/scheduled/js/wrangler.toml +104 -0
  99. package/templates/scheduled/ts/.editorconfig +0 -1
  100. package/templates/scheduled/ts/package.json +3 -2
  101. package/templates/scheduled/ts/src/index.ts +4 -21
  102. package/templates/scheduled/ts/worker-configuration.d.ts +4 -0
  103. package/templates/scheduled/ts/wrangler.toml +104 -0
  104. package/templates/solid/c3.ts +58 -11
  105. package/templates/solid/templates/wrangler.toml +84 -0
  106. package/templates/svelte/c3.ts +83 -30
  107. package/templates/svelte/js/wrangler.toml +85 -0
  108. package/templates/svelte/ts/wrangler.toml +85 -0
  109. package/templates/vue/c3.ts +7 -6
  110. package/templates/next/templates.ts +0 -281
  111. package/templates/solid/js/vite.config.js +0 -12
  112. package/templates/solid/ts/vite.config.ts +0 -12
  113. package/templates/svelte/templates.ts +0 -13
@@ -0,0 +1,68 @@
1
+ This is a [Next.js](https://nextjs.org/) project bootstrapped with [`c3`](https://developers.cloudflare.com/pages/get-started/c3).
2
+
3
+ ## Getting Started
4
+
5
+ First, run the development server:
6
+
7
+ ```bash
8
+ npm run dev
9
+ # or
10
+ yarn dev
11
+ # or
12
+ pnpm dev
13
+ # or
14
+ bun dev
15
+ ```
16
+
17
+ Open [http://localhost:3000](http://localhost:3000) with your browser to see the result.
18
+
19
+ ## Cloudflare integration
20
+
21
+ Besides the `dev` script mentioned above `c3` has added a few extra scripts that allow you to integrate the application with the [Cloudflare Pages](https://pages.cloudflare.com/) environment, these are:
22
+ - `pages:build` to build the application for Pages using the [`@cloudflare/next-on-pages`](https://github.com/cloudflare/next-on-pages) CLI
23
+ - `preview` to locally preview your Pages application using the [Wrangler](https://developers.cloudflare.com/workers/wrangler/) CLI
24
+ - `deploy` to deploy your Pages application using the [Wrangler](https://developers.cloudflare.com/workers/wrangler/) CLI
25
+
26
+ > __Note:__ while the `dev` script is optimal for local development you should preview your Pages application as well (periodically or before deployments) in order to make sure that it can properly work in the Pages environment (for more details see the [`@cloudflare/next-on-pages` recommended workflow](https://github.com/cloudflare/next-on-pages/blob/05b6256/internal-packages/next-dev/README.md#recommended-workflow))
27
+
28
+ ### Bindings
29
+
30
+ Cloudflare [Bindings](https://developers.cloudflare.com/pages/functions/bindings/) are what allows you to interact with resources available in the Cloudflare Platform.
31
+
32
+ You can use bindings during development, when previewing locally your application and of course in the deployed application:
33
+
34
+ - To use bindings in dev mode you need to define them in the `next.config.js` file under `setupDevBindings`, this mode uses the `next-dev` `@cloudflare/next-on-pages` submodule. For more details see its [documentation](https://github.com/cloudflare/next-on-pages/blob/05b6256/internal-packages/next-dev/README.md).
35
+
36
+ - To use bindings in the preview mode you need to add them to the `pages:preview` script accordingly to the `wrangler pages dev` command. For more details see its [documentation](https://developers.cloudflare.com/workers/wrangler/commands/#dev-1) or the [Pages Bindings documentation](https://developers.cloudflare.com/pages/functions/bindings/).
37
+
38
+ - To use bindings in the deployed application you will need to configure them in the Cloudflare [dashboard](https://dash.cloudflare.com/). For more details see the [Pages Bindings documentation](https://developers.cloudflare.com/pages/functions/bindings/).
39
+
40
+ #### KV Example
41
+
42
+ `c3` has added for you an example showing how you can use a KV binding.
43
+
44
+ In order to enable the example:
45
+ - Search for javascript/typescript lines containing the following comment:
46
+ ```ts
47
+ // KV Example:
48
+ ```
49
+ and uncomment the commented lines below it.
50
+ - Do the same in the `wrangler.toml` file, where
51
+ the comment is:
52
+ ```
53
+ # KV Example:
54
+ ```
55
+ - If you're using TypeScript run the `cf-typegen` script to update the `env.d.ts` file:
56
+ ```bash
57
+ npm run cf-typegen
58
+ # or
59
+ yarn cf-typegen
60
+ # or
61
+ pnpm cf-typegen
62
+ # or
63
+ bun cf-typegen
64
+ ```
65
+
66
+ After doing this you can run the `dev` or `preview` script and visit the `/api/hello` route to see the example in action.
67
+
68
+ Finally, if you also want to see the example work in the deployed application make sure to add a `MY_KV_NAMESPACE` binding to your Pages application in its [dashboard kv bindings settings section](https://dash.cloudflare.com/?to=/:account/pages/view/:pages-project/settings/functions#kv_namespace_bindings_section). After having configured it make sure to re-deploy your application.
@@ -0,0 +1,21 @@
1
+ import { getRequestContext } from '@cloudflare/next-on-pages'
2
+
3
+ export const runtime = 'edge'
4
+
5
+ export async function GET(request) {
6
+ let responseText = 'Hello World'
7
+
8
+ // In the edge runtime you can use Bindings that are available in your application
9
+ // (for more details see:
10
+ // - https://developers.cloudflare.com/pages/framework-guides/deploy-a-nextjs-site/#use-bindings-in-your-nextjs-application
11
+ // - https://developers.cloudflare.com/pages/functions/bindings/
12
+ // )
13
+ //
14
+ // KV Example:
15
+ // const myKv = getRequestContext().env.MY_KV_NAMESPACE
16
+ // await myKv.put('suffix', ' from a KV store!')
17
+ // const suffix = await myKv.get('suffix')
18
+ // responseText += suffix
19
+
20
+ return new Response(responseText)
21
+ }
@@ -0,0 +1,58 @@
1
+ export const runtime = "edge";
2
+
3
+ export default function NotFound() {
4
+ return (
5
+ <>
6
+ <title>404: This page could not be found.</title>
7
+ <div style={styles.error}>
8
+ <div>
9
+ <style
10
+ dangerouslySetInnerHTML={{
11
+ __html: `body{color:#000;background:#fff;margin:0}.next-error-h1{border-right:1px solid rgba(0,0,0,.3)}@media (prefers-color-scheme:dark){body{color:#fff;background:#000}.next-error-h1{border-right:1px solid rgba(255,255,255,.3)}}`,
12
+ }}
13
+ />
14
+ <h1 className="next-error-h1" style={styles.h1}>
15
+ 404
16
+ </h1>
17
+ <div style={styles.desc}>
18
+ <h2 style={styles.h2}>This page could not be found.</h2>
19
+ </div>
20
+ </div>
21
+ </div>
22
+ </>
23
+ );
24
+ }
25
+
26
+ const styles = {
27
+ error: {
28
+ fontFamily:
29
+ 'system-ui,"Segoe UI",Roboto,Helvetica,Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji"',
30
+ height: "100vh",
31
+ textAlign: "center",
32
+ display: "flex",
33
+ flexDirection: "column",
34
+ alignItems: "center",
35
+ justifyContent: "center",
36
+ },
37
+
38
+ desc: {
39
+ display: "inline-block",
40
+ },
41
+
42
+ h1: {
43
+ display: "inline-block",
44
+ margin: "0 20px 0 0",
45
+ padding: "0 23px 0 0",
46
+ fontSize: 24,
47
+ fontWeight: 500,
48
+ verticalAlign: "top",
49
+ lineHeight: "49px",
50
+ },
51
+
52
+ h2: {
53
+ fontSize: 14,
54
+ fontWeight: 400,
55
+ lineHeight: "49px",
56
+ margin: 0,
57
+ },
58
+ };
@@ -0,0 +1,22 @@
1
+ import type { NextRequest } from 'next/server'
2
+ import { getRequestContext } from '@cloudflare/next-on-pages'
3
+
4
+ export const runtime = 'edge'
5
+
6
+ export async function GET(request: NextRequest) {
7
+ let responseText = 'Hello World'
8
+
9
+ // In the edge runtime you can use Bindings that are available in your application
10
+ // (for more details see:
11
+ // - https://developers.cloudflare.com/pages/framework-guides/deploy-a-nextjs-site/#use-bindings-in-your-nextjs-application
12
+ // - https://developers.cloudflare.com/pages/functions/bindings/
13
+ // )
14
+ //
15
+ // KV Example:
16
+ // const myKv = getRequestContext().env.MY_KV_NAMESPACE
17
+ // await myKv.put('suffix', ' from a KV store!')
18
+ // const suffix = await myKv.get('suffix')
19
+ // responseText += suffix
20
+
21
+ return new Response(responseText)
22
+ }
@@ -0,0 +1,58 @@
1
+ export const runtime = "edge";
2
+
3
+ export default function NotFound() {
4
+ return (
5
+ <>
6
+ <title>404: This page could not be found.</title>
7
+ <div style={styles.error}>
8
+ <div>
9
+ <style
10
+ dangerouslySetInnerHTML={{
11
+ __html: `body{color:#000;background:#fff;margin:0}.next-error-h1{border-right:1px solid rgba(0,0,0,.3)}@media (prefers-color-scheme:dark){body{color:#fff;background:#000}.next-error-h1{border-right:1px solid rgba(255,255,255,.3)}}`,
12
+ }}
13
+ />
14
+ <h1 className="next-error-h1" style={styles.h1}>
15
+ 404
16
+ </h1>
17
+ <div style={styles.desc}>
18
+ <h2 style={styles.h2}>This page could not be found.</h2>
19
+ </div>
20
+ </div>
21
+ </div>
22
+ </>
23
+ );
24
+ }
25
+
26
+ const styles = {
27
+ error: {
28
+ fontFamily:
29
+ 'system-ui,"Segoe UI",Roboto,Helvetica,Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji"',
30
+ height: "100vh",
31
+ textAlign: "center",
32
+ display: "flex",
33
+ flexDirection: "column",
34
+ alignItems: "center",
35
+ justifyContent: "center",
36
+ },
37
+
38
+ desc: {
39
+ display: "inline-block",
40
+ },
41
+
42
+ h1: {
43
+ display: "inline-block",
44
+ margin: "0 20px 0 0",
45
+ padding: "0 23px 0 0",
46
+ fontSize: 24,
47
+ fontWeight: 500,
48
+ verticalAlign: "top",
49
+ lineHeight: "49px",
50
+ },
51
+
52
+ h2: {
53
+ fontSize: 14,
54
+ fontWeight: 400,
55
+ lineHeight: "49px",
56
+ margin: 0,
57
+ },
58
+ } as const;
@@ -1,29 +1,22 @@
1
- import { existsSync, mkdirSync } from "fs";
1
+ import { join } from "path";
2
2
  import { crash, updateStatus, warn } from "@cloudflare/cli";
3
3
  import { processArgument } from "@cloudflare/cli/args";
4
4
  import { brandColor, dim } from "@cloudflare/cli/colors";
5
- import { installPackages, runFrameworkGenerator } from "helpers/command";
5
+ import { spinner } from "@cloudflare/cli/interactive";
6
+ import { runFrameworkGenerator } from "frameworks/index";
6
7
  import {
7
- compatDateFlag,
8
+ copyFile,
8
9
  probePaths,
10
+ readFile,
9
11
  readJSON,
10
12
  usesEslint,
11
13
  usesTypescript,
12
14
  writeFile,
13
15
  writeJSON,
14
16
  } 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";
17
+ import { detectPackageManager } from "helpers/packageManagers";
18
+ import { installPackages } from "helpers/packages";
19
+ import { getTemplatePath } from "../../src/templates";
27
20
  import type { TemplateConfig } from "../../src/templates";
28
21
  import type { C3Args, C3Context } from "types";
29
22
 
@@ -33,27 +26,53 @@ const generate = async (ctx: C3Context) => {
33
26
  const projectName = ctx.project.name;
34
27
 
35
28
  await runFrameworkGenerator(ctx, [projectName]);
36
- };
37
29
 
38
- const getApiTemplate = (
39
- apiPath: string,
40
- isTypescript: boolean
41
- ): [string, string] => {
42
- const isAppDir = /\/app\/api$/.test(apiPath);
30
+ const wranglerToml = readFile(join(getTemplatePath(ctx), "wrangler.toml"));
43
31
 
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 });
32
+ // Note: here we add `# KV Example:` to the toml file for the KV example, we don't actually
33
+ // include the comment in the template wrangler.toml file just so to keep it identical
34
+ // and consistent with that of all the other frameworks
35
+ // (instead of making it a special case which needs extra care)
36
+ const newTomlContent = wranglerToml.replace(
37
+ /#\s+\[\[kv_namespaces\]\]\n#\s+binding\s+=\s+"MY_KV_NAMESPACE"\n#\s+id\s+=\s+"[a-zA-Z0-9]+?"/,
38
+ ($1) => `# KV Example:\n${$1}`,
39
+ );
48
40
 
49
- return isTypescript
50
- ? [`${routeHandlerPath}/route.ts`, apiAppDirHelloTs]
51
- : [`${routeHandlerPath}/route.js`, apiAppDirHelloJs];
41
+ if (!/# KV Example/.test(newTomlContent)) {
42
+ // This should never happen to users, it is a check mostly so that
43
+ // if the toml file is changed in a way that breaks the "KV Example" addition
44
+ // the C3 Next.js e2e runs will fail with this
45
+ crash("Failed to properly generate the wrangler.toml file");
52
46
  }
53
47
 
54
- return isTypescript
55
- ? [`${apiPath}/hello.ts`, apiPagesDirHelloTs]
56
- : [`${apiPath}/hello.js`, apiPagesDirHelloJs];
48
+ writeFile(join(ctx.project.path, "wrangler.toml"), newTomlContent);
49
+
50
+ updateStatus("Created wrangler.toml file");
51
+ };
52
+
53
+ const updateNextConfig = () => {
54
+ const s = spinner();
55
+
56
+ const configFile = "next.config.mjs";
57
+ s.start(`Updating \`${configFile}\``);
58
+
59
+ const configContent = readFile(configFile);
60
+
61
+ const updatedConfigFile =
62
+ `import { setupDevPlatform } from '@cloudflare/next-on-pages/next-dev';
63
+
64
+ // Here we use the @cloudflare/next-on-pages next-dev module to allow us to use bindings during local development
65
+ // (when running the application with \`next dev\`), for more information see:
66
+ // https://github.com/cloudflare/next-on-pages/blob/5712c57ea7/internal-packages/next-dev/README.md
67
+ if (process.env.NODE_ENV === 'development') {
68
+ await setupDevPlatform();
69
+ }
70
+
71
+ `.replace(/\n\t*/g, "\n") + configContent;
72
+
73
+ writeFile(configFile, updatedConfigFile);
74
+
75
+ s.stop(`${brandColor(`updated`)} ${dim(`\`${configFile}\``)}`);
57
76
  };
58
77
 
59
78
  const configure = async (ctx: C3Context) => {
@@ -73,36 +92,13 @@ const configure = async (ctx: C3Context) => {
73
92
  crash("Could not find the `/api` or `/app` directory");
74
93
  }
75
94
 
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
95
  const usesTs = usesTypescript(ctx);
80
96
 
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
97
  if (usesTs) {
105
- writeFile(`${projectPath}/env.d.ts`, envDts);
98
+ copyFile(
99
+ join(getTemplatePath(ctx), "env.d.ts"),
100
+ join(projectPath, "env.d.ts"),
101
+ );
106
102
  updateStatus("Created an env.d.ts file");
107
103
  }
108
104
 
@@ -112,25 +108,29 @@ const configure = async (ctx: C3Context) => {
112
108
  await writeEslintrc(ctx);
113
109
  }
114
110
 
115
- writeFile(`${projectPath}/next.config.mjs`, nextConfig);
116
- updateStatus("Updated the next.config.js file");
111
+ updateNextConfig();
117
112
 
118
- writeFile(`${projectPath}/README.md`, readme);
113
+ copyFile(
114
+ join(getTemplatePath(ctx), "README.md"),
115
+ join(projectPath, "README.md"),
116
+ );
119
117
  updateStatus("Updated the README file");
120
118
 
121
119
  await addDevDependencies(installEslintPlugin);
122
120
  };
123
121
 
124
122
  export const shouldInstallNextOnPagesEslintPlugin = async (
125
- ctx: C3Context
123
+ ctx: C3Context,
126
124
  ): Promise<boolean> => {
127
125
  const eslintUsage = usesEslint(ctx);
128
126
 
129
- if (!eslintUsage.used) return false;
127
+ if (!eslintUsage.used) {
128
+ return false;
129
+ }
130
130
 
131
131
  if (eslintUsage.configType !== ".eslintrc.json") {
132
132
  warn(
133
- `Expected .eslintrc.json from Next.js scaffolding but found ${eslintUsage.configType} instead`
133
+ `Expected .eslintrc.json from Next.js scaffolding but found ${eslintUsage.configType} instead`,
134
134
  );
135
135
  return false;
136
136
  }
@@ -177,11 +177,40 @@ export default {
177
177
  id: "next",
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>"