Caching
@opennextjs/cloudflare
supports caching (opens in a new tab).
How to enable caching
@opennextjs/cloudflare
supports multiple caching mechanisms through a project's OpenNext configuration.
Incremental Static Regeneration (ISR)
There are two storage options to use for the incremental cache.
- Workers KV: A fast (opens in a new tab) and uses Cloudflare's Tiered Cache (opens in a new tab) to increase cache hit rates. When you write cached data to Workers KV, you write to storage that can be read by any Cloudflare location. This means your app can fetch data, cache it in KV, and then subsequent requests anywhere around the world can read from this cache. The build time values are serverd by the Workers Assets (opens in a new tab). Pricing information can be found in the Cloudflare docs (opens in a new tab).
- R2 Object Storage: A cost-effective (opens in a new tab) S3-compatible object storage option for large amounts of unstructured data. Data is stored in a single region, meaning cache interactions may be slower - this can be mitigated with a regional cache.
Workers KV is eventually consistent, which means that it can take up to 60 seconds for updates to be reflected globally, when using the default TTL of 60 seconds.
1. Create a KV namespace
npx wrangler@latest kv namespace create <YOUR_NAMESPACE_NAME>
2. Add the KV namespace and Service Binding to your Worker
The binding name used in your app's worker is NEXT_INC_CACHE_KV
.
The WORKER_SELF_REFERENCE
service binding should be a self reference to your worker where <WORKER_NAME>
is the name in your wrangler configuration file.
// wrangler.jsonc
{
// ...
"name": "<WORKER_NAME>",
"kv_namespaces": [
{
"binding": "NEXT_INC_CACHE_KV",
"id": "<BINDING_ID>",
},
],
"services": [
{
"binding": "WORKER_SELF_REFERENCE",
"service": "<WORKER_NAME>",
},
],
}
3. Configure the cache
In your project's OpenNext config, enable the KV cache.
// open-next.config.ts
import { defineCloudflareConfig } from "@opennextjs/cloudflare";
import kvIncrementalCache from "@opennextjs/cloudflare/overrides/incremental-cache/kv-incremental-cache";
// ...
export default defineCloudflareConfig({
incrementalCache: kvIncrementalCache,
// ...
});
4. Configure the queue
In your project's OpenNext config, enable the cache and set up a queue.
The Durable Object Queue will send revalidation requests to a page when needed, and offers support for de-duplicating requests. By default there will be a maximum of 10 instance of the Durables Object Queue and they can each process up to 5 requests in parallel, for up to 50 concurrent ISR revalidations.
// open-next.config.ts
import { defineCloudflareConfig } from "@opennextjs/cloudflare";
// ...
import doQueue from "@opennextjs/cloudflare/overrides/queue/do-queue";
export default defineCloudflareConfig({
// ...
queue: doQueue,
});
You will also need to add some binding to your wrangler.jsonc
file.
"durable_objects": {
"bindings": [
{
"name": "NEXT_CACHE_DO_QUEUE",
"class_name": "DurableObjectQueueHandler"
}
]
},
"migrations": [
{
"tag": "v1",
"new_sqlite_classes": ["DurableObjectQueueHandler"]
}
],
You can customize the behaviors of the queue with environment variables:
- The max number of revalidations that can be processed by an instance of durable object at the same time (
NEXT_CACHE_DO_QUEUE_MAX_REVALIDATION
) - The max time in milliseconds that a revalidation can take before being considered as failed (
NEXT_CACHE_DO_QUEUE_REVALIDATION_TIMEOUT_MS
) - The amount of time after which a revalidation will be attempted again if it failed. If it fails again it will exponentially back off until it reaches the max retry interval (
NEXT_CACHE_DO_QUEUE_RETRY_INTERVAL_MS
) - The maximum number of attempts that can be made to revalidate a path (
NEXT_CACHE_DO_QUEUE_MAX_NUM_REVALIDATIONS
) - Disable SQLite for this durable object. It should only be used if your incremental cache is not eventually consistent (
NEXT_CACHE_DO_QUEUE_DISABLE_SQLITE
)
There is 2 additional modes that you can use for the queue direct
and the memory queue
-
The memory queue will dedupe request but only on a per isolate basis. It is not fully suitable for production deployments, you can use it at your own risk!
-
The
direct
mode for the queue is intended for debugging purposes and is not recommended for use in production. It only works in preview mode (i.e.wrangler dev
)For apps using the Page Router,
res.revalidate
requires to provide a self reference service binding namedWORKER_SELF_REFERENCE
.
On-Demand Revalidation
The tag revalidation mechanism can use either a Cloudflare D1 (opens in a new tab) database or Durable Objects (opens in a new tab) with SqliteStorage
as its backing store for information about tags, paths, and revalidation times.
To use on-demand revalidation, you should also follow the ISR setup steps.
If your app only uses the pages router, it does not need to have a tag cache and should skip this step.
You can also skip this step if your app doesn't to use revalidateTag
nor revalidatePath
.
There are 3 different options to choose from for the tag cache: d1NextTagCache
, doShardedTagCache
and d1TagCache
.
Which one to choose should be based on two key factors:
- Expected Load: Consider the volume of traffic or data you anticipate.
- Usage of
revalidateTag
/revalidatePath
: Evaluate how frequently these features will be utilized.
If either of these factors is significant, opting for a sharded database is recommended. Additionally, incorporating a regional cache can further enhance performance.
1. Create a D1 database and Service Binding
The binding name used in your app's worker is NEXT_TAG_CACHE_D1
. The WORKER_SELF_REFERENCE
service binding should be a self reference to your worker where <WORKER_NAME>
is the name in your wrangler configuration file.
// wrangler.jsonc
{
// ...
"d1_databases": [
{
"binding": "NEXT_TAG_CACHE_D1",
"database_id": "<DATABASE_ID>",
"database_name": "<DATABASE_NAME>",
},
],
"services": [
{
"binding": "WORKER_SELF_REFERENCE",
"service": "<WORKER_NAME>",
},
],
}
2. Create table for tag revalidations
The D1 tag cache requires a revalidations
table that tracks On-Demand revalidation times.
3. Configure the cache
In your project's OpenNext config, enable the KV cache and set up a queue (see above). The queue will send a revalidation request to a page when needed, but it will not dedupe requests.
// open-next.config.ts
import { defineCloudflareConfig } from "@opennextjs/cloudflare";
import kvIncrementalCache from "@opennextjs/cloudflare/kv-cache";
import d1NextTagCache from "@opennextjs/cloudflare/d1-next-tag-cache";
import memoryQueue from "@opennextjs/cloudflare/memory-queue";
export default defineCloudflareConfig({
incrementalCache: kvIncrementalCache,
tagCache: d1NextTagCache,
queue: memoryQueue,
});