1
1
import type { NetlifyPluginOptions } from '@netlify/build'
2
2
import glob from 'fast-glob'
3
+ import { Buffer } from 'node:buffer'
3
4
import { existsSync } from 'node:fs'
4
5
import { mkdir , readFile , writeFile } from 'node:fs/promises'
5
6
import { dirname , resolve } from 'node:path'
@@ -39,39 +40,48 @@ type FetchCacheValue = {
39
40
}
40
41
}
41
42
43
+ /**
44
+ * Write a cache entry to the blob upload directory using
45
+ * base64 keys to avoid collisions with directories
46
+ */
42
47
const writeCacheEntry = async ( key : string , value : CacheValue ) => {
43
- await mkdir ( dirname ( resolve ( BLOB_DIR , key ) ) , { recursive : true } )
44
- await writeFile (
45
- resolve ( BLOB_DIR , key ) ,
46
- JSON . stringify ( { lastModified : Date . now ( ) , value } satisfies CacheEntry ) ,
47
- 'utf-8' ,
48
- )
48
+ const path = resolve ( BLOB_DIR , Buffer . from ( key ) . toString ( 'base64' ) )
49
+ const entry = JSON . stringify ( {
50
+ lastModified : Date . now ( ) ,
51
+ value,
52
+ } satisfies CacheEntry )
53
+
54
+ await mkdir ( dirname ( path ) , { recursive : true } )
55
+ await writeFile ( path , entry , 'utf-8' )
49
56
}
50
57
51
- const urlPathToFilePath = ( path : string ) => ( path === '/' ? '/index' : path )
58
+ /**
59
+ * Normalize routes by stripping leading slashes and ensuring root path is index
60
+ */
61
+ const routeToFilePath = ( path : string ) => path . replace ( / ^ \/ / , '' ) || 'index'
52
62
53
63
const buildPagesCacheValue = async ( path : string ) : Promise < PageCacheValue > => ( {
54
64
kind : 'PAGE' ,
55
- html : await readFile ( resolve ( `${ path } .html` ) , 'utf-8' ) ,
56
- pageData : JSON . parse ( await readFile ( resolve ( `${ path } .json` ) , 'utf-8' ) ) ,
65
+ html : await readFile ( `${ path } .html` , 'utf-8' ) ,
66
+ pageData : JSON . parse ( await readFile ( `${ path } .json` , 'utf-8' ) ) ,
57
67
} )
58
68
59
69
const buildAppCacheValue = async ( path : string ) : Promise < PageCacheValue > => ( {
60
70
kind : 'PAGE' ,
61
- html : await readFile ( resolve ( `${ path } .html` ) , 'utf-8' ) ,
62
- pageData : await readFile ( resolve ( `${ path } .rsc` ) , 'utf-8' ) ,
63
- ...JSON . parse ( await readFile ( resolve ( `${ path } .meta` ) , 'utf-8' ) ) ,
71
+ html : await readFile ( `${ path } .html` , 'utf-8' ) ,
72
+ pageData : await readFile ( `${ path } .rsc` , 'utf-8' ) ,
73
+ ...JSON . parse ( await readFile ( `${ path } .meta` , 'utf-8' ) ) ,
64
74
} )
65
75
66
76
const buildRouteCacheValue = async ( path : string ) : Promise < RouteCacheValue > => ( {
67
77
kind : 'ROUTE' ,
68
- body : await readFile ( resolve ( `${ path } .body` ) , 'utf-8' ) ,
69
- ...JSON . parse ( await readFile ( resolve ( `${ path } .meta` ) , 'utf-8' ) ) ,
78
+ body : await readFile ( `${ path } .body` , 'utf-8' ) ,
79
+ ...JSON . parse ( await readFile ( `${ path } .meta` , 'utf-8' ) ) ,
70
80
} )
71
81
72
82
const buildFetchCacheValue = async ( path : string ) : Promise < FetchCacheValue > => ( {
73
83
kind : 'FETCH' ,
74
- ...JSON . parse ( await readFile ( resolve ( path ) , 'utf-8' ) ) ,
84
+ ...JSON . parse ( await readFile ( path , 'utf-8' ) ) ,
75
85
} )
76
86
77
87
/**
@@ -86,31 +96,27 @@ export const copyPrerenderedContent = async ({
86
96
try {
87
97
// read prerendered content and build JSON key/values for the blob store
88
98
const manifest = await getPrerenderManifest ( { PUBLISH_DIR } )
89
- const routes = Object . entries ( manifest . routes )
90
- const notFoundRoute = 'server/app/_not-found'
91
99
92
100
await Promise . all (
93
- routes . map ( async ( [ path , route ] ) => {
94
- let key , value
101
+ Object . entries ( manifest . routes ) . map ( async ( [ route , meta ] ) => {
102
+ const key = routeToFilePath ( route )
103
+ let value : CacheValue
95
104
96
105
switch ( true ) {
97
- case route . dataRoute ?. endsWith ( '.json' ) :
98
- key = `server/pages/${ urlPathToFilePath ( path ) } `
99
- value = await buildPagesCacheValue ( resolve ( PUBLISH_DIR , key ) )
106
+ case meta . dataRoute ?. endsWith ( '.json' ) :
107
+ value = await buildPagesCacheValue ( resolve ( PUBLISH_DIR , 'server/pages' , key ) )
100
108
break
101
109
102
- case route . dataRoute ?. endsWith ( '.rsc' ) :
103
- key = `server/app/${ urlPathToFilePath ( path ) } `
104
- value = await buildAppCacheValue ( resolve ( PUBLISH_DIR , key ) )
110
+ case meta . dataRoute ?. endsWith ( '.rsc' ) :
111
+ value = await buildAppCacheValue ( resolve ( PUBLISH_DIR , 'server/app' , key ) )
105
112
break
106
113
107
- case route . dataRoute === null :
108
- key = `server/app/${ urlPathToFilePath ( path ) } `
109
- value = await buildRouteCacheValue ( resolve ( PUBLISH_DIR , key ) )
114
+ case meta . dataRoute === null :
115
+ value = await buildRouteCacheValue ( resolve ( PUBLISH_DIR , 'server/app' , key ) )
110
116
break
111
117
112
118
default :
113
- throw new Error ( `Unrecognized prerendered content: ${ path } ` )
119
+ throw new Error ( `Unrecognized content: ${ route } ` )
114
120
}
115
121
116
122
await writeCacheEntry ( key , value )
@@ -119,11 +125,10 @@ export const copyPrerenderedContent = async ({
119
125
120
126
// app router 404 pages are not in the prerender manifest
121
127
// so we need to check for them manually
122
- if ( existsSync ( resolve ( PUBLISH_DIR , `${ notFoundRoute } .html` ) ) ) {
123
- await writeCacheEntry (
124
- notFoundRoute ,
125
- await buildAppCacheValue ( resolve ( PUBLISH_DIR , notFoundRoute ) ) ,
126
- )
128
+ if ( existsSync ( resolve ( PUBLISH_DIR , `server/app/_not-found.html` ) ) ) {
129
+ const key = '404'
130
+ const value = await buildAppCacheValue ( resolve ( PUBLISH_DIR , 'server/app/_not-found' ) )
131
+ await writeCacheEntry ( key , value )
127
132
}
128
133
} catch ( error ) {
129
134
failBuild (
@@ -143,14 +148,15 @@ export const copyFetchContent = async ({
143
148
} ,
144
149
} : Pick < NetlifyPluginOptions , 'constants' | 'utils' > ) => {
145
150
try {
146
- const paths = await glob ( [ `cache/fetch-cache/ !(*.*)` ] , {
147
- cwd : resolve ( PUBLISH_DIR ) ,
151
+ const paths = await glob ( [ ' !(*.*)' ] , {
152
+ cwd : resolve ( PUBLISH_DIR , 'cache/fetch-cache' ) ,
148
153
extglob : true ,
149
154
} )
150
155
151
156
await Promise . all (
152
157
paths . map ( async ( key ) => {
153
- await writeCacheEntry ( key , await buildFetchCacheValue ( resolve ( PUBLISH_DIR , key ) ) )
158
+ const value = await buildFetchCacheValue ( resolve ( PUBLISH_DIR , 'cache/fetch-cache' , key ) )
159
+ await writeCacheEntry ( key , value )
154
160
} ) ,
155
161
)
156
162
} catch ( error ) {
0 commit comments