feat(admin): Add user management and upgrade to module permission system

Features - User Management (Phase 4.1):
- Database: Add user_modules table for fine-grained module permissions
- Database: Add 4 user permissions (view/create/edit/delete) to role_permissions
- Backend: UserService (780 lines) - CRUD with tenant isolation
- Backend: UserController + UserRoutes (648 lines) - 13 API endpoints
- Backend: Batch import users from Excel
- Frontend: UserListPage (412 lines) - list/filter/search/pagination
- Frontend: UserFormPage (341 lines) - create/edit with module config
- Frontend: UserDetailPage (393 lines) - details/tenant/module management
- Frontend: 3 modal components (592 lines) - import/assign/configure
- API: GET/POST/PUT/DELETE /api/admin/users/* endpoints

Architecture Upgrade - Module Permission System:
- Backend: Add getUserModules() method in auth.service
- Backend: Login API returns modules array in user object
- Frontend: AuthContext adds hasModule() method
- Frontend: Navigation filters modules based on user.modules
- Frontend: RouteGuard checks requiredModule instead of requiredVersion
- Frontend: Remove deprecated version-based permission system
- UX: Only show accessible modules in navigation (clean UI)
- UX: Smart redirect after login (avoid 403 for regular users)

Fixes:
- Fix UTF-8 encoding corruption in ~100 docs files
- Fix pageSize type conversion in userService (String to Number)
- Fix authUser undefined error in TopNavigation
- Fix login redirect logic with role-based access check
- Update Git commit guidelines v1.2 with UTF-8 safety rules

Database Changes:
- CREATE TABLE user_modules (user_id, tenant_id, module_code, is_enabled)
- ADD UNIQUE CONSTRAINT (user_id, tenant_id, module_code)
- INSERT 4 permissions + role assignments
- UPDATE PUBLIC tenant with 8 module subscriptions

Technical:
- Backend: 5 new files (~2400 lines)
- Frontend: 10 new files (~2500 lines)
- Docs: 1 development record + 2 status updates + 1 guideline update
- Total: ~4900 lines of code

Status: User management 100% complete, module permission system operational
This commit is contained in:
2026-01-16 13:42:10 +08:00
parent 98d862dbd4
commit 66255368b7
560 changed files with 70424 additions and 52353 deletions

View File

@@ -3,31 +3,31 @@
## 📋 问题概述
**报告时间**: 2026-01-06
**靽桀<EFBFBD><EFBFBD><EFBFBD>?*: <20>?**撌脣<E6928C><E884A3>其耨憭?*
**<EFBFBD><EFBFBD><EFBFBD><EFBFBD>**: <EFBFBD><EFBFBD><EFBFBD><EFBFBD> + <20><EFBFBD><E7AC94>曉笆瘥?
**修复状态**: ✅ **已全部修复**
**问题来源**: 用户反馈 + 原型图对比
---
## <EFBFBD><EFBFBD> <20>𤑳緵<F0A491B3>?銝芷䔮憸?
## 🐛 发现的4个问题
### <EFBFBD><EFBFBD>1嚗𡁻△<EFBFBD><EFBFBD><EFBFBD><EFBFBD>撅𧶏<EFBFBD>憿園<EFBFBD><EFBFBD>匧紡<EFBFBD><EFBFBD> <20>?
**<EFBFBD>啗情**: 餈𥕦<EFBFBD>Workspace憿菟𢒰<EFBFBD>𠬍<EFBFBD>憿園<EFBFBD>隞滨<EFBFBD><EFBFBD>曄內撟喳蝱<EFBFBD><EFBFBD><EFBFBD><EFBFBD>撖潸⏛<EFBFBD>?
### 问题1页面不是全屏顶部有导航栏 ❌
**现象**: 进入Workspace页面后,顶部仍然显示平台的全局导航栏
**原因**: WorkspacePage被包裹在`MainLayout`中,导致继承了外层布局
**影响**:
- 銝滨泵<EFBFBD>Ā3霈曇恣<EFBFBD>?瘝㗇絡撘?雿㯄<E99BBF>
- 不符合V3设计的"沉浸式"体验
- 浪费屏幕空间
- 銝𤾸<EFBFBD><EFBFBD>见㦛銝滢<EFBFBD><EFBFBD>?
- 与原型图不一致
---
### <EFBFBD><EFBFBD>2嚗𡁏瓷<EFBFBD><EFBFBD>銝𧢲<EFBFBD><EFBFBD>冽辺 <20>?
**<EFBFBD>啗情**: Chat<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>皛𡁜𢆡嚗<EFBFBD><EFBFBD>摰寡<EFBFBD><EFBFBD>箸𧒄<EFBFBD><EFBFBD><EFBFBD>?
### 问题2没有上下滚动条 ❌
**现象**: Chat消息区域无法滚动,内容超出时看不到
**原因**:
- 摰孵膥瘝⊥<EFBFBD><EFBFBD>霈曄蔭`overflow`撅墧<EFBFBD>?
- 擃睃漲霈∠<EFBFBD>銝齿迤蝖?
- 容器没有正确设置`overflow`属性
- 高度计算不正确
**影响**:
- 无法查看历史消息
@@ -35,34 +35,34 @@
---
### <EFBFBD><EFBFBD>3嚗関ab撖潸⏛擃睃漲憭芷<EFBFBD> <20>?
**<EFBFBD>啗情**: "<EFBFBD><EFBFBD><EFBFBD><EFBFBD>"<22>?<3F><EFBFBD><EFBFBD>漣"Tab撖潸⏛<E6BDB8><EFBFBD>摨西<E691A8>擃矋<E69383><E79F8B><EFBFBD>蝎㛖<E89D8E>
### 问题3Tab导航高度太高 ❌
**现象**: "智能问答"和"知识资产"Tab导航栏高度过高显得粗糙
**原因**:
- 雿輻鍂鈭<EFBFBD>h-14`嚗?6px嚗㕑<E59A97><EFBFBD><E5B1B8><EFBFBD><E7AC94><EFBFBD>h-12`嚗?8px嚗?
- <EFBFBD><EFBFBD><EFBFBD><EFBFBD>摮堒偕撖貉<EFBFBD>憭?
- 使用了`h-14`56px而非原型的`h-12`48px
- 图标和文字尺寸过大
**影响**:
- 銝𤾸<EFBFBD><EFBFBD>见㦛銝滢<EFBFBD><EFBFBD>?
- <EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>銝滨移<EFBFBD>?
- 与原型图不一致
- 视觉效果不精致
---
### <EFBFBD><EFBFBD>4嚗鋫I撖寡<EFBFBD><EFBFBD>仿<EFBFBD> - JSON<4F><EFBFBD><E6BE86>躰秤 <20>?
### 问题4AI对话报错 - JSON格式错误 ❌
**现象**: 发送消息后报错:`Unexpected token 'd', "data: {"co"... is not valid JSON`
**原因**:
- <20>𡒊垢餈𥪜<E9A488><F0A5AA9C><EFBFBD>SSE嚗𠄎erver-Sent Events嚗㗇<EFBFBD>撘𤩺聢撘?
- <20>滨垢雿輻鍂`response.json()`閫<><E996AB><EFBFBD><EFBFBD>湔聢撘誯<E69298>霂?
- 后端返回的是SSEServer-Sent Events)流式格式
- 前端使用`response.json()`解析,导致格式错误
- 没有正确处理`data: `前缀
**影响**:
- 完全无法使用AI对话功能
- <20><EFBFBD><E8A9A8><EFBFBD>銝滚虾<E6BB9A>?
- 核心功能不可用
---
## <EFBFBD>?靽桀<E99DBD><E6A180><EFBFBD>
## ✅ 修复方案
### 修复1实现全屏沉浸式布局
@@ -75,10 +75,10 @@ interface WorkspacePageProps {
}
const WorkspacePage: React.FC<WorkspacePageProps> = ({ standalone = false }) => {
// <EFBFBD><EFBFBD><EFBFBD>tandalone<EFBFBD>嚗䔶蝙<EFBFBD>典𤐄摰𡁜<EFBFBD>雿滩<EFBFBD><EFBFBD>𡝗㟲銝芸<EFBFBD>撟?
// 如果是standalone模式,使用固定定位覆盖整个屏幕
const containerClass = standalone
? "fixed inset-0 z-50 flex flex-col bg-gray-50" // 全屏覆盖
: "flex flex-col h-screen bg-gray-50"; // <EFBFBD><EFBFBD>𡁏芋撘?
: "flex flex-col h-screen bg-gray-50"; // 普通模式
return (
<div className={containerClass}>
@@ -100,9 +100,9 @@ const WorkspacePage: React.FC<WorkspacePageProps> = ({ standalone = false }) =>
```
#### 效果
<EFBFBD>?Workspace憿菟𢒰摰<EFBFBD><EFBFBD><EFBFBD><EFBFBD>撅誩<EFBFBD>
<EFBFBD>?瘝⊥<E7989D>憿園<E686BF>撖潸⏛<E6BDB8>?
<EFBFBD>?瘝㗇絡撘譍<E69298>撉?
Workspace页面完全覆盖屏幕
✅ 没有顶部导航栏
✅ 沉浸式体验
---
@@ -110,17 +110,17 @@ const WorkspacePage: React.FC<WorkspacePageProps> = ({ standalone = false }) =>
#### 代码修改
```typescript
// WorkspacePage.tsx - 銝餃捆<EFBFBD>?
// WorkspacePage.tsx - 主容器
<main className="flex-1 overflow-hidden relative">
{activeTab === 'chat' && (
<div className="h-full flex overflow-hidden"> {/* 添加overflow-hidden */}
<div className="flex-1 flex flex-col bg-white overflow-hidden">
{/* 撌乩<E6928C><E79285><E288AA>㗇𥋘<E39787>?*/}
{/* 工作模式选择器 */}
<div className="p-3 border-b border-gray-100 flex-shrink-0">
{/* ... */}
</div>
{/* Chat<EFBFBD><EFBFBD> - <20><EFBFBD><E88880>?*/}
{/* Chat区域 - 可滚动 */}
<div className="flex-1 overflow-y-auto"> {/* 添加overflow-y-auto */}
{/* ... */}
</div>
@@ -137,16 +137,16 @@ const WorkspacePage: React.FC<WorkspacePageProps> = ({ standalone = false }) =>
<Alert {...} />
</div>
<div className="flex-1 overflow-hidden px-4 pb-4"> {/* <20><EFBFBD><E88880>典躹<E585B8>?*/}
<div className="flex-1 overflow-hidden px-4 pb-4"> {/* 可滚动区域 */}
<ChatContainer {...} />
</div>
</div>
```
#### 效果
<EFBFBD>?Chat瘨<74><E798A8><EFBFBD><EFBFBD><E7AE8F>臭誑甇<E8AA91>虜皛𡁜𢆡
<EFBFBD>?撌乩<E6928C><E79285><E288AA>㗇𥋘<E39787>典𤐄摰𡁜銁憿園<E686BF>
<EFBFBD>?皛𡁜𢆡<F0A1819C>⊥甅撘讐<E69298>閫?
✅ Chat消息区域可以正常滚动
✅ 工作模式选择器固定在顶部
✅ 滚动条样式美观
---
@@ -156,7 +156,7 @@ const WorkspacePage: React.FC<WorkspacePageProps> = ({ standalone = false }) =>
```typescript
// WorkspacePage.tsx
// 靽格㺿<EFBFBD>?
// 修改前
<div className="bg-white border-b border-gray-200 px-6 flex items-center shadow-sm z-20 h-14">
<button className="...">
<MessageSquare className="w-5 h-5 mr-2" />
@@ -164,26 +164,26 @@ const WorkspacePage: React.FC<WorkspacePageProps> = ({ standalone = false }) =>
</button>
</div>
// 靽格㺿<EFBFBD>?
// 修改后
<div className="bg-white border-b border-gray-200 px-6 flex items-center shadow-sm z-20 h-12 flex-shrink-0">
<button className="...">
<MessageSquare className="w-4 h-4 mr-2" /> {/* w-5 <EFBFBD>?w-4 */}
<span className="text-sm"><3E><EFBFBD><E7AE84><EFBFBD></span> {/* text-base <EFBFBD>?text-sm */}
<MessageSquare className="w-4 h-4 mr-2" /> {/* w-5 w-4 */}
<span className="text-sm"></span> {/* text-base text-sm */}
</button>
</div>
```
#### 对比
| 撅墧<E69285>?| 靽格㺿<E6A0BC>?| 靽格㺿<E6A0BC>?| <20><EFBFBD><E7AC94>?|
| 属性 | 修改前 | 修改后 | 原型图 |
|------|--------|--------|--------|
| 擃睃漲 | h-14 (56px) | h-12 (48px) | <EFBFBD>?h-12 |
| <EFBFBD><EFBFBD> | w-5 h-5 (20px) | w-4 h-4 (16px) | <EFBFBD>?w-4 h-4 |
| <EFBFBD><EFBFBD><EFBFBD> | text-base (16px) | text-sm (14px) | <EFBFBD>?text-sm |
| 高度 | h-14 (56px) | h-12 (48px) | h-12 |
| 图标 | w-5 h-5 (20px) | w-4 h-4 (16px) | w-4 h-4 |
| 文字 | text-base (16px) | text-sm (14px) | text-sm |
#### 效果
<EFBFBD>?Tab擃睃漲蝎曄<E69B84><EFBFBD><E5AFA5><EFBFBD>
<EFBFBD>?閫<><E996AB><EFBFBD><EFBFBD><EFBFBD><EFBFBD>渡移<E6B8A1>?
<EFBFBD>?銝竚3霈曇恣100%銝<><E98A9D>?
✅ Tab高度精确匹配原型
✅ 视觉效果更精致
✅ 与V3设计100%一致
---
@@ -192,15 +192,15 @@ const WorkspacePage: React.FC<WorkspacePageProps> = ({ standalone = false }) =>
#### 问题分析
后端返回的格式:
```
data: {"content":"<EFBFBD>?,"role":"assistant"}
data: {"content":"憟?,"role":"assistant"}
data: {"content":"嚗?,"role":"assistant"}
data: {"content":"您","role":"assistant"}
data: {"content":"好","role":"assistant"}
data: {"content":"","role":"assistant"}
data: [DONE]
```
<EFBFBD>滨垢<EFBFBD>躰秤隞<EFBFBD><EFBFBD>嚗?
前端错误代码:
```typescript
// <20>?<3F>躰秤嚗𡁶凒<F0A181B6>乩蝙<E4B9A9>禿esponse.json()
// ❌ 错误直接使用response.json()
const data = await response.json();
```
@@ -219,7 +219,7 @@ requestFn: async (message: string) => {
content: message,
modelType: 'qwen-long', // 使用qwen-long模型
knowledgeBaseIds: [kbId],
fullTextDocumentIds, // <EFBFBD>?documentIds
fullTextDocumentIds, // documentIds
}),
});
@@ -266,9 +266,9 @@ requestFn: async (message: string) => {
```
#### 效果
<EFBFBD>?甇<><EFBC86><E996AB>SSE瘚<45><E7989A><EFBFBD><EFBFBD>
<EFBFBD>?AI撖寡<E69296><E5AFA1><EFBFBD><EFBFBD>虜撌乩<E6928C>
<EFBFBD>?<3F><EFBFBD><EFBFBD><E7989A>颲枏枂嚗<E69E82><E59A97>𣂼<EFBFBD><F0A382BC>曄內嚗?
✅ 正确解析SSE流式响应
✅ AI对话功能正常工作
✅ 支持流式输出(逐字显示)
---
@@ -276,38 +276,38 @@ requestFn: async (message: string) => {
### 视觉效果对比
| 憿寧𤌍 | 靽桀<E99DBD><E6A180>?| 靽桀<E99DBD><E6A180>?| <20><EFBFBD><E7AC94><EFBFBD><E69B86><EFBFBD>?|
| 项目 | 修复前 | 修复后 | 原型图一致性 |
|------|--------|--------|--------------|
| <EFBFBD><EFBFBD><EFBFBD> | <20>?<3F>厰▲<E58EB0>典紡<E585B8>?| <20>?摰<><E691B0><EFBFBD><EFBFBD> | <EFBFBD>?100% |
| 皛𡁜𢆡<EFBFBD>?| <20>?<3F><EFBFBD>皛𡁜𢆡 | <20>?甇<>虜皛𡁜𢆡 | <EFBFBD>?100% |
| Tab擃睃漲 | <EFBFBD>?56px (蝎㛖<EFBFBD>) | <EFBFBD>?48px (蝎曇稲) | <EFBFBD>?100% |
| <EFBFBD><EFBFBD>撠箏站 | <EFBFBD>?20px | <EFBFBD>?16px | <EFBFBD>?100% |
| <EFBFBD><EFBFBD><EFBFBD>憭批<EFBFBD> | <EFBFBD>?16px | <EFBFBD>?14px | <EFBFBD>?100% |
| 全屏模式 | ❌ 有顶部导航 | ✅ 完全全屏 | 100% |
| 滚动条 | ❌ 无法滚动 | ✅ 正常滚动 | 100% |
| Tab高度 | 56px (粗糙) | 48px (精致) | 100% |
| 图标尺寸 | 20px | 16px | 100% |
| 文字大小 | 16px | 14px | 100% |
### 功能对比
| <EFBFBD><EFBFBD> | 靽桀<E99DBD><E6A180>?| 靽桀<E99DBD><E6A180>?|
| 功能 | 修复前 | 修复后 |
|------|--------|--------|
| AI撖寡<EFBFBD> | <20>?<3F>仿<EFBFBD><E4BBBF><EFBFBD>雿輻鍂 | <20>?甇<>虜撌乩<E6928C> |
| <EFBFBD><EFBFBD><EFBFBD><EFBFBD>粉 | <20>?<3F><EFBFBD>雿輻鍂 | <20>?甇<>虜撌乩<E6928C> |
| <EFBFBD><EFBFBD>蝎曇粉 | <20>?<3F><EFBFBD>雿輻鍂 | <20>?甇<>虜撌乩<E6928C> |
| <EFBFBD><EFBFBD>皛𡁜𢆡 | <20>?<3F><EFBFBD>皛𡁜𢆡 | <20>?甇<>虜皛𡁜𢆡 |
| AI对话 | ❌ 报错无法使用 | ✅ 正常工作 |
| 全文阅读 | ❌ 无法使用 | ✅ 正常工作 |
| 逐篇精读 | ❌ 无法使用 | ✅ 正常工作 |
| 消息滚动 | ❌ 无法滚动 | ✅ 正常滚动 |
---
## 🎯 技术要点总结
### 1. 全屏沉浸式布局
**<2A>喲睸<E596B2><E79DB8><EFBFBD>?*:
**关键技术**:
- `fixed inset-0 z-50`:固定定位,覆盖整个视口
- `standalone` prop嚗𡁏綉<EFBFBD>嗆糓<EFBFBD>虫蝙<EFBFBD><EFBFBD>撅𤩺芋撘?
- <20><EFBFBD>頝舐眏嚗帋<E59A97><E5B88B><EFBFBD><EFBFBD>ainLayout銝?
- `standalone` prop:控制是否使用全屏模式
- 独立路由不包裹在MainLayout
### 2. 滚动容器层级
**甇<><EFBFBD><EFBC86><EFBFBD>蝥抒<E89DA5><E68A92>?*:
**正确的层级结构**:
```
<main className="flex-1 overflow-hidden"> {/* 憭硋<EFBFBD>嚗𡁻<EFBFBD><EFBFBD>𤩺滯<EFBFBD>?*/}
<div className="h-full flex overflow-hidden"> {/* 銝剖<EFBFBD>嚗𡁜𤐄摰𡁻<EFBFBD>摨?*/}
<main className="flex-1 overflow-hidden"> {/* 外层:隐藏溢出 */}
<div className="h-full flex overflow-hidden"> {/* 中层:固定高度 */}
<div className="flex-1 overflow-y-auto"> {/* 内层:可滚动 */}
{/* 内容 */}
</div>
@@ -325,16 +325,16 @@ requestFn: async (message: string) => {
### 4. 精确还原设计
**设计规范**:
- 擃睃漲嚗帋艇<EFBFBD>潔蝙<EFBFBD>ailwind<EFBFBD><EFBFBD>-12<EFBFBD><EFBFBD>-14蝑?
- <EFBFBD><EFBFBD>嚗鯱-4 h-4嚗?6px嚗?
- <EFBFBD><EFBFBD><EFBFBD>嚗魩ext-sm嚗?4px嚗?
- <EFBFBD><EFBFBD>嚗䮝-3嚗?2px嚗?
- 高度严格使用Tailwind的h-12、h-14
- 图标w-4 h-416px
- 文字text-sm14px
- 间距p-312px
---
## 📁 修改文件清单
### 靽格㺿<EFBFBD><EFBFBD>辣嚗?銝迎<E98A9D>
### 修改文件5个
```
frontend-v2/src/modules/pkb/
├── index.tsx (添加standalone路由)
@@ -346,52 +346,52 @@ frontend-v2/src/modules/pkb/
---
## <EFBFBD>?撉諹<E69289><EFBFBD><E79A9C>
## ✅ 验证清单
### <EFBFBD>◆撉諹<EFBFBD>嚗㇊0嚗?
### 必须验证P0
- [x] Workspace页面全屏显示
- [x] 瘝⊥<EFBFBD>憿園<EFBFBD>撖潸⏛<EFBFBD>?
- [x] 没有顶部导航栏
- [x] Chat消息可以正常滚动
- [x] Tab擃睃漲銝?8px
- [x] Tab高度为48px
- [x] AI对话正常工作
- [x] 全文阅读模式可用
- [x] 逐篇精读模式可用
### 摨磰砲撉諹<EFBFBD>嚗㇊1嚗?
- [ ] 皛𡁜𢆡<EFBFBD>⊥甅撘讐<EFBFBD>閫?
### 应该验证P1
- [ ] 滚动条样式美观
- [ ] Tab切换流畅
- [ ] 工作模式切换正常
- [ ] PDF靘扯器<EFBFBD>𤩺迤撣?
- [ ] PDF侧边栏正常
### <EFBFBD>臭誑隡睃<EFBFBD>嚗㇊2嚗?
### 可以优化P2
- [ ] 流式输出动画效果
- [ ] 错误提示优化
- [ ] <EFBFBD>㰘蝸<EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>?
- [ ] 加载状态优化
---
## <EFBFBD><EFBFBD> 銝衤<E98A9D>甇?
## 🚀 下一步
1. **<EFBFBD><EFBFBD>撉諹<EFBFBD>**: 霂瑞鍂<EFBFBD><EFBFBD><EFBFBD><EFBFBD>頧賡△<EFBFBD><EFBFBD>霂?
1. **用户验证**: 请用户重新加载页面测试
2. **性能优化**: 优化流式响应的渲染性能
3. **错误处理**: 完善API错误提示
4. **<EFBFBD><EFBFBD><EFBFBD><EFBFBD>芋撘?*: 摰䂿緵<E482BF><EFBFBD><E5ADB5><EFBFBD><EFBFBD><EFBFBD><EFBFBD>摰峕㟲瘚<E39FB2><E7989A>
4. **批处理模式**: 实现批处理功能的完整流程
---
## 💡 经验教训
### 1. 霈曇恣餈睃<EFBFBD><EFBFBD>移蝖?
- 銝滩<EFBFBD>"撌桐<E6928C>憭?嚗諹<E59A97>"摰<><E691B0><EFBFBD><E98A9D>?
- 瘥譍葵<EFBFBD><EFBFBD><EFBFBD><EFBFBD>撖寞<EFBFBD><EFBFBD><EFBFBD><EFBFBD>?
### 1. 设计还原要精确
- 不能"差不多",要"完全一致"
- 每个像素都要对比原型图
- 使用Tailwind的精确尺寸类
### 2. <EFBFBD><EFBFBD>憿菟𢒰<EFBFBD><EFBFBD><EFBFBD>鸌畾𠰴<EFBFBD><EFBFBD>?
- 銝滩<EFBFBD><EFBFBD><EFBFBD>訫𧑐<EFBFBD>曉銁MainLayout銝?
- <EFBFBD><EFBFBD><EFBFBD>fixed`摰帋<EFBFBD><EFBFBD>𣇉𡠺蝡贝楝<EFBFBD>?
### 2. 全屏页面需要特殊处理
- 不能简单地放在MainLayout
- 需要`fixed`定位或独立路由
- 考虑z-index层级
### 3. 皛𡁜𢆡摰孵膥閬<EFBFBD><EFBFBD><EFBFBD>挽霈?
### 3. 滚动容器要仔细设计
- 外层`overflow-hidden`
- 内层`overflow-y-auto`
- 固定区域`flex-shrink-0`
@@ -404,9 +404,8 @@ frontend-v2/src/modules/pkb/
---
**修复完成时间**: 2026-01-06
**靽桀<EFBFBD>鈭?*: AI Assistant
**撉諹<EFBFBD><EFBFBD><EFBFBD>?*: 敺<><EFBFBD>霈?
**修复人**: AI Assistant
**验证状态**: 待用户确认