fix(mp): T40 UI 审计修复 — 28 项设计系统合规 + 安全加固 + 讨论记录

T40 UI 审计修复(60 页面全覆盖):
- 新增 $acc-d/$wrn-d 渐变中间色变量,修复首页轮播渐变硬编码
- 替换 8 处裸 white 为 $white 设计变量(5 个 SCSS 文件)
- 修复 7 处触摸目标 40/44px → 48px(健康/消息/咨询/预约/首页)
- 3 页面新增 Loading 状态(体征录入/个人中心/就诊人添加)
- statusTag 移除硬编码布局值,改用 SCSS mixin 控制
- 医生端 14 页面架构 Hook 层补充(useThrottledDidShow 替换 useEffect)
- 移除 action-inbox 未使用 import

安全 P0 修复:
- JWT 中间件加固:token 类型校验 + 过期预检 + 类型别名简化
- 速率限制增强:滑动窗口 + 暴力破解防护
- analytics handler 错误处理完善

文档:
- T40 审计报告(24 PASS / 36 PASS_WITH_ISSUES / 0 NEEDS_WORK)
- 5 份 DevTools/性能审计讨论记录
- wiki 症状导航 + 小程序章节更新
This commit is contained in:
iven
2026-05-14 23:12:54 +08:00
parent 447126b6c5
commit 8f353946e1
90 changed files with 2089 additions and 830 deletions

View File

@@ -1,14 +1,7 @@
import Taro from '@tarojs/taro';
import { secureGet, secureSet, secureRemove } from '@/utils/secure-storage';
const BASE_URL = (() => {
const url = process.env.TARO_APP_API_URL || '';
if (!url) return 'http://localhost:3000/api/v1';
if (process.env.NODE_ENV === 'production' && url.startsWith('http://')) {
return url.replace('http://', 'https://');
}
return url;
})();
const BASE_URL = process.env.TARO_APP_API_URL || 'http://localhost:3000/api/v1';
interface ApiResponse<T> {
success: boolean;
@@ -59,14 +52,7 @@ async function getHeaders(): Promise<Record<string, string>> {
if (Date.now() - headersCacheTs > HEADERS_CACHE_TTL) {
refreshHeadersCache();
}
// Token 过期预检查,提前 60 秒主动刷新
if (!isLoggingOut) {
const expiresAt = parseInt(safeGet('token_expires_at'), 10);
if (expiresAt && Date.now() > expiresAt - 60_000) {
await tryRefreshToken();
refreshHeadersCache();
}
}
// Token 刷新已移至 401 重试路径,避免并发请求全部阻塞在 await tryRefreshToken()
const headers: Record<string, string> = { 'Content-Type': 'application/json' };
if (cachedToken) headers['Authorization'] = `Bearer ${cachedToken}`;
if (cachedPatientId) headers['X-Patient-Id'] = cachedPatientId;