From 330dca0ceee048d05dbed4958e3c33e1008c3ae0 Mon Sep 17 00:00:00 2001 From: Michal Piechowiak Date: Thu, 13 Feb 2025 20:59:45 +0100 Subject: [PATCH 01/15] fix: apply caching headers to pages router 404 with getStaticProps --- src/build/content/prerendered.ts | 4 +++ src/run/handlers/cache.cts | 10 ++++++ src/run/handlers/request-context.cts | 1 + src/run/headers.ts | 46 ++++++++++++++++++++------- tests/integration/page-router.test.ts | 12 +++++++ 5 files changed, 61 insertions(+), 12 deletions(-) diff --git a/src/build/content/prerendered.ts b/src/build/content/prerendered.ts index 3510aefe12..6511b16c13 100644 --- a/src/build/content/prerendered.ts +++ b/src/build/content/prerendered.ts @@ -57,6 +57,7 @@ const routeToFilePath = (path: string) => { const buildPagesCacheValue = async ( path: string, + initialRevalidateSeconds: number | false | undefined, shouldUseEnumKind: boolean, shouldSkipJson = false, ): Promise => ({ @@ -65,6 +66,7 @@ const buildPagesCacheValue = async ( pageData: shouldSkipJson ? {} : JSON.parse(await readFile(`${path}.json`, 'utf-8')), headers: undefined, status: undefined, + revalidate: initialRevalidateSeconds, }) const buildAppCacheValue = async ( @@ -178,6 +180,7 @@ export const copyPrerenderedContent = async (ctx: PluginContext): Promise } value = await buildPagesCacheValue( join(ctx.publishDir, 'server/pages', key), + meta.initialRevalidateSeconds, shouldUseEnumKind, ) break @@ -210,6 +213,7 @@ export const copyPrerenderedContent = async (ctx: PluginContext): Promise const key = routeToFilePath(route) const value = await buildPagesCacheValue( join(ctx.publishDir, 'server/pages', key), + undefined, shouldUseEnumKind, true, // there is no corresponding json file for fallback, so we are skipping it for this entry ) diff --git a/src/run/handlers/cache.cts b/src/run/handlers/cache.cts index c00a4d5e14..ba6f79794c 100644 --- a/src/run/handlers/cache.cts +++ b/src/run/handlers/cache.cts @@ -260,6 +260,11 @@ export class NetlifyCacheHandler implements CacheHandlerForMultipleVersions { const { revalidate, ...restOfPageValue } = blob.value + const requestContext = getRequestContext() + if (requestContext) { + requestContext.pageHandlerRevalidate = revalidate + } + await this.injectEntryToPrerenderManifest(key, revalidate) return { @@ -272,6 +277,11 @@ export class NetlifyCacheHandler implements CacheHandlerForMultipleVersions { const { revalidate, rscData, ...restOfPageValue } = blob.value + const requestContext = getRequestContext() + if (requestContext) { + requestContext.pageHandlerRevalidate = revalidate + } + await this.injectEntryToPrerenderManifest(key, revalidate) return { diff --git a/src/run/handlers/request-context.cts b/src/run/handlers/request-context.cts index cc67739242..af4083e1ca 100644 --- a/src/run/handlers/request-context.cts +++ b/src/run/handlers/request-context.cts @@ -21,6 +21,7 @@ export type RequestContext = { didPagesRouterOnDemandRevalidate?: boolean serverTiming?: string routeHandlerRevalidate?: NetlifyCachedRouteValue['revalidate'] + pageHandlerRevalidate?: NetlifyCachedRouteValue['revalidate'] /** * Track promise running in the background and need to be waited for. * Uses `context.waitUntil` if available, otherwise stores promises to diff --git a/src/run/headers.ts b/src/run/headers.ts index c93c0a605b..d206ec4e73 100644 --- a/src/run/headers.ts +++ b/src/run/headers.ts @@ -2,6 +2,7 @@ import type { Span } from '@opentelemetry/api' import type { NextConfigComplete } from 'next/dist/server/config-shared.js' import { encodeBlobKey } from '../shared/blobkey.js' +import type { NetlifyCachedRouteValue } from '../shared/cache-types.cjs' import { getLogger, RequestContext } from './handlers/request-context.cjs' import type { RuntimeTracer } from './handlers/tracer.cjs' @@ -208,6 +209,19 @@ export const adjustDateHeader = async ({ headers.set('date', lastModifiedDate.toUTCString()) } +function setCacheControlFromRequestContext( + headers: Headers, + revalidate: NetlifyCachedRouteValue['revalidate'], +) { + const cdnCacheControl = + // if we are serving already stale response, instruct edge to not attempt to cache that response + headers.get('x-nextjs-cache') === 'STALE' + ? 'public, max-age=0, must-revalidate, durable' + : `s-maxage=${revalidate === false ? 31536000 : revalidate}, stale-while-revalidate=31536000, durable` + + headers.set('netlify-cdn-cache-control', cdnCacheControl) +} + /** * Ensure stale-while-revalidate and s-maxage don't leak to the client, but * assume the user knows what they are doing if CDN cache controls are set @@ -225,13 +239,7 @@ export const setCacheControlHeaders = ( !headers.has('netlify-cdn-cache-control') ) { // handle CDN Cache Control on Route Handler responses - const cdnCacheControl = - // if we are serving already stale response, instruct edge to not attempt to cache that response - headers.get('x-nextjs-cache') === 'STALE' - ? 'public, max-age=0, must-revalidate, durable' - : `s-maxage=${requestContext.routeHandlerRevalidate === false ? 31536000 : requestContext.routeHandlerRevalidate}, stale-while-revalidate=31536000, durable` - - headers.set('netlify-cdn-cache-control', cdnCacheControl) + setCacheControlFromRequestContext(headers, requestContext.routeHandlerRevalidate) return } @@ -242,11 +250,25 @@ export const setCacheControlHeaders = ( .log('NetlifyHeadersHandler.trailingSlashRedirect') } - if (status === 404 && request.url.endsWith('.php')) { - // temporary CDN Cache Control handling for bot probes on PHP files - // https://linear.app/netlify/issue/FRB-1344/prevent-excessive-ssr-invocations-due-to-404-routes - headers.set('cache-control', 'public, max-age=0, must-revalidate') - headers.set('netlify-cdn-cache-control', `max-age=31536000, durable`) + if (status === 404) { + if (request.url.endsWith('.php')) { + // temporary CDN Cache Control handling for bot probes on PHP files + // https://linear.app/netlify/issue/FRB-1344/prevent-excessive-ssr-invocations-due-to-404-routes + headers.set('cache-control', 'public, max-age=0, must-revalidate') + headers.set('netlify-cdn-cache-control', `max-age=31536000, durable`) + return + } + + if ( + typeof requestContext.pageHandlerRevalidate !== 'undefined' && + ['GET', 'HEAD'].includes(request.method) && + !headers.has('cdn-cache-control') && + !headers.has('netlify-cdn-cache-control') + ) { + // handle CDN Cache Control on 404 Page responses + setCacheControlFromRequestContext(headers, requestContext.pageHandlerRevalidate) + return + } } const cacheControl = headers.get('cache-control') diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index eb039eb79d..79eb960728 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -126,6 +126,10 @@ test('Should serve correct locale-aware custom 404 pages', a load(responseImplicitDefaultLocale.body)('[data-testid="locale"]').text(), 'Served 404 page content should use default locale if locale is not explicitly used in pathname (after basePath)', ).toBe('en') + expect( + responseImplicitDefaultLocale.headers['netlify-cdn-cache-control'], + 'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') const responseExplicitDefaultLocale = await invokeFunction(ctx, { url: '/base/path/en/not-existing-page', @@ -139,6 +143,10 @@ test('Should serve correct locale-aware custom 404 pages', a load(responseExplicitDefaultLocale.body)('[data-testid="locale"]').text(), 'Served 404 page content should use default locale if default locale is explicitly used in pathname (after basePath)', ).toBe('en') + expect( + responseExplicitDefaultLocale.headers['netlify-cdn-cache-control'], + 'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') const responseNonDefaultLocale = await invokeFunction(ctx, { url: '/base/path/fr/not-existing-page', @@ -152,4 +160,8 @@ test('Should serve correct locale-aware custom 404 pages', a load(responseNonDefaultLocale.body)('[data-testid="locale"]').text(), 'Served 404 page content should use non-default locale if non-default locale is explicitly used in pathname (after basePath)', ).toBe('fr') + expect( + responseNonDefaultLocale.headers['netlify-cdn-cache-control'], + 'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') }) From bdfb31a904cc36f692973a3eb400987e7a9ca715 Mon Sep 17 00:00:00 2001 From: Michal Piechowiak Date: Mon, 17 Feb 2025 11:57:18 +0100 Subject: [PATCH 02/15] chore: don't store revalidate on request context for app router pages to limit potentially unwanted impact of the change --- src/run/handlers/cache.cts | 5 ----- 1 file changed, 5 deletions(-) diff --git a/src/run/handlers/cache.cts b/src/run/handlers/cache.cts index ba6f79794c..8c2a06956d 100644 --- a/src/run/handlers/cache.cts +++ b/src/run/handlers/cache.cts @@ -277,11 +277,6 @@ export class NetlifyCacheHandler implements CacheHandlerForMultipleVersions { const { revalidate, rscData, ...restOfPageValue } = blob.value - const requestContext = getRequestContext() - if (requestContext) { - requestContext.pageHandlerRevalidate = revalidate - } - await this.injectEntryToPrerenderManifest(key, revalidate) return { From c1712f87f8c4fb587e720df5378f54ff2a5b88ea Mon Sep 17 00:00:00 2001 From: Michal Piechowiak Date: Mon, 17 Feb 2025 12:16:15 +0100 Subject: [PATCH 03/15] chore: don't apply new 404 caching header code path, if existing method would have set cacheable header --- src/run/headers.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/run/headers.ts b/src/run/headers.ts index d206ec4e73..4a1f4682ab 100644 --- a/src/run/headers.ts +++ b/src/run/headers.ts @@ -250,6 +250,7 @@ export const setCacheControlHeaders = ( .log('NetlifyHeadersHandler.trailingSlashRedirect') } + const cacheControl = headers.get('cache-control') if (status === 404) { if (request.url.endsWith('.php')) { // temporary CDN Cache Control handling for bot probes on PHP files @@ -260,6 +261,7 @@ export const setCacheControlHeaders = ( } if ( + (!cacheControl || cacheControl.includes('no-store')) && typeof requestContext.pageHandlerRevalidate !== 'undefined' && ['GET', 'HEAD'].includes(request.method) && !headers.has('cdn-cache-control') && @@ -271,7 +273,6 @@ export const setCacheControlHeaders = ( } } - const cacheControl = headers.get('cache-control') if ( cacheControl !== null && ['GET', 'HEAD'].includes(request.method) && From 7b947dc9e8d3ffb9154a0047a3bd1dc80589733f Mon Sep 17 00:00:00 2001 From: Michal Piechowiak Date: Mon, 17 Feb 2025 12:17:17 +0100 Subject: [PATCH 04/15] test: add notFound cases for 404 with getStaticProps without revalidate --- .../pages/products/[slug].js | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/tests/fixtures/page-router-base-path-i18n/pages/products/[slug].js b/tests/fixtures/page-router-base-path-i18n/pages/products/[slug].js index f41d142c67..306314ab1f 100644 --- a/tests/fixtures/page-router-base-path-i18n/pages/products/[slug].js +++ b/tests/fixtures/page-router-base-path-i18n/pages/products/[slug].js @@ -8,7 +8,19 @@ const Product = ({ time, slug }) => ( ) +/** @type {import('next').GetStaticProps} */ export async function getStaticProps({ params }) { + if (params.slug === 'not-found-no-revalidate') { + return { + notFound: true, + } + } else if (params.slug === 'not-found-with-revalidate') { + return { + notFound: true, + revalidate: 600, + } + } + return { props: { time: new Date().toISOString(), From 3423330ce1672619f1cba3e395575e52054a9521 Mon Sep 17 00:00:00 2001 From: Michal Piechowiak Date: Mon, 17 Feb 2025 12:17:42 +0100 Subject: [PATCH 05/15] test: add fixture for 404 with getStaticProps with revalidate --- .../next.config.js | 10 +++++ .../package.json | 15 +++++++ .../pages/404.js | 17 ++++++++ .../pages/products/[slug].js | 40 +++++++++++++++++++ 4 files changed, 82 insertions(+) create mode 100644 tests/fixtures/page-router-404-get-static-props-with-revalidate/next.config.js create mode 100644 tests/fixtures/page-router-404-get-static-props-with-revalidate/package.json create mode 100644 tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/404.js create mode 100644 tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/products/[slug].js diff --git a/tests/fixtures/page-router-404-get-static-props-with-revalidate/next.config.js b/tests/fixtures/page-router-404-get-static-props-with-revalidate/next.config.js new file mode 100644 index 0000000000..6346ab0742 --- /dev/null +++ b/tests/fixtures/page-router-404-get-static-props-with-revalidate/next.config.js @@ -0,0 +1,10 @@ +/** @type {import('next').NextConfig} */ +const nextConfig = { + output: 'standalone', + eslint: { + ignoreDuringBuilds: true, + }, + generateBuildId: () => 'build-id', +} + +module.exports = nextConfig diff --git a/tests/fixtures/page-router-404-get-static-props-with-revalidate/package.json b/tests/fixtures/page-router-404-get-static-props-with-revalidate/package.json new file mode 100644 index 0000000000..4506b1ddb2 --- /dev/null +++ b/tests/fixtures/page-router-404-get-static-props-with-revalidate/package.json @@ -0,0 +1,15 @@ +{ + "name": "page-router-404-get-static-props-with-revalidate", + "version": "0.1.0", + "private": true, + "scripts": { + "postinstall": "next build", + "dev": "next dev", + "build": "next build" + }, + "dependencies": { + "next": "latest", + "react": "18.2.0", + "react-dom": "18.2.0" + } +} diff --git a/tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/404.js b/tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/404.js new file mode 100644 index 0000000000..cb047699af --- /dev/null +++ b/tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/404.js @@ -0,0 +1,17 @@ +export default function NotFound({ timestamp }) { + return ( +

+ Custom 404 page with revalidate:

{timestamp}
+

+ ) +} + +/** @type {import('next').GetStaticProps} */ +export const getStaticProps = ({ locale }) => { + return { + props: { + timestamp: Date.now(), + }, + revalidate: 300, + } +} diff --git a/tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/products/[slug].js b/tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/products/[slug].js new file mode 100644 index 0000000000..714c8ac143 --- /dev/null +++ b/tests/fixtures/page-router-404-get-static-props-with-revalidate/pages/products/[slug].js @@ -0,0 +1,40 @@ +const Product = ({ time, slug }) => ( +
+

Product {slug}

+

+ This page uses getStaticProps() and getStaticPaths() to pre-fetch a Product + {time} +

+
+) + +/** @type {import('next').GetStaticProps} */ +export async function getStaticProps({ params }) { + if (params.slug === 'not-found-no-revalidate') { + return { + notFound: true, + } + } else if (params.slug === 'not-found-with-revalidate') { + return { + notFound: true, + revalidate: 600, + } + } + + return { + props: { + time: new Date().toISOString(), + slug: params.slug, + }, + } +} + +/** @type {import('next').GetStaticPaths} */ +export const getStaticPaths = () => { + return { + paths: [], + fallback: 'blocking', // false or "blocking" + } +} + +export default Product From 25ad18c12218d2134d256a4d1a3d165ece2d8b27 Mon Sep 17 00:00:00 2001 From: Michal Piechowiak Date: Mon, 17 Feb 2025 12:19:49 +0100 Subject: [PATCH 06/15] tmp: outline 404 integration test cases for page router --- tests/integration/page-router.test.ts | 147 +++++++++++++++++++++++--- 1 file changed, 134 insertions(+), 13 deletions(-) diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index 79eb960728..3c6b624862 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -4,7 +4,7 @@ import { HttpResponse, http, passthrough } from 'msw' import { setupServer } from 'msw/node' import { platform } from 'node:process' import { v4 } from 'uuid' -import { afterAll, afterEach, beforeAll, beforeEach, expect, test, vi } from 'vitest' +import { afterAll, afterEach, beforeAll, beforeEach, describe, expect, test, vi } from 'vitest' import { type FixtureTestContext } from '../utils/contexts.js' import { createFixture, invokeFunction, runPlugin } from '../utils/fixture.js' import { encodeBlobKey, generateRandomObjectID, startMockBlobStore } from '../utils/helpers.js' @@ -126,10 +126,6 @@ test('Should serve correct locale-aware custom 404 pages', a load(responseImplicitDefaultLocale.body)('[data-testid="locale"]').text(), 'Served 404 page content should use default locale if locale is not explicitly used in pathname (after basePath)', ).toBe('en') - expect( - responseImplicitDefaultLocale.headers['netlify-cdn-cache-control'], - 'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status', - ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') const responseExplicitDefaultLocale = await invokeFunction(ctx, { url: '/base/path/en/not-existing-page', @@ -143,10 +139,6 @@ test('Should serve correct locale-aware custom 404 pages', a load(responseExplicitDefaultLocale.body)('[data-testid="locale"]').text(), 'Served 404 page content should use default locale if default locale is explicitly used in pathname (after basePath)', ).toBe('en') - expect( - responseExplicitDefaultLocale.headers['netlify-cdn-cache-control'], - 'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status', - ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') const responseNonDefaultLocale = await invokeFunction(ctx, { url: '/base/path/fr/not-existing-page', @@ -160,8 +152,137 @@ test('Should serve correct locale-aware custom 404 pages', a load(responseNonDefaultLocale.body)('[data-testid="locale"]').text(), 'Served 404 page content should use non-default locale if non-default locale is explicitly used in pathname (after basePath)', ).toBe('fr') - expect( - responseNonDefaultLocale.headers['netlify-cdn-cache-control'], - 'Response for not existing route if locale is not explicitly used in pathname (after basePath) should have 404 status', - ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') +}) + +describe('404 caching', () => { + describe('default nextjs 404 (no custom 404)', () => { + test.todo('not matching dynamic paths') + test.todo('matching dynamic path without revalidate') + test.todo('matching dynamic path with revalidate') + }) + + describe('404 without getStaticProps', () => { + test.todo('not matching dynamic paths') + test.todo('matching dynamic path without revalidate') + test.todo('matching dynamic path with revalidate') + }) + + describe('404 with getStaticProps without revalidate', () => { + test('not matching dynamic paths should be cached permanently', async (ctx) => { + console.log('[test] not matching dynamic paths') + + await createFixture('page-router-base-path-i18n', ctx) + await runPlugin(ctx) + + const notExistingPage = await invokeFunction(ctx, { + url: '/base/path/not-existing-page', + }) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached permanently', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') + }) + test('matching dynamic path without revalidate should be cached permanently', async (ctx) => { + console.log('[test] matching dynamic path without revalidate') + + await createFixture('page-router-base-path-i18n', ctx) + await runPlugin(ctx) + + const notExistingPage = await invokeFunction(ctx, { + url: '/base/path/products/not-found-no-revalidate', + }) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached permanently', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') + }) + test('matching dynamic path with revalidate should be cached for revalidate time', async (ctx) => { + console.log('[test] matching dynamic path with revalidate') + + await createFixture('page-router-base-path-i18n', ctx) + await runPlugin(ctx) + + const notExistingPage = await invokeFunction(ctx, { + url: '/base/path/products/not-found-with-revalidate', + }) + + expect(notExistingPage.statusCode).toBe(404) + // this page was not prerendered, so no STALE case here + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached for 600 seconds', + ).toBe('s-maxage=600, stale-while-revalidate=31536000, durable') + }) + }) + + describe('404 with getStaticProps with revalidate', () => { + test('not matching dynamic paths should be cached for 404 page revalidate', async (ctx) => { + console.log('[test] not matching dynamic paths') + + await createFixture('page-router-404-get-static-props-with-revalidate', ctx) + await runPlugin(ctx) + + // ignoring initial stale case + await invokeFunction(ctx, { + url: 'not-existing-page', + }) + await new Promise((res) => setTimeout(res, 100)) + + const notExistingPage = await invokeFunction(ctx, { + url: 'not-existing-page', + }) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached permanently', + ).toBe('s-maxage=300, stale-while-revalidate=31536000, durable') + }) + test.only('matching dynamic path without revalidate should be cached permanently', async (ctx) => { + console.log('[test] matching dynamic path without revalidate') + + await createFixture('page-router-404-get-static-props-with-revalidate', ctx) + await runPlugin(ctx) + + // // ignoring initial stale case + // await invokeFunction(ctx, { + // url: 'products/not-found-no-revalidate', + // }) + // await new Promise((res) => setTimeout(res, 100)) + + const notExistingPage = await invokeFunction(ctx, { + url: 'products/not-found-no-revalidate', + }) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached permanently', + ).toBe('s-maxage=31536000, durable') + }) + + test.only('matching dynamic path with revalidate should be cached for revalidate time', async (ctx) => { + console.log('[test] matching dynamic path with revalidate') + + await createFixture('page-router-404-get-static-props-with-revalidate', ctx) + await runPlugin(ctx) + + // // ignoring initial stale case + // await invokeFunction(ctx, { + // url: 'products/not-found-with-revalidate', + // }) + // await new Promise((res) => setTimeout(res, 100)) + + const notExistingPage = await invokeFunction(ctx, { + url: 'products/not-found-with-revalidate', + }) + + expect(notExistingPage.statusCode).toBe(404) + // this page was not prerendered, so no STALE case here + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached for 600 seconds', + ).toBe('s-maxage=600, stale-while-revalidate=31535400, durable') + }) + }) }) From bbedbdb0ac4dc290bc919d2fc55ca44e5dc8204f Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Mon, 24 Feb 2025 10:40:36 +0000 Subject: [PATCH 07/15] test: update 404 caching tests --- tests/fixtures/page-router/pages/404.js | 3 + .../page-router/pages/products/[slug].js | 7 ++ tests/integration/page-router.test.ts | 100 ++++++++---------- 3 files changed, 54 insertions(+), 56 deletions(-) create mode 100644 tests/fixtures/page-router/pages/404.js diff --git a/tests/fixtures/page-router/pages/404.js b/tests/fixtures/page-router/pages/404.js new file mode 100644 index 0000000000..3c251e6665 --- /dev/null +++ b/tests/fixtures/page-router/pages/404.js @@ -0,0 +1,3 @@ +export default function NotFound() { + return

Custom 404 page

+} diff --git a/tests/fixtures/page-router/pages/products/[slug].js b/tests/fixtures/page-router/pages/products/[slug].js index a55c3d0991..319b1f9213 100644 --- a/tests/fixtures/page-router/pages/products/[slug].js +++ b/tests/fixtures/page-router/pages/products/[slug].js @@ -9,6 +9,13 @@ const Product = ({ time, slug }) => ( ) export async function getStaticProps({ params }) { + if (params.slug === 'not-found-with-revalidate') { + return { + notFound: true, + revalidate: 600, + } + } + return { props: { time: new Date().toISOString(), diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index 3c6b624862..8066f1e639 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -154,51 +154,59 @@ test('Should serve correct locale-aware custom 404 pages', a ).toBe('fr') }) -describe('404 caching', () => { - describe('default nextjs 404 (no custom 404)', () => { - test.todo('not matching dynamic paths') - test.todo('matching dynamic path without revalidate') - test.todo('matching dynamic path with revalidate') - }) - +describe.only('404 caching', () => { describe('404 without getStaticProps', () => { - test.todo('not matching dynamic paths') - test.todo('matching dynamic path without revalidate') - test.todo('matching dynamic path with revalidate') - }) - - describe('404 with getStaticProps without revalidate', () => { test('not matching dynamic paths should be cached permanently', async (ctx) => { - console.log('[test] not matching dynamic paths') + await createFixture('page-router', ctx) + await runPlugin(ctx) - await createFixture('page-router-base-path-i18n', ctx) + const notExistingPage = await invokeFunction(ctx, { + url: '/not-existing-page', + }) + + expect(notExistingPage.statusCode).toBe(404) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached permanently', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') + }) + test('matching dynamic path with revalidate should be cached permanently', async (ctx) => { + await createFixture('page-router', ctx) await runPlugin(ctx) const notExistingPage = await invokeFunction(ctx, { - url: '/base/path/not-existing-page', + url: '/products/not-found-with-revalidate', }) + expect(notExistingPage.statusCode).toBe(404) + expect( notExistingPage.headers['netlify-cdn-cache-control'], 'should be cached permanently', ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') }) - test('matching dynamic path without revalidate should be cached permanently', async (ctx) => { - console.log('[test] matching dynamic path without revalidate') + }) + + describe('404 with getStaticProps without revalidate', () => { + test('not matching dynamic paths should be cached permanently', async (ctx) => { + console.log('[test] not matching dynamic paths') await createFixture('page-router-base-path-i18n', ctx) await runPlugin(ctx) const notExistingPage = await invokeFunction(ctx, { - url: '/base/path/products/not-found-no-revalidate', + url: '/base/path/not-existing-page', }) + expect(notExistingPage.statusCode).toBe(404) + expect( notExistingPage.headers['netlify-cdn-cache-control'], 'should be cached permanently', ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') }) - test('matching dynamic path with revalidate should be cached for revalidate time', async (ctx) => { + test('matching dynamic path with revalidate should be cached permanently', async (ctx) => { console.log('[test] matching dynamic path with revalidate') await createFixture('page-router-base-path-i18n', ctx) @@ -209,18 +217,16 @@ describe('404 caching', () => { }) expect(notExistingPage.statusCode).toBe(404) - // this page was not prerendered, so no STALE case here + expect( notExistingPage.headers['netlify-cdn-cache-control'], - 'should be cached for 600 seconds', - ).toBe('s-maxage=600, stale-while-revalidate=31536000, durable') + 'should be cached permanently', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') }) }) describe('404 with getStaticProps with revalidate', () => { test('not matching dynamic paths should be cached for 404 page revalidate', async (ctx) => { - console.log('[test] not matching dynamic paths') - await createFixture('page-router-404-get-static-props-with-revalidate', ctx) await runPlugin(ctx) @@ -228,61 +234,43 @@ describe('404 caching', () => { await invokeFunction(ctx, { url: 'not-existing-page', }) + await new Promise((res) => setTimeout(res, 100)) const notExistingPage = await invokeFunction(ctx, { url: 'not-existing-page', }) - expect( - notExistingPage.headers['netlify-cdn-cache-control'], - 'should be cached permanently', - ).toBe('s-maxage=300, stale-while-revalidate=31536000, durable') - }) - test.only('matching dynamic path without revalidate should be cached permanently', async (ctx) => { - console.log('[test] matching dynamic path without revalidate') - - await createFixture('page-router-404-get-static-props-with-revalidate', ctx) - await runPlugin(ctx) - - // // ignoring initial stale case - // await invokeFunction(ctx, { - // url: 'products/not-found-no-revalidate', - // }) - // await new Promise((res) => setTimeout(res, 100)) - - const notExistingPage = await invokeFunction(ctx, { - url: 'products/not-found-no-revalidate', - }) + expect(notExistingPage.statusCode).toBe(404) expect( notExistingPage.headers['netlify-cdn-cache-control'], - 'should be cached permanently', - ).toBe('s-maxage=31536000, durable') + 'should be cached for 404 page revalidate', + ).toBe('s-maxage=300, stale-while-revalidate=31536000, durable') }) - test.only('matching dynamic path with revalidate should be cached for revalidate time', async (ctx) => { + test('matching dynamic path with revalidate should be cached for 404 page revalidate', async (ctx) => { console.log('[test] matching dynamic path with revalidate') await createFixture('page-router-404-get-static-props-with-revalidate', ctx) await runPlugin(ctx) - // // ignoring initial stale case - // await invokeFunction(ctx, { - // url: 'products/not-found-with-revalidate', - // }) - // await new Promise((res) => setTimeout(res, 100)) + // ignoring initial stale case + await invokeFunction(ctx, { + url: 'products/not-found-with-revalidate', + }) + await new Promise((res) => setTimeout(res, 100)) const notExistingPage = await invokeFunction(ctx, { url: 'products/not-found-with-revalidate', }) expect(notExistingPage.statusCode).toBe(404) - // this page was not prerendered, so no STALE case here + expect( notExistingPage.headers['netlify-cdn-cache-control'], - 'should be cached for 600 seconds', - ).toBe('s-maxage=600, stale-while-revalidate=31535400, durable') + 'should be cached for 404 page revalidate', + ).toBe('s-maxage=300, stale-while-revalidate=31536000, durable') }) }) }) From 3ea93759d51d6a793382c473e451de024739c3a8 Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Mon, 24 Feb 2025 10:46:35 +0000 Subject: [PATCH 08/15] fix: relax the conditions for caching 404s --- src/run/headers.ts | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/run/headers.ts b/src/run/headers.ts index 4a1f4682ab..2dd998258f 100644 --- a/src/run/headers.ts +++ b/src/run/headers.ts @@ -217,7 +217,7 @@ function setCacheControlFromRequestContext( // if we are serving already stale response, instruct edge to not attempt to cache that response headers.get('x-nextjs-cache') === 'STALE' ? 'public, max-age=0, must-revalidate, durable' - : `s-maxage=${revalidate === false ? 31536000 : revalidate}, stale-while-revalidate=31536000, durable` + : `s-maxage=${revalidate || 31536000}, stale-while-revalidate=31536000, durable` headers.set('netlify-cdn-cache-control', cdnCacheControl) } @@ -261,8 +261,6 @@ export const setCacheControlHeaders = ( } if ( - (!cacheControl || cacheControl.includes('no-store')) && - typeof requestContext.pageHandlerRevalidate !== 'undefined' && ['GET', 'HEAD'].includes(request.method) && !headers.has('cdn-cache-control') && !headers.has('netlify-cdn-cache-control') From 5d91e73462f06e374da6927bd10efb72b41e7abb Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Mon, 24 Feb 2025 10:54:53 +0000 Subject: [PATCH 09/15] test: remove only --- tests/integration/page-router.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index 8066f1e639..bd5e35b35a 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -154,7 +154,7 @@ test('Should serve correct locale-aware custom 404 pages', a ).toBe('fr') }) -describe.only('404 caching', () => { +describe('404 caching', () => { describe('404 without getStaticProps', () => { test('not matching dynamic paths should be cached permanently', async (ctx) => { await createFixture('page-router', ctx) From 2962d0b24f704eb5a54afb8f27bd0a89a2ef84a5 Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Tue, 25 Feb 2025 13:22:41 +0000 Subject: [PATCH 10/15] feat: hide 404 caching behind env var --- src/run/headers.ts | 1 + tests/integration/page-router.test.ts | 10 +++++++++- 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/src/run/headers.ts b/src/run/headers.ts index aff964aeab..3faf1f6698 100644 --- a/src/run/headers.ts +++ b/src/run/headers.ts @@ -261,6 +261,7 @@ export const setCacheControlHeaders = ( } if ( + process.env.ENABLE_404_CACHING && ['GET', 'HEAD'].includes(request.method) && !headers.has('cdn-cache-control') && !headers.has('netlify-cdn-cache-control') diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index bd5e35b35a..ee0ac1291c 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -154,7 +154,15 @@ test('Should serve correct locale-aware custom 404 pages', a ).toBe('fr') }) -describe('404 caching', () => { +describe.only('404 caching', () => { + beforeAll(() => { + process.env.ENABLE_404_CACHING = 'true' + }) + + afterAll(() => { + delete process.env.ENABLE_404_CACHING + }) + describe('404 without getStaticProps', () => { test('not matching dynamic paths should be cached permanently', async (ctx) => { await createFixture('page-router', ctx) From 238e6e6e0577c3b6086e7de6b8a6aa40849d4b33 Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Tue, 25 Feb 2025 13:28:18 +0000 Subject: [PATCH 11/15] test: remove only and console logs --- tests/integration/page-router.test.ts | 10 +--------- 1 file changed, 1 insertion(+), 9 deletions(-) diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index ee0ac1291c..9661f12bce 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -19,7 +19,6 @@ beforeAll(() => { // and passthrough everything else server = setupServer( http.post('https://api.netlify.com/api/v1/purge', () => { - console.log('intercepted purge api call') return HttpResponse.json({}) }), http.all(/.*/, () => passthrough()), @@ -91,7 +90,6 @@ test('Should revalidate path with On-demand Revalidation', a expect(staticPageRevalidated.headers?.['cache-status']).toMatch(/"Next.js"; hit/) const dateCacheRevalidated = load(staticPageRevalidated.body)('[data-testid="date-now"]').text() - console.log({ dateCacheInitial, dateCacheRevalidated }) expect(dateCacheInitial).not.toBe(dateCacheRevalidated) }) @@ -154,7 +152,7 @@ test('Should serve correct locale-aware custom 404 pages', a ).toBe('fr') }) -describe.only('404 caching', () => { +describe('404 caching', () => { beforeAll(() => { process.env.ENABLE_404_CACHING = 'true' }) @@ -198,8 +196,6 @@ describe.only('404 caching', () => { describe('404 with getStaticProps without revalidate', () => { test('not matching dynamic paths should be cached permanently', async (ctx) => { - console.log('[test] not matching dynamic paths') - await createFixture('page-router-base-path-i18n', ctx) await runPlugin(ctx) @@ -215,8 +211,6 @@ describe.only('404 caching', () => { ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') }) test('matching dynamic path with revalidate should be cached permanently', async (ctx) => { - console.log('[test] matching dynamic path with revalidate') - await createFixture('page-router-base-path-i18n', ctx) await runPlugin(ctx) @@ -258,8 +252,6 @@ describe.only('404 caching', () => { }) test('matching dynamic path with revalidate should be cached for 404 page revalidate', async (ctx) => { - console.log('[test] matching dynamic path with revalidate') - await createFixture('page-router-404-get-static-props-with-revalidate', ctx) await runPlugin(ctx) From bb64d825be2763993ab67967715fd3ce07532c81 Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Tue, 25 Feb 2025 14:58:47 +0000 Subject: [PATCH 12/15] test: update static 404 test --- tests/integration/static.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/integration/static.test.ts b/tests/integration/static.test.ts index c0fb42fd74..b332577ca4 100644 --- a/tests/integration/static.test.ts +++ b/tests/integration/static.test.ts @@ -54,7 +54,7 @@ test('requesting a non existing page route that needs to be // test that it should request the 404.html file const call1 = await invokeFunction(ctx, { url: 'static/revalidate-not-existing' }) expect(call1.statusCode).toBe(404) - expect(load(call1.body)('h1').text()).toBe('404') + expect(load(call1.body)('p').text()).toBe('Custom 404 page') // https://github.com/vercel/next.js/pull/69802 made changes to returned cache-control header, // after that (14.2.10 and canary.147) 404 pages would have `private` directive, before that it From f31d30c11cd5ae1b9e83e763e06acb9ed0c6981d Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Thu, 27 Feb 2025 18:21:51 +0000 Subject: [PATCH 13/15] feat: reduce 404 caching scope to just the 404 page itself --- src/run/headers.ts | 7 +- tests/integration/page-router.test.ts | 99 ++++++++++++++++++++++----- 2 files changed, 84 insertions(+), 22 deletions(-) diff --git a/src/run/headers.ts b/src/run/headers.ts index 3faf1f6698..cb4c1f4640 100644 --- a/src/run/headers.ts +++ b/src/run/headers.ts @@ -261,10 +261,9 @@ export const setCacheControlHeaders = ( } if ( - process.env.ENABLE_404_CACHING && - ['GET', 'HEAD'].includes(request.method) && - !headers.has('cdn-cache-control') && - !headers.has('netlify-cdn-cache-control') + process.env.CACHE_404_PAGE && + request.url.endsWith('/404') && + ['GET', 'HEAD'].includes(request.method) ) { // handle CDN Cache Control on 404 Page responses setCacheControlFromRequestContext(headers, requestContext.pageHandlerRevalidate) diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index 9661f12bce..8fda544d67 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -152,15 +152,10 @@ test('Should serve correct locale-aware custom 404 pages', a ).toBe('fr') }) -describe('404 caching', () => { - beforeAll(() => { - process.env.ENABLE_404_CACHING = 'true' - }) - - afterAll(() => { - delete process.env.ENABLE_404_CACHING - }) - +// These tests describe how the 404 caching should work, but unfortunately it doesn't work like +// this in v5 and a fix would represent a breaking change so we are skipping them for now, but +// leaving them here for future reference when considering the next major version +describe.skip('404 caching', () => { describe('404 without getStaticProps', () => { test('not matching dynamic paths should be cached permanently', async (ctx) => { await createFixture('page-router', ctx) @@ -177,7 +172,7 @@ describe('404 caching', () => { 'should be cached permanently', ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') }) - test('matching dynamic path with revalidate should be cached permanently', async (ctx) => { + test('matching dynamic path with revalidate should be cached for revalidate time', async (ctx) => { await createFixture('page-router', ctx) await runPlugin(ctx) @@ -189,8 +184,8 @@ describe('404 caching', () => { expect( notExistingPage.headers['netlify-cdn-cache-control'], - 'should be cached permanently', - ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') + 'should be cached for revalidate time', + ).toBe('s-maxage=600, stale-while-revalidate=31536000, durable') }) }) @@ -210,7 +205,7 @@ describe('404 caching', () => { 'should be cached permanently', ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') }) - test('matching dynamic path with revalidate should be cached permanently', async (ctx) => { + test('matching dynamic path with revalidate should be cached for revalidate time', async (ctx) => { await createFixture('page-router-base-path-i18n', ctx) await runPlugin(ctx) @@ -222,8 +217,8 @@ describe('404 caching', () => { expect( notExistingPage.headers['netlify-cdn-cache-control'], - 'should be cached permanently', - ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') + 'should be cached for revalidate time', + ).toBe('s-maxage=600, stale-while-revalidate=31536000, durable') }) }) @@ -251,7 +246,7 @@ describe('404 caching', () => { ).toBe('s-maxage=300, stale-while-revalidate=31536000, durable') }) - test('matching dynamic path with revalidate should be cached for 404 page revalidate', async (ctx) => { + test('matching dynamic path with revalidate should be cached for revalidate time', async (ctx) => { await createFixture('page-router-404-get-static-props-with-revalidate', ctx) await runPlugin(ctx) @@ -269,8 +264,76 @@ describe('404 caching', () => { expect( notExistingPage.headers['netlify-cdn-cache-control'], - 'should be cached for 404 page revalidate', - ).toBe('s-maxage=300, stale-while-revalidate=31536000, durable') + 'should be cached for revalidate time', + ).toBe('s-maxage=600, stale-while-revalidate=31536000, durable') + }) + }) +}) + +// This is a temporary fix to ensure that the 404 page itself is cached correctly when requested +// directly. This is a workaround for a specific customer and should be removed once the 404 caching +// is fixed in the next major version. +describe.only('404 page caching', () => { + beforeAll(() => { + process.env.CACHE_404_PAGE = 'true' + }) + + afterAll(() => { + delete process.env.CACHE_404_PAGE + }) + + test('404 without getStaticProps', async (ctx) => { + await createFixture('page-router', ctx) + await runPlugin(ctx) + + const notExistingPage = await invokeFunction(ctx, { + url: '/404', + }) + + expect(notExistingPage.statusCode).toBe(404) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached permanently', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') + }) + + test('404 with getStaticProps without revalidate', async (ctx) => { + await createFixture('page-router-base-path-i18n', ctx) + await runPlugin(ctx) + + const notExistingPage = await invokeFunction(ctx, { + url: '/base/404', + }) + + expect(notExistingPage.statusCode).toBe(404) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached permanently', + ).toBe('s-maxage=31536000, stale-while-revalidate=31536000, durable') + }) + + test('404 with getStaticProps with revalidate', async (ctx) => { + await createFixture('page-router-404-get-static-props-with-revalidate', ctx) + await runPlugin(ctx) + + // ignoring initial stale case + await invokeFunction(ctx, { + url: '/404', }) + + await new Promise((res) => setTimeout(res, 100)) + + const notExistingPage = await invokeFunction(ctx, { + url: '/404', + }) + + expect(notExistingPage.statusCode).toBe(404) + + expect( + notExistingPage.headers['netlify-cdn-cache-control'], + 'should be cached for 404 page revalidate', + ).toBe('s-maxage=300, stale-while-revalidate=31536000, durable') }) }) From a6f37e2cf630485608fbd4c36995ec5035d7cc34 Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Thu, 27 Feb 2025 19:18:19 +0000 Subject: [PATCH 14/15] test: remove only modifier --- tests/integration/page-router.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/integration/page-router.test.ts b/tests/integration/page-router.test.ts index 8fda544d67..25dbfaba41 100644 --- a/tests/integration/page-router.test.ts +++ b/tests/integration/page-router.test.ts @@ -273,7 +273,7 @@ describe.skip('404 caching', () => { // This is a temporary fix to ensure that the 404 page itself is cached correctly when requested // directly. This is a workaround for a specific customer and should be removed once the 404 caching // is fixed in the next major version. -describe.only('404 page caching', () => { +describe('404 page caching', () => { beforeAll(() => { process.env.CACHE_404_PAGE = 'true' }) From 70a3ae6e6b3f59666c525406dba01d92d67d73c3 Mon Sep 17 00:00:00 2001 From: Rob Stanford Date: Fri, 28 Feb 2025 14:46:28 +0000 Subject: [PATCH 15/15] test: fix tests for new custom 404 pate --- tests/e2e/page-router.test.ts | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/tests/e2e/page-router.test.ts b/tests/e2e/page-router.test.ts index 06f8abc29d..7be19bee58 100644 --- a/tests/e2e/page-router.test.ts +++ b/tests/e2e/page-router.test.ts @@ -1,6 +1,6 @@ import { expect } from '@playwright/test' -import { test } from '../utils/playwright-helpers.js' import { nextVersionSatisfies } from '../utils/next-version-helpers.mjs' +import { test } from '../utils/playwright-helpers.js' export function waitFor(millis: number) { return new Promise((resolve) => setTimeout(resolve, millis)) @@ -405,7 +405,7 @@ test.describe('Simple Page Router (no basePath, no i18n)', () => { const headers = response?.headers() || {} expect(response?.status()).toBe(404) - expect(await page.textContent('h1')).toBe('404') + expect(await page.textContent('p')).toBe('Custom 404 page') // https://github.com/vercel/next.js/pull/69802 made changes to returned cache-control header, // after that (14.2.10 and canary.147) 404 pages would have `private` directive, before that @@ -429,7 +429,7 @@ test.describe('Simple Page Router (no basePath, no i18n)', () => { const headers = response?.headers() || {} expect(response?.status()).toBe(404) - expect(await page.textContent('h1')).toBe('404') + expect(await page.textContent('p')).toBe('Custom 404 page') expect(headers['netlify-cdn-cache-control']).toBe( nextVersionSatisfies('>=15.0.0-canary.187') @@ -1269,7 +1269,7 @@ test.describe('Page Router with basePath and i18n', () => { const headers = response?.headers() || {} expect(response?.status()).toBe(404) - expect(await page.textContent('h1')).toBe('404') + expect(await page.textContent('p')).toBe('Custom 404 page') // https://github.com/vercel/next.js/pull/69802 made changes to returned cache-control header, // after that 404 pages would have `private` directive, before that it would not @@ -1292,7 +1292,7 @@ test.describe('Page Router with basePath and i18n', () => { const headers = response?.headers() || {} expect(response?.status()).toBe(404) - expect(await page.textContent('h1')).toBe('404') + expect(await page.textContent('p')).toBe('Custom 404 page') expect(headers['netlify-cdn-cache-control']).toBe( nextVersionSatisfies('>=15.0.0-canary.187')