Skip to content

Commit bac472e

Browse files
committed
fix: apply caching headers to pages router 404 with getStaticProps
1 parent f4b59b6 commit bac472e

File tree

5 files changed

+61
-12
lines changed

5 files changed

+61
-12
lines changed

src/build/content/prerendered.ts

+4
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,7 @@ const routeToFilePath = (path: string) => {
5757

5858
const buildPagesCacheValue = async (
5959
path: string,
60+
initialRevalidateSeconds: number | false,
6061
shouldUseEnumKind: boolean,
6162
shouldSkipJson = false,
6263
): Promise<NetlifyCachedPageValue> => ({
@@ -65,6 +66,7 @@ const buildPagesCacheValue = async (
6566
pageData: shouldSkipJson ? {} : JSON.parse(await readFile(`${path}.json`, 'utf-8')),
6667
headers: undefined,
6768
status: undefined,
69+
revalidate: initialRevalidateSeconds,
6870
})
6971

7072
const buildAppCacheValue = async (
@@ -178,6 +180,7 @@ export const copyPrerenderedContent = async (ctx: PluginContext): Promise<void>
178180
}
179181
value = await buildPagesCacheValue(
180182
join(ctx.publishDir, 'server/pages', key),
183+
meta.initialRevalidateSeconds,
181184
shouldUseEnumKind,
182185
)
183186
break
@@ -210,6 +213,7 @@ export const copyPrerenderedContent = async (ctx: PluginContext): Promise<void>
210213
const key = routeToFilePath(route)
211214
const value = await buildPagesCacheValue(
212215
join(ctx.publishDir, 'server/pages', key),
216+
false,
213217
shouldUseEnumKind,
214218
true, // there is no corresponding json file for fallback, so we are skipping it for this entry
215219
)

src/run/handlers/cache.cts

+10
Original file line numberDiff line numberDiff line change
@@ -260,6 +260,11 @@ export class NetlifyCacheHandler implements CacheHandlerForMultipleVersions {
260260

261261
const { revalidate, ...restOfPageValue } = blob.value
262262

263+
const requestContext = getRequestContext()
264+
if (requestContext) {
265+
requestContext.pageHandlerRevalidate = revalidate
266+
}
267+
263268
await this.injectEntryToPrerenderManifest(key, revalidate)
264269

265270
return {
@@ -272,6 +277,11 @@ export class NetlifyCacheHandler implements CacheHandlerForMultipleVersions {
272277

273278
const { revalidate, rscData, ...restOfPageValue } = blob.value
274279

280+
const requestContext = getRequestContext()
281+
if (requestContext) {
282+
requestContext.pageHandlerRevalidate = revalidate
283+
}
284+
275285
await this.injectEntryToPrerenderManifest(key, revalidate)
276286

277287
return {

src/run/handlers/request-context.cts

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ export type RequestContext = {
2121
didPagesRouterOnDemandRevalidate?: boolean
2222
serverTiming?: string
2323
routeHandlerRevalidate?: NetlifyCachedRouteValue['revalidate']
24+
pageHandlerRevalidate?: NetlifyCachedRouteValue['revalidate']
2425
/**
2526
* Track promise running in the background and need to be waited for.
2627
* Uses `context.waitUntil` if available, otherwise stores promises to

src/run/headers.ts

+34-12
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import type { Span } from '@opentelemetry/api'
22
import type { NextConfigComplete } from 'next/dist/server/config-shared.js'
33

44
import { encodeBlobKey } from '../shared/blobkey.js'
5+
import type { NetlifyCachedRouteValue } from '../shared/cache-types.cjs'
56

67
import { getLogger, RequestContext } from './handlers/request-context.cjs'
78
import type { RuntimeTracer } from './handlers/tracer.cjs'
@@ -208,6 +209,19 @@ export const adjustDateHeader = async ({
208209
headers.set('date', lastModifiedDate.toUTCString())
209210
}
210211

212+
function setCacheControlFromRequestContext(
213+
headers: Headers,
214+
revalidate: NetlifyCachedRouteValue['revalidate'],
215+
) {
216+
const cdnCacheControl =
217+
// if we are serving already stale response, instruct edge to not attempt to cache that response
218+
headers.get('x-nextjs-cache') === 'STALE'
219+
? 'public, max-age=0, must-revalidate, durable'
220+
: `s-maxage=${revalidate === false ? 31536000 : revalidate}, stale-while-revalidate=31536000, durable`
221+
222+
headers.set('netlify-cdn-cache-control', cdnCacheControl)
223+
}
224+
211225
/**
212226
* Ensure stale-while-revalidate and s-maxage don't leak to the client, but
213227
* assume the user knows what they are doing if CDN cache controls are set
@@ -225,13 +239,7 @@ export const setCacheControlHeaders = (
225239
!headers.has('netlify-cdn-cache-control')
226240
) {
227241
// handle CDN Cache Control on Route Handler responses
228-
const cdnCacheControl =
229-
// if we are serving already stale response, instruct edge to not attempt to cache that response
230-
headers.get('x-nextjs-cache') === 'STALE'
231-
? 'public, max-age=0, must-revalidate, durable'
232-
: `s-maxage=${requestContext.routeHandlerRevalidate === false ? 31536000 : requestContext.routeHandlerRevalidate}, stale-while-revalidate=31536000, durable`
233-
234-
headers.set('netlify-cdn-cache-control', cdnCacheControl)
242+
setCacheControlFromRequestContext(headers, requestContext.routeHandlerRevalidate)
235243
return
236244
}
237245

@@ -242,11 +250,25 @@ export const setCacheControlHeaders = (
242250
.log('NetlifyHeadersHandler.trailingSlashRedirect')
243251
}
244252

245-
if (status === 404 && request.url.endsWith('.php')) {
246-
// temporary CDN Cache Control handling for bot probes on PHP files
247-
// https://linear.app/netlify/issue/FRB-1344/prevent-excessive-ssr-invocations-due-to-404-routes
248-
headers.set('cache-control', 'public, max-age=0, must-revalidate')
249-
headers.set('netlify-cdn-cache-control', `max-age=31536000, durable`)
253+
if (status === 404) {
254+
if (request.url.endsWith('.php')) {
255+
// temporary CDN Cache Control handling for bot probes on PHP files
256+
// https://linear.app/netlify/issue/FRB-1344/prevent-excessive-ssr-invocations-due-to-404-routes
257+
headers.set('cache-control', 'public, max-age=0, must-revalidate')
258+
headers.set('netlify-cdn-cache-control', `max-age=31536000, durable`)
259+
return
260+
}
261+
262+
if (
263+
typeof requestContext.pageHandlerRevalidate !== 'undefined' &&
264+
['GET', 'HEAD'].includes(request.method) &&
265+
!headers.has('cdn-cache-control') &&
266+
!headers.has('netlify-cdn-cache-control')
267+
) {
268+
// handle CDN Cache Control on 404 Page responses
269+
setCacheControlFromRequestContext(headers, requestContext.pageHandlerRevalidate)
270+
return
271+
}
250272
}
251273

252274
const cacheControl = headers.get('cache-control')

tests/integration/page-router.test.ts

+12
Original file line numberDiff line numberDiff line change
@@ -126,6 +126,10 @@ test<FixtureTestContext>('Should serve correct locale-aware custom 404 pages', a
126126
load(responseImplicitDefaultLocale.body)('[data-testid="locale"]').text(),
127127
'Served 404 page content should use default locale if locale is not explicitly used in pathname (after basePath)',
128128
).toBe('en')
129+
expect(
130+
responseImplicitDefaultLocale.headers['netlify-cdn-cache-control'],
131+
'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status',
132+
).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable')
129133

130134
const responseExplicitDefaultLocale = await invokeFunction(ctx, {
131135
url: '/base/path/en/not-existing-page',
@@ -139,6 +143,10 @@ test<FixtureTestContext>('Should serve correct locale-aware custom 404 pages', a
139143
load(responseExplicitDefaultLocale.body)('[data-testid="locale"]').text(),
140144
'Served 404 page content should use default locale if default locale is explicitly used in pathname (after basePath)',
141145
).toBe('en')
146+
expect(
147+
responseExplicitDefaultLocale.headers['netlify-cdn-cache-control'],
148+
'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status',
149+
).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable')
142150

143151
const responseNonDefaultLocale = await invokeFunction(ctx, {
144152
url: '/base/path/fr/not-existing-page',
@@ -152,4 +160,8 @@ test<FixtureTestContext>('Should serve correct locale-aware custom 404 pages', a
152160
load(responseNonDefaultLocale.body)('[data-testid="locale"]').text(),
153161
'Served 404 page content should use non-default locale if non-default locale is explicitly used in pathname (after basePath)',
154162
).toBe('fr')
163+
expect(
164+
responseNonDefaultLocale.headers['netlify-cdn-cache-control'],
165+
'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status',
166+
).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable')
155167
})

0 commit comments

Comments
 (0)