From b40963bf9539ec7ebc9759202c07e10428fa476c Mon Sep 17 00:00:00 2001 From: Andrew Clark Date: Wed, 4 Dec 2024 22:26:13 -0500 Subject: [PATCH] [Segment Cache] Skip dynamic request if possible During a navigation, if all the data has been prefetched, and the target route does not contain any dynamic data, then we should skip a request to the server. This uses the `isPartial` field I added in the previous PRs to track whether the prefetched data is complete or not. --- .../router-reducer/ppr-navigations.ts | 210 ++++++++++++++---- .../reducers/navigate-reducer.ts | 25 ++- .../components/segment-cache/navigation.ts | 90 +++++--- .../basic/app/fully-static/page.tsx | 18 ++ .../app/fully-static/target-page/page.tsx | 3 + .../basic/segment-cache-basic.test.ts | 28 +++ 6 files changed, 292 insertions(+), 82 deletions(-) create mode 100644 test/e2e/app-dir/segment-cache/basic/app/fully-static/page.tsx create mode 100644 test/e2e/app-dir/segment-cache/basic/app/fully-static/target-page/page.tsx diff --git a/packages/next/src/client/components/router-reducer/ppr-navigations.ts b/packages/next/src/client/components/router-reducer/ppr-navigations.ts index 52ec6b8653698..68c48d0ed88db 100644 --- a/packages/next/src/client/components/router-reducer/ppr-navigations.ts +++ b/packages/next/src/client/components/router-reducer/ppr-navigations.ts @@ -20,13 +20,17 @@ import type { FetchServerResponseResult } from './fetch-server-response' // request. We can't use the Cache Node tree or Route State tree directly // because those include reused nodes, too. This tree is discarded as soon as // the navigation response is received. -type Task = { +export type Task = { // The router state that corresponds to the tree that this Task represents. route: FlightRouterState - // This is usually non-null. It represents a brand new Cache Node tree whose - // data is still pending. If it's null, it means there's no pending data but - // the client patched the router state. + // The CacheNode that corresponds to the tree that this Task represents. If + // `children` is null (i.e. if this is a terminal task node), then `node` + // represents a brand new Cache Node tree, which way or may not need to be + // filled with dynamic data from the server. node: CacheNode | null + // Whether anything in this tree contains dynamic holes that need to be filled + // by the server. + needsDynamicRequest: boolean children: Map | null } @@ -64,7 +68,8 @@ export function updateCacheNodeOnNavigation( oldRouterState: FlightRouterState, newRouterState: FlightRouterState, prefetchData: CacheNodeSeedData | null, - prefetchHead: React.ReactNode | null + prefetchHead: React.ReactNode | null, + isPrefetchHeadPartial: boolean ): Task | null { // Diff the old and new trees to reuse the shared layouts. const oldRouterStateChildren = oldRouterState[1] @@ -96,14 +101,15 @@ export function updateCacheNodeOnNavigation( } = {} let taskChildren = null - // For most navigations, we need to issue a "dynamic" request to fetch the - // full RSC data from the server since during rendering, we'll only serve - // the prefetch shell. For some navigations, we re-use the existing cache node - // (via `spawnReusedTask`), and don't actually need fresh data from the server. - // In those cases, we use this `needsDynamicRequest` flag to return a `null` - // cache node, which signals to the caller that we don't need to issue a - // dynamic request. We start off with a `false` value, and then for each parallel - // route, we set it to `true` if we encounter a segment that needs a dynamic request. + // Most navigations require a request to fetch additional data from the + // server, either because the data was not already prefetched, or because the + // target route contains dynamic data that cannot be prefetched. + // + // However, if the target route is fully static, and it's already completely + // loaded into the segment cache, then we can skip the server request. + // + // This starts off as `false`, and is set to `true` if any of the child + // routes requires a dynamic request. let needsDynamicRequest = false for (let parallelRouteKey in newRouterStateChildren) { @@ -144,10 +150,11 @@ export function updateCacheNodeOnNavigation( taskChild = spawnReusedTask(oldRouterStateChild) } else { // There's no currently active segment. Switch to the "create" path. - taskChild = spawnPendingTask( + taskChild = createCacheNodeOnNavigation( newRouterStateChild, prefetchDataChild !== undefined ? prefetchDataChild : null, - prefetchHead + prefetchHead, + isPrefetchHeadPartial ) } } else if ( @@ -165,24 +172,27 @@ export function updateCacheNodeOnNavigation( oldRouterStateChild, newRouterStateChild, prefetchDataChild, - prefetchHead + prefetchHead, + isPrefetchHeadPartial ) } else { // Either there's no existing Cache Node for this segment, or this // segment doesn't exist in the old Router State tree. Switch to the // "create" path. - taskChild = spawnPendingTask( + taskChild = createCacheNodeOnNavigation( newRouterStateChild, prefetchDataChild !== undefined ? prefetchDataChild : null, - prefetchHead + prefetchHead, + isPrefetchHeadPartial ) } } else { // This is a new tree. Switch to the "create" path. - taskChild = spawnPendingTask( + taskChild = createCacheNodeOnNavigation( newRouterStateChild, prefetchDataChild !== undefined ? prefetchDataChild : null, - prefetchHead + prefetchHead, + isPrefetchHeadPartial ) } @@ -197,8 +207,9 @@ export function updateCacheNodeOnNavigation( const newSegmentMapChild: ChildSegmentMap = new Map(oldSegmentMapChild) newSegmentMapChild.set(newSegmentKeyChild, newCacheNodeChild) prefetchParallelRoutes.set(parallelRouteKey, newSegmentMapChild) - // a non-null taskChild.node means we're waiting for a dynamic request to - // fill in the missing data + } + + if (taskChild.needsDynamicRequest) { needsDynamicRequest = true } @@ -241,9 +252,110 @@ export function updateCacheNodeOnNavigation( newRouterState, patchedRouterStateChildren ), - // Only return the new cache node if there are pending tasks that need to be resolved - // by the dynamic data from the server. If they don't, we don't need to trigger a dynamic request. - node: needsDynamicRequest ? newCacheNode : null, + node: newCacheNode, + needsDynamicRequest, + children: taskChildren, + } +} + +function createCacheNodeOnNavigation( + routerState: FlightRouterState, + prefetchData: CacheNodeSeedData | null, + possiblyPartialPrefetchHead: React.ReactNode | null, + isPrefetchHeadPartial: boolean +): Task { + // Same traversal as updateCacheNodeNavigation, but we switch to this path + // once we reach the part of the tree that was not in the previous route. We + // don't need to diff against the old tree, we just need to create a new one. + if (prefetchData === null) { + // There's no prefetch for this segment. Everything from this point will be + // requested from the server, even if there are static children below it. + // Create a terminal task node that will later be fulfilled by + // server response. + return spawnPendingTask( + routerState, + null, + possiblyPartialPrefetchHead, + isPrefetchHeadPartial + ) + } + + const routerStateChildren = routerState[1] + const isPrefetchRscPartial = prefetchData[4] + + // The head is assigned to every leaf segment delivered by the server. Based + // on corresponding logic in fill-lazy-items-till-leaf-with-head.ts + const isLeafSegment = Object.keys(routerStateChildren).length === 0 + + // If prefetch data is available for a segment, and it's fully static (i.e. + // does not contain any dynamic holes), we don't need to request it from + // the server. + if ( + // Check if the segment data is partial + isPrefetchRscPartial || + // Check if the head is partial (only relevant if this is a leaf segment) + (isPrefetchHeadPartial && isLeafSegment) + ) { + // We only have partial data from this segment. Like missing segments, we + // must request the full data from the server. + return spawnPendingTask( + routerState, + prefetchData, + possiblyPartialPrefetchHead, + isPrefetchHeadPartial + ) + } + + // The prefetched segment is fully static, so we don't need to request a new + // one from the server. Keep traversing down the tree until we reach something + // that requires a dynamic request. + const prefetchDataChildren = prefetchData[2] + const taskChildren = new Map() + const cacheNodeChildren = new Map() + let needsDynamicRequest = false + for (let parallelRouteKey in routerStateChildren) { + const routerStateChild: FlightRouterState = + routerStateChildren[parallelRouteKey] + const prefetchDataChild: CacheNodeSeedData | void | null = + prefetchDataChildren !== null + ? prefetchDataChildren[parallelRouteKey] + : null + const segmentChild = routerStateChild[0] + const segmentKeyChild = createRouterCacheKey(segmentChild) + const taskChild = createCacheNodeOnNavigation( + routerStateChild, + prefetchDataChild, + possiblyPartialPrefetchHead, + isPrefetchHeadPartial + ) + taskChildren.set(parallelRouteKey, taskChild) + if (taskChild.needsDynamicRequest) { + needsDynamicRequest = true + } + const newCacheNodeChild = taskChild.node + if (newCacheNodeChild !== null) { + const newSegmentMapChild: ChildSegmentMap = new Map() + newSegmentMapChild.set(segmentKeyChild, newCacheNodeChild) + cacheNodeChildren.set(parallelRouteKey, newSegmentMapChild) + } + } + + const rsc = prefetchData[1] + const loading = prefetchData[3] + return { + route: routerState, + node: { + lazyData: null, + // Since this is a fully static segment, we don't need to use the + // `prefetchRsc` field. + rsc, + prefetchRsc: null, + head: isLeafSegment ? possiblyPartialPrefetchHead : null, + prefetchHead: null, + loading, + parallelRoutes: cacheNodeChildren, + }, + needsDynamicRequest, children: taskChildren, } } @@ -271,19 +383,26 @@ function patchRouterStateWithNewChildren( function spawnPendingTask( routerState: FlightRouterState, prefetchData: CacheNodeSeedData | null, - prefetchHead: React.ReactNode | null + prefetchHead: React.ReactNode | null, + isPrefetchHeadPartial: boolean ): Task { // Create a task that will later be fulfilled by data from the server. - const pendingCacheNode = createPendingCacheNode( - routerState, - prefetchData, - prefetchHead - ) - return { + const newTask: Task = { route: routerState, - node: pendingCacheNode, + + // Corresponds to the part of the route that will be rendered on the server. + node: createPendingCacheNode( + routerState, + prefetchData, + prefetchHead, + isPrefetchHeadPartial + ), + // Set this to true to indicate that this tree is missing data. This will + // be propagated to all the parent tasks. + needsDynamicRequest: true, children: null, } + return newTask } function spawnReusedTask(reusedRouterState: FlightRouterState): Task { @@ -292,6 +411,7 @@ function spawnReusedTask(reusedRouterState: FlightRouterState): Task { return { route: reusedRouterState, node: null, + needsDynamicRequest: false, children: null, } } @@ -413,6 +533,11 @@ function finishTaskUsingDynamicDataPayload( dynamicData: CacheNodeSeedData, dynamicHead: React.ReactNode ) { + if (!task.needsDynamicRequest) { + // Everything in this subtree is already complete. Bail out. + return + } + // dynamicData may represent a larger subtree than the task. Before we can // finish the task, we need to line them up. const taskChildren = task.children @@ -429,8 +554,8 @@ function finishTaskUsingDynamicDataPayload( dynamicData, dynamicHead ) - // Null this out to indicate that the task is complete. - task.node = null + // Set this to false to indicate that this task is now complete. + task.needsDynamicRequest = false } return } @@ -472,7 +597,8 @@ function finishTaskUsingDynamicDataPayload( function createPendingCacheNode( routerState: FlightRouterState, prefetchData: CacheNodeSeedData | null, - prefetchHead: React.ReactNode | null + prefetchHead: React.ReactNode | null, + isPrefetchHeadPartial: boolean ): ReadyCacheNode { const routerStateChildren = routerState[1] const prefetchDataChildren = prefetchData !== null ? prefetchData[2] : null @@ -492,7 +618,8 @@ function createPendingCacheNode( const newCacheNodeChild = createPendingCacheNode( routerStateChild, prefetchDataChild === undefined ? null : prefetchDataChild, - prefetchHead + prefetchHead, + isPrefetchHeadPartial ) const newSegmentMapChild: ChildSegmentMap = new Map() @@ -503,7 +630,6 @@ function createPendingCacheNode( // The head is assigned to every leaf segment delivered by the server. Based // on corresponding logic in fill-lazy-items-till-leaf-with-head.ts const isLeafSegment = parallelRoutes.size === 0 - const maybePrefetchRsc = prefetchData !== null ? prefetchData[1] : null const maybePrefetchLoading = prefetchData !== null ? prefetchData[3] : null return { @@ -512,6 +638,10 @@ function createPendingCacheNode( prefetchRsc: maybePrefetchRsc !== undefined ? maybePrefetchRsc : null, prefetchHead: isLeafSegment ? prefetchHead : null, + + // TODO: Technically, a loading boundary could contain dynamic data. We must + // have separate `loading` and `prefetchLoading` fields to handle this, like + // we do for the segment data and head. loading: maybePrefetchLoading !== undefined ? maybePrefetchLoading : null, // Create a deferred promise. This will be fulfilled once the dynamic @@ -645,8 +775,8 @@ export function abortTask(task: Task, error: any): void { } } - // Null this out to indicate that the task is complete. - task.node = null + // Set this to false to indicate that this task is now complete. + task.needsDynamicRequest = false } function abortPendingCacheNode( 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 0ee5246873cab..9921442edc6a4 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 @@ -266,6 +266,7 @@ export function navigateReducer( pathToSegment: flightSegmentPath, seedData, head, + isHeadPartial, isRootRender, } = normalizedFlightData let treePatch = normalizedFlightData.tree @@ -316,13 +317,11 @@ export function navigateReducer( currentTree, treePatch, seedData, - head + head, + isHeadPartial ) if (task !== null) { - // We've created a new Cache Node tree that contains a prefetched - // version of the next page. This can be rendered instantly. - // Use the tree computed by updateCacheNodeOnNavigation instead // of the one computed by applyRouterStatePatchToTree. // TODO: We should remove applyRouterStatePatchToTree @@ -330,12 +329,13 @@ export function navigateReducer( const patchedRouterState: FlightRouterState = task.route newTree = patchedRouterState - // It's possible that `updateCacheNodeOnNavigation` only spawned tasks to reuse the existing cache, - // in which case `task.node` will be null, signaling we don't need to wait for a dynamic request - // and can simply apply the patched `FlightRouterState`. - if (task.node !== null) { - const newCache = task.node - + const newCache = task.node + if (newCache !== null) { + // We've created a new Cache Node tree that contains a prefetched + // version of the next page. This can be rendered instantly. + mutable.cache = newCache + } + if (task.needsDynamicRequest) { // The prefetched tree has dynamic holes in it. We initiate a // dynamic request to fill them in. // @@ -359,8 +359,9 @@ export function navigateReducer( // because we're not going to await the dynamic request here. Since we're not blocking // on the dynamic request, `layout-router` will // task.node.lazyData = dynamicRequest - - mutable.cache = newCache + } else { + // The prefetched tree does not contain dynamic holes — it's + // fully static. We can skip the dynamic request. } } else { // Nothing changed, so reuse the old cache. diff --git a/packages/next/src/client/components/segment-cache/navigation.ts b/packages/next/src/client/components/segment-cache/navigation.ts index 39c2b2021df70..2f00f1d1767e7 100644 --- a/packages/next/src/client/components/segment-cache/navigation.ts +++ b/packages/next/src/client/components/segment-cache/navigation.ts @@ -8,13 +8,11 @@ import type { LoadingModuleData, } from '../../../shared/lib/app-router-context.shared-runtime' import type { NormalizedFlightData } from '../../flight-data-helpers' -import { - fetchServerResponse, - type FetchServerResponseResult, -} from '../router-reducer/fetch-server-response' +import { fetchServerResponse } from '../router-reducer/fetch-server-response' import { updateCacheNodeOnNavigation, listenForDynamicRequest, + type Task as PPRNavigationTask, } from '../router-reducer/ppr-navigations' import { createHrefFromUrl as createCanonicalUrl } from '../router-reducer/create-href-from-url' import { @@ -94,19 +92,18 @@ export function navigate( const prefetchFlightRouterState = snapshot.flightRouterState const prefetchSeedData = snapshot.seedData const prefetchHead = route.head + const isPrefetchHeadPartial = route.isHeadPartial const canonicalUrl = route.canonicalUrl - const promiseForDynamicServerResponse = fetchServerResponse(url, { - flightRouterState: currentFlightRouterState, - nextUrl, - }) return navigateUsingPrefetchedRouteTree( + url, + nextUrl, currentCacheNode, currentFlightRouterState, prefetchFlightRouterState, prefetchSeedData, prefetchHead, - canonicalUrl, - promiseForDynamicServerResponse + isPrefetchHeadPartial, + canonicalUrl ) } // There's no matching prefetch for this route in the cache. @@ -114,21 +111,23 @@ export function navigate( tag: NavigationResultTag.Async, data: navigateDynamicallyWithNoPrefetch( url, + nextUrl, currentCacheNode, - currentFlightRouterState, - nextUrl + currentFlightRouterState ), } } function navigateUsingPrefetchedRouteTree( + url: URL, + nextUrl: string | null, currentCacheNode: CacheNode, currentFlightRouterState: FlightRouterState, prefetchFlightRouterState: FlightRouterState, prefetchSeedData: CacheNodeSeedData | null, prefetchHead: React.ReactNode | null, - canonicalUrl: string, - promiseForDynamicServerResponse: Promise + isPrefetchHeadPartial: boolean, + canonicalUrl: string ): SuccessfulNavigationResult | NoOpNavigationResult { // Recursively construct a prefetch tree by reading from the Segment Cache. To // maintain compatibility, we output the same data structures as the old @@ -141,26 +140,42 @@ function navigateUsingPrefetchedRouteTree( currentFlightRouterState, prefetchFlightRouterState, prefetchSeedData, - prefetchHead + prefetchHead, + isPrefetchHeadPartial ) if (task !== null) { - const newCacheNode = task.node - if (newCacheNode !== null) { + if (task.needsDynamicRequest) { + const promiseForDynamicServerResponse = fetchServerResponse(url, { + flightRouterState: currentFlightRouterState, + nextUrl, + }) listenForDynamicRequest(task, promiseForDynamicServerResponse) + } else { + // The prefetched tree does not contain dynamic holes — it's + // fully static. We can skip the dynamic request. } - return { - tag: NavigationResultTag.Success, - data: { - flightRouterState: task.route, - cacheNode: newCacheNode !== null ? newCacheNode : currentCacheNode, - canonicalUrl, - }, - } + return navigationTaskToResult(task, currentCacheNode, canonicalUrl) } // The server sent back an empty tree patch. There's nothing to update. return noOpNavigationResult } +function navigationTaskToResult( + task: PPRNavigationTask, + currentCacheNode: CacheNode, + canonicalUrl: string +): SuccessfulNavigationResult { + const newCacheNode = task.node + return { + tag: NavigationResultTag.Success, + data: { + flightRouterState: task.route, + cacheNode: newCacheNode !== null ? newCacheNode : currentCacheNode, + canonicalUrl, + }, + } +} + function readRenderSnapshotFromCache( now: number, tree: TreePrefetch @@ -243,9 +258,9 @@ function readRenderSnapshotFromCache( async function navigateDynamicallyWithNoPrefetch( url: URL, + nextUrl: string | null, currentCacheNode: CacheNode, - currentFlightRouterState: FlightRouterState, - nextUrl: string | null + currentFlightRouterState: FlightRouterState ): Promise< MPANavigationResult | SuccessfulNavigationResult | NoOpNavigationResult > { @@ -292,17 +307,32 @@ async function navigateDynamicallyWithNoPrefetch( // nor a prefetch head. const prefetchSeedData = null const prefetchHead = null + const isPrefetchHeadPartial = true + + const canonicalUrl = createCanonicalUrl( + canonicalUrlOverride ? canonicalUrlOverride : url + ) // Now we proceed exactly as we would for normal navigation. - return navigateUsingPrefetchedRouteTree( + const task = updateCacheNodeOnNavigation( currentCacheNode, currentFlightRouterState, prefetchFlightRouterState, prefetchSeedData, prefetchHead, - createCanonicalUrl(canonicalUrlOverride ? canonicalUrlOverride : url), - promiseForDynamicServerResponse + isPrefetchHeadPartial ) + if (task !== null) { + if (task.needsDynamicRequest) { + listenForDynamicRequest(task, promiseForDynamicServerResponse) + } else { + // The prefetched tree does not contain dynamic holes — it's + // fully static. We can skip the dynamic request. + } + return navigationTaskToResult(task, currentCacheNode, canonicalUrl) + } + // The server sent back an empty tree patch. There's nothing to update. + return noOpNavigationResult } function simulatePrefetchTreeUsingDynamicTreePatch( diff --git a/test/e2e/app-dir/segment-cache/basic/app/fully-static/page.tsx b/test/e2e/app-dir/segment-cache/basic/app/fully-static/page.tsx new file mode 100644 index 0000000000000..feb16d95c7031 --- /dev/null +++ b/test/e2e/app-dir/segment-cache/basic/app/fully-static/page.tsx @@ -0,0 +1,18 @@ +import Link from 'next/link' + +export default function FullyStaticStart() { + return ( + <> +

