-
Notifications
You must be signed in to change notification settings - Fork 367
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #10292 from linode/release-v1.115.0
Release v1.115.0 - release → staging
- Loading branch information
Showing
339 changed files
with
6,223 additions
and
2,587 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,5 +1,5 @@ | ||
module.exports = { | ||
root: true, | ||
ignorePatterns: ["**/node_modules/", "**/build/"], | ||
parser: "@babel/eslint-parser", | ||
parser: "@typescript-eslint/parser", | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,135 @@ | ||
# React Query | ||
|
||
[TanStack Query](https://tanstack.com/query/latest) (formerly React Query) is Cloud Manager's primary tool for fetching and caching API data. For a quick introduction, read our [Fetching Data](../development-guide/05-fetching-data.md#react-query) development guide. | ||
|
||
## Query Keys | ||
|
||
React Query's cache is a simple key-value store. Query Keys are serializable strings that uniquely identify a query's data in the cache. You can read more about the concept [here](https://tanstack.com/query/latest/docs/framework/react/guides/query-keys) in the TanStack Query docs. | ||
|
||
Because of Cloud Manager's complexity, we use [`@lukemorales/query-key-factory`](https://github.com/lukemorales/query-key-factory) to manage our query keys. This package allows us to define query key _factories_ that enable typesafe standardized query keys that can be reused and referenced throughout the application. | ||
|
||
### Examples | ||
|
||
#### Simple Query | ||
|
||
```ts | ||
import { useQuery } from "@tanstack/react-query"; | ||
import { getProfile } from "@linode/api-v4"; | ||
import type { APIError, Profile } from "@linode/api-v4"; | ||
|
||
const profileQueries = createQueryKeys('profile', { | ||
profile: { | ||
queryFn: getProfile, | ||
queryKey: null, | ||
}, | ||
}); | ||
|
||
export const useProfile = () => | ||
useQuery<Profile, APIError[]>(profileQueries.profile); | ||
``` | ||
|
||
#### Query with parameters | ||
|
||
> [!important] | ||
> Queries that have parameters should always include the parameters in the `queryKey` | ||
```ts | ||
import { useQuery } from "@tanstack/react-query"; | ||
import { getLinode } from "@linode/api-v4"; | ||
import type { APIError, Linode } from "@linode/api-v4"; | ||
|
||
const linodeQueries = createQueryKeys('linodes', { | ||
linode: (id: number) => ({ | ||
queryFn: () => getLinode(id), | ||
queryKey: [id], | ||
}), | ||
}); | ||
|
||
export const useLinodeQuery = (id: number) => | ||
useQuery<Linode, APIError[]>(linodeQueries.linode(1)); | ||
``` | ||
|
||
## Maintaining the Cache | ||
|
||
> A significant challenge of React Query is keeping the client state in sync with the server. | ||
The two easiest ways of updating the cache using React Query are | ||
- Using `invalidateQueries` to mark data as stale (which will trigger a refetch the next time the query is mounted) | ||
- Using `setQueryData` to manually update the cache | ||
|
||
### `invalidateQueries` | ||
|
||
This will mark data as stale in the React Query cache, which will cause Cloud Manager to refetch the data if the corresponding query is mounted. | ||
|
||
Use `invalidateQueries` when: | ||
- You are dealing with any *paginated data* (because order may have changed) | ||
- You want fresh data from the API | ||
|
||
> [!note] | ||
> When using `invalidateQueries`, use a query key factory to ensure you are invalidating the data at the correct query key. | ||
### `setQueryData` | ||
|
||
Use this if you have data readily available to put in the cache. This often happens when you make a PUT request. | ||
|
||
### Example | ||
|
||
This example shows how we keep the cache up to date when performing create / update / delete operations | ||
on an entity. | ||
|
||
```ts | ||
import { useQuery, useMutation } from "@tanstack/react-query"; | ||
import { getLinode, getLinodes, updateLinode, deleteLinode, createLinode } from "@linode/api-v4"; | ||
import type { APIError, Linode, ResourcePage } from "@linode/api-v4"; | ||
|
||
const linodeQueries = createQueryKeys('linodes', { | ||
linode: (id: number) => ({ | ||
queryFn: () => getLinode(id), | ||
queryKey: [id], | ||
}), | ||
linodes: (params: Params = {}, filter: Filter = {}) => ({ | ||
queryFn: () => getLinodes(params, filter), | ||
queryKey: [params, filter], | ||
}), | ||
}); | ||
|
||
export const useLinodeQuery = (id: number) => | ||
useQuery<Linode, APIError[]>(linodeQueries.linode(1)); | ||
|
||
export const useLinodeUpdateMutation = (id: number) => { | ||
const queryClient = useQueryClient(); | ||
return useMutation<Linode, APIError[], Partial<Linode>>({ | ||
mutationFn: (data) => updateLinode(id, data), | ||
onSuccess(linode) { | ||
// Invalidate all paginated pages in the cache. | ||
queryClient.invalidateQueries(linodeQueries.linodes._def); | ||
// Because we have the updated Linode, we can manually set the cache for the `useLinode` query. | ||
queryClient.setQueryData(linodeQueries.linode(id).queryKey, linode); | ||
}, | ||
}); | ||
} | ||
|
||
export const useDeleteLinodeMutation = (id: number) => { | ||
const queryClient = useQueryClient(); | ||
return useMutation<{}, APIError[]>({ | ||
mutationFn: () => deleteLinode(id), | ||
onSuccess() { | ||
queryClient.removeQueries(linodeQueries.linode(id).queryKey); | ||
queryClient.invalidateQueries(linodeQueries.linodes._def); | ||
}, | ||
}); | ||
}; | ||
|
||
export const useCreateLinodeMutation = () => { | ||
const queryClient = useQueryClient(); | ||
return useMutation<Linode, APIError[], CreateLinodeRequest>({ | ||
mutationFn: createLinode, | ||
onSuccess(linode) { | ||
// Invalidate all paginated pages in the cache. We don't know what page the new Linode will be on. | ||
queryClient.invalidateQueries(linodeQueries.linodes._def); | ||
// Because we have the new Linode, we can manually set the cache for the `useLinode` query. | ||
queryClient.setQueryData(linodeQueries.linode(id).queryKey, linode); | ||
}, | ||
}); | ||
} | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.