From 798928a71d271fac7a4dae9c50f46e3447656ad6 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Fri, 8 Sep 2023 00:44:54 +0200 Subject: [PATCH 01/12] added config for _routes.json generation --- packages/integrations/cloudflare/README.md | 30 +++++ packages/integrations/cloudflare/src/index.ts | 71 ++++++++--- .../cloudflare/test/routesJson.js | 115 ++++++++++++++++++ 3 files changed, 199 insertions(+), 17 deletions(-) diff --git a/packages/integrations/cloudflare/README.md b/packages/integrations/cloudflare/README.md index f49824cd9364..926a9cf69e62 100644 --- a/packages/integrations/cloudflare/README.md +++ b/packages/integrations/cloudflare/README.md @@ -75,6 +75,36 @@ export default defineConfig({ Note that this adapter does not support using [Cloudflare Pages Middleware](https://developers.cloudflare.com/pages/platform/functions/middleware/). Astro will bundle the [Astro middleware](https://docs.astro.build/en/guides/middleware/) into each page. +### routes.strategy + +`routes.strategy: "auto" | "include" | "exclude"` + +default `"auto"` + +If no [custom `_routes.json`](#custom-_routesjson) is provided, `@astrojs/cloudflare` will generate one for you. There are two ways to generate the `_routes.json`: + +1. For each page or endpoint in your application that is not prerendered, an entry in the `include` array will be generated. For each page that is prerendered and whoose path is matched by an `include` entry, an entry in the `exclude` array will be generated. + +2. One `"/*"` entry in the `include` array will be generated. For each page that is prerendered, an entry in the `exclude` array will be generated. + +Setting `routes.strategy` to `"include"` will generate a `_routes.json` with the first strategy. Setting it to `"exclude"` will use the second strategy. Setting it to `"auto"` will use the strategy that generates the least amount of entries. + +### routes.include + +`routes.include: string[]` + +default `[]` + +If you want to use the automatic `_routes.json` generation, but want to include additional routes (e.g. when having custom functions in the `functions` folder), you can use the `routes.include` option to add additional routes to the `include` array. + +### routes.exclude + +`routes.exclude: string[]` + +default `[]` + +If you want to use the automatic `_routes.json` generation, but want to exclude additional routes, you can use the `routes.exclude` option to add additional routes to the `exclude` array. + ## Enabling Preview In order for preview to work you must install `wrangler` diff --git a/packages/integrations/cloudflare/src/index.ts b/packages/integrations/cloudflare/src/index.ts index 718b1efa8648..a175f38f90a1 100644 --- a/packages/integrations/cloudflare/src/index.ts +++ b/packages/integrations/cloudflare/src/index.ts @@ -13,6 +13,19 @@ export type { DirectoryRuntime } from './server.directory'; type Options = { mode: 'directory' | 'advanced'; functionPerRoute?: boolean; + /** Configure automatic `routes.json` generation */ + routes?: { + /** Strategy for generating `include` and `exclude` patterns + * - `auto`: Will use the strategy that generates the least amount of entries. + * - `include`: For each page or endpoint in your application that is not prerendered, an entry in the `include` array will be generated. For each page that is prerendered and whoose path is matched by an `include` entry, an entry in the `exclude` array will be generated. + * - `exclude`: One `"/*"` entry in the `include` array will be generated. For each page that is prerendered, an entry in the `exclude` array will be generated. + * */ + strategy?: 'auto' | 'include' | 'exclude'; + /** Additional `include` patterns */ + include?: string[]; + /** Additional `exclude` patterns */ + exclude?: string[]; + }; }; interface BuildConfig { @@ -373,39 +386,63 @@ export default function createIntegration(args?: Options): AstroIntegration { staticPathList.push(...routes.filter((r) => r.type === 'redirect').map((r) => r.route)); + const strategy = args?.routes?.strategy ?? 'auto'; + // In order to product the shortest list of patterns, we first try to // include all function endpoints, and then exclude all static paths - let include = deduplicatePatterns( - functionEndpoints.map((endpoint) => endpoint.includePattern) - ); - let exclude = deduplicatePatterns( - staticPathList.filter((file: string) => - functionEndpoints.some((endpoint) => endpoint.regexp.test(file)) - ) - ); + const includeStrategy = + strategy === 'exclude' + ? undefined + : { + include: deduplicatePatterns( + functionEndpoints + .map((endpoint) => endpoint.includePattern) + .concat(args?.routes?.include ?? []) + ), + exclude: deduplicatePatterns( + staticPathList + .filter((file: string) => + functionEndpoints.some((endpoint) => endpoint.regexp.test(file)) + ) + .concat(args?.routes?.exclude ?? []) + ), + }; // Cloudflare requires at least one include pattern: // https://developers.cloudflare.com/pages/platform/functions/routing/#limits // So we add a pattern that we immediately exclude again - if (include.length === 0) { - include = ['/']; - exclude = ['/']; + if (includeStrategy?.include.length === 0) { + includeStrategy.include = ['/']; + includeStrategy.exclude = ['/']; } // If using only an exclude list would produce a shorter list of patterns, // we use that instead - if (include.length + exclude.length > staticPathList.length) { - include = ['/*']; - exclude = deduplicatePatterns(staticPathList); - } + const excludeStrategy = + strategy === 'include' + ? undefined + : { + include: ['/*'], + exclude: deduplicatePatterns(staticPathList.concat(args?.routes?.exclude ?? [])), + }; + + const includeStrategyLength = includeStrategy + ? includeStrategy.include.length + includeStrategy.exclude.length + : Infinity; + + const excludeStrategyLength = excludeStrategy + ? excludeStrategy.include.length + excludeStrategy.exclude.length + : Infinity; + + const winningStrategy = + includeStrategyLength <= excludeStrategyLength ? includeStrategy : excludeStrategy; await fs.promises.writeFile( new URL('./_routes.json', _config.outDir), JSON.stringify( { version: 1, - include, - exclude, + ...winningStrategy, }, null, 2 diff --git a/packages/integrations/cloudflare/test/routesJson.js b/packages/integrations/cloudflare/test/routesJson.js index 1714dfb89c0c..adc7c41e7012 100644 --- a/packages/integrations/cloudflare/test/routesJson.js +++ b/packages/integrations/cloudflare/test/routesJson.js @@ -1,5 +1,6 @@ import { expect } from 'chai'; import { loadFixture } from './test-utils.js'; +import cloudflare from '../dist/index.js'; /** @type {import('./test-utils.js').Fixture} */ describe('_routes.json generation', () => { @@ -75,4 +76,118 @@ describe('_routes.json generation', () => { }); }); }); + + describe('with strategy `"include"`', () => { + let fixture; + + before(async () => { + process.env.SRC = './src/dynamicOnly'; + fixture = await loadFixture({ + root: './fixtures/routesJson/', + adapter: cloudflare({ + mode: 'directory', + routes: { strategy: 'include' }, + }), + }); + await fixture.build(); + }); + + it('creates `include` entries even though the `"exclude"` strategy would have produced less entries.', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/', '/_image', '/another'], + exclude: [], + }); + }); + }); + + describe('with strategy `"exclude"`', () => { + let fixture; + + before(async () => { + process.env.SRC = './src/staticOnly'; + fixture = await loadFixture({ + root: './fixtures/routesJson/', + adapter: cloudflare({ + mode: 'directory', + routes: { strategy: 'exclude' }, + }), + }); + await fixture.build(); + }); + + it('creates `exclude` entries even though the `"include"` strategy would have produced less entries.', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/*'], + exclude: ['/', '/index.html', '/a/redirect'], + }); + }); + }); + + describe('with additional `include` entries', () => { + let fixture; + + before(async () => { + process.env.SRC = './src/mixed'; + fixture = await loadFixture({ + root: './fixtures/routesJson/', + adapter: cloudflare({ + mode: 'directory', + routes: { + strategy: 'include', + include: ['/another', '/a/redundant'], + }, + }), + }); + await fixture.build(); + }); + + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image', '/another'], + exclude: ['/a/', '/a/redirect', '/a/index.html'], + }); + }); + }); + + describe('with additional `exclude` entries', () => { + let fixture; + + before(async () => { + process.env.SRC = './src/mixed'; + fixture = await loadFixture({ + root: './fixtures/routesJson/', + adapter: cloudflare({ + mode: 'directory', + routes: { + strategy: 'include', + exclude: ['/another', '/a/*', '/a/index.html'], + }, + }), + }); + await fixture.build(); + }); + + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image'], + exclude: ['/a/', '/a/*', '/another'], + }); + }); + }); }); From 9713bd39c275e6f6d42c7148c6a33bafd6bc98ea Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Fri, 8 Sep 2023 08:40:44 +0200 Subject: [PATCH 02/12] added changeset --- .changeset/famous-seas-obey.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 .changeset/famous-seas-obey.md diff --git a/.changeset/famous-seas-obey.md b/.changeset/famous-seas-obey.md new file mode 100644 index 000000000000..32fa7231749e --- /dev/null +++ b/.changeset/famous-seas-obey.md @@ -0,0 +1,5 @@ +--- +'@astrojs/cloudflare': minor +--- + +added config for `_routes.json` generation From fce6c3d5944c76beb10bda20dfb117dbaad17f92 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Mon, 11 Sep 2023 14:12:59 +0200 Subject: [PATCH 03/12] renamed test file --- .../cloudflare/test/{routesJson.js => routes-json.js} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename packages/integrations/cloudflare/test/{routesJson.js => routes-json.js} (100%) diff --git a/packages/integrations/cloudflare/test/routesJson.js b/packages/integrations/cloudflare/test/routes-json.js similarity index 100% rename from packages/integrations/cloudflare/test/routesJson.js rename to packages/integrations/cloudflare/test/routes-json.js From c3cb0087f3d1ecd15cd591ec0f9572d741703d4e Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Mon, 11 Sep 2023 15:54:08 +0200 Subject: [PATCH 04/12] updated comments --- packages/integrations/cloudflare/src/index.ts | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/packages/integrations/cloudflare/src/index.ts b/packages/integrations/cloudflare/src/index.ts index a175f38f90a1..8345e4695350 100644 --- a/packages/integrations/cloudflare/src/index.ts +++ b/packages/integrations/cloudflare/src/index.ts @@ -388,8 +388,7 @@ export default function createIntegration(args?: Options): AstroIntegration { const strategy = args?.routes?.strategy ?? 'auto'; - // In order to product the shortest list of patterns, we first try to - // include all function endpoints, and then exclude all static paths + // Strategy `include`: include all function endpoints, and then exclude static paths that would be matched by an include pattern const includeStrategy = strategy === 'exclude' ? undefined @@ -416,8 +415,7 @@ export default function createIntegration(args?: Options): AstroIntegration { includeStrategy.exclude = ['/']; } - // If using only an exclude list would produce a shorter list of patterns, - // we use that instead + // Strategy `exclude`: include everything, and then exclude all static paths const excludeStrategy = strategy === 'include' ? undefined From 36acb341c654f66e01ae6824788b0f91de63e5c1 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Wed, 13 Sep 2023 16:24:36 +0200 Subject: [PATCH 05/12] Apply suggestions from code review Co-authored-by: Sarah Rainsberger --- .changeset/famous-seas-obey.md | 2 +- packages/integrations/cloudflare/README.md | 26 ++++++++++++++++++---- 2 files changed, 23 insertions(+), 5 deletions(-) diff --git a/.changeset/famous-seas-obey.md b/.changeset/famous-seas-obey.md index 32fa7231749e..29aca3c054d0 100644 --- a/.changeset/famous-seas-obey.md +++ b/.changeset/famous-seas-obey.md @@ -2,4 +2,4 @@ '@astrojs/cloudflare': minor --- -added config for `_routes.json` generation +Adds three new config options for `_routes.json` generation: `routes.strategy`, `routes.include`, and `routes.exclude`. diff --git a/packages/integrations/cloudflare/README.md b/packages/integrations/cloudflare/README.md index 926a9cf69e62..f6199a0365c4 100644 --- a/packages/integrations/cloudflare/README.md +++ b/packages/integrations/cloudflare/README.md @@ -81,13 +81,16 @@ Note that this adapter does not support using [Cloudflare Pages Middleware](http default `"auto"` -If no [custom `_routes.json`](#custom-_routesjson) is provided, `@astrojs/cloudflare` will generate one for you. There are two ways to generate the `_routes.json`: -1. For each page or endpoint in your application that is not prerendered, an entry in the `include` array will be generated. For each page that is prerendered and whoose path is matched by an `include` entry, an entry in the `exclude` array will be generated. +Determines how `routes.json` will be generated if no [custom `_routes.json`](#custom-_routesjson) is provided. -2. One `"/*"` entry in the `include` array will be generated. For each page that is prerendered, an entry in the `exclude` array will be generated. +There are three options available: -Setting `routes.strategy` to `"include"` will generate a `_routes.json` with the first strategy. Setting it to `"exclude"` will use the second strategy. Setting it to `"auto"` will use the strategy that generates the least amount of entries. +- **`include`:** For each page or endpoint in your application that is not pre-rendered, an entry in the `include` array will be generated. For each page that is pre-rendered and whose path is matched by an `include` entry, an entry in the `exclude` array will be generated. + +- **`exclude`:** One `"/*"` entry in the `include` array will be generated. For each page that is pre-rendered, an entry in the `exclude` array will be generated. + +- **`"auto"` (default):** will compare the methods used for both `include` and `exclude` and will automatically select the option that generates the fewest entries. ### routes.include @@ -105,6 +108,21 @@ default `[]` If you want to use the automatic `_routes.json` generation, but want to exclude additional routes, you can use the `routes.exclude` option to add additional routes to the `exclude` array. +The following example automatically generates `_routes.json` while including additional routes for `/functions/` and excluding routes for `/assets/`: + +```diff +// astro.config.mjs +export default defineConfig({ + adapter: cloudflare({ + mode: 'directory', ++ routes: { ++ strategy: 'include', ++ include: ['/functions'], ++ exclude: ['/assets'], ++ }, + }), +}); +``` ## Enabling Preview In order for preview to work you must install `wrangler` From a341a70b3cf8e443f582fe88b5648efdd3e8a645 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Wed, 13 Sep 2023 17:10:43 +0200 Subject: [PATCH 06/12] worked on tests --- packages/integrations/cloudflare/src/index.ts | 4 + .../astro.config.mjs | 4 +- .../{routesJson => routes-json}/package.json | 0 .../src/dynamicOnly/pages/another.astro | 0 .../src/dynamicOnly/pages/index.astro | 0 .../src/mixed/pages/a/[...rest].astro | 0 .../src/mixed/pages/a/[id].astro | 0 .../src/mixed/pages/a/endpoint.ts | 0 .../src/mixed/pages/a/index.astro | 0 .../src/mixed/pages/b/index.html | 0 .../src/mixed/public/public.txt | 0 .../src/staticOnly/pages/index.astro | 0 .../cloudflare/test/routes-json.js | 193 ----------------- .../cloudflare/test/routes-json.test.js | 203 ++++++++++++++++++ 14 files changed, 208 insertions(+), 196 deletions(-) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/astro.config.mjs (54%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/package.json (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/dynamicOnly/pages/another.astro (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/dynamicOnly/pages/index.astro (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/mixed/pages/a/[...rest].astro (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/mixed/pages/a/[id].astro (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/mixed/pages/a/endpoint.ts (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/mixed/pages/a/index.astro (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/mixed/pages/b/index.html (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/mixed/public/public.txt (100%) rename packages/integrations/cloudflare/test/fixtures/{routesJson => routes-json}/src/staticOnly/pages/index.astro (100%) delete mode 100644 packages/integrations/cloudflare/test/routes-json.js create mode 100644 packages/integrations/cloudflare/test/routes-json.test.js diff --git a/packages/integrations/cloudflare/src/index.ts b/packages/integrations/cloudflare/src/index.ts index 8345e4695350..04e572812184 100644 --- a/packages/integrations/cloudflare/src/index.ts +++ b/packages/integrations/cloudflare/src/index.ts @@ -266,6 +266,10 @@ export default function createIntegration(args?: Options): AstroIntegration { // move cloudflare specific files to the root const cloudflareSpecialFiles = ['_headers', '_redirects', '_routes.json']; + if (!isModeDirectory) { + cloudflareSpecialFiles.push('_worker.js'); + } + if (_config.base !== '/') { for (const file of cloudflareSpecialFiles) { try { diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/astro.config.mjs b/packages/integrations/cloudflare/test/fixtures/routes-json/astro.config.mjs similarity index 54% rename from packages/integrations/cloudflare/test/fixtures/routesJson/astro.config.mjs rename to packages/integrations/cloudflare/test/fixtures/routes-json/astro.config.mjs index 66b50c0986ff..6e03bbc353fc 100644 --- a/packages/integrations/cloudflare/test/fixtures/routesJson/astro.config.mjs +++ b/packages/integrations/cloudflare/test/fixtures/routes-json/astro.config.mjs @@ -1,11 +1,9 @@ import { defineConfig } from 'astro/config'; -import cloudflare from '@astrojs/cloudflare'; export default defineConfig({ - adapter: cloudflare({ mode: 'directory' }), + // adapter will be set dynamically by the test output: 'hybrid', redirects: { '/a/redirect': '/', }, - srcDir: process.env.SRC }); diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/package.json b/packages/integrations/cloudflare/test/fixtures/routes-json/package.json similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/package.json rename to packages/integrations/cloudflare/test/fixtures/routes-json/package.json diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/dynamicOnly/pages/another.astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/another.astro similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/dynamicOnly/pages/another.astro rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/another.astro diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/dynamicOnly/pages/index.astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/index.astro similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/dynamicOnly/pages/index.astro rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/index.astro diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/[...rest].astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/[...rest].astro similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/[...rest].astro rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/[...rest].astro diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/[id].astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/[id].astro similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/[id].astro rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/[id].astro diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/endpoint.ts b/packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/endpoint.ts similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/endpoint.ts rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/endpoint.ts diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/index.astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/index.astro similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/a/index.astro rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/a/index.astro diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/b/index.html b/packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/b/index.html similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/pages/b/index.html rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/pages/b/index.html diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/public/public.txt b/packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/public/public.txt similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/mixed/public/public.txt rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/public/public.txt diff --git a/packages/integrations/cloudflare/test/fixtures/routesJson/src/staticOnly/pages/index.astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/staticOnly/pages/index.astro similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routesJson/src/staticOnly/pages/index.astro rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/staticOnly/pages/index.astro diff --git a/packages/integrations/cloudflare/test/routes-json.js b/packages/integrations/cloudflare/test/routes-json.js deleted file mode 100644 index adc7c41e7012..000000000000 --- a/packages/integrations/cloudflare/test/routes-json.js +++ /dev/null @@ -1,193 +0,0 @@ -import { expect } from 'chai'; -import { loadFixture } from './test-utils.js'; -import cloudflare from '../dist/index.js'; - -/** @type {import('./test-utils.js').Fixture} */ -describe('_routes.json generation', () => { - after(() => { - delete process.env.SRC; - }); - - describe('of both functions and static files', () => { - let fixture; - - before(async () => { - process.env.SRC = './src/mixed'; - fixture = await loadFixture({ - root: './fixtures/routesJson/', - }); - await fixture.build(); - }); - - it('creates `include` for functions and `exclude` for static files where needed', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); - - expect(routes).to.deep.equal({ - version: 1, - include: ['/a/*', '/_image'], - exclude: ['/a/', '/a/redirect', '/a/index.html'], - }); - }); - }); - - describe('of only functions', () => { - let fixture; - - before(async () => { - process.env.SRC = './src/dynamicOnly'; - fixture = await loadFixture({ - root: './fixtures/routesJson/', - }); - await fixture.build(); - }); - - it('creates a wildcard `include` and `exclude` only for the redirect', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); - - expect(routes).to.deep.equal({ - version: 1, - include: ['/*'], - exclude: ['/a/redirect'], - }); - }); - }); - - describe('of only static files', () => { - let fixture; - - before(async () => { - process.env.SRC = './src/staticOnly'; - fixture = await loadFixture({ - root: './fixtures/routesJson/', - }); - await fixture.build(); - }); - - it('create only one `include` and `exclude` that are supposed to match nothing', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); - - expect(routes).to.deep.equal({ - version: 1, - include: ['/_image'], - exclude: [], - }); - }); - }); - - describe('with strategy `"include"`', () => { - let fixture; - - before(async () => { - process.env.SRC = './src/dynamicOnly'; - fixture = await loadFixture({ - root: './fixtures/routesJson/', - adapter: cloudflare({ - mode: 'directory', - routes: { strategy: 'include' }, - }), - }); - await fixture.build(); - }); - - it('creates `include` entries even though the `"exclude"` strategy would have produced less entries.', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); - - expect(routes).to.deep.equal({ - version: 1, - include: ['/', '/_image', '/another'], - exclude: [], - }); - }); - }); - - describe('with strategy `"exclude"`', () => { - let fixture; - - before(async () => { - process.env.SRC = './src/staticOnly'; - fixture = await loadFixture({ - root: './fixtures/routesJson/', - adapter: cloudflare({ - mode: 'directory', - routes: { strategy: 'exclude' }, - }), - }); - await fixture.build(); - }); - - it('creates `exclude` entries even though the `"include"` strategy would have produced less entries.', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); - - expect(routes).to.deep.equal({ - version: 1, - include: ['/*'], - exclude: ['/', '/index.html', '/a/redirect'], - }); - }); - }); - - describe('with additional `include` entries', () => { - let fixture; - - before(async () => { - process.env.SRC = './src/mixed'; - fixture = await loadFixture({ - root: './fixtures/routesJson/', - adapter: cloudflare({ - mode: 'directory', - routes: { - strategy: 'include', - include: ['/another', '/a/redundant'], - }, - }), - }); - await fixture.build(); - }); - - it('creates `include` for functions and `exclude` for static files where needed', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); - - expect(routes).to.deep.equal({ - version: 1, - include: ['/a/*', '/_image', '/another'], - exclude: ['/a/', '/a/redirect', '/a/index.html'], - }); - }); - }); - - describe('with additional `exclude` entries', () => { - let fixture; - - before(async () => { - process.env.SRC = './src/mixed'; - fixture = await loadFixture({ - root: './fixtures/routesJson/', - adapter: cloudflare({ - mode: 'directory', - routes: { - strategy: 'include', - exclude: ['/another', '/a/*', '/a/index.html'], - }, - }), - }); - await fixture.build(); - }); - - it('creates `include` for functions and `exclude` for static files where needed', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); - - expect(routes).to.deep.equal({ - version: 1, - include: ['/a/*', '/_image'], - exclude: ['/a/', '/a/*', '/another'], - }); - }); - }); -}); diff --git a/packages/integrations/cloudflare/test/routes-json.test.js b/packages/integrations/cloudflare/test/routes-json.test.js new file mode 100644 index 000000000000..58791480f748 --- /dev/null +++ b/packages/integrations/cloudflare/test/routes-json.test.js @@ -0,0 +1,203 @@ +import { expect } from 'chai'; +import { loadFixture } from './test-utils.js'; +import cloudflare from '../dist/index.js'; + +/** @type {import('./test-utils.js').Fixture} */ +describe('_routes.json generation', () => { + for (const mode of ['directory', 'advanced']) { + describe(`in mode "${mode}"`, () => { + describe('of both functions and static files', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/mixed', + adapter: cloudflare({ + mode, + functionPerRoute: true, + }), + }); + await fixture.build(); + }); + + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image'], + exclude: ['/a/', '/a/redirect', '/a/index.html'], + }); + }); + }); + + describe('of only functions', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/dynamicOnly', + adapter: cloudflare({ + mode, + }), + }); + await fixture.build(); + }); + + it('creates a wildcard `include` and `exclude` only for the redirect', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/*'], + exclude: ['/a/redirect'], + }); + }); + }); + + describe('of only static files', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/staticOnly', + adapter: cloudflare({ + mode, + }), + }); + await fixture.build(); + }); + + it('create only one `include` and `exclude` that are supposed to match nothing', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/_image'], + exclude: [], + }); + }); + }); + + describe('with strategy `"include"`', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/dynamicOnly', + adapter: cloudflare({ + mode, + routes: { strategy: 'include' }, + }), + }); + await fixture.build(); + }); + + it('creates `include` entries even though the `"exclude"` strategy would have produced less entries.', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/', '/_image', '/another'], + exclude: [], + }); + }); + }); + + describe('with strategy `"exclude"`', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/staticOnly', + adapter: cloudflare({ + mode, + routes: { strategy: 'exclude' }, + }), + }); + await fixture.build(); + }); + + it('creates `exclude` entries even though the `"include"` strategy would have produced less entries.', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/*'], + exclude: ['/', '/index.html', '/a/redirect'], + }); + }); + }); + + describe('with additional `include` entries', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/mixed', + adapter: cloudflare({ + mode, + routes: { + strategy: 'include', + include: ['/another', '/a/redundant'], + }, + }), + }); + await fixture.build(); + }); + + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image', '/another'], + exclude: ['/a/', '/a/redirect', '/a/index.html'], + }); + }); + }); + + describe('with additional `exclude` entries', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/mixed', + adapter: cloudflare({ + mode, + routes: { + strategy: 'include', + exclude: ['/another', '/a/*', '/a/index.html'], + }, + }), + }); + await fixture.build(); + }); + + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); + + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image'], + exclude: ['/a/', '/a/*', '/another'], + }); + }); + }); + }); + } +}); From e910655f5651ba5ee6b41bffdf3335e1b129a836 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Wed, 13 Sep 2023 17:36:11 +0200 Subject: [PATCH 07/12] worked on docs --- packages/integrations/cloudflare/README.md | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/packages/integrations/cloudflare/README.md b/packages/integrations/cloudflare/README.md index f6199a0365c4..cd8d786a8ed7 100644 --- a/packages/integrations/cloudflare/README.md +++ b/packages/integrations/cloudflare/README.md @@ -81,16 +81,15 @@ Note that this adapter does not support using [Cloudflare Pages Middleware](http default `"auto"` - Determines how `routes.json` will be generated if no [custom `_routes.json`](#custom-_routesjson) is provided. -There are three options available: +There are three options available: - **`include`:** For each page or endpoint in your application that is not pre-rendered, an entry in the `include` array will be generated. For each page that is pre-rendered and whose path is matched by an `include` entry, an entry in the `exclude` array will be generated. - **`exclude`:** One `"/*"` entry in the `include` array will be generated. For each page that is pre-rendered, an entry in the `exclude` array will be generated. -- **`"auto"` (default):** will compare the methods used for both `include` and `exclude` and will automatically select the option that generates the fewest entries. +- **`"auto"` (default):** will compare the methods used for both `include` and `exclude` and will automatically select the option that generates the fewest entries. ### routes.include @@ -108,7 +107,7 @@ default `[]` If you want to use the automatic `_routes.json` generation, but want to exclude additional routes, you can use the `routes.exclude` option to add additional routes to the `exclude` array. -The following example automatically generates `_routes.json` while including additional routes for `/functions/` and excluding routes for `/assets/`: +The following example automatically generates `_routes.json` while including and excluding additional routes. Note that that is only necessary if you have custom functions in the `functions` folder that are not handled by Astro. ```diff // astro.config.mjs @@ -117,12 +116,13 @@ export default defineConfig({ mode: 'directory', + routes: { + strategy: 'include', -+ include: ['/functions'], -+ exclude: ['/assets'], ++ include: ['/users/*'], // handled by custom function: functions/users/[id].js ++ exclude: ['/users/faq'], // handled by static page: pages/users/faq.astro + }, }), }); ``` + ## Enabling Preview In order for preview to work you must install `wrangler` From 00ba7fceb063f832794fe81306ba223a1a466918 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Thu, 14 Sep 2023 09:53:45 +0200 Subject: [PATCH 08/12] worked on docs --- packages/integrations/cloudflare/README.md | 48 +++++++++++++++++++--- 1 file changed, 43 insertions(+), 5 deletions(-) diff --git a/packages/integrations/cloudflare/README.md b/packages/integrations/cloudflare/README.md index cd8d786a8ed7..010dad86a9aa 100644 --- a/packages/integrations/cloudflare/README.md +++ b/packages/integrations/cloudflare/README.md @@ -85,11 +85,49 @@ Determines how `routes.json` will be generated if no [custom `_routes.json`](#cu There are three options available: -- **`include`:** For each page or endpoint in your application that is not pre-rendered, an entry in the `include` array will be generated. For each page that is pre-rendered and whose path is matched by an `include` entry, an entry in the `exclude` array will be generated. - -- **`exclude`:** One `"/*"` entry in the `include` array will be generated. For each page that is pre-rendered, an entry in the `exclude` array will be generated. - -- **`"auto"` (default):** will compare the methods used for both `include` and `exclude` and will automatically select the option that generates the fewest entries. +- **`"auto"` (default):** Will automatically select the strategy that generates the fewest entries. This should almost always be sufficient, so choose this option unless you have a specific reason not to. + +- **`include`:** Pages and endpoints that are not pre-rendered are listed as `include` entries, telling Cloudflare to invoke these routes as functions. `exclude` entries are only used to resolve conflicts. Usually the best strategy when your website has mostly static pages and only a few dynamic pages or endpoints. + + Example: For `src/pages/index.astro` (static), `src/pages/company.astro` (static), `src/pages/users/faq.astro` (static) and `/src/pages/users/[id].astro` (SSR) this will produce the following `_routes.json`: + + ```json + { + "version": 1, + "include": [ + "/_image", // Astro's image endpoint + "/users/*" // Dynamic route + ], + "exclude": [ + // Static routes that needs to be exempted from the dynamic wildcard route above + "/users/faq/", + "/users/faq/index.html" + ] + } + ``` + +- **`exclude`:** Pre-rendered pages are listed as `exclude` entries (telling Cloudflare to handle these routes as static assets). Usually the best strategy when your website has mostly dynamic pages or endpoints and only a few static pages. + + Example: For the same pages as in the previous example this will produce the following `_routes.json`: + + ```json + { + "version": 1, + "include": [ + "/*" // Handle everything as function except the routes below + ], + "exclude": [ + // All static assets + "/", + "/company/", + "/index.html", + "/users/faq/", + "/favicon.png", + "/company/index.html", + "/users/faq/index.html" + ] + } + ``` ### routes.include From 0002115474e633f330dcb6892124a16dec6b558e Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Thu, 14 Sep 2023 10:23:59 +0200 Subject: [PATCH 09/12] worked on tests --- .../test/fixtures/routes-json/public/_redirects | 1 + .../routes-json/{src/mixed => }/public/public.txt | 0 .../dynamicOnly/pages/{another.astro => dynamic1.astro} | 0 .../routes-json/src/dynamicOnly/pages/dynamic2.astro | 5 +++++ .../routes-json/src/dynamicOnly/pages/dynamic3.astro | 5 +++++ packages/integrations/cloudflare/test/routes-json.test.js | 8 ++++---- 6 files changed, 15 insertions(+), 4 deletions(-) create mode 100644 packages/integrations/cloudflare/test/fixtures/routes-json/public/_redirects rename packages/integrations/cloudflare/test/fixtures/routes-json/{src/mixed => }/public/public.txt (100%) rename packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/{another.astro => dynamic1.astro} (100%) create mode 100644 packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic2.astro create mode 100644 packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic3.astro diff --git a/packages/integrations/cloudflare/test/fixtures/routes-json/public/_redirects b/packages/integrations/cloudflare/test/fixtures/routes-json/public/_redirects new file mode 100644 index 000000000000..14e38c465f84 --- /dev/null +++ b/packages/integrations/cloudflare/test/fixtures/routes-json/public/_redirects @@ -0,0 +1 @@ +/redirectme / 302 diff --git a/packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/public/public.txt b/packages/integrations/cloudflare/test/fixtures/routes-json/public/public.txt similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routes-json/src/mixed/public/public.txt rename to packages/integrations/cloudflare/test/fixtures/routes-json/public/public.txt diff --git a/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/another.astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic1.astro similarity index 100% rename from packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/another.astro rename to packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic1.astro diff --git a/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic2.astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic2.astro new file mode 100644 index 000000000000..9a2306b864fa --- /dev/null +++ b/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic2.astro @@ -0,0 +1,5 @@ +--- +export const prerender=false; +--- + +ok diff --git a/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic3.astro b/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic3.astro new file mode 100644 index 000000000000..9a2306b864fa --- /dev/null +++ b/packages/integrations/cloudflare/test/fixtures/routes-json/src/dynamicOnly/pages/dynamic3.astro @@ -0,0 +1,5 @@ +--- +export const prerender=false; +--- + +ok diff --git a/packages/integrations/cloudflare/test/routes-json.test.js b/packages/integrations/cloudflare/test/routes-json.test.js index 58791480f748..0ca0b60aae0f 100644 --- a/packages/integrations/cloudflare/test/routes-json.test.js +++ b/packages/integrations/cloudflare/test/routes-json.test.js @@ -47,14 +47,14 @@ describe('_routes.json generation', () => { await fixture.build(); }); - it('creates a wildcard `include` and `exclude` only for the redirect', async () => { + it('creates a wildcard `include` and `exclude` only for static assets and redirects', async () => { const _routesJson = await fixture.readFile('/_routes.json'); const routes = JSON.parse(_routesJson); expect(routes).to.deep.equal({ version: 1, include: ['/*'], - exclude: ['/a/redirect'], + exclude: ['/public.txt', '/redirectme', '/a/redirect'], }); }); }); @@ -106,7 +106,7 @@ describe('_routes.json generation', () => { expect(routes).to.deep.equal({ version: 1, - include: ['/', '/_image', '/another'], + include: ['/', '/_image', '/dynamic1', '/dynamic2', '/dynamic3'], exclude: [], }); }); @@ -134,7 +134,7 @@ describe('_routes.json generation', () => { expect(routes).to.deep.equal({ version: 1, include: ['/*'], - exclude: ['/', '/index.html', '/a/redirect'], + exclude: ['/', '/index.html', '/public.txt', '/redirectme', '/a/redirect'], }); }); }); From 74a6a2d8eda24b960817303627ef0ec615acc8c1 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Thu, 14 Sep 2023 10:28:43 +0200 Subject: [PATCH 10/12] updated pnpm-lock.yaml --- pnpm-lock.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index e178a0ee91bb..797ec056267f 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -3711,7 +3711,7 @@ importers: specifier: workspace:* version: link:../../../../../astro - packages/integrations/cloudflare/test/fixtures/routesJson: + packages/integrations/cloudflare/test/fixtures/routes-json: dependencies: '@astrojs/cloudflare': specifier: workspace:* From 8ee05ad6adebd102019e265fb1a24e0c1f2380a2 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Thu, 14 Sep 2023 13:38:53 +0200 Subject: [PATCH 11/12] worked on tests --- .../cloudflare/test/routes-json.test.js | 312 +++++++++--------- 1 file changed, 160 insertions(+), 152 deletions(-) diff --git a/packages/integrations/cloudflare/test/routes-json.test.js b/packages/integrations/cloudflare/test/routes-json.test.js index 0ca0b60aae0f..9c5cfad4a181 100644 --- a/packages/integrations/cloudflare/test/routes-json.test.js +++ b/packages/integrations/cloudflare/test/routes-json.test.js @@ -5,199 +5,207 @@ import cloudflare from '../dist/index.js'; /** @type {import('./test-utils.js').Fixture} */ describe('_routes.json generation', () => { for (const mode of ['directory', 'advanced']) { - describe(`in mode "${mode}"`, () => { - describe('of both functions and static files', () => { - let fixture; - - before(async () => { - fixture = await loadFixture({ - root: './fixtures/routes-json/', - srcDir: './src/mixed', - adapter: cloudflare({ - mode, - functionPerRoute: true, - }), + for (const functionPerRoute of [false, true]) { + describe(`with mode=${mode}, functionPerRoute=${functionPerRoute}`, () => { + describe('of both functions and static files', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/mixed', + adapter: cloudflare({ + mode, + functionPerRoute, + }), + }); + await fixture.build(); }); - await fixture.build(); - }); - it('creates `include` for functions and `exclude` for static files where needed', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); - expect(routes).to.deep.equal({ - version: 1, - include: ['/a/*', '/_image'], - exclude: ['/a/', '/a/redirect', '/a/index.html'], + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image'], + exclude: ['/a/', '/a/redirect', '/a/index.html'], + }); }); }); - }); - describe('of only functions', () => { - let fixture; - - before(async () => { - fixture = await loadFixture({ - root: './fixtures/routes-json/', - srcDir: './src/dynamicOnly', - adapter: cloudflare({ - mode, - }), + describe('of only functions', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/dynamicOnly', + adapter: cloudflare({ + mode, + functionPerRoute, + }), + }); + await fixture.build(); }); - await fixture.build(); - }); - it('creates a wildcard `include` and `exclude` only for static assets and redirects', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); + it('creates a wildcard `include` and `exclude` only for static assets and redirects', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); - expect(routes).to.deep.equal({ - version: 1, - include: ['/*'], - exclude: ['/public.txt', '/redirectme', '/a/redirect'], + expect(routes).to.deep.equal({ + version: 1, + include: ['/*'], + exclude: ['/public.txt', '/redirectme', '/a/redirect'], + }); }); }); - }); - - describe('of only static files', () => { - let fixture; - before(async () => { - fixture = await loadFixture({ - root: './fixtures/routes-json/', - srcDir: './src/staticOnly', - adapter: cloudflare({ - mode, - }), + describe('of only static files', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/staticOnly', + adapter: cloudflare({ + mode, + functionPerRoute, + }), + }); + await fixture.build(); }); - await fixture.build(); - }); - it('create only one `include` and `exclude` that are supposed to match nothing', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); + it('create only one `include` and `exclude` that are supposed to match nothing', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); - expect(routes).to.deep.equal({ - version: 1, - include: ['/_image'], - exclude: [], + expect(routes).to.deep.equal({ + version: 1, + include: ['/_image'], + exclude: [], + }); }); }); - }); - describe('with strategy `"include"`', () => { - let fixture; - - before(async () => { - fixture = await loadFixture({ - root: './fixtures/routes-json/', - srcDir: './src/dynamicOnly', - adapter: cloudflare({ - mode, - routes: { strategy: 'include' }, - }), + describe('with strategy `"include"`', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/dynamicOnly', + adapter: cloudflare({ + mode, + functionPerRoute, + routes: { strategy: 'include' }, + }), + }); + await fixture.build(); }); - await fixture.build(); - }); - it('creates `include` entries even though the `"exclude"` strategy would have produced less entries.', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); + it('creates `include` entries even though the `"exclude"` strategy would have produced less entries.', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); - expect(routes).to.deep.equal({ - version: 1, - include: ['/', '/_image', '/dynamic1', '/dynamic2', '/dynamic3'], - exclude: [], + expect(routes).to.deep.equal({ + version: 1, + include: ['/', '/_image', '/dynamic1', '/dynamic2', '/dynamic3'], + exclude: [], + }); }); }); - }); - describe('with strategy `"exclude"`', () => { - let fixture; - - before(async () => { - fixture = await loadFixture({ - root: './fixtures/routes-json/', - srcDir: './src/staticOnly', - adapter: cloudflare({ - mode, - routes: { strategy: 'exclude' }, - }), + describe('with strategy `"exclude"`', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/staticOnly', + adapter: cloudflare({ + mode, + functionPerRoute, + routes: { strategy: 'exclude' }, + }), + }); + await fixture.build(); }); - await fixture.build(); - }); - it('creates `exclude` entries even though the `"include"` strategy would have produced less entries.', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); + it('creates `exclude` entries even though the `"include"` strategy would have produced less entries.', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); - expect(routes).to.deep.equal({ - version: 1, - include: ['/*'], - exclude: ['/', '/index.html', '/public.txt', '/redirectme', '/a/redirect'], + expect(routes).to.deep.equal({ + version: 1, + include: ['/*'], + exclude: ['/', '/index.html', '/public.txt', '/redirectme', '/a/redirect'], + }); }); }); - }); - describe('with additional `include` entries', () => { - let fixture; - - before(async () => { - fixture = await loadFixture({ - root: './fixtures/routes-json/', - srcDir: './src/mixed', - adapter: cloudflare({ - mode, - routes: { - strategy: 'include', - include: ['/another', '/a/redundant'], - }, - }), + describe('with additional `include` entries', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/mixed', + adapter: cloudflare({ + mode, + functionPerRoute, + routes: { + strategy: 'include', + include: ['/another', '/a/redundant'], + }, + }), + }); + await fixture.build(); }); - await fixture.build(); - }); - it('creates `include` for functions and `exclude` for static files where needed', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); - expect(routes).to.deep.equal({ - version: 1, - include: ['/a/*', '/_image', '/another'], - exclude: ['/a/', '/a/redirect', '/a/index.html'], + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image', '/another'], + exclude: ['/a/', '/a/redirect', '/a/index.html'], + }); }); }); - }); - describe('with additional `exclude` entries', () => { - let fixture; - - before(async () => { - fixture = await loadFixture({ - root: './fixtures/routes-json/', - srcDir: './src/mixed', - adapter: cloudflare({ - mode, - routes: { - strategy: 'include', - exclude: ['/another', '/a/*', '/a/index.html'], - }, - }), + describe('with additional `exclude` entries', () => { + let fixture; + + before(async () => { + fixture = await loadFixture({ + root: './fixtures/routes-json/', + srcDir: './src/mixed', + adapter: cloudflare({ + mode, + functionPerRoute, + routes: { + strategy: 'include', + exclude: ['/another', '/a/*', '/a/index.html'], + }, + }), + }); + await fixture.build(); }); - await fixture.build(); - }); - it('creates `include` for functions and `exclude` for static files where needed', async () => { - const _routesJson = await fixture.readFile('/_routes.json'); - const routes = JSON.parse(_routesJson); + it('creates `include` for functions and `exclude` for static files where needed', async () => { + const _routesJson = await fixture.readFile('/_routes.json'); + const routes = JSON.parse(_routesJson); - expect(routes).to.deep.equal({ - version: 1, - include: ['/a/*', '/_image'], - exclude: ['/a/', '/a/*', '/another'], + expect(routes).to.deep.equal({ + version: 1, + include: ['/a/*', '/_image'], + exclude: ['/a/', '/a/*', '/another'], + }); }); }); }); - }); + } } }); From b07d3ff13a1f6ac242d978de6cca4c791e097844 Mon Sep 17 00:00:00 2001 From: Marco Schumacher Date: Sat, 23 Sep 2023 21:05:08 +0200 Subject: [PATCH 12/12] moved the _worker.js in cloudflareSpecialFiles statement --- packages/integrations/cloudflare/src/index.ts | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/packages/integrations/cloudflare/src/index.ts b/packages/integrations/cloudflare/src/index.ts index 1965419c7c3c..792aec196b2f 100644 --- a/packages/integrations/cloudflare/src/index.ts +++ b/packages/integrations/cloudflare/src/index.ts @@ -480,9 +480,6 @@ export default function createIntegration(args?: Options): AstroIntegration { // move cloudflare specific files to the root const cloudflareSpecialFiles = ['_headers', '_redirects', '_routes.json']; - if (!isModeDirectory) { - cloudflareSpecialFiles.push('_worker.js'); - } if (_config.base !== '/') { for (const file of cloudflareSpecialFiles) { @@ -497,6 +494,11 @@ export default function createIntegration(args?: Options): AstroIntegration { } } + // Add also the worker file so it's excluded from the _routes.json generation + if (!isModeDirectory) { + cloudflareSpecialFiles.push('_worker.js'); + } + const routesExists = await fs.promises .stat(new URL('./_routes.json', _config.outDir)) .then((stat) => stat.isFile())