import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query"; import type { AxiosError } from "axios"; import { Activity, BookOpenText, Copy, Laptop, Plus, Search, ServerCog, ShieldHalf, } from "lucide-react"; import { Link, useNavigate } from "react-router-dom"; import { Avatar, AvatarFallback, AvatarImage, } from "../../components/ui/avatar"; import { Badge } from "../../components/ui/badge"; import { Button } from "../../components/ui/button"; import { Card, CardContent, CardDescription, CardHeader, CardTitle, } from "../../components/ui/card"; import { Input } from "../../components/ui/input"; import { Separator } from "../../components/ui/separator"; import { Switch } from "../../components/ui/switch"; import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow, } from "../../components/ui/table"; import { deleteClient, fetchClients, updateClientStatus, } from "../../lib/devApi"; import { cn } from "../../lib/utils"; import { CopyButton } from "../../components/ui/copy-button"; import { toast } from "../../components/ui/use-toast"; function ClientsPage() { const navigate = useNavigate(); const queryClient = useQueryClient(); const { data, isLoading, error } = useQuery({ queryKey: ["clients"], queryFn: fetchClients, }); const updateStatusMutation = useMutation({ mutationFn: (payload: { id: string; status: "active" | "inactive" }) => updateClientStatus(payload.id, payload.status), onSuccess: (_, variables) => { const statusText = variables.status === "active" ? "활성화" : "비활성화"; toast(`클라이언트가 ${statusText}되었습니다.`); queryClient.invalidateQueries({ queryKey: ["clients"] }); }, onError: (error: AxiosError<{ error?: string }>) => { const errMsg = error.response?.data?.error ?? error.message ?? "Failed to update client status"; toast(errMsg, "error"); }, }); const deleteMutation = useMutation({ mutationFn: (clientId: string) => deleteClient(clientId), onSuccess: () => queryClient.invalidateQueries({ queryKey: ["clients"] }), }); const clients = data?.items || []; const totalClients = clients.length; // TODO: Add real stats for active sessions and auth failures const stats = [ { label: "총 클라이언트", value: totalClients.toString(), delta: "Realtime", tone: "up" as const, }, { label: "활성 세션", value: "-", delta: "Not impl", tone: "stable" as const, }, { label: "인증 실패 (24h)", value: "0", delta: "Stable", tone: "stable" as const, }, ]; if (isLoading) { return
RP registry
Docs & Examples
Includes PKCE, client_secret_basic, redirect URI validation tips.
AI Admin Bot
admin@brsw.kr