Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

test: expand test fixture and cases for dynamic-cms tests with added i18n to fixture and testing i18n cases and json requests #2813

Merged
merged 1 commit into from
Apr 3, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
154 changes: 102 additions & 52 deletions tests/e2e/dynamic-cms.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,57 +2,107 @@ import { expect } from '@playwright/test'
import { test } from '../utils/playwright-helpers.js'

test.describe('Dynamic CMS', () => {
test('Invalidates 404 pages from durable cache', async ({ page, dynamicCms }) => {
// 1. Verify the status and headers of the dynamic page
const response1 = await page.goto(new URL('/content/blog', dynamicCms.url).href)
const headers1 = response1?.headers() || {}

expect(response1?.status()).toEqual(404)
expect(headers1['cache-control']).toEqual('public,max-age=0,must-revalidate')
expect(headers1['cache-status']).toEqual(
'"Next.js"; fwd=miss, "Netlify Durable"; fwd=uri-miss; stored, "Netlify Edge"; fwd=miss',
)
expect(headers1['netlify-cache-tag']).toEqual('_n_t_/content/blog')
expect(headers1['netlify-cdn-cache-control']).toMatch(
/s-maxage=31536000,( stale-while-revalidate=31536000,)? durable/,
)

// 2. Publish the blob, revalidate the dynamic page, and wait to regenerate
await page.goto(new URL('/cms/publish', dynamicCms.url).href)
await page.goto(new URL('/api/revalidate?path=/content/blog', dynamicCms.url).href)
await page.waitForTimeout(1000)

// 3. Verify the status and headers of the dynamic page
const response2 = await page.goto(new URL('/content/blog', dynamicCms.url).href)
const headers2 = response2?.headers() || {}

expect(response2?.status()).toEqual(200)
expect(headers2['cache-control']).toEqual('public,max-age=0,must-revalidate')
expect(headers2['cache-status']).toMatch(
/"Next.js"; hit, "Netlify Durable"; fwd=stale; ttl=[0-9]+; stored, "Netlify Edge"; fwd=(stale|miss)/,
)
expect(headers2['netlify-cache-tag']).toEqual('_n_t_/content/blog')
expect(headers2['netlify-cdn-cache-control']).toMatch(
/s-maxage=31536000,( stale-while-revalidate=31536000,)? durable/,
)

// 4. Unpublish the blob, revalidate the dynamic page, and wait to regenerate
await page.goto(new URL('/cms/unpublish', dynamicCms.url).href)
await page.goto(new URL('/api/revalidate?path=/content/blog', dynamicCms.url).href)
await page.waitForTimeout(1000)

// 5. Verify the status and headers of the dynamic page
const response3 = await page.goto(new URL('/content/blog', dynamicCms.url).href)
const headers3 = response3?.headers() || {}

expect(response3?.status()).toEqual(404)
expect(headers3['cache-control']).toEqual('public,max-age=0,must-revalidate')
expect(headers3['cache-status']).toMatch(
/"Next.js"; fwd=miss, "Netlify Durable"; fwd=stale; ttl=[0-9]+; stored, "Netlify Edge"; fwd=(stale|miss)/,
)
expect(headers3['netlify-cache-tag']).toEqual('_n_t_/content/blog')
expect(headers3['netlify-cdn-cache-control']).toMatch(
/s-maxage=31536000,( stale-while-revalidate=31536000,)? durable/,
)
test.describe('Invalidates 404 pages from durable cache', () => {
// using postFix allows to rerun tests without having to redeploy the app because paths/keys will be unique for each test run
const postFix = Date.now()
for (const { label, contentKey, expectedCacheTag, urlPath, pathToRevalidate } of [
{
label: 'Invalidates 404 html from durable cache (implicit default locale)',
urlPath: `/content/html-implicit-default-locale-${postFix}`,
contentKey: `html-implicit-default-locale-${postFix}`,
expectedCacheTag: `_n_t_/en/content/html-implicit-default-locale-${postFix}`,
},
{
label: 'Invalidates 404 html from durable cache (explicit default locale)',
urlPath: `/en/content/html-explicit-default-locale-${postFix}`,
contentKey: `html-explicit-default-locale-${postFix}`,
expectedCacheTag: `_n_t_/en/content/html-explicit-default-locale-${postFix}`,
},
// json paths don't have implicit locale routing
{
label: 'Invalidates 404 json from durable cache (default locale)',
urlPath: `/_next/data/build-id/en/content/json-default-locale-${postFix}.json`,
// for html, we can use html path as param for revalidate,
// for json we can't use json path and instead use one of html paths
// let's use implicit default locale here, as we will have another case for
// non-default locale which will have to use explicit one
pathToRevalidate: `/content/json-default-locale-${postFix}`,
contentKey: `json-default-locale-${postFix}`,
expectedCacheTag: `_n_t_/en/content/json-default-locale-${postFix}`,
},
{
label: 'Invalidates 404 html from durable cache (non-default locale)',
urlPath: `/fr/content/html-non-default-locale-${postFix}`,
contentKey: `html-non-default-locale-${postFix}`,
expectedCacheTag: `_n_t_/fr/content/html-non-default-locale-${postFix}`,
},
{
label: 'Invalidates 404 json from durable cache (non-default locale)',
urlPath: `/_next/data/build-id/fr/content/json-non-default-locale-${postFix}.json`,
pathToRevalidate: `/fr/content/json-non-default-locale-${postFix}`,
contentKey: `json-non-default-locale-${postFix}`,
expectedCacheTag: `_n_t_/fr/content/json-non-default-locale-${postFix}`,
},
]) {
test(label, async ({ page, dynamicCms }) => {
const routeUrl = new URL(urlPath, dynamicCms.url).href
const revalidateAPiUrl = new URL(
`/api/revalidate?path=${pathToRevalidate ?? urlPath}`,
dynamicCms.url,
).href

// 1. Verify the status and headers of the dynamic page
const response1 = await page.goto(routeUrl)
const headers1 = response1?.headers() || {}

expect(response1?.status()).toEqual(404)
expect(headers1['cache-control']).toEqual('public,max-age=0,must-revalidate')
expect(headers1['cache-status']).toEqual(
'"Next.js"; fwd=miss, "Netlify Durable"; fwd=uri-miss; stored, "Netlify Edge"; fwd=miss',
)
expect(headers1['netlify-cache-tag']).toEqual(expectedCacheTag)
expect(headers1['netlify-cdn-cache-control']).toMatch(
/s-maxage=31536000,( stale-while-revalidate=31536000,)? durable/,
)

// 2. Publish the blob, revalidate the dynamic page, and wait to regenerate
await page.goto(new URL(`/cms/publish/${contentKey}`, dynamicCms.url).href)
await page.goto(revalidateAPiUrl)
await page.waitForTimeout(1000)

// 3. Verify the status and headers of the dynamic page
const response2 = await page.goto(routeUrl)
const headers2 = response2?.headers() || {}

expect(response2?.status()).toEqual(200)
expect(headers2['cache-control']).toEqual('public,max-age=0,must-revalidate')
expect(headers2['cache-status']).toMatch(
/"Next.js"; hit, "Netlify Durable"; fwd=stale; ttl=[0-9]+; stored, "Netlify Edge"; fwd=(stale|miss)/,
)
expect(headers2['netlify-cache-tag']).toEqual(expectedCacheTag)
expect(headers2['netlify-cdn-cache-control']).toMatch(
/s-maxage=31536000,( stale-while-revalidate=31536000,)? durable/,
)

// 4. Unpublish the blob, revalidate the dynamic page, and wait to regenerate
await page.goto(new URL(`/cms/unpublish/${contentKey}`, dynamicCms.url).href)
await page.goto(revalidateAPiUrl)
await page.waitForTimeout(1000)

// 5. Verify the status and headers of the dynamic page
const response3 = await page.goto(routeUrl)
const headers3 = response3?.headers() || {}

expect(response3?.status()).toEqual(404)
expect(headers3['cache-control']).toEqual('public,max-age=0,must-revalidate')
expect(headers3['cache-status']).toMatch(
/"Next.js"; fwd=miss, "Netlify Durable"; fwd=stale; ttl=[0-9]+; stored, "Netlify Edge"; fwd=(stale|miss)/,
)
expect(headers3['netlify-cache-tag']).toEqual(expectedCacheTag)
expect(headers3['netlify-cdn-cache-control']).toMatch(
/s-maxage=31536000,( stale-while-revalidate=31536000,)? durable/,
)
})
}
})
})
Original file line number Diff line number Diff line change
Expand Up @@ -4,21 +4,25 @@ import { Context } from '@netlify/functions'
// publish or unpublish "cms content" depending on the sent operation
export default async function handler(_request: Request, context: Context) {
const store = getDeployStore({ name: 'cms-content', consistency: 'strong' })
const BLOB_KEY = 'key'

const operation = context.params['operation']

// root of optional catch-all route in Next.js sets 'index.html' as param
// while it's undefined in the Netlify function, because we need to declare
// path without wildcard
const contentKey = context.params['0'] ?? 'index.html'

if (operation === 'publish') {
await store.setJSON(BLOB_KEY, { content: true })
await store.setJSON(contentKey, { content: true })
}

if (operation === 'unpublish') {
await store.delete(BLOB_KEY)
await store.delete(contentKey)
}

return Response.json({ ok: true })
}

export const config = {
path: '/cms/:operation',
path: ['/cms/:operation/*', '/cms/:operation'],
}
4 changes: 4 additions & 0 deletions tests/fixtures/dynamic-cms/next.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,10 @@ const nextConfig = {
eslint: {
ignoreDuringBuilds: true,
},
i18n: {
locales: ['en', 'fr'],
defaultLocale: 'en',
},
generateBuildId: () => 'build-id',
}

Expand Down
1 change: 0 additions & 1 deletion tests/fixtures/dynamic-cms/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@
"dependencies": {
"@netlify/blobs": "^8.1.0",
"@netlify/functions": "^2.7.0",
"@netlify/plugin-nextjs": "^5.10.1",
"netlify-cli": "^19.0.3",
"next": "latest",
"react": "18.2.0",
Expand Down
9 changes: 5 additions & 4 deletions tests/fixtures/dynamic-cms/pages/content/[...slug].js
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,12 @@ const Content = ({ value }) => (
</div>
)

export async function getStaticProps() {
export async function getStaticProps({ params }) {
const contentKey = params.slug.join('/')

const store = getDeployStore({ name: 'cms-content', consistency: 'strong' })
const BLOB_KEY = 'key'

const value = await store.get(BLOB_KEY, { type: 'json' })
const value = await store.get(contentKey, { type: 'json' })

if (!value) {
return {
Expand All @@ -22,7 +23,7 @@ export async function getStaticProps() {

return {
props: {
value: value,
value,
},
}
}
Expand Down
Loading