Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enabling a disabled query does not unsuspend #8499

Open
hector opened this issue Jan 3, 2025 · 19 comments · May be fixed by #8501
Open

Enabling a disabled query does not unsuspend #8499

hector opened this issue Jan 3, 2025 · 19 comments · May be fixed by #8501

Comments

@hector
Copy link

hector commented Jan 3, 2025

Describe the bug

I am using the new mechanism to suspend explained here: https://tanstack.com/query/latest/docs/framework/react/guides/suspense#using-usequerypromise-and-reactuse-experimental

Disabled queries return a pending promise (which is ok), but the promise does not resolve when the query is enabled afterwards.
This happens because the query is not refetched when enabled changes from false to true.

Your minimal, reproducible example

https://codesandbox.io/p/devbox/mystifying-mopsa-qwhjwx?workspaceId=ws_5WeinpZuX6zrSSrhdSgo5C

Steps to reproduce

Click the button to enable the query

Expected behavior

When the query is enabled it should fetch and the promise should finally resolve

How often does this bug happen?

Every time

Screenshots or Videos

No response

Platform

  • OS: macOS
  • Browser: Chrome
  • Version: 131.0.6778.205

Tanstack Query adapter

react-query

TanStack Query version

@tanstack/[email protected]

TypeScript version

[email protected]

Additional context

No response

@TkDodo
Copy link
Collaborator

TkDodo commented Jan 4, 2025

enabled is an option on the observer - but you unmount the observer completely when the Suspense fallback is rendered instead of the Example component. Since the promise was already passed to use, but the promise is also stored on the observer, there isn’t really a way for us to tell the Suspense Boundary that the promise has been resolved.

I tried switching to a conditional use:

const data = enabled ? use(promise) : null

This works a bit better - the component renders with null data when disabled, and when we trigger towards enabled, we suspend because we use the promise.

However, it still doesn’t trigger the fetch. The reason is that because a cache entry exists already, we can’t trigger the fetch during render without violating the rules of react:

const promise = isNewCacheEntry
? // Fetch immediately on render in order to ensure `.promise` is resolved even if the component is unmounted
fetchOptimistic(defaultedOptions, observer, errorResetBoundary)
: // subscribe to the "cache promise" so that we can finalize the currentThenable once data comes in
client.getQueryCache().get(defaultedOptions.queryHash)?.promise

isNewCacheEntry is false, so we just subscribe to the query cache promise. However, since the only thing that would actually trigger the fetch is the useEffect that updates the options inside the Example component, we never get to trigger the fetch because this effect doesn’t run as the Example component is unmounted as a whole due to suspense 🙃 .

The only thing that works right now is to pass the promise as a prop to a child component, where only the child component then suspends, keeping the component with useQuery mounted. Here’s my fork:

https://codesandbox.io/p/devbox/hopeful-flower-f6t8qp?workspaceId=ws_XiBCv7MgHSszKyXHXG5ray

@KATT what do you think ? If we changed our code to call fetchOptimistic during render whenever we have no data yet and e.g. have no promise in the cache yet, I think we could get it to work. However, it’s a slippery slope to call side-effects during render unless they are guaranteed to be not observable side-effects. This is guaranteed right now because we only run it if the cache entry gets created by useQuery, which means there can’t be any other observer. The widening would throw this out the window. @Ephem probably also has opinions on this :)

@Ephem
Copy link
Collaborator

Ephem commented Jan 4, 2025

I don't have many opinions really, but I do have observations. Your explanation why this is happening is great and makes total sense, but if you only look at the public API, this still seems like a bug. I do think avoiding observable side effects in render is something we should strive for though, so where does that leave us?

