import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query"; import type { AxiosError } from "axios"; import { useState } from "react"; import { fetchAllTenants, fetchMe, fetchUsers, fetchSystemRelations, addSystemRelation, removeSystemRelation, type TenantRelation, } from "../../../lib/adminApi"; import { t } from "../../../lib/i18n"; import { TenantFineGrainedPermissionsTab } from "./TenantFineGrainedPermissionsTab"; import { ShieldCheck, Search, Plus, UserPlus, Trash2, Settings, Shield } from "lucide-react"; import { Badge } from "../../../components/ui/badge"; import { Button } from "../../../components/ui/button"; import { Card, CardContent, CardDescription, CardHeader, CardTitle, } from "../../../components/ui/card"; import { Dialog, DialogContent, DialogDescription, DialogHeader, DialogTitle, } from "../../../components/ui/dialog"; import { Input } from "../../../components/ui/input"; import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow, } from "../../../components/ui/table"; import { toast } from "../../../components/ui/use-toast"; export function TenantFineGrainedPermissionsPage() { const queryClient = useQueryClient(); const [activeTab, setActiveTab] = useState<"tenant" | "system">("tenant"); const [selectedTenantId, setSelectedTenantId] = useState(""); const [searchTerm, setSearchTerm] = useState(""); const [isDialogOpen, setIsDialogOpen] = useState(false); const { data: profile } = useQuery({ queryKey: ["me"], queryFn: fetchMe, }); const isSuperAdmin = profile?.role === "super_admin"; const tenantsQuery = useQuery({ queryKey: ["tenants", "list-all"], queryFn: () => fetchAllTenants(), enabled: isSuperAdmin, }); const tenants = isSuperAdmin ? (tenantsQuery.data?.items ?? []) : (profile?.manageableTenants ?? []); // System Relations (Admin Control) Queries & Mutations const systemRelationsQuery = useQuery({ queryKey: ["system-relations"], queryFn: fetchSystemRelations, enabled: isSuperAdmin && activeTab === "system", }); const systemRelations = systemRelationsQuery.data ?? []; const addSystemRelationMutation = useMutation({ mutationFn: (payload: { userId: string; relation: string }) => addSystemRelation(payload.userId, payload.relation), onSuccess: () => { queryClient.invalidateQueries({ queryKey: ["system-relations"] }); toast.success(t("msg.admin.system.relations.add_success", "시스템 메뉴 권한이 추가되었습니다.")); }, onError: (err: AxiosError<{ error?: string }>) => { toast.error(err.response?.data?.error || t("msg.common.error", "오류가 발생했습니다.")); }, }); const removeSystemRelationMutation = useMutation({ mutationFn: (payload: { userId: string; relation: string }) => removeSystemRelation(payload.userId, payload.relation), onSuccess: () => { queryClient.invalidateQueries({ queryKey: ["system-relations"] }); toast.success(t("msg.admin.system.relations.remove_success", "시스템 메뉴 권한이 회수되었습니다.")); }, onError: (err: AxiosError<{ error?: string }>) => { toast.error(err.response?.data?.error || t("msg.common.error", "오류가 발생했습니다.")); }, }); const handleSystemRelationChange = async ( userId: string, relation: string, hasAccess: boolean, ) => { if (hasAccess) { await addSystemRelationMutation.mutateAsync({ userId, relation }); } else { await removeSystemRelationMutation.mutateAsync({ userId, relation }); } }; const handleRemoveAllSystemRelations = async (userId: string, userRelations: string[]) => { if (!window.confirm(t("msg.admin.system.relations.remove_all_confirm", "이 사용자의 모든 시스템 메뉴 권한을 삭제하시겠습니까?"))) { return; } for (const rel of userRelations) { await removeSystemRelationMutation.mutateAsync({ userId, relation: rel }); } }; const usersQuery = useQuery({ queryKey: ["admin-users-search", searchTerm], queryFn: () => fetchUsers(20, 0, searchTerm), enabled: isDialogOpen && searchTerm.length >= 2, }); const handleAddSystemUser = (userId: string) => { addSystemRelationMutation.mutate({ userId, relation: "overview_viewers" }); setIsDialogOpen(false); setSearchTerm(""); }; const searchResults = usersQuery.data?.items || []; const systemMenus = [ { label: t("ui.admin.nav.overview", "개요"), relation: "overview_viewers" }, { label: t("ui.admin.nav.tenants", "테넌트"), relation: "tenants_viewers" }, { label: t("ui.admin.nav.org_chart", "조직도"), relation: "org_chart_viewers" }, { label: t("ui.admin.nav.worksmobile", "Worksmobile"), relation: "worksmobile_viewers" }, { label: t("ui.admin.nav.ory_ssot", "Ory SSOT"), relation: "ory_ssot_viewers" }, { label: t("ui.admin.nav.data_integrity", "정합성"), relation: "data_integrity_viewers" }, { label: t("ui.admin.nav.users", "사용자"), relation: "users_viewers" }, { label: t("ui.admin.nav.permissions_direct", "권한 부여"), relation: "permissions_direct_viewers" }, { label: t("ui.admin.nav.auth_guard", "인증 가드"), relation: "auth_guard_viewers" }, { label: t("ui.admin.nav.api_keys", "API 키"), relation: "api_keys_viewers" }, { label: t("ui.admin.nav.audit_logs", "감사 로그"), relation: "audit_logs_viewers" }, ]; return (
{t( "msg.admin.permissions_direct.description", "테넌트의 세부 기능 권한 및 글로벌 사이드바 메뉴 탭 접근 권한을 지정하고 부여합니다.", )}