1
Fork 0
mirror of https://github.com/Steffo99/festa.git synced 2025-01-09 23:39:44 +00:00
festa/components/auth/requests.tsx
2022-07-19 20:31:25 +02:00

81 lines
2.4 KiB
TypeScript

import { AxiosInstance, AxiosRequestConfig, AxiosResponse, default as axios } from "axios";
import { AuthContext, AuthContextContents } from "./base";
import { ReactNode, useCallback, useContext } from "react";
import { usePromise } from "../generic/loading/promise";
import { default as useSWR, SWRConfig } from "swr"
/**
* Hook which creates an {@link AxiosInstance} based on the current {@link AuthContext}, using the appropriate `Authentication` header the context is non-null.
*/
export function useAxios(): AxiosInstance {
const authContext = useContext(AuthContext)
let auth: AuthContextContents | undefined = authContext?.[0]
return axios.create({
headers: {
Authorization: auth ? `Bearer ${auth.token}` : false,
},
})
}
/**
* Hook which returns the callback to use as fetcher in {@link useSWR} calls.
*
* Preferably set through {@link SWRConfig}.
*/
export function useAxiosSWRFetcher() {
const axios = useAxios()
return useCallback(
async (resource: string, init: AxiosRequestConfig<any>) => {
const response = await axios.get(resource, init)
return response.data
},
[axios]
)
}
export type AxiosSWRFetcherProviderProps = {
children: ReactNode,
}
/**
* Component which provides the fetcher to {@link useSWR} via {@link useAxiosSWRFetcher}.
*/
export const AxiosSWRFetcherProvider = ({ children }: AxiosSWRFetcherProviderProps) => {
return (
<SWRConfig
value={{
fetcher: useAxiosSWRFetcher(),
onError: (error, key) => {
throw error
},
}}
>
{children}
</SWRConfig>
)
}
/**
* Hook which uses {@link useAxios} to perform a single HTTP request with the given configuration, tracking the status of the request and any errors that may arise from it.
*/
export function useAxiosRequest<Output = any, Input = any>(config: AxiosRequestConfig<Input> = {}) {
const axios = useAxios()
const performRequest = useCallback(
async (funcConfig: AxiosRequestConfig<Input> = {}): Promise<AxiosResponse<Output, Input>> => {
return await axios.request({ ...config, ...funcConfig })
},
[axios, config]
)
const promiseHook = usePromise(performRequest)
return { ...promiseHook, data: promiseHook.result?.data }
}