Skip to content

Commit 27a5645

Browse files
authored
fix: handle pages router notFound pages (#318)
* test: add page router not found case * test: add app router not found case * test: add i18n basePath case * fix: don't try to upload not found routes * fix: app router notFound() should have 404 status
1 parent 639fc52 commit 27a5645

File tree

7 files changed

+128
-7
lines changed

7 files changed

+128
-7
lines changed

src/build/content/prerendered.ts

+19-6
Original file line numberDiff line numberDiff line change
@@ -46,12 +46,21 @@ const buildPagesCacheValue = async (path: string): Promise<CachedPageValue> => (
4646
status: undefined,
4747
})
4848

49-
const buildAppCacheValue = async (path: string): Promise<CachedPageValue> => ({
50-
kind: 'PAGE',
51-
html: await readFile(`${path}.html`, 'utf-8'),
52-
pageData: await readFile(`${path}.rsc`, 'utf-8'),
53-
...JSON.parse(await readFile(`${path}.meta`, 'utf-8')),
54-
})
49+
const buildAppCacheValue = async (path: string): Promise<CachedPageValue> => {
50+
const meta = JSON.parse(await readFile(`${path}.meta`, 'utf-8'))
51+
const rsc = await readFile(`${path}.rsc`, 'utf-8')
52+
53+
if (!meta.status && rsc.includes('NEXT_NOT_FOUND')) {
54+
meta.status = 404
55+
}
56+
57+
return {
58+
kind: 'PAGE',
59+
html: await readFile(`${path}.html`, 'utf-8'),
60+
pageData: rsc,
61+
...meta,
62+
}
63+
}
5564

