diff --git a/packages/next/src/client/components/router-reducer/fetch-server-response.ts b/packages/next/src/client/components/router-reducer/fetch-server-response.ts index 2af83aa8a203c4..a9251096c20d19 100644 --- a/packages/next/src/client/components/router-reducer/fetch-server-response.ts +++ b/packages/next/src/client/components/router-reducer/fetch-server-response.ts @@ -27,8 +27,9 @@ import { } from '../app-router-headers' import { urlToUrlWithoutFlightMarker } from '../app-router' import { callServer } from '../../app-call-server' -import { PrefetchKind } from './router-reducer-types' +import { PrefetchKind, type PrefetchCacheEntry } from './router-reducer-types' import { hexHash } from '../../../shared/lib/hash' +import { prefixExistingPrefetchCacheEntry } from './prefetch-cache-utils' export type FetchServerResponseResult = [ flightData: FlightData, @@ -49,6 +50,7 @@ export async function fetchServerResponse( flightRouterState: FlightRouterState, nextUrl: string | null, currentBuildId: string, + prefetchCache?: Map, prefetchKind?: PrefetchKind ): Promise { const headers: { @@ -116,6 +118,16 @@ export async function fetchServerResponse( const interception = !!res.headers.get('vary')?.includes(NEXT_URL) let isFlightResponse = contentType === RSC_CONTENT_TYPE_HEADER + if (prefetchCache && interception) { + // If this fetch response corresponds with a prefetch to an interception route, we want to move it to + // a prefixed cache key to avoid clobbering an existing entry. + prefixExistingPrefetchCacheEntry({ + url, + nextUrl, + prefetchCache, + }) + } + if (process.env.NODE_ENV === 'production') { if (process.env.__NEXT_CONFIG_OUTPUT === 'export') { if (!isFlightResponse) { diff --git a/packages/next/src/client/components/router-reducer/prefetch-cache-utils.ts b/packages/next/src/client/components/router-reducer/prefetch-cache-utils.ts index d91aaca4a9d7b2..b87d029e81f8e3 100644 --- a/packages/next/src/client/components/router-reducer/prefetch-cache-utils.ts +++ b/packages/next/src/client/components/router-reducer/prefetch-cache-utils.ts @@ -1,11 +1,12 @@ +import { createHrefFromUrl } from './create-href-from-url' +import { fetchServerResponse } from './fetch-server-response' import { PrefetchCacheEntryStatus, - type AppRouterState, type PrefetchCacheEntry, + PrefetchKind, + type ReadonlyReducerState, } from './router-reducer-types' -import { addPathPrefix } from '../../../shared/lib/router/utils/add-path-prefix' -import { pathHasPrefix } from '../../../shared/lib/router/utils/path-has-prefix' -import { createHrefFromUrl } from './create-href-from-url' +import { prefetchQueue } from './reducers/prefetch-reducer' /** * Creates a cache key for the router prefetch cache @@ -14,27 +15,171 @@ import { createHrefFromUrl } from './create-href-from-url' * @param nextUrl - an internal URL, primarily used for handling rewrites. Defaults to '/'. * @return The generated prefetch cache key. */ -export function createPrefetchCacheKey(url: URL, nextUrl: string | null) { +function createPrefetchCacheKey(url: URL, nextUrl?: string | null) { const pathnameFromUrl = createHrefFromUrl( url, // Ensures the hash is not part of the cache key as it does not impact the server fetch false ) - // delimit the prefix so we don't conflict with other pages - const nextUrlPrefix = `${nextUrl}%` - - // Route interception depends on `nextUrl` values which aren't a 1:1 mapping to a URL - // The cache key that we store needs to use `nextUrl` to properly distinguish cache entries - if (nextUrl && !pathHasPrefix(pathnameFromUrl, nextUrl)) { - return addPathPrefix(pathnameFromUrl, nextUrlPrefix) + // nextUrl is used as a cache key delimiter since entries can vary based on the Next-URL header + if (nextUrl) { + return `${nextUrl}%${pathnameFromUrl}` } return pathnameFromUrl } +export function prefixExistingPrefetchCacheEntry({ + url, + nextUrl, + prefetchCache, +}: Pick & { + url: URL +}) { + const existingCacheKey = createPrefetchCacheKey(url) + const existingCacheEntry = prefetchCache.get(existingCacheKey) + if (!existingCacheEntry) { + // no-op -- there wasn't an entry to move + return + } + + const newCacheKey = createPrefetchCacheKey(url, nextUrl) + prefetchCache.set(newCacheKey, existingCacheEntry) + prefetchCache.delete(existingCacheKey) +} + +type GetOrCreatePrefetchCacheEntryParams = Pick< + ReadonlyReducerState, + 'nextUrl' | 'prefetchCache' | 'tree' | 'buildId' +> & { + url: URL + createIfNotFound?: true +} + +type GetPrefetchCacheEntryParams = Pick< + ReadonlyReducerState, + 'nextUrl' | 'prefetchCache' +> & { + tree?: ReadonlyReducerState['tree'] + buildId?: ReadonlyReducerState['buildId'] + url: URL + createIfNotFound?: false +} + +/** + * Returns a prefetch cache entry if one exists. Optionally creates a new one. + */ +export function getPrefetchCacheEntry({ + url, + nextUrl, + tree, + buildId, + prefetchCache, + createIfNotFound, +}: GetOrCreatePrefetchCacheEntryParams): PrefetchCacheEntry +export function getPrefetchCacheEntry({ + url, + nextUrl, + prefetchCache, + createIfNotFound, +}: GetPrefetchCacheEntryParams): PrefetchCacheEntry | undefined +export function getPrefetchCacheEntry({ + url, + nextUrl, + tree, + buildId, + prefetchCache, + createIfNotFound, +}: GetOrCreatePrefetchCacheEntryParams | GetPrefetchCacheEntryParams): + | PrefetchCacheEntry + | undefined { + let existingCacheEntry: PrefetchCacheEntry | undefined = undefined + // We first check if there's a more specific interception route prefetch entry + // This is because when we detect a prefetch that corresponds with an interception route, we prefix it with nextUrl (see `createPrefetchCacheKey`) + // to avoid conflicts with other pages that may have the same URL but render different things depending on the `Next-URL` header. + const interceptionCacheKey = createPrefetchCacheKey(url, nextUrl) + const interceptionData = prefetchCache.get(interceptionCacheKey) + + if (interceptionData) { + existingCacheEntry = interceptionData + } else { + // If we dont find a more specific interception route prefetch entry, we check for a regular prefetch entry + const prefetchCacheKey = createPrefetchCacheKey(url) + const prefetchData = prefetchCache.get(prefetchCacheKey) + if (prefetchData) { + existingCacheEntry = prefetchData + } + } + + // We found an entry, so we can return it + if (existingCacheEntry) { + // Grab the latest status of the cache entry and update it + existingCacheEntry.status = getPrefetchEntryCacheStatus(existingCacheEntry) + return existingCacheEntry + } + + // When retrieving a prefetch entry, we usually want to create one if it doesn't exist + // This let's us create a new one if it doesn't exist to avoid needing typeguards in the calling code + if (createIfNotFound) { + // If we don't have a prefetch value, we need to create one + return createLazyPrefetchEntry({ + tree, + url, + buildId, + nextUrl, + prefetchCache, + // in dev, there's never gonna be a prefetch entry so we want to prefetch here + kind: + process.env.NODE_ENV === 'development' + ? PrefetchKind.AUTO + : PrefetchKind.TEMPORARY, + }) + } +} + +/** + * Creates a prefetch entry for data that has not been resolved. This will add the prefetch request to a promise queue. + */ +export function createLazyPrefetchEntry({ + url, + kind, + tree, + nextUrl, + buildId, + prefetchCache, +}: Pick< + ReadonlyReducerState, + 'nextUrl' | 'tree' | 'buildId' | 'prefetchCache' +> & { + url: URL + kind: PrefetchKind +}): PrefetchCacheEntry { + const prefetchCacheKey = createPrefetchCacheKey(url) + + // initiates the fetch request for the prefetch and attaches a listener + // to the promise to update the prefetch cache entry when the promise resolves (if necessary) + const data = prefetchQueue.enqueue(() => + fetchServerResponse(url, tree, nextUrl, buildId, prefetchCache, kind) + ) + + const prefetchEntry = { + treeAtTimeOfPrefetch: tree, + data, + kind, + prefetchTime: Date.now(), + lastUsedTime: null, + key: prefetchCacheKey, + status: PrefetchCacheEntryStatus.fresh, + } + + prefetchCache.set(prefetchCacheKey, prefetchEntry) + + return prefetchEntry +} + export function prunePrefetchCache( - prefetchCache: AppRouterState['prefetchCache'] + prefetchCache: ReadonlyReducerState['prefetchCache'] ) { for (const [href, prefetchCacheEntry] of prefetchCache) { if ( @@ -49,7 +194,7 @@ export function prunePrefetchCache( const FIVE_MINUTES = 5 * 60 * 1000 const THIRTY_SECONDS = 30 * 1000 -export function getPrefetchEntryCacheStatus({ +function getPrefetchEntryCacheStatus({ kind, prefetchTime, lastUsedTime, diff --git a/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.test.tsx b/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.test.tsx index b7cf15ce625f4c..2f145df1a8c58a 100644 --- a/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.test.tsx +++ b/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.test.tsx @@ -260,6 +260,7 @@ describe('navigateReducer', () => { "kind": "temporary", "lastUsedTime": 1690329600000, "prefetchTime": 1690329600000, + "status": "fresh", "treeAtTimeOfPrefetch": [ "", { @@ -455,6 +456,7 @@ describe('navigateReducer', () => { "kind": "temporary", "lastUsedTime": 1690329600000, "prefetchTime": 1690329600000, + "status": "fresh", "treeAtTimeOfPrefetch": [ "", { @@ -893,6 +895,7 @@ describe('navigateReducer', () => { "kind": "temporary", "lastUsedTime": 1690329600000, "prefetchTime": 1690329600000, + "status": "fresh", "treeAtTimeOfPrefetch": [ "", { @@ -1120,6 +1123,7 @@ describe('navigateReducer', () => { "kind": "auto", "lastUsedTime": 1690329600000, "prefetchTime": 1690329600000, + "status": "fresh", "treeAtTimeOfPrefetch": [ "", { @@ -1375,6 +1379,7 @@ describe('navigateReducer', () => { "kind": "temporary", "lastUsedTime": 1690329600000, "prefetchTime": 1690329600000, + "status": "fresh", "treeAtTimeOfPrefetch": [ "", { @@ -1719,6 +1724,7 @@ describe('navigateReducer', () => { "kind": "temporary", "lastUsedTime": 1690329600000, "prefetchTime": 1690329600000, + "status": "fresh", "treeAtTimeOfPrefetch": [ "", { diff --git a/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.ts b/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.ts index 001fa3aad2ece7..d480298f4235e0 100644 --- a/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.ts +++ b/packages/next/src/client/components/router-reducer/reducers/navigate-reducer.ts @@ -17,7 +17,7 @@ import type { ReadonlyReducerState, ReducerState, } from '../router-reducer-types' -import { PrefetchKind, PrefetchCacheEntryStatus } from '../router-reducer-types' +import { PrefetchCacheEntryStatus } from '../router-reducer-types' import { handleMutable } from '../handle-mutable' import { applyFlightData } from '../apply-flight-data' import { prefetchQueue } from './prefetch-reducer' @@ -28,12 +28,9 @@ import { updateCacheNodeOnNavigation, } from '../ppr-navigations' import { - createPrefetchCacheKey, getPrefetchCacheEntry, - getPrefetchEntryCacheStatus, prunePrefetchCache, - createPrefetchCacheEntry, -} from './prefetch-cache-utils' +} from '../prefetch-cache-utils' export function handleExternalUrl( state: ReadonlyReducerState, @@ -128,27 +125,20 @@ function navigateReducer_noPPR( return handleExternalUrl(state, mutable, url.toString(), pendingPush) } - let prefetchValues = getPrefetchCacheEntry(url, state) - // If we don't have a prefetch value, we need to create one - if (!prefetchValues) { - const cacheKey = createPrefetchCacheKey(url) - prefetchValues = createPrefetchCacheEntry({ - state, - url, - // in dev, there's never gonna be a prefetch entry so we want to prefetch here - kind: - process.env.NODE_ENV === 'development' - ? PrefetchKind.AUTO - : PrefetchKind.TEMPORARY, - prefetchCacheKey: cacheKey, - }) - - state.prefetchCache.set(cacheKey, prefetchValues) - } - - const prefetchEntryCacheStatus = getPrefetchEntryCacheStatus(prefetchValues) + const prefetchValues = getPrefetchCacheEntry({ + url, + nextUrl: state.nextUrl, + tree: state.tree, + buildId: state.buildId, + prefetchCache: state.prefetchCache, + createIfNotFound: true, + }) + const { + treeAtTimeOfPrefetch, + data, + status: prefetchEntryCacheStatus, + } = prefetchValues - const { treeAtTimeOfPrefetch, data } = prefetchValues prefetchQueue.bump(data) return data.then( @@ -305,27 +295,20 @@ function navigateReducer_PPR( return handleExternalUrl(state, mutable, url.toString(), pendingPush) } - let prefetchValues = getPrefetchCacheEntry(url, state) - // If we don't have a prefetch value, we need to create one - if (!prefetchValues) { - const cacheKey = createPrefetchCacheKey(url) - prefetchValues = createPrefetchCacheEntry({ - state, - url, - // in dev, there's never gonna be a prefetch entry so we want to prefetch here - kind: - process.env.NODE_ENV === 'development' - ? PrefetchKind.AUTO - : PrefetchKind.TEMPORARY, - prefetchCacheKey: cacheKey, - }) - - state.prefetchCache.set(cacheKey, prefetchValues) - } - - const prefetchEntryCacheStatus = getPrefetchEntryCacheStatus(prefetchValues) + const prefetchValues = getPrefetchCacheEntry({ + url, + nextUrl: state.nextUrl, + tree: state.tree, + buildId: state.buildId, + prefetchCache: state.prefetchCache, + createIfNotFound: true, + }) + const { + treeAtTimeOfPrefetch, + data, + status: prefetchEntryCacheStatus, + } = prefetchValues - const { treeAtTimeOfPrefetch, data } = prefetchValues prefetchQueue.bump(data) return data.then( diff --git a/packages/next/src/client/components/router-reducer/reducers/prefetch-cache-utils.ts b/packages/next/src/client/components/router-reducer/reducers/prefetch-cache-utils.ts deleted file mode 100644 index e96432f4a53423..00000000000000 --- a/packages/next/src/client/components/router-reducer/reducers/prefetch-cache-utils.ts +++ /dev/null @@ -1,151 +0,0 @@ -import { createHrefFromUrl } from '../create-href-from-url' -import { fetchServerResponse } from '../fetch-server-response' -import type { - PrefetchCacheEntry, - PrefetchKind, - ReadonlyReducerState, -} from '../router-reducer-types' -import { prefetchQueue } from './prefetch-reducer' - -/** - * Creates a cache key for the router prefetch cache - * - * @param url - The URL being navigated to - * @param nextUrl - an internal URL, primarily used for handling rewrites. Defaults to '/'. - * @return The generated prefetch cache key. - */ -export function createPrefetchCacheKey(url: URL, nextUrl?: string | null) { - const pathnameFromUrl = createHrefFromUrl( - url, - // Ensures the hash is not part of the cache key as it does not impact the server fetch - false - ) - - // delimit the prefix so we don't conflict with other pages - - // Route interception depends on `nextUrl` values which aren't a 1:1 mapping to a URL - // The cache key that we store needs to use `nextUrl` to properly distinguish cache entries - if (nextUrl) { - return `${nextUrl}%${pathnameFromUrl}` - } - - return pathnameFromUrl -} - -export function getPrefetchCacheEntry( - url: URL, - state: ReadonlyReducerState -): PrefetchCacheEntry | undefined { - // We first check if there's a more specific interception route prefetch entry - // This is because when we detect a prefetch that corresponds with an interception route, we prefix it with nextUrl (see `createPrefetchCacheKey`) - // to avoid conflicts with other pages that may have the same URL but render different things depending on the `Next-URL` header. - const interceptionCacheKey = createPrefetchCacheKey(url, state.nextUrl) - const interceptionData = state.prefetchCache.get(interceptionCacheKey) - - if (interceptionData) { - return interceptionData - } - - // If we dont find a more specific interception route prefetch entry, we check for a regular prefetch entry - const prefetchCacheKey = createPrefetchCacheKey(url) - return state.prefetchCache.get(prefetchCacheKey) -} - -export function createPrefetchCacheEntry({ - state, - url, - kind, - prefetchCacheKey, -}: { - state: ReadonlyReducerState - url: URL - kind: PrefetchKind - prefetchCacheKey: string -}): PrefetchCacheEntry { - // initiates the fetch request for the prefetch and attaches a listener - // to the promise to update the prefetch cache entry when the promise resolves (if necessary) - const getPrefetchData = () => - fetchServerResponse( - url, - state.tree, - state.nextUrl, - state.buildId, - kind - ).then((prefetchResponse) => { - /* [flightData, canonicalUrlOverride, postpone, intercept] */ - const [, , , intercept] = prefetchResponse - const existingPrefetchEntry = state.prefetchCache.get(prefetchCacheKey) - // If we discover that the prefetch corresponds with an interception route, we want to move it to - // a prefixed cache key to avoid clobbering an existing entry. - if (intercept && existingPrefetchEntry) { - const prefixedCacheKey = createPrefetchCacheKey(url, state.nextUrl) - state.prefetchCache.set(prefixedCacheKey, existingPrefetchEntry) - state.prefetchCache.delete(prefetchCacheKey) - } - - return prefetchResponse - }) - - const data = prefetchQueue.enqueue(getPrefetchData) - - return { - treeAtTimeOfPrefetch: state.tree, - data, - kind, - prefetchTime: Date.now(), - lastUsedTime: null, - key: prefetchCacheKey, - } -} - -export function prunePrefetchCache( - prefetchCache: ReadonlyReducerState['prefetchCache'] -) { - for (const [href, prefetchCacheEntry] of prefetchCache) { - if ( - getPrefetchEntryCacheStatus(prefetchCacheEntry) === - PrefetchCacheEntryStatus.expired - ) { - prefetchCache.delete(href) - } - } -} - -const FIVE_MINUTES = 5 * 60 * 1000 -const THIRTY_SECONDS = 30 * 1000 - -export enum PrefetchCacheEntryStatus { - fresh = 'fresh', - reusable = 'reusable', - expired = 'expired', - stale = 'stale', -} - -export function getPrefetchEntryCacheStatus({ - kind, - prefetchTime, - lastUsedTime, -}: PrefetchCacheEntry): PrefetchCacheEntryStatus { - // if the cache entry was prefetched or read less than 30s ago, then we want to re-use it - if (Date.now() < (lastUsedTime ?? prefetchTime) + THIRTY_SECONDS) { - return lastUsedTime - ? PrefetchCacheEntryStatus.reusable - : PrefetchCacheEntryStatus.fresh - } - - // if the cache entry was prefetched less than 5 mins ago, then we want to re-use only the loading state - if (kind === 'auto') { - if (Date.now() < prefetchTime + FIVE_MINUTES) { - return PrefetchCacheEntryStatus.stale - } - } - - // if the cache entry was prefetched less than 5 mins ago and was a "full" prefetch, then we want to re-use it "full - if (kind === 'full') { - if (Date.now() < prefetchTime + FIVE_MINUTES) { - return PrefetchCacheEntryStatus.reusable - } - } - - return PrefetchCacheEntryStatus.expired -} diff --git a/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.test.tsx b/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.test.tsx index 88efb764f6b3ad..cf45bd6375dae6 100644 --- a/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.test.tsx +++ b/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.test.tsx @@ -4,7 +4,11 @@ import type { FlightData } from '../../../../server/app-render/types' import type { FlightRouterState } from '../../../../server/app-render/types' import type { CacheNode } from '../../../../shared/lib/app-router-context.shared-runtime' import { createInitialRouterState } from '../create-initial-router-state' -import { ACTION_PREFETCH, PrefetchKind } from '../router-reducer-types' +import { + ACTION_PREFETCH, + PrefetchCacheEntryStatus, + PrefetchKind, +} from '../router-reducer-types' import type { PrefetchAction } from '../router-reducer-types' import { prefetchReducer } from './prefetch-reducer' import { fetchServerResponse } from '../fetch-server-response' @@ -127,6 +131,8 @@ describe('prefetchReducer', () => { url, initialTree, null, + state.buildId, + state.prefetchCache, PrefetchKind.AUTO ) const action: PrefetchAction = { @@ -153,6 +159,7 @@ describe('prefetchReducer', () => { kind: PrefetchKind.AUTO, lastUsedTime: null, prefetchTime: expect.any(Number), + status: PrefetchCacheEntryStatus.fresh, treeAtTimeOfPrefetch: [ '', { @@ -282,6 +289,7 @@ describe('prefetchReducer', () => { initialTree, null, state.buildId, + state.prefetchCache, PrefetchKind.AUTO ) const action: PrefetchAction = { @@ -310,6 +318,7 @@ describe('prefetchReducer', () => { prefetchTime: expect.any(Number), kind: PrefetchKind.AUTO, lastUsedTime: null, + status: PrefetchCacheEntryStatus.fresh, treeAtTimeOfPrefetch: [ '', { diff --git a/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.ts b/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.ts index 851a648538303a..90d34a0803d0f6 100644 --- a/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.ts +++ b/packages/next/src/client/components/router-reducer/reducers/prefetch-reducer.ts @@ -7,11 +7,10 @@ import { PrefetchKind } from '../router-reducer-types' import { NEXT_RSC_UNION_QUERY } from '../../app-router-headers' import { PromiseQueue } from '../../promise-queue' import { - createPrefetchCacheKey, - createPrefetchCacheEntry, + createLazyPrefetchEntry, getPrefetchCacheEntry, prunePrefetchCache, -} from './prefetch-cache-utils' +} from '../prefetch-cache-utils' export const prefetchQueue = new PromiseQueue(5) @@ -25,7 +24,11 @@ export function prefetchReducer( const { url } = action url.searchParams.delete(NEXT_RSC_UNION_QUERY) - const cacheEntry = getPrefetchCacheEntry(url, state) + const cacheEntry = getPrefetchCacheEntry({ + url, + nextUrl: state.nextUrl, + prefetchCache: state.prefetchCache, + }) if (cacheEntry) { /** @@ -53,15 +56,14 @@ export function prefetchReducer( } } - const prefetchCacheKey = createPrefetchCacheKey(url) - const newEntry = createPrefetchCacheEntry({ - state, + createLazyPrefetchEntry({ url, + tree: state.tree, + buildId: state.buildId, + prefetchCache: state.prefetchCache, + nextUrl: state.nextUrl, kind: action.kind, - prefetchCacheKey, }) - state.prefetchCache.set(prefetchCacheKey, newEntry) - return state } diff --git a/packages/next/src/client/components/router-reducer/reducers/server-patch-reducer.test.tsx b/packages/next/src/client/components/router-reducer/reducers/server-patch-reducer.test.tsx index 3ad5b14e2b146a..76a6a8708fe674 100644 --- a/packages/next/src/client/components/router-reducer/reducers/server-patch-reducer.test.tsx +++ b/packages/next/src/client/components/router-reducer/reducers/server-patch-reducer.test.tsx @@ -448,6 +448,7 @@ describe('serverPatchReducer', () => { "kind": "temporary", "lastUsedTime": 1690329600000, "prefetchTime": 1690329600000, + "status": "fresh", "treeAtTimeOfPrefetch": [ "", { diff --git a/packages/next/src/client/components/router-reducer/router-reducer-types.ts b/packages/next/src/client/components/router-reducer/router-reducer-types.ts index bd6c90f11f6d27..9268aca1fa0f60 100644 --- a/packages/next/src/client/components/router-reducer/router-reducer-types.ts +++ b/packages/next/src/client/components/router-reducer/router-reducer-types.ts @@ -203,6 +203,7 @@ export type PrefetchCacheEntry = { prefetchTime: number lastUsedTime: number | null key: string + status: PrefetchCacheEntryStatus } export enum PrefetchCacheEntryStatus { diff --git a/packages/next/src/server/base-server.ts b/packages/next/src/server/base-server.ts index 1ddbbabfda7ea8..13d3a0f3abadb9 100644 --- a/packages/next/src/server/base-server.ts +++ b/packages/next/src/server/base-server.ts @@ -2032,26 +2032,6 @@ export default abstract class Server { } if (isAppPath) { - res.setHeader('vary', RSC_VARY_HEADER) - - const couldBeIntercepted = this.interceptionRouteRewrites?.some( - (rewrite) => { - return new RegExp(rewrite.regex).test(resolvedUrlPathname) - } - ) - - // Interception route responses can vary based on the `Next-URL` header as they're rewritten to different components. - // This means that multiple route interception responses can resolve to the same URL. We use the Vary header to signal this - // behavior to the client so that it can properly cache the response. - // If the request that we're handling is one that could have a different response based on the `Next-URL` header, or if - // we're handling an interception route, then we include `Next-URL` in the Vary header. - if ( - couldBeIntercepted || - isInterceptionRouteAppPath(resolvedUrlPathname) - ) { - res.setHeader('vary', `${RSC_VARY_HEADER}, ${NEXT_URL}`) - } - if (!this.renderOpts.dev && !isPreviewMode && isSSG && isRSCRequest) { // If this is an RSC request but we aren't in minimal mode, then we mark // that this is a data request so that we can generate the flight data