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

SWR state has undocumented and unexpected values on key change #4112

Open
TmanTman opened this issue Mar 28, 2025 · 1 comment
Open

SWR state has undocumented and unexpected values on key change #4112

TmanTman opened this issue Mar 28, 2025 · 1 comment

Comments

@TmanTman
Copy link

TmanTman commented Mar 28, 2025

Bug report

Description / Observed Behavior

Consider the following values during a key change:

const { data, isLoading } = useSWRImmutable(someKey)

If "someKey" changes, the "data" variable can go undefined before "isLoading" goes to true. This is the state progression:

someKey data isLoading Notes
'a' defined false
'b' undefined false unexpected
'b' undefined true
'b' defined false

This behaviour is not expected, and not documented on SWR under "key change + previous data" docs

A use cases that breaks here is, for example, if a useEffect looks at these variables and if someKey leads to undefined results, makes update to state

Expected Behavior

The following state progression:

someKey data isLoading
'a' defined false
'b' undefined true
'b' defined false

Repro Steps / Code Example

Here is a link to a StackBlitz. The console shows this state.

Additional Context

SWR version: Previous versions + Latest

@kawaaaas
Copy link

kawaaaas commented Apr 5, 2025

Hello,
I'm a big fan of SWR and use it regularly. I'd like to express my gratitude to the maintainers, contributors, and those who open issues.

Problem

I've encountered the same issue myself. This problem occurs when revalidateIfStale = false is set, which has been previously reported in other issues.
#2930

Proposed Solution

Here's my proposal for fixing this:
Modify the getSnapshot() logic to add an edge case handling for revalidateIfStale:

const getSnapshot = useMemo(() => {
    const shouldStartRequest = (() => {
      if (!key) return false
      if (!fetcher) return false
      // If `revalidateOnMount` is set, we take the value directly.
      if (!isUndefined(revalidateOnMount)) return revalidateOnMount
      // If it's paused, we skip revalidation.
      if (getConfig().isPaused()) return false
      return !suspense
    })()
    
    // Get the cache and merge it with expected states.
    const getSelectedCache = (state: ReturnType<typeof getCache>) => {
      // We only select the needed fields from the state.
      const snapshot = mergeObjects(state)
      delete snapshot._k
      
      if (!shouldStartRequest) {
        return snapshot
      }
      
   // Added condition to handle revalidateIfStale=false edge case
      if (!(revalidateIfStale !== false)) {
        if (isUndefined(snapshot.data) && isUndefined(fallbackData)) {
          return {
            isValidating: true,
            isLoading: true,
            ...snapshot
          }
        } else {
          return snapshot
        }
      }
      
      return {
        isValidating: true,
        isLoading: true,
        ...snapshot
      }
    }
    // Rest of the code remains unchanged

Verification

I've verified that all tests pass, it returns true when there's no data, and the issue is resolved.
Would it be appropriate for me to create a PR with this approach? Please let me know if there are any requirements I haven't addressed in this solution.
Also, if this issue reflects the intended behavior, please close this and same issue.

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

No branches or pull requests

2 participants