302 lines
13 KiB
TypeScript
302 lines
13 KiB
TypeScript
import { createBrowserRouter } from 'react-router-dom'
|
|
import * as Sentry from '@sentry/react'
|
|
import { Suspense } from 'react'
|
|
import { AppLayout, ProtectedRoute } from '@/components/layout'
|
|
import { RouteError } from '@/components/common/RouteError'
|
|
import { ErrorBoundary } from '@/components/common/ErrorBoundary'
|
|
import { PageLoader } from '@/components/common/PageLoader'
|
|
import { lazyWithRetry } from '@/lib/lazyWithRetry'
|
|
|
|
const sentryCreateBrowserRouter = Sentry.wrapCreateBrowserRouterV7(createBrowserRouter)
|
|
import {
|
|
LoginPage,
|
|
RegisterPage,
|
|
} from '@/pages'
|
|
|
|
// Public pages
|
|
const LandingPage = lazyWithRetry(() => import('@/pages/LandingPage'))
|
|
const PublicTemplatesPage = lazyWithRetry(() => import('@/pages/PublicTemplatesPage'))
|
|
const SharedSessionPage = lazyWithRetry(() => import('@/pages/SharedSessionPage'))
|
|
const SurveyPage = lazyWithRetry(() => import('@/pages/SurveyPage'))
|
|
const SurveyThankYouPage = lazyWithRetry(() => import('@/pages/SurveyThankYouPage'))
|
|
const PrivacyPage = lazyWithRetry(() => import('@/pages/PrivacyPage'))
|
|
const TermsPage = lazyWithRetry(() => import('@/pages/TermsPage'))
|
|
|
|
// Standalone auth pages
|
|
const VerifyEmailPage = lazyWithRetry(() => import('@/pages/VerifyEmailPage'))
|
|
const ChangePasswordPage = lazyWithRetry(() => import('@/pages/ChangePasswordPage'))
|
|
const ForgotPasswordPage = lazyWithRetry(() => import('@/pages/ForgotPasswordPage'))
|
|
const ResetPasswordPage = lazyWithRetry(() => import('@/pages/ResetPasswordPage'))
|
|
|
|
// Lazy load heavy pages for code splitting
|
|
const QuickStartPage = lazyWithRetry(() => import('@/pages/QuickStartPage'))
|
|
const TreeLibraryPage = lazyWithRetry(() => import('@/pages/TreeLibraryPage'))
|
|
const MyTreesPage = lazyWithRetry(() => import('@/pages/MyTreesPage'))
|
|
const TreeNavigationPage = lazyWithRetry(() => import('@/pages/TreeNavigationPage'))
|
|
const TreeEditorPage = lazyWithRetry(() => import('@/pages/TreeEditorPage'))
|
|
const ProceduralEditorPage = lazyWithRetry(() => import('@/pages/ProceduralEditorPage'))
|
|
const ProceduralNavigationPage = lazyWithRetry(() => import('@/pages/ProceduralNavigationPage'))
|
|
const MaintenanceFlowDetailPage = lazyWithRetry(() => import('@/pages/MaintenanceFlowDetailPage'))
|
|
const BatchStatusPage = lazyWithRetry(() => import('@/pages/BatchStatusPage'))
|
|
const SessionHistoryPage = lazyWithRetry(() => import('@/pages/SessionHistoryPage'))
|
|
const SessionDetailPage = lazyWithRetry(() => import('@/pages/SessionDetailPage'))
|
|
const MySharesPage = lazyWithRetry(() => import('@/pages/MySharesPage'))
|
|
const TeamAnalyticsPage = lazyWithRetry(() => import('@/pages/TeamAnalyticsPage'))
|
|
const MyAnalyticsPage = lazyWithRetry(() => import('@/pages/MyAnalyticsPage'))
|
|
const FeedbackPage = lazyWithRetry(() => import('@/pages/FeedbackPage'))
|
|
const StepLibraryPage = lazyWithRetry(() => import('@/pages/StepLibraryPage'))
|
|
const ScriptLibraryPage = lazyWithRetry(() => import('@/pages/ScriptLibraryPage'))
|
|
const ScriptManagePage = lazyWithRetry(() => import('@/pages/ScriptManagePage'))
|
|
const FlowPilotPage = lazyWithRetry(() => import('@/pages/FlowPilotPage'))
|
|
const CockpitPage = lazyWithRetry(() => import('@/pages/CockpitPage'))
|
|
const FlowAssistPage = lazyWithRetry(() => import('@/pages/FlowAssistPage'))
|
|
const FlowPilotSessionPage = lazyWithRetry(() => import('@/pages/FlowPilotSessionPage'))
|
|
const EscalationQueuePage = lazyWithRetry(() => import('@/pages/EscalationQueuePage'))
|
|
const ReviewQueuePage = lazyWithRetry(() => import('@/pages/ReviewQueuePage'))
|
|
const FlowPilotAnalyticsPage = lazyWithRetry(() => import('@/pages/FlowPilotAnalyticsPage'))
|
|
const ScriptBuilderPage = lazyWithRetry(() => import('@/pages/ScriptBuilderPage'))
|
|
const KBAcceleratorPage = lazyWithRetry(() => import('@/pages/KBAcceleratorPage'))
|
|
const SessionQueuePage = lazyWithRetry(() => import('@/pages/SessionQueuePage'))
|
|
const DevBranchingPage = lazyWithRetry(() => import('@/pages/DevBranchingPage'))
|
|
const GuidesHubPage = lazyWithRetry(() => import('@/pages/GuidesHubPage'))
|
|
const GuideDetailPage = lazyWithRetry(() => import('@/pages/GuideDetailPage'))
|
|
const AccountSettingsPage = lazyWithRetry(() => import('@/pages/AccountSettingsPage'))
|
|
const NetworkDiagramsPage = lazyWithRetry(() => import('@/pages/NetworkDiagrams'))
|
|
const DiagramEditorPage = lazyWithRetry(() => import('@/pages/NetworkDiagrams/DiagramEditor'))
|
|
// Admin pages
|
|
const AdminLayout = lazyWithRetry(() => import('@/components/admin/AdminLayout'))
|
|
const AdminDashboardPage = lazyWithRetry(() => import('@/pages/admin/DashboardPage'))
|
|
const AdminAccountsPage = lazyWithRetry(() => import('@/pages/admin/AccountsPage'))
|
|
const AdminAccountDetailPage = lazyWithRetry(() => import('@/pages/admin/AccountDetailPage'))
|
|
const AdminUserDetailPage = lazyWithRetry(() => import('@/pages/admin/UserDetailPage'))
|
|
const AdminInviteCodesPage = lazyWithRetry(() => import('@/pages/admin/InviteCodesPage'))
|
|
const AdminAuditLogsPage = lazyWithRetry(() => import('@/pages/admin/AuditLogsPage'))
|
|
const AdminPlanLimitsPage = lazyWithRetry(() => import('@/pages/admin/PlanLimitsPage'))
|
|
const AdminFeatureFlagsPage = lazyWithRetry(() => import('@/pages/admin/FeatureFlagsPage'))
|
|
const AdminSettingsPage = lazyWithRetry(() => import('@/pages/admin/SettingsPage'))
|
|
const AdminGlobalCategoriesPage = lazyWithRetry(() => import('@/pages/admin/GlobalCategoriesPage'))
|
|
const AdminSurveyInvitesPage = lazyWithRetry(() => import('@/pages/admin/SurveyInvitesPage'))
|
|
const AdminSurveyResponsesPage = lazyWithRetry(() => import('@/pages/admin/SurveyResponsesPage'))
|
|
const AdminGalleryManagementPage = lazyWithRetry(() => import('@/pages/admin/GalleryManagementPage'))
|
|
|
|
// Account pages
|
|
const AccountLayout = lazyWithRetry(() => import('@/components/account/AccountLayout'))
|
|
const ProfileSettingsPage = lazyWithRetry(() => import('@/pages/account/ProfileSettingsPage'))
|
|
const TeamCategoriesPage = lazyWithRetry(() => import('@/pages/account/TeamCategoriesPage'))
|
|
const TargetListsPage = lazyWithRetry(() => import('@/pages/account/TargetListsPage'))
|
|
const ChatRetentionSettingsPage = lazyWithRetry(() => import('@/pages/account/ChatRetentionSettingsPage'))
|
|
const IntegrationsPage = lazyWithRetry(() => import('@/pages/account/IntegrationsPage'))
|
|
const BrandingSettingsPage = lazyWithRetry(() => import('@/pages/account/BrandingSettingsPage'))
|
|
|
|
/** Wraps a lazy-loaded page with Suspense + ErrorBoundary */
|
|
function page(Component: React.LazyExoticComponent<React.ComponentType>) {
|
|
return (
|
|
<ErrorBoundary>
|
|
<Suspense fallback={<PageLoader />}>
|
|
<Component />
|
|
</Suspense>
|
|
</ErrorBoundary>
|
|
)
|
|
}
|
|
|
|
export const router = sentryCreateBrowserRouter([
|
|
{
|
|
path: '/landing',
|
|
element: page(LandingPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/templates',
|
|
element: page(PublicTemplatesPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/privacy',
|
|
element: page(PrivacyPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/terms',
|
|
element: page(TermsPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/login',
|
|
element: <LoginPage />,
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/register',
|
|
element: <RegisterPage />,
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/forgot-password',
|
|
element: page(ForgotPasswordPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/reset-password',
|
|
element: page(ResetPasswordPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/verify-email',
|
|
element: page(VerifyEmailPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/survey',
|
|
element: page(SurveyPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/survey/thank-you',
|
|
element: page(SurveyThankYouPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/share/:shareToken',
|
|
element: page(SharedSessionPage),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/change-password',
|
|
element: (
|
|
<ProtectedRoute>
|
|
{page(ChangePasswordPage)}
|
|
</ProtectedRoute>
|
|
),
|
|
errorElement: <RouteError />,
|
|
},
|
|
{
|
|
path: '/',
|
|
element: (
|
|
<ProtectedRoute>
|
|
<AppLayout />
|
|
</ProtectedRoute>
|
|
),
|
|
errorElement: <RouteError />,
|
|
children: [
|
|
{ index: true, element: page(QuickStartPage) },
|
|
{ path: 'trees', element: page(TreeLibraryPage) },
|
|
{ path: 'my-trees', element: page(MyTreesPage) },
|
|
{ path: 'trees/new', element: page(TreeEditorPage) },
|
|
{ path: 'trees/:id/edit', element: page(TreeEditorPage) },
|
|
{ path: 'flows/new', element: page(ProceduralEditorPage) },
|
|
{ path: 'flows/:id/edit', element: page(ProceduralEditorPage) },
|
|
{ path: 'flows/:id/navigate', element: page(ProceduralNavigationPage) },
|
|
{ path: 'flows/:id/maintenance', element: page(MaintenanceFlowDetailPage) },
|
|
{ path: 'flows/:id/batches/:batchId', element: page(BatchStatusPage) },
|
|
{ path: 'trees/:id/navigate', element: page(TreeNavigationPage) },
|
|
{ path: 'sessions', element: page(SessionHistoryPage) },
|
|
{ path: 'sessions/:id', element: page(SessionDetailPage) },
|
|
{ path: 'shares', element: page(MySharesPage) },
|
|
{ path: 'analytics', element: page(TeamAnalyticsPage) },
|
|
{ path: 'analytics/me', element: page(MyAnalyticsPage) },
|
|
{ path: 'feedback', element: page(FeedbackPage) },
|
|
{ path: 'step-library', element: page(StepLibraryPage) },
|
|
{ path: 'scripts', element: page(ScriptLibraryPage) },
|
|
{ path: 'scripts/manage', element: page(ScriptManagePage) },
|
|
{ path: 'script-builder', element: page(ScriptBuilderPage) },
|
|
{ path: 'network-diagrams', element: page(NetworkDiagramsPage) },
|
|
{ path: 'network-diagrams/new', element: page(DiagramEditorPage) },
|
|
{ path: 'network-diagrams/:id', element: page(DiagramEditorPage) },
|
|
{ path: 'kb-accelerator', element: page(KBAcceleratorPage) },
|
|
{ path: 'assistant', element: page(FlowPilotPage) },
|
|
{ path: 'assistant/:sessionId', element: page(FlowPilotPage) },
|
|
{ path: 'cockpit', element: page(CockpitPage) },
|
|
{ path: 'cockpit/:sessionId', element: page(CockpitPage) },
|
|
{ path: 'flow-assist', element: page(FlowAssistPage) },
|
|
{ path: 'pilot', element: page(FlowPilotSessionPage) },
|
|
{ path: 'pilot/:sessionId', element: page(FlowPilotSessionPage) },
|
|
{ path: 'escalations', element: page(EscalationQueuePage) },
|
|
{ path: 'queue', element: page(SessionQueuePage) },
|
|
{ path: 'review-queue', element: page(ReviewQueuePage) },
|
|
{ path: 'analytics/flowpilot', element: page(FlowPilotAnalyticsPage) },
|
|
{ path: 'dev/branching', element: page(DevBranchingPage) },
|
|
{ path: 'guides', element: page(GuidesHubPage) },
|
|
{ path: 'guides/:slug', element: page(GuideDetailPage) },
|
|
// Admin routes
|
|
{
|
|
path: 'admin',
|
|
element: (
|
|
<ErrorBoundary>
|
|
<Suspense fallback={<PageLoader />}>
|
|
<ProtectedRoute requiredRole="super_admin">
|
|
<AdminLayout />
|
|
</ProtectedRoute>
|
|
</Suspense>
|
|
</ErrorBoundary>
|
|
),
|
|
children: [
|
|
{ index: true, element: page(AdminDashboardPage) },
|
|
{ path: 'accounts', element: page(AdminAccountsPage) },
|
|
{ path: 'accounts/:accountId', element: page(AdminAccountDetailPage) },
|
|
{ path: 'users', element: page(AdminAccountsPage) },
|
|
{ path: 'users/:userId', element: page(AdminUserDetailPage) },
|
|
{ path: 'invite-codes', element: page(AdminInviteCodesPage) },
|
|
{ path: 'audit-logs', element: page(AdminAuditLogsPage) },
|
|
{ path: 'plan-limits', element: page(AdminPlanLimitsPage) },
|
|
{ path: 'feature-flags', element: page(AdminFeatureFlagsPage) },
|
|
{ path: 'settings', element: page(AdminSettingsPage) },
|
|
{ path: 'categories', element: page(AdminGlobalCategoriesPage) },
|
|
{ path: 'survey-invites', element: page(AdminSurveyInvitesPage) },
|
|
{ path: 'survey-responses', element: page(AdminSurveyResponsesPage) },
|
|
{ path: 'gallery', element: page(AdminGalleryManagementPage) },
|
|
],
|
|
},
|
|
// Account routes
|
|
{
|
|
path: 'account',
|
|
element: (
|
|
<ErrorBoundary>
|
|
<Suspense fallback={<PageLoader />}>
|
|
<AccountLayout />
|
|
</Suspense>
|
|
</ErrorBoundary>
|
|
),
|
|
children: [
|
|
{ index: true, element: page(AccountSettingsPage) },
|
|
{ path: 'profile', element: page(ProfileSettingsPage) },
|
|
{
|
|
path: 'categories',
|
|
element: (
|
|
<ProtectedRoute requiredRole="owner">
|
|
{page(TeamCategoriesPage)}
|
|
</ProtectedRoute>
|
|
),
|
|
},
|
|
{
|
|
path: 'chat-retention',
|
|
element: (
|
|
<ProtectedRoute requiredRole="owner">
|
|
{page(ChatRetentionSettingsPage)}
|
|
</ProtectedRoute>
|
|
),
|
|
},
|
|
{ path: 'target-lists', element: page(TargetListsPage) },
|
|
{
|
|
path: 'integrations',
|
|
element: (
|
|
<ProtectedRoute requiredRole="owner">
|
|
{page(IntegrationsPage)}
|
|
</ProtectedRoute>
|
|
),
|
|
},
|
|
{
|
|
path: 'branding',
|
|
element: (
|
|
<ProtectedRoute requiredRole="owner">
|
|
{page(BrandingSettingsPage)}
|
|
</ProtectedRoute>
|
|
),
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
])
|
|
|
|
export default router
|