Skip to content

Commit 0463451

Browse files
dario-piotrowiczbmuenzenmeyerIgorMinar
committed
make site work with the Cloudflare OpenNext adapter
update the site application so that it can be build using the Cloudflare OpenNext adapter (`@opennextjs/cloudflare`) and thus deployed on Cloudflare Workers > [!Note] > This is very experimental and full of hacks > it's very much a work-in-progress right now ___ Co-authored-by: Brian Muenzenmeyer <[email protected]> Co-authored-by: Igor Minar <[email protected]>
1 parent 060f050 commit 0463451

27 files changed

+4997
-6811
lines changed

.gitignore

+7
Original file line numberDiff line numberDiff line change
@@ -32,3 +32,10 @@ cache
3232
tsconfig.tsbuildinfo
3333

3434
dist/
35+
36+
# Ignore worker artifacts
37+
apps/site/.open-next
38+
apps/site/.wrangler
39+
40+
# Pre-build generated files
41+
apps/site/.generated

apps/site/.cloudflare/empty.mjs

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export default {};

apps/site/.cloudflare/node/fs.mjs

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
import { files } from '../../.generated/next.helpers.mjs';
2+
3+
export function readdir(params, cb) {
4+
console.log('fs#readdir', params);
5+
cb(null, []);
6+
}
7+
8+
export function exists(path, cb) {
9+
const result =
10+
files.includes(path) || files.includes(path.replace(/^\//, ''));
11+
console.log('fs#exists', path, result);
12+
cb(result);
13+
}
14+
15+
export default {
16+
readdir,
17+
exists,
18+
};
+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import { getCloudflareContext } from '@opennextjs/cloudflare';
2+
3+
export async function readFile(path) {
4+
console.log('fs/promies#readFile', path);
5+
6+
const { env } = await getCloudflareContext();
7+
8+
const text = await env.ASSETS.fetch(
9+
new URL(path, 'https://jamesrocks/')
10+
).then(response => response.text());
11+
return text;
12+
}
13+
14+
export async function readdir(params) {
15+
console.log('fs/promises#readdir', params);
16+
return Promise.resolve([]);
17+
}
18+
19+
export async function exists(...args) {
20+
console.log('fs/promises#exists', args);
21+
return Promise.resolve(false);
22+
}
23+
24+
export default {
25+
readdir,
26+
exists,
27+
readFile,
28+
};
+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
// we shim @opentelemetry/api to the throwing shim so that it will throw right away, this is so that we throw inside the
2+
// try block here: https://github.com/vercel/next.js/blob/9e8266a7/packages/next/src/server/lib/trace/tracer.ts#L27-L31
3+
// causing the code to require the 'next/dist/compiled/@opentelemetry/api' module instead (which properly works)
4+
5+
// IMPORTANT: we already do that in the open-next Cloudflare adapter, it shouldn't be necessary here too
6+
// (https://github.com/opennextjs/opennextjs-cloudflare/issues/219 seems to be the same issue)
7+
throw new Error();

apps/site/.cloudflare/server-only.mjs

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
// In our aliased fs code: apps/site/.cloudflare/node/fs/promises.mjs we are importing `getCloudflareContext`
2+
// from `@opennextjs/cloudflare`, this in turn imports from `server-only`, this aliasing makes it so that
3+
// server-only is not actually removed from the final bundle as it would otherwise cause an incorrect server
4+
// internal error

apps/site/CLOUDFLARE.md

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
# nodejs.org on OpenNext for Cloudflare
2+
3+
## Getting started
4+
5+
To develop, build, preview, and deploy nodejs.org, execute the following commands to get started:
6+
7+
```
8+
nvm use
9+
npm install
10+
cd apps/site
11+
```
12+
13+
## Developing locally
14+
15+
To develop locally, run the usual:
16+
17+
```
18+
npm run dev
19+
```
20+
21+
## Build nodejs.org production distribution using OpenNext
22+
23+
To build you need connection to the Internet because the build system will try to fetch the following files:
24+
25+
- https://nodejs.org/dist/index.json
26+
- https://raw.githubusercontent.com/nodejs/Release/master/schedule.json
27+
28+
```
29+
npm run cf:build
30+
```
31+
32+
## Preview a production build locally
33+
34+
You can preview production build locally using [wrangler](https://developers.cloudflare.com/workers/wrangler/):
35+
36+
```
37+
npm run cf:preview
38+
```
39+
40+
## Deploying a build to production
41+
42+
To build and deploy the application run:
43+
44+
```
45+
npm run cf:deploy
46+
```
47+
48+
The build is currently deployed to a dedicated "nodejs.org" (Cloudflare account id: 8ed4d03ac99f77561d0e8c9cbcc76cb6): https://nodejs-website.web-experiments.workers.dev
49+
50+
You can monitor and configure the project at https://dash.cloudflare.com/8ed4d03ac99f77561d0e8c9cbcc76cb6/workers/services/view/nodejs-website/production
51+
52+
## TODOs
53+
54+
The following is an incomplete list of tasks and problems that still need to be resolved:
55+
56+
- [x] update `@opennextjs/cloudflare` to the latest in `/apps/site/package.json`
57+
- [x] sort out issues with `eval` and MDX (Claudio is looking into this one)
58+
- [x] and undo edits in `./app/[locale]/[[...path]]/page.tsx`
59+
- [x] reimplement `getMarkdownFiles` in `next.helpers.mjs` to be generated at build time
60+
- this can be accomplished either via a npm/turbo prebuild task, or possibly as part of next.js SSG/staticProps but
61+
- [ ] we need to ensure that we don't end up accidentally downloading this big file to the client as part of hydration
62+
- [x] once we have easy access to the list of files, we should roll back changes to `next-data/providers/blogData.ts`
63+
- [x] back out most changes from `next.dynamic.mjs`
64+
- [x] instead of using runtime detection via `globalThis.navigator?.userAgent`, we should instead use `alias` feature in `wrangler.toml` to override the implementation of `node:fs` calls but only when running in workerd as we need the build to keep on running in node.js for SSG to work
65+
- [x] could we reimplement the `existsAsync` call as sync `exists` which consults `getMarkdownFiles` from the task above?
66+
- [ ] remove symlink hack in `package.json#build:cloudflare`
67+
- would it be possible to make the pages directory part of assets in a less hacky way?
68+
- [ ] move these files under `.open-next/assets/cdn-cgi/pages` so that these raw md files are not publicly accessible as that could become a maintenance burden down the road.
69+
- [ ] wire up the changes with turborepo (right now just plain npm scripts are used)
70+
- [ ] reenable minification in `next.config.mjs`
71+
- [ ] remove as many `alias`es as possible from the `wrangler.toml` file
72+
(the `alias`es that can't be removed should be fully investigated and documented)
73+
- [ ] fix flashes of unstyled content present on hard navigation
74+
- [x] enable caching
75+
- [x] fix routes for languages besides `en` 404ing

apps/site/app/[locale]/next-data/api-data/route.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ const getPathnameForApiFile = (name: string, version: string) =>
2121
// for a digest and metadata of all API pages from the Node.js Website
2222
// @see https://nextjs.org/docs/app/building-your-application/routing/router-handlers
2323
export const GET = async () => {
24-
const releases = provideReleaseData();
24+
const releases = await provideReleaseData();
2525

2626
const { versionWithPrefix } = releases.find(
2727
release => release.status === 'LTS'

apps/site/app/[locale]/next-data/release-data/route.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ import { defaultLocale } from '@/next.locales.mjs';
55
// for generating static data related to the Node.js Release Data
66
// @see https://nextjs.org/docs/app/building-your-application/routing/router-handlers
77
export const GET = async () => {
8-
const releaseData = provideReleaseData();
8+
const releaseData = await provideReleaseData();
99

1010
return Response.json(releaseData);
1111
};

apps/site/instrumentation.ts

-5
This file was deleted.

apps/site/next-data/generators/blogData.mjs

+12-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ import readline from 'node:readline';
66

77
import graymatter from 'gray-matter';
88

9-
import { getMarkdownFiles } from '../../next.helpers.mjs';
9+
import { getMarkdownFiles } from '../../.generated/next.helpers.mjs';
1010

1111
// gets the current blog path based on local module path
1212
const blogPath = join(process.cwd(), 'pages/en/blog');
@@ -63,6 +63,17 @@ const generateBlogData = async () => {
6363
'**/index.md',
6464
]);
6565

66+
const result = {
67+
/* generated at build time */
68+
};
69+
70+
if (Object.keys(result).length > 0) {
71+
return {
72+
...result,
73+
posts: result.posts.map(post => ({ ...post, date: new Date(post.date) })),
74+
};
75+
}
76+
6677
return new Promise(resolve => {
6778
const posts = [];
6879
const rawFrontmatter = [];

apps/site/next-data/generators/downloadSnippets.mjs

+18-1
Original file line numberDiff line numberDiff line change
@@ -7,11 +7,15 @@ import { glob } from 'glob';
77

88
import { availableLocaleCodes } from '../../next.locales.mjs';
99

10+
const preGeneratedDownloadSnippets = [
11+
/* generated at build time */
12+
];
13+
1014
/**
1115
* This method is used to generate the Node.js Website Download Snippets
1216
* for self-consumption during RSC and Static Builds
1317
*/
14-
const generateDownloadSnippets = async () => {
18+
export const generateRawDownloadSnippets = async () => {
1519
/**
1620
* This generates all the Download Snippets for each available Locale
1721
*
@@ -37,6 +41,19 @@ const generateDownloadSnippets = async () => {
3741

3842
return [locale, await Promise.all(snippets)];
3943
});
44+
return await Promise.all(downloadSnippets);
45+
};
46+
47+
/**
48+
* This method is used to generate the Node.js Website Download Snippets
49+
* for self-consumption during RSC and Static Builds
50+
*/
51+
const generateDownloadSnippets = async () => {
52+
if (preGeneratedDownloadSnippets) {
53+
return new Map(preGeneratedDownloadSnippets);
54+
}
55+
56+
const downloadSnippets = generateRawDownloadSnippets();
4057

4158
return new Map(await Promise.all(downloadSnippets));
4259
};

apps/site/next-data/providers/blogData.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,16 @@
11
import { cache } from 'react';
22

3-
import generateBlogData from '@/next-data/generators/blogData.mjs';
43
import { BLOG_POSTS_PER_PAGE } from '@/next.constants.mjs';
5-
import type { BlogCategory, BlogPostsRSC } from '@/types';
4+
import type { BlogCategory, BlogPost, BlogPostsRSC } from '@/types';
5+
import generateBlogData from '@generated/blogData.mjs';
66

77
const { categories, posts } = await generateBlogData();
88

99
export const provideBlogCategories = cache(() => categories);
1010

1111
export const provideBlogPosts = cache(
1212
(category: BlogCategory): BlogPostsRSC => {
13-
const categoryPosts = posts
13+
const categoryPosts = (posts as Array<BlogPost>)
1414
.filter(post => post.categories.includes(category))
1515
.sort((a, b) => b.date.getTime() - a.date.getTime());
1616

apps/site/next-data/providers/downloadSnippets.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { cache } from 'react';
22

3-
import generateDownloadSnippets from '@/next-data/generators/downloadSnippets.mjs';
3+
import generateDownloadSnippets from '@generated/downloadSnippets.mjs';
44

55
const downloadSnippets = await generateDownloadSnippets();
66

apps/site/next-data/providers/releaseData.ts

+1-3
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,6 @@ import { cache } from 'react';
22

33
import generateReleaseData from '@/next-data/generators/releaseData.mjs';
44

5-
const releaseData = await generateReleaseData();
6-
7-
const provideReleaseData = cache(() => releaseData);
5+
const provideReleaseData = cache(() => generateReleaseData());
86

97
export default provideReleaseData;

apps/site/next.config.mjs

+9
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,15 @@ const nextConfig = {
6969
// as we already check it on the CI within each Pull Request
7070
// we also configure ESLint to run its lint checking on all files (next lint)
7171
eslint: { dirs: ['.'], ignoreDuringBuilds: true },
72+
// Adds custom WebPack configuration to our Next.js setup
73+
webpack: function (config) {
74+
// CF hacking: temporarily disable minification to make debugging easier
75+
config.optimization = {
76+
minimize: false,
77+
};
78+
79+
return config;
80+
},
7281
experimental: {
7382
// Ensure that server-side code is also minified
7483
serverMinification: true,

apps/site/next.dynamic.mjs

+29-5
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,15 @@
11
'use strict';
22

3+
import { exists } from 'node:fs';
34
import { readFile } from 'node:fs/promises';
45
import { join, normalize, sep } from 'node:path';
56

67
import matter from 'gray-matter';
78
import { cache } from 'react';
89
import { VFile } from 'vfile';
910

11+
import { getMarkdownFiles } from '@generated/next.helpers.mjs';
12+
1013
import {
1114
BASE_PATH,
1215
BASE_URL,
@@ -19,7 +22,6 @@ import {
1922
IGNORED_ROUTES,
2023
PAGE_METADATA,
2124
} from './next.dynamic.constants.mjs';
22-
import { getMarkdownFiles } from './next.helpers.mjs';
2325
import { siteConfig } from './next.json.mjs';
2426
import { availableLocaleCodes, defaultLocale } from './next.locales.mjs';
2527
import { compile } from './next.mdx.compiler.mjs';
@@ -62,7 +64,7 @@ const getDynamicRouter = async () => {
6264

6365
const websitePages = await getMarkdownFiles(
6466
process.cwd(),
65-
`pages/${defaultLocale.code}`
67+
`/pages/${defaultLocale.code}`
6668
);
6769

6870
websitePages.forEach(filename => {
@@ -110,9 +112,20 @@ const getDynamicRouter = async () => {
110112

111113
// This verifies if the given pathname actually exists on our Map
112114
// meaning that the route exists on the website and can be rendered
113-
if (pathnameToFilename.has(normalizedPathname)) {
114-
const filename = pathnameToFilename.get(normalizedPathname);
115-
const filepath = join(process.cwd(), 'pages', locale, filename);
115+
if (
116+
pathnameToFilename.has(normalizedPathname) ||
117+
pathnameToFilename.has(
118+
`pages/en${normalizedPathname ? `/${normalizedPathname}` : ''}`
119+
)
120+
) {
121+
const filename = (
122+
pathnameToFilename.get(normalizedPathname) ??
123+
pathnameToFilename.get(
124+
`pages/en${normalizedPathname ? `/${normalizedPathname}` : ''}`
125+
)
126+
).replace(new RegExp(`^pages/en/`), '');
127+
128+
let filepath = join(process.cwd(), 'pages', locale, filename);
116129

117130
// We verify if our Markdown cache already has a cache entry for a localized
118131
// version of this file, because if not, it means that either
@@ -143,6 +156,17 @@ const getDynamicRouter = async () => {
143156
return { source: fileLanguageContent, filename };
144157
}
145158

159+
const existsPromise = path =>
160+
new Promise(resolve => exists(path, resolve));
161+
162+
// and return the current fetched result; If the file does not exist
163+
// we fallback to the English source
164+
if (await existsPromise(join(filepath, locale, filename))) {
165+
filepath = join(filepath, locale, filename);
166+
167+
return { source: fileLanguageContent, filename };
168+
}
169+
146170
// Prevent infinite loops as if at this point the file does not exist with the default locale
147171
// then there must be an issue on the file system or there's an error on the mapping of paths to files
148172
if (locale === defaultLocale.code) {

0 commit comments

Comments
 (0)