+ Demonstrates that when navigating to a fully prefetched route that does + not contain any dynamic data, we do not need to perform an additional + request on navigation. +

+
    +
  • + Target +
  • +
+ + ) +} diff --git a/test/e2e/app-dir/segment-cache/basic/app/fully-static/target-page/page.tsx b/test/e2e/app-dir/segment-cache/basic/app/fully-static/target-page/page.tsx new file mode 100644 index 0000000000000..dafd08c46fa0b --- /dev/null +++ b/test/e2e/app-dir/segment-cache/basic/app/fully-static/target-page/page.tsx @@ -0,0 +1,3 @@ +export default function Target() { + return
Target
+} diff --git a/test/e2e/app-dir/segment-cache/basic/segment-cache-basic.test.ts b/test/e2e/app-dir/segment-cache/basic/segment-cache-basic.test.ts index d44d1e2f424e6..623a6155f3b64 100644 --- a/test/e2e/app-dir/segment-cache/basic/segment-cache-basic.test.ts +++ b/test/e2e/app-dir/segment-cache/basic/segment-cache-basic.test.ts @@ -104,6 +104,33 @@ describe('segment cache (basic tests)', () => { navigationsLock.release() }) + + it('skips dynamic request if prefetched data is fully static', async () => { + const interceptor = createRequestInterceptor() + const browser = await next.browser('/fully-static', { + beforePageLoad(page: Page) { + page.route('**/*', async (route: Route) => { + await interceptor.interceptRoute(route) + }) + }, + }) + + // Rendering the link triggers a prefetch of the test page. + const link = await browser.elementByCss( + 'a[href="/fully-static/target-page"]' + ) + const navigationsLock = interceptor.lockNavigations() + await link.click() + + // The page should render immediately because it was prefetched. + const div = await browser.elementById('target-page') + expect(await div.innerHTML()).toBe('Target') + + // We should have skipped the navigation request because all the data was + // fully static. + const numberOfNavigationRequests = (await navigationsLock.release()).size + expect(numberOfNavigationRequests).toBe(0) + }) }) function createRequestInterceptor() { @@ -131,6 +158,7 @@ function createRequestInterceptor() { for (const route of routes) { route.continue() } + return routes }, } },