Skip to content

Commit 5dd95e3

Browse files
gnoffunstubbable
andauthored
[dynamicIO] only abort once per prerender (#77747)
Prior to this change multiple sync dynamic APIs could trigger multiple aborts. Generally this isn't a problem however it is wasteful compute and blocks a later refactor where we stop explicitly tracking whether the sync dynamic access is what caused the prerender to abort or not. To achieve better perf and future semantics we now only abort if the prerender is not yet aborted. In effect this means that there can only be one sync abort error per prerender in RSC and SSR respectively. --------- Co-authored-by: Hendrik Liebau <[email protected]>
1 parent 940de49 commit 5dd95e3

File tree

2 files changed

+49
-36
lines changed

2 files changed

+49
-36
lines changed

packages/next/src/server/app-render/dynamic-rendering.ts

+19-11
Original file line numberDiff line numberDiff line change
@@ -294,7 +294,7 @@ export function abortOnSynchronousPlatformIOAccess(
294294
dynamicTracking.syncDynamicErrorWithStack = errorWithStack
295295
}
296296
}
297-
return abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore)
297+
abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore)
298298
}
299299

300300
export function trackSynchronousPlatformIOAccessInDev(
@@ -321,19 +321,27 @@ export function abortAndThrowOnSynchronousRequestDataAccess(
321321
errorWithStack: Error,
322322
prerenderStore: PrerenderStoreModern
323323
): never {
324-
const dynamicTracking = prerenderStore.dynamicTracking
325-
if (dynamicTracking) {
326-
if (dynamicTracking.syncDynamicErrorWithStack === null) {
327-
dynamicTracking.syncDynamicExpression = expression
328-
dynamicTracking.syncDynamicErrorWithStack = errorWithStack
329-
if (prerenderStore.validating === true) {
330-
// We always log Request Access in dev at the point of calling the function
331-
// So we mark the dynamic validation as not requiring it to be printed
332-
dynamicTracking.syncDynamicLogged = true
324+
const prerenderSignal = prerenderStore.controller.signal
325+
if (prerenderSignal.aborted === false) {
326+
// TODO it would be better to move this aborted check into the callsite so we can avoid making
327+
// the error object when it isn't relevant to the aborting of the prerender however
328+
// since we need the throw semantics regardless of whether we abort it is easier to land
329+
// this way. See how this was handled with `abortOnSynchronousPlatformIOAccess` for a closer
330+
// to ideal implementation
331+
const dynamicTracking = prerenderStore.dynamicTracking
332+
if (dynamicTracking) {
333+
if (dynamicTracking.syncDynamicErrorWithStack === null) {
334+
dynamicTracking.syncDynamicExpression = expression
335+
dynamicTracking.syncDynamicErrorWithStack = errorWithStack
336+
if (prerenderStore.validating === true) {
337+
// We always log Request Access in dev at the point of calling the function
338+
// So we mark the dynamic validation as not requiring it to be printed
339+
dynamicTracking.syncDynamicLogged = true
340+
}
333341
}
334342
}
343+
abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore)
335344
}
336-
abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore)
337345
throw createPrerenderInterruptedError(
338346
`Route ${route} needs to bail out of prerendering at this point because it used ${expression}.`
339347
)

packages/next/src/server/node-environment-extensions/utils.tsx

+30-25
Original file line numberDiff line numberDiff line change
@@ -12,32 +12,37 @@ export function io(expression: string, type: ApiType) {
1212
const workUnitStore = workUnitAsyncStorage.getStore()
1313
if (workUnitStore) {
1414
if (workUnitStore.type === 'prerender') {
15-
const workStore = workAsyncStorage.getStore()
16-
if (workStore) {
17-
let message: string
18-
switch (type) {
19-
case 'time':
20-
message = `Route "${workStore.route}" used ${expression} instead of using \`performance\` or without explicitly calling \`await connection()\` beforehand. See more info here: https://nextjs.org/docs/messages/next-prerender-current-time`
21-
break
22-
case 'random':
23-
message = `Route "${workStore.route}" used ${expression} outside of \`"use cache"\` and without explicitly calling \`await connection()\` beforehand. See more info here: https://nextjs.org/docs/messages/next-prerender-random`
24-
break
25-
case 'crypto':
26-
message = `Route "${workStore.route}" used ${expression} outside of \`"use cache"\` and without explicitly calling \`await connection()\` beforehand. See more info here: https://nextjs.org/docs/messages/next-prerender-crypto`
27-
break
28-
default:
29-
throw new InvariantError(
30-
'Unknown expression type in abortOnSynchronousPlatformIOAccess.'
31-
)
32-
}
33-
const errorWithStack = new Error(message)
15+
const prerenderSignal = workUnitStore.controller.signal
16+
if (prerenderSignal.aborted === false) {
17+
// If the prerender signal is already aborted we don't need to construct any stacks
18+
// because something else actually terminated the prerender.
19+
const workStore = workAsyncStorage.getStore()
20+
if (workStore) {
21+
let message: string
22+
switch (type) {
23+
case 'time':
24+
message = `Route "${workStore.route}" used ${expression} instead of using \`performance\` or without explicitly calling \`await connection()\` beforehand. See more info here: https://nextjs.org/docs/messages/next-prerender-current-time`
25+
break
26+
case 'random':
27+
message = `Route "${workStore.route}" used ${expression} outside of \`"use cache"\` and without explicitly calling \`await connection()\` beforehand. See more info here: https://nextjs.org/docs/messages/next-prerender-random`
28+
break
29+
case 'crypto':
30+
message = `Route "${workStore.route}" used ${expression} outside of \`"use cache"\` and without explicitly calling \`await connection()\` beforehand. See more info here: https://nextjs.org/docs/messages/next-prerender-crypto`
31+
break
32+
default:
33+
throw new InvariantError(
34+
'Unknown expression type in abortOnSynchronousPlatformIOAccess.'
35+
)
36+
}
37+
const errorWithStack = new Error(message)
3438

35-
abortOnSynchronousPlatformIOAccess(
36-
workStore.route,
37-
expression,
38-
errorWithStack,
39-
workUnitStore
40-
)
39+
abortOnSynchronousPlatformIOAccess(
40+
workStore.route,
41+
expression,
42+
errorWithStack,
43+
workUnitStore
44+
)
45+
}
4146
}
4247
} else if (
4348
workUnitStore.type === 'request' &&

0 commit comments

Comments
 (0)