-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathclient.ts
92 lines (83 loc) · 3.22 KB
/
client.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
*
* Copyright Oxide Computer Company
*/
import {
QueryClient as QueryClientOrig,
useQuery,
type UseQueryOptions,
} from '@tanstack/react-query'
import { Api } from './__generated__/Api'
import { type ApiError } from './errors'
import {
ensurePrefetched,
getApiQueryOptions,
getListQueryOptionsFn,
getUseApiMutation,
getUseApiQuery,
getUseApiQueryErrorsAllowed,
getUsePrefetchedApiQuery,
wrapQueryClient,
} from './hooks'
export const api = new Api({
// unit tests run in Node, whose fetch implementation requires a full URL
host: process.env.NODE_ENV === 'test' ? 'http://testhost' : '',
})
export type ApiMethods = typeof api.methods
/** API-specific query options helper. */
export const apiq = getApiQueryOptions(api.methods)
/**
* Query options helper that only supports list endpoints. Returns
* a function `(limit, pageToken) => QueryOptions` for use with
* `useQueryTable`.
*/
export const getListQFn = getListQueryOptionsFn(api.methods)
export const useApiQuery = getUseApiQuery(api.methods)
/**
* Same as `useApiQuery`, except we use `invariant(data)` to ensure the data is
* already there in the cache at request time, which means it has been
* prefetched in a loader. Whenever this hook is used, there should be an e2e
* test loading the page to exercise the invariant in CI.
*/
export const usePrefetchedApiQuery = getUsePrefetchedApiQuery(api.methods)
export const useApiQueryErrorsAllowed = getUseApiQueryErrorsAllowed(api.methods)
export const useApiMutation = getUseApiMutation(api.methods)
export const usePrefetchedQuery = <TData>(options: UseQueryOptions<TData, ApiError>) =>
ensurePrefetched(useQuery(options), options.queryKey)
/**
* Extends React Query's `QueryClient` with a couple of API-specific methods.
* Existing methods are never modified.
*/
class QueryClient extends QueryClientOrig {
/**
* Invalidate all cached queries for a given endpoint.
*
* Note that we only take a single argument, `method`, rather than allowing
* the full query key `[query, params]` to be specified. This is to avoid
* accidentally overspecifying and therefore failing to match the desired query.
* The params argument can be added in if we ever have a use case for it.
*/
invalidateEndpoint(method: keyof typeof api.methods) {
this.invalidateQueries({ queryKey: [method] })
}
}
// Needs to be defined here instead of in app so we can use it to define
// `apiQueryClient`, which provides API-typed versions of QueryClient methods
export const queryClient = new QueryClient({
defaultOptions: {
queries: {
retry: false,
staleTime: 2000,
refetchOnWindowFocus: false,
},
},
})
// to be used in loaders, which are outside the component tree and therefore
// don't have access to context
export const apiQueryClient = wrapQueryClient(api.methods, queryClient)
// used to retrieve the typed query client in components. doesn't need to exist:
// we could import apiQueryClient directly everywhere, but the change is noisy
export const useApiQueryClient = () => apiQueryClient