diff --git a/packages/next/src/server/app-render/dynamic-rendering.ts b/packages/next/src/server/app-render/dynamic-rendering.ts index e0755550e3fef..405770c9c104b 100644 --- a/packages/next/src/server/app-render/dynamic-rendering.ts +++ b/packages/next/src/server/app-render/dynamic-rendering.ts @@ -294,7 +294,7 @@ export function abortOnSynchronousPlatformIOAccess( dynamicTracking.syncDynamicErrorWithStack = errorWithStack } } - return abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore) + abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore) } export function trackSynchronousPlatformIOAccessInDev( @@ -321,19 +321,27 @@ export function abortAndThrowOnSynchronousRequestDataAccess( errorWithStack: Error, prerenderStore: PrerenderStoreModern ): never { - const dynamicTracking = prerenderStore.dynamicTracking - if (dynamicTracking) { - if (dynamicTracking.syncDynamicErrorWithStack === null) { - dynamicTracking.syncDynamicExpression = expression - dynamicTracking.syncDynamicErrorWithStack = errorWithStack - if (prerenderStore.validating === true) { - // We always log Request Access in dev at the point of calling the function - // So we mark the dynamic validation as not requiring it to be printed - dynamicTracking.syncDynamicLogged = true + const prerenderSignal = prerenderStore.controller.signal + if (prerenderSignal.aborted === false) { + // TODO it would be better to move this aborted check into the callsite so we can avoid making + // the error object when it isn't relevant to the aborting of the prerender however + // since we need the throw semantics regardless of whether we abort it is easier to land + // this way. See how this was handled with `abortOnSynchronousPlatformIOAccess` for a closer + // to ideal implementation + const dynamicTracking = prerenderStore.dynamicTracking + if (dynamicTracking) { + if (dynamicTracking.syncDynamicErrorWithStack === null) { + dynamicTracking.syncDynamicExpression = expression + dynamicTracking.syncDynamicErrorWithStack = errorWithStack + if (prerenderStore.validating === true) { + // We always log Request Access in dev at the point of calling the function + // So we mark the dynamic validation as not requiring it to be printed + dynamicTracking.syncDynamicLogged = true + } } } + abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore) } - abortOnSynchronousDynamicDataAccess(route, expression, prerenderStore) throw createPrerenderInterruptedError( `Route ${route} needs to bail out of prerendering at this point because it used ${expression}.` ) diff --git a/packages/next/src/server/node-environment-extensions/utils.tsx b/packages/next/src/server/node-environment-extensions/utils.tsx index 20de81ba529ec..ee3a2e0b2cea2 100644 --- a/packages/next/src/server/node-environment-extensions/utils.tsx +++ b/packages/next/src/server/node-environment-extensions/utils.tsx @@ -12,32 +12,37 @@ export function io(expression: string, type: ApiType) { const workUnitStore = workUnitAsyncStorage.getStore() if (workUnitStore) { if (workUnitStore.type === 'prerender') { - const workStore = workAsyncStorage.getStore() - if (workStore) { - let message: string - switch (type) { - case 'time': - 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` - break - case 'random': - 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` - break - case 'crypto': - 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` - break - default: - throw new InvariantError( - 'Unknown expression type in abortOnSynchronousPlatformIOAccess.' - ) - } - const errorWithStack = new Error(message) + const prerenderSignal = workUnitStore.controller.signal + if (prerenderSignal.aborted === false) { + // If the prerender signal is already aborted we don't need to construct any stacks + // because something else actually terminated the prerender. + const workStore = workAsyncStorage.getStore() + if (workStore) { + let message: string + switch (type) { + case 'time': + 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` + break + case 'random': + 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` + break + case 'crypto': + 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` + break + default: + throw new InvariantError( + 'Unknown expression type in abortOnSynchronousPlatformIOAccess.' + ) + } + const errorWithStack = new Error(message) - abortOnSynchronousPlatformIOAccess( - workStore.route, - expression, - errorWithStack, - workUnitStore - ) + abortOnSynchronousPlatformIOAccess( + workStore.route, + expression, + errorWithStack, + workUnitStore + ) + } } } else if ( workUnitStore.type === 'request' &&