mirror of
https://github.com/MODSetter/SurfSense.git
synced 2026-06-30 21:59:46 +02:00
Merge pull request #584 from CREDO23/feat/add-jotai-tanstack-user
[Feat] User | Add jotai & tanstack
This commit is contained in:
commit
fc7c4bc58e
9 changed files with 57 additions and 63 deletions
|
|
@ -35,10 +35,10 @@ import {
|
||||||
} from "@/components/ui/card";
|
} from "@/components/ui/card";
|
||||||
import { Spotlight } from "@/components/ui/spotlight";
|
import { Spotlight } from "@/components/ui/spotlight";
|
||||||
import { Tilt } from "@/components/ui/tilt";
|
import { Tilt } from "@/components/ui/tilt";
|
||||||
import { useUser } from "@/hooks";
|
|
||||||
import { searchSpacesAtom } from "@/atoms/search-spaces/search-space-query.atoms";
|
import { searchSpacesAtom } from "@/atoms/search-spaces/search-space-query.atoms";
|
||||||
import { deleteSearchSpaceMutationAtom } from "@/atoms/search-spaces/search-space-mutation.atoms";
|
import { deleteSearchSpaceMutationAtom } from "@/atoms/search-spaces/search-space-mutation.atoms";
|
||||||
import { authenticatedFetch } from "@/lib/auth-utils";
|
import { authenticatedFetch } from "@/lib/auth-utils";
|
||||||
|
import { currentUserAtom } from "@/atoms/user/user-query.atoms";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Formats a date string into a readable format
|
* Formats a date string into a readable format
|
||||||
|
|
@ -159,8 +159,7 @@ const DashboardPage = () => {
|
||||||
const { data: searchSpaces = [], isLoading: loading, error, refetch: refreshSearchSpaces } = useAtomValue(searchSpacesAtom);
|
const { data: searchSpaces = [], isLoading: loading, error, refetch: refreshSearchSpaces } = useAtomValue(searchSpacesAtom);
|
||||||
const { mutateAsync: deleteSearchSpace } = useAtomValue(deleteSearchSpaceMutationAtom);
|
const { mutateAsync: deleteSearchSpace } = useAtomValue(deleteSearchSpaceMutationAtom);
|
||||||
|
|
||||||
// Fetch user details
|
const { data: user, isPending: isLoadingUser, error: userError } = useAtomValue(currentUserAtom);
|
||||||
const { user, loading: isLoadingUser, error: userError } = useUser();
|
|
||||||
|
|
||||||
// Create user object for UserDropdown
|
// Create user object for UserDropdown
|
||||||
const customUser = {
|
const customUser = {
|
||||||
|
|
|
||||||
13
surfsense_web/atoms/user/user-query.atoms.ts
Normal file
13
surfsense_web/atoms/user/user-query.atoms.ts
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
import { atomWithQuery } from "jotai-tanstack-query";
|
||||||
|
import { userApiService } from "@/lib/apis/user-api.service";
|
||||||
|
import { cacheKeys } from "@/lib/query-client/cache-keys";
|
||||||
|
|
||||||
|
export const currentUserAtom = atomWithQuery(() => {
|
||||||
|
return {
|
||||||
|
queryKey: cacheKeys.user.current(),
|
||||||
|
staleTime: 5 * 60 * 1000, // 5 minutes
|
||||||
|
queryFn: async () => {
|
||||||
|
return userApiService.getMe();
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
@ -17,10 +17,10 @@ import {
|
||||||
DialogHeader,
|
DialogHeader,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
} from "@/components/ui/dialog";
|
} from "@/components/ui/dialog";
|
||||||
import { useUser } from "@/hooks";
|
|
||||||
import { useQuery } from "@tanstack/react-query";
|
import { useQuery } from "@tanstack/react-query";
|
||||||
import { searchSpacesApiService } from "@/lib/apis/search-spaces-api.service";
|
import { searchSpacesApiService } from "@/lib/apis/search-spaces-api.service";
|
||||||
import { cacheKeys } from "@/lib/query-client/cache-keys";
|
import { cacheKeys } from "@/lib/query-client/cache-keys";
|
||||||
|
import { currentUserAtom } from "@/atoms/user/user-query.atoms";
|
||||||
|
|
||||||
interface AppSidebarProviderProps {
|
interface AppSidebarProviderProps {
|
||||||
searchSpaceId: string;
|
searchSpaceId: string;
|
||||||
|
|
@ -68,7 +68,7 @@ export function AppSidebarProvider({
|
||||||
enabled: !!searchSpaceId,
|
enabled: !!searchSpaceId,
|
||||||
});
|
});
|
||||||
|
|
||||||
const { user } = useUser();
|
const { data: user } = useAtomValue(currentUserAtom);
|
||||||
|
|
||||||
const [showDeleteDialog, setShowDeleteDialog] = useState(false);
|
const [showDeleteDialog, setShowDeleteDialog] = useState(false);
|
||||||
const [chatToDelete, setChatToDelete] = useState<{ id: number; name: string } | null>(null);
|
const [chatToDelete, setChatToDelete] = useState<{ id: number; name: string } | null>(null);
|
||||||
|
|
|
||||||
|
|
@ -38,7 +38,8 @@ import {
|
||||||
DropdownMenuSeparator,
|
DropdownMenuSeparator,
|
||||||
DropdownMenuTrigger,
|
DropdownMenuTrigger,
|
||||||
} from "@/components/ui/dropdown-menu";
|
} from "@/components/ui/dropdown-menu";
|
||||||
import { useUser } from "@/hooks/use-user";
|
import { useAtomValue } from "jotai";
|
||||||
|
import { currentUserAtom } from "@/atoms/user/user-query.atoms";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generates a consistent color based on a string (email)
|
* Generates a consistent color based on a string (email)
|
||||||
|
|
@ -262,7 +263,7 @@ export const AppSidebar = memo(function AppSidebar({
|
||||||
}: AppSidebarProps) {
|
}: AppSidebarProps) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const { theme, setTheme } = useTheme();
|
const { theme, setTheme } = useTheme();
|
||||||
const { user, loading: isLoadingUser } = useUser();
|
const { data: user, isPending: isLoadingUser } = useAtomValue(currentUserAtom);
|
||||||
const [isClient, setIsClient] = useState(false);
|
const [isClient, setIsClient] = useState(false);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
|
|
||||||
19
surfsense_web/contracts/types/user.types.ts
Normal file
19
surfsense_web/contracts/types/user.types.ts
Normal file
|
|
@ -0,0 +1,19 @@
|
||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
export const user = z.object({
|
||||||
|
id: z.string().uuid(),
|
||||||
|
email: z.string().email(),
|
||||||
|
is_active: z.boolean(),
|
||||||
|
is_superuser: z.boolean(),
|
||||||
|
is_verified: z.boolean(),
|
||||||
|
pages_limit: z.number(),
|
||||||
|
pages_used: z.number(),
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get current user
|
||||||
|
*/
|
||||||
|
export const getMeResponse = user;
|
||||||
|
|
||||||
|
export type User = z.infer<typeof user>;
|
||||||
|
export type GetMeResponse = z.infer<typeof getMeResponse>;
|
||||||
|
|
@ -1,4 +1,3 @@
|
||||||
export * from "./use-logs";
|
export * from "./use-logs";
|
||||||
export * from "./use-rbac";
|
export * from "./use-rbac";
|
||||||
export * from "./use-search-source-connectors";
|
export * from "./use-search-source-connectors";
|
||||||
export * from "./use-user";
|
|
||||||
|
|
|
||||||
|
|
@ -1,53 +0,0 @@
|
||||||
"use client";
|
|
||||||
|
|
||||||
import { useEffect, useState } from "react";
|
|
||||||
import { toast } from "sonner";
|
|
||||||
import { authenticatedFetch } from "@/lib/auth-utils";
|
|
||||||
|
|
||||||
interface User {
|
|
||||||
id: string;
|
|
||||||
email: string;
|
|
||||||
is_active: boolean;
|
|
||||||
is_superuser: boolean;
|
|
||||||
is_verified: boolean;
|
|
||||||
pages_limit: number;
|
|
||||||
pages_used: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useUser() {
|
|
||||||
const [user, setUser] = useState<User | null>(null);
|
|
||||||
const [loading, setLoading] = useState(true);
|
|
||||||
const [error, setError] = useState<string | null>(null);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const fetchUser = async () => {
|
|
||||||
try {
|
|
||||||
// Only run on client-side
|
|
||||||
if (typeof window === "undefined") return;
|
|
||||||
|
|
||||||
setLoading(true);
|
|
||||||
const response = await authenticatedFetch(
|
|
||||||
`${process.env.NEXT_PUBLIC_FASTAPI_BACKEND_URL}/users/me`,
|
|
||||||
{ method: "GET" }
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error(`Failed to fetch user: ${response.status}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = await response.json();
|
|
||||||
setUser(data);
|
|
||||||
setError(null);
|
|
||||||
} catch (err: any) {
|
|
||||||
setError(err.message || "Failed to fetch user");
|
|
||||||
console.error("Error fetching user:", err);
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
fetchUser();
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return { user, loading, error };
|
|
||||||
}
|
|
||||||
13
surfsense_web/lib/apis/user-api.service.ts
Normal file
13
surfsense_web/lib/apis/user-api.service.ts
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
import { getMeResponse } from "@/contracts/types/user.types";
|
||||||
|
import { baseApiService } from "./base-api.service";
|
||||||
|
|
||||||
|
class UserApiService {
|
||||||
|
/**
|
||||||
|
* Get current authenticated user
|
||||||
|
*/
|
||||||
|
getMe = async () => {
|
||||||
|
return baseApiService.get(`/users/me`, getMeResponse);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export const userApiService = new UserApiService();
|
||||||
|
|
@ -40,5 +40,8 @@ export const cacheKeys = {
|
||||||
["search-spaces", ...(queries ? Object.values(queries) : [])] as const,
|
["search-spaces", ...(queries ? Object.values(queries) : [])] as const,
|
||||||
detail: (searchSpaceId: string) => ["search-spaces", searchSpaceId] as const,
|
detail: (searchSpaceId: string) => ["search-spaces", searchSpaceId] as const,
|
||||||
communityPrompts: ["search-spaces", "community-prompts"] as const,
|
communityPrompts: ["search-spaces", "community-prompts"] as const,
|
||||||
}
|
},
|
||||||
};
|
user: {
|
||||||
|
current: () => ["user", "me"] as const,
|
||||||
|
},
|
||||||
|
};
|
||||||
Loading…
Add table
Add a link
Reference in a new issue