1
1
import { existsSync } from 'node:fs'
2
- import { readFile } from 'node:fs/promises'
3
- import { join } from 'node:path'
2
+ import { mkdir , readFile , writeFile } from 'node:fs/promises'
3
+ import { dirname , join } from 'node:path'
4
4
5
- import glob from 'fast-glob'
5
+ import { glob } from 'fast-glob'
6
+ import type { CacheHandlerValue } from 'next/dist/server/lib/incremental-cache/index.js'
7
+ import type { IncrementalCacheValue } from 'next/dist/server/response-cache/types.js'
6
8
7
- import type {
8
- CacheValue ,
9
- FetchCacheValue ,
10
- PageCacheValue ,
11
- PluginContext ,
12
- RouteCacheValue ,
13
- } from '../plugin-context.js'
9
+ import { encodeBlobKey } from '../../shared/blobkey.js'
10
+ import type { PluginContext } from '../plugin-context.js'
11
+
12
+ type CachedPageValue = Extract < IncrementalCacheValue , { kind : 'PAGE' } >
13
+ type CachedRouteValue = Extract < IncrementalCacheValue , { kind : 'ROUTE' } >
14
+ type CachedFetchValue = Extract < IncrementalCacheValue , { kind : 'FETCH' } >
15
+
16
+ /**
17
+ * Write a cache entry to the blob upload directory.
18
+ */
19
+ const writeCacheEntry = async (
20
+ route : string ,
21
+ value : IncrementalCacheValue ,
22
+ lastModified : number ,
23
+ ctx : PluginContext ,
24
+ ) : Promise < void > => {
25
+ const path = join ( ctx . blobDir , await encodeBlobKey ( route ) )
26
+ const entry = JSON . stringify ( {
27
+ lastModified,
28
+ value,
29
+ } satisfies CacheHandlerValue )
30
+
31
+ await mkdir ( dirname ( path ) , { recursive : true } )
32
+ await writeFile ( path , entry , 'utf-8' )
33
+ }
14
34
15
35
/**
16
36
* Normalize routes by stripping leading slashes and ensuring root path is index
17
37
*/
18
38
const routeToFilePath = ( path : string ) => ( path === '/' ? '/index' : path )
19
39
20
- const buildPagesCacheValue = async ( path : string ) : Promise < PageCacheValue > => ( {
40
+ const buildPagesCacheValue = async ( path : string ) : Promise < CachedPageValue > => ( {
21
41
kind : 'PAGE' ,
22
42
html : await readFile ( `${ path } .html` , 'utf-8' ) ,
23
43
pageData : JSON . parse ( await readFile ( `${ path } .json` , 'utf-8' ) ) ,
44
+ postponed : undefined ,
45
+ headers : undefined ,
46
+ status : undefined ,
24
47
} )
25
48
26
- const buildAppCacheValue = async ( path : string ) : Promise < PageCacheValue > => ( {
49
+ const buildAppCacheValue = async ( path : string ) : Promise < CachedPageValue > => ( {
27
50
kind : 'PAGE' ,
28
51
html : await readFile ( `${ path } .html` , 'utf-8' ) ,
29
52
pageData : await readFile ( `${ path } .rsc` , 'utf-8' ) ,
30
53
...JSON . parse ( await readFile ( `${ path } .meta` , 'utf-8' ) ) ,
31
54
} )
32
55
33
- const buildRouteCacheValue = async ( path : string ) : Promise < RouteCacheValue > => ( {
56
+ const buildRouteCacheValue = async ( path : string ) : Promise < CachedRouteValue > => ( {
34
57
kind : 'ROUTE' ,
35
58
body : await readFile ( `${ path } .body` , 'base64' ) ,
36
59
...JSON . parse ( await readFile ( `${ path } .meta` , 'utf-8' ) ) ,
37
60
} )
38
61
39
- const buildFetchCacheValue = async ( path : string ) : Promise < FetchCacheValue > => ( {
62
+ const buildFetchCacheValue = async ( path : string ) : Promise < CachedFetchValue > => ( {
40
63
kind : 'FETCH' ,
41
64
...JSON . parse ( await readFile ( path , 'utf-8' ) ) ,
42
65
} )
@@ -51,45 +74,38 @@ export const copyPrerenderedContent = async (ctx: PluginContext): Promise<void>
51
74
52
75
await Promise . all (
53
76
Object . entries ( manifest . routes ) . map ( async ( [ route , meta ] ) : Promise < void > => {
77
+ const lastModified = meta . initialRevalidateSeconds ? 1 : Date . now ( )
54
78
const key = routeToFilePath ( route )
55
- let value : CacheValue
56
- let path : string
79
+ let value : IncrementalCacheValue
57
80
switch ( true ) {
58
81
// Parallel route default layout has no prerendered page
59
82
case meta . dataRoute ?. endsWith ( '/default.rsc' ) &&
60
83
! existsSync ( join ( ctx . publishDir , 'server/app' , `${ key } .html` ) ) :
61
84
return
62
85
case meta . dataRoute ?. endsWith ( '.json' ) :
63
- path = join ( ctx . publishDir , 'server/pages' , key )
64
- value = await buildPagesCacheValue ( path )
86
+ value = await buildPagesCacheValue ( join ( ctx . publishDir , 'server/pages' , key ) )
65
87
break
66
88
case meta . dataRoute ?. endsWith ( '.rsc' ) :
67
- path = join ( ctx . publishDir , 'server/app' , key )
68
- value = await buildAppCacheValue ( path )
89
+ value = await buildAppCacheValue ( join ( ctx . publishDir , 'server/app' , key ) )
69
90
break
70
91
case meta . dataRoute === null :
71
- path = join ( ctx . publishDir , 'server/app' , key )
72
- value = await buildRouteCacheValue ( path )
92
+ value = await buildRouteCacheValue ( join ( ctx . publishDir , 'server/app' , key ) )
73
93
break
74
94
default :
75
95
throw new Error ( `Unrecognized content: ${ route } ` )
76
96
}
77
97
78
- await ctx . writeCacheEntry (
79
- key ,
80
- value ,
81
- meta . dataRoute === null ? `${ path } .body` : `${ path } .html` ,
82
- )
98
+ await writeCacheEntry ( key , value , lastModified , ctx )
83
99
} ) ,
84
100
)
85
101
86
102
// app router 404 pages are not in the prerender manifest
87
103
// so we need to check for them manually
88
104
if ( existsSync ( join ( ctx . publishDir , `server/app/_not-found.html` ) ) ) {
105
+ const lastModified = Date . now ( )
89
106
const key = '/404'
90
- const path = join ( ctx . publishDir , 'server/app/_not-found' )
91
- const value = await buildAppCacheValue ( path )
92
- await ctx . writeCacheEntry ( key , value , `${ path } .html` )
107
+ const value = await buildAppCacheValue ( join ( ctx . publishDir , 'server/app/_not-found' ) )
108
+ await writeCacheEntry ( key , value , lastModified , ctx )
93
109
}
94
110
} catch ( error ) {
95
111
ctx . failBuild ( 'Failed assembling prerendered content for upload' , error )
@@ -108,9 +124,10 @@ export const copyFetchContent = async (ctx: PluginContext): Promise<void> => {
108
124
109
125
await Promise . all (
110
126
paths . map ( async ( key ) : Promise < void > => {
127
+ const lastModified = 1
111
128
const path = join ( ctx . publishDir , 'cache/fetch-cache' , key )
112
129
const value = await buildFetchCacheValue ( path )
113
- await ctx . writeCacheEntry ( key , value , path )
130
+ await writeCacheEntry ( key , value , lastModified , ctx )
114
131
} ) ,
115
132
)
116
133
} catch ( error ) {
0 commit comments