Skip to content

Commit 87fe9b4

Browse files
committed
test: add e2e case testing storing fresh responses in cdn when handling background SWR requests
1 parent 0a28ab9 commit 87fe9b4

File tree

4 files changed

+133
-25
lines changed

4 files changed

+133
-25
lines changed

tests/e2e/page-router.test.ts

+57
Original file line numberDiff line numberDiff line change
@@ -390,6 +390,63 @@ test.describe('Simple Page Router (no basePath, no i18n)', () => {
390390
expect(beforeFetch.localeCompare(date2)).toBeLessThan(0)
391391
})
392392

393+
test('Background SWR invocations can store fresh responses in CDN cache', async ({
394+
page,
395+
pageRouter,
396+
}) => {
397+
const slug = Date.now()
398+
const pathname = `/revalidate-60/${slug}`
399+
400+
const beforeFirstFetch = new Date().toISOString()
401+
402+
const response1 = await page.goto(new URL(pathname, pageRouter.url).href)
403+
expect(response1?.status()).toBe(200)
404+
expect(response1?.headers()['cache-status']).toMatch(
405+
/"Netlify (Edge|Durable)"; fwd=(uri-miss(; stored)?|miss)/m,
406+
)
407+
expect(response1?.headers()['netlify-cdn-cache-control']).toMatch(
408+
/s-maxage=60, stale-while-revalidate=[0-9]+, durable/,
409+
)
410+
411+
// ensure response was NOT produced before invocation
412+
const date1 = (await page.textContent('[data-testid="date-now"]')) ?? ''
413+
expect(date1.localeCompare(beforeFirstFetch)).toBeGreaterThan(0)
414+
415+
// allow page to get stale
416+
await page.waitForTimeout(60_000)
417+
418+
const response2 = await page.goto(new URL(pathname, pageRouter.url).href)
419+
expect(response2?.status()).toBe(200)
420+
expect(response2?.headers()['cache-status']).toMatch(
421+
/"Netlify (Edge|Durable)"; hit; fwd=stale/m,
422+
)
423+
expect(response2?.headers()['netlify-cdn-cache-control']).toMatch(
424+
/s-maxage=60, stale-while-revalidate=[0-9]+, durable/,
425+
)
426+
427+
const date2 = (await page.textContent('[data-testid="date-now"]')) ?? ''
428+
expect(date2).toBe(date1)
429+
430+
// wait a bit to ensure background work has a chance to finish
431+
// (it should take at least 5 seconds to regenerate, so we should wait at least that much to get fresh response)
432+
await page.waitForTimeout(10_000)
433+
434+
// subsequent request should be served with fresh response from cdn cache, as previous request
435+
// should result in background SWR invocation that serves fresh response that was stored in CDN cache
436+
const response3 = await page.goto(new URL(pathname, pageRouter.url).href)
437+
expect(response3?.status()).toBe(200)
438+
expect(response3?.headers()['cache-status']).toMatch(
439+
// hit, without being followed by ';fwd=stale'
440+
/"Netlify (Edge|Durable)"; hit(?!; fwd=stale)/m,
441+
)
442+
expect(response3?.headers()['netlify-cdn-cache-control']).toMatch(
443+
/s-maxage=60, stale-while-revalidate=[0-9]+, durable/,
444+
)
445+
446+
const date3 = (await page.textContent('[data-testid="date-now"]')) ?? ''
447+
expect(date3.localeCompare(date2)).toBeGreaterThan(0)
448+
})
449+
393450
test('should serve 404 page when requesting non existing page (no matching route)', async ({
394451
page,
395452
pageRouter,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import { purgeCache, Config } from '@netlify/functions'
2+
3+
export default async function handler(request: Request) {
4+
const url = new URL(request.url)
5+
const pathToPurge = url.searchParams.get('path')
6+
7+
if (!pathToPurge) {
8+
return Response.json(
9+
{
10+
status: 'error',
11+
error: 'missing "path" query parameter',
12+
},
13+
{ status: 400 },
14+
)
15+
}
16+
try {
17+
await purgeCache({ tags: [`_N_T_${encodeURI(pathToPurge)}`] })
18+
return Response.json(
19+
{
20+
status: 'ok',
21+
},
22+
{
23+
status: 200,
24+
},
25+
)
26+
} catch (error) {
27+
return Response.json(
28+
{
29+
status: 'error',
30+
error: error.toString(),
31+
},
32+
{
33+
status: 500,
34+
},
35+
)
36+
}
37+
}
38+
39+
export const config: Config = {
40+
path: '/api/purge-cdn',
41+
}

tests/fixtures/page-router/pages/api/purge-cdn.js

-25
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
const Show = ({ time, easyTimeToCompare, slug }) => (
2+
<div>
3+
<p>
4+
This page uses getStaticProps() at
5+
<span data-testid="date-now">{time}</span>
6+
</p>
7+
<p>
8+
Time string: <span data-testid="date-easy-time">{easyTimeToCompare}</span>
9+
</p>
10+
<p>Slug {slug}</p>
11+
</div>
12+
)
13+
14+
/** @type {import('next').getStaticPaths} */
15+
export const getStaticPaths = () => {
16+
return {
17+
paths: [],
18+
fallback: 'blocking',
19+
}
20+
}
21+
22+
/** @type {import('next').GetStaticProps} */
23+
export async function getStaticProps({ params }) {
24+
const date = new Date()
25+
return {
26+
props: {
27+
slug: params.slug,
28+
time: date.toISOString(),
29+
easyTimeToCompare: date.toTimeString(),
30+
},
31+
revalidate: 60,
32+
}
33+
}
34+
35+
export default Show

0 commit comments

Comments
 (0)