AeThex-OS/client/src/pages/admin-architects.tsx
sirpiglr cf72b31513 Secure all admin routes and fix achievement icon display
Introduces a ProtectedRoute component to secure all admin routes, centralizing authentication logic. Removes redundant individual auth checks from admin pages. Updates admin-achievements.tsx to use a new `iconMap` for consistent icon rendering, resolving display issues.

Replit-Commit-Author: Agent
Replit-Commit-Session-Id: 279f1558-c0e3-40e4-8217-be7e9f4c6eca
Replit-Commit-Checkpoint-Type: full_checkpoint
Replit-Commit-Event-Id: 5a2fcd4b-aa24-41db-8542-c8df6e959cd0
Replit-Commit-Screenshot-Url: https://storage.googleapis.com/screenshot-production-us-central1/b984cb14-1d19-4944-922b-bc79e821ed35/279f1558-c0e3-40e4-8217-be7e9f4c6eca/ugufFZw
Replit-Helium-Checkpoint-Created: true
2025-12-16 01:02:07 +00:00

339 lines
15 KiB
TypeScript

import { useState } from "react";
import { motion } from "framer-motion";
import { Link, useLocation } from "wouter";
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
import { useAuth } from "@/lib/auth";
import {
Users, FileCode, Shield, Activity, LogOut,
Home, BarChart3, Settings, User, Search,
CheckCircle, XCircle, Eye, Edit, ChevronRight
} from "lucide-react";
export default function AdminArchitects() {
const { user, logout } = useAuth();
const [, setLocation] = useLocation();
const [searchQuery, setSearchQuery] = useState("");
const [selectedProfile, setSelectedProfile] = useState<any>(null);
const queryClient = useQueryClient();
const { data: profiles, isLoading } = useQuery({
queryKey: ["profiles"],
queryFn: async () => {
const res = await fetch("/api/profiles");
return res.json();
},
});
const updateProfileMutation = useMutation({
mutationFn: async ({ id, updates }: { id: string; updates: any }) => {
const res = await fetch(`/api/profiles/${id}`, {
method: "PATCH",
headers: { "Content-Type": "application/json" },
body: JSON.stringify(updates),
});
if (!res.ok) throw new Error("Failed to update");
return res.json();
},
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["profiles"] });
setSelectedProfile(null);
},
});
const filteredProfiles = profiles?.filter((p: any) =>
p.username?.toLowerCase().includes(searchQuery.toLowerCase()) ||
p.email?.toLowerCase().includes(searchQuery.toLowerCase())
) || [];
const handleLogout = async () => {
await logout();
setLocation("/");
};
const toggleVerified = (profile: any) => {
updateProfileMutation.mutate({
id: profile.id,
updates: { is_verified: !profile.is_verified }
});
};
return (
<div className="min-h-screen bg-background text-foreground font-mono flex">
{/* Sidebar */}
<div className="w-64 bg-card border-r border-white/10 flex flex-col">
<div className="p-6 border-b border-white/10">
<h1 className="text-xl font-display font-bold text-white uppercase tracking-wider">
AeThex
</h1>
<p className="text-xs text-primary mt-1">Command Center</p>
</div>
<nav className="flex-1 p-4 space-y-2">
<NavItem icon={<BarChart3 className="w-4 h-4" />} label="Dashboard" href="/admin" />
<NavItem icon={<Users className="w-4 h-4" />} label="Architects" href="/admin/architects" active />
<NavItem icon={<FileCode className="w-4 h-4" />} label="Credentials" href="/admin/credentials" />
<NavItem icon={<Activity className="w-4 h-4" />} label="Projects" href="/admin/projects" />
<NavItem icon={<Shield className="w-4 h-4" />} label="Aegis Monitor" href="/admin/aegis" />
</nav>
<div className="p-4 border-t border-white/10">
<div className="flex items-center gap-3 mb-4">
<div className="w-8 h-8 bg-primary/20 rounded-full flex items-center justify-center">
<User className="w-4 h-4 text-primary" />
</div>
<div>
<div className="text-sm text-white font-bold">{user?.username}</div>
<div className="text-xs text-muted-foreground">
{user?.isAdmin ? "Administrator" : "Member"}
</div>
</div>
</div>
<button
onClick={handleLogout}
className="w-full flex items-center gap-2 text-muted-foreground hover:text-white text-sm py-2 px-3 hover:bg-white/5 transition-colors"
>
<LogOut className="w-4 h-4" /> Sign Out
</button>
</div>
</div>
{/* Main Content */}
<div className="flex-1 overflow-auto">
<div className="p-8">
<div className="flex justify-between items-center mb-8">
<div>
<h2 className="text-2xl font-display font-bold text-white uppercase tracking-wider">
Architects
</h2>
<p className="text-muted-foreground text-sm mt-1">
{profiles?.length || 0} registered architects
</p>
</div>
{/* Search */}
<div className="relative">
<Search className="w-4 h-4 absolute left-3 top-1/2 -translate-y-1/2 text-muted-foreground" />
<input
type="text"
placeholder="Search architects..."
value={searchQuery}
onChange={(e) => setSearchQuery(e.target.value)}
className="bg-card border border-white/10 pl-10 pr-4 py-2 text-sm text-white placeholder-muted-foreground focus:border-primary/50 focus:outline-none w-64"
data-testid="input-search"
/>
</div>
</div>
{/* Table */}
<div className="bg-card/50 border border-white/10 overflow-hidden">
<table className="w-full">
<thead>
<tr className="border-b border-white/10 text-left">
<th className="p-4 text-xs text-muted-foreground uppercase tracking-wider font-bold">User</th>
<th className="p-4 text-xs text-muted-foreground uppercase tracking-wider font-bold">Role</th>
<th className="p-4 text-xs text-muted-foreground uppercase tracking-wider font-bold">Level</th>
<th className="p-4 text-xs text-muted-foreground uppercase tracking-wider font-bold">XP</th>
<th className="p-4 text-xs text-muted-foreground uppercase tracking-wider font-bold">Status</th>
<th className="p-4 text-xs text-muted-foreground uppercase tracking-wider font-bold">Verified</th>
<th className="p-4 text-xs text-muted-foreground uppercase tracking-wider font-bold">Actions</th>
</tr>
</thead>
<tbody>
{isLoading ? (
<tr>
<td colSpan={7} className="p-8 text-center text-muted-foreground">
Loading...
</td>
</tr>
) : filteredProfiles.length === 0 ? (
<tr>
<td colSpan={7} className="p-8 text-center text-muted-foreground">
No architects found
</td>
</tr>
) : (
filteredProfiles.map((profile: any) => (
<tr key={profile.id} className="border-b border-white/5 hover:bg-white/5 transition-colors">
<td className="p-4">
<div className="flex items-center gap-3">
<img
src={profile.avatar_url}
alt={profile.username}
className="w-8 h-8 rounded-full object-cover"
/>
<div>
<div className="text-sm text-white font-bold">{profile.username}</div>
<div className="text-xs text-muted-foreground">{profile.email}</div>
</div>
</div>
</td>
<td className="p-4">
<span className={`text-xs px-2 py-1 rounded uppercase font-bold ${
profile.role === 'admin' ? 'bg-primary/10 text-primary' :
profile.role === 'employee' ? 'bg-secondary/10 text-secondary' :
'bg-white/5 text-muted-foreground'
}`}>
{profile.role}
</span>
</td>
<td className="p-4 text-white font-bold">{profile.level}</td>
<td className="p-4 text-primary font-bold">{profile.total_xp}</td>
<td className="p-4">
<span className={`text-xs ${
profile.status === 'online' ? 'text-green-500' : 'text-muted-foreground'
}`}>
{profile.status}
</span>
</td>
<td className="p-4">
<button
onClick={() => toggleVerified(profile)}
className={`p-1 rounded transition-colors ${
profile.is_verified
? 'text-green-500 hover:bg-green-500/10'
: 'text-muted-foreground hover:bg-white/5'
}`}
data-testid={`button-verify-${profile.id}`}
>
{profile.is_verified ? (
<CheckCircle className="w-5 h-5" />
) : (
<XCircle className="w-5 h-5" />
)}
</button>
</td>
<td className="p-4">
<button
onClick={() => setSelectedProfile(profile)}
className="text-muted-foreground hover:text-white transition-colors p-1"
data-testid={`button-view-${profile.id}`}
>
<Eye className="w-4 h-4" />
</button>
</td>
</tr>
))
)}
</tbody>
</table>
</div>
</div>
</div>
{/* Profile Detail Modal */}
{selectedProfile && (
<div className="fixed inset-0 bg-black/80 flex items-center justify-center z-50 p-4">
<motion.div
initial={{ opacity: 0, scale: 0.95 }}
animate={{ opacity: 1, scale: 1 }}
className="bg-card border border-white/10 w-full max-w-2xl max-h-[80vh] overflow-auto"
>
<div className="p-6 border-b border-white/10 flex justify-between items-start">
<div className="flex items-center gap-4">
<img
src={selectedProfile.avatar_url}
alt={selectedProfile.username}
className="w-16 h-16 rounded-full object-cover"
/>
<div>
<h3 className="text-xl font-display text-white uppercase">{selectedProfile.username}</h3>
<p className="text-muted-foreground text-sm">{selectedProfile.email}</p>
</div>
</div>
<button
onClick={() => setSelectedProfile(null)}
className="text-muted-foreground hover:text-white"
>
</button>
</div>
<div className="p-6 grid grid-cols-2 gap-6">
<div>
<label className="text-xs text-muted-foreground uppercase tracking-wider">Role</label>
<div className="text-white font-bold">{selectedProfile.role}</div>
</div>
<div>
<label className="text-xs text-muted-foreground uppercase tracking-wider">Level</label>
<div className="text-white font-bold">{selectedProfile.level}</div>
</div>
<div>
<label className="text-xs text-muted-foreground uppercase tracking-wider">Total XP</label>
<div className="text-primary font-bold">{selectedProfile.total_xp}</div>
</div>
<div>
<label className="text-xs text-muted-foreground uppercase tracking-wider">Status</label>
<div className={selectedProfile.status === 'online' ? 'text-green-500' : 'text-muted-foreground'}>
{selectedProfile.status}
</div>
</div>
<div>
<label className="text-xs text-muted-foreground uppercase tracking-wider">Passport ID</label>
<div className="text-white font-mono text-xs">{selectedProfile.aethex_passport_id}</div>
</div>
<div>
<label className="text-xs text-muted-foreground uppercase tracking-wider">Verified</label>
<div className={selectedProfile.is_verified ? 'text-green-500' : 'text-destructive'}>
{selectedProfile.is_verified ? 'Yes' : 'No'}
</div>
</div>
<div className="col-span-2">
<label className="text-xs text-muted-foreground uppercase tracking-wider">Bio</label>
<div className="text-white">{selectedProfile.bio || 'No bio'}</div>
</div>
<div className="col-span-2">
<label className="text-xs text-muted-foreground uppercase tracking-wider">Skills</label>
<div className="flex flex-wrap gap-2 mt-2">
{selectedProfile.skills?.map((skill: string, i: number) => (
<span key={i} className="bg-primary/10 text-primary px-2 py-1 text-xs rounded">
{skill}
</span>
)) || <span className="text-muted-foreground">No skills listed</span>}
</div>
</div>
</div>
<div className="p-6 border-t border-white/10 flex justify-end gap-4">
<button
onClick={() => setSelectedProfile(null)}
className="px-4 py-2 text-muted-foreground hover:text-white transition-colors"
>
Close
</button>
<button
onClick={() => toggleVerified(selectedProfile)}
className={`px-4 py-2 font-bold uppercase tracking-wider transition-colors ${
selectedProfile.is_verified
? 'bg-destructive/10 text-destructive hover:bg-destructive/20'
: 'bg-green-500/10 text-green-500 hover:bg-green-500/20'
}`}
>
{selectedProfile.is_verified ? 'Revoke Verification' : 'Verify Architect'}
</button>
</div>
</motion.div>
</div>
)}
</div>
);
}
function NavItem({ icon, label, href, active = false }: {
icon: React.ReactNode;
label: string;
href: string;
active?: boolean;
}) {
return (
<Link href={href}>
<div className={`flex items-center gap-3 px-3 py-2 text-sm transition-colors cursor-pointer ${
active
? 'bg-primary/10 text-primary border-l-2 border-primary'
: 'text-muted-foreground hover:text-white hover:bg-white/5'
}`}>
{icon}
{label}
</div>
</Link>
);
}