React Query and TypeScript
Last Update: 2023-10-21
- Earlier parts of the series are hidden
- #5: Testing React Query
- #6: React Query and TypeScriptCurrent
- #7: Using WebSockets with React Query
- Later parts of the series are hidden
All 32 parts in the series
- #1: Practical React Query
- #2: React Query Data Transformations
- #3: React Query Render Optimizations
- #4: Status Checks in React Query
- #5: Testing React Query
- #6: React Query and TypeScriptCurrent
- #7: Using WebSockets with React Query
- #8: Effective React Query Keys
- #8a: Leveraging the Query Function Context
- #9: Placeholder and Initial Data in React Query
- #10: React Query as a State Manager
- #11: React Query Error Handling
- #12: Mastering Mutations in React Query
- #13: Offline React Query
- #14: React Query and Forms
- #15: React Query FAQs
- #16: React Query meets React Router
- #17: Seeding the Query Cache
- #18: Inside React Query
- #19: Type-safe React Query
- #20: You Might Not Need React Query
- #21: Thinking in React Query
- #22: React Query and React Context
- #23: Why You Want React Query
- #24: The Query Options API
- #25: Automatic Query Invalidation after Mutations
- #26: How Infinite Queries work
- #27: React Query API Design - Lessons Learned
- #28: React Query - The Bad Parts
- #29: Concurrent Optimistic Updates in React Query
- #30: React Query Selectors, Supercharged
- #31: Creating Query AbstractionsLatest
TypeScript (opens in a new window) is 🔥 - this seems to be a common understanding now in the frontend community. Many developers expect libraries to either be written in TypeScript, or at least provide good type definitions. For me, if a library is written in TypeScript, the type definitions are the best documentation there is. It’s never wrong because it directly reflects the implementation. I frequently look at type definitions before I read API docs.
React Query was initially written in JavaScript (v1), and was then re-written to TypeScript with v2. This means that right now, there is very good support for TypeScript consumers.
There are however a couple of “gotchas” when working with TypeScript due to how dynamic and unopinionated React Query is. Let’s go through them one by one to make your experience with it even better.
React Query heavily uses Generics (opens in a new window). This is necessary because the library does not actually fetch data for you, and it cannot know what type the data will have that your api returns.
The TypeScript section in the official docs (opens in a new window) is not very extensive, and it tells us to explicitly specify the Generics that useQuery expects when calling it:
function useGroups() { return useQuery<Group[], Error>({ queryKey: ['groups'], queryFn: fetchGroups, })}Over time, React Query has added more Generics to the useQuery hook (there are now four of them), mainly because more functionality was added. The above code works, and it will make sure that the data property of our custom hook is correctly typed to Group[] | undefined as well as that our error will be of type Error | undefined. But it will not work like that for more advanced use-cases, especially when the other two Generics are needed.
This is the current definition of the useQuery hook:
export function useQuery< TQueryFnData = unknown, TError = unknown, TData = TQueryFnData, TQueryKey extends QueryKey = QueryKey>There’s a lot of stuff going on, so let’s try to break it down:
TQueryFnData: the type returned from thequeryFn. In the above example, it’sGroup[].TError: the type of Errors to expect from thequeryFn.Errorin the example.TData: the type ourdataproperty will eventually have. Only relevant if you use theselectoption, because then thedataproperty can be different from what thequeryFnreturns. Otherwise, it will default to whatever thequeryFnreturns.TQueryKey: the type of ourqueryKey, only relevant if you use thequeryKeythat is passed to yourqueryFn.
As you can also see, all those Generics have default values, which means that if you don’t provide them, TypeScript will fall back to those types. This works pretty much the same as default parameters in JavaScript:
function multiply(a, b = 2) { return a * b}
multiply(10) // ✅ 20multiply(10, 3) // ✅ 30TypeScript works best if you let it infer (or figure out) what type something should be on its own. Not only does it make code easier to write (because you don’t have to type all the types 😅), but it will also make it easier to read. In many instances, it can make code look exactly like JavaScript. Some simple examples of type inference would be:
const num = Math.random() + 5 // ✅ `number`
// 🚀 both greeting and the result of greet will be stringfunction greet(greeting = 'ciao') { return `${greeting}, ${getName()}`}When it comes to Generics, they can also generally be inferred from their usage, which is super awesome. You could also provide them manually, but in many cases, you don’t need to.
function identity<T>(value: T): T { return value}
// 🚨 no need to provide the genericlet result = identity<number>(23)
// ⚠️ or to annotate the resultlet result: number = identity(23)
// 😎 infers correctly to `string`let result = identity('react-query')…doesn’t exist in TypeScript yet (see this open issue (opens in a new window)). This basically means that if you provide one Generic, you have to provide all of them. But because React Query has default values for Generics, we might not notice right away that they will be taken. The resulting error messages can be quite cryptic. Let’s look at an example where this actually backfires:
function useGroupCount() { return useQuery<Group[], Error>({ queryKey: ['groups'], queryFn: fetchGroups, select: (groups) => groups.length, // 🚨 Type '(groups: Group[]) => number' is not assignable to type '(data: Group[]) => Group[]'. // Type 'number' is not assignable to type 'Group[]'.ts(2322) })}Because we haven’t provided the 3rd Generic, the default value kicks in, which is also Group[], but we return number from our select function. One fix is to simply add the 3rd Generic:
function useGroupCount() { // ✅ fixed it return useQuery<Group[], Error, number>({ queryKey: ['groups'], queryFn: fetchGroups, select: (groups) => groups.length, })}As long as we don’t have Partial Type Argument Inference, we have to work with what we got.
So what’s the alternative?
Let’s start by not passing in any Generics at all and let TypeScript figure out what to do. For this to work, we need the queryFn to have a good return type. Of course, if you inline that function without an explicit return type, you will have any - because that’s what axios or fetch give you:
function useGroups() { // 🚨 data will be `any` here return useQuery({ queryKey: ['groups'], queryFn: () => axios.get('groups').then((response) => response.data), })}If you (like me) like to keep your api layer separated from your queries, you’ll need to add type definitions anyways to avoid implicit any, so React Query can infer the rest:
function fetchGroups(): Promise<Group[]> { return axios.get('groups').then((response) => response.data)}
// ✅ data will be `Group[] | undefined` herefunction useGroups() { return useQuery({ queryKey: ['groups'], queryFn: fetchGroups })}
// ✅ data will be `number | undefined` herefunction useGroupCount() { return useQuery({ queryKey: ['groups'], queryFn: fetchGroups, select: (groups) => groups.length, })}Advantages of this approach are:
- no more manually specifying Generics
- works for cases where the 3rd (select) and 4th (QueryKey) Generic are needed
- will continue to work if more Generics are added
- code is less confusing / looks more like JavaScript
What about error, you might ask? Per default, without any Generics, error will be inferred to unknown. This might sound like a bug, why is it not Error? But it is actually on purpose, because in JavaScript, you can throw anything - it doesn’t have to be of type Error:
Since React Query is not in charge of the function that returns the Promise, it also can’t know what type of errors it might produce. So unknown is correct. Once TypeScript allows skipping some generics when calling a function with multiple generics (see this issue for more information (opens in a new window)), we could handle this better, but for now, if we need to work with errors and don’t want to resort to passing Generics, we can narrow the type with an instanceof check:
const groups = useGroups()
if (groups.error) { // 🚨 this doesn't work because: Object is of type 'unknown'.ts(2571) return <div>An error occurred: {groups.error.message}</div>}
// ✅ the instanceOf check narrows to type `Error`if (groups.error instanceof Error) { return <div>An error occurred: {groups.error.message}</div>}Since we need to make some kind of check anyways to see if we have an error, the instanceof check doesn’t look like a bad idea at all, and it will also make sure that our error actually has a property message at runtime. This is also in line with what TypeScript has planned for the 4.4 release, where they’ll introduce a new compiler flag useUnknownInCatchVariables, where catch variables will be unknown instead of any (see here (opens in a new window)).
I rarely use destructuring when working with React Query. First of all, names like data and error are quite universal (purposefully so), so you’ll likely rename them anyway. Keeping the whole object will keep the context of what data it is or where the error is coming from. It will further help TypeScript to narrow types when using the status field or one of the status booleans, which it cannot do if you use destructuring:
const { data, isSuccess } = useGroups()if (isSuccess) { // 🚨 data will still be `Group[] | undefined` here}
const groupsQuery = useGroups()if (groupsQuery.isSuccess) { // ✅ groupsQuery.data will now be `Group[]`}This has nothing to do with React Query, it is just how TypeScript works. @danvdk (opens in a new window) has a good explanation for this behaviour

The comment from @TkDodo is exactly right, TypeScript does refinement on the types of individual symbols. Once you split them apart, it can’t keep track of the relationship any more. Doing this in general would be computationally hard. It can also be hard for people.
I’ve expressed my ♥️ for the enabled option right from the start, but it can be a bit tricky on type level if you want to use it for dependent queries (opens in a new window) and disable your query for as long as some parameters are not yet defined:
function fetchGroup(id: number): Promise<Group> { return axios.get(`group/${id}`).then((response) => response.data)}
function useGroup(id: number | undefined) { return useQuery({ queryKey: ['group', id], queryFn: () => fetchGroup(id), enabled: Boolean(id), }) // 🚨 Argument of type 'number | undefined' is not assignable to parameter of type 'number'. // Type 'undefined' is not assignable to type 'number'.ts(2345)}Technically, TypeScript is right, id is possibly undefined: the enabled option does not perform any type narrowing. Also, there are ways to bypass the enabled option, for example by calling the refetch method returned from useQuery. In that case, the id might really be undefined.
I’ve found the best way to go here, if you don’t like the non-null assertion operator (opens in a new window), is to accept that id can be undefined and reject the Promise in the queryFn. It’s a bit of duplication, but it’s also explicit and safe:
function fetchGroup(id: number | undefined): Promise<Group> { // ✅ check id at runtime because it can be `undefined` return typeof id === 'undefined' ? Promise.reject(new Error('Invalid id')) : axios.get(`group/${id}`).then((response) => response.data)}
function useGroup(id: number | undefined) { return useQuery({ queryKey: ['group', id], queryFn: () => fetchGroup(id), enabled: Boolean(id), })}Getting optimistic updates right in TypeScript is not an easy feat, so we’ve decided to add it as a comprehensive example (opens in a new window) to the docs.
The important part is: You have to explicitly type the variables argument passed to onMutate in order to get the best type inference. I don’t fully comprehend why that is, but it again seems to have something to do with inference of Generics. Have a look at this comment (opens in a new window) for more information.
For the most parts, typing useInfiniteQuery is no different from typing useQuery. One noticeable gotcha is that the pageParam value, which is passed to the queryFn, is typed as any. Could be improved in the library for sure, but as long as it’s any, it’s probably best to explicitly annotate it:
type GroupResponse = { next?: number; groups: Group[] }const queryInfo = useInfiniteQuery({ queryKey: ['groups'], // ⚠️ explicitly type pageParam to override `any` queryFn: ({ pageParam = 0, }: { pageParam: GroupResponse['next'] }) => fetchGroups(groups, pageParam), getNextPageParam: (lastGroup) => lastGroup.next,})If fetchGroups returns a GroupResponse, lastGroup will have its type nicely inferred, and we can use the same type to annotate pageParam.
I am personally not using a defaultQueryFn (opens in a new window), but I know many people are. It’s a neat way to leverage the passed queryKey to directly build your request url. If you inline the function when creating the queryClient, the type of the passed QueryFunctionContext will also be inferred for you. TypeScript is just so much better when you inline stuff :)
const queryClient = new QueryClient({ defaultOptions: { queries: { queryFn: async ({ queryKey: [url] }) => { const { data } = await axios.get(`${baseUrl}/${url}`) return data }, }, },})This just works, however, url is inferred to type unknown, because the whole queryKey is an unknown Array. At the time of the creation of the queryClient, there is absolutely no guarantee how the queryKeys will be constructed when calling useQuery, so there is only so much React Query can do. That is just the nature of this highly dynamic feature. It’s not a bad thing though because it means you now have to work defensively and narrow the type with runtime checks to work with it, for example:
const queryClient = new QueryClient({ defaultOptions: { queries: { queryFn: async ({ queryKey: [url] }) => { // ✅ narrow the type of url to string // so that we can work with it if (typeof url === 'string') { const { data } = await axios.get( `${baseUrl}/${url.toLowerCase()}` ) return data } throw new Error('Invalid QueryKey') }, }, },})I think this shows quite well why unknown is such a great (and underused) type compared to any. It has become my favourite type lately - but that is subject for another blog post. 😊
That’s it for today. Feel free to reach out to me on bluesky (opens in a new window) if you have any questions, or just leave a comment below. ⬇️