Files
leaudit-platform-frontend/app/components/auth/ClientAuthGuard.tsx
T

62 lines
2.0 KiB
TypeScript

/**
* 客户端认证守卫组件
*
* 在客户端检查 localStorage 中的 token
* 如果未认证且访问的是需要认证的路径,则跳转到登录页
*/
import { useEffect } from 'react';
import { useNavigate, useLocation } from '@remix-run/react';
import { isAuthenticated } from '~/utils/auth-storage';
interface ClientAuthGuardProps {
isPublicPath: boolean;
frontendJWT?: string;
userInfo?: Record<string, unknown>;
}
export function ClientAuthGuard({ isPublicPath, frontendJWT, userInfo }: ClientAuthGuardProps) {
const navigate = useNavigate();
const location = useLocation();
useEffect(() => {
console.log('🔍 [Auth Guard] useEffect 触发', {
isPublicPath,
pathname: location.pathname
});
// 如果是公共路径,不需要检查认证
if (isPublicPath) {
console.log('✅ [Auth Guard] 公共路径,跳过认证检查');
return;
}
// 优先用服务端 session 回传的数据回填 localStorage,避免刚登录时客户端误判未登录
const token = localStorage.getItem('access_token');
if (!token && frontendJWT) {
localStorage.setItem('access_token', frontendJWT);
if (userInfo) {
localStorage.setItem('user_info', JSON.stringify(userInfo));
}
console.log('✅ [Auth Guard] 已根据服务端 session 回填本地认证数据');
}
const authenticated = isAuthenticated() || !!frontendJWT;
if (!authenticated) {
console.log('🔒 [Auth Guard] 未认证,重定向到登录页');
// 保存当前路径,登录后可以跳转回来
const redirectTo = location.pathname !== '/login' ? location.pathname : '/';
// 跳转到登录页,并传递重定向目标
navigate(`/login?redirect=${encodeURIComponent(redirectTo)}`, { replace: true });
} else {
// console.log('✅ [Auth Guard] 已认证,允许访问');
}
}, [isPublicPath, navigate, location.pathname, frontendJWT, userInfo]);
// 这个组件不渲染任何内容
return null;
}