On the one hand, maybe this should lead us to question the current implementation, why are we triggering the fetch in the observer and not on the outside of it? I do think Suspense has been pushing the current RQ implementation for a while now. (It's interesting that <Suspense> could actually be seen as a new type of observer that only observes a single promise)

On the other hand, I think we have something like three cases:

  • The query is already in the cache and has data - Enabling should resolve the promise to that
  • The query is already in the cache and pending
    • It's pending because it is already fetching - Enabling should resolve with the data from that when it finishes
    • It's pending because it was previously disabled - Maybe in this case triggering the fetch in render is still safe because it wont be observable?

I don't have strong arguments why exactly, but I am starting to feel the current model/implementation chafing a bit.

@KATT

This comment has been minimized.

@KATT

This comment has been minimized.

@KATT
Copy link
Contributor

KATT commented Jan 4, 2025

Oh yeah, it's because the promise is colocated in the same component, I missed looking at the original reproduction 🤔

Agreed this is a bug

@Ephem
Copy link
Collaborator

Ephem commented Jan 4, 2025

FWIW I agree this specific bug is a bit niche and there's indeed a good userland workaround, hoist the useQuery to outside the Suspense boundary as you demonstrate, so I agree it's not super high priority and better to fix "right" than "fast". 😄

@hector
Copy link
Author

hector commented Jan 4, 2025

I don't know about the internals of the library but I believe this is how it should work:

  1. First render:
    On the first render of Example component: query is disabled, query.promise is pending and the component suspends
  2. Second render:
    When enabled prop is set to true, Example component renders again but "from zero", meaning that since it was suspended you need to think as if all the hooks (useRef, useEffect, etc...) run for the first time. No state from the first render is there.
    This is what React docs say:
    React does not preserve any state for renders that got suspended before they were able to mount for the first time. When the component has loaded, React will retry rendering the suspended tree from scratch.
    In this render, you should make useQuery return a new promise that will resolve once the data is fetched
    Again, in this render, the promise will be pending in the beginning and the component will suspend
  3. Third render:
    When the promise finally resolves because the data has been fetched, React will render the component from scratch but since the query will be in the cache it will just read it and render normally.

@hector
Copy link
Author

hector commented Jan 4, 2025

By the way I found another workaround for userland, I am sharing it here:
https://codesandbox.io/p/devbox/morning-tree-drm5xf?workspaceId=ws_5WeinpZuX6zrSSrhdSgo5C

In short, it consists on issuing a refetch if the query becomes enabled and hasn't been fetched before:

if (enabled && isPending) refetch();

Probably has its issues, point them out if needed.

@hector
Copy link
Author

hector commented Jan 4, 2025

Also, in case context helps, what I am trying to do is simply extending my hooks to be able to do this:

const { data: user } = useQuery(...).suspense();

suspense function is basically a shortcut to:

React.use(useQuery(...).promise)

So then I have my custom hooks and can use them with or without suspense:

const user = useUser().data; // user might be null
// or
const user = useUser().suspense().data; // user cannot be null

Then I can easily use suspense but with parallel fetching:

// Start both queries
const userQuery = useUser();
const articlesQuery = useArticles();

// Suspend until both queries have data
userQuery.suspend();
articlesQuery.suspend();

// Natively it would have been:
React.use(Promise.all([userQuery.promise, articlesQuery.promise]));

The thing is that all this logic works inside hooks, I cannot create suspense boundaries to make it work.

@TkDodo
Copy link
Collaborator

TkDodo commented Jan 5, 2025

but I believe this is how it should work:

@hector what you describe is not how it works. In the second render, enabled is true, but fetching will happen in an effect. That effect never executes because use(promise) will unmount the component that has the effect that would trigger the fetch. experimental_prefetchInRender only triggers a fetch during render IF there is no cache entry yet, which is only true on the first render (where we don’t want a fetch due to enabled:false), but not on the second render, as the entry exists already. As it exists, it can potentially be observed by another component, so strictly speaking, the rules of react don’t allow us to trigger a fetch in render here.

It's pending because it was previously disabled - Maybe in this case triggering the fetch in render is still safe because it wont be observable?

@Ephem I think I disagree here. The cache entry could definitely be observed by a different component, for example, the devools will show it. We could try to bend the rules here but I’m not really a fan.


Generally, it seems that relying on useQuery to trigger the fetch, while at the same time, calling use(promise) in the same component unconditionally feels like an anti-pattern to me. This is literally the same as what useSuspenseQuery does at the moment, and it’s what we want to move away from because of potential waterfalls. There’s also a reason why enabled doesn’t exist on useSuspenseQuery, but with use + useQuery, it’s kinda “coming back”.

I think all of this would be easier for us if use(observable) would exist. Maybe it will exist in the future, but it doesn’t right now. The best thing would be to trigger the fetch once the promise is passed to use, because that’s when we can guarantee that the component will suspend, but we don’t know about that either. Proxies also don’t help here because destructing promise from useQuery is enough to trigger the read.


I’m all in all more inclined to leave this pattern as-is, and document it, or even go the opposite direction and remove experimental_prefetchInRender to enforce splitting the promise initialization and suspension.

@hector
Copy link
Author

hector commented Jan 5, 2025

@TkDodo I see, thanks for the explanation.

experimental_prefetchInRender only triggers a fetch during render IF there is no cache entry yet, which is only true on the first render (where we don’t want a fetch due to enabled:false), but not on the second render, as the entry exists already. As it exists, it can potentially be observed by another component, so strictly speaking, the rules of react don’t allow us to trigger a fetch in render here.

What if you did not create the cache entry for a disabled query? Seems like it would work fine then.

@KATT
Copy link
Contributor

KATT commented Jan 6, 2025

I need to think about this issue some more, but, I kinda think that we should have a .promise that always rejects whenever the query is disabled - having a spinner that loads forever is a bit confusing DX which will easily creep out into apps.

I'm not convinced what the right thing to do is but I fixed it so it works according to current expectations in #8501.

@Ephem
Copy link
Collaborator

Ephem commented Jan 10, 2025

The cache entry could definitely be observed by a different component, for example, the devools will show it. We could try to bend the rules here but I’m not really a fan.

This is very true, I was thinking nothing could be observing the data in this specific circumstance, but you are right the query itself could be observed.

Generally, it seems that relying on useQuery to trigger the fetch, while at the same time, calling use(promise) in the same component unconditionally feels like an anti-pattern to me. This is literally the same as what useSuspenseQuery does at the moment, and it’s what we want to move away from because of potential waterfalls.

It is only an anti-pattern if you have not prefetched the query elsewhere, which is the common pattern with loaders. So if we want to at some point deprecate useSuspenseQuery in favour of this pattern, I think we'd need to fully support useQuery+use in the same component?

There’s also a reason why enabled doesn’t exist on useSuspenseQuery, but with use + useQuery, it’s kinda “coming back”.

Can you remind me about the reason again? I know there are some implementation trickiness involved here, but I kind of think we should support this long term if we can. I've been struggling with this myself, how would we solve dependent queries with Suspense otherwise? I don't think I've seen a good pattern currently. To be fair I think this is trickier with useSuspenseQuery than useQuery+use, because:

I kinda think that we should have a .promise that always rejects whenever the query is disabled

@KATT Yes! I think this makes perfect sense. Trying to read from a disabled query does feel like it should be an error. Since a common reason for disabling a query is that it's a dependent one, it makes sense to me to have to make that check both when you declare the fetch and when you try to read. This also solves the type issue that data should always be defined with Suspense.

@TkDodo
Copy link
Collaborator

TkDodo commented Jan 10, 2025

I think we'd need to fully support useQuery+use in the same component?

Yes, the not supported part I was referring to is expecting this combo to trigger a fetch, which we currently do with the experimental_prefetchInRender flag. But as this issue has shown, this only works for new cache entries, and not situations where it transitions from disabled to enabled, as the entry already exists. I’m sure there are other situations where it won’t work, too.

So if we get rid of experimental_prefetchInRender altogether, you would be forced to either prefetch or pass the promise down to a component that has its own suspense boundary. It seems pretty limiting and I’m back-and-forth between keeping the flag and removing it 😅. But I don’t think there is a good way to fix this opened issue here. The only way I see is also triggering a fetch during render if we have no data in the cache and we were previously disabled; I think it’s possible technically but again, not a fan.

Can you remind me about the reason again?

You’d expect TypeScript to give you data: T and not data: T | undefined. We can’t guarantee this with enabled: false.

how would we solve dependent queries with Suspense otherwise?

The suspense drawback that it creates waterfalls works to our advantage here. A simple dependent query with suspense is just ... calling useSuspenseQuery twice in the same component. As it waterfalls, you’d have the data available from the first query when the second query fetches. Our example in the docs simply becomes:

// Get the user
const { data: user } = useSuspenseQuery({
  queryKey: ['user', email],
  queryFn: getUserByEmail,
})

const userId = user.id

// Then get the user's projects
const {
  status,
  fetchStatus,
  data: projects,
} = useSuspenseQuery({
  queryKey: ['projects', userId],
  queryFn: getProjectsByUser,
})

see that we can just access user.id and not user?.id thanks to suspense.

Trying to read from a disabled query does feel like it should be an error.

Yeah, you can just not use(promise) if the query is disabled. It wouldn’t solve this issue though, right? Transitioning from disabled to enabled would not trigger the fetch in the component unless we bend the rules :/

I think the user-land workaround @hector showed is exactly that:

if (enabled && isPending) refetch();

triggering a fetch inside render when you know you’re going to suspend. But again, I think it’s dangerous for us to do this in the lib itself.

@TkDodo
Copy link
Collaborator

TkDodo commented Jan 10, 2025

What if you did not create the cache entry for a disabled query? Seems like it would work fine then.

Sounds fine in theory, but it’s not how react-query works. We couldn’t return anything from useQuery in those cases - we need a cache entry and an observer that’s subscribed to that entry. Also, if you call queryClient.setQueryData somewhere else that puts some data into the cache, your disabled observer needs to still reflect those changes.

@Ephem
Copy link
Collaborator

Ephem commented Jan 10, 2025

Yeah, you can just not use(promise) if the query is disabled. It wouldn’t solve this issue though, right?

For sure, I'm definitely discussing several things here. API design/optimal behaviour and pragmatic/reasonable solutions might differ.

Yes, the not supported part I was referring to is expecting this combo to trigger a fetch, which we currently do with the experimental_prefetchInRender flag.

Ah, of course, thanks for clarifying! I do agree this is limiting and surprising though so if we could fix this, I think that would be optimal.

But I don’t think there is a good way to fix this opened issue here.

I definitely think it's hard, and maybe not worth it, but I do think it should be possible. One way could be to queue the fetch in render, but execute it elsewhere (higher up/outside the render tree)? That does also seem like a more robust model in general with Suspense?

This is complex of course. The component has to create the promise since it needs to be useable in render, but it can't fulfil the promise since it hasn't started a fetch. So the queue would have to include both these things, and an external controller (queryClient) would have to keep a map of fetches and observer promises and resolve them when the fetch resolves.

Might not be worth it just for this and it's definitely not a short term thing, but I remember seeing other issues that might benefit from separating the promise from the fetch, so there might be more reasons to do this.

You’d expect TypeScript to give you data: T and not data: T | undefined. We can’t guarantee this with enabled: false.

If we fail the use(), this is only a problem for useSuspenseQuery, but even so, couldn't we type useSuspenseQuery as always returning undefined when enabled: false and always T when enabled? (I ask naively 😇) That way this would only be a problem when you actually use the option, and at that point it would be expected?

As it waterfalls, you’d have the data available from the first query when the second query fetches.

This is a super nice thing about suspense, but, what if I don't want to make the second query at all? The article does not contain any authorId? You can of course add this behaviour to the queryFn of the author query, returning null if no id is passed in (since skipToken also does not work), but that's also not optimal at a type level since both the input and the output are now optional, making reuse harder etc. If this would be considered good enough DX, I don't think we would need to have enabled/skipToken for useQuery either?


I haven't mentioned this in the thread, but I do have the exact situation @hector is reporting about here and I was hoping using useQuery+use would be a nice way to solve it before I saw this issue. The case is I have two queries, the second dependent on the first. Both are prefetched (the second conditionally), but, there are situations where the first can change client-side and in those cases it can switch from "disabled" to "enabled" or the other way around. This is pretty common when using suspense to resolve feature flags/settings etc. If a setting is enabled, I'd like that part of the app to fetch its data. Because I often rely on the loading.tsx from NextJS for loading states, it's pretty cumbersome to move the useQuerys to above the Suspense boundary (which also requires passing the promise down through multiple layers).

Just wanted to add some extra context on usecases. 😄

@TkDodo
Copy link
Collaborator

TkDodo commented Jan 10, 2025

but execute it elsewhere (higher up/outside the render tree)? That does also seem like a more robust model in general with Suspense?

that’s certainly interesting - so you think writing to a context / outside store during render that is not observed by any react component would be fine?

If we fail the use(), this is only a problem for useSuspenseQuery, but even so, couldn't we type useSuspenseQuery as always returning undefined when enabled: false and always T when enabled?

yes, conditional return types could do this I think, but it’s complex. The longer term goal though would be to deprecate / get rid of useSuspenseQuery and switch to use(promise). If we can fix it for that case, that would be enough for me.

but, what if I don't want to make the second query at all?

I think right now, you’d have to conditionally render a child component.

@Ephem
Copy link
Collaborator

Ephem commented Jan 10, 2025

that’s certainly interesting - so you think writing to a context / outside store during render that is not observed by any react component would be fine?

I'm not entirely sure exactly what is safe and not, but if the queue is not observable on the outside and the actual observable side effect happens in an effect (so maybe not in queryClient), I would at least think so? Feels like we can formulate a pretty clear question to the React team about this.

I don't think we could remove the fetch from the queue if the Suspense boundary quickly stops rendering though, but I don't think that's a huge deal.

If we can fix it for that case, that would be enough for me.

Same! Very reasonable tradeoff and a good way to push for useQuery(); use(promise); too (when we feel confident enough about that).

I think right now, you’d have to conditionally render a child component.

Yeah, or do the queryFn thing, but I don't wanna 😛

@Ephem
Copy link
Collaborator

Ephem commented Jan 16, 2025

Re: useSuspenseQuery enabled

conditional return types could do this I think

I experimented with this a bit and turns out that it sure can work with explicit true and false, but that's not what we want of course. Thinking about this it's quite obvious that this is runtime behaviour that we can't type away.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

4 participants