import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query"; import type { AxiosError } from "axios"; import { BookOpenText, 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 { CopyButton } from "../../components/ui/copy-button"; 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 { toast } from "../../components/ui/use-toast"; import { deleteClient, fetchClients, updateClientStatus, } from "../../lib/devApi"; import { t } from "../../lib/i18n"; import { cn } from "../../lib/utils"; 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" ? t("ui.common.status.active", "활성화") : t("ui.common.status.inactive", "비활성화"); toast( t( "msg.dev.clients.status_updated", "클라이언트가 {{status}}되었습니다.", { status: statusText, }, ), ); queryClient.invalidateQueries({ queryKey: ["clients"] }); }, onError: (error: AxiosError<{ error?: string }>) => { const errMsg = error.response?.data?.error ?? error.message ?? t( "msg.dev.clients.status_update_error", "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 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: "총 클라이언트", value: totalClients.toString(), deltaKey: "ui.dev.clients.stats.realtime", deltaFallback: "Realtime", tone: "up" as const, }, { labelKey: "ui.dev.clients.stats.active_sessions", labelFallback: "활성 세션", value: "-", deltaKey: "ui.dev.clients.stats.not_impl", deltaFallback: "Not impl", tone: "stable" as const, }, { labelKey: "ui.dev.clients.stats.auth_failures", labelFallback: "인증 실패 (24h)", value: "0", deltaKey: "ui.dev.clients.stats.stable", deltaFallback: "Stable", tone: "stable" as const, }, ]; if (isLoading) { return (
{t("msg.dev.clients.loading", "Loading clients...")}
); } if (error) { const errMsg = (error as AxiosError<{ error?: string }>).response?.data?.error ?? (error as Error).message; return (
{t("msg.dev.clients.load_error", "Error loading clients: {{error}}", { error: errMsg, })}
); } return (

{t("ui.dev.clients.registry.title", "RP registry")}

{t("ui.dev.clients.registry.subtitle", "Relying Parties")} {t( "msg.dev.clients.registry.description", "OIDC 클라이언트, 인증 방식, 리다이렉트 URI, 비밀키 재발행을 감사 로그와 함께 관리합니다.", )}
{t("ui.dev.clients.badge.tenant_selected", "테넌트: 선택됨")} {t("ui.dev.clients.badge.admin_session", "관리자 세션")}
{stats.map((item) => ( {t(item.labelKey, item.labelFallback)}
{item.value} {t(item.deltaKey, item.deltaFallback)}
))}
{t("ui.dev.clients.list.title", "클라이언트 목록")}
{t("ui.dev.clients.table.application", "애플리케이션")} {t("ui.dev.clients.table.client_id", "Client ID")} {t("ui.dev.clients.table.type", "유형")} {t("ui.dev.clients.table.status", "상태")} {t("ui.dev.clients.table.created_at", "생성일")} {t("ui.dev.clients.table.actions", "액션")} {clients.map((client) => (
{client.type === "confidential" ? ( ) : ( )}

{client.name || t("ui.dev.clients.untitled", "Untitled")}

{t("ui.dev.clients.tenant_scoped", "Tenant-scoped")}

{client.id} toast( t( "msg.dev.clients.copy_client_id", "클라이언트 ID가 복사되었습니다.", ), ) } />
{client.type === "confidential" ? t( "ui.dev.clients.type.confidential", "기밀(Confidential)", ) : t("ui.dev.clients.type.public", "Public")}
updateStatusMutation.mutate({ id: client.id, status: checked ? "active" : "inactive", }) } /> {client.status === "active" ? t("ui.common.status.active", "활성") : t("ui.common.status.inactive", "비활성")}
{client.createdAt ? new Date(client.createdAt).toLocaleDateString() : "-"}
))}
{t( "msg.dev.clients.showing", "Showing {{shown}} of {{total}} clients", { shown: clients.length, total: totalClients }, )}
{t( "ui.dev.clients.help.title", "Need help with OIDC configuration?", )} {t( "msg.dev.clients.help.subtitle", "Developer guides for Confidential/Public clients, redirect URIs, and auth methods.", )}

{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.title", "Owner")} {t("ui.dev.clients.owner.subtitle", "Tenant admin on-call")}
AR

{t("ui.dev.clients.owner.name", "AI Admin Bot")}

{t("ui.dev.clients.owner.email", "admin@brsw.kr")}

{t("ui.dev.clients.owner.role", "Role: Tenant Admin")} {t("ui.dev.clients.owner.scope", "Scope: TENANT-12")}
); } export default ClientsPage;