- User model with email/hashed_password/is_admin/notification_prefs - JWT auth: POST /api/auth/register, /login, /me - First registered user auto-promoted to admin - Migration 0005: users table + user_id FK on follows (clears global follows) - Follows, dashboard, settings, admin endpoints all require authentication - Admin endpoints (settings writes, celery triggers) require is_admin - Frontend: login/register pages, Zustand auth store (localStorage persist) - AuthGuard component gates all app routes, shows app shell only when authed - Sidebar shows user email + logout; Admin nav link visible to admins only - Admin panel (/settings): user list with delete + promote/demote, LLM config, data source settings, and manual celery controls Authored-By: Jack Levy
28 lines
573 B
TypeScript
28 lines
573 B
TypeScript
import { create } from "zustand";
|
|
import { persist } from "zustand/middleware";
|
|
|
|
interface AuthUser {
|
|
id: number;
|
|
email: string;
|
|
is_admin: boolean;
|
|
}
|
|
|
|
interface AuthState {
|
|
token: string | null;
|
|
user: AuthUser | null;
|
|
setAuth: (token: string, user: AuthUser) => void;
|
|
logout: () => void;
|
|
}
|
|
|
|
export const useAuthStore = create<AuthState>()(
|
|
persist(
|
|
(set) => ({
|
|
token: null,
|
|
user: null,
|
|
setAuth: (token, user) => set({ token, user }),
|
|
logout: () => set({ token: null, user: null }),
|
|
}),
|
|
{ name: "pocketveto-auth" }
|
|
)
|
|
);
|