Skip to content

Commit

Permalink
Merge pull request reduxjs#4731 from reduxjs/bugfix/3353-isSuccess-fl…
Browse files Browse the repository at this point in the history
…ashing

Keep `isSuccess: true` when switching to an uninitialized cache entry
  • Loading branch information
markerikson authored Nov 23, 2024
2 parents 22cb3d0 + 02248d0 commit 4d92026
Show file tree
Hide file tree
Showing 2 changed files with 66 additions and 8 deletions.
10 changes: 8 additions & 2 deletions packages/toolkit/src/query/react/buildHooks.ts
Original file line number Diff line number Diff line change
Expand Up @@ -927,13 +927,19 @@ export function buildHooks<Definitions extends EndpointDefinitions>({

// isFetching = true any time a request is in flight
const isFetching = currentState.isLoading

// isLoading = true only when loading while no data is present yet (initial load with no data in the cache)
const isLoading =
(!lastResult || lastResult.isLoading || lastResult.isUninitialized) &&
!hasData &&
isFetching
// isSuccess = true when data is present
const isSuccess = currentState.isSuccess || (isFetching && hasData)

// isSuccess = true when data is present.
// That includes cases where the _current_ item is either actively
// fetching or about to fetch due to an uninitialized entry.
const isSuccess =
currentState.isSuccess ||
((isFetching || currentState.isUninitialized) && hasData)

return {
...currentState,
Expand Down
64 changes: 58 additions & 6 deletions packages/toolkit/src/query/tests/buildHooks.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -126,10 +126,10 @@ const api = createApi({
},
}),
queryWithDeepArg: build.query<string, { param: { nested: string } }>({
query: ({ param: { nested }}) => nested,
query: ({ param: { nested } }) => nested,
serializeQueryArgs: ({ queryArgs }) => {
return queryArgs.param.nested
}
},
}),
}),
})
Expand Down Expand Up @@ -383,6 +383,54 @@ describe('hooks tests', () => {
expect(fetchingHist).toEqual([true, false, true, false])
})

test('`isSuccess` does not jump back false on subsequent queries', async () => {
type LoadingState = {
id: number
isFetching: boolean
isSuccess: boolean
}
const loadingHistory: LoadingState[] = []

function User({ id }: { id: number }) {
const queryRes = api.endpoints.getUser.useQuery(id)

useEffect(() => {
const { isFetching, isSuccess } = queryRes
loadingHistory.push({ id, isFetching, isSuccess })
}, [id, queryRes])
return (
<div data-testid="status">
{queryRes.status === QueryStatus.fulfilled && id}
</div>
)
}

let { rerender } = render(<User id={1} />, { wrapper: storeRef.wrapper })

await waitFor(() =>
expect(screen.getByTestId('status').textContent).toBe('1'),
)
rerender(<User id={2} />)

await waitFor(() =>
expect(screen.getByTestId('status').textContent).toBe('2'),
)

expect(loadingHistory).toEqual([
// Initial render(s)
{ id: 1, isFetching: true, isSuccess: false },
{ id: 1, isFetching: true, isSuccess: false },
// Data returned
{ id: 1, isFetching: false, isSuccess: true },
// ID changed, there's an uninitialized cache entry.
// IMPORTANT: `isSuccess` should not be false here.
// We have valid data already for the old item.
{ id: 2, isFetching: true, isSuccess: true },
{ id: 2, isFetching: true, isSuccess: true },
{ id: 2, isFetching: false, isSuccess: true },
])
})

test('useQuery hook respects refetchOnMountOrArgChange: true', async () => {
let data, isLoading, isFetching
function User() {
Expand Down Expand Up @@ -674,9 +722,11 @@ describe('hooks tests', () => {
})

test(`useQuery shouldn't call args serialization if request skipped`, async () => {
expect(() => renderHook(() => api.endpoints.queryWithDeepArg.useQuery(skipToken), {
wrapper: storeRef.wrapper,
})).not.toThrow()
expect(() =>
renderHook(() => api.endpoints.queryWithDeepArg.useQuery(skipToken), {
wrapper: storeRef.wrapper,
}),
).not.toThrow()
})

test(`useQuery gracefully handles bigint types`, async () => {
Expand Down Expand Up @@ -2861,6 +2911,7 @@ describe('skip behavior', () => {
})
expect(result.current).toEqual({
...uninitialized,
isSuccess: true,
currentData: undefined,
data: { name: 'Timmy' },
})
Expand Down Expand Up @@ -2898,6 +2949,7 @@ describe('skip behavior', () => {
})
expect(result.current).toEqual({
...uninitialized,
isSuccess: true,
currentData: undefined,
data: { name: 'Timmy' },
})
Expand Down Expand Up @@ -2936,7 +2988,7 @@ describe('skip behavior', () => {
// even though it's skipped. `currentData` is undefined, since that matches the current arg.
expect(result.current).toMatchObject({
status: QueryStatus.uninitialized,
isSuccess: false,
isSuccess: true,
data: { name: 'Timmy' },
currentData: undefined,
})
Expand Down

0 comments on commit 4d92026

Please sign in to comment.