Skip to content

Commit f4c1493

Browse files
authored
Cleanup Render Result (#58782)
This improves some of the typings around the `RenderResult` returned during renders. Previously it had a single large metadata object that was shared across both the pages and app render pipelines. To add more type safety, this splits the types used by each of the render pipelines into their own types while still allowing the default `RenderResult` to reference metadata from either render pipeline. This also improved the flight data generation for app renders. Previously, the promise was inlined, and errors were swallowed. With the advent of improvements in #58779 the postpone errors are no longer returned by the flight generation and are instead handled correctly inside the render by React so it can emit properly postponed flight data. Besides that there was some whitespace changes, so hiding whitespace differences during review should make it much clearer to review!
1 parent 8d1c619 commit f4c1493

11 files changed

+271
-189
lines changed

packages/next/src/build/utils.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -1127,10 +1127,16 @@ export async function buildStaticPaths({
11271127
}
11281128
}
11291129

1130+
export type AppConfigDynamic =
1131+
| 'auto'
1132+
| 'error'
1133+
| 'force-static'
1134+
| 'force-dynamic'
1135+
11301136
export type AppConfig = {
11311137
revalidate?: number | false
11321138
dynamicParams?: true | false
1133-
dynamic?: 'auto' | 'error' | 'force-static' | 'force-dynamic'
1139+
dynamic?: AppConfigDynamic
11341140
fetchCache?: 'force-cache' | 'only-cache'
11351141
preferredRegion?: string
11361142
}

packages/next/src/client/components/static-generation-async-storage.external.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@ import type { AsyncLocalStorage } from 'async_hooks'
22
import type { IncrementalCache } from '../../server/lib/incremental-cache'
33
import type { DynamicServerError } from './hooks-server-context'
44
import type { FetchMetrics } from '../../server/base-http'
5+
import type { Revalidate } from '../../server/lib/revalidate'
6+
57
import { createAsyncLocalStorage } from './async-local-storage'
68

79
export interface StaticGenerationStore {
@@ -23,7 +25,7 @@ export interface StaticGenerationStore {
2325
| 'default-no-store'
2426
| 'only-no-store'
2527

26-
revalidate?: false | number
28+
revalidate?: Revalidate
2729
forceStatic?: boolean
2830
dynamicShouldError?: boolean
2931
pendingRevalidates?: Record<string, Promise<any>>

packages/next/src/client/components/static-generation-bailout.ts

+7-5
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
import type { AppConfigDynamic } from '../../build/utils'
2+
13
import { DynamicServerError } from './hooks-server-context'
24
import { maybePostpone } from './maybe-postpone'
35
import { staticGenerationAsyncStorage } from './static-generation-async-storage.external'
@@ -6,7 +8,7 @@ class StaticGenBailoutError extends Error {
68
code = 'NEXT_STATIC_GEN_BAILOUT'
79
}
810

9-
type BailoutOpts = { dynamic?: string; link?: string }
11+
type BailoutOpts = { dynamic?: AppConfigDynamic; link?: string }
1012

1113
export type StaticGenerationBailout = (
1214
reason: string,
@@ -23,7 +25,7 @@ function formatErrorMessage(reason: string, opts?: BailoutOpts) {
2325

2426
export const staticGenerationBailout: StaticGenerationBailout = (
2527
reason,
26-
opts
28+
{ dynamic, link } = {}
2729
) => {
2830
const staticGenerationStore = staticGenerationAsyncStorage.getStore()
2931
if (!staticGenerationStore) return false
@@ -34,12 +36,12 @@ export const staticGenerationBailout: StaticGenerationBailout = (
3436

3537
if (staticGenerationStore.dynamicShouldError) {
3638
throw new StaticGenBailoutError(
37-
formatErrorMessage(reason, { ...opts, dynamic: opts?.dynamic ?? 'error' })
39+
formatErrorMessage(reason, { link, dynamic: dynamic ?? 'error' })
3840
)
3941
}
4042

4143
const message = formatErrorMessage(reason, {
42-
...opts,
44+
dynamic,
4345
// this error should be caught by Next to bail out of static generation
4446
// in case it's uncaught, this link provides some additional context as to why
4547
link: 'https://nextjs.org/docs/messages/dynamic-server-error',
@@ -51,7 +53,7 @@ export const staticGenerationBailout: StaticGenerationBailout = (
5153
// to 0.
5254
staticGenerationStore.revalidate = 0
5355

54-
if (!opts?.dynamic) {
56+
if (!dynamic) {
5557
// we can statically prefetch pages that opt into dynamic,
5658
// but not things like headers/cookies
5759
staticGenerationStore.staticPrefetchBailout = true

packages/next/src/export/routes/app-page.ts

+10-6
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ import { lazyRenderAppPage } from '../../server/future/route-modules/app-page/mo
2525
export const enum ExportedAppPageFiles {
2626
HTML = 'HTML',
2727
FLIGHT = 'FLIGHT',
28+
PREFETCH_FLIGHT = 'PREFETCH_FLIGHT',
2829
META = 'META',
2930
POSTPONED = 'POSTPONED',
3031
}
@@ -128,10 +129,8 @@ export async function exportAppPage(
128129

129130
const html = result.toUnchunkedString()
130131

131-
const {
132-
metadata: { pageData, revalidate = false, postponed, fetchTags },
133-
} = result
134132
const { metadata } = result
133+
const { flightData, revalidate = false, postponed, fetchTags } = metadata
135134

136135
// Ensure we don't postpone without having PPR enabled.
137136
if (postponed && !renderOpts.experimental.ppr) {
@@ -169,23 +168,28 @@ export async function exportAppPage(
169168

170169
return { revalidate: 0 }
171170
}
171+
// If page data isn't available, it means that the page couldn't be rendered
172+
// properly.
173+
else if (!flightData) {
174+
throw new Error(`Invariant: failed to get page data for ${path}`)
175+
}
172176
// If PPR is enabled, we want to emit a prefetch rsc file for the page
173177
// instead of the standard rsc. This is because the standard rsc will
174178
// contain the dynamic data.
175179
else if (renderOpts.experimental.ppr) {
176180
// If PPR is enabled, we should emit the flight data as the prefetch
177181
// payload.
178182
await fileWriter(
179-
ExportedAppPageFiles.FLIGHT,
183+
ExportedAppPageFiles.PREFETCH_FLIGHT,
180184
htmlFilepath.replace(/\.html$/, RSC_PREFETCH_SUFFIX),
181-
pageData
185+
flightData
182186
)
183187
} else {
184188
// Writing the RSC payload to a file if we don't have PPR enabled.
185189
await fileWriter(
186190
ExportedAppPageFiles.FLIGHT,
187191
htmlFilepath.replace(/\.html$/, RSC_SUFFIX),
188-
pageData
192+
flightData
189193
)
190194
}
191195

packages/next/src/server/app-render/action-handler.ts

+3-2
Original file line numberDiff line numberDiff line change
@@ -210,7 +210,8 @@ async function createRedirectRenderResult(
210210
console.error(`failed to get redirect response`, err)
211211
}
212212
}
213-
return new RenderResult(JSON.stringify({}))
213+
214+
return RenderResult.fromStatic('{}')
214215
}
215216

216217
// Used to compare Host header and Origin header.
@@ -607,7 +608,7 @@ To configure the body size limit for Server Actions, see: https://nextjs.org/doc
607608
res.statusCode = 303
608609
return {
609610
type: 'done',
610-
result: new RenderResult(''),
611+
result: RenderResult.fromStatic(''),
611612
}
612613
} else if (isNotFoundError(err)) {
613614
res.statusCode = 404

0 commit comments

Comments
 (0)