web/.agents/skills/tanstack-query-best-practices/rules/cache-stale-time.md
staleTime determines how long data is considered fresh. The default is 0ms, meaning data is immediately stale and will refetch on every new query mount. Set appropriate staleTime based on how often your data actually changes to reduce unnecessary network requests.
// Default staleTime of 0 - refetches on every component mount
const { data } = useQuery({
queryKey: ['user-profile', userId],
queryFn: () => fetchUserProfile(userId),
// No staleTime set - always considered stale
})
// User profile probably doesn't change every second
// This causes unnecessary API calls on navigation
// Setting same staleTime everywhere regardless of data type
const queryClient = new QueryClient({
defaultOptions: {
queries: {
staleTime: 60 * 1000, // 1 minute for everything - too simple
},
},
})
// Match staleTime to data volatility
const { data: profile } = useQuery({
queryKey: ['user-profile', userId],
queryFn: () => fetchUserProfile(userId),
staleTime: 5 * 60 * 1000, // 5 minutes - profile rarely changes
})
const { data: notifications } = useQuery({
queryKey: ['notifications'],
queryFn: fetchNotifications,
staleTime: 30 * 1000, // 30 seconds - changes more frequently
})
const { data: stockPrice } = useQuery({
queryKey: ['stock', symbol],
queryFn: () => fetchStockPrice(symbol),
staleTime: 0, // Real-time data - always refetch
})
// Set sensible defaults, override per-query
const queryClient = new QueryClient({
defaultOptions: {
queries: {
staleTime: 60 * 1000, // 1 minute default
},
},
})
| Data Type | staleTime | Rationale |
|---|---|---|
| Real-time (stocks, live feeds) | 0 | Must always be current |
| Frequently changing (notifications) | 30s - 1min | Balance freshness and requests |
| User-generated content | 1 - 5min | Changes on user action |
| Reference data (categories, config) | 10 - 30min | Rarely changes |
| Static content | Infinity | Never changes |
staleTime: 0 (default) triggers background refetch on every mountstaleTime: Infinity never considers data stale (manual invalidation only)queryOptions factory to centralize staleTime per data type