@@ -4,9 +4,21 @@ import { cp } from 'node:fs/promises'
4
4
import { createRequire } from 'node:module'
5
5
import { join } from 'node:path'
6
6
import { gunzipSync } from 'node:zlib'
7
+ import { HttpResponse , http , passthrough } from 'msw'
8
+ import { setupServer } from 'msw/node'
7
9
import { gt , prerelease } from 'semver'
8
10
import { v4 } from 'uuid'
9
- import { Mock , afterAll , beforeAll , beforeEach , describe , expect , test , vi } from 'vitest'
11
+ import {
12
+ Mock ,
13
+ afterAll ,
14
+ afterEach ,
15
+ beforeAll ,
16
+ beforeEach ,
17
+ describe ,
18
+ expect ,
19
+ test ,
20
+ vi ,
21
+ } from 'vitest'
10
22
import { getPatchesToApply } from '../../src/build/content/server.js'
11
23
import { type FixtureTestContext } from '../utils/contexts.js'
12
24
import {
@@ -36,9 +48,32 @@ vi.mock('node:fs/promises', async (importOriginal) => {
36
48
}
37
49
} )
38
50
51
+ let server : ReturnType < typeof setupServer >
52
+
39
53
// Disable the verbose logging of the lambda-local runtime
40
54
getLogger ( ) . level = 'alert'
41
55
56
+ const purgeAPI = vi . fn ( )
57
+
58
+ beforeAll ( ( ) => {
59
+ server = setupServer (
60
+ http . post ( 'https://api.netlify.com/api/v1/purge' , async ( { request } ) => {
61
+ purgeAPI ( await request . json ( ) )
62
+
63
+ return HttpResponse . json ( {
64
+ ok : true ,
65
+ } )
66
+ } ) ,
67
+ http . all ( / .* / , ( ) => passthrough ( ) ) ,
68
+ )
69
+ server . listen ( )
70
+ } )
71
+
72
+ afterAll ( ( ) => {
73
+ // Disable API mocking after the tests are done.
74
+ server . close ( )
75
+ } )
76
+
42
77
beforeEach < FixtureTestContext > ( async ( ctx ) => {
43
78
// set for each test a new deployID and siteID
44
79
ctx . deployID = generateRandomObjectID ( )
@@ -48,9 +83,15 @@ beforeEach<FixtureTestContext>(async (ctx) => {
48
83
// hide debug logs in tests
49
84
vi . spyOn ( console , 'debug' ) . mockImplementation ( ( ) => { } )
50
85
86
+ purgeAPI . mockClear ( )
87
+
51
88
await startMockBlobStore ( ctx )
52
89
} )
53
90
91
+ afterEach ( ( ) => {
92
+ vi . unstubAllEnvs ( )
93
+ } )
94
+
54
95
test < FixtureTestContext > ( 'Test that the simple next app is working' , async ( ctx ) => {
55
96
await createFixture ( 'simple' , ctx )
56
97
await runPlugin ( ctx )
@@ -210,6 +251,39 @@ test<FixtureTestContext>('cacheable route handler is cached on cdn (revalidate=f
210
251
)
211
252
} )
212
253
254
+ test < FixtureTestContext > ( 'purge API is not used when unstable_cache cache entry gets stale' , async ( ctx ) => {
255
+ await createFixture ( 'simple' , ctx )
256
+ await runPlugin ( ctx )
257
+
258
+ // set the NETLIFY_PURGE_API_TOKEN to get pass token check and allow fetch call to be made
259
+ vi . stubEnv ( 'NETLIFY_PURGE_API_TOKEN' , 'mock' )
260
+
261
+ const page1 = await invokeFunction ( ctx , {
262
+ url : '/unstable_cache' ,
263
+ } )
264
+ const data1 = load ( page1 . body ) ( 'pre' ) . text ( )
265
+
266
+ // allow for cache entry to get stale
267
+ await new Promise ( ( res ) => setTimeout ( res , 2000 ) )
268
+
269
+ const page2 = await invokeFunction ( ctx , {
270
+ url : '/unstable_cache' ,
271
+ } )
272
+ const data2 = load ( page2 . body ) ( 'pre' ) . text ( )
273
+
274
+ const page3 = await invokeFunction ( ctx , {
275
+ url : '/unstable_cache' ,
276
+ } )
277
+ const data3 = load ( page3 . body ) ( 'pre' ) . text ( )
278
+
279
+ expect ( purgeAPI , 'Purge API should not be hit' ) . toHaveBeenCalledTimes ( 0 )
280
+ expect (
281
+ data2 ,
282
+ 'Should use stale cache entry for current request and invalidate it in background' ,
283
+ ) . toBe ( data1 )
284
+ expect ( data3 , 'Should use updated cache entry' ) . not . toBe ( data2 )
285
+ } )
286
+
213
287
test < FixtureTestContext > ( 'cacheable route handler is cached on cdn (revalidate=15)' , async ( ctx ) => {
214
288
await createFixture ( 'simple' , ctx )
215
289
await runPlugin ( ctx )
0 commit comments