Change aftersales serial to YYMMNN monthly sequence

User-requested format: zjbf-sh-260501 (26=year, 05=month, 01=monthly seq).
Sequence resets each month and skips soft-deleted entries to avoid reuse.
Also documents aftersales API and new technician role in AGENTS.md / README.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Frudrax Cheng
2026-05-26 10:51:16 +08:00
parent 1029d5966b
commit b1e3adaf6a
3 changed files with 57 additions and 14 deletions
+8 -2
View File
@@ -44,6 +44,7 @@ backend-go/
├── config/ # Configuration management ├── config/ # Configuration management
│ └── config.go # Config loading (.env, config.yaml, env vars) │ └── config.go # Config loading (.env, config.yaml, env vars)
├── controllers/ # HTTP request handlers ├── controllers/ # HTTP request handlers
│ ├── aftersales_controller.go # Aftersales orders: create, query, submit, confirm, qrcode
│ ├── auth_controller.go # Auth: login, profile, password change │ ├── auth_controller.go # Auth: login, profile, password change
│ ├── companies_controller.go # Company CRUD │ ├── companies_controller.go # Company CRUD
│ ├── employees_controller.go # Employee serials: generate, query, update, revoke, qrcode │ ├── employees_controller.go # Employee serials: generate, query, update, revoke, qrcode
@@ -55,12 +56,13 @@ backend-go/
├── logger/ # Structured logging ├── logger/ # Structured logging
│ └── logger.go # Zap logger wrapper │ └── logger.go # Zap logger wrapper
├── middleware/ # Middleware ├── middleware/ # Middleware
│ └── auth.go # JWT auth, Admin permission check │ └── auth.go # JWT auth, Admin / Technician permission checks
├── models/ # Data models and DTOs ├── models/ # Data models and DTOs
│ └── models.go # User, Company, Serial, EmployeeSerial and DTOs │ └── models.go # User, Company, Serial, EmployeeSerial, AftersalesOrder and DTOs
├── routes/ # Route configuration ├── routes/ # Route configuration
│ └── routes.go # API route registration │ └── routes.go # API route registration
├── services/ # Business logic layer ├── services/ # Business logic layer
│ ├── aftersales_service.go # Aftersales orders: create, list, update, submit, customer confirm, qrcode
│ ├── auth_service.go # Auth: validate user, generate token, password management │ ├── auth_service.go # Auth: validate user, generate token, password management
│ ├── companies_service.go # Company CRUD │ ├── companies_service.go # Company CRUD
│ ├── employees_service.go # Employee serials: generate, query, update, revoke, qrcode │ ├── employees_service.go # Employee serials: generate, query, update, revoke, qrcode
@@ -87,6 +89,9 @@ backend-go/
- **Serials**: `POST /api/serials/generate`, `POST /api/serials/generate-with-prefix`, `POST /api/serials/:serialNumber/qrcode`, `GET /api/serials/:serialNumber/query`, `GET /api/serials`, `PATCH /api/serials/:serialNumber`, `PUT /api/serials/:serialNumber`, `POST /api/serials/:serialNumber/revoke` - **Serials**: `POST /api/serials/generate`, `POST /api/serials/generate-with-prefix`, `POST /api/serials/:serialNumber/qrcode`, `GET /api/serials/:serialNumber/query`, `GET /api/serials`, `PATCH /api/serials/:serialNumber`, `PUT /api/serials/:serialNumber`, `POST /api/serials/:serialNumber/revoke`
- **Companies**: `GET /api/companies/stats/overview`, `GET /api/companies`, `GET /api/companies/:companyName`, `POST /api/companies`, `PATCH /api/companies/:companyName`, `PUT /api/companies/:companyName`, `POST /api/companies/:companyName/revoke`, `DELETE /api/companies/:companyName/serials/:serialNumber`, `DELETE /api/companies/:companyName` - **Companies**: `GET /api/companies/stats/overview`, `GET /api/companies`, `GET /api/companies/:companyName`, `POST /api/companies`, `PATCH /api/companies/:companyName`, `PUT /api/companies/:companyName`, `POST /api/companies/:companyName/revoke`, `DELETE /api/companies/:companyName/serials/:serialNumber`, `DELETE /api/companies/:companyName`
- **Employee Serials**: `POST /api/employee-serials/generate`, `POST /api/employee-serials/:serialNumber/qrcode`, `GET /api/employee-serials/:serialNumber/query`, `GET /api/employee-serials`, `PATCH /api/employee-serials/:serialNumber`, `PUT /api/employee-serials/:serialNumber`, `POST /api/employee-serials/:serialNumber/revoke` - **Employee Serials**: `POST /api/employee-serials/generate`, `POST /api/employee-serials/:serialNumber/qrcode`, `GET /api/employee-serials/:serialNumber/query`, `GET /api/employee-serials`, `PATCH /api/employee-serials/:serialNumber`, `PUT /api/employee-serials/:serialNumber`, `POST /api/employee-serials/:serialNumber/revoke`
- **Aftersales** (公开): `GET /api/aftersales/:serialNumber/query`, `POST /api/aftersales/:serialNumber/confirm`
- **Aftersales** (技术员+管理员): `POST /api/aftersales`, `GET /api/aftersales`, `GET /api/aftersales/:serialNumber`, `PATCH /api/aftersales/:serialNumber`, `POST /api/aftersales/:serialNumber/qrcode`, `POST /api/aftersales/:serialNumber/submit`
- **Aftersales** (仅管理员): `POST /api/aftersales/:serialNumber/reassign`, `POST /api/aftersales/:serialNumber/force-close`, `DELETE /api/aftersales/:serialNumber`
### Import Organization ### Import Organization
Standard imports followed by third-party imports, then project imports (sorted alphabetically): Standard imports followed by third-party imports, then project imports (sorted alphabetically):
@@ -219,6 +224,7 @@ After modifying Swagger annotations, run `make swagger`.
### Middleware ### Middleware
- **JWTAuthMiddleware**: Validates JWT tokens, sets user in context - **JWTAuthMiddleware**: Validates JWT tokens, sets user in context
- **AdminMiddleware**: Checks if user has admin role - **AdminMiddleware**: Checks if user has admin role
- **TechnicianMiddleware**: Allows admin and technician roles (used for aftersales endpoints)
- Access current user: `user, ok := GetCurrentUser(ctx)` - Access current user: `user, ok := GetCurrentUser(ctx)`
### Git Hooks ### Git Hooks
+28 -2
View File
@@ -29,6 +29,7 @@ backend-go/
├── config/ # 配置管理 ├── config/ # 配置管理
│ └── config.go # 配置加载和解析(支持 .env 文件和环境变量) │ └── config.go # 配置加载和解析(支持 .env 文件和环境变量)
├── controllers/ # 控制器层,处理 HTTP 请求 ├── controllers/ # 控制器层,处理 HTTP 请求
│ ├── aftersales_controller.go # 售后工单接口
│ ├── auth_controller.go # 认证相关接口 │ ├── auth_controller.go # 认证相关接口
│ ├── companies_controller.go # 企业管理接口 │ ├── companies_controller.go # 企业管理接口
│ ├── employees_controller.go # 员工赋码接口 │ ├── employees_controller.go # 员工赋码接口
@@ -43,12 +44,13 @@ backend-go/
├── logger/ # 日志管理 ├── logger/ # 日志管理
│ └── logger.go # 结构化日志(使用 Zap) │ └── logger.go # 结构化日志(使用 Zap)
├── middleware/ # 中间件层 ├── middleware/ # 中间件层
│ └── auth.go # JWT 认证权限检查 │ └── auth.go # JWT 认证、管理员/技术员权限检查
├── models/ # 数据模型和 DTO ├── models/ # 数据模型和 DTO
│ └── models.go # User、Company、Serial 等模型定义 │ └── models.go # User、Company、Serial、AftersalesOrder 等模型定义
├── routes/ # 路由配置 ├── routes/ # 路由配置
│ └── routes.go # API 路由注册 │ └── routes.go # API 路由注册
├── services/ # 业务逻辑层 ├── services/ # 业务逻辑层
│ ├── aftersales_service.go # 售后工单业务逻辑
│ ├── auth_service.go # 认证业务逻辑 │ ├── auth_service.go # 认证业务逻辑
│ ├── companies_service.go # 企业管理业务逻辑 │ ├── companies_service.go # 企业管理业务逻辑
│ ├── employees_service.go # 员工赋码业务逻辑 │ ├── employees_service.go # 员工赋码业务逻辑
@@ -292,6 +294,30 @@ swag init -g main.go
- 包含部门(department)和员工姓名(employeeName)信息 - 包含部门(department)和员工姓名(employeeName)信息
- 序列号格式: `EMP26xxxxxx`(EMP + 年份后两位 + 6位随机字符) - 序列号格式: `EMP26xxxxxx`(EMP + 年份后两位 + 6位随机字符)
### 售后工单
| 方法 | 路径 | 描述 | 需要认证 | 角色 |
| ------ | --------------------------------------------- | -------------------------- | -------- | --------------- |
| GET | `/api/aftersales/:serialNumber/query` | 公开查询工单(脱敏) | 否 | 任何 |
| POST | `/api/aftersales/:serialNumber/confirm` | 客户授权/未授权确认 | 否 | 任何 |
| POST | `/api/aftersales` | 创建售后工单 | 是 | 管理员/技术员 |
| GET | `/api/aftersales` | 工单列表(支持筛选) | 是 | 管理员/技术员 |
| GET | `/api/aftersales/:serialNumber` | 工单详情 | 是 | 管理员/技术员 |
| PATCH | `/api/aftersales/:serialNumber` | 更新工单(仅负责人或管理员)| 是 | 管理员/技术员 |
| POST | `/api/aftersales/:serialNumber/qrcode` | 生成工单二维码 | 是 | 管理员/技术员 |
| POST | `/api/aftersales/:serialNumber/submit` | 提交客户确认 | 是 | 管理员/技术员 |
| POST | `/api/aftersales/:serialNumber/reassign` | 重新分配技术员 | 是 | 管理员 |
| POST | `/api/aftersales/:serialNumber/force-close` | 强制关闭工单 | 是 | 管理员 |
| DELETE | `/api/aftersales/:serialNumber` | 删除工单 | 是 | 管理员 |
**售后工单特点**:
- 工单号格式: `zjbf-sh-YYMMNN`(年份后 2 位 + 月份 2 位 + 当月序号至少 2 位,例:`zjbf-sh-260501`
- 序号按月重置,软删除工单不释放编号(避免回收造成混淆)
- 二维码扫码后客户输入手机号后 4 位进行身份校验
- 客户确认接口每分钟同一工单最多 5 次请求
- 工单状态机: `created``pending_confirmation``closed` / `rejected`,被退回后可重新提交
- 公开查询不返回手机号(脱敏)
## 测试 ## 测试
### 运行所有测试 ### 运行所有测试
+21 -10
View File
@@ -1,9 +1,7 @@
package services package services
import ( import (
"crypto/rand"
"encoding/base64" "encoding/base64"
"encoding/hex"
"errors" "errors"
"fmt" "fmt"
"os" "os"
@@ -70,20 +68,33 @@ func normalizeAftersalesSerial(sn string) string {
} }
// generateUniqueSerial 生成唯一的售后工单序列号 // generateUniqueSerial 生成唯一的售后工单序列号
// 格式:zjbf-sh-YYMMNNYY=年份后两位,MM=月份,NN=当月第几单(至少 2 位,溢出自然加宽)
func (s *AftersalesService) generateUniqueSerial() (string, error) { func (s *AftersalesService) generateUniqueSerial() (string, error) {
for attempt := 0; attempt < 10; attempt++ { now := time.Now()
randomBytes := make([]byte, 3) yy := now.Year() % 100
if _, err := rand.Read(randomBytes); err != nil { mm := int(now.Month())
return "", fmt.Errorf("生成随机数失败: %w", err)
} monthStart := time.Date(now.Year(), now.Month(), 1, 0, 0, 0, 0, now.Location())
randomPart := strings.ToLower(hex.EncodeToString(randomBytes))[:6] nextMonth := monthStart.AddDate(0, 1, 0)
candidate := aftersalesSerialPrefix + randomPart
// 统计本月已创建工单数(含软删除,避免编号回收)
var count int64
if err := database.DB.Unscoped().Model(&models.AftersalesOrder{}).
Where("created_at >= ? AND created_at < ?", monthStart, nextMonth).
Count(&count).Error; err != nil {
return "", fmt.Errorf("统计当月工单数失败: %w", err)
}
seq := int(count) + 1
for attempt := 0; attempt < 100; attempt++ {
candidate := fmt.Sprintf("%s%02d%02d%02d", aftersalesSerialPrefix, yy, mm, seq)
var existing models.AftersalesOrder var existing models.AftersalesOrder
result := database.DB.Where("serial_number = ?", candidate).First(&existing) result := database.DB.Unscoped().Where("serial_number = ?", candidate).First(&existing)
if result.Error != nil { if result.Error != nil {
return candidate, nil return candidate, nil
} }
seq++
} }
return "", errors.New("生成唯一序列号失败,请重试") return "", errors.New("生成唯一序列号失败,请重试")
} }