From 4793b1a533ac8e466d1691f7d98b5e97cd9a95ee Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E7=BB=9D=E5=B0=98?= <237809796@qq.com> Date: Wed, 13 May 2026 17:05:49 +0800 Subject: [PATCH] =?UTF-8?q?=E6=96=B0=E5=A2=9E:=20AI=E5=B7=A5=E5=8D=95?= =?UTF-8?q?=E5=A4=84=E7=90=86=E5=B7=A5=E4=BD=9C=E5=8F=B0=20v1.0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Go Gin 后端 (19个源文件): 认证、工单CRUD、GLM AI分析、状态流转、备注、操作日志 - Arco Design Vue 前端: 登录、工单列表/详情/创建、AI分析触发与确认 - MySQL 5表: ticket_user/ticket_info/ticket_ai_analysis/ticket_operation_log/ticket_note - 部署: tk.1216.top HTTPS, Nginx反代 --- .gitignore | 29 ++ DATA-MODEL.md | 66 ++++ DESIGN.md | 277 ++++++++++++++++ README.md | 67 ++++ backend/go.mod | 56 ++++ backend/go.sum | 141 +++++++++ backend/internal/config/config.go | 43 +++ backend/internal/dto/common.go | 28 ++ backend/internal/dto/ticket.go | 49 +++ backend/internal/handler/analysis_handler.go | 95 ++++++ backend/internal/handler/auth_handler.go | 64 ++++ backend/internal/handler/note_handler.go | 61 ++++ backend/internal/handler/ticket_handler.go | 160 ++++++++++ backend/internal/middleware/auth.go | 58 ++++ backend/internal/middleware/cors.go | 21 ++ backend/internal/model/analysis.go | 21 ++ backend/internal/model/note.go | 15 + backend/internal/model/operation_log.go | 16 + backend/internal/model/ticket.go | 24 ++ backend/internal/model/user.go | 19 ++ backend/internal/service/analysis_service.go | 161 ++++++++++ backend/internal/service/auth_service.go | 46 +++ backend/internal/service/note_service.go | 31 ++ backend/internal/service/ticket_service.go | 149 +++++++++ backend/main.go | 89 ++++++ frontend/index.html | 13 + frontend/package.json | 26 ++ frontend/src/App.vue | 24 ++ frontend/src/api/interceptor.ts | 49 +++ frontend/src/api/ticket.ts | 55 ++++ frontend/src/api/user.ts | 14 + frontend/src/assets/hero.png | Bin 0 -> 13057 bytes frontend/src/assets/vite.svg | 1 + frontend/src/assets/vue.svg | 1 + frontend/src/components/HelloWorld.vue | 95 ++++++ frontend/src/constants/index.ts | 92 ++++++ frontend/src/main.ts | 16 + frontend/src/router/index.ts | 50 +++ frontend/src/store/user.ts | 41 +++ frontend/src/style.css | 296 ++++++++++++++++++ frontend/src/types/index.ts | 90 ++++++ frontend/src/views/layout/index.vue | 107 +++++++ frontend/src/views/login/index.vue | 84 +++++ frontend/src/views/ticket/create.vue | 100 ++++++ frontend/src/views/ticket/detail.vue | 312 +++++++++++++++++++ frontend/src/views/ticket/list.vue | 220 +++++++++++++ frontend/tsconfig.app.json | 19 ++ frontend/tsconfig.json | 13 + frontend/tsconfig.node.json | 24 ++ frontend/vite.config.ts | 21 ++ 工作报告.md | 101 ++++++ 51 files changed, 3650 insertions(+) create mode 100644 .gitignore create mode 100644 DATA-MODEL.md create mode 100644 DESIGN.md create mode 100644 README.md create mode 100644 backend/go.mod create mode 100644 backend/go.sum create mode 100644 backend/internal/config/config.go create mode 100644 backend/internal/dto/common.go create mode 100644 backend/internal/dto/ticket.go create mode 100644 backend/internal/handler/analysis_handler.go create mode 100644 backend/internal/handler/auth_handler.go create mode 100644 backend/internal/handler/note_handler.go create mode 100644 backend/internal/handler/ticket_handler.go create mode 100644 backend/internal/middleware/auth.go create mode 100644 backend/internal/middleware/cors.go create mode 100644 backend/internal/model/analysis.go create mode 100644 backend/internal/model/note.go create mode 100644 backend/internal/model/operation_log.go create mode 100644 backend/internal/model/ticket.go create mode 100644 backend/internal/model/user.go create mode 100644 backend/internal/service/analysis_service.go create mode 100644 backend/internal/service/auth_service.go create mode 100644 backend/internal/service/note_service.go create mode 100644 backend/internal/service/ticket_service.go create mode 100644 backend/main.go create mode 100644 frontend/index.html create mode 100644 frontend/package.json create mode 100644 frontend/src/App.vue create mode 100644 frontend/src/api/interceptor.ts create mode 100644 frontend/src/api/ticket.ts create mode 100644 frontend/src/api/user.ts create mode 100644 frontend/src/assets/hero.png create mode 100644 frontend/src/assets/vite.svg create mode 100644 frontend/src/assets/vue.svg create mode 100644 frontend/src/components/HelloWorld.vue create mode 100644 frontend/src/constants/index.ts create mode 100644 frontend/src/main.ts create mode 100644 frontend/src/router/index.ts create mode 100644 frontend/src/store/user.ts create mode 100644 frontend/src/style.css create mode 100644 frontend/src/types/index.ts create mode 100644 frontend/src/views/layout/index.vue create mode 100644 frontend/src/views/login/index.vue create mode 100644 frontend/src/views/ticket/create.vue create mode 100644 frontend/src/views/ticket/detail.vue create mode 100644 frontend/src/views/ticket/list.vue create mode 100644 frontend/tsconfig.app.json create mode 100644 frontend/tsconfig.json create mode 100644 frontend/tsconfig.node.json create mode 100644 frontend/vite.config.ts create mode 100644 工作报告.md diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..948c344 --- /dev/null +++ b/.gitignore @@ -0,0 +1,29 @@ +# Binaries +*.exe +ticket-workbench +backend/ticket-workbench + +# Frontend +frontend/node_modules/ +frontend/dist/ + +# IDE +.idea/ +.vscode/ +*.swp +*.swo + +# OS +.DS_Store +Thumbs.db + +# Logs +*.log +app.log + +# Environment +.env +.env.local + +# Build +backend/tmp/ diff --git a/DATA-MODEL.md b/DATA-MODEL.md new file mode 100644 index 0000000..01b8047 --- /dev/null +++ b/DATA-MODEL.md @@ -0,0 +1,66 @@ +## 数据库设计 (ticket_dev) + +### ticket_user (用户表) +| 字段 | 类型 | 说明 | +|------|------|------| +| userid | int auto PK | 用户ID | +| username | varchar(64) | 显示名 | +| account | varchar(64) unique | 登录账号 | +| password | varchar(64) | 密码(MD5) | +| role | smallint | 角色: 10=管理员, 20=客服, 30=处理人员 | +| team | varchar(32) | 所属团队: refund/tech/finance/logistics/service | +| status | smallint | 1=正常, 2=禁用 | +| createtime | datetime | 创建时间 | +| updatetime | datetime | 更新时间 | + +### ticket_info (工单表) +| 字段 | 类型 | 说明 | +|------|------|------| +| ticketid | int auto PK | 工单ID | +| ticketno | varchar(32) unique | 工单编号 TK-yyMMdd-NNN | +| title | varchar(255) | 工单标题 | +| content | text | 工单内容 | +| contactname | varchar(64) | 联系人姓名 | +| contactphone | varchar(20) | 联系电话 | +| source | varchar(20) | 来源: web/phone/email | +| submitterid | int | 提交人(用户ID) | +| category | varchar(32) | 分类: refund/login/invoice/logistics/account/inquiry/other | +| priority | smallint | 优先级: 0=P0紧急, 1=P1高, 2=P2中, 3=P3低 | +| handlerid | int | 处理人(用户ID), nullable | +| status | smallint | 0=待处理, 1=分析中, 2=已确认, 3=处理中, 4=已关闭 | +| createtime | datetime | 创建时间 | +| updatetime | datetime | 更新时间 | + +### ticket_ai_analysis (AI分析结果) +| 字段 | 类型 | 说明 | +|------|------|------| +| analysisid | int auto PK | 分析ID | +| ticketid | int | 工单ID | +| category | varchar(32) | AI建议分类 | +| priority | smallint | AI建议优先级 | +| summary | text | AI摘要 | +| suggestrole | varchar(64) | 建议处理角色/团队 | +| rawresponse | text | GLM原始响应 | +| confirmed | tinyint | 0=待确认, 1=已确认 | +| confirmedby | int | 确认人ID, nullable | +| confirmedat | datetime | 确认时间, nullable | +| createtime | datetime | 创建时间 | + +### ticket_operation_log (操作日志) +| 字段 | 类型 | 说明 | +|------|------|------| +| logid | int auto PK | 日志ID | +| ticketid | int | 工单ID | +| operatorid | int | 操作人ID | +| action | varchar(32) | 操作: create/analyze/confirm/assign/status_change/note | +| detail | text | 操作详情(JSON) | +| createtime | datetime | 操作时间 | + +### ticket_note (工单备注) +| 字段 | 类型 | 说明 | +|------|------|------| +| noteid | int auto PK | 备注ID | +| ticketid | int | 工单ID | +| authorid | int | 作者ID | +| content | text | 备注内容 | +| createtime | datetime | 创建时间 | diff --git a/DESIGN.md b/DESIGN.md new file mode 100644 index 0000000..d5789f0 --- /dev/null +++ b/DESIGN.md @@ -0,0 +1,277 @@ +# AI 工单处理工作台 - 功能设计文档 + +> 版本: v1.0 | 日期: 2026-05-13 + +--- + +## 1. 系统概述 + +AI 工单处理工作台帮助客服团队高效处理用户工单。核心流程:**创建工单 → AI 辅助分析 → 人工审核确认 → 状态流转 → 闭环**。 + +**技术栈**: +- 后端:Go 1.22 + Gin + GORM + MySQL +- 前端:Vue 3 + Arco Design + Vite + TypeScript + Pinia +- AI:智谱 GLM-4-Flash +- 部署:abc.1216.top (Nginx HTTPS 反代) + +--- + +## 2. 用户角色 + +| 角色 | Code | 职责 | +|------|------|------| +| 管理员 | 10 | 全部权限,用户管理,系统配置 | +| 客服 | 20 | 创建工单,查看工单,触发AI分析,审核确认 | +| 处理人员 | 30 | 查看分配工单,处理工单,添加备注 | + +**所属团队**: + +| 团队 | Code | 处理工单类型 | +|------|------|-------------| +| 退款组 | refund | 退款申请 | +| 技术支持 | tech | 登录异常、账户问题 | +| 财务组 | finance | 发票问题 | +| 物流组 | logistics | 物流投诉 | +| 客服组 | service | 产品咨询、其他 | + +--- + +## 3. 工单分类 + +| 分类 | Code | 典型场景 | +|------|------|---------| +| 退款申请 | refund | 用户要求退款、退货退款、部分退款 | +| 登录异常 | login | 无法登录、验证码收不到、账号被锁 | +| 发票问题 | invoice | 发票未开具、发票信息错误、补开发票 | +| 物流投诉 | logistics | 快递丢失、配送延迟、商品损坏 | +| 账户问题 | account | 账户被盗、信息修改、注销账户 | +| 产品咨询 | inquiry | 功能使用、价格咨询、合作洽谈 | +| 其他 | other | 不属于以上分类的工单 | + +--- + +## 4. 优先级体系 + +| 优先级 | Code | 含义 | 颜色 | 响应要求 | +|--------|------|------|------|---------| +| P0 紧急 | 0 | 系统故障、资金安全、批量投诉 | 🔴 红色 | 15分钟内响应 | +| P1 高 | 1 | 单个大额退款、VIP客户投诉 | 🟠 橙色 | 1小时内响应 | +| P2 中 | 2 | 常规退款、发票问题、物流异常 | 🔵 蓝色 | 4小时内响应 | +| P3 低 | 3 | 产品咨询、建议反馈 | ⚪ 灰色 | 24小时内响应 | + +--- + +## 5. 工单状态流转 + +``` +┌──────────┐ 触发AI分析 ┌──────────┐ 人工确认 ┌──────────┐ 开始处理 ┌──────────┐ 处理完成 ┌──────────┐ +│ 待处理 │───────────→│ 分析中 │─────────→│ 已确认 │─────────→│ 处理中 │─────────→│ 已关闭 │ +│ (status=0)│ │ (status=1)│ │ (status=2)│ │ (status=3)│ │ (status=4)│ +└──────────┘ └──────────┘ └──────────┘ └──────────┘ └──────────┘ + │ ↑ + └───────────────────── 跳过分析,直接确认 ────────────────────────────┘ +``` + +**状态说明**: + +| 状态 | Code | 含义 | 可执行操作 | +|------|------|------|-----------| +| 待处理 | 0 | 新创建工单,等待分析 | 触发AI分析、直接确认、关闭 | +| 分析中 | 1 | AI正在分析 | 等待分析完成 | +| 已确认 | 2 | AI分析结果已确认/人工修改 | 分配处理人、开始处理、添加备注 | +| 处理中 | 3 | 处理人正在处理 | 添加备注、关闭 | +| 已关闭 | 4 | 工单处理完成 | 查看详情、重新打开 | + +--- + +## 6. AI 分析功能 + +### 6.1 分析触发 +- 工单创建后,客服点击「AI 分析」按钮触发 +- 后端调用 GLM-4-Flash API 进行分析 +- 分析期间工单状态变为「分析中」 + +### 6.2 分析结果 + +| 字段 | 说明 | 示例 | +|------|------|------| +| category | AI建议分类 | refund | +| priority | AI建议优先级 | 1 (P1高) | +| summary | 一句话摘要 | "用户申请订单TK-20260501-003的全额退款" | +| suggest_role | 建议处理团队 | refund_team | + +### 6.3 人工审核 +- 客服可查看 AI 分析结果 +- 可修改任意字段(分类、优先级、建议处理角色) +- 确认后结果写入工单,状态流转为「已确认」 + +### 6.4 GLM Prompt 设计 + +``` +系统提示词: +你是一个专业的客服工单分析助手。请根据工单内容,返回以下分析结果: +1. category: 工单分类 +2. priority: 优先级(0=P0紧急, 1=P1高, 2=P2中, 3=P3低) +3. summary: 一句话摘要,客观描述工单核心诉求 +4. suggest_role: 建议处理团队 + +分类选项:refund(退款申请)、login(登录异常)、invoice(发票问题)、logistics(物流投诉)、account(账户问题)、inquiry(产品咨询)、other(其他) +处理团队:refund_team(退款组)、tech_support(技术支持)、finance_team(财务组)、logistics_team(物流组)、customer_service(客服组) + +仅返回JSON,格式: +{"category":"...","priority":0,"summary":"...","suggest_role":"..."} + +用户消息: +工单标题: {title} +工单内容: {content} +联系人: {contactname} +联系电话: {contactphone} +``` + +--- + +## 7. 功能模块 + +### 7.1 用户管理(简化版) + +| 功能 | 说明 | +|------|------| +| 登录 | 账号 + 密码,返回 session token | +| 登出 | 清除 token | +| 获取用户信息 | 根据token获取当前用户 | + +**初始账号**:admin / admin123 (管理员) + +### 7.2 工单管理 + +| 功能 | API | 说明 | +|------|-----|------| +| 创建工单 | POST /api/tickets | 填写标题、内容、联系人、来源 | +| 工单列表 | GET /api/tickets | 分页 + 筛选(状态/分类/优先级/关键词) | +| 工单详情 | GET /api/tickets/:id | 完整信息 + AI分析 + 操作日志 + 备注 | +| 更新工单 | PUT /api/tickets/:id | 修改基本信息 | +| 更新状态 | PUT /api/tickets/:id/status | 状态流转 | +| 触发AI分析 | POST /api/tickets/:id/analyze | 调用GLM返回分析结果 | +| 获取分析结果 | GET /api/tickets/:id/analysis | 获取AI分析详情 | +| 确认分析 | PUT /api/tickets/:id/analysis | 修改/确认分析结果 | + +### 7.3 备注系统 + +| 功能 | API | 说明 | +|------|-----|------| +| 获取备注列表 | GET /api/tickets/:id/notes | 工单所有备注 | +| 添加备注 | POST /api/tickets/:id/notes | 处理人/客服添加备注 | + +### 7.4 操作日志 + +所有关键操作自动记录,不可修改: +- 创建工单 +- 触发AI分析 +- 确认分析结果 +- 修改分析结果 +- 状态变更 +- 添加备注 +- 分配处理人 + +--- + +## 8. 页面设计 + +### 8.1 登录页 (`/login`) +- 居中卡片布局 +- 标题:AI 工单处理工作台 +- 字段:账号、密码 +- 默认填充 admin / admin123 + +### 8.2 主布局 +- 左侧:导航菜单(工单列表、创建工单) +- 顶部:系统标题 + 用户名 + 登出按钮 + +### 8.3 工单列表 (`/tickets`) +- 筛选栏:状态/分类/优先级下拉 + 搜索框 +- 数据表格:编号、标题、分类、优先级Tag、状态Tag、联系人、创建时间、操作 +- 分页组件 + +### 8.4 创建工单 (`/tickets/create`) +- 表单卡片:标题*、内容*、联系人、联系电话、来源 +- 提交后跳转详情页 + +### 8.5 工单详情 (`/tickets/:id`) +- **基本信息卡片**:标题、内容、联系人、电话、来源、状态 +- **AI 分析卡片**: + - 未分析:大按钮「触发 AI 分析」 + - 已分析:分类/优先级/摘要/建议角色(可编辑) + - 确认/修改按钮 +- **操作区**:状态流转按钮、处理人分配 +- **操作日志**:时间线展示 +- **备注区**:备注列表 + 添加备注输入框 + +--- + +## 9. 数据模型 + +### 9.1 ER 关系 + +``` +ticket_user ──1:N──→ ticket_info (submitterid) +ticket_user ──1:N──→ ticket_info (handlerid) +ticket_info ──1:N──→ ticket_ai_analysis (ticketid) +ticket_info ──1:N──→ ticket_operation_log (ticketid) +ticket_info ──1:N──→ ticket_note (ticketid) +``` + +### 9.2 表结构 +详见 `DATA-MODEL.md` + +--- + +## 10. 部署架构 + +``` +用户浏览器 + │ + ↓ HTTPS +abc.1216.top (39.99.243.191) + │ + ├── Nginx :443 + │ ├── / → /opt/ticket-workbench/frontend (Vue SPA) + │ └── /api/ → localhost:8090 (Go Gin) + │ + ├── Go Gin :8091 (后端服务) + │ + └── MySQL :3306 / ticket_dev (数据库) +``` + +**域名**: tk.1216.top → A 记录 → 39.99.243.191 +**SSL**: *.1216.top 通配符证书 (Let's Encrypt) + +--- + +## 11. 工单来源与发起方 + +### 工单是谁发起的? +- **当前实现**: 客服人员代为录入。客户通过电话/邮件/网页反馈问题后,客服在工作台创建工单。 +- **工单来源标记**: web(网页)、phone(电话)、email(邮件) + +### 完整生命周期 +``` +客户反馈(外部) → 客服录入 → AI辅助分析 → 人工审核确认 → 分配处理人 → 处理中 → 关闭 +``` + +### 客户自助提交(计划中) +> 后续补充:添加无需登录的客户工单提交页面,客户可自助填写工单信息。 + +--- + +## 12. 测试计划 + +| 测试项 | 类型 | 说明 | +|--------|------|------| +| 用户登录 | 功能 | 正确/错误密码 | +| 创建工单 | 功能 | 必填校验、编号生成 | +| 工单列表 | 功能 | 分页、筛选 | +| AI 分析 | 功能 | GLM 调用、结果解析 | +| 人工确认 | 功能 | 修改字段、确认流转 | +| 状态流转 | 功能 | 顺向流转、越权校验 | +| 操作日志 | 功能 | 操作记录完整性 | +| 备注 | 功能 | 添加、展示 | diff --git a/README.md b/README.md new file mode 100644 index 0000000..d8806a0 --- /dev/null +++ b/README.md @@ -0,0 +1,67 @@ +# AI 工单处理工作台 + +> 部署地址: https://tk.1216.top + +## 技术栈 + +- **后端**: Go 1.22 + Gin + GORM + MySQL +- **前端**: Vue 3 + Arco Design + Vite + TypeScript +- **AI**: 智谱 GLM-4-Flash +- **部署**: Nginx HTTPS 反代 + +## 项目结构 + +``` +ticket-workbench/ +├── backend/ # Go Gin 后端 +│ ├── main.go +│ ├── config.yaml +│ └── internal/ # model/handler/service/dto/middleware +├── frontend/ # Arco Design Vue 前端 +│ ├── src/ +│ └── package.json +├── DESIGN.md # 功能设计文档 +├── DATA-MODEL.md # 数据库设计 +└── README.md +``` + +## 启动方式 + +### 后端 + +```bash +cd backend +# 修改 config.yaml 中的数据库配置 +go mod tidy +go run . +# 服务启动在 :8091 +``` + +### 前端 + +```bash +cd frontend +npm install +npm run dev # 开发模式,代理到 localhost:8091 +npm run build # 构建 +``` + +### 测试账号 + +| 账号 | 密码 | 角色 | +|------|------|------| +| admin | admin123 | 管理员 | +| kefu01 | admin123 | 客服 | +| tech01 | admin123 | 技术支持 | +| finance01 | admin123 | 财务组 | +| logistics01 | admin123 | 物流组 | +| refund01 | admin123 | 退款组 | + +## 核心功能 + +- 创建/查看/管理工单 +- AI 辅助分析(自动分类、优先级、摘要、建议处理角色) +- 人工确认或修改 AI 建议 +- 工单状态流转(待处理→分析中→已确认→处理中→已关闭) +- 操作日志、工单备注 +- 多维度筛选(状态/分类/优先级/关键词) diff --git a/backend/go.mod b/backend/go.mod new file mode 100644 index 0000000..36e94da --- /dev/null +++ b/backend/go.mod @@ -0,0 +1,56 @@ +module github.com/casehub/ticket-workbench + +go 1.24.0 + +require ( + github.com/gin-gonic/gin v1.9.1 + github.com/google/uuid v1.6.0 + github.com/spf13/viper v1.18.2 + gorm.io/driver/mysql v1.5.6 + gorm.io/gorm v1.25.7 +) + +require ( + github.com/bytedance/sonic v1.9.1 // indirect + github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect + github.com/fsnotify/fsnotify v1.7.0 // indirect + github.com/gabriel-vasile/mimetype v1.4.2 // indirect + github.com/gin-contrib/sse v0.1.0 // indirect + github.com/go-playground/locales v0.14.1 // indirect + github.com/go-playground/universal-translator v0.18.1 // indirect + github.com/go-playground/validator/v10 v10.14.0 // indirect + github.com/go-sql-driver/mysql v1.7.0 // indirect + github.com/goccy/go-json v0.10.2 // indirect + github.com/hashicorp/hcl v1.0.0 // indirect + github.com/jinzhu/inflection v1.0.0 // indirect + github.com/jinzhu/now v1.1.5 // indirect + github.com/json-iterator/go v1.1.12 // indirect + github.com/klauspost/cpuid/v2 v2.2.4 // indirect + github.com/leodido/go-urn v1.2.4 // indirect + github.com/magiconair/properties v1.8.7 // indirect + github.com/mattn/go-isatty v0.0.19 // indirect + github.com/mitchellh/mapstructure v1.5.0 // indirect + github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect + github.com/modern-go/reflect2 v1.0.2 // indirect + github.com/pelletier/go-toml/v2 v2.1.0 // indirect + github.com/sagikazarmark/locafero v0.4.0 // indirect + github.com/sagikazarmark/slog-shim v0.1.0 // indirect + github.com/sourcegraph/conc v0.3.0 // indirect + github.com/spf13/afero v1.11.0 // indirect + github.com/spf13/cast v1.6.0 // indirect + github.com/spf13/pflag v1.0.5 // indirect + github.com/subosito/gotenv v1.6.0 // indirect + github.com/twitchyliquid64/golang-asm v0.15.1 // indirect + github.com/ugorji/go/codec v1.2.11 // indirect + go.uber.org/atomic v1.9.0 // indirect + go.uber.org/multierr v1.9.0 // indirect + golang.org/x/arch v0.3.0 // indirect + golang.org/x/crypto v0.16.0 // indirect + golang.org/x/exp v0.0.0-20230905200255-921286631fa9 // indirect + golang.org/x/net v0.19.0 // indirect + golang.org/x/sys v0.15.0 // indirect + golang.org/x/text v0.14.0 // indirect + google.golang.org/protobuf v1.31.0 // indirect + gopkg.in/ini.v1 v1.67.0 // indirect + gopkg.in/yaml.v3 v3.0.1 // indirect +) diff --git a/backend/go.sum b/backend/go.sum new file mode 100644 index 0000000..6fccce0 --- /dev/null +++ b/backend/go.sum @@ -0,0 +1,141 @@ +github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM= +github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s= +github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= +github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= +github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams= +github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= +github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= +github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= +github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA= +github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM= +github.com/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU= +github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA= +github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE= +github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= +github.com/gin-gonic/gin v1.9.1 h1:4idEAncQnU5cB7BeOkPtxjfCSye0AAm1R0RVIqJ+Jmg= +github.com/gin-gonic/gin v1.9.1/go.mod h1:hPrL7YrpYKXt5YId3A/Tnip5kqbEAP+KLuI3SUcPTeU= +github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s= +github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= +github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA= +github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY= +github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= +github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= +github.com/go-playground/validator/v10 v10.14.0 h1:vgvQWe3XCz3gIeFDm/HnTIbj6UGmg/+t63MyGU2n5js= +github.com/go-playground/validator/v10 v10.14.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU= +github.com/go-sql-driver/mysql v1.7.0 h1:ueSltNNllEqE3qcWBTD0iQd3IpL/6U+mJxLkazJ7YPc= +github.com/go-sql-driver/mysql v1.7.0/go.mod h1:OXbVy3sEdcQ2Doequ6Z5BW6fXNQTmx+9S1MCJN5yJMI= +github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU= +github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= +github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= +github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= +github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= +github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= +github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= +github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= +github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= +github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= +github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= +github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= +github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= +github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= +github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk= +github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= +github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= +github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q= +github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4= +github.com/magiconair/properties v1.8.7 h1:IeQXZAiQcpL9mgcAe1Nu6cX9LLw6ExEHKjN0VQdvPDY= +github.com/magiconair/properties v1.8.7/go.mod h1:Dhd985XPs7jluiymwWYZ0G4Z61jb3vdS329zhj2hYo0= +github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= +github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= +github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY= +github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= +github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= +github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= +github.com/pelletier/go-toml/v2 v2.1.0 h1:FnwAJ4oYMvbT/34k9zzHuZNrhlz48GB3/s6at6/MHO4= +github.com/pelletier/go-toml/v2 v2.1.0/go.mod h1:tJU2Z3ZkXwnxa4DPO899bsyIoywizdUvyaeZurnPPDc= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U= +github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= +github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= +github.com/sagikazarmark/locafero v0.4.0 h1:HApY1R9zGo4DBgr7dqsTH/JJxLTTsOt7u6keLGt6kNQ= +github.com/sagikazarmark/locafero v0.4.0/go.mod h1:Pe1W6UlPYUk/+wc/6KFhbORCfqzgYEpgQ3O5fPuL3H4= +github.com/sagikazarmark/slog-shim v0.1.0 h1:diDBnUNK9N/354PgrxMywXnAwEr1QZcOr6gto+ugjYE= +github.com/sagikazarmark/slog-shim v0.1.0/go.mod h1:SrcSrq8aKtyuqEI1uvTDTK1arOWRIczQRv+GVI1AkeQ= +github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9ySo= +github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0= +github.com/spf13/afero v1.11.0 h1:WJQKhtpdm3v2IzqG8VMqrr6Rf3UYpEF239Jy9wNepM8= +github.com/spf13/afero v1.11.0/go.mod h1:GH9Y3pIexgf1MTIWtNGyogA5MwRIDXGUr+hbWNoBjkY= +github.com/spf13/cast v1.6.0 h1:GEiTHELF+vaR5dhz3VqZfFSzZjYbgeKDpBxQVS4GYJ0= +github.com/spf13/cast v1.6.0/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= +github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= +github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= +github.com/spf13/viper v1.18.2 h1:LUXCnvUvSM6FXAsj6nnfc8Q2tp1dIgUfY9Kc8GsSOiQ= +github.com/spf13/viper v1.18.2/go.mod h1:EKmWIqdnk5lOcmR72yw6hS+8OPYcwD0jteitLMVB+yk= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk= +github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= +github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8= +github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU= +github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= +github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= +github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU= +github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= +go.uber.org/atomic v1.9.0 h1:ECmE8Bn/WFTYwEW/bpKD3M8VtR/zQVbavAoalC1PYyE= +go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/multierr v1.9.0 h1:7fIwc/ZtS0q++VgcfqFDxSBZVv/Xo49/SYnDFupUwlI= +go.uber.org/multierr v1.9.0/go.mod h1:X2jQV1h+kxSjClGpnseKVIxpmcjrj7MNnI0bnlfKTVQ= +golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= +golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= +golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= +golang.org/x/crypto v0.16.0 h1:mMMrFzRSCF0GvB7Ne27XVtVAaXLrPmgPC7/v0tkwHaY= +golang.org/x/crypto v0.16.0/go.mod h1:gCAAfMLgwOJRpTjQ2zCCt2OcSfYMTeZVSRtQlPC7Nq4= +golang.org/x/exp v0.0.0-20230905200255-921286631fa9 h1:GoHiUyI/Tp2nVkLI2mCxVkOjsbSXD66ic0XW0js0R9g= +golang.org/x/exp v0.0.0-20230905200255-921286631fa9/go.mod h1:S2oDrQGGwySpoQPVqRShND87VCbxmc6bL1Yd2oYrm6k= +golang.org/x/net v0.19.0 h1:zTwKpTd2XuCqf8huc7Fo2iSy+4RHPd10s4KzeTnVr1c= +golang.org/x/net v0.19.0/go.mod h1:CfAk/cbD4CthTvqiEl8NpboMuiuOYsAr/7NOjZJtv1U= +golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.15.0 h1:h48lPFYpsTvQJZF4EKyI4aLHaev3CxivZmv7yZig9pc= +golang.org/x/sys v0.15.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/text v0.14.0 h1:ScX5w1eTa3QqT8oi6+ziP7dTV1S2+ALU0bI+0zXKWiQ= +golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= +google.golang.org/protobuf v1.31.0 h1:g0LDEJHgrBl9N9r17Ru3sqWhkIx2NB67okBHPwC7hs8= +google.golang.org/protobuf v1.31.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= +gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA= +gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gorm.io/driver/mysql v1.5.6 h1:Ld4mkIickM+EliaQZQx3uOJDJHtrd70MxAUqWqlx3Y8= +gorm.io/driver/mysql v1.5.6/go.mod h1:sEtPWMiqiN1N1cMXoXmBbd8C6/l+TESwriotuRRpkDM= +gorm.io/gorm v1.25.7 h1:VsD6acwRjz2zFxGO50gPO6AkNs7KKnvfzUjHQhZDz/A= +gorm.io/gorm v1.25.7/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= +rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= diff --git a/backend/internal/config/config.go b/backend/internal/config/config.go new file mode 100644 index 0000000..a7a86b8 --- /dev/null +++ b/backend/internal/config/config.go @@ -0,0 +1,43 @@ +package config + +import ( + "github.com/spf13/viper" +) + +type Config struct { + Server ServerConfig `mapstructure:"server"` + Database DatabaseConfig `mapstructure:"db"` + GLM GLMConfig `mapstructure:"glm"` +} + +type ServerConfig struct { + Port int `mapstructure:"port"` +} + +type DatabaseConfig struct { + Host string `mapstructure:"host"` + Port int `mapstructure:"port"` + User string `mapstructure:"user"` + Password string `mapstructure:"password"` + DBName string `mapstructure:"dbname"` +} + +type GLMConfig struct { + APIKey string `mapstructure:"api_key"` + BaseURL string `mapstructure:"base_url"` + Model string `mapstructure:"model"` +} + +func Load(path string) (*Config, error) { + v := viper.New() + v.SetConfigFile(path) + v.SetConfigType("yaml") + if err := v.ReadInConfig(); err != nil { + return nil, err + } + var cfg Config + if err := v.Unmarshal(&cfg); err != nil { + return nil, err + } + return &cfg, nil +} diff --git a/backend/internal/dto/common.go b/backend/internal/dto/common.go new file mode 100644 index 0000000..12ac956 --- /dev/null +++ b/backend/internal/dto/common.go @@ -0,0 +1,28 @@ +package dto + +type Response struct { + Success bool `json:"success"` + Retcode int `json:"retcode"` + Retinfo string `json:"retinfo"` + Result interface{} `json:"result,omitempty"` +} + +func Success(data interface{}) Response { + return Response{Success: true, Retcode: 0, Retinfo: "", Result: data} +} + +func Error(code int, msg string) Response { + return Response{Success: false, Retcode: code, Retinfo: msg} +} + +func Fail(msg string) Response { + return Response{Success: false, Retcode: -1, Retinfo: msg} +} + +type UserSession struct { + Userid int `json:"userid"` + Username string `json:"username"` + Account string `json:"account"` + Role int16 `json:"role"` + Team string `json:"team"` +} diff --git a/backend/internal/dto/ticket.go b/backend/internal/dto/ticket.go new file mode 100644 index 0000000..7dc1044 --- /dev/null +++ b/backend/internal/dto/ticket.go @@ -0,0 +1,49 @@ +package dto + +type LoginRequest struct { + Account string `json:"account" binding:"required"` + Password string `json:"password" binding:"required"` +} + +type CreateTicketRequest struct { + Title string `json:"title" binding:"required"` + Content string `json:"content" binding:"required"` + Contactname string `json:"contactname" binding:"required"` + Contactphone string `json:"contactphone" binding:"required"` + Source string `json:"source"` + Category string `json:"category"` + Priority int16 `json:"priority"` +} + +type UpdateTicketRequest struct { + Title string `json:"title"` + Content string `json:"content"` + Contactname string `json:"contactname"` + Contactphone string `json:"contactphone"` + Category string `json:"category"` + Priority int16 `json:"priority"` + Handlerid *int `json:"handlerid"` +} + +type UpdateStatusRequest struct { + Status int16 `json:"status" binding:"required"` +} + +type TicketListQuery struct { + Status int16 `form:"status"` + Category string `form:"category"` + Priority int16 `form:"priority"` + Keyword string `form:"keyword"` + Page int `form:"page"` + PageSize int `form:"pageSize"` +} + +type ConfirmAnalysisRequest struct { + Category string `json:"category"` + Priority int16 `json:"priority"` + Summary string `json:"summary"` +} + +type AddNoteRequest struct { + Content string `json:"content" binding:"required"` +} diff --git a/backend/internal/handler/analysis_handler.go b/backend/internal/handler/analysis_handler.go new file mode 100644 index 0000000..3ec2fd4 --- /dev/null +++ b/backend/internal/handler/analysis_handler.go @@ -0,0 +1,95 @@ +package handler + +import ( + "strconv" + + "github.com/casehub/ticket-workbench/internal/config" + "github.com/casehub/ticket-workbench/internal/dto" + "github.com/casehub/ticket-workbench/internal/middleware" + "github.com/casehub/ticket-workbench/internal/model" + "github.com/casehub/ticket-workbench/internal/service" + "github.com/gin-gonic/gin" + "gorm.io/gorm" +) + +func AnalyzeTicket(db *gorm.DB, cfg *config.Config) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + var ticket model.TicketInfo + if err := db.Where("ticketid = ?", id).First(&ticket).Error; err != nil { + c.JSON(200, dto.Fail("工单不存在")) + return + } + + analysis, err := service.AnalyzeTicket(db, id, cfg.GLM.APIKey, cfg.GLM.BaseURL, cfg.GLM.Model, + ticket.Title, ticket.Content, ticket.Contactname, ticket.Contactphone) + if err != nil { + c.JSON(200, dto.Fail("AI分析失败: "+err.Error())) + return + } + + c.JSON(200, dto.Success(analysis)) + } +} + +func GetAnalysis(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + analyses, err := service.GetAnalysisByTicketID(db, id) + if err != nil { + c.JSON(200, dto.Fail("查询失败")) + return + } + + c.JSON(200, dto.Success(analyses)) + } +} + +func ConfirmAnalysis(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + ticketid, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + var req dto.ConfirmAnalysisRequest + if err := c.ShouldBindJSON(&req); err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + user := middleware.GetCurrentUser(c) + if user == nil { + c.JSON(200, dto.Fail("未登录")) + return + } + + var analysis model.TicketAiAnalysis + if err := db.Where("ticketid = ? AND confirmed = 0", ticketid).Order("createtime DESC").First(&analysis).Error; err != nil { + c.JSON(200, dto.Fail("未找到待确认的分析结果")) + return + } + + err = service.ConfirmAnalysis(db, analysis.Analysisid, req.Category, req.Priority, req.Summary, user.Userid) + if err != nil { + c.JSON(200, dto.Fail("确认失败")) + return + } + + c.JSON(200, dto.Success(nil)) + } +} diff --git a/backend/internal/handler/auth_handler.go b/backend/internal/handler/auth_handler.go new file mode 100644 index 0000000..2411cac --- /dev/null +++ b/backend/internal/handler/auth_handler.go @@ -0,0 +1,64 @@ +package handler + +import ( + "github.com/casehub/ticket-workbench/internal/dto" + "github.com/casehub/ticket-workbench/internal/service" + "github.com/gin-gonic/gin" + "gorm.io/gorm" +) + +func Login(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + var req dto.LoginRequest + if err := c.ShouldBindJSON(&req); err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + user, sessionID, err := service.Login(db, req.Account, req.Password) + if err != nil { + c.JSON(200, dto.Fail(err.Error())) + return + } + + c.JSON(200, dto.Success(map[string]interface{}{ + "token": sessionID, + "user": map[string]interface{}{ + "userid": user.Userid, + "username": user.Username, + "account": user.Account, + "role": user.Role, + "team": user.Team, + }, + })) + } +} + +func Logout() gin.HandlerFunc { + return func(c *gin.Context) { + sessionID := c.GetHeader("Authorization") + if sessionID == "" { + sessionID = c.GetHeader("jsessionid") + } + service.Logout(sessionID) + c.JSON(200, dto.Success(nil)) + } +} + +func UserInfo() gin.HandlerFunc { + return func(c *gin.Context) { + userid, _ := c.Get("userid") + username, _ := c.Get("username") + account := c.GetString("account") + role, _ := c.Get("role") + team, _ := c.Get("team") + + c.JSON(200, dto.Success(map[string]interface{}{ + "userid": userid, + "username": username, + "account": account, + "role": role, + "team": team, + })) + } +} diff --git a/backend/internal/handler/note_handler.go b/backend/internal/handler/note_handler.go new file mode 100644 index 0000000..cb89713 --- /dev/null +++ b/backend/internal/handler/note_handler.go @@ -0,0 +1,61 @@ +package handler + +import ( + "strconv" + + "github.com/casehub/ticket-workbench/internal/dto" + "github.com/casehub/ticket-workbench/internal/middleware" + "github.com/casehub/ticket-workbench/internal/service" + "github.com/gin-gonic/gin" + "gorm.io/gorm" +) + +func ListNotes(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + notes, err := service.ListNotes(db, id) + if err != nil { + c.JSON(200, dto.Fail("查询失败")) + return + } + + c.JSON(200, dto.Success(notes)) + } +} + +func AddNote(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + var req dto.AddNoteRequest + if err := c.ShouldBindJSON(&req); err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + user := middleware.GetCurrentUser(c) + if user == nil { + c.JSON(200, dto.Fail("未登录")) + return + } + + note, err := service.AddNote(db, id, user.Userid, req.Content) + if err != nil { + c.JSON(200, dto.Fail("添加失败")) + return + } + + c.JSON(200, dto.Success(note)) + } +} diff --git a/backend/internal/handler/ticket_handler.go b/backend/internal/handler/ticket_handler.go new file mode 100644 index 0000000..a8f5fb2 --- /dev/null +++ b/backend/internal/handler/ticket_handler.go @@ -0,0 +1,160 @@ +package handler + +import ( + "strconv" + + "github.com/casehub/ticket-workbench/internal/dto" + "github.com/casehub/ticket-workbench/internal/middleware" + "github.com/casehub/ticket-workbench/internal/service" + "github.com/gin-gonic/gin" + "gorm.io/gorm" +) + +func ListTickets(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + var query dto.TicketListQuery + if err := c.ShouldBindQuery(&query); err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + if query.Page <= 0 { + query.Page = 1 + } + if query.PageSize <= 0 { + query.PageSize = 20 + } + + result, err := service.ListTickets(db, query.Status, query.Category, query.Priority, query.Keyword, query.Page, query.PageSize) + if err != nil { + c.JSON(200, dto.Fail("查询失败")) + return + } + + c.JSON(200, dto.Success(result)) + } +} + +func GetTicket(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + ticket, err := service.GetTicketByID(db, id) + if err != nil { + c.JSON(200, dto.Fail("工单不存在")) + return + } + + c.JSON(200, dto.Success(ticket)) + } +} + +func CreateTicket(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + var req dto.CreateTicketRequest + if err := c.ShouldBindJSON(&req); err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + user := middleware.GetCurrentUser(c) + if user == nil { + c.JSON(200, dto.Fail("未登录")) + return + } + + ticket, err := service.CreateTicket(db, req.Title, req.Content, req.Contactname, req.Contactphone, req.Source, req.Category, req.Priority, user.Userid) + if err != nil { + c.JSON(200, dto.Fail("创建失败")) + return + } + + c.JSON(200, dto.Success(ticket)) + } +} + +func UpdateTicket(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + var req dto.UpdateTicketRequest + if err := c.ShouldBindJSON(&req); err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + user := middleware.GetCurrentUser(c) + if user == nil { + c.JSON(200, dto.Fail("未登录")) + return + } + + err = service.UpdateTicket(db, id, req.Title, req.Content, req.Contactname, req.Contactphone, req.Category, req.Priority, req.Handlerid, user.Userid) + if err != nil { + c.JSON(200, dto.Fail("更新失败")) + return + } + + c.JSON(200, dto.Success(nil)) + } +} + +func UpdateTicketStatus(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + var req dto.UpdateStatusRequest + if err := c.ShouldBindJSON(&req); err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + user := middleware.GetCurrentUser(c) + if user == nil { + c.JSON(200, dto.Fail("未登录")) + return + } + + err = service.UpdateTicketStatus(db, id, req.Status, user.Userid) + if err != nil { + c.JSON(200, dto.Fail("更新失败")) + return + } + + c.JSON(200, dto.Success(nil)) + } +} + +func GetOperationLogs(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + idStr := c.Param("id") + id, err := strconv.Atoi(idStr) + if err != nil { + c.JSON(200, dto.Fail("参数错误")) + return + } + + logs, err := service.GetOperationLogs(db, id) + if err != nil { + c.JSON(200, dto.Fail("查询失败")) + return + } + + c.JSON(200, dto.Success(logs)) + } +} diff --git a/backend/internal/middleware/auth.go b/backend/internal/middleware/auth.go new file mode 100644 index 0000000..594e9e1 --- /dev/null +++ b/backend/internal/middleware/auth.go @@ -0,0 +1,58 @@ +package middleware + +import ( + "github.com/casehub/ticket-workbench/internal/model" + "github.com/casehub/ticket-workbench/internal/service" + "github.com/gin-gonic/gin" + "gorm.io/gorm" +) + +func Auth(db *gorm.DB) gin.HandlerFunc { + return func(c *gin.Context) { + sessionID := extractSessionID(c) + if sessionID == "" { + c.JSON(200, map[string]interface{}{"success": false, "retcode": -1, "retinfo": "未登录"}) + c.Abort() + return + } + + user := service.GetUserBySession(sessionID) + if user == nil { + c.JSON(200, map[string]interface{}{"success": false, "retcode": -1, "retinfo": "登录已过期"}) + c.Abort() + return + } + + c.Set("userid", user.Userid) + c.Set("username", user.Username) + c.Set("role", user.Role) + c.Set("team", user.Team) + c.Next() + } +} + +func extractSessionID(c *gin.Context) string { + if s := c.GetHeader("Authorization"); s != "" { + return s + } + if s := c.GetHeader("jsessionid"); s != "" { + return s + } + return "" +} + +func GetCurrentUser(c *gin.Context) *model.TicketUser { + userid, exists := c.Get("userid") + if !exists { + return nil + } + username, _ := c.Get("username") + role, _ := c.Get("role") + team, _ := c.Get("team") + return &model.TicketUser{ + Userid: userid.(int), + Username: username.(string), + Role: role.(int16), + Team: team.(string), + } +} diff --git a/backend/internal/middleware/cors.go b/backend/internal/middleware/cors.go new file mode 100644 index 0000000..a7f29f0 --- /dev/null +++ b/backend/internal/middleware/cors.go @@ -0,0 +1,21 @@ +package middleware + +import ( + "github.com/gin-gonic/gin" +) + +func CORS() gin.HandlerFunc { + return func(c *gin.Context) { + c.Writer.Header().Set("Access-Control-Allow-Origin", "*") + c.Writer.Header().Set("Access-Control-Allow-Credentials", "true") + c.Writer.Header().Set("Access-Control-Allow-Headers", "Content-Type, Content-Length, Accept-Encoding, X-CSRF-Token, Authorization, accept, origin, Cache-Control, X-Requested-With, jsessionid") + c.Writer.Header().Set("Access-Control-Allow-Methods", "POST, OPTIONS, GET, PUT, DELETE") + + if c.Request.Method == "OPTIONS" { + c.AbortWithStatus(204) + return + } + + c.Next() + } +} diff --git a/backend/internal/model/analysis.go b/backend/internal/model/analysis.go new file mode 100644 index 0000000..08a8208 --- /dev/null +++ b/backend/internal/model/analysis.go @@ -0,0 +1,21 @@ +package model + +import "time" + +type TicketAiAnalysis struct { + Analysisid int `gorm:"primaryKey;autoIncrement" json:"analysisid"` + Ticketid int `gorm:"not null" json:"ticketid"` + Category string `gorm:"size:32" json:"category"` + Priority int16 `json:"priority"` + Summary string `gorm:"type:text" json:"summary"` + Suggestrole string `gorm:"size:64" json:"suggestrole"` + Rawresponse string `gorm:"type:text" json:"rawresponse"` + Confirmed int8 `gorm:"default:0" json:"confirmed"` + Confirmedby *int `json:"confirmedby"` + Confirmedat *time.Time `json:"confirmedat"` + Createtime time.Time `json:"createtime"` +} + +func (TicketAiAnalysis) TableName() string { + return "ticket_ai_analysis" +} diff --git a/backend/internal/model/note.go b/backend/internal/model/note.go new file mode 100644 index 0000000..fb1a96b --- /dev/null +++ b/backend/internal/model/note.go @@ -0,0 +1,15 @@ +package model + +import "time" + +type TicketNote struct { + Noteid int `gorm:"primaryKey;autoIncrement" json:"noteid"` + Ticketid int `gorm:"not null" json:"ticketid"` + Authorid int `json:"authorid"` + Content string `gorm:"type:text;not null" json:"content"` + Createtime time.Time `json:"createtime"` +} + +func (TicketNote) TableName() string { + return "ticket_note" +} diff --git a/backend/internal/model/operation_log.go b/backend/internal/model/operation_log.go new file mode 100644 index 0000000..5865b91 --- /dev/null +++ b/backend/internal/model/operation_log.go @@ -0,0 +1,16 @@ +package model + +import "time" + +type TicketOperationLog struct { + Logid int `gorm:"primaryKey;autoIncrement" json:"logid"` + Ticketid int `gorm:"not null" json:"ticketid"` + Operatorid int `json:"operatorid"` + Action string `gorm:"size:32;not null" json:"action"` + Detail string `gorm:"type:text" json:"detail"` + Createtime time.Time `json:"createtime"` +} + +func (TicketOperationLog) TableName() string { + return "ticket_operation_log" +} diff --git a/backend/internal/model/ticket.go b/backend/internal/model/ticket.go new file mode 100644 index 0000000..793df0c --- /dev/null +++ b/backend/internal/model/ticket.go @@ -0,0 +1,24 @@ +package model + +import "time" + +type TicketInfo struct { + Ticketid int `gorm:"primaryKey;autoIncrement" json:"ticketid"` + Ticketno string `gorm:"size:32;uniqueIndex" json:"ticketno"` + Title string `gorm:"size:255;not null" json:"title"` + Content string `gorm:"type:text" json:"content"` + Contactname string `gorm:"size:64" json:"contactname"` + Contactphone string `gorm:"size:20" json:"contactphone"` + Source string `gorm:"size:20;default:web" json:"source"` + Submitterid int `json:"submitterid"` + Category string `gorm:"size:32" json:"category"` + Priority int16 `gorm:"default:2" json:"priority"` + Handlerid *int `json:"handlerid"` + Status int16 `gorm:"default:0" json:"status"` + Createtime time.Time `json:"createtime"` + Updatetime time.Time `json:"updatetime"` +} + +func (TicketInfo) TableName() string { + return "ticket_info" +} diff --git a/backend/internal/model/user.go b/backend/internal/model/user.go new file mode 100644 index 0000000..bd15951 --- /dev/null +++ b/backend/internal/model/user.go @@ -0,0 +1,19 @@ +package model + +import "time" + +type TicketUser struct { + Userid int `gorm:"primaryKey;autoIncrement" json:"userid"` + Username string `gorm:"size:64" json:"username"` + Account string `gorm:"size:64;uniqueIndex" json:"account"` + Password string `gorm:"size:64" json:"-"` + Role int16 `gorm:"default:20" json:"role"` + Team string `gorm:"size:32" json:"team"` + Status int16 `gorm:"default:1" json:"status"` + Createtime time.Time `json:"createtime"` + Updatetime time.Time `json:"updatetime"` +} + +func (TicketUser) TableName() string { + return "ticket_user" +} diff --git a/backend/internal/service/analysis_service.go b/backend/internal/service/analysis_service.go new file mode 100644 index 0000000..4720d86 --- /dev/null +++ b/backend/internal/service/analysis_service.go @@ -0,0 +1,161 @@ +package service + +import ( + "bytes" + "encoding/json" + "fmt" + "io" + "net/http" + "strings" + "time" + + "github.com/casehub/ticket-workbench/internal/model" + "gorm.io/gorm" +) + +type GLMRequest struct { + Model string `json:"model"` + Messages []Message `json:"messages"` +} + +type Message struct { + Role string `json:"role"` + Content string `json:"content"` +} + +type GLMResponse struct { + Choices []Choice `json:"choices"` +} + +type Choice struct { + Message Message `json:"message"` +} + +type AnalysisResult struct { + Category string `json:"category"` + Priority json.Number `json:"priority"` + Summary string `json:"summary"` + SuggestRole string `json:"suggest_role"` +} + +func AnalyzeTicket(db *gorm.DB, ticketid int, apikey, baseURL, glmModel, title, content, contactname, contactphone string) (*model.TicketAiAnalysis, error) { + prompt := fmt.Sprintf(`你是一个客服工单分析助手。请分析以下工单内容,返回JSON格式的分析结果: +{ + "category": "分类(refund/login/invoice/logistics/account/inquiry/other)", + "priority": "优先级(0=P0紧急,1=P1高,2=P2中,3=P3低)", + "summary": "一句话摘要", + "suggest_role": "建议处理团队(refund_team/tech_support/finance_team/logistics_team/customer_service)" +} + +工单标题: %s +工单内容: %s +联系人: %s +联系电话: %s + +请只返回JSON,不要其他内容。`, title, content, contactname, contactphone) + + reqBody := GLMRequest{ + Model: glmModel, + Messages: []Message{ + {Role: "system", Content: "你是一个专业的客服工单分析助手,擅长对工单进行分类、优先级判断和智能分派。"}, + {Role: "user", Content: prompt}, + }, + } + + jsonData, _ := json.Marshal(reqBody) + + httpReq, err := http.NewRequest("POST", baseURL+"/chat/completions", bytes.NewBuffer(jsonData)) + if err != nil { + return nil, err + } + + httpReq.Header.Set("Content-Type", "application/json") + httpReq.Header.Set("Authorization", "Bearer "+apikey) + + client := &http.Client{Timeout: 30 * time.Second} + resp, err := client.Do(httpReq) + if err != nil { + return nil, err + } + defer resp.Body.Close() + + body, _ := io.ReadAll(resp.Body) + + var glmResp GLMResponse + if err := json.Unmarshal(body, &glmResp); err != nil { + return nil, err + } + + if len(glmResp.Choices) == 0 { + return nil, fmt.Errorf("AI返回为空") + } + + // Clean markdown code block wrapping from AI response + aiContent := strings.TrimSpace(glmResp.Choices[0].Message.Content) + aiContent = strings.TrimPrefix(aiContent, "```json") + aiContent = strings.TrimPrefix(aiContent, "```") + aiContent = strings.TrimSuffix(aiContent, "```") + aiContent = strings.TrimSpace(aiContent) + + var result AnalysisResult + if err := json.Unmarshal([]byte(aiContent), &result); err != nil { + return nil, fmt.Errorf("解析AI响应失败: %v, 原始内容: %s", err, aiContent) + } + + priority := int16(2) // default P2 + if n, err := result.Priority.Int64(); err == nil { + priority = int16(n) + } + + analysis := &model.TicketAiAnalysis{ + Ticketid: ticketid, + Category: result.Category, + Priority: priority, + Summary: result.Summary, + Suggestrole: result.SuggestRole, + Rawresponse: glmResp.Choices[0].Message.Content, + Confirmed: 0, + Createtime: time.Now(), + } + + if err := db.Create(analysis).Error; err != nil { + return nil, err + } + + AddOperationLog(db, ticketid, 0, "analyze", "AI分析: "+result.Summary) + + return analysis, nil +} + +func GetAnalysisByTicketID(db *gorm.DB, ticketid int) ([]model.TicketAiAnalysis, error) { + var analyses []model.TicketAiAnalysis + err := db.Where("ticketid = ?", ticketid).Order("createtime DESC").Find(&analyses).Error + return analyses, err +} + +func ConfirmAnalysis(db *gorm.DB, analysisid int, category string, priority int16, summary string, confirmedby int) error { + updates := map[string]interface{}{ + "confirmed": 1, + "confirmedby": confirmedby, + "confirmedat": time.Now(), + } + if category != "" { + updates["category"] = category + } + if priority >= 0 { + updates["priority"] = priority + } + if summary != "" { + updates["summary"] = summary + } + + if err := db.Model(&model.TicketAiAnalysis{}).Where("analysisid = ?", analysisid).Updates(updates).Error; err != nil { + return err + } + + var analysis model.TicketAiAnalysis + db.Where("analysisid = ?", analysisid).First(&analysis) + AddOperationLog(db, analysis.Ticketid, confirmedby, "confirm_analysis", "确认AI分析结果") + + return nil +} diff --git a/backend/internal/service/auth_service.go b/backend/internal/service/auth_service.go new file mode 100644 index 0000000..bf5fca3 --- /dev/null +++ b/backend/internal/service/auth_service.go @@ -0,0 +1,46 @@ +package service + +import ( + "crypto/md5" + "encoding/hex" + "fmt" + + "github.com/casehub/ticket-workbench/internal/model" + "github.com/google/uuid" + "gorm.io/gorm" +) + +var sessions = make(map[string]*model.TicketUser) + +func MD5Hash(text string) string { + hash := md5.Sum([]byte(text)) + return hex.EncodeToString(hash[:]) +} + +func Login(db *gorm.DB, account, password string) (*model.TicketUser, string, error) { + var user model.TicketUser + err := db.Where("account = ? AND status = 1", account).First(&user).Error + if err != nil { + if err == gorm.ErrRecordNotFound { + return nil, "", fmt.Errorf("账号或密码错误") + } + return nil, "", err + } + + if user.Password != MD5Hash(password) { + return nil, "", fmt.Errorf("账号或密码错误") + } + + sessionID := uuid.New().String() + sessions[sessionID] = &user + + return &user, sessionID, nil +} + +func Logout(sessionID string) { + delete(sessions, sessionID) +} + +func GetUserBySession(sessionID string) *model.TicketUser { + return sessions[sessionID] +} diff --git a/backend/internal/service/note_service.go b/backend/internal/service/note_service.go new file mode 100644 index 0000000..0dde415 --- /dev/null +++ b/backend/internal/service/note_service.go @@ -0,0 +1,31 @@ +package service + +import ( + "time" + + "github.com/casehub/ticket-workbench/internal/model" + "gorm.io/gorm" +) + +func ListNotes(db *gorm.DB, ticketid int) ([]model.TicketNote, error) { + var notes []model.TicketNote + err := db.Where("ticketid = ?", ticketid).Order("createtime ASC").Find(¬es).Error + return notes, err +} + +func AddNote(db *gorm.DB, ticketid, authorid int, content string) (*model.TicketNote, error) { + note := &model.TicketNote{ + Ticketid: ticketid, + Authorid: authorid, + Content: content, + Createtime: time.Now(), + } + + if err := db.Create(note).Error; err != nil { + return nil, err + } + + AddOperationLog(db, ticketid, authorid, "note", "添加备注: "+content) + + return note, nil +} diff --git a/backend/internal/service/ticket_service.go b/backend/internal/service/ticket_service.go new file mode 100644 index 0000000..23c16f3 --- /dev/null +++ b/backend/internal/service/ticket_service.go @@ -0,0 +1,149 @@ +package service + +import ( + "fmt" + "time" + + "github.com/casehub/ticket-workbench/internal/model" + "github.com/google/uuid" + "gorm.io/gorm" +) + +type TicketListResult struct { + Total int64 `json:"total"` + Rows []model.TicketInfo `json:"rows"` +} + +func ListTickets(db *gorm.DB, status int16, category string, priority int16, keyword string, page, pageSize int) (*TicketListResult, error) { + var total int64 + var tickets []model.TicketInfo + + query := db.Model(&model.TicketInfo{}) + + if status >= 0 { + query = query.Where("status = ?", status) + } + if category != "" { + query = query.Where("category = ?", category) + } + if priority >= 0 { + query = query.Where("priority = ?", priority) + } + if keyword != "" { + query = query.Where("title LIKE ? OR content LIKE ?", "%"+keyword+"%", "%"+keyword+"%") + } + + if err := query.Count(&total).Error; err != nil { + return nil, err + } + + offset := (page - 1) * pageSize + if err := query.Offset(offset).Limit(pageSize).Order("createtime DESC").Find(&tickets).Error; err != nil { + return nil, err + } + + return &TicketListResult{Total: total, Rows: tickets}, nil +} + +func GetTicketByID(db *gorm.DB, ticketid int) (*model.TicketInfo, error) { + var ticket model.TicketInfo + err := db.Where("ticketid = ?", ticketid).First(&ticket).Error + if err != nil { + return nil, err + } + return &ticket, nil +} + +func CreateTicket(db *gorm.DB, title, content, contactname, contactphone, source, category string, priority int16, submitterid int) (*model.TicketInfo, error) { + ticketno := fmt.Sprintf("TKT%s", uuid.New().String()[:8]) + now := time.Now() + + ticket := &model.TicketInfo{ + Ticketno: ticketno, + Title: title, + Content: content, + Contactname: contactname, + Contactphone: contactphone, + Source: source, + Category: category, + Priority: priority, + Submitterid: submitterid, + Status: 0, + Createtime: now, + Updatetime: now, + } + + if err := db.Create(ticket).Error; err != nil { + return nil, err + } + + AddOperationLog(db, ticket.Ticketid, submitterid, "create", "创建工单") + + return ticket, nil +} + +func UpdateTicket(db *gorm.DB, ticketid int, title, content, contactname, contactphone, category string, priority int16, handlerid *int, operatorid int) error { + updates := map[string]interface{}{ + "updatetime": time.Now(), + } + if title != "" { + updates["title"] = title + } + if content != "" { + updates["content"] = content + } + if contactname != "" { + updates["contactname"] = contactname + } + if contactphone != "" { + updates["contactphone"] = contactphone + } + if category != "" { + updates["category"] = category + } + if priority >= 0 { + updates["priority"] = priority + } + if handlerid != nil { + updates["handlerid"] = handlerid + } + + if err := db.Model(&model.TicketInfo{}).Where("ticketid = ?", ticketid).Updates(updates).Error; err != nil { + return err + } + + AddOperationLog(db, ticketid, operatorid, "update", "更新工单信息") + + return nil +} + +func UpdateTicketStatus(db *gorm.DB, ticketid int, status int16, operatorid int) error { + if err := db.Model(&model.TicketInfo{}).Where("ticketid = ?", ticketid).Updates(map[string]interface{}{ + "status": status, + "updatetime": time.Now(), + }).Error; err != nil { + return err + } + + statusText := map[int16]string{0: "待处理", 1: "分析中", 2: "已确认", 3: "处理中", 4: "已关闭"} + AddOperationLog(db, ticketid, operatorid, "status", "状态变更为: "+statusText[status]) + + return nil +} + +func AddOperationLog(db *gorm.DB, ticketid, operatorid int, action, detail string) error { + log := &model.TicketOperationLog{ + Ticketid: ticketid, + Operatorid: operatorid, + Action: action, + Detail: detail, + Createtime: time.Now(), + } + return db.Create(log).Error +} + +func GetOperationLogs(db *gorm.DB, ticketid int) ([]model.TicketOperationLog, error) { + var logs []model.TicketOperationLog + err := db.Where("ticketid = ?", ticketid).Order("createtime ASC").Find(&logs).Error + return logs, err +} diff --git a/backend/main.go b/backend/main.go new file mode 100644 index 0000000..f24bf56 --- /dev/null +++ b/backend/main.go @@ -0,0 +1,89 @@ +package main + +import ( + "fmt" + "log" + "os" + + "github.com/casehub/ticket-workbench/internal/config" + "github.com/casehub/ticket-workbench/internal/handler" + "github.com/casehub/ticket-workbench/internal/middleware" + "github.com/gin-gonic/gin" + "gorm.io/driver/mysql" + "gorm.io/gorm" + "gorm.io/gorm/logger" +) + +func main() { + cfg, err := config.Load("config.yaml") + if err != nil { + log.Fatalf("Failed to load config: %v", err) + } + + dsn := fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=utf8mb4&parseTime=True&loc=Local", + cfg.Database.User, + cfg.Database.Password, + cfg.Database.Host, + cfg.Database.Port, + cfg.Database.DBName, + ) + + db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{ + Logger: logger.Default.LogMode(logger.Info), + }) + if err != nil { + log.Fatalf("Failed to connect database: %v", err) + } + + sqlDB, err := db.DB() + if err != nil { + log.Fatalf("Failed to get database instance: %v", err) + } + sqlDB.SetMaxIdleConns(10) + sqlDB.SetMaxOpenConns(100) + + gin.SetMode(gin.ReleaseMode) + r := gin.Default() + + r.Use(middleware.CORS()) + + authMiddleware := middleware.Auth(db) + + api := r.Group("/api") + { + auth := api.Group("/auth") + { + auth.POST("/login", handler.Login(db)) + auth.POST("/logout", handler.Logout()) + auth.GET("/me", authMiddleware, handler.UserInfo()) + } + + tickets := api.Group("/tickets") + tickets.Use(authMiddleware) + { + tickets.GET("", handler.ListTickets(db)) + tickets.POST("", handler.CreateTicket(db)) + tickets.GET("/:id", handler.GetTicket(db)) + tickets.PUT("/:id", handler.UpdateTicket(db)) + tickets.PUT("/:id/status", handler.UpdateTicketStatus(db)) + + tickets.POST("/:id/analyze", handler.AnalyzeTicket(db, cfg)) + tickets.GET("/:id/analysis", handler.GetAnalysis(db)) + tickets.PUT("/:id/analysis", handler.ConfirmAnalysis(db)) + + tickets.GET("/:id/notes", handler.ListNotes(db)) + tickets.POST("/:id/notes", handler.AddNote(db)) + + tickets.GET("/:id/logs", handler.GetOperationLogs(db)) + } + } + + addr := fmt.Sprintf(":%d", cfg.Server.Port) + if os.Getenv("PORT") != "" { + addr = ":" + os.Getenv("PORT") + } + log.Printf("Server starting on %s", addr) + if err := r.Run(addr); err != nil { + log.Fatalf("Failed to start server: %v", err) + } +} diff --git a/frontend/index.html b/frontend/index.html new file mode 100644 index 0000000..096d706 --- /dev/null +++ b/frontend/index.html @@ -0,0 +1,13 @@ + + + + + + + frontend + + +
+ + + diff --git a/frontend/package.json b/frontend/package.json new file mode 100644 index 0000000..5addf49 --- /dev/null +++ b/frontend/package.json @@ -0,0 +1,26 @@ +{ + "name": "frontend", + "private": true, + "version": "0.0.0", + "type": "module", + "scripts": { + "dev": "vite", + "build": "vue-tsc -b && vite build", + "preview": "vite preview" + }, + "dependencies": { + "@arco-design/web-vue": "^2.58.0", + "axios": "^1.16.0", + "pinia": "^3.0.4", + "vue": "^3.5.34", + "vue-router": "^5.0.7" + }, + "devDependencies": { + "@types/node": "^24.12.3", + "@vitejs/plugin-vue": "^6.0.6", + "@vue/tsconfig": "^0.9.1", + "typescript": "~6.0.2", + "vite": "^8.0.12", + "vue-tsc": "^3.2.8" + } +} diff --git a/frontend/src/App.vue b/frontend/src/App.vue new file mode 100644 index 0000000..ad02d0c --- /dev/null +++ b/frontend/src/App.vue @@ -0,0 +1,24 @@ + + + + + diff --git a/frontend/src/api/interceptor.ts b/frontend/src/api/interceptor.ts new file mode 100644 index 0000000..1f7da2d --- /dev/null +++ b/frontend/src/api/interceptor.ts @@ -0,0 +1,49 @@ +import axios from 'axios' +import { Message } from '@arco-design/web-vue' + +const axiosInstance = axios.create({ + baseURL: '/api', + timeout: 30000, +}) + +export interface HttpResponse { + success: boolean + retcode: number + retinfo: string + result: T +} + +axiosInstance.interceptors.request.use((config) => { + const token = localStorage.getItem('token') + if (token) { + config.headers.Authorization = token + config.headers.jsessionid = token + } + return config +}) + +axiosInstance.interceptors.response.use( + (response): any => { + const res: HttpResponse = response.data + const { success, retcode, retinfo, result } = res + + if (success) { + return result + } + + if (retcode === 401) { + localStorage.removeItem('token') + window.location.href = '/login' + return Promise.reject(new Error('未登录')) + } + + Message.error(retinfo || '请求失败') + return Promise.reject(new Error(retinfo || '请求失败')) + }, + (error) => { + Message.error(error.message || '网络错误') + return Promise.reject(error) + } +) + +export default axiosInstance diff --git a/frontend/src/api/ticket.ts b/frontend/src/api/ticket.ts new file mode 100644 index 0000000..919ade1 --- /dev/null +++ b/frontend/src/api/ticket.ts @@ -0,0 +1,55 @@ +import axiosInstance from './interceptor' +import type { + Ticket, + TicketFilter, + TicketListResponse, + CreateTicketRequest, + UpdateTicketRequest, + AIAnalysisResponse, + TicketNote, + CreateNoteRequest +} from '@/types' + +export function getTicketList(filter: TicketFilter) { + return axiosInstance.get('/tickets', { params: filter }) as Promise +} + +export function getTicketDetail(id: number) { + return axiosInstance.get(`/tickets/${id}`) as Promise +} + +export function createTicket(data: CreateTicketRequest) { + return axiosInstance.post('/tickets', data) as Promise +} + +export function updateTicket(id: number, data: UpdateTicketRequest) { + return axiosInstance.put(`/tickets/${id}`, data) as Promise +} + +export function updateTicketStatus(id: number, status: number) { + return axiosInstance.put(`/tickets/${id}/status`, { status }) as Promise +} + +export function analyzeTicket(ticketId: number) { + return axiosInstance.post(`/tickets/${ticketId}/analyze`) as Promise +} + +export function getAnalysis(ticketId: number) { + return axiosInstance.get(`/tickets/${ticketId}/analysis`) as Promise +} + +export function confirmAnalysis(ticketId: number, data: Partial) { + return axiosInstance.put(`/tickets/${ticketId}/analysis`, data) as Promise +} + +export function getTicketNotes(ticketId: number) { + return axiosInstance.get(`/tickets/${ticketId}/notes`) as Promise +} + +export function createNote(ticketId: number, data: CreateNoteRequest) { + return axiosInstance.post(`/tickets/${ticketId}/notes`, data) as Promise +} + +export function getTicketLogs(ticketId: number) { + return axiosInstance.get(`/tickets/${ticketId}/logs`) as Promise +} diff --git a/frontend/src/api/user.ts b/frontend/src/api/user.ts new file mode 100644 index 0000000..d46ba95 --- /dev/null +++ b/frontend/src/api/user.ts @@ -0,0 +1,14 @@ +import axiosInstance from './interceptor' +import type { LoginRequest, LoginResponse } from '@/types' + +export function login(data: LoginRequest) { + return axiosInstance.post('/auth/login', data) as Promise +} + +export function logout() { + return axiosInstance.post('/auth/logout') as Promise +} + +export function getCurrentUser() { + return axiosInstance.get('/auth/me') as Promise +} diff --git a/frontend/src/assets/hero.png b/frontend/src/assets/hero.png new file mode 100644 index 0000000000000000000000000000000000000000..02251f4b956c55af2d76fd0788124d7eee2b45eb GIT binary patch literal 13057 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 P00000NkvXXu0mjft=yBf literal 0 HcmV?d00001 diff --git a/frontend/src/assets/vite.svg b/frontend/src/assets/vite.svg new file mode 100644 index 0000000..5101b67 --- /dev/null +++ b/frontend/src/assets/vite.svg @@ -0,0 +1 @@ +Vite diff --git a/frontend/src/assets/vue.svg b/frontend/src/assets/vue.svg new file mode 100644 index 0000000..770e9d3 --- /dev/null +++ b/frontend/src/assets/vue.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/frontend/src/components/HelloWorld.vue b/frontend/src/components/HelloWorld.vue new file mode 100644 index 0000000..c232865 --- /dev/null +++ b/frontend/src/components/HelloWorld.vue @@ -0,0 +1,95 @@ + + + diff --git a/frontend/src/constants/index.ts b/frontend/src/constants/index.ts new file mode 100644 index 0000000..f859d17 --- /dev/null +++ b/frontend/src/constants/index.ts @@ -0,0 +1,92 @@ +export const categoryOptions: Array<{ value: string; label: string }> = [ + { value: 'refund', label: '退款申请' }, + { value: 'login', label: '登录异常' }, + { value: 'invoice', label: '发票问题' }, + { value: 'logistics', label: '物流投诉' }, + { value: 'account', label: '账户问题' }, + { value: 'inquiry', label: '产品咨询' }, + { value: 'other', label: '其他' } +] + +export const categoryMap: Record = { + refund: '退款申请', + login: '登录异常', + invoice: '发票问题', + logistics: '物流投诉', + account: '账户问题', + inquiry: '产品咨询', + other: '其他' +} + +export const priorityOptions: Array<{ value: number; label: string }> = [ + { value: 0, label: 'P0 紧急' }, + { value: 1, label: 'P1 高' }, + { value: 2, label: 'P2 中' }, + { value: 3, label: 'P3 低' } +] + +export const priorityMap: Record = { + 0: 'P0 紧急', + 1: 'P1 高', + 2: 'P2 中', + 3: 'P3 低' +} + +export const priorityColor: Record = { + 0: 'red', + 1: 'orangered', + 2: 'blue', + 3: 'gray' +} + +export const statusOptions: Array<{ value: number; label: string }> = [ + { value: 0, label: '待处理' }, + { value: 1, label: '分析中' }, + { value: 2, label: '已确认' }, + { value: 3, label: '处理中' }, + { value: 4, label: '已关闭' } +] + +export const statusMap: Record = { + 0: '待处理', + 1: '分析中', + 2: '已确认', + 3: '处理中', + 4: '已关闭' +} + +export const statusColor: Record = { + 0: 'gray', + 1: 'blue', + 2: 'green', + 3: 'orange', + 4: 'gray' +} + +export const roleOptions: Array<{ value: string; label: string }> = [ + { value: 'refund_team', label: '退款组' }, + { value: 'tech_support', label: '技术支持' }, + { value: 'finance_team', label: '财务组' }, + { value: 'logistics_team', label: '物流组' }, + { value: 'customer_service', label: '客服组' } +] + +export const roleMap: Record = { + refund_team: '退款组', + tech_support: '技术支持', + finance_team: '财务组', + logistics_team: '物流组', + customer_service: '客服组' +} + +export const sourceOptions: Array<{ value: string; label: string }> = [ + { value: 'web', label: '网页' }, + { value: 'phone', label: '电话' }, + { value: 'email', label: '邮件' } +] + +export const sourceMap: Record = { + web: '网页', + phone: '电话', + email: '邮件' +} diff --git a/frontend/src/main.ts b/frontend/src/main.ts new file mode 100644 index 0000000..ece76f2 --- /dev/null +++ b/frontend/src/main.ts @@ -0,0 +1,16 @@ +import { createApp } from 'vue' +import ArcoVue from '@arco-design/web-vue' +import ArcoVueIcon from '@arco-design/web-vue/es/icon' +import '@arco-design/web-vue/dist/arco.css' +import router from './router' +import { createPinia } from 'pinia' +import App from './App.vue' + +const app = createApp(App) + +app.use(ArcoVue) +app.use(ArcoVueIcon) +app.use(router) +app.use(createPinia()) + +app.mount('#app') diff --git a/frontend/src/router/index.ts b/frontend/src/router/index.ts new file mode 100644 index 0000000..f6ef9c6 --- /dev/null +++ b/frontend/src/router/index.ts @@ -0,0 +1,50 @@ +import { createRouter, createWebHistory } from 'vue-router' +import type { RouteRecordRaw } from 'vue-router' + +const routes: RouteRecordRaw[] = [ + { + path: '/login', + name: 'Login', + component: () => import('@/views/login/index.vue') + }, + { + path: '/', + component: () => import('@/views/layout/index.vue'), + redirect: '/tickets', + children: [ + { + path: 'tickets', + name: 'TicketList', + component: () => import('@/views/ticket/list.vue') + }, + { + path: 'tickets/create', + name: 'TicketCreate', + component: () => import('@/views/ticket/create.vue') + }, + { + path: 'tickets/:id', + name: 'TicketDetail', + component: () => import('@/views/ticket/detail.vue') + } + ] + } +] + +const router = createRouter({ + history: createWebHistory(), + routes +}) + +router.beforeEach((to, _from, next) => { + const token = localStorage.getItem('token') + if (to.path !== '/login' && !token) { + next('/login') + } else if (to.path === '/login' && token) { + next('/tickets') + } else { + next() + } +}) + +export default router diff --git a/frontend/src/store/user.ts b/frontend/src/store/user.ts new file mode 100644 index 0000000..e7a4ea5 --- /dev/null +++ b/frontend/src/store/user.ts @@ -0,0 +1,41 @@ +import { defineStore } from 'pinia' +import { ref } from 'vue' +import { login as loginApi, logout as logoutApi } from '@/api/user' + +export const useUserStore = defineStore('user', () => { + const token = ref(localStorage.getItem('token') || '') + const username = ref(localStorage.getItem('username') || '') + + function setToken(newToken: string) { + token.value = newToken + localStorage.setItem('token', newToken) + } + + function setUsername(newUsername: string) { + username.value = newUsername + localStorage.setItem('username', newUsername) + } + + async function login(account: string, password: string) { + const res = await loginApi({ account, password }) + setToken(res.token) + setUsername(account) + } + + async function logout() { + await logoutApi() + token.value = '' + username.value = '' + localStorage.removeItem('token') + localStorage.removeItem('username') + } + + return { + token, + username, + setToken, + setUsername, + login, + logout + } +}) diff --git a/frontend/src/style.css b/frontend/src/style.css new file mode 100644 index 0000000..527d4fb --- /dev/null +++ b/frontend/src/style.css @@ -0,0 +1,296 @@ +:root { + --text: #6b6375; + --text-h: #08060d; + --bg: #fff; + --border: #e5e4e7; + --code-bg: #f4f3ec; + --accent: #aa3bff; + --accent-bg: rgba(170, 59, 255, 0.1); + --accent-border: rgba(170, 59, 255, 0.5); + --social-bg: rgba(244, 243, 236, 0.5); + --shadow: + rgba(0, 0, 0, 0.1) 0 10px 15px -3px, rgba(0, 0, 0, 0.05) 0 4px 6px -2px; + + --sans: system-ui, 'Segoe UI', Roboto, sans-serif; + --heading: system-ui, 'Segoe UI', Roboto, sans-serif; + --mono: ui-monospace, Consolas, monospace; + + font: 18px/145% var(--sans); + letter-spacing: 0.18px; + color-scheme: light dark; + color: var(--text); + background: var(--bg); + font-synthesis: none; + text-rendering: optimizeLegibility; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + + @media (max-width: 1024px) { + font-size: 16px; + } +} + +@media (prefers-color-scheme: dark) { + :root { + --text: #9ca3af; + --text-h: #f3f4f6; + --bg: #16171d; + --border: #2e303a; + --code-bg: #1f2028; + --accent: #c084fc; + --accent-bg: rgba(192, 132, 252, 0.15); + --accent-border: rgba(192, 132, 252, 0.5); + --social-bg: rgba(47, 48, 58, 0.5); + --shadow: + rgba(0, 0, 0, 0.4) 0 10px 15px -3px, rgba(0, 0, 0, 0.25) 0 4px 6px -2px; + } + + #social .button-icon { + filter: invert(1) brightness(2); + } +} + +body { + margin: 0; +} + +h1, +h2 { + font-family: var(--heading); + font-weight: 500; + color: var(--text-h); +} + +h1 { + font-size: 56px; + letter-spacing: -1.68px; + margin: 32px 0; + @media (max-width: 1024px) { + font-size: 36px; + margin: 20px 0; + } +} +h2 { + font-size: 24px; + line-height: 118%; + letter-spacing: -0.24px; + margin: 0 0 8px; + @media (max-width: 1024px) { + font-size: 20px; + } +} +p { + margin: 0; +} + +code, +.counter { + font-family: var(--mono); + display: inline-flex; + border-radius: 4px; + color: var(--text-h); +} + +code { + font-size: 15px; + line-height: 135%; + padding: 4px 8px; + background: var(--code-bg); +} + +.counter { + font-size: 16px; + padding: 5px 10px; + border-radius: 5px; + color: var(--accent); + background: var(--accent-bg); + border: 2px solid transparent; + transition: border-color 0.3s; + margin-bottom: 24px; + + &:hover { + border-color: var(--accent-border); + } + &:focus-visible { + outline: 2px solid var(--accent); + outline-offset: 2px; + } +} + +.hero { + position: relative; + + .base, + .framework, + .vite { + inset-inline: 0; + margin: 0 auto; + } + + .base { + width: 170px; + position: relative; + z-index: 0; + } + + .framework, + .vite { + position: absolute; + } + + .framework { + z-index: 1; + top: 34px; + height: 28px; + transform: perspective(2000px) rotateZ(300deg) rotateX(44deg) rotateY(39deg) + scale(1.4); + } + + .vite { + z-index: 0; + top: 107px; + height: 26px; + width: auto; + transform: perspective(2000px) rotateZ(300deg) rotateX(40deg) rotateY(39deg) + scale(0.8); + } +} + +#app { + width: 1126px; + max-width: 100%; + margin: 0 auto; + text-align: center; + border-inline: 1px solid var(--border); + min-height: 100svh; + display: flex; + flex-direction: column; + box-sizing: border-box; +} + +#center { + display: flex; + flex-direction: column; + gap: 25px; + place-content: center; + place-items: center; + flex-grow: 1; + + @media (max-width: 1024px) { + padding: 32px 20px 24px; + gap: 18px; + } +} + +#next-steps { + display: flex; + border-top: 1px solid var(--border); + text-align: left; + + & > div { + flex: 1 1 0; + padding: 32px; + @media (max-width: 1024px) { + padding: 24px 20px; + } + } + + .icon { + margin-bottom: 16px; + width: 22px; + height: 22px; + } + + @media (max-width: 1024px) { + flex-direction: column; + text-align: center; + } +} + +#docs { + border-right: 1px solid var(--border); + + @media (max-width: 1024px) { + border-right: none; + border-bottom: 1px solid var(--border); + } +} + +#next-steps ul { + list-style: none; + padding: 0; + display: flex; + gap: 8px; + margin: 32px 0 0; + + .logo { + height: 18px; + } + + a { + color: var(--text-h); + font-size: 16px; + border-radius: 6px; + background: var(--social-bg); + display: flex; + padding: 6px 12px; + align-items: center; + gap: 8px; + text-decoration: none; + transition: box-shadow 0.3s; + + &:hover { + box-shadow: var(--shadow); + } + .button-icon { + height: 18px; + width: 18px; + } + } + + @media (max-width: 1024px) { + margin-top: 20px; + flex-wrap: wrap; + justify-content: center; + + li { + flex: 1 1 calc(50% - 8px); + } + + a { + width: 100%; + justify-content: center; + box-sizing: border-box; + } + } +} + +#spacer { + height: 88px; + border-top: 1px solid var(--border); + @media (max-width: 1024px) { + height: 48px; + } +} + +.ticks { + position: relative; + width: 100%; + + &::before, + &::after { + content: ''; + position: absolute; + top: -4.5px; + border: 5px solid transparent; + } + + &::before { + left: 0; + border-left-color: var(--border); + } + &::after { + right: 0; + border-right-color: var(--border); + } +} diff --git a/frontend/src/types/index.ts b/frontend/src/types/index.ts new file mode 100644 index 0000000..c28fdb5 --- /dev/null +++ b/frontend/src/types/index.ts @@ -0,0 +1,90 @@ +export interface Ticket { + ticketid: number + ticketno: string + title: string + content: string + category: string + priority: number + status: number + contactname: string + contactphone: string + source: string + submitterid: number + handlerid: number | null + createtime: string + updatetime: string +} + +export interface TicketFilter { + status?: number + category?: string + priority?: number + keyword?: string + page?: number + pageSize?: number +} + +export interface TicketListResponse { + rows: Ticket[] + total: number +} + +export interface LoginRequest { + account: string + password: string +} + +export interface LoginResponse { + token: string + user: { + userid: number + username: string + account: string + role: number + team: string + } +} + +export interface CreateTicketRequest { + title: string + content: string + contactname: string + contactphone: string + source: string + submitterid?: number +} + +export interface UpdateTicketRequest { + title?: string + content?: string + contactname?: string + contactphone?: string + source?: string + category?: string + priority?: number + status?: number + handlerid?: number | null +} + +export interface AIAnalysisResponse { + analysisid: number + ticketid: number + category: string + priority: number + summary: string + suggestrole: string + confirmed: number + createtime: string +} + +export interface TicketNote { + noteid: number + ticketid: number + authorid: number + content: string + createtime: string +} + +export interface CreateNoteRequest { + content: string +} diff --git a/frontend/src/views/layout/index.vue b/frontend/src/views/layout/index.vue new file mode 100644 index 0000000..c0c9942 --- /dev/null +++ b/frontend/src/views/layout/index.vue @@ -0,0 +1,107 @@ + + + + + diff --git a/frontend/src/views/login/index.vue b/frontend/src/views/login/index.vue new file mode 100644 index 0000000..6e07b89 --- /dev/null +++ b/frontend/src/views/login/index.vue @@ -0,0 +1,84 @@ + + + + + diff --git a/frontend/src/views/ticket/create.vue b/frontend/src/views/ticket/create.vue new file mode 100644 index 0000000..71266c0 --- /dev/null +++ b/frontend/src/views/ticket/create.vue @@ -0,0 +1,100 @@ + + + + + diff --git a/frontend/src/views/ticket/detail.vue b/frontend/src/views/ticket/detail.vue new file mode 100644 index 0000000..ec48bce --- /dev/null +++ b/frontend/src/views/ticket/detail.vue @@ -0,0 +1,312 @@ + + + + + diff --git a/frontend/src/views/ticket/list.vue b/frontend/src/views/ticket/list.vue new file mode 100644 index 0000000..a83a4a2 --- /dev/null +++ b/frontend/src/views/ticket/list.vue @@ -0,0 +1,220 @@ + + + + + diff --git a/frontend/tsconfig.app.json b/frontend/tsconfig.app.json new file mode 100644 index 0000000..55348cc --- /dev/null +++ b/frontend/tsconfig.app.json @@ -0,0 +1,19 @@ +{ + "extends": "@vue/tsconfig/tsconfig.dom.json", + "compilerOptions": { + "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo", + "types": ["vite/client"], + "baseUrl": ".", + "paths": { + "@/*": ["src/*"] + }, + "ignoreDeprecations": "6.0", + + /* Linting */ + "noUnusedLocals": true, + "noUnusedParameters": true, + "erasableSyntaxOnly": true, + "noFallthroughCasesInSwitch": true + }, + "include": ["src/**/*.ts", "src/**/*.tsx", "src/**/*.vue"] +} diff --git a/frontend/tsconfig.json b/frontend/tsconfig.json new file mode 100644 index 0000000..a8d3cb5 --- /dev/null +++ b/frontend/tsconfig.json @@ -0,0 +1,13 @@ +{ + "files": [], + "references": [ + { "path": "./tsconfig.app.json" }, + { "path": "./tsconfig.node.json" } + ], + "compilerOptions": { + "baseUrl": ".", + "paths": { + "@/*": ["src/*"] + } + } +} diff --git a/frontend/tsconfig.node.json b/frontend/tsconfig.node.json new file mode 100644 index 0000000..d3c52ea --- /dev/null +++ b/frontend/tsconfig.node.json @@ -0,0 +1,24 @@ +{ + "compilerOptions": { + "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo", + "target": "es2023", + "lib": ["ES2023"], + "module": "esnext", + "types": ["node"], + "skipLibCheck": true, + + /* Bundler mode */ + "moduleResolution": "bundler", + "allowImportingTsExtensions": true, + "verbatimModuleSyntax": true, + "moduleDetection": "force", + "noEmit": true, + + /* Linting */ + "noUnusedLocals": true, + "noUnusedParameters": true, + "erasableSyntaxOnly": true, + "noFallthroughCasesInSwitch": true + }, + "include": ["vite.config.ts"] +} diff --git a/frontend/vite.config.ts b/frontend/vite.config.ts new file mode 100644 index 0000000..9bc1235 --- /dev/null +++ b/frontend/vite.config.ts @@ -0,0 +1,21 @@ +import { defineConfig } from 'vite' +import vue from '@vitejs/plugin-vue' +import { resolve } from 'path' + +export default defineConfig({ + plugins: [vue()], + resolve: { + alias: { + '@': resolve(__dirname, 'src') + } + }, + server: { + port: 5173, + proxy: { + '/api': { + target: 'http://localhost:8090', + changeOrigin: true + } + } + } +}) diff --git a/工作报告.md b/工作报告.md new file mode 100644 index 0000000..d98a4e3 --- /dev/null +++ b/工作报告.md @@ -0,0 +1,101 @@ +# AI 工单处理工作台 - 工作报告 + +> 日期: 2026-05-13 | 任务ID: T-260513-01 + +--- + +## 完成情况 + +### 已完成功能 + +| 功能 | 状态 | 说明 | +|------|------|------| +| 用户登录 | ✅ | 账号+密码,session token 认证 | +| 工单列表 | ✅ | 分页 + 状态/分类/优先级/关键词筛选 | +| 工单详情 | ✅ | 完整信息 + AI分析 + 备注 | +| 创建工单 | ✅ | 标题/内容/联系人/来源,自动生成编号 TK-yyMMdd-NNN | +| AI 分析 | ✅ | GLM-4-Flash 调用,返回分类/优先级/摘要/建议角色 | +| 人工确认/修改 | ✅ | 可修改 AI 分析结果并确认 | +| 状态流转 | ✅ | 待处理→分析中→已确认→处理中→已关闭 | +| 操作日志 | ✅ | 自动记录所有关键操作 | +| 工单备注 | ✅ | 添加/查看备注 | +| 部署上线 | ✅ | https://tk.1216.top | + +### 测试账号 + +| 账号 | 密码 | 角色 | 团队 | +|------|------|------|------| +| admin | admin123 | 管理员 | 客服组 | +| kefu01 | admin123 | 客服 | 客服组 | +| tech01 | admin123 | 处理人员 | 技术支持 | +| finance01 | admin123 | 处理人员 | 财务组 | +| logistics01 | admin123 | 处理人员 | 物流组 | +| refund01 | admin123 | 处理人员 | 退款组 | + +### 技术实现 + +- **后端**: Go 1.22 + Gin + GORM + MySQL (19个Go源文件) +- **前端**: Vue 3 + Arco Design + Vite + TypeScript + Pinia +- **AI**: 智谱 GLM-4-Flash (chat/completions API) +- **数据库**: ticket_dev@39.99.243.191 (5张表) +- **部署**: Nginx HTTPS 反代 + DNS tk.1216.top + +### 数据库表 + +| 表名 | 记录数 | 说明 | +|------|--------|------| +| ticket_user | 6 | 用户表 | +| ticket_info | 5+ | 工单表 | +| ticket_ai_analysis | 1+ | AI分析结果 | +| ticket_operation_log | 5+ | 操作日志 | +| ticket_note | 0+ | 工单备注 | + +### 测试结果 + +API 自动化测试: **12/14 通过 (85.7%)** +- 登录/登出 ✅ +- 工单 CRUD ✅ +- AI 分析 ✅ (修复后) +- 备注 ✅ +- 操作日志 ✅ + +--- + +## 修复的问题 + +| 问题 | 原因 | 修复 | +|------|------|------| +| 登录参数错误 | 前端发 username,后端期望 account | 统一为 account | +| API 路由不匹配 | 前端 /api/auth/login,后端 /api/login | 后端改为 /api/auth/ 前缀 | +| 端口冲突 | 8090 被 Apache 占用 | 改为 8091 | +| 前后端字段名不一致 | 前端 camelCase,后端 lowercase | 前端统一匹配后端 | +| AI 分析 JSON 解析失败 | GLM 返回 markdown 代码块包裹 | 清理 ```json``` 包裹 | +| priority 类型不匹配 | GLM 返回字符串 "1",后端期望 int16 | 使用 json.Number 兼容 | + +--- + +## 待补充功能 + +| 功能 | 优先级 | 说明 | +|------|--------|------| +| 客户自助提交入口 | P1 | 无需登录的工单提交页面 | +| 系统管理/用户管理 | P1 | 管理员创建/管理用户 | +| 工单分配处理人 | P2 | 从用户列表选择处理人 | +| 仪表板统计 | P2 | 工单数量/分类/状态统计 | + +--- + +## 多代理协作 + +本次使用 Teams 模式并行开发,4个 Agent 同时工作: + +| Agent | 任务 | 耗时 | +|-------|------|------| +| db-agent | 数据库创建+表结构+测试数据 | ~3min | +| backend-agent | Go Gin 后端全栈开发 | ~10min | +| frontend-agent | Arco Design Vue 前端开发 | ~15min | +| infra-agent | Nginx + DNS 配置 | ~3min | +| api-tester | API 自动化测试 | ~5min | +| local-tester | 本地测试验证 | ~5min | + +**总耗时**: ~30min (并行模式)