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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "create-cloudflare",
3
- "version": "2.39.0",
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"
@@ -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(resolve("angular.json"));
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
+ };
@@ -24,5 +24,6 @@ const config: TemplateConfig = {
24
24
  }),
25
25
  devScript: "preview",
26
26
  deployScript: "deploy",
27
+ previewScript: "preview",
27
28
  };
28
29
  export default config;
@@ -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,9 @@
1
+ [project]
2
+ name = "<TBD>"
3
+ version = "0.1.0"
4
+ description = "Add your description here"
5
+ readme = "README.md"
6
+ requires-python = ">=3.12"
7
+ dependencies = [
8
+ "webtypy>=0.1.7",
9
+ ]
@@ -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
+ ]
@@ -65,5 +65,6 @@ const config: TemplateConfig = {
65
65
  }),
66
66
  devScript: "dev",
67
67
  deployScript: "deploy",
68
+ previewScript: "dev",
68
69
  };
69
70
  export default config;
@@ -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"),
@@ -10,7 +10,7 @@ import * as recast from "recast";
10
10
  import type { TemplateConfig } from "../../src/templates";
11
11
  import type { C3Context } from "types";
12
12
 
13
- const { npm, npx } = detectPackageManager();
13
+ const { npm, npx, name } = detectPackageManager();
14
14
 
15
15
  const generate = async (ctx: C3Context) => {
16
16
  await runFrameworkGenerator(ctx, ["playground", ctx.project.name]);
@@ -18,7 +18,8 @@ const generate = async (ctx: C3Context) => {
18
18
 
19
19
  const configure = async (ctx: C3Context) => {
20
20
  // Add the pages integration
21
- const cmd = [npx, "qwik", "add", "cloudflare-pages"];
21
+ // For some reason `pnpx qwik add` fails for qwik so we use `pnpm qwik add` instead.
22
+ const cmd = [name === "pnpm" ? npm : npx, "qwik", "add", "cloudflare-pages"];
22
23
  endSection(`Running ${quoteShellArgs(cmd)}`);
23
24
  await runCommand(cmd);
24
25
 
@@ -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
+ };
@@ -28,5 +28,6 @@ const config: TemplateConfig = {
28
28
  }),
29
29
  devScript: "start",
30
30
  deployScript: "deploy",
31
+ previewScript: "preview",
31
32
  };
32
33
  export default config;
@@ -39,5 +39,6 @@ const config: TemplateConfig = {
39
39
  }),
40
40
  devScript: "dev",
41
41
  deployScript: "deploy",
42
+ previewScript: "dev",
42
43
  };
43
44
  export default config;
@@ -10,7 +10,7 @@ import * as recast from "recast";
10
10
  import type { TemplateConfig } from "../../src/templates";
11
11
  import type { C3Context } from "types";
12
12
 
13
- const { npm, npx } = detectPackageManager();
13
+ const { npm, npx, name } = detectPackageManager();
14
14
 
