From c4a9cad1c50e89365e2a58b50e259af642ed3b8c Mon Sep 17 00:00:00 2001 From: Codex Assistant <codex@example.com> Date: 星期二, 07 十月 2025 16:12:20 +0800 Subject: [PATCH] feat(review): 调整评审详情展示顺序与样式,描述支持多行,项目信息列宽40/60 fix(auth): 登录页与首页循环跳转保护;api.ts 在登录页不再重定向;401分支在登录页不跳转 fix(router): /login 放行策略优化,避免死循环;评审列表跳转到 /project-review/:id/detail fix(frontend): 补齐 utils/appConfig.ts,避免启动白屏 fix(review): 详情页提交评分缺少stageId时回退使用项目详情的stageId feat(backend): ActivityPlayerDetailResponse.playerInfo 补充 avatarUrl/avatar,服务组装时填充用户头像 chore(dev): 启动脚本注入本地JWT密钥,重启前后端 --- web/src/utils/auth.ts | 44 ++++++++++++++++++++++++++++++++++++++++++++ 1 files changed, 44 insertions(+), 0 deletions(-) diff --git a/web/src/utils/auth.ts b/web/src/utils/auth.ts index a55e90c..3cb965e 100644 --- a/web/src/utils/auth.ts +++ b/web/src/utils/auth.ts @@ -75,6 +75,50 @@ return !!(token && userInfo) } +/** + * 瑙f瀽骞跺垽鏂璊WT鏄惁杩囨湡 + * 瑙勫垯锛氳嫢token涓虹┖鎴栨棤娉曡В鏋恊xp锛岃涓鸿繃鏈燂紱exp鍗曚綅涓虹 + */ +export function isTokenExpired(token?: string | null): boolean { + if (!token) return true + try { + const parts = token.split('.') + if (parts.length !== 3) return true + const payloadJson = JSON.parse(decodeBase64Url(parts[1])) + const exp = payloadJson?.exp + if (!exp || typeof exp !== 'number') return true + const now = Math.floor(Date.now() / 1000) + return exp <= now + } catch (e) { + console.error('瑙f瀽JWT澶辫触:', e) + return true + } +} + +/** + * Base64Url 瑙g爜 + */ +function decodeBase64Url(input: string): string { + // 鏇挎崲URL瀹夊叏瀛楃骞惰ˉ榻�'=' + let base64 = input.replace(/-/g, '+').replace(/_/g, '/') + const pad = base64.length % 4 + if (pad) { + base64 += '='.repeat(4 - pad) + } + const decoded = atob(base64) + // 澶勭悊UTF-8 + try { + return decodeURIComponent( + decoded + .split('') + .map(c => '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2)) + .join('') + ) + } catch { + return decoded + } +} + // 娓呴櫎鎵�鏈夎璇佹暟鎹� export function clearAuth(): void { removeToken() -- Gitblit v1.8.0