5665
const buildRouteCacheValue = async (path: string): Promise<CachedRouteValue> => ({
5766
kind: 'ROUTE',
@@ -83,6 +92,10 @@ export const copyPrerenderedContent = async (ctx: PluginContext): Promise<void>
8392
!existsSync(join(ctx.publishDir, 'server/app', `${key}.html`)):
8493
return
8594
case meta.dataRoute?.endsWith('.json'):
95+
if (manifest.notFoundRoutes.includes(route)) {
96+
// if pages router returns 'notFound: true', build won't produce html and json files
97+
return
98+
}
8699
value = await buildPagesCacheValue(join(ctx.publishDir, 'server/pages', key))
87100
break
88101
case meta.dataRoute?.endsWith('.rsc'):

tests/e2e/page-router.test.ts

+48
Original file line numberDiff line numberDiff line change
@@ -208,6 +208,22 @@ test.describe('Simple Page Router (no basePath, no i18n)', () => {
208208
expect(headers['cache-control']).toBe('no-cache,no-store,max-age=0,must-revalidate')
209209
})
210210

211+
test('requesting a non existing page route that needs to be fetched from the blob store like 404.html (notFound: true)', async ({
212+
page,
213+
pageRouter,
214+
}) => {
215+
const response = await page.goto(new URL('static/not-found', pageRouter.url).href)
216+
const headers = response?.headers() || {}
217+
expect(response?.status()).toBe(404)
218+
219+
expect(await page.textContent('h1')).toBe('404')
220+
221+
expect(headers['netlify-cdn-cache-control']).toBe(
222+
's-maxage=31536000, stale-while-revalidate=31536000',
223+
)
224+
expect(headers['cache-control']).toBe('public,max-age=0,must-revalidate')
225+
})
226+
211227
test('requesting a page with a very long name works', async ({ page, pageRouter }) => {
212228
const response = await page.goto(
213229
new URL(
@@ -740,4 +756,36 @@ test.describe('Page Router with basePath and i18n', () => {
740756
expect(data3?.pageProps?.time).toBe(date3)
741757
})
742758
})
759+
760+
test('requesting a non existing page route that needs to be fetched from the blob store like 404.html', async ({
761+
page,
762+
pageRouter,
763+
}) => {
764+
const response = await page.goto(new URL('non-exisitng', pageRouter.url).href)
765+
const headers = response?.headers() || {}
766+
expect(response?.status()).toBe(404)
767+
768+
expect(await page.textContent('h1')).toBe('404')
769+
770+
expect(headers['netlify-cdn-cache-control']).toBe(
771+
'no-cache, no-store, max-age=0, must-revalidate',
772+
)
773+
expect(headers['cache-control']).toBe('no-cache,no-store,max-age=0,must-revalidate')
774+
})
775+
776+
test('requesting a non existing page route that needs to be fetched from the blob store like 404.html (notFound: true)', async ({
777+
page,
778+
pageRouter,
779+
}) => {
780+
const response = await page.goto(new URL('static/not-found', pageRouter.url).href)
781+
const headers = response?.headers() || {}
782+
expect(response?.status()).toBe(404)
783+
784+
expect(await page.textContent('h1')).toBe('404')
785+
786+
expect(headers['netlify-cdn-cache-control']).toBe(
787+
's-maxage=31536000, stale-while-revalidate=31536000',
788+
)
789+
expect(headers['cache-control']).toBe('public,max-age=0,must-revalidate')
790+
})
743791
})

tests/e2e/simple-app.test.ts

+32
Original file line numberDiff line numberDiff line change
@@ -170,3 +170,35 @@ test.describe('next/image is using Netlify Image CDN', () => {
170170
await expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
171171
})
172172
})
173+
174+
test('requesting a non existing page route that needs to be fetched from the blob store like 404.html', async ({
175+
page,
176+
simpleNextApp,
177+
}) => {
178+
const response = await page.goto(new URL('non-exisitng', simpleNextApp.url).href)
179+
const headers = response?.headers() || {}
180+
expect(response?.status()).toBe(404)
181+
182+
expect(await page.textContent('h1')).toBe('404')
183+
184+
expect(headers['netlify-cdn-cache-control']).toBe(
185+
'private, no-cache, no-store, max-age=0, must-revalidate',
186+
)
187+
expect(headers['cache-control']).toBe('private,no-cache,no-store,max-age=0,must-revalidate')
188+
})
189+
190+
test('requesting a non existing page route that needs to be fetched from the blob store like 404.html (notFound())', async ({
191+
page,
192+
simpleNextApp,
193+
}) => {
194+
const response = await page.goto(new URL('not-found', simpleNextApp.url).href)
195+
const headers = response?.headers() || {}
196+
expect(response?.status()).toBe(404)
197+
198+
expect(await page.textContent('h1')).toBe('404')
199+
200+
expect(headers['netlify-cdn-cache-control']).toBe(
201+
's-maxage=31536000, stale-while-revalidate=31536000',
202+
)
203+
expect(headers['cache-control']).toBe('public,max-age=0,must-revalidate')
204+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
const Show = () => <div>Won't be used</div>
2+
3+
export async function getStaticProps() {
4+
return {
5+
notFound: true,
6+
}
7+
}
8+
9+
export default Show
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
const Show = () => <div>Won't be used</div>
2+
3+
export async function getStaticProps() {
4+
return {
5+
notFound: true,
6+
}
7+
}
8+
9+
export default Show
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
import { notFound } from 'next/navigation'
2+
3+
export default async function Page() {
4+
notFound()
5+
}

tests/integration/simple-app.test.ts

+6-1
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ test<FixtureTestContext>('Test that the simple next app is working', async (ctx)
4242
'/image/remote-pattern-1',
4343
'/image/remote-pattern-2',
4444
'/index',
45+
'/not-found',
4546
'/other',
4647
'/redirect',
4748
'/redirect/response',
@@ -60,7 +61,11 @@ test<FixtureTestContext>('Test that the simple next app is working', async (ctx)
6061

6162
const notFound = await invokeFunction(ctx, { url: 'not-found' })
6263
expect(notFound.statusCode).toBe(404)
63-
expect(load(notFound.body)('h1').text()).toBe('404')
64+
expect(notFound.body).toContain('NEXT_NOT_FOUND')
65+
66+
const notExisting = await invokeFunction(ctx, { url: 'non-exisitng' })
67+
expect(notExisting.statusCode).toBe(404)
68+
expect(load(notExisting.body)('h1').text()).toBe('404')
6469
})
6570

6671
test<FixtureTestContext>('Should warn if publish dir is root', async (ctx) => {

0 commit comments

Comments
 (0)