Files
RaidBuilderWeb/src/providers/AuthProvider.tsx
2025-03-15 18:23:04 -04:00

190 lines
5.8 KiB
TypeScript

import { useGetTutorialsStatus, useUpdateTutorialsStatus } from "@/hooks/AccountHooks";
import { AccountPermission } from "@/interface/AccountPermission";
import { AccountTutorialStatus } from "@/interface/AccountTutorialStatus";
import { GamePermission } from "@/interface/GamePermission";
import { RaidGroupPermission } from "@/interface/RaidGroupPermission";
import { RaidGroupRequest } from "@/interface/RaidGroupRequest";
import { api } from "@/util/AxiosUtil";
import { createContext, useCallback, useContext, useEffect, useLayoutEffect, useMemo, useState } from "react";
import { Navigate, Outlet } from "react-router";
type AuthProviderProps = {
children: React.ReactNode;
jwtStorageKey?: string;
refreshTokenStorageKey?: string;
}
type AuthProviderState = {
jwt: string | null;
setJwt: (token: string | null) => void;
expiration: Date | null;
setExpiration: (expiration: Date | null) => void;
accountId: string | null;
accountPermissions: AccountPermission[];
raidGroupPermissions: RaidGroupPermission[];
gamePermissions: GamePermission[];
raidGroupRequests: RaidGroupRequest[];
tutorialsStatus: AccountTutorialStatus;
setTutorialsStatus: (tutorialsStatus: AccountTutorialStatus) => void;
}
const initialState: AuthProviderState = {
jwt: null,
setJwt: () => null,
expiration: null,
setExpiration: () => null,
accountId: null,
accountPermissions: [],
raidGroupPermissions: [],
gamePermissions: [],
raidGroupRequests: [],
tutorialsStatus: {} as AccountTutorialStatus,
setTutorialsStatus: () => null
}
const AuthContext = createContext<AuthProviderState>(initialState);
export function AuthProvider({
children
}: AuthProviderProps){
const [ jwt, setJwt ] = useState<string | null>(null);
const [ expiration, setExpiration ] = useState<Date | null>(null);
const [ firstFetch, setFirstFetch ] = useState(true);
const [ accountId, setAccountId ] = useState<string | null>(null);
const [ accountPermissions, setAccountPermissions ] = useState<AccountPermission[]>([]);
const [ raidGroupPermissions, setRaidGroupPermissions ] = useState<RaidGroupPermission[]>([]);
const [ gamePermissions, setGamePermissions ] = useState<GamePermission[]>([]);
const [ raidGroupRequests, setRaidGroupRequests ] = useState<RaidGroupRequest[]>([]);
const [ tutorialsStatus, setTutorialsStatus ] = useState<AccountTutorialStatus>({} as AccountTutorialStatus);
const tutorialsStatusQuery = useGetTutorialsStatus(accountId);
const { mutate: tutorialsStatusMutation } = useUpdateTutorialsStatus();
const fetchToken = useCallback(async () => {
//console.log("Fetching token");
try{
const response = await api.get("/auth/refresh");
//If the token is retrieved
if((response.status === 200) && (!response.data.errors)){
setJwt(response.data.token);
const decodedToken = JSON.parse(atob(response.data.token.split(".")[1]));
//console.log("decodedToken = ");
//console.log(decodedToken);
setExpiration(new Date(decodedToken.exp * 1000));
setFirstFetch(false);
setAccountId(decodedToken.accountId);
setAccountPermissions(JSON.parse(decodedToken.accountPermissions));
setRaidGroupPermissions(JSON.parse(decodedToken.raidGroupPermissions));
setGamePermissions(JSON.parse(decodedToken.gamePermissions));
setRaidGroupRequests(JSON.parse(decodedToken.raidGroupRequests));
return response.data.token;
}
//If the token cannot be retrieved
else{
setJwt(null);
setExpiration(null);
setFirstFetch(false);
}
}
//If the token cannot be retrieved
catch{
setJwt(null);
setExpiration(null);
setFirstFetch(false);
}
}, [ setJwt, setExpiration, setFirstFetch ]);
//Add the token to all queries
useLayoutEffect(() => {
const authInterceptor = api.interceptors.request.use(async (config) => {
if(config.url?.endsWith("/auth/refresh")){
return config;
}
let currentJwt = jwt;
if((expiration) && (expiration < new Date()) && (!config.url?.endsWith("/auth/refresh"))){
currentJwt = await fetchToken();
config.headers.Authorization = jwt ? `Bearer ${currentJwt}` : config.headers.Authorization;
}
config.headers.Authorization = jwt ? `Bearer ${currentJwt}` : config.headers.Authorization;
return config;
});
return () => { api.interceptors.request.eject(authInterceptor); };
}, [ jwt, expiration, fetchToken ]);
//Try to get the token on page load
useEffect(() => {
fetchToken();
}, [ fetchToken ]);
//Update the tutorial status when fetched
useEffect(() => {
if(tutorialsStatusQuery.status === "success"){
setTutorialsStatus(tutorialsStatusQuery.data);
}
}, [ tutorialsStatusQuery.status, tutorialsStatusQuery.data ]);
const updateTutorialsStatus = useCallback((newTutorialsStatus: AccountTutorialStatus) => {
setTutorialsStatus(newTutorialsStatus);
tutorialsStatusMutation(newTutorialsStatus);
}, [ setTutorialsStatus, tutorialsStatusMutation ]);
const currentTokens = useMemo(() => ({
jwt, setJwt,
expiration, setExpiration,
accountId,
accountPermissions,
raidGroupPermissions,
gamePermissions,
raidGroupRequests,
tutorialsStatus, setTutorialsStatus: updateTutorialsStatus
}), [ jwt, expiration, accountId, accountPermissions, raidGroupPermissions, gamePermissions, raidGroupRequests, tutorialsStatus, updateTutorialsStatus ]);
//TODO: Return a spinner while the first token is being fetched
if(firstFetch){
return (
<main
className="text-4xl"
>
Loading...
</main>
);
}
return (
<AuthContext.Provider value={currentTokens}>
{children}
</AuthContext.Provider>
);
}
export function ProtectedRoute(){
const { jwt } = useAuth();
if(!jwt){
return <Navigate to="/login"/>;
}
return <Outlet/>;
}
export const useAuth = () => {
const context = useContext(AuthContext);
if(context === undefined){
throw new Error("useAuth must be used within an AuthProvider");
}
return context;
}