import { useQuery } from "@tanstack/react-query"; import type { AxiosError } from "axios"; import { BookOpenText, Filter, Plus, Search, ServerCog, ShieldHalf, } from "lucide-react"; import { useState } from "react"; import { useAuth } from "react-oidc-context"; import { Link, useNavigate } from "react-router-dom"; import { ForbiddenMessage } from "../../components/common/ForbiddenMessage"; 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 { Table, TableBody, TableCell, TableHead, TableHeader, TableRow, } from "../../components/ui/table"; import { fetchClients, fetchDevStats } from "../../lib/devApi"; import { t } from "../../lib/i18n"; import { cn } from "../../lib/utils"; function ClientsPage() { const navigate = useNavigate(); const auth = useAuth(); const hasAccessToken = Boolean(auth.user?.access_token); const { data, isLoading: isLoadingClients, error: clientError, } = useQuery({ queryKey: ["clients"], queryFn: fetchClients, enabled: hasAccessToken, }); const { data: statsData, isLoading: isLoadingStats } = useQuery({ queryKey: ["dev-stats"], queryFn: fetchDevStats, enabled: hasAccessToken, }); const [searchQuery, setSearchQuery] = useState(""); const [typeFilter, setTypeFilter] = useState("all"); const [statusFilter, setStatusFilter] = useState("all"); const [isAdvancedFilterOpen, setIsAdvancedFilterOpen] = useState(false); const clients = data?.items || []; const filteredClients = clients.filter((client) => { const matchesSearch = !searchQuery || client.name?.toLowerCase().includes(searchQuery.toLowerCase()) || client.id.toLowerCase().includes(searchQuery.toLowerCase()); const matchesType = typeFilter === "all" || client.type === typeFilter; const matchesStatus = statusFilter === "all" || client.status === statusFilter; return matchesSearch && matchesType && matchesStatus; }); const totalClients = statsData?.total_clients ?? clients.length; const activeSessions = statsData?.active_sessions ?? 0; const authFailures = statsData?.auth_failures_24h ?? 0; type StatTone = "up" | "down" | "stable"; type StatItem = { labelKey: string; labelFallback: string; value: string; deltaKey: string; deltaFallback: string; tone: StatTone; }; const stats: StatItem[] = [ { labelKey: "ui.dev.clients.stats.total", labelFallback: "Total Applications", value: totalClients.toString(), deltaKey: "ui.dev.clients.stats.realtime", deltaFallback: "Realtime", tone: "up" as const, }, { labelKey: "ui.dev.clients.stats.active_sessions", labelFallback: "Active Sessions", value: activeSessions.toString(), deltaKey: "ui.dev.clients.stats.realtime", deltaFallback: "Realtime", tone: "up" as const, }, { labelKey: "ui.dev.clients.stats.auth_failures", labelFallback: "Auth Failures (24h)", value: authFailures.toString(), deltaKey: authFailures > 0 ? "ui.dev.clients.stats.alert" : "ui.dev.clients.stats.stable", deltaFallback: authFailures > 0 ? "Check Logs" : "Stable", tone: authFailures > 0 ? ("down" as const) : ("stable" as const), }, ]; const isLoading = isLoadingClients || isLoadingStats; if (auth.isLoading || !hasAccessToken || isLoading) { return (
{t("ui.dev.clients.registry.title", "RP registry")}
{t("ui.dev.clients.help.docs_title", "Docs & Examples")}
{t( "msg.dev.clients.help.docs_body", "Includes PKCE, client_secret_basic, redirect URI validation tips.", )}
{t("ui.dev.clients.owner.name", "AI Admin Bot")}
{t("ui.dev.clients.owner.email", "admin@brsw.kr")}