2025-01-01 00:00:00 +00:00
|
|
|
import { Toaster } from "@/components/ui/toaster";
|
|
|
|
|
import { Toaster as Sonner } from "@/components/ui/sonner";
|
|
|
|
|
import { TooltipProvider } from "@/components/ui/tooltip";
|
|
|
|
|
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
|
|
|
|
|
import { BrowserRouter, Routes, Route } from "react-router-dom";
|
2026-01-06 14:46:23 +00:00
|
|
|
|
|
|
|
|
// Layouts
|
|
|
|
|
import PublicLayout from "@/components/layouts/PublicLayout";
|
2026-01-16 17:31:25 +10:30
|
|
|
import ProtectedLayout from "@/components/layouts/ProtectedLayout";
|
2026-01-06 14:46:23 +00:00
|
|
|
|
|
|
|
|
// Public pages
|
|
|
|
|
import Index from "@/pages/Index";
|
|
|
|
|
import LoginPage from "@/pages/auth/LoginPage";
|
|
|
|
|
import RegisterPage from "@/pages/auth/RegisterPage";
|
|
|
|
|
import VerifyEmailPage from "@/pages/auth/VerifyEmailPage";
|
|
|
|
|
import ForgotPasswordPage from "@/pages/auth/ForgotPasswordPage";
|
|
|
|
|
import ResetPasswordPage from "@/pages/auth/ResetPasswordPage";
|
|
|
|
|
import InviteAcceptPage from "@/pages/auth/InviteAcceptPage";
|
|
|
|
|
import OIDCConsentPage from "@/pages/auth/OIDCConsentPage";
|
|
|
|
|
import OIDCErrorPage from "@/pages/auth/OIDCErrorPage";
|
2026-01-20 15:54:11 +10:30
|
|
|
import OAuthCallbackPage from "@/pages/auth/OAuthCallbackPage";
|
2026-02-28 23:35:32 +05:45
|
|
|
import ActivatePage from "@/pages/auth/ActivatePage";
|
2026-01-06 14:46:23 +00:00
|
|
|
|
|
|
|
|
// User pages
|
|
|
|
|
import ProfilePage from "@/pages/user/ProfilePage";
|
|
|
|
|
import SecurityPage from "@/pages/user/SecurityPage";
|
|
|
|
|
import LinkedAccountsPage from "@/pages/user/LinkedAccountsPage";
|
|
|
|
|
import ActivityPage from "@/pages/user/ActivityPage";
|
2026-02-28 23:35:32 +05:45
|
|
|
import SSHKeysPage from "@/pages/user/SSHKeysPage";
|
2026-01-06 14:46:23 +00:00
|
|
|
|
|
|
|
|
// Organization pages
|
|
|
|
|
import OrgOverviewPage from "@/pages/org/OrgOverviewPage";
|
|
|
|
|
import MembersPage from "@/pages/org/MembersPage";
|
|
|
|
|
import PoliciesPage from "@/pages/org/PoliciesPage";
|
2026-01-16 17:31:25 +10:30
|
|
|
import CompliancePage from "@/pages/org/CompliancePage";
|
2026-01-06 14:46:23 +00:00
|
|
|
import OrgAuditPage from "@/pages/org/OrgAuditPage";
|
|
|
|
|
import OIDCClientsPage from "@/pages/org/OIDCClientsPage";
|
2026-02-28 23:35:32 +05:45
|
|
|
import CAsPage from "@/pages/org/CAsPage";
|
2026-02-27 21:08:16 +05:45
|
|
|
import DepartmentsPage from "@/pages/org/DepartmentsPage";
|
|
|
|
|
import PrincipalsPage from "@/pages/org/PrincipalsPage";
|
|
|
|
|
import SystemAuditPage from "@/pages/admin/SystemAuditPage";
|
2026-02-28 23:35:32 +05:45
|
|
|
import AdminUsersPage from "@/pages/admin/AdminUsersPage";
|
2026-01-06 14:46:23 +00:00
|
|
|
|
|
|
|
|
import NotFound from "@/pages/NotFound";
|
2026-01-06 15:39:14 +00:00
|
|
|
import ApiDevTools from "@/components/dev/ApiDevTools";
|
2025-01-01 00:00:00 +00:00
|
|
|
|
2026-01-16 17:31:25 +10:30
|
|
|
const queryClient = new QueryClient({
|
|
|
|
|
defaultOptions: {
|
|
|
|
|
queries: {
|
|
|
|
|
retry: (failureCount, error) => {
|
|
|
|
|
// Don't retry on 403 authorization errors
|
|
|
|
|
if (error && typeof error === 'object' && 'code' in error && error.code === 403) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
// Default retry behavior for other errors (max 3 retries)
|
|
|
|
|
return failureCount < 3;
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
});
|
2025-01-01 00:00:00 +00:00
|
|
|
|
|
|
|
|
const App = () => (
|
|
|
|
|
<QueryClientProvider client={queryClient}>
|
|
|
|
|
<TooltipProvider>
|
|
|
|
|
<Toaster />
|
|
|
|
|
<Sonner />
|
|
|
|
|
<BrowserRouter>
|
2026-01-06 15:33:03 +00:00
|
|
|
<AppRoutes />
|
2025-01-01 00:00:00 +00:00
|
|
|
</BrowserRouter>
|
|
|
|
|
</TooltipProvider>
|
|
|
|
|
</QueryClientProvider>
|
|
|
|
|
);
|
|
|
|
|
|
2026-01-06 15:33:03 +00:00
|
|
|
// Separate component so AuthProvider can use useNavigate
|
2026-02-28 23:35:32 +05:45
|
|
|
import { AuthProvider, useAuth } from "@/contexts/AuthContext";
|
|
|
|
|
import { Navigate } from "react-router-dom";
|
|
|
|
|
|
|
|
|
|
/** Redirects already-authenticated users away from guest-only pages (e.g. /login). */
|
|
|
|
|
function GuestRoute({ children }: { children: React.ReactNode }) {
|
|
|
|
|
const { isAuthenticated, isLoading } = useAuth();
|
|
|
|
|
if (isLoading) return null; // wait for auth state to resolve
|
|
|
|
|
if (isAuthenticated) return <Navigate to="/profile" replace />;
|
|
|
|
|
return <>{children}</>;
|
|
|
|
|
}
|
2026-01-06 15:33:03 +00:00
|
|
|
|
|
|
|
|
function AppRoutes() {
|
|
|
|
|
return (
|
|
|
|
|
<AuthProvider>
|
|
|
|
|
<Routes>
|
|
|
|
|
{/* Index redirect */}
|
|
|
|
|
<Route path="/" element={<Index />} />
|
|
|
|
|
|
|
|
|
|
{/* Public routes */}
|
|
|
|
|
<Route element={<PublicLayout />}>
|
2026-02-28 23:35:32 +05:45
|
|
|
<Route path="/login" element={<GuestRoute><LoginPage /></GuestRoute>} />
|
2026-01-06 15:33:03 +00:00
|
|
|
<Route path="/register" element={<RegisterPage />} />
|
|
|
|
|
<Route path="/verify-email" element={<VerifyEmailPage />} />
|
|
|
|
|
<Route path="/forgot-password" element={<ForgotPasswordPage />} />
|
|
|
|
|
<Route path="/reset-password" element={<ResetPasswordPage />} />
|
|
|
|
|
<Route path="/invite" element={<InviteAcceptPage />} />
|
|
|
|
|
<Route path="/consent" element={<OIDCConsentPage />} />
|
|
|
|
|
<Route path="/error" element={<OIDCErrorPage />} />
|
2026-01-20 15:54:11 +10:30
|
|
|
<Route path="/oauth/callback" element={<OAuthCallbackPage />} />
|
2026-02-28 23:35:32 +05:45
|
|
|
<Route path="/activate" element={<ActivatePage />} />
|
2026-01-06 15:33:03 +00:00
|
|
|
</Route>
|
|
|
|
|
|
2026-01-16 17:31:25 +10:30
|
|
|
{/* Protected routes - handles auth and MFA enforcement */}
|
|
|
|
|
<Route element={<ProtectedLayout />}>
|
2026-01-06 15:33:03 +00:00
|
|
|
{/* User routes */}
|
|
|
|
|
<Route path="/profile" element={<ProfilePage />} />
|
|
|
|
|
<Route path="/security" element={<SecurityPage />} />
|
|
|
|
|
<Route path="/linked-accounts" element={<LinkedAccountsPage />} />
|
|
|
|
|
<Route path="/activity" element={<ActivityPage />} />
|
2026-02-28 23:35:32 +05:45
|
|
|
<Route path="/ssh-keys" element={<SSHKeysPage />} />
|
2026-01-06 15:33:03 +00:00
|
|
|
|
|
|
|
|
{/* Organization routes */}
|
|
|
|
|
<Route path="/org" element={<OrgOverviewPage />} />
|
|
|
|
|
<Route path="/org/members" element={<MembersPage />} />
|
2026-02-27 21:08:16 +05:45
|
|
|
<Route path="/org/departments" element={<DepartmentsPage />} />
|
|
|
|
|
<Route path="/org/principals" element={<PrincipalsPage />} />
|
2026-01-06 15:33:03 +00:00
|
|
|
<Route path="/org/policies" element={<PoliciesPage />} />
|
2026-01-16 17:31:25 +10:30
|
|
|
<Route path="/org/policies/compliance" element={<CompliancePage />} />
|
2026-01-06 15:33:03 +00:00
|
|
|
<Route path="/org/audit" element={<OrgAuditPage />} />
|
|
|
|
|
<Route path="/org/clients" element={<OIDCClientsPage />} />
|
2026-02-28 23:35:32 +05:45
|
|
|
<Route path="/org/cas" element={<CAsPage />} />
|
2026-02-27 21:08:16 +05:45
|
|
|
|
|
|
|
|
{/* Admin routes */}
|
|
|
|
|
<Route path="/admin/audit" element={<SystemAuditPage />} />
|
2026-02-28 23:35:32 +05:45
|
|
|
<Route path="/admin/users" element={<AdminUsersPage />} />
|
2026-01-06 15:33:03 +00:00
|
|
|
</Route>
|
|
|
|
|
|
|
|
|
|
{/* Catch-all */}
|
|
|
|
|
<Route path="*" element={<NotFound />} />
|
|
|
|
|
</Routes>
|
2026-01-06 15:39:14 +00:00
|
|
|
|
|
|
|
|
{/* Dev tools - only shown in development */}
|
|
|
|
|
<ApiDevTools />
|
2026-01-06 15:33:03 +00:00
|
|
|
</AuthProvider>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2025-01-01 00:00:00 +00:00
|
|
|
export default App;
|