From 5228e5170cdb927d9071855d193cccda4515cd79 Mon Sep 17 00:00:00 2001 From: Dominik Dorfmeister Date: Thu, 10 Oct 2024 20:55:29 +0200 Subject: [PATCH] fix(core): don't consider queries as enabled if they have no observers and have never fetched (successfully or erroneously) (#8161) it's very likely that this used to be a disabled observer; one other case would be canceling a query while you were initially fetching it, but this is a weird corner case that brings all sorts of troubles additionally, we can check for the queryFn being a skipToken; even if we have data in the cache, when the queryFn is currently set to skipToken, this is means we never want to see this query fetch --- .../src/__tests__/queryClient.test.tsx | 40 +++++++++++++++++++ packages/query-core/src/query.ts | 10 ++++- 2 files changed, 49 insertions(+), 1 deletion(-) diff --git a/packages/query-core/src/__tests__/queryClient.test.tsx b/packages/query-core/src/__tests__/queryClient.test.tsx index b8f58e2ec7..0ad7baed30 100644 --- a/packages/query-core/src/__tests__/queryClient.test.tsx +++ b/packages/query-core/src/__tests__/queryClient.test.tsx @@ -1409,6 +1409,46 @@ describe('queryClient', () => { expect(queryFn2).toHaveBeenCalledTimes(2) }) + test('should not refetch disabled inactive queries even if "refetchType" is "all', async () => { + const queryFn = vi + .fn<(...args: Array) => string>() + .mockReturnValue('data1') + const observer = new QueryObserver(queryClient, { + queryKey: queryKey(), + queryFn: queryFn, + staleTime: Infinity, + enabled: false, + }) + const unsubscribe = observer.subscribe(() => undefined) + unsubscribe() + await queryClient.invalidateQueries({ + refetchType: 'all', + }) + expect(queryFn).toHaveBeenCalledTimes(0) + }) + + test('should not refetch inactive queries that have a skipToken queryFn even if "refetchType" is "all', async () => { + const key = queryKey() + const observer = new QueryObserver(queryClient, { + queryKey: key, + queryFn: skipToken, + staleTime: Infinity, + }) + + queryClient.setQueryData(key, 'data1') + + const unsubscribe = observer.subscribe(() => undefined) + unsubscribe() + + expect(queryClient.getQueryState(key)?.dataUpdateCount).toBe(1) + + await queryClient.invalidateQueries({ + refetchType: 'all', + }) + + expect(queryClient.getQueryState(key)?.dataUpdateCount).toBe(1) + }) + test('should cancel ongoing fetches if cancelRefetch option is set (default value)', async () => { const key = queryKey() const abortFn = vi.fn() diff --git a/packages/query-core/src/query.ts b/packages/query-core/src/query.ts index 1160b1acfd..9850c75160 100644 --- a/packages/query-core/src/query.ts +++ b/packages/query-core/src/query.ts @@ -3,6 +3,7 @@ import { noop, replaceData, resolveEnabled, + skipToken, timeUntilStale, } from './utils' import { notifyManager } from './notifyManager' @@ -256,7 +257,14 @@ export class Query< } isDisabled(): boolean { - return this.getObserversCount() > 0 && !this.isActive() + if (this.getObserversCount() > 0) { + return !this.isActive() + } + // if a query has no observers, it should still be considered disabled if it never attempted a fetch + return ( + this.options.queryFn === skipToken || + this.state.dataUpdateCount + this.state.errorUpdateCount === 0 + ) } isStale(): boolean {