-
Notifications
You must be signed in to change notification settings - Fork 87
/
Copy pathprerendered.ts
263 lines (237 loc) · 9.11 KB
/
prerendered.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
import { existsSync } from 'node:fs'
import { mkdir, readFile, writeFile } from 'node:fs/promises'
import { join } from 'node:path'
import { trace } from '@opentelemetry/api'
import { wrapTracer } from '@opentelemetry/api/experimental'
import { glob } from 'fast-glob'
import pLimit from 'p-limit'
import { satisfies } from 'semver'
import { encodeBlobKey } from '../../shared/blobkey.js'
import type {
CachedFetchValueForMultipleVersions,
NetlifyCachedAppPageValue,
NetlifyCachedPageValue,
NetlifyCachedRouteValue,
NetlifyCacheHandlerValue,
NetlifyIncrementalCacheValue,
} from '../../shared/cache-types.cjs'
import type { PluginContext } from '../plugin-context.js'
import { verifyNetlifyForms } from '../verification.js'
const tracer = wrapTracer(trace.getTracer('Next runtime'))
/**
* Write a cache entry to the blob upload directory.
*/
const writeCacheEntry = async (
route: string,
value: NetlifyIncrementalCacheValue,
lastModified: number,
ctx: PluginContext,
): Promise<void> => {
const path = join(ctx.blobDir, await encodeBlobKey(route))
const entry = JSON.stringify({
lastModified,
value,
} satisfies NetlifyCacheHandlerValue)
await writeFile(path, entry, 'utf-8')
}
/**
* Normalize routes by ensuring leading slashes and ensuring root path is /index
*/
const routeToFilePath = (path: string) => {
if (path === '/') {
return '/index'
}
if (path.startsWith('/')) {
return path
}
return `/${path}`
}
const buildPagesCacheValue = async (
path: string,
initialRevalidateSeconds: number | false | undefined,
shouldUseEnumKind: boolean,
shouldSkipJson = false,
): Promise<NetlifyCachedPageValue> => ({
kind: shouldUseEnumKind ? 'PAGES' : 'PAGE',
html: await readFile(`${path}.html`, 'utf-8'),
pageData: shouldSkipJson ? {} : JSON.parse(await readFile(`${path}.json`, 'utf-8')),
headers: undefined,
status: undefined,
revalidate: initialRevalidateSeconds,
})
const buildAppCacheValue = async (
path: string,
shouldUseAppPageKind: boolean,
): Promise<NetlifyCachedAppPageValue | NetlifyCachedPageValue> => {
const meta = JSON.parse(await readFile(`${path}.meta`, 'utf-8'))
const html = await readFile(`${path}.html`, 'utf-8')
// supporting both old and new cache kind for App Router pages - https://github.com/vercel/next.js/pull/65988
if (shouldUseAppPageKind) {
return {
kind: 'APP_PAGE',
html,
rscData: await readFile(`${path}.rsc`, 'base64').catch(() =>
readFile(`${path}.prefetch.rsc`, 'base64'),
),
...meta,
}
}
const rsc = await readFile(`${path}.rsc`, 'utf-8').catch(() =>
readFile(`${path}.prefetch.rsc`, 'utf-8'),
)
// Next < v14.2.0 does not set meta.status when notFound() is called directly on a page
// Exclude Parallel routes, they are 404s when visited directly
if (
!meta.status &&
rsc.includes('NEXT_NOT_FOUND') &&
!meta.headers['x-next-cache-tags'].includes('/@')
) {
meta.status = 404
}
return {
kind: 'PAGE',
html,
pageData: rsc,
...meta,
}
}
const buildRouteCacheValue = async (
path: string,
initialRevalidateSeconds: number | false,
shouldUseEnumKind: boolean,
): Promise<NetlifyCachedRouteValue> => ({
kind: shouldUseEnumKind ? 'APP_ROUTE' : 'ROUTE',
body: await readFile(`${path}.body`, 'base64'),
...JSON.parse(await readFile(`${path}.meta`, 'utf-8')),
revalidate: initialRevalidateSeconds,
})
const buildFetchCacheValue = async (
path: string,
): Promise<CachedFetchValueForMultipleVersions> => ({
kind: 'FETCH',
...JSON.parse(await readFile(path, 'utf-8')),
})
/**
* Upload prerendered content to the blob store
*/
export const copyPrerenderedContent = async (ctx: PluginContext): Promise<void> => {
return tracer.withActiveSpan('copyPrerenderedContent', async () => {
try {
// ensure the blob directory exists
await mkdir(ctx.blobDir, { recursive: true })
// read prerendered content and build JSON key/values for the blob store
const manifest = await ctx.getPrerenderManifest()
const limitConcurrentPrerenderContentHandling = pLimit(10)
// https://github.com/vercel/next.js/pull/65988 introduced Cache kind specific to pages in App Router (`APP_PAGE`).
// Before this change there was common kind for both Pages router and App router pages
// so we check Next.js version to decide how to generate cache values for App Router pages.
// Note: at time of writing this code, released 15@rc uses old kind for App Router pages, while [email protected] and newer canaries use new kind.
// Looking at 15@rc release branch it was merging `canary` branch in, so the version constraint assumes that future 15@rc (and 15@latest) versions
// will use new kind for App Router pages.
const shouldUseAppPageKind = ctx.nextVersion
? satisfies(ctx.nextVersion, '>=15.0.0-canary.13 <15.0.0-d || >15.0.0-rc.0', {
includePrerelease: true,
})
: false
// https://github.com/vercel/next.js/pull/68602 changed the cache kind for Pages router pages from `PAGE` to `PAGES` and from `ROUTE` to `APP_ROUTE`.
const shouldUseEnumKind = ctx.nextVersion
? satisfies(ctx.nextVersion, '>=15.0.0-canary.114 <15.0.0-d || >15.0.0-rc.0', {
includePrerelease: true,
})
: false
await Promise.all([
...Object.entries(manifest.routes).map(
([route, meta]): Promise<void> =>
limitConcurrentPrerenderContentHandling(async () => {
const lastModified = meta.initialRevalidateSeconds
? Date.now() - 31536000000
: Date.now()
const key = routeToFilePath(route)
let value: NetlifyIncrementalCacheValue
switch (true) {
// Parallel route default layout has no prerendered page
case meta.dataRoute?.endsWith('/default.rsc') &&
!existsSync(join(ctx.publishDir, 'server/app', `${key}.html`)):
return
case meta.dataRoute?.endsWith('.json'):
if (manifest.notFoundRoutes.includes(route)) {
// if pages router returns 'notFound: true', build won't produce html and json files
return
}
value = await buildPagesCacheValue(
join(ctx.publishDir, 'server/pages', key),
meta.initialRevalidateSeconds,
shouldUseEnumKind,
)
break
case meta.dataRoute?.endsWith('.rsc'):
value = await buildAppCacheValue(
join(ctx.publishDir, 'server/app', key),
shouldUseAppPageKind,
)
break
case meta.dataRoute === null:
value = await buildRouteCacheValue(
join(ctx.publishDir, 'server/app', key),
meta.initialRevalidateSeconds,
shouldUseEnumKind,
)
break
default:
throw new Error(`Unrecognized content: ${route}`)
}
// Netlify Forms are not support and require a workaround
if (value.kind === 'PAGE' || value.kind === 'PAGES' || value.kind === 'APP_PAGE') {
verifyNetlifyForms(ctx, value.html)
}
await writeCacheEntry(key, value, lastModified, ctx)
}),
),
...ctx.getFallbacks(manifest).map(async (route) => {
const key = routeToFilePath(route)
const value = await buildPagesCacheValue(
join(ctx.publishDir, 'server/pages', key),
undefined,
shouldUseEnumKind,
true, // there is no corresponding json file for fallback, so we are skipping it for this entry
)
await writeCacheEntry(key, value, Date.now(), ctx)
}),
])
// app router 404 pages are not in the prerender manifest
// so we need to check for them manually
if (existsSync(join(ctx.publishDir, `server/app/_not-found.html`))) {
const lastModified = Date.now()
const key = '/404'
const value = await buildAppCacheValue(
join(ctx.publishDir, 'server/app/_not-found'),
shouldUseAppPageKind,
)
await writeCacheEntry(key, value, lastModified, ctx)
}
} catch (error) {
ctx.failBuild('Failed assembling prerendered content for upload', error)
}
})
}
/**
* Upload fetch content to the blob store
*/
export const copyFetchContent = async (ctx: PluginContext): Promise<void> => {
try {
const paths = await glob(['!(*.*)'], {
cwd: join(ctx.publishDir, 'cache/fetch-cache'),
extglob: true,
})
await Promise.all(
paths.map(async (key): Promise<void> => {
const lastModified = Date.now() - 31536000000
const path = join(ctx.publishDir, 'cache/fetch-cache', key)
const value = await buildFetchCacheValue(path)
await writeCacheEntry(key, value, lastModified, ctx)
}),
)
} catch (error) {
ctx.failBuild('Failed assembling fetch content for upload', error)
}
}