App.tsx 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. import { Component, type ReactNode, type ErrorInfo } from 'react';
  2. import { BrowserRouter, Routes, Route, Navigate } from 'react-router-dom';
  3. import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
  4. import { Layout } from './components/Layout';
  5. import { PrintersPage } from './pages/PrintersPage';
  6. import { ArchivesPage } from './pages/ArchivesPage';
  7. import { QueuePage } from './pages/QueuePage';
  8. import { StatsPage } from './pages/StatsPage';
  9. import { SettingsPage } from './pages/SettingsPage';
  10. import { ProfilesPage } from './pages/ProfilesPage';
  11. import { MaintenancePage } from './pages/MaintenancePage';
  12. import { ProjectsPage } from './pages/ProjectsPage';
  13. import { ProjectDetailPage } from './pages/ProjectDetailPage';
  14. import { FileManagerPage } from './pages/FileManagerPage';
  15. import { LibraryTrashPage } from './pages/LibraryTrashPage';
  16. import { CameraPage } from './pages/CameraPage';
  17. import { StreamOverlayPage } from './pages/StreamOverlayPage';
  18. import { ExternalLinkPage } from './pages/ExternalLinkPage';
  19. import { GroupEditPage } from './pages/GroupEditPage';
  20. import InventoryPage from './pages/InventoryPage';
  21. import { MakerworldPage } from './pages/MakerworldPage';
  22. import { SystemInfoPage } from './pages/SystemInfoPage';
  23. import { LoginPage } from './pages/LoginPage';
  24. import { SetupPage } from './pages/SetupPage';
  25. import { NotificationsPage } from './pages/NotificationsPage';
  26. import { GCodeViewerPage } from './pages/GCodeViewerPage';
  27. import { useWebSocket } from './hooks/useWebSocket';
  28. import { useStreamTokenSync } from './hooks/useCameraStreamToken';
  29. import { ThemeProvider } from './contexts/ThemeContext';
  30. import { ToastProvider } from './contexts/ToastContext';
  31. import { AuthProvider, useAuth } from './contexts/AuthContext';
  32. import { ColorCatalogProvider } from './contexts/ColorCatalogContext';
  33. import { SpoolBuddyLayout } from './components/spoolbuddy/SpoolBuddyLayout';
  34. import { SpoolBuddyDashboard } from './pages/spoolbuddy/SpoolBuddyDashboard';
  35. import { SpoolBuddyAmsPage } from './pages/spoolbuddy/SpoolBuddyAmsPage';
  36. import { SpoolBuddySettingsPage } from './pages/spoolbuddy/SpoolBuddySettingsPage';
  37. import { SpoolBuddyCalibrationPage } from './pages/spoolbuddy/SpoolBuddyCalibrationPage';
  38. import { SpoolBuddyWriteTagPage } from './pages/spoolbuddy/SpoolBuddyWriteTagPage';
  39. import { SpoolBuddyInventoryPage } from './pages/spoolbuddy/SpoolBuddyInventoryPage';
  40. class ErrorBoundary extends Component<{ children: ReactNode }, { error: Error | null; errorInfo: ErrorInfo | null }> {
  41. state = { error: null as Error | null, errorInfo: null as ErrorInfo | null };
  42. static getDerivedStateFromError(error: Error) {
  43. return { error };
  44. }
  45. componentDidCatch(error: Error, errorInfo: ErrorInfo) {
  46. this.setState({ errorInfo });
  47. console.error('React crash:', error, errorInfo);
  48. }
  49. render() {
  50. if (this.state.error) {
  51. return (
  52. <div style={{ padding: 24, color: '#ef4444', backgroundColor: '#18181b', minHeight: '100vh', fontFamily: 'monospace' }}>
  53. <h1 style={{ fontSize: 20, marginBottom: 12 }}>UI Crash</h1>
  54. <pre style={{ whiteSpace: 'pre-wrap', fontSize: 14 }}>{this.state.error.message}</pre>
  55. <pre style={{ whiteSpace: 'pre-wrap', fontSize: 12, color: '#a1a1aa', marginTop: 12 }}>
  56. {this.state.error.stack}
  57. </pre>
  58. <button
  59. onClick={() => { this.setState({ error: null, errorInfo: null }); }}
  60. style={{ marginTop: 16, padding: '8px 16px', backgroundColor: '#3b82f6', color: '#fff', border: 'none', borderRadius: 8, cursor: 'pointer' }}
  61. >
  62. Retry
  63. </button>
  64. </div>
  65. );
  66. }
  67. return this.props.children;
  68. }
  69. }
  70. const queryClient = new QueryClient({
  71. defaultOptions: {
  72. queries: {
  73. staleTime: 1000 * 60,
  74. retry: 1,
  75. },
  76. },
  77. });
  78. function StreamTokenSync() {
  79. useStreamTokenSync();
  80. return null;
  81. }
  82. function WebSocketProvider({ children }: { children: React.ReactNode }) {
  83. useWebSocket();
  84. return <>{children}</>;
  85. }
  86. function ProtectedRoute({ children }: { children: React.ReactNode }) {
  87. const { authEnabled, loading, user } = useAuth();
  88. if (loading) {
  89. return <div className="min-h-screen flex items-center justify-center">Loading...</div>;
  90. }
  91. if (authEnabled && !user) {
  92. return <Navigate to="/login" replace />;
  93. }
  94. return <>{children}</>;
  95. }
  96. function PermissionRoute({ permission, children }: { permission: string; children: React.ReactNode }) {
  97. // Permission-gated route: any user with the given permission can enter, not
  98. // just admins. Individual components below this guard apply their own
  99. // per-action permission checks. Used for pages where delegation is supported
  100. // (e.g. settings:read grants read-only access to Settings; specific tabs
  101. // require their own permissions like users:read, groups:update, etc.).
  102. const { authEnabled, loading, user, hasPermission } = useAuth();
  103. if (loading) {
  104. return <div className="min-h-screen flex items-center justify-center">Loading...</div>;
  105. }
  106. // Auth disabled → open access (backward compatibility)
  107. if (!authEnabled) {
  108. return <>{children}</>;
  109. }
  110. if (!user) {
  111. return <Navigate to="/login" replace />;
  112. }
  113. if (!hasPermission(permission as Parameters<typeof hasPermission>[0])) {
  114. return <Navigate to="/" replace />;
  115. }
  116. return <>{children}</>;
  117. }
  118. function SetupRoute({ children }: { children: React.ReactNode }) {
  119. const { authEnabled, loading } = useAuth();
  120. if (loading) {
  121. return <div className="min-h-screen flex items-center justify-center">Loading...</div>;
  122. }
  123. // If auth is already enabled, redirect to login
  124. // Otherwise, allow access to setup page (even if setup was completed before)
  125. // This allows users to enable auth later if they skipped it during initial setup
  126. if (authEnabled) {
  127. return <Navigate to="/login" replace />;
  128. }
  129. return <>{children}</>;
  130. }
  131. function App() {
  132. return (
  133. <ErrorBoundary>
  134. <ThemeProvider>
  135. <ToastProvider>
  136. <QueryClientProvider client={queryClient}>
  137. <AuthProvider>
  138. <ColorCatalogProvider>
  139. <StreamTokenSync />
  140. <BrowserRouter>
  141. <Routes>
  142. {/* Setup page - only accessible if auth not enabled */}
  143. <Route path="/setup" element={<SetupRoute><SetupPage /></SetupRoute>} />
  144. {/* Login page */}
  145. <Route path="/login" element={<LoginPage />} />
  146. {/* Camera page - standalone, no layout, no WebSocket (doesn't need real-time updates) */}
  147. <Route path="/camera/:printerId" element={<CameraPage />} />
  148. {/* Stream overlay page - standalone for OBS/streaming embeds, no auth required */}
  149. <Route path="/overlay/:printerId" element={<StreamOverlayPage />} />
  150. {/* SpoolBuddy kiosk UI */}
  151. <Route element={<ProtectedRoute><WebSocketProvider><SpoolBuddyLayout /></WebSocketProvider></ProtectedRoute>}>
  152. <Route path="spoolbuddy" element={<SpoolBuddyDashboard />} />
  153. <Route path="spoolbuddy/ams" element={<SpoolBuddyAmsPage />} />
  154. <Route path="spoolbuddy/write-tag" element={<SpoolBuddyWriteTagPage />} />
  155. <Route path="spoolbuddy/inventory" element={<SpoolBuddyInventoryPage />} />
  156. <Route path="spoolbuddy/settings" element={<SpoolBuddySettingsPage />} />
  157. <Route path="spoolbuddy/calibration" element={<SpoolBuddyCalibrationPage />} />
  158. </Route>
  159. {/* Main app with WebSocket for real-time updates */}
  160. <Route element={<ProtectedRoute><WebSocketProvider><Layout /></WebSocketProvider></ProtectedRoute>}>
  161. <Route index element={<PrintersPage />} />
  162. <Route path="archives" element={<ArchivesPage />} />
  163. <Route path="queue" element={<QueuePage />} />
  164. <Route path="stats" element={<StatsPage />} />
  165. <Route path="profiles" element={<ProfilesPage />} />
  166. <Route path="maintenance" element={<MaintenancePage />} />
  167. <Route path="projects" element={<ProjectsPage />} />
  168. <Route path="projects/:id" element={<ProjectDetailPage />} />
  169. <Route path="inventory" element={<InventoryPage />} />
  170. <Route path="files" element={<FileManagerPage />} />
  171. <Route path="files/trash" element={<LibraryTrashPage />} />
  172. <Route path="makerworld" element={<MakerworldPage />} />
  173. <Route path="settings" element={<PermissionRoute permission="settings:read"><SettingsPage /></PermissionRoute>} />
  174. <Route path="groups/new" element={<PermissionRoute permission="groups:create"><GroupEditPage /></PermissionRoute>} />
  175. <Route path="groups/:id/edit" element={<PermissionRoute permission="groups:update"><GroupEditPage /></PermissionRoute>} />
  176. <Route path="users" element={<Navigate to="/settings?tab=users" replace />} />
  177. <Route path="groups" element={<Navigate to="/settings?tab=users" replace />} />
  178. <Route path="system" element={<SystemInfoPage />} />
  179. <Route path="notifications" element={<NotificationsPage />} />
  180. <Route path="gcode-viewer" element={<GCodeViewerPage />} />
  181. <Route path="external/:id" element={<ExternalLinkPage />} />
  182. </Route>
  183. </Routes>
  184. </BrowserRouter>
  185. </ColorCatalogProvider>
  186. </AuthProvider>
  187. </QueryClientProvider>
  188. </ToastProvider>
  189. </ThemeProvider>
  190. </ErrorBoundary>
  191. );
  192. }
  193. export default App;