Skip to content

Commit

Permalink
test: add integration test for site-wide purge calls
Browse files Browse the repository at this point in the history
  • Loading branch information
pieh committed Dec 19, 2024
1 parent 6a2c39c commit ae4eaa9
Show file tree
Hide file tree
Showing 2 changed files with 85 additions and 0 deletions.
21 changes: 21 additions & 0 deletions tests/fixtures/simple/app/unstable_cache/page.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
import { unstable_cache } from 'next/cache'

export const dynamic = 'force-dynamic'

const getData = unstable_cache(
async () => {
return {
timestamp: Date.now(),
}
},
[],
{
revalidate: 1,
},
)

export default async function Page() {
const data = await getData()

return <pre>{JSON.stringify(data, null, 2)}</pre>
}
64 changes: 64 additions & 0 deletions tests/integration/simple-app.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@ import { cp } from 'node:fs/promises'
import { createRequire } from 'node:module'
import { join } from 'node:path'
import { gunzipSync } from 'node:zlib'
import { HttpResponse, http, passthrough } from 'msw'
import { setupServer } from 'msw/node'
import { gt, prerelease } from 'semver'
import { v4 } from 'uuid'
import { Mock, afterAll, beforeAll, beforeEach, describe, expect, test, vi } from 'vitest'
Expand All @@ -22,6 +24,8 @@ import {
startMockBlobStore,
} from '../utils/helpers.js'
import { nextVersionSatisfies } from '../utils/next-version-helpers.mjs'
import { purgeCache } from '@netlify/functions'
import { afterEach } from 'node:test'

const mockedCp = cp as Mock<
Parameters<(typeof import('node:fs/promises'))['cp']>,
Expand All @@ -36,9 +40,32 @@ vi.mock('node:fs/promises', async (importOriginal) => {
}
})

let server: ReturnType<typeof setupServer>

// Disable the verbose logging of the lambda-local runtime
getLogger().level = 'alert'

const purgeAPI = vi.fn()

beforeAll(() => {
server = setupServer(
http.post('https://api.netlify.com/api/v1/purge', async ({ request }) => {
purgeAPI(await request.json())

return HttpResponse.json({
ok: true,
})
}),
http.all(/.*/, () => passthrough()),
)
server.listen()
})

afterAll(() => {
// Disable API mocking after the tests are done.
server.close()
})

beforeEach<FixtureTestContext>(async (ctx) => {
// set for each test a new deployID and siteID
ctx.deployID = generateRandomObjectID()
Expand All @@ -51,6 +78,10 @@ beforeEach<FixtureTestContext>(async (ctx) => {
await startMockBlobStore(ctx)
})

afterEach(() => {
vi.unstubAllEnvs()
})

test<FixtureTestContext>('Test that the simple next app is working', async (ctx) => {
await createFixture('simple', ctx)
await runPlugin(ctx)
Expand Down Expand Up @@ -210,6 +241,39 @@ test<FixtureTestContext>('cacheable route handler is cached on cdn (revalidate=f
)
})

test<FixtureTestContext>('purge API is not used when unstable_cache cache entry gets stale', async (ctx) => {
await createFixture('simple', ctx)
await runPlugin(ctx)

// set the NETLIFY_PURGE_API_TOKEN to get pass token check and allow fetch call to be made
vi.stubEnv('NETLIFY_PURGE_API_TOKEN', 'mock')

const page1 = await invokeFunction(ctx, {
url: '/unstable_cache',
})
const data1 = load(page1.body)('pre').text()

// allow for cache entry to get stale
await new Promise((res) => setTimeout(res, 2000))

const page2 = await invokeFunction(ctx, {
url: '/unstable_cache',
})
const data2 = load(page2.body)('pre').text()

const page3 = await invokeFunction(ctx, {
url: '/unstable_cache',
})
const data3 = load(page3.body)('pre').text()

expect(purgeAPI, 'Purge API should not be hit').toHaveBeenCalledTimes(0)
expect(
data2,
'Should use stale cache entry for current request and invalidate it in background',
).toBe(data1)
expect(data3, 'Should use updated cache entry').not.toBe(data2)
})

test<FixtureTestContext>('cacheable route handler is cached on cdn (revalidate=15)', async (ctx) => {
await createFixture('simple', ctx)
await runPlugin(ctx)
Expand Down

0 comments on commit ae4eaa9

Please sign in to comment.