install
source · Clone the upstream repo
git clone https://github.com/Intense-Visions/harness-engineering
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/Intense-Visions/harness-engineering "$T" && mkdir -p ~/.claude/skills && cp -r "$T/agents/skills/claude-code/tanstack-cache-management" ~/.claude/skills/intense-visions-harness-engineering-tanstack-cache-management && rm -rf "$T"
manifest:
agents/skills/claude-code/tanstack-cache-management/SKILL.mdsource content
TanStack Query: Cache Management
Directly read, write, and remove cache entries with QueryClient methods
When to Use
- Updating related cache entries after a mutation without an extra network request
- Pre-populating the cache with data received from a WebSocket or server-sent event
- Removing cached data when it is known to be invalid (e.g., after logout)
- Inspecting cached values in tests to assert cache state after operations
- Implementing advanced patterns like cache federation or cross-query updates
Instructions
- Use
to write a value directly to the cache — the query does not refetch.queryClient.setQueryData(key, updaterOrValue) - Use
to read a cached value synchronously — returnsqueryClient.getQueryData(key)
if not cached.undefined - Use
before manual cache writes to prevent in-flight fetches from overwriting your change.queryClient.cancelQueries({ queryKey }) - Use
to evict cache entries entirely — they refetch fresh on next mount.queryClient.removeQueries({ queryKey }) - Use
to clear the entire cache — call this on logout to remove all user data.queryClient.clear() - Use
to bulk-read multiple cache entries matching a filter.queryClient.getQueriesData(filter) - Use updater functions in
for immutable updates — receive current value, return new value.setQueryData
// Real-time cache update from WebSocket function useLivePostUpdates() { const queryClient = useQueryClient(); useEffect(() => { const ws = new WebSocket('/ws/posts'); ws.onmessage = (event) => { const updatedPost: Post = JSON.parse(event.data); // Update the specific post detail queryClient.setQueryData(postKeys.detail(updatedPost.id), updatedPost); // Update the post in all list caches queryClient.setQueriesData<PostsPage>({ queryKey: postKeys.lists() }, (old) => { if (!old) return old; return { ...old, posts: old.posts.map((p) => (p.id === updatedPost.id ? updatedPost : p)), }; }); }; return () => ws.close(); }, [queryClient]); } // Logout — clear all user data from cache function useLogout() { const queryClient = useQueryClient(); const router = useRouter(); return async () => { await fetch('/api/auth/logout', { method: 'POST' }); queryClient.clear(); // remove all cached data router.push('/login'); }; } // Seeding cache after create mutation — avoid refetch const { mutate: createPost } = useMutation({ mutationFn: (data: CreatePostInput) => fetch('/api/posts', { method: 'POST', body: JSON.stringify(data) }).then((r) => r.json()), onSuccess: (newPost: Post) => { // Add to detail cache directly queryClient.setQueryData(postKeys.detail(newPost.id), newPost); // Invalidate lists — server determines list membership queryClient.invalidateQueries({ queryKey: postKeys.lists() }); }, });
Details
TanStack Query's cache is a
Map-like structure keyed by serialized query keys. setQueryData, getQueryData, and removeQueries give direct programmatic access to this cache outside the normal fetch lifecycle.
vs setQueryData
: invalidateQueries
setQueryData writes a value and marks the query as fresh (it will not refetch until staleTime expires). invalidateQueries marks the query as stale and triggers a background refetch. Use setQueryData when you have the correct server response in hand (e.g., from a create mutation); use invalidateQueries when you know data has changed but do not have the new value.
for list updates: When a single entity update needs to reflect in multiple list queries (e.g., filtered views), use setQueriesData
setQueriesData(filter, updater) to update all matching caches in one call. The filter uses the same matching logic as invalidateQueries.
Observer pattern: Use
queryClient.getQueryCache().subscribe(callback) to observe all cache changes. This is useful for syncing query state to external systems (analytics, debug logging). Unsubscribe on cleanup.
Type safety:
getQueryData<T>(key) returns T | undefined. Always handle the undefined case — the cache may not have the value if it was never fetched or was garbage collected.
Testing assertions: In tests, call
queryClient.getQueryData(key) after mutations to assert that the cache was updated correctly — without needing to render a component and inspect the DOM.
Source
https://tanstack.com/query/latest/docs/reference/QueryClient
Process
- Read the instructions and examples in this document.
- Apply the patterns to your implementation, adapting to your specific context.
- Verify your implementation against the details and edge cases listed above.
Harness Integration
- Type: knowledge — this skill is a reference document, not a procedural workflow.
- No tools or state — consumed as context by other skills and agents.
Success Criteria
- The patterns described in this document are applied correctly in the implementation.
- Edge cases and anti-patterns listed in this document are avoided.