From afeeed281e60466b576fbe74d339634cc5d07b82 Mon Sep 17 00:00:00 2001 From: Codex Assistant <codex@example.com> Date: 星期三, 08 十月 2025 08:56:42 +0800 Subject: [PATCH] 修复评审功能和用户认证问题 --- web/src/config/api.ts | 90 +++++++++++++++++++++++++++++++++++---------- 1 files changed, 70 insertions(+), 20 deletions(-) diff --git a/web/src/config/api.ts b/web/src/config/api.ts index 485abb2..ecbdb5e 100644 --- a/web/src/config/api.ts +++ b/web/src/config/api.ts @@ -17,36 +17,86 @@ // GraphQL璇锋眰宸ュ叿鍑芥暟 export const graphqlRequest = async (query: string, variables: any = {}) => { - // 鑾峰彇JWT token - const { getToken } = await import('@/utils/auth'); + // 鑾峰彇JWT token涓庡伐鍏� + const { getToken, isTokenExpired, clearAuth } = await import('@/utils/auth'); const token = getToken(); + + // 鑻oken杩囨湡锛岀洿鎺ユ竻鐞嗗苟璺崇櫥褰� + if (!token || isTokenExpired(token)) { + clearAuth(); + // 閬垮厤鍦ㄧ櫥褰曢〉閲嶅璺宠浆閫犳垚鐧藉睆/寰幆 + const atLogin = typeof window !== 'undefined' && window.location && window.location.hash?.startsWith('#/login'); + if (!atLogin) { + window.location.href = '/#/login'; + } + throw new Error('Token expired or missing') + } + + // 鏋勫缓璇锋眰澶� const headers: Record<string, string> = { 'Content-Type': 'application/json', }; + if (token) { headers['Authorization'] = `Bearer ${token}`; } - const response = await fetch(API_CONFIG.GRAPHQL_ENDPOINT, { - method: 'POST', - headers: headers, - body: JSON.stringify({ - query, - variables, - }), - }) + // 鏋勫缓璇锋眰浣� + const requestBody = JSON.stringify({ + query, + variables, + }); - if (!response.ok) { - throw new Error(`HTTP error! status: ${response.status}`) + try { + // 鍙戦�佽姹� + const response = await fetch(API_CONFIG.GRAPHQL_ENDPOINT, { + method: 'POST', + headers, + body: requestBody, + }); + + if (!response.ok) { + // 澶勭悊401鏈巿鏉� + if (response.status === 401) { + const { clearAuth } = await import('@/utils/auth'); + clearAuth(); + const atLogin = typeof window !== 'undefined' && window.location && window.location.hash?.startsWith('#/login'); + if (!atLogin) { + window.location.href = '/#/login'; + } + } + throw new Error(`HTTP error! status: ${response.status}`); + } + + const result = await response.json(); + + if (result.errors) { + const msg = JSON.stringify(result.errors) || '' + // 璇嗗埆璁よ瘉绫婚敊璇叧閿瓧 + const isAuthError = + msg.includes('Unauthorized') || + msg.includes('璁よ瘉') || + msg.includes('unauthorized') || + msg.includes('invalid token') || + msg.includes('expired') + + if (isAuthError) { + const { clearAuth } = await import('@/utils/auth'); + clearAuth(); + const atLogin = typeof window !== 'undefined' && window.location && window.location.hash?.startsWith('#/login'); + if (!atLogin) { + window.location.href = '/#/login'; + } + } + throw new Error(`GraphQL errors: ${msg}`); + } + + return result; + } catch (error) { + console.error('=== GraphQL璇锋眰澶辫触 ==='); + console.error('閿欒璇︽儏:', error); + throw error; } - - const result = await response.json() - - if (result.errors) { - throw new Error(result.errors[0].message) - } - - return result.data } // 閫氱敤API璇锋眰宸ュ叿鍑芥暟 -- Gitblit v1.8.0