App.tsx 9.9 KB

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