create-cloudflare 2.39.0 → 2.40.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.js +15816 -14009
- package/package.json +4 -3
- package/templates/angular/c3.ts +24 -5
- package/templates/docusaurus/c3.ts +1 -0
- package/templates/hello-world/c3.ts +17 -1
- package/templates/hello-world/py/.python-version +1 -0
- package/templates/hello-world/py/README.md +18 -0
- package/templates/hello-world/py/pyproject.toml +9 -0
- package/templates/hello-world/py/uv.lock +22 -0
- package/templates/hono/c3.ts +1 -0
- package/templates/next/c3.ts +4 -1
- package/templates/pre-existing/c3.ts +1 -1
- package/templates/qwik/c3.ts +3 -2
- package/templates-experimental/angular/c3.ts +23 -5
- package/templates-experimental/docusaurus/c3.ts +1 -0
- package/templates-experimental/hono/c3.ts +1 -0
- package/templates-experimental/qwik/c3.ts +3 -2
- package/templates-experimental/react/c3.ts +182 -0
- package/templates-experimental/react/js/api/index.js +13 -0
- package/templates-experimental/react/js/src/App.jsx +58 -0
- package/templates-experimental/react/js/src/assets/Cloudflare_Logo.svg +26 -0
- package/templates-experimental/react/js/wrangler.json +9 -0
- package/templates-experimental/react/ts/api/index.ts +17 -0
- package/templates-experimental/react/ts/src/App.tsx +58 -0
- package/templates-experimental/react/ts/src/assets/Cloudflare_Logo.svg +26 -0
- package/templates-experimental/react/ts/tsconfig.worker.json +8 -0
- package/templates-experimental/react/ts/wrangler.json +9 -0
- package/templates/pre-existing/js/wrangler.jsonc +0 -5
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "create-cloudflare",
|
|
3
|
-
"version": "2.
|
|
3
|
+
"version": "2.40.0",
|
|
4
4
|
"description": "A CLI for creating and deploying new applications to Cloudflare.",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"cloudflare",
|
|
@@ -46,6 +46,7 @@
|
|
|
46
46
|
"@typescript-eslint/parser": "^6.9.0",
|
|
47
47
|
"chalk": "^5.2.0",
|
|
48
48
|
"command-exists": "^1.2.9",
|
|
49
|
+
"comment-json": "^4.2.5",
|
|
49
50
|
"cross-spawn": "^7.0.3",
|
|
50
51
|
"deepmerge": "^4.3.1",
|
|
51
52
|
"degit": "^2.8.4",
|
|
@@ -71,9 +72,9 @@
|
|
|
71
72
|
"yargs": "^17.7.2",
|
|
72
73
|
"@cloudflare/cli": "1.1.1",
|
|
73
74
|
"@cloudflare/eslint-config-worker": "1.1.0",
|
|
75
|
+
"wrangler": "3.112.0",
|
|
74
76
|
"@cloudflare/mock-npm-registry": "0.0.0",
|
|
75
|
-
"@cloudflare/workers-tsconfig": "0.0.0"
|
|
76
|
-
"wrangler": "3.111.0"
|
|
77
|
+
"@cloudflare/workers-tsconfig": "0.0.0"
|
|
77
78
|
},
|
|
78
79
|
"engines": {
|
|
79
80
|
"node": ">=18.14.1"
|
package/templates/angular/c3.ts
CHANGED
|
@@ -8,7 +8,7 @@ import { readFile, readJSON, writeFile } from "helpers/files";
|
|
|
8
8
|
import { detectPackageManager } from "helpers/packageManagers";
|
|
9
9
|
import { installPackages } from "helpers/packages";
|
|
10
10
|
import type { TemplateConfig } from "../../src/templates";
|
|
11
|
-
import type { C3Context } from "types";
|
|
11
|
+
import type { C3Context, PackageJson } from "types";
|
|
12
12
|
|
|
13
13
|
const { npm } = detectPackageManager();
|
|
14
14
|
|
|
@@ -64,10 +64,10 @@ async function updateAppCode() {
|
|
|
64
64
|
// Remove unwanted dependencies
|
|
65
65
|
s.start(`Updating package.json`);
|
|
66
66
|
const packageJsonPath = resolve("package.json");
|
|
67
|
-
const packageManifest = readJSON(packageJsonPath);
|
|
67
|
+
const packageManifest = readJSON(packageJsonPath) as PackageJson;
|
|
68
68
|
|
|
69
|
-
delete packageManifest["dependencies"]["express"];
|
|
70
|
-
delete packageManifest["devDependencies"]["@types/express"];
|
|
69
|
+
delete packageManifest["dependencies"]?.["express"];
|
|
70
|
+
delete packageManifest["devDependencies"]?.["@types/express"];
|
|
71
71
|
|
|
72
72
|
writeFile(packageJsonPath, JSON.stringify(packageManifest, null, 2));
|
|
73
73
|
s.stop(`${brandColor(`updated`)} ${dim(`\`package.json\``)}`);
|
|
@@ -76,7 +76,8 @@ async function updateAppCode() {
|
|
|
76
76
|
function updateAngularJson(ctx: C3Context) {
|
|
77
77
|
const s = spinner();
|
|
78
78
|
s.start(`Updating angular.json config`);
|
|
79
|
-
const angularJson = readJSON(
|
|
79
|
+
const angularJson = readJSON("angular.json") as AngularJson;
|
|
80
|
+
|
|
80
81
|
// Update builder
|
|
81
82
|
const architectSection = angularJson.projects[ctx.project.name].architect;
|
|
82
83
|
architectSection.build.options.outputPath = "dist";
|
|
@@ -112,3 +113,21 @@ const config: TemplateConfig = {
|
|
|
112
113
|
}),
|
|
113
114
|
};
|
|
114
115
|
export default config;
|
|
116
|
+
|
|
117
|
+
type AngularJson = {
|
|
118
|
+
projects: Record<
|
|
119
|
+
string,
|
|
120
|
+
{
|
|
121
|
+
architect: {
|
|
122
|
+
build: {
|
|
123
|
+
options: {
|
|
124
|
+
outputPath: string;
|
|
125
|
+
outputMode: string;
|
|
126
|
+
ssr: Record<string, unknown>;
|
|
127
|
+
assets: string[];
|
|
128
|
+
};
|
|
129
|
+
};
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
>;
|
|
133
|
+
};
|
|
@@ -1,9 +1,25 @@
|
|
|
1
|
+
import { readFile, writeFile } from "node:fs/promises";
|
|
2
|
+
import { resolve } from "node:path";
|
|
3
|
+
import type { TemplateConfig } from "../../src/templates";
|
|
4
|
+
|
|
1
5
|
export default {
|
|
2
6
|
configVersion: 1,
|
|
3
7
|
id: "hello-world",
|
|
4
8
|
displayName: "Hello World Worker",
|
|
5
9
|
description: "Get started with a basic Worker in the language of your choice",
|
|
6
10
|
platform: "workers",
|
|
11
|
+
async configure(ctx) {
|
|
12
|
+
if (ctx.args.lang === "python") {
|
|
13
|
+
for (const file of ["pyproject.toml", "uv.lock"]) {
|
|
14
|
+
const contents = await readFile(
|
|
15
|
+
resolve(ctx.project.path, file),
|
|
16
|
+
"utf8",
|
|
17
|
+
);
|
|
18
|
+
const updated = contents.replaceAll(/<TBD>/g, ctx.project.name);
|
|
19
|
+
await writeFile(resolve(ctx.project.path, file), updated);
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
},
|
|
7
23
|
copyFiles: {
|
|
8
24
|
variants: {
|
|
9
25
|
js: {
|
|
@@ -17,4 +33,4 @@ export default {
|
|
|
17
33
|
},
|
|
18
34
|
},
|
|
19
35
|
},
|
|
20
|
-
};
|
|
36
|
+
} satisfies TemplateConfig;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
3.12
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
## Usage
|
|
2
|
+
|
|
3
|
+
You can run the worker defined by your new project by executing `wrangler dev` in this
|
|
4
|
+
directory. This will start up an HTTP server and will allow you to iterate on your
|
|
5
|
+
worker without having to restart `wrangler`.
|
|
6
|
+
|
|
7
|
+
### Types and autocomplete
|
|
8
|
+
|
|
9
|
+
This project also includes a pyproject.toml and uv.lock file with some requirements which
|
|
10
|
+
set up autocomplete and type hints for this Python Workers project. To get these installed
|
|
11
|
+
you can run the following:
|
|
12
|
+
|
|
13
|
+
```
|
|
14
|
+
uv venv
|
|
15
|
+
uv sync
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
Then point your editor's Python plugin at the `.venv` directory.
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
version = 1
|
|
2
|
+
requires-python = ">=3.12"
|
|
3
|
+
|
|
4
|
+
[[package]]
|
|
5
|
+
name = "<TBD>"
|
|
6
|
+
version = "0.1.0"
|
|
7
|
+
source = { virtual = "." }
|
|
8
|
+
dependencies = [
|
|
9
|
+
{ name = "webtypy" },
|
|
10
|
+
]
|
|
11
|
+
|
|
12
|
+
[package.metadata]
|
|
13
|
+
requires-dist = [{ name = "webtypy", specifier = ">=0.1.7" }]
|
|
14
|
+
|
|
15
|
+
[[package]]
|
|
16
|
+
name = "webtypy"
|
|
17
|
+
version = "0.1.7"
|
|
18
|
+
source = { registry = "https://pypi.org/simple" }
|
|
19
|
+
sdist = { url = "https://files.pythonhosted.org/packages/5e/89/c7a0311fdc73809fc2415be97767f085ff3e00c86546430034dc8465fee7/webtypy-0.1.7.tar.gz", hash = "sha256:1b7212719a949c802f3d60fac5f0d952eb503a92121409cf1ad9847d7c76a336", size = 104505 }
|
|
20
|
+
wheels = [
|
|
21
|
+
{ url = "https://files.pythonhosted.org/packages/61/91/c731bdaa605279e00b28bfd2bf0ae67f48061d16890fb1c026924bfbd242/webtypy-0.1.7-py3-none-any.whl", hash = "sha256:f35e6d73a4e08783e23adfac271a11cda3a2bd1105499db70e4819244efed0ae", size = 103519 },
|
|
22
|
+
]
|
package/templates/hono/c3.ts
CHANGED
package/templates/next/c3.ts
CHANGED
|
@@ -125,7 +125,10 @@ export const shouldInstallNextOnPagesEslintPlugin = async (
|
|
|
125
125
|
};
|
|
126
126
|
|
|
127
127
|
export const writeEslintrc = async (ctx: C3Context): Promise<void> => {
|
|
128
|
-
const eslintConfig = readJSON(`${ctx.project.path}/.eslintrc.json`)
|
|
128
|
+
const eslintConfig = readJSON(`${ctx.project.path}/.eslintrc.json`) as {
|
|
129
|
+
plugins: string[];
|
|
130
|
+
extends: string | string[];
|
|
131
|
+
};
|
|
129
132
|
|
|
130
133
|
eslintConfig.plugins ??= [];
|
|
131
134
|
eslintConfig.plugins.push("eslint-plugin-next-on-pages");
|
|
@@ -56,7 +56,7 @@ export async function copyExistingWorkerFiles(ctx: C3Context) {
|
|
|
56
56
|
{ recursive: true },
|
|
57
57
|
);
|
|
58
58
|
|
|
59
|
-
// copy wrangler.toml from the downloaded Worker
|
|
59
|
+
// copy ./wrangler.toml from the downloaded Worker
|
|
60
60
|
await cp(
|
|
61
61
|
join(tempdir, ctx.args.existingScript, "wrangler.toml"),
|
|
62
62
|
join(ctx.project.path, "wrangler.toml"),
|
package/templates/qwik/c3.ts
CHANGED
|
@@ -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
|
-
|
|
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
|
|
|
@@ -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
|
+
};
|
|
@@ -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
|
-
|
|
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
|
|
|
@@ -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,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,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>;
|