UIKit/src/api/makeRequest.ts

97 lines
2.6 KiB
TypeScript
Raw Normal View History

2023-06-11 09:55:06 +00:00
import { create } from "zustand";
import { persist } from "zustand/middleware";
2023-06-06 10:02:17 +00:00
import axios, { AxiosResponse, Method } from "axios";
2023-06-11 09:55:06 +00:00
interface AuthStore {
token: string;
2023-06-06 10:02:17 +00:00
}
2023-06-11 09:55:06 +00:00
export const useAuthStore = create<AuthStore>()(
persist(
(set, get) => ({
token: "",
}),
{
name: "token",
}
)
);
export const getAuthToken = () => useAuthStore.getState().token;
export const setAuthToken = (token: string) => useAuthStore.setState({ token });
export const clearAuthToken = () => useAuthStore.setState({ token: "" });
export function useToken() {
return useAuthStore(state => state.token);
}
export async function makeRequest<TRequest = unknown, TResponse = unknown>({
2023-06-06 10:02:17 +00:00
method = "post",
url,
body,
useToken = true,
contentType = false,
signal,
withCredentials,
2023-06-11 09:55:06 +00:00
}: {
method?: Method;
url: string;
body?: TRequest;
/** Send access token */
useToken?: boolean;
contentType?: boolean;
signal?: AbortSignal;
/** Send refresh token */
withCredentials?: boolean;
}): Promise<TResponse> {
2023-06-06 10:02:17 +00:00
const headers: Record<string, string> = {};
2023-06-11 09:55:06 +00:00
if (useToken) headers["Authorization"] = `Bearer ${getAuthToken()}`;
2023-06-06 10:02:17 +00:00
if (contentType) headers["Content-Type"] = "application/json";
try {
const response = await axios<TRequest, AxiosResponse<TResponse & { accessToken?: string; }>>({
url,
method,
headers,
data: body,
signal,
withCredentials,
});
if (response.data?.accessToken) {
2023-06-11 09:55:06 +00:00
setAuthToken(response.data.accessToken);
2023-06-06 10:02:17 +00:00
}
return response.data;
} catch (error) {
if (axios.isAxiosError(error) && error.response?.status === 401 && !withCredentials) {
2023-06-11 09:55:06 +00:00
const refreshResponse = await refresh(getAuthToken());
if (refreshResponse.data?.accessToken) setAuthToken(refreshResponse.data.accessToken);
2023-06-06 10:02:17 +00:00
headers["Authorization"] = refreshResponse.data.accessToken;
const response = await axios.request<TRequest, AxiosResponse<TResponse>>({
url,
method,
headers,
data: body,
signal,
});
return response.data;
}
throw error;
}
};
function refresh(token: string) {
return axios<never, AxiosResponse<{ accessToken: string; }>>("https://admin.pena.digital/auth/refresh", {
headers: {
"Authorization": token,
"Content-Type": "application/json",
},
});
}