15
15
  const generate = async (ctx: C3Context) => {
16
16
  await runFrameworkGenerator(ctx, ["playground", ctx.project.name]);
@@ -18,7 +18,8 @@ const generate = async (ctx: C3Context) => {
18
18
 
19
19
  const configure = async (ctx: C3Context) => {
20
20
  // Add the pages integration
21
- const cmd = [npx, "qwik", "add", "cloudflare-pages"];
21
+ // For some reason `pnpx qwik add` fails for qwik so we use `pnpm qwik add` instead.
22
+ const cmd = [name === "pnpm" ? npm : npx, "qwik", "add", "cloudflare-pages"];
22
23
  endSection(`Running ${quoteShellArgs(cmd)}`);
23
24
  await runCommand(cmd);
24
25
 
@@ -0,0 +1,182 @@
1
+ import assert from "assert";
2
+ import { logRaw } from "@cloudflare/cli";
3
+ import { brandColor, dim } from "@cloudflare/cli/colors";
4
+ import { inputPrompt, spinner } from "@cloudflare/cli/interactive";
5
+ import { runFrameworkGenerator } from "frameworks/index";
6
+ import { transformFile } from "helpers/codemod";
7
+ import { readJSON, usesTypescript, writeJSON } from "helpers/files";
8
+ import { detectPackageManager } from "helpers/packageManagers";
9
+ import { installPackages } from "helpers/packages";
10
+ import * as recast from "recast";
11
+ import type { TemplateConfig } from "../../src/templates";
12
+ import type { types } from "recast";
13
+ import type { C3Context } from "types";
14
+
15
+ const b = recast.types.builders;
16
+ const t = recast.types.namedTypes;
17
+ const { npm } = detectPackageManager();
18
+
19
+ const generate = async (ctx: C3Context) => {
20
+ const variant = await getVariant();
21
+ ctx.args.lang = variant.lang;
22
+
23
+ await runFrameworkGenerator(ctx, [
24
+ ctx.project.name,
25
+ "--template",
26
+ variant.value,
27
+ ]);
28
+
29
+ logRaw("");
30
+ };
31
+
32
+ const configure = async (ctx: C3Context) => {
33
+ await installPackages(["@cloudflare/vite-plugin"], {
34
+ dev: true,
35
+ startText: "Installing the Cloudflare Vite plugin",
36
+ doneText: `${brandColor(`installed`)} ${dim("@cloudflare/vite-plugin")}`,
37
+ });
38
+
39
+ await transformViteConfig(ctx);
40
+
41
+ if (usesTypescript(ctx)) {
42
+ updateTsconfigJson();
43
+ }
44
+ };
45
+
46
+ function transformViteConfig(ctx: C3Context) {
47
+ const filePath = `vite.config.${usesTypescript(ctx) ? "ts" : "js"}`;
48
+
49
+ transformFile(filePath, {
50
+ visitProgram(n) {
51
+ // Add an import of the @cloudflare/vite-plugin
52
+ // ```
53
+ // import {cloudflare} from "@cloudflare/vite-plugin";
54
+ // ```
55
+ const lastImportIndex = n.node.body.findLastIndex(
56
+ (statement) => statement.type === "ImportDeclaration",
57
+ );
58
+ const lastImport = n.get("body", lastImportIndex);
59
+ const importAst = b.importDeclaration(
60
+ [b.importSpecifier(b.identifier("cloudflare"))],
61
+ b.stringLiteral("@cloudflare/vite-plugin"),
62
+ );
63
+ lastImport.insertAfter(importAst);
64
+
65
+ return this.traverse(n);
66
+ },
67
+ visitCallExpression: function (n) {
68
+ // Add the imported plugin to the config
69
+ // ```
70
+ // defineConfig({
71
+ // plugins: [react(), cloudflare()],
72
+ // });
73
+ const callee = n.node.callee as types.namedTypes.Identifier;
74
+ if (callee.name !== "defineConfig") {
75
+ return this.traverse(n);
76
+ }
77
+
78
+ const config = n.node.arguments[0];
79
+ assert(t.ObjectExpression.check(config));
80
+ const pluginsProp = config.properties.find((prop) => isPluginsProp(prop));
81
+ assert(pluginsProp && t.ArrayExpression.check(pluginsProp.value));
82
+ pluginsProp.value.elements.push(
83
+ b.callExpression(b.identifier("cloudflare"), []),
84
+ );
85
+
86
+ return false;
87
+ },
88
+ });
89
+ }
90
+
91
+ function isPluginsProp(
92
+ prop: unknown,
93
+ ): prop is types.namedTypes.ObjectProperty | types.namedTypes.Property {
94
+ return (
95
+ (t.Property.check(prop) || t.ObjectProperty.check(prop)) &&
96
+ t.Identifier.check(prop.key) &&
97
+ prop.key.name === "plugins"
98
+ );
99
+ }
100
+
101
+ function updateTsconfigJson() {
102
+ const s = spinner();
103
+ s.start(`Updating tsconfig.json config`);
104
+ // Add a reference to the extra tsconfig.worker.json file.
105
+ // ```
106
+ // "references": [ ..., { path: "./tsconfig.worker.json" } ]
107
+ // ```
108
+ const tsconfig = readJSON("tsconfig.json") as { references: object[] };
109
+ if (tsconfig && typeof tsconfig === "object") {
110
+ tsconfig.references ??= [];
111
+ tsconfig.references.push({ path: "./tsconfig.worker.json" });
112
+ }
113
+ writeJSON("tsconfig.json", tsconfig);
114
+ s.stop(`${brandColor(`updated`)} ${dim(`\`tsconfig.json\``)}`);
115
+ }
116
+
117
+ async function getVariant() {
118
+ const variantsOptions = [
119
+ {
120
+ value: "react-ts",
121
+ lang: "ts",
122
+ label: "TypeScript",
123
+ },
124
+ {
125
+ value: "react-swc-ts",
126
+ lang: "ts",
127
+ label: "TypeScript + SWC",
128
+ },
129
+ {
130
+ value: "react",
131
+ lang: "js",
132
+ label: "JavaScript",
133
+ },
134
+ {
135
+ value: "react-swc",
136
+ lang: "js",
137
+ label: "JavaScript + SWC",
138
+ },
139
+ ];
140
+ const value = await inputPrompt({
141
+ type: "select",
142
+ question: "Select a variant:",
143
+ label: "variant",
144
+ options: variantsOptions,
145
+ defaultValue: variantsOptions[0].value,
146
+ });
147
+
148
+ const selected = variantsOptions.find((variant) => variant.value === value);
149
+ assert(selected, "Expected a variant to be selected");
150
+ return selected;
151
+ }
152
+
153
+ const config: TemplateConfig = {
154
+ configVersion: 1,
155
+ id: "react",
156
+ frameworkCli: "create-vite",
157
+ displayName: "React",
158
+ platform: "workers",
159
+ path: "templates-experimental/react",
160
+ copyFiles: {
161
+ variants: {
162
+ ts: {
163
+ path: "./ts",
164
+ },
165
+ js: {
166
+ path: "./js",
167
+ },
168
+ },
169
+ },
170
+ generate,
171
+ configure,
172
+ transformPackageJson: async () => ({
173
+ scripts: {
174
+ deploy: `${npm} run build && wrangler deploy`,
175
+ preview: `${npm} run build && vite preview`,
176
+ },
177
+ }),
178
+ devScript: "dev",
179
+ deployScript: "deploy",
180
+ previewScript: "preview",
181
+ };
182
+ export default config;
@@ -0,0 +1,13 @@
1
+ export default {
2
+ fetch(request, env) {
3
+ const url = new URL(request.url);
4
+
5
+ if (url.pathname.startsWith("/api/")) {
6
+ return Response.json({
7
+ name: "Cloudflare",
8
+ });
9
+ }
10
+
11
+ return env.ASSETS.fetch(request);
12
+ },
13
+ }
@@ -0,0 +1,58 @@
1
+ import { useState } from 'react'
2
+ import reactLogo from './assets/react.svg'
3
+ import viteLogo from '/vite.svg'
4
+ import cloudflareLogo from './assets/Cloudflare_Logo.svg'
5
+ import './App.css'
6
+
7
+ function App() {
8
+ const [count, setCount] = useState(0)
9
+ const [name, setName] = useState('unknown')
10
+
11
+ return (
12
+ <>
13
+ <div>
14
+ <a href='https://vite.dev' target='_blank'>
15
+ <img src={viteLogo} className='logo' alt='Vite logo' />
16
+ </a>
17
+ <a href='https://react.dev' target='_blank'>
18
+ <img src={reactLogo} className='logo react' alt='React logo' />
19
+ </a>
20
+ <a href='https://workers.cloudflare.com/' target='_blank'>
21
+ <img src={cloudflareLogo} className='logo cloudflare' alt='Cloudflare logo' />
22
+ </a>
23
+ </div>
24
+ <h1>Vite + React + Cloudflare</h1>
25
+ <div className='card'>
26
+ <button
27
+ onClick={() => setCount((count) => count + 1)}
28
+ aria-label='increment'
29
+ >
30
+ count is {count}
31
+ </button>
32
+ <p>
33
+ Edit <code>src/App.tsx</code> and save to test HMR
34
+ </p>
35
+ </div>
36
+ <div className='card'>
37
+ <button
38
+ onClick={() => {
39
+ fetch('/api/')
40
+ .then((res) => res.json())
41
+ .then((data) => setName(data.name))
42
+ }}
43
+ aria-label='get name'
44
+ >
45
+ Name from API is: {name}
46
+ </button>
47
+ <p>
48
+ Edit <code>api/index.js</code> to change the name
49
+ </p>
50
+ </div>
51
+ <p className='read-the-docs'>
52
+ Click on the Vite and React logos to learn more
53
+ </p>
54
+ </>
55
+ )
56
+ }
57
+
58
+ export default App
@@ -0,0 +1,26 @@
1
+ <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 101.4 33.5">
2
+ <defs>
3
+ <style>
4
+ .a {
5
+ fill: #fff;
6
+ }
7
+
8
+ .b {
9
+ fill: #f48120;
10
+ }
11
+
12
+ .c {
13
+ fill: #faad3f;
14
+ }
15
+
16
+ .d {
17
+ fill: #404041;
18
+ }
19
+ </style>
20
+ </defs>
21
+ <title>Cloudflare logo</title>
22
+ <path class="a" d="M94.7,10.6,89.1,9.3l-1-.4-25.7.2V21.5l32.3.1Z"/>
23
+ <path class="b" d="M84.2,20.4a2.85546,2.85546,0,0,0-.3-2.6,3.09428,3.09428,0,0,0-2.1-1.1l-17.4-.2c-.1,0-.2-.1-.3-.1a.1875.1875,0,0,1,0-.3c.1-.2.2-.3.4-.3L82,15.6a6.29223,6.29223,0,0,0,5.1-3.8l1-2.6c0-.1.1-.2,0-.3A11.39646,11.39646,0,0,0,66.2,7.7a5.45941,5.45941,0,0,0-3.6-1A5.20936,5.20936,0,0,0,58,11.3a5.46262,5.46262,0,0,0,.1,1.8A7.30177,7.30177,0,0,0,51,20.4a4.102,4.102,0,0,0,.1,1.1.3193.3193,0,0,0,.3.3H83.5c.2,0,.4-.1.4-.3Z"/>
24
+ <path class="c" d="M89.7,9.2h-.5c-.1,0-.2.1-.3.2l-.7,2.4a2.85546,2.85546,0,0,0,.3,2.6,3.09428,3.09428,0,0,0,2.1,1.1l3.7.2c.1,0,.2.1.3.1a.1875.1875,0,0,1,0,.3c-.1.2-.2.3-.4.3l-3.8.2a6.29223,6.29223,0,0,0-5.1,3.8l-.2.9c-.1.1,0,.3.2.3H98.5a.26517.26517,0,0,0,.3-.3,10.87184,10.87184,0,0,0,.4-2.6,9.56045,9.56045,0,0,0-9.5-9.5"/>
25
+ <path class="d" d="M100.5,27.2a.9.9,0,1,1,.9-.9.89626.89626,0,0,1-.9.9m0-1.6a.7.7,0,1,0,.7.7.68354.68354,0,0,0-.7-.7m.4,1.2h-.2l-.2-.3h-.2v.3h-.2v-.9h.5a.26517.26517,0,0,1,.3.3c0,.1-.1.2-.2.3l.2.3Zm-.3-.5c.1,0,.1,0,.1-.1a.09794.09794,0,0,0-.1-.1h-.3v.3h.3Zm-89.7-.9h2.2v6h3.8v1.9h-6Zm8.3,3.9a4.10491,4.10491,0,0,1,4.3-4.1,4.02,4.02,0,0,1,4.2,4.1,4.10491,4.10491,0,0,1-4.3,4.1,4.07888,4.07888,0,0,1-4.2-4.1m6.3,0a2.05565,2.05565,0,0,0-2-2.2,2.1025,2.1025,0,0,0,0,4.2c1.2.2,2-.8,2-2m4.9.5V25.4h2.2v4.4c0,1.1.6,1.7,1.5,1.7a1.39926,1.39926,0,0,0,1.5-1.6V25.4h2.2v4.4c0,2.6-1.5,3.7-3.7,3.7-2.3-.1-3.7-1.2-3.7-3.7m10.7-4.4h3.1c2.8,0,4.5,1.6,4.5,3.9s-1.7,4-4.5,4h-3V25.4Zm3.1,5.9a2.00909,2.00909,0,1,0,0-4h-.9v4Zm7.6-5.9h6.3v1.9H54v1.3h3.7v1.8H54v2.9H51.8Zm9.4,0h2.2v6h3.8v1.9h-6Zm11.7-.1h2.2l3.4,8H76.1l-.6-1.4H72.4l-.6,1.4H69.5Zm2,4.9L74,28l-.9,2.2Zm6.4-4.8H85a3.41818,3.41818,0,0,1,2.6.9,2.62373,2.62373,0,0,1-.9,4.2l1.9,2.8H86.1l-1.6-2.4h-1v2.4H81.3Zm3.6,3.8c.7,0,1.2-.4,1.2-.9,0-.6-.5-.9-1.2-.9H83.5v1.9h1.4Zm6.5-3.8h6.4v1.8H93.6v1.2h3.8v1.8H93.6v1.2h4.3v1.9H91.4ZM6.1,30.3a1.97548,1.97548,0,0,1-1.8,1.2,2.1025,2.1025,0,0,1,0-4.2,2.0977,2.0977,0,0,1,1.9,1.3H8.5a4.13459,4.13459,0,0,0-4.2-3.3A4.1651,4.1651,0,0,0,0,29.4a4.07888,4.07888,0,0,0,4.2,4.1,4.31812,4.31812,0,0,0,4.2-3.2Z"/>
26
+ </svg>
@@ -0,0 +1,9 @@
1
+ {
2
+ "name": "<TBD>",
3
+ "main": "api/index.js",
4
+ "compatibility_date": "<TBD>",
5
+ "assets": { "not_found_handling": "single-page-application", "binding": "ASSETS" },
6
+ "observability": {
7
+ "enabled": true
8
+ }
9
+ }
@@ -0,0 +1,17 @@
1
+ interface Env {
2
+ ASSETS: Fetcher;
3
+ }
4
+
5
+ export default {
6
+ fetch(request, env) {
7
+ const url = new URL(request.url);
8
+
9
+ if (url.pathname.startsWith("/api/")) {
10
+ return Response.json({
11
+ name: "Cloudflare",
12
+ });
13
+ }
14
+
15
+ return env.ASSETS.fetch(request);
16
+ },
17
+ } satisfies ExportedHandler<Env>;