diff --git a/.changeset/cuddly-bulldogs-deny.md b/.changeset/cuddly-bulldogs-deny.md new file mode 100644 index 000000000..17a73a792 --- /dev/null +++ b/.changeset/cuddly-bulldogs-deny.md @@ -0,0 +1,6 @@ +--- +'eslint-plugin-next-on-pages': patch +'@cloudflare/next-on-pages': patch +--- + +Simplify documentation diff --git a/README.md b/README.md index 83db8f802..4de589385 100644 --- a/README.md +++ b/README.md @@ -1,42 +1,17 @@ -

-

⚡▲ @cloudflare/next-on-pages ▲⚡

+# `@cloudflare/next-on-pages` -

Build, develop, and deploy Next.js apps for Cloudflare Pages.

-

+`@cloudflare/next-on-pages` lets you build and deploy [Next.js](https://nextjs.org/) apps to [Cloudflare Pages](https://pages.cloudflare.com/). -`@cloudflare/next-on-pages` is a CLI tool that you can use to build and develop [Next.js](https://nextjs.org/) applications so that they can run on the [Cloudflare Pages](https://pages.cloudflare.com/) platform. +## Get started -Alongside the `@cloudflare/next-on-pages` there is an additional package `eslint-plugin-next-on-pages` implementing an Eslint plugin which aim is to aid developers at using the `@cloudflare/next-on-pages` more efficiently and improve their overall developer experience when working with it. +- [Getting Started Guide](https://developers.cloudflare.com/pages/framework-guides/nextjs/deploy-a-nextjs-site/) -You can see the packages contents (with their documentation) in their respective package directories: +## Components -- [`@cloudflare/next-on-pages`](https://github.com/cloudflare/next-on-pages/tree/main/packages/next-on-pages#cloudflarenext-on-pages) -- [`eslint-plugin-next-on-pages`](https://github.com/cloudflare/next-on-pages/tree/main/packages/eslint-plugin-next-on-pages#eslint-plugin-next-on-pages) - -Additionally there is also the `next-dev` submodule which is implemented as a separate package in this repository but included as a submodule of the main `@cloudflare/next-on-pages` package, you can see the submodule's content here: - -- [`@cloudflare/next-on-pages/next-dev`](https://github.com/cloudflare/next-on-pages/tree/main/internal-packages/next-dev) +- [`@cloudflare/next-on-pages`](https://github.com/cloudflare/next-on-pages/tree/main/packages/next-on-pages) — a CLI that runs `next build`, then transforms its output to run on Cloudflare Pages. +- [`@cloudflare/next-on-pages/next-dev`](https://github.com/cloudflare/next-on-pages/tree/main/internal-packages/next-dev) — lets you access [bindings](https://developers.cloudflare.com/workers/runtime-apis/bindings/) in local development of your Next.js app. +- [`eslint-plugin-next-on-pages`](https://github.com/cloudflare/next-on-pages/tree/main/packages/eslint-plugin-next-on-pages#eslint-plugin-next-on-pages) — lints your Next.js app to ensure it is configured to work on Cloudflare Pages. ## Contributing -If you want to contribute to this project (both to the main package and the eslint one) please refer to the [Contributing document](./docs/contributing.md). - -## References - -Extra references you might be interested in: - -- [Blog post](https://blog.cloudflare.com/next-on-pages) - - The original blog post introducing `@cloudflare/next-on-pages` (24/10/2022), it goes into details on the inspiration for this package and provides some details on how it works. - -- [Cloudflare Next.js Guide](https://developers.cloudflare.com/pages/framework-guides/deploy-a-nextjs-site/) - - Cloudflare guide on how to create and deploy a Next.js application. The application can be either static (and deployed as simple static assets) or dynamic using the edge runtime (using `@cloudflare/next-on-pages`). - -- [Caching and Data Revalidation](./packages/next-on-pages/docs/caching.md) - - Documentation on how the caching and data revalidation for applications built using `@cloudflare/next-on-pages` works. - -- [Technical Documentation](./docs/technical) - - Explanations and insights into how `@cloudflare/next-on-pages` works, design decisions behind different aspects, and how it handles different Next.js features. +We welcome external contributions — please refer to the [contribution guidelines](./docs/contributing.md). diff --git a/docs/contributing.md b/docs/contributing.md index 5bcb07676..860aa2fb5 100644 --- a/docs/contributing.md +++ b/docs/contributing.md @@ -1,23 +1,5 @@ # Contributing -We're thrilled that you're considering contributing to the project! All contributions are greatly appreciated, regardless if they are simple issue reporting, bug fixes, documentation changes, etc... - -No contribution is too small and all help moving the project forward! - -## Submitting an bug report or feature request - -If you find an bug or want to request a feature, open a [new issue](https://github.com/cloudflare/next-on-pages/issues/new) documenting it, please try to fill as much of the requested information as possible as that will help us dealing with the issue more effectively. - -## Pull Requests - -When opening a new PR make sure to set up an informative title for it and provide as much details as you can in the PR's description so that it is clear what its intentions are. This helps in simplifying and speeding up the reviewing process. - -If your PR is addressing an existing issue make sure that it references the issue (as per the [Github PR Issues linking documentation](https://docs.github.com/en/issues/tracking-your-work-with-issues/linking-a-pull-request-to-an-issue)) so that the two can be linked correctly (and the issue gets automatically closed on merge). - -Smaller PRs are preferred since they are easier to review and merge quickly. If you're planning to make multiple unrelated (or loosely related) changes please consider splitting them in multiple PRs. - - - ### Not sure what to contribute on? If you're looking for something to help us with, what we believe would be most valuable to us is to either: try to implement Next.js functionality which we currently don't support (which you can see in the [Supported Versions and Features document](https://github.com/cloudflare/next-on-pages/blob/main/packages/next-on-pages/docs/supported.md)), or help us fix existing [issues](https://github.com/cloudflare/next-on-pages/issues). diff --git a/internal-packages/next-dev/README.md b/internal-packages/next-dev/README.md index 3ba75a0dd..d81e215f0 100644 --- a/internal-packages/next-dev/README.md +++ b/internal-packages/next-dev/README.md @@ -1,82 +1,32 @@ -# Next-on-pages Next-Dev +# `next-dev` (`setupDevPlatform`) -The `next-dev` submodule of the `@cloudflare/next-on-pages` package implements a utility that allows you to run the [standard Next.js development server](https://nextjs.org/docs/app/api-reference/next-cli#development) (with hot-code reloading, error reporting, HMR and everything it has to offer) with also local Cloudflare integration. - -**IMPORTANT**: As mentioned above the module allows you to run the standard Next.js dev server as is and it only makes sure that a simulation of the Cloudflare specific data (retrievable using `getRequestContext`) is available. It does not generate a worker nor faithfully represent the final application that will be deployed to Cloudflare Pages, so please use this only as a development tool and make sure to properly test your application with `wrangler pages dev` before actually deploying it to Cloudflare. - -## How to use the module - -The module is part of the `@cloudflare/next-on-pages` package so it does not need installation, it exports the `setupDevPlatform` function which you need to import and call in your Next.js config file (`next.config.mjs` or `next.config.(c)js`). The utility will read your [`wrangler.toml`](https://developers.cloudflare.com/workers/wrangler/configuration/) gather information from it and use it to simulate the Cloudflare platform in the in the dev server. - -After having created an appropriate `wrangler.toml` file and added the `setupDevPlatform` call to the Next.js config file you can simply run `next dev` and inside your edge routes you will be able to access your bindings via `getRequestContext` in the exact same way as you would in your production code. - -`setupDevPlatform` uses wrangler's `getPlatformProxy` utility under the hood, it accepts the same exact arguments and supports bindings in the same exact manner, for more details please refer to the official [`getPlatformProxy` documentation](https://developers.cloudflare.com/workers/wrangler/api/#getplatformproxy). +`setupDevPlatform` lets you access [bindings](https://developers.cloudflare.com/workers/runtime-apis/bindings/) in local development of your Next.js app. ### Example -Let's see an example of how to use the utility, in a Next.js application built in TypeScript using the App router. - -Firstly let's define a simple `wrangler.toml` file which only declares bindings: - -```toml -[[kv_namespaces]] -binding = "MY_KV_1" -id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" - -[[kv_namespaces]] -binding = "MY_KV_2" -id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" - -[[durable_objects.bindings]] -name = "MY_DO" -script_name = "do-worker" -class_name = "DurableObjectClass" - -[[r2_buckets]] -binding = "MY_R2" -bucket_name = "my-bucket" -``` - -Then we need update the `next.config.mjs` file: +[`next.config.mjs`](https://nextjs.org/docs/app/api-reference/next-config-js): ```js -// file: next.config.mjs - -// we import the utility from the next-dev submodule import { setupDevPlatform } from '@cloudflare/next-on-pages/next-dev'; -/** @type {import('next').NextConfig} */ const nextConfig = {}; export default nextConfig; -// we only need to use the utility during development so we can check NODE_ENV -// (note: this check is recommended but completely optional) if (process.env.NODE_ENV === 'development') { - // `await`ing the call is not necessary but it helps making sure that the setup has succeeded. - // If you cannot use top level awaits you could use the following to avoid an unhandled rejection: - // `setupDevPlatform().catch(e => console.error(e));` await setupDevPlatform(); } ``` -Next (optional but highly recommended) we create a [TypeScript declaration file](https://www.typescriptlang.org/docs/handbook/2/type-declarations.html) that defines/extends a `CloudflareEnv` interface (the same interface used by `getRequestContext`): +[`wrangler.toml`](https://developers.cloudflare.com/pages/functions/wrangler-configuration/): -```ts -// file: env.d.ts - -interface CloudflareEnv { - MY_KV_1: KVNamespace; - MY_KV_2: KVNamespace; - MY_R2: R2Bucket; - MY_DO: DurableObjectNamespace; -} +```toml +[[kv_namespaces]] +binding = "MY_KV_1" +id = "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" ``` -> **Note** -> The binding types used in the above file come from `@cloudflare/workers-types`, in order to use them make sure that you've installed the package as a dev dependency and you've added it to your `tsconfig.js` file under `compilerOptions.types`. - -Then we can simply use the platform and any of its bindings inside our Next.js application, for example in the following API route: +A route in your Next.js app: ```ts export const runtime = 'edge'; @@ -90,23 +40,3 @@ export async function GET(request: NextRequest) { return new Response(`The value of key-a in MY_KV is: ${valueA}`); } ``` - -## Recommended development workflow - -When developing a `next-on-pages` application, this is the development workflow that we recommend: - -### Develop using the standard Next.js dev server - -Develop your application using the [standard development server provided by Next.js](https://nextjs.org/docs/getting-started/installation#run-the-development-server). It is the best available option for a fast and polished development experience and the `next-dev` submodule makes it possible to use it while also being able to access a faithful simulation of the Cloudflare platform. - -### Build and preview your application locally - -In order to make sure that your application is being built in a manner that is fully compatible with Cloudflare Pages, before deploying it, or whenever you're comfortable checking the correctness of the application during your development process you'll want to build and preview it locally using Cloudflare's `workerd` JavaScript runtime. - -To do so build your worker by using `@cloudflare/next-on-pages` and preview it locally via `wrangler pages dev .vercel/output/static --compatibility-flag nodejs_compat`. - -By doing this, you can run your application locally to make sure everything is working as you expect it to. - -### Deploy your app and iterate - -Once you've previewed your application locally then you can deploy it to Cloudflare Pages (both via [direct uploads](https://developers.cloudflare.com/pages/get-started/direct-upload/) or [git integration](https://developers.cloudflare.com/pages/configuration/git-integration/)) and iterate over the process to make new changes. diff --git a/packages/eslint-plugin-next-on-pages/README.md b/packages/eslint-plugin-next-on-pages/README.md index ed58f14db..42e4c9dc9 100644 --- a/packages/eslint-plugin-next-on-pages/README.md +++ b/packages/eslint-plugin-next-on-pages/README.md @@ -1,24 +1,18 @@ # `eslint-plugin-next-on-pages` -`eslint-plugin-next-on-pages` is an ESlint plugin intended to support developers developing Next.js application via `@cloudflare/next-on-pages`. +`eslint-plugin-next-on-pages` lints your Next.js app to ensure it is configured to work on Cloudflare Pages. ## Setup To install the plugin run: ```sh - npm i --save-dev eslint-plugin-next-on-pages@V + npm i --save-dev eslint-plugin-next-on-pages ``` -where `V` indicates the version of your `@cloudflare/next-on-pages` package. - -> **Note** -> The `eslint-plugin-next-on-pages` package is versioned identically to `@cloudflare/next-on-pages`, this can be used to ensure that the two packages are in sync with each other. For best results make sure that the versions of the two packages are always the same. - -Then simply register the plugin in your eslintrc file. As part of this we suggest to also extend the recommended configuration. After that you can also further tweak the available rules: +Add the plugin to your `.eslintrc.json`: ```diff -// .eslintrc.json { "plugins": [ + "next-on-pages" @@ -36,4 +30,7 @@ Then simply register the plugin in your eslintrc file. As part of this we sugges ## Rules -For more details check out the [rules documentation](https://github.com/cloudflare/next-on-pages/tree/main/packages/eslint-plugin-next-on-pages/docs/rules). +- [`next-on-pages/no-app-nodejs-dynamic-ssg`](https://github.com/cloudflare/next-on-pages/tree/main/packages/eslint-plugin-next-on-pages/docs/rules/no-app-nodejs-dynamic-ssg) +- [`next-on-pages/no-nodejs-runtime`](https://github.com/cloudflare/next-on-pages/tree/main/packages/eslint-plugin-next-on-pages/docs/rules/no-app-nodejs-runtime) +- [`next-on-pages/no-pages-nodejs-dynamic-ssg`](https://github.com/cloudflare/next-on-pages/tree/main/packages/eslint-plugin-next-on-pages/docs/rules/no-pages-nodejs-dynamic-ssg) +- [`next-on-pages/no-unsupported-configs`](https://github.com/cloudflare/next-on-pages/tree/main/packages/eslint-plugin-next-on-pages/docs/rules/no-unsupported-configs) diff --git a/packages/next-on-pages/README.md b/packages/next-on-pages/README.md index a6e9c7512..bfd608abd 100644 --- a/packages/next-on-pages/README.md +++ b/packages/next-on-pages/README.md @@ -1,137 +1,10 @@ # `@cloudflare/next-on-pages` -`@cloudflare/next-on-pages` is a CLI tool that you can use to build and develop [Next.js](https://nextjs.org/) applications so that they can run on the [Cloudflare Pages](https://pages.cloudflare.com/) platform (and integrate with Cloudflare's various other [product offerings](https://developers.cloudflare.com/pages/platform/functions/bindings/), such as KV, D1, R2, and Durable Objects). +## Get Started -This tool is a best-effort library implemented by the Cloudflare team and the community. As such, most, but not all, Next.js features are supported. See the [Supported Versions and Features document](https://github.com/cloudflare/next-on-pages/blob/main/packages/next-on-pages/docs/supported.md) for more details. +Follow [this guide](https://developers.cloudflare.com/pages/framework-guides/nextjs/deploy-a-nextjs-site/) to get started. -## Quick Start +## Components -This section describes how to bundle and deploy a (new or existing) Next.js application to [Cloudflare Pages](https://pages.cloudflare.com), using `@cloudflare/next-on-pages`. - -### 1. Select your Next.js app - -To start using `@cloudflare/next-on-pages`, you must have a Next.js project that you wish to deploy. If you already have one, change to its directory. Otherwise, you can use the `create-next-app` command to start a new one. - -```sh -npx create-next-app@latest my-next-app -cd my-next-app -``` - -
-Note on the Next.js version - -We have confirmed support for the current version of Next.js at the time of writing, `13.4.2`. Although we'll endeavor to keep support for newer versions, we cannot guarantee that we'll always be up-to-date with the latest version. If you experience any problems with `@cloudflare/next-on-pages`, you may wish to try pinning to `13.4.2` while we work on supporting any recent breaking changes. - -
- -### 2. Configure the application to use the Edge Runtime - -For your application to run on Cloudflare Pages, it needs to opt in to use the Edge Runtime for routes containing server-side code (e.g. API Routes or pages that use `getServerSideProps`). To do this, export a `runtime` [route segment config](https://nextjs.org/docs/app/api-reference/file-conventions/route-segment-config#runtime) option from each file, specifying that it should use the Edge Runtime. - -```typescript -export const runtime = 'edge'; -``` - - - -For more examples of this and for Next.js versions prior to v13.3.1, take a look at our [examples document](https://github.com/cloudflare/next-on-pages/blob/main/packages/next-on-pages/docs/examples.md). Additionally, ensure that your application is not using any unsupported [APIs](https://nextjs.org/docs/app/api-reference/edge#unsupported-apis) or [features](https://github.com/cloudflare/next-on-pages/blob/main/packages/next-on-pages/docs/supported.md). - -### 3. Deploy your application to Cloudflare Pages - -To deploy your application to Cloudflare Pages, you need to install the `@cloudflare/next-on-pages` package. - -```sh -npm install -D @cloudflare/next-on-pages -``` - -Then you can deploy to Cloudflare Pages via the [automatic Git integration](https://developers.cloudflare.com/pages/platform/git-integration/). To do so, start by committing and pushing your application's code to a GitHub/GitLab repository. - -Next, in the [Cloudflare Dashboard](https://dash.cloudflare.com/?to=/:account/pages), create a new Pages project: - -- Navigate to the project creation pages (_Your account Home_ > _Workers & Pages_ > _Create application_ > _Pages_ > _Connect to Git_). -- Select the GitHub/GitLab repository you pushed your code to. -- Choose a project name and your production branch. -- Select _Next.js_ as the _Framework preset_ and provide the following options: - | Option | Value | - | ---------------------- | ---------------------------------- | - | Build command | `npx @cloudflare/next-on-pages@1` | - | Build output directory | `.vercel/output/static` | -- In the _Environment variables (advanced)_ section, add a new variable named `NODE_VERSION` set to `16` or greater. -- Click on _Save and Deploy_ to start the deployment (this first deployment won't be fully functional as the next step is also necessary). -- Go to the Pages project settings page (_Settings_ > _Functions_ > _Compatibility Flags_), **add the `nodejs_compat` flag** for both production and preview, and make sure that the **Compatibility Date** for both production and preview is set to at least `2022-11-30`. - -> If you don't want to set up a Git repository, you can build your application (as indicated in [Local Development](#local-development)) and publish it manually via the [`wrangler pages publish` command](https://developers.cloudflare.com/workers/wrangler/commands/#publish-1) instead (you'll still need to set the **`nodejs_compat`** flag for your project in the Cloudflare dashboard). - -> **Note**: -> When deploying via the Git integration, for better compatibility with tools such as `yarn` and `pnpm` we recommend using the Build system version 2 (that is the default so no action is required). - -## Recommended development workflow - -When developing a `next-on-pages` application, this is the development workflow that Cloudflare recommends: - -### Develop using the standard Next.js dev server - -The [standard development server provided by Next.js](https://nextjs.org/docs/getting-started/installation#run-the-development-server) is the best available option for a fast and polished development experience. The `next-dev` submodule makes it possible to use Next.js' standard development server while still having access to your Cloudflare bindings. - -### Build and preview your application locally - -To ensure that your application is being built in a manner that is fully compatible with Cloudflare Pages, before deploying it, or whenever you are comfortable checking the correctness of the application during your development process, you will want to build and preview it locally using Cloudflare's `workerd` JavaScript runtime. - -Do this by running: - -```sh -npx @cloudflare/next-on-pages -``` - -And preview your project by running: - -```sh -npx wrangler pages dev .vercel/output/static -``` - -> [!NOTE] -> The [`wrangler pages dev`](/workers/wrangler/commands/#dev-1) command needs to run the application using the [`nodejs_compat`](/workers/configuration/compatibility-dates/#nodejs-compatibility-flag) compatibility flag. The `nodejs_compat` flag can be specified in either your project's `wrangler.toml` file or provided to the command as an inline argument: `--compatibility-flag=nodejs_compat`. - -### Deploy your application and iterate - -After you have previewed your application locally, you can deploy it to Cloudflare Pages (both via [Direct Uploads](/pages/get-started/direct-upload/) or [Git integration](/pages/configuration/git-integration/)) and iterate over the process to make new changes. - -## Cloudflare Platform Integration - -Next.js applications built using `@cloudflare/next-on-pages` get access to resources and information only available or relevant on the Cloudflare platform, such are: - -- [Bindings (`env`)](https://developers.cloudflare.com/pages/platform/functions/bindings/), which allows you to take advantage of Cloudflare specific resources. -- [Cloudflare properties (`cf`)](https://developers.cloudflare.com/workers/runtime-apis/request/#incomingrequestcfproperties), object containing information about the request provided by Cloudflare’s global network. -- [Lifecycle methods (`ctx`)](https://developers.cloudflare.com/workers/runtime-apis/handlers/fetch/#lifecycle-methods), methods to augment or control how the request is handled. - -Such can be accessed by calling the `getRequestContext` function in server only code. - -For example: - -```ts -import { getRequestContext } from '@cloudflare/next-on-pages'; - -// ... - -const { env, cf, ctx } = getRequestContext(); -``` - -> **Warning**: The function cannot be called in code from components using the Pages router. - -> **Note**: In order to make the function work in development mode (using the standard Next.js dev server) use the [`@cloudflare/next-on-pages/next-dev`](https://github.com/cloudflare/next-on-pages/tree/main/internal-packages/next-dev) submodule. - -> **TypeScript Env Type**: the `env` object returned by `getRequestContext` implements the `CloudflareEnv` interface, add your binding types to such interface in order for get a correctly typed `env` object. - -> **Note**: `getRequestContext` throws an error if invoked when the request context is not available, if you prefer to receive `undefined` in such cases use `getOptionalRequestContext` instead, the latter is identical to `getRequestContext` except from the fact that it returns `undefined` when the context is not available. - -## Examples - -To see some examples on how to use Next.js features with `@cloudflare/next-on-pages`, see the [Examples document](https://github.com/cloudflare/next-on-pages/blob/main/packages/next-on-pages/docs/examples.md). - -## Troubleshooting - -If you find yourself hitting some issues with `@cloudflare/next-on-pages` please check out our [official troubleshooting documentation](https://developers.cloudflare.com/pages/framework-guides/nextjs/deploy-a-nextjs-site/#troubleshooting). - -## More Information - -For more information on the project please check out the [README](https://github.com/cloudflare/next-on-pages/blob/main/README.md) in the next-on-pages github repository. +- `npx @cloudflare/next-on-pages` — a CLI that runs `next build`, then transforms its output to run on Cloudflare Pages. +- [`setupDevPlatform`](https://github.com/cloudflare/next-on-pages/tree/main/internal-packages/next-dev) — lets you access [bindings](https://developers.cloudflare.com/workers/runtime-apis/bindings/) in local development of your Next.js app. diff --git a/packages/next-on-pages/docs/advanced-usage.md b/packages/next-on-pages/docs/advanced-usage.md deleted file mode 100644 index 35ad6ac20..000000000 --- a/packages/next-on-pages/docs/advanced-usage.md +++ /dev/null @@ -1,34 +0,0 @@ -# Advanced Usage - -## Custom Worker Entrypoint - -Certain use cases may require the ability the control what happens in your Pages project's worker. Observability requirements, for instance, might benefit from being able to intercept console logs, catch uncaught exceptions, or monitor the time spent doing work in the next-on-pages router. - -All of these would require modifying the worker to add some code before and/or after next-on-pages' logic runs. - -To achieve this, next-on-pages exposes an option to use your own worker entrypoint. Within it, you can directly import and use the next-on-pages fetch handler. - -1. Create a handler in your project. - -```ts -// file: ./custom-entrypoint.ts -import nextOnPagesHandler from '@cloudflare/next-on-pages/fetch-handler'; - -export default { - async fetch(request, env, ctx) { - // do something before running the next-on-pages handler - - const response = await nextOnPagesHandler.fetch(request, env, ctx); - - // do something after running the next-on-pages handler - - return response; - }, -} as ExportedHandler<{ ASSETS: Fetcher }>; -``` - -2. Pass the entrypoint argument to the next-on-pages CLI with the path to your handler. - -```sh -npx @cloudflare/next-on-pages --custom-entrypoint=./custom-entrypoint.ts -``` diff --git a/packages/next-on-pages/docs/caching.md b/packages/next-on-pages/docs/caching.md deleted file mode 100644 index 6b9d82ebd..000000000 --- a/packages/next-on-pages/docs/caching.md +++ /dev/null @@ -1,31 +0,0 @@ -# Caching and Data Revalidation - -`@cloudflare/next-on-pages` comes with support for data revalidation and caching for fetch requests. This is done in our router and acts as an extension to Next.js' built-in functionality. - -Caching in Next.js applications (thus applications built using `@cloudflare/next-on-pages` as well) is enabled by default. If you wish to opt-out of this caching please refer to the [official Next.js documentation](https://nextjs.org/docs/app/building-your-application/caching#opting-out-1). - -Also please note that the cache is persisted across deployments, again inline with what the [Next.js documented behavior](https://nextjs.org/docs/app/building-your-application/caching#data-cache). You are responsible for revalidating/purging this cache. It is not handled for you by `@cloudflare/next-on-pages` or Cloudflare Pages. - -## Storage Options - -There are currently two different storage options that `@cloudflare/next-on-pages` supports, the Cache API and Workers KV. - -In the future, support will be available for creating custom cache interfaces and using different bindings. - -### Cache API - -The [Cache API](https://developers.cloudflare.com/workers/runtime-apis/cache/) is a per data-center cache that is ideal for storing data that is not required to be accessible globally. - -It is worth noting that Vercel's Data Cache is regional (and not global) the same applies to Workers Cache API, so with this storage option there is no difference in terms of data availability. This includes cache related operations such as the [Next.js' on-demand revalidation](https://nextjs.org/docs/app/building-your-application/data-fetching/fetching-caching-and-revalidating#on-demand-revalidation). - -### Workers KV - -[Workers KV](https://developers.cloudflare.com/kv/) is a low-latency key-value store that is ideal for storing data that should be globally distributed. KV is eventually consistent, which means that it will take up to 60 seconds for updates to be reflected globally. - -It is important to note that, contrary to the Vercel Data Cache and Workers Cache API, the KV storage is global (and not regional), this naturally changes the data availability and the effect of operations such as the on-demand revalidation. - -To use Workers KV for caching all you need to do is to add a KV binding (as documented in the [Cloudflare Pages documentation](https://developers.cloudflare.com/pages/functions/bindings/#kv-namespaces)) to your Pages project with the name `__NEXT_ON_PAGES__KV_SUSPENSE_CACHE`. - -> [!IMPORTANT] -> The Workers KV storage solution is, compared to the Cache API, less surprising but still equally performant. It adds the extra benefits of being able to easily inspect the cache content and easily invalidate/purge it (in the case of production cache via the [Dashboard KV UI](https://dash.cloudflare.com/?to=/:account/workers/kv/namespaces)).\ -> So for the above reasons the Workers KV storage solution is the one we recommend. diff --git a/packages/next-on-pages/docs/examples.md b/packages/next-on-pages/docs/examples.md deleted file mode 100644 index 32fc125bd..000000000 --- a/packages/next-on-pages/docs/examples.md +++ /dev/null @@ -1,155 +0,0 @@ -# Examples - - - -## Edge Runtime - -To opt a route into the edge runtime in Next.js, it has to export a `runtime` [route segment config](https://nextjs.org/docs/app/api-reference/file-conventions/route-segment-config#runtime) option. - -```diff -+ export const runtime = 'edge'; -``` - -
-Prior to Next.js v13.3.1 - -When using a Next.js version that is older than v13.3.1, it is possible to export a `config` object from a route and specify a `runtime` option inside that object. This can opt the route into the edge runtime. - -```diff -export const config = { -+ runtime: 'edge', -}; -``` - -
- -### App Directory - -#### [Edge Route Handlers](https://nextjs.org/docs/app/building-your-application/routing/router-handlers#edge-and-nodejs-runtimes) - -```typescript -// ./app/api/hello/route.ts - -import { cookies } from 'next/headers'; - -export const runtime = 'edge'; - -export async function GET(request: Request) { - const cookieStore = cookies(); - const token = cookieStore.get('token'); - - return new Response('Hello, Next.js!', { - status: 200, - headers: { 'Set-Cookie': `token=${token}` }, - }); -} -``` - -#### [Server Side Rendering](https://nextjs.org/docs/app/building-your-application/rendering/static-and-dynamic-rendering#dynamic-rendering) - -```typescript -// ./app/ssr/page.tsx - -export const runtime = 'edge'; - -export default async function Page({ searchParams }: { searchParams: any }) { - return ( -
-

- Updating searchParams -

-

- searchParams is {JSON.stringify(searchParams)} -

-
- ); -} -``` - -### Pages Directory - -#### [Edge API Routes](https://nextjs.org/docs/pages/building-your-application/routing/api-routes#edge-api-routes) - -```typescript -// ./pages/api/some_route.ts - -import type { NextRequest } from 'next/server'; - -export const config = { - runtime: 'edge', -}; - -export default async function handler(req: NextRequest) { - return new Response( - JSON.stringify({ - name: process.env.NEXT_RUNTIME, - }), - { - status: 200, - headers: { - 'content-type': 'application/json', - }, - }, - ); -} -``` - -#### Server-side rendering (SSR) pages with [`getServerSideProps()`](https://nextjs.org/docs/pages/building-your-application/data-fetching/get-server-side-props) - -```typescript -// ./pages/ssr.tsx - -import styles from '../styles/Home.module.css'; - -export const config = { - runtime: 'edge', -}; - -export const getServerSideProps = async () => { - return { - props: { - runtime: process.env.NEXT_RUNTIME, - uuid: await fetch('https://uuid.rocks/plain').then(response => - response.text(), - ), - }, - }; -}; - -type Props = { runtime: string; uuid: string }; - -export default function Page({ runtime, uuid }: Props) { - return ( -
-
-

