Describe the bug
If a useQuery is rendered before a HydrationBouhndary, hydrations for that key will be skipped despite the fact that the useQuery would have not fetched in the first place.
In our case we have a prefetch where we pass the dehydrated queryClient to a HydrationBoundary. But there is a useQuery using the same key as the prefetched query above the hydration boundary that (in some global component higher up) gets rendered. The hydration boundary skips this hydration and the child components of the HydrationBoundary that have useSuspenseQuery for that key still run their queries in SSR.
Small example:
function Page({ params }) {
const queryClient = getQueryClient();
void queryClient.prefetchQuery(
buildTodoQueryOptions(params.todoId, makeServerRequest),
);
return (
<>
<SomeHeaderComponent /> {/* a `useQuery` lives in here */}
<HydrationBoundary state={dehydrate(queryClient)}>
<MyChildComponent /> {/* a `useSuspenseQuery` lives in here and fires despite the above prefetch */}
</HydrationBoundary>
</>
);
}
Also, in our particular case this triggers a guard we have to ensure we do not client fetch on the server during SSR. This also means the request is being duplicated and firing twice (in our prefetch and during SSR, which should have been skipped)
Flow (all on the server):
- Parent/route boundary prefetches the detail query
todoId on the server.
- A higher component in layout/header renders a
useQuery with the key todoId.
- Pass the dehydrated and prefetched
queryClient to a HydrationBoundary lower than the useQuery.
- The HydrationBoundary` skips this key and does not hydrate the prefetched query as expected.
- Child
useSuspenseQuery then attempts a fetch with client requestor during SSR and hits the guard error.
Repro
Server prefetch at detail layout/page:
// In app/(dashboard)/todos/[todoId]/layout.tsx or page.tsx
function Page({ params }) {
const queryClient = getQueryClient();
void queryClient.prefetchQuery(buildTodoQueryOptions(todoId, makeServerRequest));
return <HydrationBoundary state={dehydrate(queryClient)}>{children}</HydrationBoundary>;
}
Global/header client component that also reads URL todoId (lives OUTSIDE/above the above layout.tsx)
"use client";
import { useQuery, useParams } from "@tanstack/react-query";
function TodoHeaderMenu() {
const params = useParams();
const todoId = typeof params.todoId === "string" ? params.todoId : "";
const { data: todo } = useQuery({
...buildTodoQueryOptions(todoId),
enabled: Boolean(todoId),
});
return <span>{todo?.title}</span>;
}
Child detail component:
"use client";
function TodoPage({ todoId }: { todoId: string }) {
const { data: todo } = useSuspenseQuery(buildTodoQueryOptions(todoId));
return <TodoDetails todo={todo} />;
}
buildTodoQueryOptions defaults to client requestor. Client-request guard throws:
if (typeof window === "undefined") {
throw new Error(
"Warning, aborting SSR. You attempted to fetch from the server with clientRequest."
);
}
AI disclosure: I had codex help me with the quick repro.
Questions
- Is this current behavior expected?
- Should hydration logic be changed to only skip queries that are fetching (aka don't skip
useQueries that fill the cache key on the server but do not initiate filling the value on the server)
Your minimal, reproducible example
https://codesandbox.io/p/devbox/pr7c4g
Steps to reproduce
The home page on that will bring up the error in the next dev error popup
Expected behavior
I would expect a useQuery in a higher component with the same key to not STOP the server fetch/hydration given useQuery would not trigger the client request guard like useSuspenseQuery does. useQuery does not start the queryFn on the server, so it should not stop useSuspenseQueries that CAN imo.
This also comes with the additional problem that because of the useQuery, the useSuspenseQuery is firing again, ruining deduplication.
How often does this bug happen?
Every time
Screenshots or Videos
No response
Platform
macOS
Chromium
next: 16.1.1
Tanstack Query adapter
react-query
TanStack Query version
5.90.21
TypeScript version
No response
Additional context
I can technically gate the useQuery in the parent component with something like
const isDealQueryReady = !!(
state &&
state.data !== undefined &&
state.status !== "pending" &&
state.fetchStatus !== "fetching"
)
//and then
if (!isDealQueryReady) {
return null;
}
//...
//...`useQuery`
and then I do not get the error but that is obnoxious.
Describe the bug
If a
useQueryis rendered before aHydrationBouhndary, hydrations for that key will be skipped despite the fact that theuseQuerywould have not fetched in the first place.In our case we have a prefetch where we pass the dehydrated
queryClientto aHydrationBoundary. But there is auseQueryusing the same key as the prefetched query above the hydration boundary that (in some global component higher up) gets rendered. The hydration boundary skips this hydration and the child components of theHydrationBoundarythat haveuseSuspenseQueryfor that key still run their queries in SSR.Small example:
Also, in our particular case this triggers a guard we have to ensure we do not client fetch on the server during SSR. This also means the request is being duplicated and firing twice (in our prefetch and during SSR, which should have been skipped)
Flow (all on the server):
todoIdon the server.useQuerywith the keytodoId.queryClientto aHydrationBoundarylower than theuseQuery.useSuspenseQuerythen attempts a fetch with client requestor during SSR and hits the guard error.Repro
Server prefetch at detail layout/page:
Global/header client component that also reads URL
todoId(lives OUTSIDE/above the above layout.tsx)Child detail component:
buildTodoQueryOptionsdefaults to client requestor. Client-request guard throws:AI disclosure: I had codex help me with the quick repro.
Questions
useQueriesthat fill the cache key on the server but do not initiate filling the value on the server)Your minimal, reproducible example
https://codesandbox.io/p/devbox/pr7c4g
Steps to reproduce
The home page on that will bring up the error in the next dev error popup
Expected behavior
I would expect a
useQueryin a higher component with the same key to not STOP the server fetch/hydration givenuseQuerywould not trigger the client request guard likeuseSuspenseQuerydoes.useQuerydoes not start thequeryFnon the server, so it should not stopuseSuspenseQueriesthat CAN imo.This also comes with the additional problem that because of the
useQuery, theuseSuspenseQueryis firing again, ruining deduplication.How often does this bug happen?
Every time
Screenshots or Videos
No response
Platform
macOS
Chromium
next: 16.1.1
Tanstack Query adapter
react-query
TanStack Query version
5.90.21
TypeScript version
No response
Additional context
I can technically gate the
useQueryin the parent component with something likeand then I do not get the error but that is obnoxious.