fix: persist auth across refreshes with onFinishHydration and safety timeout
Some checks failed
CI Pipeline / Rust Format Check (push) Failing after 4s
CI Pipeline / Clippy Lints (push) Successful in 45s
CI Pipeline / Rust Unit Tests (push) Successful in 1m2s
CI Pipeline / Security Audit (push) Successful in 4s
CI Pipeline / Frontend Lint & Type Check (push) Failing after 10s
CI Pipeline / Build .deb & Release (push) Has been skipped
Some checks failed
CI Pipeline / Rust Format Check (push) Failing after 4s
CI Pipeline / Clippy Lints (push) Successful in 45s
CI Pipeline / Rust Unit Tests (push) Successful in 1m2s
CI Pipeline / Security Audit (push) Successful in 4s
CI Pipeline / Frontend Lint & Type Check (push) Failing after 10s
CI Pipeline / Build .deb & Release (push) Has been skipped
This commit is contained in:
@ -1,8 +1,8 @@
|
|||||||
|
import { useEffect } from 'react'
|
||||||
import { Routes, Route, Navigate } from 'react-router-dom'
|
import { Routes, Route, Navigate } from 'react-router-dom'
|
||||||
import { CssBaseline, ThemeProvider } from '@mui/material'
|
import { CssBaseline, ThemeProvider, CircularProgress, Box } from '@mui/material'
|
||||||
import { darkTheme } from './theme/theme'
|
import { darkTheme } from './theme/theme'
|
||||||
import { useAuthStore } from './store/authStore'
|
import { useAuthStore } from './store/authStore'
|
||||||
import { CircularProgress, Box } from '@mui/material'
|
|
||||||
import AppLayout from './components/AppLayout'
|
import AppLayout from './components/AppLayout'
|
||||||
import LoginPage from './pages/LoginPage'
|
import LoginPage from './pages/LoginPage'
|
||||||
import MfaSetupPage from './pages/MfaSetupPage'
|
import MfaSetupPage from './pages/MfaSetupPage'
|
||||||
@ -33,33 +33,82 @@ function RequireAuth({ children }: { children: React.ReactNode }) {
|
|||||||
return isAuthenticated ? <>{children}</> : <Navigate to="/login" replace />
|
return isAuthenticated ? <>{children}</> : <Navigate to="/login" replace />
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Waits for Zustand persist to finish rehydrating from localStorage,
|
||||||
|
* then calls restoreSession() so it can see the persisted refreshToken.
|
||||||
|
* Includes a safety timeout in case anything hangs.
|
||||||
|
*/
|
||||||
|
function AuthRestorer({ children }: { children: React.ReactNode }) {
|
||||||
|
const restoreSession = useAuthStore((s) => s.restoreSession)
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
let cancelled = false
|
||||||
|
|
||||||
|
// Safety timeout: force isRestoring=false if restoration doesn't complete in 15s
|
||||||
|
const timeout = setTimeout(() => {
|
||||||
|
if (!cancelled) {
|
||||||
|
console.warn('[auth] Restoration timeout — forcing isRestoring=false')
|
||||||
|
useAuthStore.setState({ isRestoring: false })
|
||||||
|
}
|
||||||
|
}, 15_000)
|
||||||
|
|
||||||
|
const doRestore = () => {
|
||||||
|
if (!cancelled) restoreSession()
|
||||||
|
}
|
||||||
|
|
||||||
|
let unsub: (() => void) | undefined
|
||||||
|
|
||||||
|
// Only call restoreSession AFTER Zustand has rehydrated the persisted state
|
||||||
|
if (useAuthStore.persist.hasHydrated()) {
|
||||||
|
console.log('[auth] Store already hydrated, restoring session')
|
||||||
|
doRestore()
|
||||||
|
} else {
|
||||||
|
console.log('[auth] Waiting for Zustand hydration...')
|
||||||
|
unsub = useAuthStore.persist.onFinishHydration(() => {
|
||||||
|
console.log('[auth] Hydration complete, restoring session')
|
||||||
|
doRestore()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
cancelled = true
|
||||||
|
clearTimeout(timeout)
|
||||||
|
unsub?.()
|
||||||
|
}
|
||||||
|
}, [restoreSession])
|
||||||
|
|
||||||
|
return <>{children}</>
|
||||||
|
}
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
return (
|
return (
|
||||||
<ThemeProvider theme={darkTheme}>
|
<ThemeProvider theme={darkTheme}>
|
||||||
<CssBaseline />
|
<CssBaseline />
|
||||||
<Routes>
|
<AuthRestorer>
|
||||||
{/* Public */}
|
<Routes>
|
||||||
<Route path="/login" element={<LoginPage />} />
|
{/* Public */}
|
||||||
|
<Route path="/login" element={<LoginPage />} />
|
||||||
|
|
||||||
{/* Protected — wrapped in AppLayout with sidebar navigation */}
|
{/* Protected — wrapped in AppLayout with sidebar navigation */}
|
||||||
<Route element={<RequireAuth><AppLayout /></RequireAuth>}>
|
<Route element={<RequireAuth><AppLayout /></RequireAuth>}>
|
||||||
<Route path="/" element={<Navigate to="/dashboard" replace />} />
|
<Route path="/" element={<Navigate to="/dashboard" replace />} />
|
||||||
<Route path="/mfa/setup" element={<MfaSetupPage />} />
|
<Route path="/mfa/setup" element={<MfaSetupPage />} />
|
||||||
<Route path="/dashboard" element={<DashboardPage />} />
|
<Route path="/dashboard" element={<DashboardPage />} />
|
||||||
<Route path="/hosts" element={<HostsPage />} />
|
<Route path="/hosts" element={<HostsPage />} />
|
||||||
<Route path="/hosts/:id" element={<HostDetailPage />} />
|
<Route path="/hosts/:id" element={<HostDetailPage />} />
|
||||||
<Route path="/groups" element={<GroupsPage />} />
|
<Route path="/groups" element={<GroupsPage />} />
|
||||||
<Route path="/users" element={<UsersPage />} />
|
<Route path="/users" element={<UsersPage />} />
|
||||||
<Route path="/jobs" element={<JobsPage />} />
|
<Route path="/jobs" element={<JobsPage />} />
|
||||||
<Route path="/deployment" element={<PatchDeploymentPage />} />
|
<Route path="/deployment" element={<PatchDeploymentPage />} />
|
||||||
<Route path="/maintenance" element={<MaintenanceWindowsPage />} />
|
<Route path="/maintenance" element={<MaintenanceWindowsPage />} />
|
||||||
<Route path="/reports" element={<ReportsPage />} />
|
<Route path="/reports" element={<ReportsPage />} />
|
||||||
<Route path="/certificates" element={<CertificatesPage />} />
|
<Route path="/certificates" element={<CertificatesPage />} />
|
||||||
<Route path="/settings" element={<SettingsPage />} />
|
<Route path="/settings" element={<SettingsPage />} />
|
||||||
</Route>
|
</Route>
|
||||||
|
|
||||||
<Route path="*" element={<Navigate to="/dashboard" replace />} />
|
<Route path="*" element={<Navigate to="/dashboard" replace />} />
|
||||||
</Routes>
|
</Routes>
|
||||||
|
</AuthRestorer>
|
||||||
</ThemeProvider>
|
</ThemeProvider>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@ -12,11 +12,12 @@ interface AuthState {
|
|||||||
setTokens: (access: string, refresh: string) => void
|
setTokens: (access: string, refresh: string) => void
|
||||||
setUser: (user: User) => void
|
setUser: (user: User) => void
|
||||||
logout: () => void
|
logout: () => void
|
||||||
|
restoreSession: () => Promise<void>
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useAuthStore = create<AuthState>()(
|
export const useAuthStore = create<AuthState>()(
|
||||||
persist(
|
persist(
|
||||||
(set) => ({
|
(set, get) => ({
|
||||||
accessToken: null,
|
accessToken: null,
|
||||||
refreshToken: null,
|
refreshToken: null,
|
||||||
user: null,
|
user: null,
|
||||||
@ -30,40 +31,47 @@ export const useAuthStore = create<AuthState>()(
|
|||||||
|
|
||||||
logout: () =>
|
logout: () =>
|
||||||
set({ accessToken: null, refreshToken: null, user: null, isAuthenticated: false, isRestoring: false }),
|
set({ accessToken: null, refreshToken: null, user: null, isAuthenticated: false, isRestoring: false }),
|
||||||
|
|
||||||
|
restoreSession: async () => {
|
||||||
|
const { refreshToken } = get()
|
||||||
|
if (!refreshToken) {
|
||||||
|
console.log('[auth] No refresh token found, skipping restoration')
|
||||||
|
set({ isRestoring: false })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { data } = await axios.post(
|
||||||
|
'/api/v1/auth/refresh',
|
||||||
|
{ refresh_token: refreshToken },
|
||||||
|
{ timeout: 10000 }
|
||||||
|
)
|
||||||
|
console.log('[auth] Token refresh successful')
|
||||||
|
set({
|
||||||
|
accessToken: data.access_token,
|
||||||
|
refreshToken: data.refresh_token,
|
||||||
|
user: data.user ?? get().user,
|
||||||
|
isAuthenticated: true,
|
||||||
|
isRestoring: false,
|
||||||
|
})
|
||||||
|
} catch (err: unknown) {
|
||||||
|
const status = (err as { response?: { status?: number } })?.response?.status
|
||||||
|
const message = (err as Error)?.message
|
||||||
|
console.warn('[auth] Token refresh failed:', status, message)
|
||||||
|
set({
|
||||||
|
accessToken: null,
|
||||||
|
refreshToken: null,
|
||||||
|
user: null,
|
||||||
|
isAuthenticated: false,
|
||||||
|
isRestoring: false,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
}),
|
}),
|
||||||
{
|
{
|
||||||
name: 'pm-auth',
|
name: 'pm-auth',
|
||||||
// Only persist refresh token; access token regenerated on load
|
// Only persist refresh token; access token regenerated on load
|
||||||
partialize: (state) => ({ refreshToken: state.refreshToken, user: state.user }),
|
partialize: (state) => ({ refreshToken: state.refreshToken, user: state.user }),
|
||||||
onRehydrateStorage: () => {
|
|
||||||
return (state) => {
|
|
||||||
if (state?.refreshToken) {
|
|
||||||
// Proactively refresh the access token using the persisted refresh token
|
|
||||||
axios.post('/api/v1/auth/refresh', { refresh_token: state.refreshToken })
|
|
||||||
.then(({ data }) => {
|
|
||||||
useAuthStore.setState({
|
|
||||||
accessToken: data.access_token,
|
|
||||||
refreshToken: data.refresh_token,
|
|
||||||
isAuthenticated: true,
|
|
||||||
isRestoring: false,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
.catch(() => {
|
|
||||||
// Refresh token expired or invalid — clear all auth state
|
|
||||||
useAuthStore.setState({
|
|
||||||
accessToken: null,
|
|
||||||
refreshToken: null,
|
|
||||||
user: null,
|
|
||||||
isAuthenticated: false,
|
|
||||||
isRestoring: false,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
// No refresh token — not logged in, skip restoration
|
|
||||||
useAuthStore.setState({ isRestoring: false })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|||||||
Reference in New Issue
Block a user