- Welcome to{' '} - Next.js, running at the {runtime}! -

- -

- Here's a server-side UUID: - {uuid} -

-
-
- ); -} -``` - -## [Middleware](https://nextjs.org/docs/pages/building-your-application/routing/middleware) - -```typescript -// ./middleware.ts -import { NextResponse } from 'next/server'; -import type { NextRequest } from 'next/server'; - -// This function can be marked `async` if using `await` inside -export function middleware(request: NextRequest) { - return NextResponse.redirect(new URL('/home', request.url)); -} - -export const config = { - matcher: '/about/:path*', -}; -``` diff --git a/packages/next-on-pages/docs/static-assets-routing.md b/packages/next-on-pages/docs/static-assets-routing.md deleted file mode 100644 index f58a4be3d..000000000 --- a/packages/next-on-pages/docs/static-assets-routing.md +++ /dev/null @@ -1,18 +0,0 @@ -# \_routes.json - -Cloudflare Pages supports defining a list of patterns that should (or should not) invoke your worker script. This is useful when using a library like `@cloudflare/next-on-pages` as it allows you to exclude certain static assets, like a favicon, from invoking the routing system on each request, saving you money and improving performance. - -To opt-out certain static assets, you can create an `_routes.json` file in your project. This file can specify which assets to include or exclude from invoking the worker script, as per the [Cloudflare docs](https://developers.cloudflare.com/pages/platform/functions/routing/#create-a-_routesjson-file). - -For example, to exclude the `/favicon.ico` asset from invoking the worker script, you can create the following `_routes.json` file in the root directory of your project: - -```json -{ - "version": 1, - "exclude": ["/favicon.ico"] -} -``` - -During the build process, `@cloudflare/next-on-pages` will automatically generate an `_routes.json` file in the output directory. Any entries that are provided in your own `_routes.json` file (in the project's root directory) will be merged with the generated file and take effect when deployed to Cloudflare Pages. - -The `_routes.json` file **should only be used for static assets** that do not need to go through the routing system. It **should not be used for routes** as this could lead to unexpected behavior and incorrect routing. diff --git a/packages/next-on-pages/docs/supported.md b/packages/next-on-pages/docs/supported.md deleted file mode 100644 index fd1f7ed72..000000000 --- a/packages/next-on-pages/docs/supported.md +++ /dev/null @@ -1,177 +0,0 @@ -# Supported Versions and Features - -## Operating Systems - -`@cloudflare/next-on-pages` can be run on Linux, Mac OS and Windows but its usage under the latter is discouraged as we've noticed that one of the CLI's dependencies, the [Vercel CLI](https://vercel.com/docs/cli) (used to build the Next.js application) seems not to work reliably on Windows. If you need to run `@cloudflare/next-on-pages` on Windows we advise you to run it under the [Windows Subsystem for Linux](https://learn.microsoft.com/en-us/windows/wsl/). - -## Supported Next.js versions - -`@cloudflare/next-on-pages` supports all minor and patch version of Next.js 13 and 14. We regularly run manual and automated tests to ensure such compatibility. - -Next.js canary versions not actively being tested and we don't currently extend support to these versions. - -### Node.js - -Next.js offers two different [runtimes](https://nextjs.org/docs/app/building-your-application/rendering/edge-and-nodejs-runtimes#nodejs-runtime) for your application's routes: `Node.js` and `Edge`. - -Routes using the `Node.js` runtime get built as Node.js serverless functions, such are fundamentally incompatible with the Cloudflare network and so only routes using the `Edge` runtime are supported when using `@cloudflare/next-on-pages`. - -The [Cloudflare workers runtime supports certain Node.js APIs](https://developers.cloudflare.com/workers/platform/nodejs-compatibility/), but currently Next.js in its `Edge` runtime only supports a [subset](https://github.com/vercel/next.js/blob/06f505c78bcc25ad4756f0e1665d239c0f45a3e2/packages/next/src/build/webpack/plugins/middleware-plugin.ts#L788-L794) of them, resulting in only the following Node.js built-in modules being currently supported in `@cloudflare/next-on-pages`: - -- `buffer` -- `events` -- `assert` -- `util` -- `async_hooks` - -## External Packages - -You are free to use any external npm package with your Next.js application as long as it doesn't use: - -- unsupported Node.js APIs (see above) -- [JavaScript APIs](https://developers.cloudflare.com/workers/runtime-apis/web-standards/#javascript-standards) disabled by Cloudflare due to security concerns -- features not supported by `@cloudflare/next-on-pages` (see below) - -## Supported Features - -### Routers - -Both the [Pages](https://nextjs.org/docs/pages) and [App](https://nextjs.org/docs/app) routers are supported, however we recommend using the App router as there are a few unsupported features in the Pages one (including [custom error pages](https://github.com/cloudflare/next-on-pages/issues/174)), and since the latter is not the recommended one by the Next.js team, there is a good chance that it won't be improved/worked on to such a degree to allows us to increase our support of it. - -The App router should be fully supported, but unfortunately a few of its features aren't currently supported (because of Vercel implementation details that are outside of our control) like for example [custom not-found pages with server side runtime logic](https://github.com/cloudflare/next-on-pages/issues/413). - -To check the latest state of the routers and possible missing features you can check our GitHub [app router](https://github.com/cloudflare/next-on-pages/issues?q=is%3Aopen+is%3Aissue+label%3A%22app+router%22) and [pages router](https://github.com/cloudflare/next-on-pages/issues?q=is%3Aopen+is%3Aissue+label%3A%22pages+router%22) issues. - -### Build Output Configuration - -> The following options have been gathered from the [Vercel Build Output API (v3) documentation](https://vercel.com/docs/build-output-api/v3#build-output-configuration): - -| `config.json` property | Support | -| ----------------------- | ------- | -| version | `3` | -| routes `src` | ✅ | -| routes `dest` | ✅ | -| routes `headers` | ✅ | -| routes `methods` | ✅ | -| routes `continue` | ✅ | -| routes `caseSensitive` | ✅ | -| routes `check` | ✅ | -| routes `status` | ✅ | -| routes `has` | ✅ | -| routes `missing` | ✅ | -| routes `locale` | ✅ | -| routes `middlewarePath` | ✅ | -| images1 | ✅ | -| wildcard | ✅ | -| overrides | ✅ | -| cache | ❌ | -| crons | ❌ | - - - ✅: Supported - - 🔄: Not currently supported, but it's probably possible and we may add support in the future - - ❌: Not supported and unlikely to be supported in the future - -- _1_ - **images**: If you want to use `next/image`, there are two options; allow the library to take care of incoming requests, or using a custom loader. Requests are intercepted in the router and image resizing is attempted to be used (due to limitations with Pages, it is not currently possible to use image resizing) - if image resizing is not available, it falls back to fetching the normal image URL. Alternatively, you can provide your own [custom loader](https://nextjs.org/docs/api-reference/next/image#loader) and use Cloudflare Image Resizing, as per [Cloudflare's Image Resizing documentation](https://developers.cloudflare.com/images/image-resizing/integration-with-frameworks/#nextjs). - -### next.config.mjs Properties - -> The following options have been gathered from the Next.js' next.config.js [app](https://nextjs.org/docs/app/api-reference/next-config-js) and [pages](https://nextjs.org/docs/pages/api-reference/next-config-js) documentations. - -| Option | Next Docs | Support | -| ----------------------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------- | -| appDir | [app](https://nextjs.org/docs/app/api-reference/next-config-js/appDir) | ✅ | -| assetPrefix | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/assetPrefix), [app](https://nextjs.org/docs/app/api-reference/next-config-js/assetPrefix) | 🔄 | -| basePath | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/basePath), [app](https://nextjs.org/docs/app/api-reference/next-config-js/basePath) | ✅ | -| compress | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/compress), [app](https://nextjs.org/docs/app/api-reference/next-config-js/compress) | `N/A`1 | -| devIndicators | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/devIndicators), [app](https://nextjs.org/docs/app/api-reference/next-config-js/devIndicators) | `N/A`2 | -| distDir | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/distDir), [app](https://nextjs.org/docs/app/api-reference/next-config-js/distDir) | `N/A`3 | -| env | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/env), [app](https://nextjs.org/docs/app/api-reference/next-config-js/env) | ✅ | -| eslint | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/eslint), [app](https://nextjs.org/docs/app/api-reference/next-config-js/eslint) | ✅ | -| exportPathMap | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/exportPathMap), [app](https://nextjs.org/docs/app/api-reference/next-config-js/exportPathMap) | `N/A`4 | -| generateBuildId | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/generateBuildId), [app](https://nextjs.org/docs/app/api-reference/next-config-js/generateBuildId) | ✅ | -| generateEtags | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/generateEtags), [app](https://nextjs.org/docs/app/api-reference/next-config-js/generateEtags) | 🔄 | -| headers | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/headers), [app](https://nextjs.org/docs/app/api-reference/next-config-js/headers) | ✅ | -| httpAgentOptions | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/httpAgentOptions), [app](https://nextjs.org/docs/app/api-reference/next-config-js/httpAgentOptions) | `N/A` | -| images | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/images), [app](https://nextjs.org/docs/app/api-reference/next-config-js/images) | ✅ | -| incrementalCacheHandlerPath | [app](https://nextjs.org/docs/app/api-reference/next-config-js/incrementalCacheHandlerPath) | 🔄 | -| logging | [app](https://nextjs.org/docs/app/api-reference/next-config-js/logging) | `N/A`5 | -| mdxRs | [app](https://nextjs.org/docs/app/api-reference/next-config-js/mdxRs) | ✅ | -| onDemandEntries | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/onDemandEntries), [app](https://nextjs.org/docs/app/api-reference/next-config-js/onDemandEntries) | `N/A`6 | -| optimizePackageImports | [app](https://nextjs.org/docs/app/api-reference/next-config-js/optimizePackageImports) | ✅/`N/A`7 | -| output | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/output), [app](https://nextjs.org/docs/app/api-reference/next-config-js/output) | `N/A`8 | -| pageExtensions | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/pageExtensions), [app](https://nextjs.org/docs/app/api-reference/next-config-js/pageExtensions) | ✅ | -| Partial Prerendering (experimental) | [app](https://nextjs.org/docs/app/api-reference/next-config-js/partial-prerendering) | ❌9 | -| poweredByHeader | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/poweredByHeader), [app](https://nextjs.org/docs/app/api-reference/next-config-js/poweredByHeader) | 🔄 | -| productionBrowserSourceMaps | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/productionBrowserSourceMaps), [app](https://nextjs.org/docs/app/api-reference/next-config-js/productionBrowserSourceMaps) | 🔄10 | -| reactStrictMode | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/reactStrictMode), [app](https://nextjs.org/docs/app/api-reference/next-config-js/reactStrictMode) | ❌11 | -| redirects | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/redirects), [app](https://nextjs.org/docs/app/api-reference/next-config-js/redirects) | ✅ | -| rewrites | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/rewrites), [app](https://nextjs.org/docs/app/api-reference/next-config-js/rewrites) | ✅ | -| Runtime Config | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/runtime-configuration), [app](https://nextjs.org/docs/app/api-reference/next-config-js/runtime-configuration) | ❌12 | -| serverActions | [app](https://nextjs.org/docs/app/api-reference/next-config-js/serverActions) | ✅ | -| serverComponentsExternalPackages | [app](https://nextjs.org/docs/app/api-reference/next-config-js/serverComponentsExternalPackages) | `N/A`13 | -| trailingSlash | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/trailingSlash), [app](https://nextjs.org/docs/app/api-reference/next-config-js/trailingSlash) | ✅ | -| transpilePackages | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/transpilePackages), [app](https://nextjs.org/docs/app/api-reference/next-config-js/transpilePackages) | ✅ | -| turbo | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/turbo), [app](https://nextjs.org/docs/app/api-reference/next-config-js/turbo) | 🔄 | -| typedRoutes | [app](https://nextjs.org/docs/app/api-reference/next-config-js/typedRoutes) | ✅ | -| typescript | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/typescript), [app](https://nextjs.org/docs/app/api-reference/next-config-js/typescript) | ✅ | -| urlImports | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/urlImports), [app](https://nextjs.org/docs/app/api-reference/next-config-js/urlImports) | ✅ | -| webpack | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/webpack), [app](https://nextjs.org/docs/app/api-reference/next-config-js/webpack) | ✅ | -| webVitalsAttribution | [pages](https://nextjs.org/docs/pages/api-reference/next-config-js/webVitalsAttribution), [app](https://nextjs.org/docs/app/api-reference/next-config-js/webVitalsAttribution) | ✅ | - - - ✅: Supported - - 🔄: Not currently supported, but it's probably possible and we may add support in the future - - ❌: Not supported and unlikely we ever will support this - - N/A: Not applicable - -- _1_ - **compression**: [Cloudflare applies gzip or brotli compression](https://developers.cloudflare.com/support/speed/optimization-file-size/what-will-cloudflare-compress) automatically. When developing locally with Wrangler, no compression is applied. - -- _2_ - **dev indicators**: If you're developing using `wrangler pages dev`, it hard refreshes your application the dev indicator doesn't appear. If you run your app locally using `next dev`, this option works fine. - -- _3_ - **setting custom build directory**: Applications built using `@cloudflare/next-on-pages` don't rely on the `.next` directory so this option isn't really applicable (the `@cloudflare/next-on-pages` equivalent is to use the `--outdir` flag). - -- _4_ - **exportPathMap**: Option used for SSG not applicable for apps built using `@cloudflare/next-on-pages`. - -- _5_ - **logging**: If you're developing using `wrangler pages dev`, the extra logging is not applied (since you are effectively running a production build). If you run your app locally using `next dev`, this option works fine. - -- _6_ - **onDemandEntries**: Not applicable since it's an option for the Next.js server during development which we don't rely on. - -- _7_ - **optimizePackageImports**: `@cloudflare/next-on-pages` performs chunks deduplication and provides an implementation based on modules lazy loading, based on this applying an `optimizePackageImports` doesn't have an impact on the output produced by the CLI. This configuration can still however be used to speed up the build process (both when running `next dev` or when generating a production build). - -- _8_ - **output**: `@cloudflare/next-on-pages` works with the standard Next.js output, `standalone` is incompatible with it, `export` is used to generate a static site which doesn't need `@cloudflare/next-on-pages` to run. - -- _9_ - **Partial Prerendering (experimental)**: As presented in the official [Next.js documentation](https://nextjs.org/docs/app/api-reference/next-config-js/partial-prerendering): `Partial Prerendering is designed for the Node.js runtime only.`, as such it is fundamentally incompatibly with `@cloudflare/next-on-pages` (which only works on the edge runtime). - -- _10_ - **productionBrowserSourceMaps**: The webpack chunks deduplication performed by `@cloudflare/next-on-pages` doesn't currently preserve source maps in any case so this option can't be implemented either. In the future we might try to preserver source maps, in such case it should be simple to also support this option. - -- _11_ - **reactStrictMode**: Currently we build the application so react strict mode (being a local dev feature) doesn't work either way. If we can make strict mode work, this option will most likely work straight away. - -- _12_ - **runtime configuration**: We could look into implementing the runtime configuration but it is probably not worth it since it is a legacy configuration and environment variables should be used instead. - -- _13_ - **serverComponentsExternalPackages**: This option is for applications running on Node.js so it's not relevant to applications running on Cloudflare Pages. - -### Internationalization - -Besides the above mentioned `next.config.js` properties, there is also the `i18n` one, that is also fully supported meaning that `@cloudflare/next-on-pages` does support Next.js' built-in internationalization system. For more details on the option see the [Next.js Internationalization documentation](https://nextjs.org/docs/pages/building-your-application/routing/internationalization). - -### Rendering and Data Fetching - -#### Incremental Static Regeneration - -Incremental Static Regeneration (ISR) is a rendering mode in Next.js that allows you to automatically cache and periodically regenerate pages with fresh data. Next.js [does not support](https://nextjs.org/docs/pages/building-your-application/rendering/incremental-static-regeneration) building ISR pages for the edge runtime, and as such, pages should be changed to use server side rendering (SSR) instead. - -ISR pages are built by the Vercel CLI to generate Vercel [Prerender Functions](https://vercel.com/docs/build-output-api/v3/primitives#prerender-functions). These are Node.js serverless functions that can be called in the background while serving the page from the cache. It is not possible to use these with Cloudflare Pages and they are not compatible with the [edge runtime](https://nextjs.org/docs/app/api-reference/edge) currently. - -In case the Vercel build process generates prerendered pages for your application, `@cloudflare/next-on-pages` will use static fallback files that are generated by the build process so that your application will still correctly serve your ISR/prerendered pages (but without the regeneration aspect). - -#### Dynamic handling of static routes - -`@cloudflare/next-on-pages` supports standard statically generated routes, it does however not support dynamic Node.js-based on-demand handling of such routes. - -For more details see: - -- [troubleshooting `generateStaticParams`](https://developers.cloudflare.com/pages/framework-guides/nextjs/deploy-a-nextjs-site/#generatestaticparams) -- [troubleshooting `getStaticPaths` ](https://developers.cloudflare.com/pages/framework-guides/nextjs/deploy-a-nextjs-site/#getstaticpaths) - -#### Caching and Data Revalidation - -Revalidation and `next/cache` are supported on Cloudflare Pages, and can use various bindings. For more information, see our [caching documentation](./caching.md).