aboutsummaryrefslogtreecommitdiffhomepage
path: root/frontend/src/contexts/AuthContext.tsx
diff options
context:
space:
mode:
authornsfisis <nsfisis@gmail.com>2025-11-06 04:10:55 +0900
committernsfisis <nsfisis@gmail.com>2025-11-08 05:04:02 +0900
commite0cc2915f22fe74d5be9e8f51d6b73437192e0ba (patch)
treee2e27e3cba8b5e7205732eef7b6df9789e83396f /frontend/src/contexts/AuthContext.tsx
parentba1e0c904f810193f25d4f88cc2bb168f1d625fe (diff)
downloadfeedaka-e0cc2915f22fe74d5be9e8f51d6b73437192e0ba.tar.gz
feedaka-e0cc2915f22fe74d5be9e8f51d6b73437192e0ba.tar.zst
feedaka-e0cc2915f22fe74d5be9e8f51d6b73437192e0ba.zip
feat: Support multi-user
Diffstat (limited to 'frontend/src/contexts/AuthContext.tsx')
-rw-r--r--frontend/src/contexts/AuthContext.tsx97
1 files changed, 97 insertions, 0 deletions
diff --git a/frontend/src/contexts/AuthContext.tsx b/frontend/src/contexts/AuthContext.tsx
new file mode 100644
index 0000000..fc6b237
--- /dev/null
+++ b/frontend/src/contexts/AuthContext.tsx
@@ -0,0 +1,97 @@
+import {
+ createContext,
+ type ReactNode,
+ useContext,
+ useEffect,
+ useState,
+} from "react";
+import { useMutation, useQuery } from "urql";
+import {
+ GetCurrentUserDocument,
+ LoginDocument,
+ LogoutDocument,
+} from "../graphql/generated/graphql";
+
+interface User {
+ id: string;
+ username: string;
+}
+
+interface AuthContextType {
+ user: User | null;
+ isLoading: boolean;
+ login: (username: string, password: string) => Promise<boolean>;
+ logout: () => Promise<void>;
+}
+
+const AuthContext = createContext<AuthContextType | undefined>(undefined);
+
+export function AuthProvider({ children }: { children: ReactNode }) {
+ const [user, setUser] = useState<User | null>(null);
+ const [isLoading, setIsLoading] = useState(true);
+
+ const [, executeLogin] = useMutation(LoginDocument);
+ const [, executeLogout] = useMutation(LogoutDocument);
+ const [currentUserResult, reexecuteCurrentUser] = useQuery({
+ query: GetCurrentUserDocument,
+ });
+
+ // Update user from CurrentUser query
+ useEffect(() => {
+ if (currentUserResult.data?.currentUser) {
+ setUser(currentUserResult.data.currentUser);
+ } else {
+ setUser(null);
+ }
+ if (!currentUserResult.fetching) {
+ setIsLoading(false);
+ }
+ }, [currentUserResult.data, currentUserResult.fetching]);
+
+ const login = async (
+ username: string,
+ password: string,
+ ): Promise<boolean> => {
+ try {
+ const result = await executeLogin({ username, password });
+
+ if (result.data?.login?.user) {
+ setUser(result.data.login.user);
+ // Refetch CurrentUser query to ensure session is established
+ reexecuteCurrentUser({ requestPolicy: "network-only" });
+ return true;
+ }
+
+ return false;
+ } catch (error) {
+ console.error("Login failed:", error);
+ return false;
+ }
+ };
+
+ const logout = async () => {
+ try {
+ await executeLogout({});
+ } catch (error) {
+ console.error("Logout failed:", error);
+ } finally {
+ setUser(null);
+ // Refetch CurrentUser query to ensure session is cleared
+ reexecuteCurrentUser({ requestPolicy: "network-only" });
+ }
+ };
+
+ return (
+ <AuthContext.Provider value={{ user, isLoading, login, logout }}>
+ {children}
+ </AuthContext.Provider>
+ );
+}
+
+export function useAuth() {
+ const context = useContext(AuthContext);
+ if (context === undefined) {
+ throw new Error("useAuth must be used within an AuthProvider");
+ }
+ return context;
+}