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

fetchQuery returns stale data when used with createPersister #8075

Open
fa-sharp opened this issue Sep 21, 2024 · 4 comments
Open

fetchQuery returns stale data when used with createPersister #8075

fa-sharp opened this issue Sep 21, 2024 · 4 comments

Comments

@fa-sharp
Copy link

fa-sharp commented Sep 21, 2024

Describe the bug

I noticed that when using createPersister and then calling queryClient.fetchQuery, it will always return stale data from the persisted cache (if there's no data in the local cache).

Taking a look at the createPersister code, I can see that it signals if the query is stale, but it still returns the stale data. While this makes sense in a useQuery context, it doesn't make sense when using a procedural function like queryClient.fetchQuery, where we expect it to fetch the latest data if the cache is stale.

This makes it difficult to use the persister in a non-React context, e.g. in an extension service worker, where we might want to make procedural calls to fetchQuery to update some data in the background. I'm using a very questionable hack right now to get around this, calling fetchQuery two times. The first time, it returns the stale persisted data and loads it into the local cache. The second time, it actually runs the query function, as fetchQuery now detects stale data in the local cache. This has the desired effect of fetching the latest data, and automatically saving it to the persister (indexedDB in my case).

const staleData = await queryClient.fetchQuery(...)
await new Promise((r) => setTimeout(r, 100)); // this is necessary because the local cache isn't loaded immediately
const freshData = await queryClient.fetchQuery(...)

Your minimal, reproducible example

https://codesandbox.io/p/sandbox/loving-brook-9ln67f

Steps to reproduce

  1. Click on the "Fetch query" button a couple times. The counter will advance, as expected. The button is calling fetchQuery, and the counter data is also persisted to localStorage using createPersister.
  2. Click on "Clear local cache," which will clear React Query's local cache using queryClient.clear()
  3. Click "Fetch query" again. This time the counter will not advance, as fetchQuery is returning the stale data from createPersister, even though the staleTime is set to 10ms.
  4. If you click "Fetch query" one more time, this time the counter will advance, as fetchQuery now detects stale data in the local cache.

Expected behavior

I expect fetchQuery to run the query function and return the latest data, if it sees that the data from the persister is stale.

How often does this bug happen?

Every time

Screenshots or Videos

No response

Platform

macOS, Chrome 128

Tanstack Query adapter

react-query

TanStack Query version

v5.56.2

TypeScript version

v5.6.2

Additional context

This is tangentially related to the discussions in #6310 and #7677

@0xd8d
Copy link

0xd8d commented Oct 28, 2024

I also experience this issue with ensureQueryData and revalidateIfStale set to true. It doesn't revalidate on page load.

@TkDodo
Copy link
Collaborator

TkDodo commented Jan 4, 2025

The way the persister is implemented is that it returns cached data from the persister, and triggers a background refetch if the restored data has been considered stale:

if (query.isStale()) {
query.fetch()
}

as you can see, the fetch is now awaited. @DamianOsipiuk do you think it makes sense to await this fetch in some situations and return the fresh data? Maybe with a new option on the persister?

@DamianOsipiuk
Copy link
Contributor

Yeah.
I think in the spirit of SWR we should keep current behaviour as default.
But we can definietly add an option to wait for fresh data when persisted data is stale.
awaitWhenStale or something similar.

@TkDodo
Copy link
Collaborator

TkDodo commented Jan 6, 2025

we should consolidate naming a bit. On ensureQueryData, we have revalidateIfStale: boolean. I’m not super happy with different namings, so I might want to re-think this a bit for a v6.

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

No branches or pull requests

4 participants