diff --git a/api/README.md b/api/README.md
new file mode 100644
index 0000000..52ec97f
--- /dev/null
+++ b/api/README.md
@@ -0,0 +1,211 @@
+# Coder Common Thin Backend API 文档
+
+## 项目概述
+
+Coder Common Thin Backend 是一个基于Spring Boot 3.5.0的企业级开发框架,采用插件化架构设计,提供了完整的后台管理系统功能。
+
+## 技术栈
+
+- **后端框架**: Spring Boot 3.5.0 + Java 17
+- **数据库**: MySQL 8 + MyBatis Plus 3.5.12
+- **缓存**: Redis (Spring Data Redis)
+- **安全认证**: Sa-Token 1.43.0
+- **权限控制**: RBAC (Role-Based Access Control)
+- **API文档**: OpenAPI 3.0 (Swagger)
+- **工具库**: Hutool 5.8.38, Fastjson2 2.0.57, Guava 33.4.8
+
+## 服务器信息
+
+- **默认端口**: 18099
+- **基础URL**: http://localhost:18099
+- **API前缀**: /coder (除登录认证接口外)
+
+## 认证方式
+
+系统使用Sa-Token进行身份认证和权限管理:
+- **Token类型**: Bearer Token
+- **Token位置**: 请求头 `Authorization` 字段
+- **Token过期**: 支持自定义过期时间
+- **权限验证**: 基于注解的权限验证
+
+## 统一响应格式
+
+所有接口均采用统一的JSON响应格式:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ // 具体数据
+ },
+ "traceId": "trace-id-value"
+}
+```
+
+### 状态码说明
+
+- **200**: 成功
+- **400**: 请求参数错误
+- **401**: 认证失败/未登录
+- **403**: 权限不足
+- **500**: 服务器内部错误
+
+## 分页查询格式
+
+分页查询请求参数:
+```json
+{
+ "pageNo": 1,
+ "pageSize": 10,
+ "params": {
+ // 查询参数
+ }
+}
+```
+
+分页查询响应格式:
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "records": [
+ // 数据记录
+ ],
+ "total": 100,
+ "size": 10,
+ "current": 1,
+ "pages": 10,
+ "searchCount": true
+ },
+ "traceId": "trace-id-value"
+}
+```
+
+## 权限系统
+
+系统采用RBAC权限模型:
+- **用户 (User)**: 系统使用者
+- **角色 (Role)**: 权限的集合
+- **权限 (Permission)**: 具体的操作权限
+- **菜单 (Menu)**: 系统菜单和按钮权限
+
+### 权限验证流程
+
+1. 用户登录获取Token
+2. 请求接口时携带Token
+3. 系统验证Token有效性
+4. 根据用户角色验证权限
+5. 允许或拒绝访问
+
+## API模块分类
+
+### 1. 认证模块 (Authentication)
+- [登录认证API](./authentication/登录认证API)
+- [验证码API](./authentication/验证码API)
+
+### 2. 用户管理模块 (User Management)
+- [用户管理API](./user/用户管理API)
+
+### 3. 权限管理模块 (Permission Management)
+- [菜单管理API](./permission/菜单管理API)
+- [角色管理API](./permission/角色管理API)
+
+### 4. 系统管理模块 (System Management)
+- [文件管理API](./system/文件管理API)
+- [图片管理API](./system/图片管理API)
+- [登录日志API](./system/登录日志API)
+
+## 错误处理
+
+系统提供了完善的错误处理机制:
+
+### 常见错误类型
+
+1. **业务异常**: 自定义业务逻辑错误
+2. **认证异常**: 登录认证相关错误
+3. **权限异常**: 权限验证失败
+4. **参数异常**: 请求参数验证失败
+5. **系统异常**: 服务器内部错误
+
+### 错误响应格式
+
+```json
+{
+ "status": 400,
+ "msg": "具体错误信息",
+ "data": null,
+ "traceId": "trace-id-value"
+}
+```
+
+## 开发环境配置
+
+### 数据库配置
+```yaml
+spring:
+ datasource:
+ url: jdbc:mysql://localhost:3306/coder_common_thin
+ username: root
+ password: your_password
+```
+
+### Redis配置
+```yaml
+spring:
+ redis:
+ host: localhost
+ port: 6379
+ password: your_password
+```
+
+### Sa-Token配置
+```yaml
+sa-token:
+ token-name: Authorization
+ timeout: 2592000
+ activity-timeout: -1
+ is-concurrent: true
+ is-share: false
+ is-read-head: true
+ is-read-cookie: false
+```
+
+## 接口调用示例
+
+### 登录接口调用
+```bash
+curl -X POST \
+ http://localhost:18099/auth/login \
+ -H 'Content-Type: application/json' \
+ -d '{
+ "loginName": "admin",
+ "password": "123456",
+ "codeKey": "uuid-key",
+ "securityCode": "1234"
+ }'
+```
+
+### 携带Token调用接口
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysLoginUser/getLoginUserInformation \
+ -H 'Authorization: Bearer your-token-value'
+```
+
+## 注意事项
+
+1. 所有接口都需要进行认证,除了登录接口和验证码接口
+2. 权限验证基于角色和权限码进行
+3. 系统支持多设备登录
+4. 敏感操作会记录操作日志
+5. 建议在生产环境中启用HTTPS
+
+## 更新日志
+
+- **v1.0.0** (2025-07-05): 初始版本发布
+
+## 联系方式
+
+如有问题,请联系开发团队。
\ No newline at end of file
diff --git a/api/authentication/登录认证API.md b/api/authentication/登录认证API.md
new file mode 100644
index 0000000..5f2be0b
--- /dev/null
+++ b/api/authentication/登录认证API.md
@@ -0,0 +1,255 @@
+# 登录认证API
+
+## 概述
+
+登录认证模块提供用户登录、退出、注册等功能,使用Sa-Token进行身份认证和会话管理。
+
+## 接口列表
+
+### 1. 用户登录
+
+**接口地址**: `POST /auth/login`
+
+**接口描述**: 用户使用账号密码登录系统
+
+**是否需要认证**: 否
+
+**请求参数**:
+
+```json
+{
+ "loginName": "admin",
+ "password": "123456",
+ "codeKey": "uuid-key",
+ "securityCode": "1234",
+ "rememberMe": false
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| loginName | String | 是 | 登录账号 | 长度3-16位,只能包含字母和数字 |
+| password | String | 是 | 登录密码 | 不能为空 |
+| codeKey | String | 是 | 验证码UUID | 不能为空 |
+| securityCode | String | 是 | 验证码 | 不能为空 |
+| rememberMe | Boolean | 否 | 记住登录 | 默认false |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "tokenName": "Authorization",
+ "tokenValue": "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9..."
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**响应参数说明**:
+
+| 参数名 | 类型 | 说明 |
+|--------|------|------|
+| tokenName | String | Token名称 |
+| tokenValue | String | Token值 |
+
+**错误码**:
+
+| 状态码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 账号长度为 3-16 位 | 登录账号长度不符合要求 |
+| 400 | 账号格式为数字以及字母 | 登录账号格式不正确 |
+| 400 | 请输入登录名 | 登录名为空 |
+| 400 | 请输入密码 | 密码为空 |
+| 400 | 验证码已失效 | 验证码Key无效 |
+| 400 | 请输入验证码 | 验证码为空 |
+| 400 | 验证码错误 | 验证码不正确 |
+| 400 | 用户不存在 | 用户账号不存在 |
+| 400 | 密码错误 | 密码不正确 |
+| 400 | 账号已被停用 | 用户账号被禁用 |
+| 400 | 账号已被锁定 | 用户账号被锁定 |
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/auth/login \
+ -H 'Content-Type: application/json' \
+ -d '{
+ "loginName": "admin",
+ "password": "123456",
+ "codeKey": "550e8400-e29b-41d4-a716-446655440000",
+ "securityCode": "1234",
+ "rememberMe": false
+ }'
+```
+
+---
+
+### 2. 用户退出
+
+**接口地址**: `GET /auth/logout`
+
+**接口描述**: 用户退出登录,清除会话信息
+
+**是否需要认证**: 是
+
+**请求头**:
+
+```
+Authorization: eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9...
+```
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "退出成功",
+ "traceId": "trace-123456"
+}
+```
+
+**错误码**:
+
+| 状态码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 401 | 当前会话未登录 | 用户未登录或Token无效 |
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/auth/logout \
+ -H 'Authorization: your-token-value'
+```
+
+---
+
+### 3. 用户注册
+
+**接口地址**: `POST /auth/register`
+
+**接口描述**: 新用户注册账号
+
+**是否需要认证**: 否
+
+**请求参数**:
+
+```json
+{
+ "loginName": "newuser",
+ "password": "123456",
+ "userName": "新用户",
+ "codeKey": "uuid-key",
+ "securityCode": "1234"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| loginName | String | 是 | 登录账号 | 长度3-16位,只能包含字母和数字 |
+| password | String | 是 | 登录密码 | 不能为空 |
+| userName | String | 是 | 用户姓名 | 不能为空 |
+| codeKey | String | 是 | 验证码UUID | 不能为空 |
+| securityCode | String | 是 | 验证码 | 不能为空 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "注册成功",
+ "traceId": "trace-123456"
+}
+```
+
+**错误码**:
+
+| 状态码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 账号长度为 3-16 位 | 登录账号长度不符合要求 |
+| 400 | 账号格式为数字以及字母 | 登录账号格式不正确 |
+| 400 | 请输入登录名 | 登录名为空 |
+| 400 | 请输入密码 | 密码为空 |
+| 400 | 请输入用户姓名 | 用户姓名为空 |
+| 400 | 验证码已失效 | 验证码Key无效 |
+| 400 | 请输入验证码 | 验证码为空 |
+| 400 | 验证码错误 | 验证码不正确 |
+| 400 | 账号已存在 | 登录账号已被注册 |
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/auth/register \
+ -H 'Content-Type: application/json' \
+ -d '{
+ "loginName": "newuser",
+ "password": "123456",
+ "userName": "新用户",
+ "codeKey": "550e8400-e29b-41d4-a716-446655440000",
+ "securityCode": "1234"
+ }'
+```
+
+---
+
+## 认证流程说明
+
+### 1. 登录流程
+
+1. 用户获取验证码(调用验证码接口)
+2. 用户输入账号、密码、验证码
+3. 系统验证验证码有效性
+4. 系统验证用户账号和密码
+5. 系统检查用户状态(是否禁用/锁定)
+6. 生成Token并返回给用户
+7. 用户后续请求携带Token
+
+### 2. Token使用
+
+- Token需要在请求头中携带:`Authorization: token-value`
+- Token有过期时间,过期后需要重新登录
+- 系统支持多设备登录
+- 管理员可以强制用户下线
+
+### 3. 权限验证
+
+- 登录后系统会根据用户角色加载权限
+- 每个接口都会验证用户是否有相应权限
+- 超级管理员拥有所有权限
+
+### 4. 会话管理
+
+- 用户登录信息存储在Redis中
+- 支持会话延长
+- 支持记住登录功能
+- 系统会记录用户登录日志
+
+## 安全特性
+
+1. **密码加密**: 使用盐值加密存储密码
+2. **验证码保护**: 防止暴力破解
+3. **账号锁定**: 连续错误3次自动锁定
+4. **IP限制**: 可配置IP白名单/黑名单
+5. **会话安全**: Token有效期管理
+6. **日志记录**: 记录所有登录操作
+
+## 注意事项
+
+1. 验证码有效期为5分钟
+2. Token默认有效期为30天
+3. 账号锁定时间为10分钟
+4. 注册功能可通过配置开启/关闭
+5. 建议在生产环境中启用HTTPS
\ No newline at end of file
diff --git a/api/authentication/验证码API.md b/api/authentication/验证码API.md
new file mode 100644
index 0000000..c25b3f9
--- /dev/null
+++ b/api/authentication/验证码API.md
@@ -0,0 +1,397 @@
+# 验证码API
+
+## 概述
+
+验证码模块提供图形验证码生成功能,支持PNG和GIF两种格式,用于防止恶意攻击和机器人注册。
+
+## 接口列表
+
+### 1. 生成PNG验证码
+
+**接口地址**: `GET /captcha/png`
+
+**接口描述**: 生成PNG格式的图形验证码
+
+**是否需要认证**: 否
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "uuid": "550e8400-e29b-41d4-a716-446655440000",
+ "base64": "...",
+ "captchaText": "1234"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**响应参数说明**:
+
+| 参数名 | 类型 | 说明 |
+|--------|------|------|
+| uuid | String | 验证码唯一标识,用于后续验证 |
+| base64 | String | 验证码图片Base64编码 |
+| captchaText | String | 验证码文本(开发环境返回,生产环境不返回) |
+
+**特性说明**:
+
+- 验证码长度:4位数字
+- 图片尺寸:102x38像素
+- 有效期:5分钟
+- 字体:随机字体
+- 背景:随机背景色
+- 干扰线:随机干扰线
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/captcha/png \
+ -H 'Content-Type: application/json'
+```
+
+**前端使用示例**:
+
+```javascript
+// 获取验证码
+function getCaptcha() {
+ fetch('/captcha/png')
+ .then(response => response.json())
+ .then(data => {
+ if (data.status === 200) {
+ // 显示验证码图片
+ document.getElementById('captchaImg').src = data.data.base64;
+ // 保存验证码UUID,用于登录时提交
+ document.getElementById('codeKey').value = data.data.uuid;
+ }
+ });
+}
+
+// 点击刷新验证码
+document.getElementById('captchaImg').onclick = getCaptcha;
+```
+
+---
+
+### 2. 生成GIF验证码
+
+**接口地址**: `GET /captcha/gif`
+
+**接口描述**: 生成GIF动画格式的图形验证码
+
+**是否需要认证**: 否
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "uuid": "550e8400-e29b-41d4-a716-446655440001",
+ "base64": "...",
+ "captchaText": "5678"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**响应参数说明**:
+
+| 参数名 | 类型 | 说明 |
+|--------|------|------|
+| uuid | String | 验证码唯一标识,用于后续验证 |
+| base64 | String | 验证码图片Base64编码 |
+| captchaText | String | 验证码文本(开发环境返回,生产环境不返回) |
+
+**特性说明**:
+
+- 验证码长度:4位数字
+- 图片尺寸:102x38像素
+- 有效期:5分钟
+- 动画效果:字符摆动动画
+- 帧数:10帧
+- 动画速度:100ms/帧
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/captcha/gif \
+ -H 'Content-Type: application/json'
+```
+
+**前端使用示例**:
+
+```javascript
+// 获取GIF验证码
+function getGifCaptcha() {
+ fetch('/captcha/gif')
+ .then(response => response.json())
+ .then(data => {
+ if (data.status === 200) {
+ // 显示验证码图片
+ document.getElementById('captchaImg').src = data.data.base64;
+ // 保存验证码UUID,用于登录时提交
+ document.getElementById('codeKey').value = data.data.uuid;
+ }
+ });
+}
+```
+
+---
+
+## 验证码配置
+
+### 1. 验证码参数配置
+
+```yaml
+# application.yml
+captcha:
+ # 验证码长度
+ length: 4
+ # 验证码宽度
+ width: 102
+ # 验证码高度
+ height: 38
+ # 验证码有效期(分钟)
+ timeout: 5
+ # 字体大小
+ font-size: 25
+ # 干扰线数量
+ line-count: 100
+ # 是否开启验证码
+ enabled: true
+```
+
+### 2. 验证码存储
+
+- 验证码使用Redis存储
+- Key格式:`captcha:uuid`
+- 过期时间:5分钟自动删除
+- 值存储:验证码文本(忽略大小写)
+
+### 3. 验证码验证
+
+```java
+// 验证码验证逻辑
+public boolean validateCaptcha(String uuid, String inputCode) {
+ // 从Redis获取验证码
+ String correctCode = redisTemplate.opsForValue().get("captcha:" + uuid);
+
+ if (correctCode == null) {
+ throw new BusinessException("验证码已过期");
+ }
+
+ // 不区分大小写验证
+ if (!correctCode.equalsIgnoreCase(inputCode)) {
+ throw new BusinessException("验证码错误");
+ }
+
+ // 验证成功后立即删除验证码(防止重复使用)
+ redisTemplate.delete("captcha:" + uuid);
+
+ return true;
+}
+```
+
+---
+
+## 安全特性
+
+### 1. 防暴力破解
+
+- 验证码一次性使用
+- 验证后立即删除
+- 5分钟自动过期
+- 支持IP限制
+
+### 2. 防机器识别
+
+- 随机字体和颜色
+- 随机背景和干扰线
+- 字符位置随机偏移
+- GIF动画增加识别难度
+
+### 3. 防重放攻击
+
+- 每次验证码都有唯一UUID
+- 验证码使用后立即失效
+- 不允许重复使用
+
+## 错误处理
+
+### 常见错误情况
+
+| 错误场景 | 错误码 | 错误信息 |
+|----------|--------|----------|
+| 验证码过期 | 400 | 验证码已失效 |
+| 验证码错误 | 400 | 验证码错误 |
+| 验证码为空 | 400 | 请输入验证码 |
+| UUID无效 | 400 | 验证码已失效 |
+| 系统异常 | 500 | 验证码生成失败 |
+
+### 错误响应示例
+
+```json
+{
+ "status": 400,
+ "msg": "验证码已失效",
+ "data": null,
+ "traceId": "trace-123456"
+}
+```
+
+---
+
+## 使用建议
+
+### 1. 前端集成
+
+```html
+
+
+
![验证码]()
+
+
+
+
+```
+
+```javascript
+// 初始化验证码
+function initCaptcha() {
+ fetch('/captcha/png')
+ .then(response => response.json())
+ .then(data => {
+ if (data.status === 200) {
+ document.getElementById('captchaImg').src = data.data.base64;
+ document.getElementById('codeKey').value = data.data.uuid;
+ }
+ });
+}
+
+// 刷新验证码
+function refreshCaptcha() {
+ initCaptcha();
+}
+
+// 页面加载时获取验证码
+window.onload = initCaptcha;
+```
+
+### 2. 表单提交
+
+```javascript
+// 登录表单提交
+function login() {
+ const formData = {
+ loginName: document.getElementById('loginName').value,
+ password: document.getElementById('password').value,
+ codeKey: document.getElementById('codeKey').value,
+ securityCode: document.getElementById('securityCode').value
+ };
+
+ fetch('/auth/login', {
+ method: 'POST',
+ headers: {
+ 'Content-Type': 'application/json'
+ },
+ body: JSON.stringify(formData)
+ })
+ .then(response => response.json())
+ .then(data => {
+ if (data.status === 200) {
+ // 登录成功
+ localStorage.setItem('token', data.data.tokenValue);
+ window.location.href = '/dashboard';
+ } else {
+ // 登录失败,刷新验证码
+ alert(data.msg);
+ refreshCaptcha();
+ }
+ });
+}
+```
+
+### 3. 移动端适配
+
+```css
+/* 移动端验证码样式 */
+@media (max-width: 768px) {
+ .captcha-container {
+ display: flex;
+ align-items: center;
+ gap: 10px;
+ }
+
+ #captchaImg {
+ width: 102px;
+ height: 38px;
+ cursor: pointer;
+ }
+
+ #securityCode {
+ width: 100px;
+ height: 38px;
+ }
+}
+```
+
+## 性能优化
+
+### 1. 缓存策略
+
+- 验证码图片不缓存
+- 验证码文本缓存5分钟
+- 使用Redis集群提高性能
+
+### 2. 并发控制
+
+- 支持高并发验证码生成
+- 每秒可生成1000+验证码
+- 使用连接池管理Redis连接
+
+### 3. 资源优化
+
+- 图片大小优化(小于2KB)
+- 内存使用优化
+- 垃圾回收优化
+
+## 监控和日志
+
+### 1. 指标监控
+
+- 验证码生成次数
+- 验证码验证次数
+- 验证码成功率
+- 验证码过期率
+
+### 2. 日志记录
+
+- 验证码生成日志
+- 验证码验证日志
+- 异常错误日志
+- 性能监控日志
+
+### 3. 告警机制
+
+- 验证码成功率过低告警
+- 验证码生成异常告警
+- Redis连接异常告警
+
+## 注意事项
+
+1. 生产环境不返回验证码文本
+2. 验证码区分大小写(可配置)
+3. 验证码只能使用一次
+4. 建议定期清理过期验证码
+5. 支持自定义验证码样式和难度
\ No newline at end of file
diff --git a/api/permission/菜单管理API.md b/api/permission/菜单管理API.md
new file mode 100644
index 0000000..2a699f2
--- /dev/null
+++ b/api/permission/菜单管理API.md
@@ -0,0 +1,1315 @@
+# 菜单管理API
+
+## 概述
+
+菜单管理模块是权限系统的核心组成部分,负责管理系统菜单、权限配置和路由生成。支持无限级菜单嵌套,提供目录、菜单、按钮三种类型的权限控制。
+
+## 权限说明
+
+菜单管理接口需要相应的权限才能访问:
+
+| 操作 | 权限码 | 说明 |
+|------|--------|------|
+| 查询菜单列表 | `system:menu:list` | 查看菜单列表权限 |
+| 新增菜单 | `system:menu:add` | 新增菜单权限 |
+| 修改菜单 | `system:menu:edit` | 修改菜单信息权限 |
+| 删除菜单 | `system:menu:remove` | 删除菜单权限 |
+| 分配权限 | `system:menu:assign` | 分配菜单权限 |
+
+## 菜单类型说明
+
+| 类型 | 值 | 说明 | 用途 |
+|------|---|------|------|
+| 目录 | 1 | 菜单目录 | 用于菜单分组,不对应具体页面 |
+| 菜单 | 2 | 菜单项 | 对应具体的页面路由 |
+| 按钮 | 3 | 按钮权限 | 用于控制页面内的按钮显示 |
+
+## 接口列表
+
+### 1. 分页查询菜单列表
+
+**接口地址**: `GET /coder/sysMenu/listPage`
+
+**接口描述**: 分页查询系统菜单列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:list`
+
+**请求头**:
+```
+Authorization: your-token-value
+```
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 | 示例 |
+|--------|------|------|------|------|
+| pageNo | Integer | 否 | 页码 | 1 |
+| pageSize | Integer | 否 | 每页大小 | 10 |
+| menuName | String | 否 | 菜单名称 | 用户管理 |
+| menuStatus | String | 否 | 菜单状态 | 0 |
+| auth | String | 否 | 权限标识 | system:user:list |
+
+**响应示例**:
+
+```json
+{
+ "code": 1,
+ "success": true,
+ "msg": "操作成功",
+ "data": [
+ {
+ "menuId": 1,
+ "menuName": "系统管理",
+ "enName": "System Manage",
+ "parentId": 0,
+ "menuType": "1",
+ "name": "systemPage",
+ "path": "/system",
+ "component": "",
+ "icon": "Tools",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "/system/user",
+ "activeMenu": null
+ },
+ {
+ "menuId": 2,
+ "menuName": "用户管理",
+ "enName": "User Manage",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "userPage",
+ "path": "/system/user",
+ "component": "system/user/index",
+ "icon": "UserFilled",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "1",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 12,
+ "menuName": "角色管理",
+ "enName": "Role Manage",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "rolePage",
+ "path": "/system/role",
+ "component": "system/role/index",
+ "icon": "Avatar",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 72,
+ "menuName": "文件管理",
+ "enName": "Files Manage",
+ "parentId": 70,
+ "menuType": "2",
+ "name": "filePage",
+ "path": "/tools/file",
+ "component": "system/file/index",
+ "icon": "FolderOpened",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": ""
+ },
+ {
+ "menuId": 19,
+ "menuName": "菜单管理",
+ "enName": "Menu Manage",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "menuPage",
+ "path": "/system/menu",
+ "component": "system/menu/index",
+ "icon": "Grid",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 77,
+ "menuName": "图库管理",
+ "enName": "Pictures Manage",
+ "parentId": 70,
+ "menuType": "2",
+ "name": "picturePage",
+ "path": "/tools/picture",
+ "component": "system/picture/index",
+ "icon": "Picture",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": ""
+ },
+ {
+ "menuId": 39,
+ "menuName": "登录日志",
+ "enName": "Login Logs",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "loginlogPage",
+ "path": "/system/loginlog",
+ "component": "system/loginlog/index",
+ "icon": "Calendar",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 42,
+ "menuName": "操作日志",
+ "enName": "Operate Logs",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "operlogPage",
+ "path": "/system/operlog",
+ "component": "system/operlog/index",
+ "icon": "Notebook",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 50,
+ "menuName": "个人中心",
+ "enName": "Personage Center",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "personagePage",
+ "path": "/system/personage",
+ "component": "system/personage/index",
+ "icon": "User",
+ "isHide": "0",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": null,
+ "activeMenu": null
+ },
+ {
+ "menuId": 166,
+ "menuName": "外部链接",
+ "enName": "External Link",
+ "parentId": 0,
+ "menuType": "1",
+ "name": "linkPage",
+ "path": "/link",
+ "component": "",
+ "icon": "Link",
+ "isHide": "0",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": null,
+ "activeMenu": null
+ }
+ ]
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysMenu/listPage?pageNo=1&pageSize=10&menuName=系统管理" \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 2. 查询菜单列表
+
+**接口地址**: `GET /coder/sysMenu/list`
+
+**接口描述**: 查询系统菜单列表(树形结构,不分页)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:list`
+
+**请求参数**: 同分页查询(除pageNo、pageSize外)
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "menuId": 1,
+ "menuName": "系统管理",
+ "enName": "System",
+ "parentId": 0,
+ "menuType": "1",
+ "path": "/system",
+ "name": "system",
+ "component": "Layout",
+ "icon": "system",
+ "auth": "",
+ "menuStatus": "0",
+ "activeMenu": "",
+ "isHide": "1",
+ "isLink": "1",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "isSpread": "0",
+ "sorted": 1,
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00",
+ "children": [
+ // 子菜单...
+ ]
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysMenu/list" \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 3. 根据ID查询菜单
+
+**接口地址**: `GET /coder/sysMenu/getById/{id}`
+
+**接口描述**: 根据菜单ID查询菜单详细信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:list`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 菜单ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "menuId": 1,
+ "menuName": "系统管理",
+ "enName": "System",
+ "parentId": 0,
+ "menuType": "1",
+ "path": "/system",
+ "name": "system",
+ "component": "Layout",
+ "icon": "system",
+ "auth": "",
+ "menuStatus": "0",
+ "activeMenu": "",
+ "isHide": "1",
+ "isLink": "1",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "isSpread": "0",
+ "sorted": 1,
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysMenu/getById/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 4. 新增菜单
+
+**接口地址**: `POST /coder/sysMenu/add`
+
+**接口描述**: 新增系统菜单
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:add`
+
+**请求参数**:
+
+```json
+{
+ "menuName": "用户管理",
+ "enName": "User Management",
+ "parentId": 1,
+ "menuType": "2",
+ "path": "/system/user",
+ "name": "user",
+ "component": "/system/user/index",
+ "icon": "user",
+ "auth": "system:user:list",
+ "menuStatus": "0",
+ "activeMenu": "",
+ "isHide": "1",
+ "isLink": "1",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "isSpread": "0",
+ "sorted": 1
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| menuName | String | 是 | 菜单名称 | 不能为空 |
+| enName | String | 否 | 英文名称 | 可为空 |
+| parentId | Long | 是 | 父菜单ID | 不能为空,0表示根菜单 |
+| menuType | String | 是 | 菜单类型 | 1-目录 2-菜单 3-按钮 |
+| path | String | 否 | 路由地址 | 菜单类型为2时必填 |
+| name | String | 否 | 路由名称 | 菜单类型为2时必填 |
+| component | String | 否 | 组件路径 | 菜单类型为2时必填 |
+| icon | String | 否 | 菜单图标 | 可为空 |
+| auth | String | 否 | 权限标识 | 按钮类型时必填 |
+| menuStatus | String | 是 | 菜单状态 | 0-启用 1-停用 |
+| activeMenu | String | 否 | 选中路由 | 可为空 |
+| isHide | String | 是 | 是否隐藏 | 0-隐藏 1-显示 |
+| isLink | String | 否 | 是否外链 | 0-是 1-否 |
+| isKeepAlive | String | 否 | 是否缓存 | 0-是 1-否 |
+| isFull | String | 否 | 是否全屏 | 0-是 1-否 |
+| isAffix | String | 否 | 是否固定 | 0-是 1-否 |
+| isSpread | String | 否 | 是否展开 | 0-是 1-否 |
+| sorted | Integer | 是 | 显示顺序 | 不能为空 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "新增成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysMenu/add \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "menuName": "用户管理",
+ "enName": "User Management",
+ "parentId": 1,
+ "menuType": "2",
+ "path": "/system/user",
+ "name": "user",
+ "component": "/system/user/index",
+ "icon": "user",
+ "auth": "system:user:list",
+ "menuStatus": "0",
+ "activeMenu": "",
+ "isHide": "1",
+ "isLink": "1",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "isSpread": "0",
+ "sorted": 1
+ }'
+```
+
+---
+
+### 5. 修改菜单信息
+
+**接口地址**: `POST /coder/sysMenu/update`
+
+**接口描述**: 修改系统菜单信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:edit`
+
+**请求参数**:
+
+```json
+{
+ "menuId": 1,
+ "menuName": "用户管理",
+ "enName": "User Management",
+ "parentId": 1,
+ "menuType": "2",
+ "path": "/system/user",
+ "name": "user",
+ "component": "/system/user/index",
+ "icon": "user",
+ "auth": "system:user:list",
+ "menuStatus": "0",
+ "activeMenu": "",
+ "isHide": "1",
+ "isLink": "1",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "isSpread": "0",
+ "sorted": 1
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| menuId | Long | 是 | 菜单ID | 必须是有效的菜单ID |
+| 其他参数 | - | - | 同新增菜单 | - |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysMenu/update \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "menuId": 1,
+ "menuName": "用户管理",
+ "parentId": 1,
+ "menuType": "2",
+ "menuStatus": "0",
+ "sorted": 1
+ }'
+```
+
+---
+
+### 6. 删除菜单
+
+**接口地址**: `POST /coder/sysMenu/deleteById/{id}`
+
+**接口描述**: 根据ID删除菜单
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:remove`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 菜单ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysMenu/deleteById/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 7. 批量删除菜单
+
+**接口地址**: `POST /coder/sysMenu/batchDelete`
+
+**接口描述**: 批量删除菜单
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:remove`
+
+**请求参数**:
+
+```json
+{
+ "ids": [1, 2, 3]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| ids | Long[] | 是 | 菜单ID数组 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysMenu/batchDelete \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "ids": [1, 2, 3]
+ }'
+```
+
+---
+
+### 8. 修改菜单状态
+
+**接口地址**: `POST /coder/sysMenu/updateStatus/{id}/{menuStatus}`
+
+**接口描述**: 修改菜单状态(启用/停用)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:edit`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 菜单ID |
+| menuStatus | String | 是 | 菜单状态(0-启用 1-停用) |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysMenu/updateStatus/1/0 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 9. 修改菜单展开状态
+
+**接口地址**: `POST /coder/sysMenu/updateSpread/{id}/{isSpread}`
+
+**接口描述**: 修改菜单展开状态
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:edit`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 菜单ID |
+| isSpread | String | 是 | 展开状态(0-是 1-否) |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysMenu/updateSpread/1/0 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 10. 菜单级联下拉框
+
+**接口地址**: `GET /coder/sysMenu/cascaderList`
+
+**接口描述**: 获取菜单级联下拉框数据
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:list`
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "label": "系统管理",
+ "value": 1,
+ "parentId": "0",
+ "children": [
+ {
+ "label": "用户管理",
+ "value": 2,
+ "parentId": "1"
+ },
+ {
+ "label": "角色管理",
+ "value": 3,
+ "parentId": "1"
+ }
+ ]
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysMenu/cascaderList \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 11. 生成用户菜单路由
+
+**接口地址**: `GET /coder/sysMenu/listRouters`
+
+**接口描述**: 根据当前用户权限生成前端菜单路由
+
+**是否需要认证**: 是
+
+**权限要求**: 无(已登录用户可访问)
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "code": 1,
+ "success": true,
+ "msg": "操作成功",
+ "data": [
+ {
+ "menuId": 1,
+ "menuName": "系统管理",
+ "enName": "System Manage",
+ "parentId": 0,
+ "menuType": "1",
+ "name": "systemPage",
+ "path": "/system",
+ "component": "",
+ "icon": "Tools",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "/system/user",
+ "activeMenu": null
+ },
+ {
+ "menuId": 2,
+ "menuName": "用户管理",
+ "enName": "User Manage",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "userPage",
+ "path": "/system/user",
+ "component": "system/user/index",
+ "icon": "UserFilled",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "1",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 70,
+ "menuName": "系统工具",
+ "enName": "System Tools",
+ "parentId": 0,
+ "menuType": "1",
+ "name": "toolsPage",
+ "path": "/tools",
+ "component": "",
+ "icon": "Tools",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "/tools/file",
+ "activeMenu": null
+ },
+ {
+ "menuId": 12,
+ "menuName": "角色管理",
+ "enName": "Role Manage",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "rolePage",
+ "path": "/system/role",
+ "component": "system/role/index",
+ "icon": "Avatar",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 72,
+ "menuName": "文件管理",
+ "enName": "Files Manage",
+ "parentId": 70,
+ "menuType": "2",
+ "name": "filePage",
+ "path": "/tools/file",
+ "component": "system/file/index",
+ "icon": "FolderOpened",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": ""
+ },
+ {
+ "menuId": 19,
+ "menuName": "菜单管理",
+ "enName": "Menu Manage",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "menuPage",
+ "path": "/system/menu",
+ "component": "system/menu/index",
+ "icon": "Grid",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 77,
+ "menuName": "图库管理",
+ "enName": "Pictures Manage",
+ "parentId": 70,
+ "menuType": "2",
+ "name": "picturePage",
+ "path": "/tools/picture",
+ "component": "system/picture/index",
+ "icon": "Picture",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": ""
+ },
+ {
+ "menuId": 39,
+ "menuName": "登录日志",
+ "enName": "Login Logs",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "loginlogPage",
+ "path": "/system/loginlog",
+ "component": "system/loginlog/index",
+ "icon": "Calendar",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 42,
+ "menuName": "操作日志",
+ "enName": "Operate Logs",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "operlogPage",
+ "path": "/system/operlog",
+ "component": "system/operlog/index",
+ "icon": "Notebook",
+ "isHide": "1",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": "",
+ "activeMenu": null
+ },
+ {
+ "menuId": 50,
+ "menuName": "个人中心",
+ "enName": "Personage Center",
+ "parentId": 1,
+ "menuType": "2",
+ "name": "personagePage",
+ "path": "/system/personage",
+ "component": "system/personage/index",
+ "icon": "User",
+ "isHide": "0",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": null,
+ "activeMenu": null
+ },
+ {
+ "menuId": 166,
+ "menuName": "外部链接",
+ "enName": "External Link",
+ "parentId": 0,
+ "menuType": "1",
+ "name": "linkPage",
+ "path": "/link",
+ "component": "",
+ "icon": "Link",
+ "isHide": "0",
+ "isLink": "",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "redirect": null,
+ "activeMenu": null
+ }
+ ]
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysMenu/listRouters \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 12. 查询正常菜单列表
+
+**接口地址**: `GET /coder/sysMenu/listMenuNormal`
+
+**接口描述**: 查询状态正常的菜单列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:list`
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "menuId": 1,
+ "menuName": "系统管理",
+ "enName": "System",
+ "parentId": 0,
+ "menuType": "1",
+ "path": "/system",
+ "name": "system",
+ "component": "Layout",
+ "icon": "system",
+ "auth": "",
+ "menuStatus": "0",
+ "activeMenu": "",
+ "isHide": "1",
+ "isLink": "1",
+ "isKeepAlive": "0",
+ "isFull": "1",
+ "isAffix": "1",
+ "isSpread": "0",
+ "sorted": 1,
+ "children": [
+ // 子菜单...
+ ]
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysMenu/listMenuNormal \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 13. 根据角色ID查询菜单
+
+**接口地址**: `GET /coder/sysMenu/listMenuIdsByRoleId/{roleId}`
+
+**接口描述**: 根据角色ID查询该角色拥有的菜单ID列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:list`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| roleId | Long | 是 | 角色ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [1, 2, 3, 4, 5],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysMenu/listMenuIdsByRoleId/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 14. 保存角色菜单权限
+
+**接口地址**: `POST /coder/sysMenu/saveRoleMenu/{roleId}/{menuIds}`
+
+**接口描述**: 保存角色的菜单权限
+
+**是否需要认证**: 是
+
+**权限要求**: `system:menu:assign`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| roleId | Long | 是 | 角色ID |
+| menuIds | String | 是 | 菜单ID列表(逗号分隔) |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "保存成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysMenu/saveRoleMenu/1/1,2,3,4,5 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+## 菜单字段说明
+
+### 基础字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| menuId | Long | 菜单ID | 1 |
+| menuName | String | 菜单名称 | 系统管理 |
+| enName | String | 英文名称 | System |
+| parentId | Long | 父菜单ID | 0(根菜单) |
+| menuType | String | 菜单类型 | 1-目录 2-菜单 3-按钮 |
+| sorted | Integer | 显示顺序 | 1 |
+
+### 路由字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| path | String | 路由地址 | /system/user |
+| name | String | 路由名称 | user |
+| component | String | 组件路径 | /system/user/index |
+| redirect | String | 重定向地址 | /system/user |
+| activeMenu | String | 选中路由 | /system/user |
+
+### 权限字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| auth | String | 权限标识 | system:user:list |
+| menuStatus | String | 菜单状态 | 0-启用 1-停用 |
+
+### 显示字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| icon | String | 菜单图标 | user |
+| isHide | String | 是否隐藏 | 0-隐藏 1-显示 |
+| isLink | String | 是否外链 | 0-是 1-否 |
+| isKeepAlive | String | 是否缓存 | 0-是 1-否 |
+| isFull | String | 是否全屏 | 0-是 1-否 |
+| isAffix | String | 是否固定 | 0-是 1-否 |
+| isSpread | String | 是否展开 | 0-是 1-否 |
+
+---
+
+## 前端路由配置
+
+### Vue Router 配置
+
+```javascript
+// 根据后端返回的菜单数据生成路由
+function generateRoutes(menuData) {
+ const routes = [];
+
+ menuData.forEach(menu => {
+ const route = {
+ path: menu.path,
+ name: menu.name,
+ component: () => import(`@/views${menu.component}.vue`),
+ meta: {
+ title: menu.menuName,
+ icon: menu.icon,
+ isHide: menu.isHide === '0',
+ isLink: menu.isLink === '0',
+ isKeepAlive: menu.isKeepAlive === '0',
+ isFull: menu.isFull === '0',
+ isAffix: menu.isAffix === '0',
+ activeMenu: menu.activeMenu
+ }
+ };
+
+ if (menu.children && menu.children.length > 0) {
+ route.children = generateRoutes(menu.children);
+ }
+
+ if (menu.redirect) {
+ route.redirect = menu.redirect;
+ }
+
+ routes.push(route);
+ });
+
+ return routes;
+}
+```
+
+### 权限控制
+
+```javascript
+// 页面权限控制
+router.beforeEach((to, from, next) => {
+ const userPermissions = store.getters.permissions;
+
+ if (to.meta.auth) {
+ if (userPermissions.includes(to.meta.auth) || userPermissions.includes('*:*:*')) {
+ next();
+ } else {
+ next('/403');
+ }
+ } else {
+ next();
+ }
+});
+
+// 按钮权限控制
+Vue.directive('permission', {
+ inserted(el, binding) {
+ const { value } = binding;
+ const permissions = store.getters.permissions;
+
+ if (value) {
+ const hasPermission = permissions.includes(value) || permissions.includes('*:*:*');
+ if (!hasPermission) {
+ el.parentNode && el.parentNode.removeChild(el);
+ }
+ }
+ }
+});
+```
+
+---
+
+## 错误码说明
+
+| 错误码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 菜单名称不能为空 | 菜单名称为空 |
+| 400 | 上级菜单不能为空 | 父菜单ID为空 |
+| 400 | 菜单类型不能为空 | 菜单类型为空 |
+| 400 | 菜单状态不能为空 | 菜单状态为空 |
+| 400 | 是否隐藏菜单不能为空 | 隐藏状态为空 |
+| 400 | 显示顺序不能为空 | 排序值为空 |
+| 400 | 菜单不存在 | 菜单ID不存在 |
+| 400 | 存在子菜单,不允许删除 | 菜单有子菜单时不能删除 |
+| 400 | 菜单已分配,不允许删除 | 菜单已分配给角色时不能删除 |
+| 400 | 不能选择自己作为父菜单 | 父菜单不能是自己 |
+| 400 | 路由地址不能为空 | 菜单类型为菜单时路由地址必填 |
+| 400 | 组件路径不能为空 | 菜单类型为菜单时组件路径必填 |
+| 400 | 权限标识不能为空 | 菜单类型为按钮时权限标识必填 |
+| 401 | 当前会话未登录 | 未登录或Token无效 |
+| 403 | 权限不足 | 没有相应的操作权限 |
+| 500 | 系统异常 | 服务器内部错误 |
+
+---
+
+## 使用建议
+
+### 1. 菜单设计规范
+
+- **层级结构**: 建议不超过3级菜单
+- **命名规范**: 使用有意义的英文名称作为路由名称
+- **权限标识**: 使用模块:功能:操作的格式
+- **图标使用**: 统一使用Element UI或其他图标库
+
+### 2. 权限设计
+
+- **粒度控制**: 按钮级别的权限控制
+- **角色分离**: 不同角色分配不同的菜单权限
+- **继承关系**: 子菜单权限依赖于父菜单权限
+- **缓存策略**: 用户权限信息缓存到前端
+
+### 3. 性能优化
+
+- **懒加载**: 菜单组件使用懒加载
+- **缓存机制**: 菜单数据缓存到本地存储
+- **权限缓存**: 权限信息缓存到内存中
+- **树形结构**: 使用高效的树形数据结构
+
+### 4. 安全考虑
+
+- **权限验证**: 前后端都要进行权限验证
+- **敏感操作**: 重要操作需要二次确认
+- **日志记录**: 记录权限变更操作日志
+- **最小权限**: 遵循最小权限原则
+
+---
+
+## 注意事项
+
+1. **菜单删除**: 删除菜单前需要检查是否有子菜单和角色关联
+2. **权限继承**: 子菜单的权限依赖于父菜单
+3. **缓存更新**: 菜单权限变更后需要清理相关缓存
+4. **前端同步**: 菜单结构变更后前端需要同步更新
+5. **权限验证**: 前后端都需要进行权限验证
+6. **状态管理**: 菜单状态变更会影响用户访问
+7. **排序规则**: 菜单按照sorted字段升序排列
+8. **数据完整性**: 保证菜单数据的完整性和一致性
\ No newline at end of file
diff --git a/api/permission/角色管理API.md b/api/permission/角色管理API.md
new file mode 100644
index 0000000..673a18f
--- /dev/null
+++ b/api/permission/角色管理API.md
@@ -0,0 +1,882 @@
+# 角色管理API
+
+## 概述
+
+角色管理模块是权限系统的重要组成部分,负责管理系统角色、角色权限分配和用户角色关联。基于RBAC权限模型,实现了灵活的角色权限管理机制。
+
+## 权限说明
+
+角色管理接口需要相应的权限才能访问:
+
+| 操作 | 权限码 | 说明 |
+|------|--------|------|
+| 查询角色列表 | `system:role:list` | 查看角色列表权限 |
+| 新增角色 | `system:role:add` | 新增角色权限 |
+| 修改角色 | `system:role:edit` | 修改角色信息权限 |
+| 删除角色 | `system:role:remove` | 删除角色权限 |
+| 分配权限 | `system:role:assign` | 分配角色权限 |
+| 用户授权 | `system:user:role` | 分配用户角色权限 |
+
+## 接口列表
+
+### 1. 分页查询角色列表
+
+**接口地址**: `GET /coder/sysRole/listPage`
+
+**接口描述**: 分页查询系统角色列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:list`
+
+**请求头**:
+```
+Authorization: your-token-value
+```
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 | 示例 |
+|--------|------|------|------|------|
+| pageNo | Integer | 否 | 页码 | 1 |
+| pageSize | Integer | 否 | 每页大小 | 10 |
+| roleName | String | 否 | 角色名称 | 管理员 |
+| roleCode | String | 否 | 角色编码 | admin |
+| roleStatus | String | 否 | 角色状态 | 0 |
+| beginTime | String | 否 | 开始时间 | 2024-01-01 |
+| endTime | String | 否 | 结束时间 | 2024-12-31 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "records": [
+ {
+ "roleId": 1,
+ "roleName": "超级管理员",
+ "roleCode": "admin",
+ "roleStatus": "0",
+ "dataScope": "1",
+ "deptCheckStrictly": "1",
+ "menuCheckStrictly": "1",
+ "sorted": 1,
+ "remark": "超级管理员",
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00",
+ "menuIds": [1, 2, 3, 4, 5],
+ "deptIds": [100, 101, 102]
+ }
+ ],
+ "total": 1,
+ "size": 10,
+ "current": 1,
+ "pages": 1
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysRole/listPage?pageNo=1&pageSize=10&roleName=管理员" \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 2. 查询所有角色
+
+**接口地址**: `GET /coder/sysRole/list`
+
+**接口描述**: 查询所有系统角色(不分页)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:list`
+
+**请求参数**: 同分页查询(除pageNo、pageSize外)
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "roleId": 1,
+ "roleName": "超级管理员",
+ "roleCode": "admin",
+ "roleStatus": "0",
+ "dataScope": "1",
+ "deptCheckStrictly": "1",
+ "menuCheckStrictly": "1",
+ "sorted": 1,
+ "remark": "超级管理员",
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00"
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysRole/list" \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 3. 根据ID查询角色
+
+**接口地址**: `GET /coder/sysRole/getById/{id}`
+
+**接口描述**: 根据角色ID查询角色详细信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:list`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 角色ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "roleId": 1,
+ "roleName": "超级管理员",
+ "roleCode": "admin",
+ "roleStatus": "0",
+ "dataScope": "1",
+ "deptCheckStrictly": "1",
+ "menuCheckStrictly": "1",
+ "sorted": 1,
+ "remark": "超级管理员",
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00",
+ "menuIds": [1, 2, 3, 4, 5],
+ "deptIds": [100, 101, 102]
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysRole/getById/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 4. 新增角色
+
+**接口地址**: `POST /coder/sysRole/add`
+
+**接口描述**: 新增系统角色
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:add`
+
+**请求参数**:
+
+```json
+{
+ "roleName": "普通用户",
+ "roleCode": "user",
+ "roleStatus": "0",
+ "dataScope": "2",
+ "deptCheckStrictly": "1",
+ "menuCheckStrictly": "1",
+ "sorted": 2,
+ "remark": "普通用户角色",
+ "menuIds": [1, 2, 3],
+ "deptIds": [100, 101]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| roleName | String | 是 | 角色名称 | 不能为空,最长30字符 |
+| roleCode | String | 是 | 角色编码 | 不能为空,最长100字符,唯一 |
+| roleStatus | String | 是 | 角色状态 | 0-正常 1-停用 |
+| dataScope | String | 否 | 数据范围 | 1-全部数据 2-自定义数据 3-本部门数据 4-本部门及以下数据 5-仅本人数据 |
+| deptCheckStrictly | String | 否 | 部门树选择项是否关联显示 | 0-父子不互相关联显示 1-父子互相关联显示 |
+| menuCheckStrictly | String | 否 | 菜单树选择项是否关联显示 | 0-父子不互相关联显示 1-父子互相关联显示 |
+| sorted | Integer | 是 | 显示顺序 | 不能为空 |
+| remark | String | 否 | 备注信息 | 最长200字符 |
+| menuIds | Long[] | 否 | 菜单ID数组 | 有效的菜单ID |
+| deptIds | Long[] | 否 | 部门ID数组 | 有效的部门ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "新增成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysRole/add \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "roleName": "普通用户",
+ "roleCode": "user",
+ "roleStatus": "0",
+ "dataScope": "2",
+ "deptCheckStrictly": "1",
+ "menuCheckStrictly": "1",
+ "sorted": 2,
+ "remark": "普通用户角色",
+ "menuIds": [1, 2, 3],
+ "deptIds": [100, 101]
+ }'
+```
+
+---
+
+### 5. 获取最新排序号
+
+**接口地址**: `GET /coder/sysRole/getSorted`
+
+**接口描述**: 获取角色的最新排序号
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:add`
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": 10,
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysRole/getSorted \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 6. 修改角色信息
+
+**接口地址**: `POST /coder/sysRole/update`
+
+**接口描述**: 修改系统角色信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:edit`
+
+**请求参数**:
+
+```json
+{
+ "roleId": 1,
+ "roleName": "超级管理员",
+ "roleCode": "admin",
+ "roleStatus": "0",
+ "dataScope": "1",
+ "deptCheckStrictly": "1",
+ "menuCheckStrictly": "1",
+ "sorted": 1,
+ "remark": "超级管理员",
+ "menuIds": [1, 2, 3, 4, 5],
+ "deptIds": [100, 101, 102]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| roleId | Long | 是 | 角色ID | 必须是有效的角色ID |
+| 其他参数 | - | - | 同新增角色 | - |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysRole/update \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "roleId": 1,
+ "roleName": "超级管理员",
+ "roleCode": "admin",
+ "roleStatus": "0",
+ "sorted": 1
+ }'
+```
+
+---
+
+### 7. 删除角色
+
+**接口地址**: `POST /coder/sysRole/deleteById/{id}`
+
+**接口描述**: 根据ID删除角色
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:remove`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 角色ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysRole/deleteById/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 8. 批量删除角色
+
+**接口地址**: `POST /coder/sysRole/batchDelete`
+
+**接口描述**: 批量删除角色
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:remove`
+
+**请求参数**:
+
+```json
+{
+ "ids": [1, 2, 3]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| ids | Long[] | 是 | 角色ID数组 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysRole/batchDelete \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "ids": [1, 2, 3]
+ }'
+```
+
+---
+
+### 9. 修改角色状态
+
+**接口地址**: `POST /coder/sysRole/updateStatus/{roleId}/{roleStatus}`
+
+**接口描述**: 修改角色状态(正常/停用)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:edit`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| roleId | Long | 是 | 角色ID |
+| roleStatus | String | 是 | 角色状态(0-正常 1-停用) |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysRole/updateStatus/1/0 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 10. 查询正常角色穿梭框
+
+**接口地址**: `GET /coder/sysRole/listNormalRole/{userId}`
+
+**接口描述**: 查询正常状态的角色用于穿梭框显示
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:role`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| userId | Long | 是 | 用户ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "label": "超级管理员",
+ "value": 1,
+ "parentId": ""
+ },
+ {
+ "label": "普通用户",
+ "value": 2,
+ "parentId": ""
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysRole/listNormalRole/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 11. 分配用户角色
+
+**接口地址**: `GET /coder/sysRole/assignUserRole/{userId}/{roleIds}`
+
+**接口描述**: 为用户分配角色
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:role`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| userId | Long | 是 | 用户ID |
+| roleIds | String | 是 | 角色ID列表(逗号分隔) |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "分配成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysRole/assignUserRole/1/1,2,3 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 12. 获取角色下拉框
+
+**接口地址**: `GET /coder/sysRole/listRoleElSelect`
+
+**接口描述**: 获取角色下拉框数据
+
+**是否需要认证**: 是
+
+**权限要求**: `system:role:list`
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "label": "超级管理员",
+ "value": 1
+ },
+ {
+ "label": "普通用户",
+ "value": 2
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysRole/listRoleElSelect \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+## 角色字段说明
+
+### 基础字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| roleId | Long | 角色ID | 1 |
+| roleName | String | 角色名称 | 超级管理员 |
+| roleCode | String | 角色编码 | admin |
+| roleStatus | String | 角色状态 | 0-正常 1-停用 |
+| sorted | Integer | 显示顺序 | 1 |
+| remark | String | 备注信息 | 超级管理员 |
+
+### 权限字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| dataScope | String | 数据权限范围 | 1-全部数据 2-自定义数据 3-本部门数据 4-本部门及以下数据 5-仅本人数据 |
+| deptCheckStrictly | String | 部门树选择项是否关联显示 | 0-父子不互相关联显示 1-父子互相关联显示 |
+| menuCheckStrictly | String | 菜单树选择项是否关联显示 | 0-父子不互相关联显示 1-父子互相关联显示 |
+
+### 关联字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| menuIds | Long[] | 菜单ID数组 | [1, 2, 3, 4, 5] |
+| deptIds | Long[] | 部门ID数组 | [100, 101, 102] |
+
+---
+
+## 数据字典
+
+### 角色状态 (roleStatus)
+
+| 值 | 说明 |
+|----|------|
+| 0 | 正常 |
+| 1 | 停用 |
+
+### 数据权限范围 (dataScope)
+
+| 值 | 说明 |
+|----|------|
+| 1 | 全部数据权限 |
+| 2 | 自定数据权限 |
+| 3 | 本部门数据权限 |
+| 4 | 本部门及以下数据权限 |
+| 5 | 仅本人数据权限 |
+
+### 树选择项关联显示
+
+| 值 | 说明 |
+|----|------|
+| 0 | 父子不互相关联显示 |
+| 1 | 父子互相关联显示 |
+
+---
+
+## 权限验证机制
+
+### 1. 角色权限验证
+
+```java
+// Sa-Token权限验证
+@SaCheckPermission("system:role:list")
+public List list() {
+ // 业务逻辑
+}
+```
+
+### 2. 数据权限控制
+
+```java
+// 根据用户数据权限过滤数据
+public List getRolesByDataScope(Long userId) {
+ SysUser user = getCurrentUser();
+ String dataScope = user.getDataScope();
+
+ switch (dataScope) {
+ case "1": // 全部数据权限
+ return getAllRoles();
+ case "2": // 自定义数据权限
+ return getRolesByDeptIds(user.getDeptIds());
+ case "3": // 本部门数据权限
+ return getRolesByDept(user.getDeptId());
+ case "4": // 本部门及以下数据权限
+ return getRolesByDeptAndChildren(user.getDeptId());
+ case "5": // 仅本人数据权限
+ return getRolesByUserId(userId);
+ default:
+ return new ArrayList<>();
+ }
+}
+```
+
+### 3. 角色权限缓存
+
+```java
+// 缓存用户角色权限
+@Cacheable(value = "userRoles", key = "#userId")
+public List getRolesByUserId(Long userId) {
+ return roleMapper.selectRolePermissionByUserId(userId);
+}
+
+// 清除角色权限缓存
+@CacheEvict(value = "userRoles", key = "#userId")
+public void clearUserRoleCache(Long userId) {
+ // 角色变更时清除缓存
+}
+```
+
+---
+
+## 错误码说明
+
+| 错误码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 角色名称不能为空 | 角色名称为空 |
+| 400 | 角色权限编码不能为空 | 角色编码为空 |
+| 400 | 显示顺序不能为空 | 排序值为空 |
+| 400 | 角色不存在 | 角色ID不存在 |
+| 400 | 角色编码已存在 | 角色编码重复 |
+| 400 | 角色已分配用户,不允许删除 | 角色已分配给用户时不能删除 |
+| 400 | 不能删除超级管理员角色 | 超级管理员角色不能删除 |
+| 400 | 不能停用超级管理员角色 | 超级管理员角色不能停用 |
+| 400 | 用户不存在 | 用户ID不存在 |
+| 400 | 角色ID列表不能为空 | 分配角色时角色ID为空 |
+| 401 | 当前会话未登录 | 未登录或Token无效 |
+| 403 | 权限不足 | 没有相应的操作权限 |
+| 500 | 系统异常 | 服务器内部错误 |
+
+---
+
+## RBAC权限模型
+
+### 1. 基本概念
+
+- **用户 (User)**: 系统的使用者
+- **角色 (Role)**: 权限的集合,连接用户和权限的桥梁
+- **权限 (Permission)**: 对系统资源的操作权限
+- **会话 (Session)**: 用户与系统的一次交互过程
+
+### 2. 关系模型
+
+```
+用户 ←→ 用户角色关系 ←→ 角色 ←→ 角色权限关系 ←→ 权限
+```
+
+### 3. 数据库设计
+
+```sql
+-- 用户表
+CREATE TABLE sys_login_user (
+ user_id BIGINT PRIMARY KEY,
+ login_name VARCHAR(32) UNIQUE NOT NULL,
+ user_name VARCHAR(32) NOT NULL,
+ -- 其他字段...
+);
+
+-- 角色表
+CREATE TABLE sys_role (
+ role_id BIGINT PRIMARY KEY,
+ role_name VARCHAR(32) NOT NULL,
+ role_code VARCHAR(32) UNIQUE NOT NULL,
+ role_status CHAR(1) DEFAULT '0',
+ -- 其他字段...
+);
+
+-- 权限表(菜单表)
+CREATE TABLE sys_menu (
+ menu_id BIGINT PRIMARY KEY,
+ menu_name VARCHAR(64) NOT NULL,
+ auth VARCHAR(128),
+ -- 其他字段...
+);
+
+-- 用户角色关联表
+CREATE TABLE sys_user_role (
+ user_id BIGINT NOT NULL,
+ role_id BIGINT NOT NULL,
+ PRIMARY KEY (user_id, role_id)
+);
+
+-- 角色权限关联表
+CREATE TABLE sys_role_menu (
+ role_id BIGINT NOT NULL,
+ menu_id BIGINT NOT NULL,
+ PRIMARY KEY (role_id, menu_id)
+);
+```
+
+---
+
+## 使用建议
+
+### 1. 角色设计原则
+
+- **最小权限原则**: 每个角色只分配必要的权限
+- **职责分离**: 不同职责的操作分配给不同角色
+- **角色层次**: 建立角色层次结构,高级角色继承低级角色权限
+- **定期审查**: 定期审查角色权限,及时调整
+
+### 2. 权限分配策略
+
+- **默认无权限**: 新建角色默认无任何权限
+- **逐步授权**: 根据业务需要逐步授权
+- **权限组合**: 通过多个角色组合实现复杂权限需求
+- **临时授权**: 支持临时权限授权机制
+
+### 3. 数据权限控制
+
+- **部门数据权限**: 基于用户所属部门控制数据访问范围
+- **自定义数据权限**: 支持自定义数据访问范围
+- **数据隔离**: 确保不同用户只能访问授权范围内的数据
+- **审计日志**: 记录数据访问日志
+
+### 4. 性能优化
+
+- **权限缓存**: 将用户权限信息缓存到Redis
+- **批量操作**: 支持批量分配和撤销权限
+- **索引优化**: 对关联表建立合适的索引
+- **分页查询**: 大数据量时使用分页查询
+
+---
+
+## 安全考虑
+
+### 1. 权限验证
+
+- **双重验证**: 前后端都要进行权限验证
+- **接口鉴权**: 每个接口都要验证权限
+- **数据鉴权**: 数据层面的权限验证
+- **操作鉴权**: 敏感操作的二次验证
+
+### 2. 权限变更
+
+- **变更日志**: 记录所有权限变更操作
+- **审批流程**: 重要权限变更需要审批
+- **通知机制**: 权限变更及时通知相关人员
+- **回滚机制**: 支持权限变更回滚
+
+### 3. 会话安全
+
+- **会话超时**: 设置会话超时时间
+- **单点登录**: 防止重复登录
+- **强制下线**: 支持强制用户下线
+- **设备绑定**: 支持设备级别的访问控制
+
+---
+
+## 注意事项
+
+1. **角色删除**: 删除角色前需要检查是否有用户关联
+2. **权限继承**: 考虑角色之间的权限继承关系
+3. **缓存一致性**: 角色权限变更后需要清理相关缓存
+4. **数据完整性**: 保证角色权限数据的完整性
+5. **超级管理员**: 超级管理员角色不能删除和停用
+6. **权限验证**: 前后端都需要进行权限验证
+7. **审计日志**: 记录所有角色操作日志
+8. **性能考虑**: 大量用户时注意权限验证性能
+9. **数据权限**: 正确配置数据权限范围
+10. **定期清理**: 定期清理无效的角色和权限关联
\ No newline at end of file
diff --git a/api/system/图片管理API.md b/api/system/图片管理API.md
new file mode 100644
index 0000000..67339a6
--- /dev/null
+++ b/api/system/图片管理API.md
@@ -0,0 +1,871 @@
+# 图片管理API
+
+## 概述
+
+图片管理模块专门用于管理系统中的图片资源,提供图片的上传、存储、展示和管理功能。支持多种图片格式,具备图片压缩、水印、缩略图生成等高级功能。
+
+## 权限说明
+
+图片管理接口需要相应的权限才能访问:
+
+| 操作 | 权限码 | 说明 |
+|------|--------|------|
+| 查询图片列表 | `system:picture:list` | 查看图片列表权限 |
+| 上传图片 | `system:picture:upload` | 上传图片权限 |
+| 删除图片 | `system:picture:remove` | 删除图片权限 |
+| 编辑图片 | `system:picture:edit` | 编辑图片信息权限 |
+
+## 接口列表
+
+### 1. 分页查询图片列表
+
+**接口地址**: `GET /coder/sysPicture/listPage`
+
+**接口描述**: 分页查询系统图片列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:picture:list`
+
+**请求头**:
+```
+Authorization: Bearer your-token-value
+```
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 | 示例 |
+|--------|------|------|------|------|
+| pageNo | Integer | 否 | 页码 | 1 |
+| pageSize | Integer | 否 | 每页大小 | 10 |
+| pictureName | String | 否 | 图片名称 | banner.jpg |
+| pictureType | String | 否 | 图片类型 | jpg |
+| albumName | String | 否 | 相册名称 | banner |
+| pictureStatus | String | 否 | 图片状态 | 0 |
+| beginTime | String | 否 | 开始时间 | 2024-01-01 |
+| endTime | String | 否 | 结束时间 | 2024-12-31 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "records": [
+ {
+ "pictureId": 1,
+ "pictureName": "banner_20240705_001.jpg",
+ "originalName": "主页横幅.jpg",
+ "picturePath": "/upload/pictures/banner/2024/07/05/banner_20240705_001.jpg",
+ "pictureUrl": "http://localhost:18099/upload/pictures/banner/2024/07/05/banner_20240705_001.jpg",
+ "thumbnailUrl": "http://localhost:18099/upload/pictures/banner/2024/07/05/thumb_banner_20240705_001.jpg",
+ "albumName": "banner",
+ "pictureSize": 2048000,
+ "pictureType": "jpg",
+ "pictureFormat": "JPEG",
+ "width": 1920,
+ "height": 1080,
+ "pictureStatus": "0",
+ "isWatermark": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "description": "主页横幅图片",
+ "tags": "横幅,主页,展示",
+ "viewCount": 100,
+ "downloadCount": 10,
+ "remark": "网站主页横幅",
+ "createBy": "admin",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-07-05 10:00:00"
+ }
+ ],
+ "total": 1,
+ "size": 10,
+ "current": 1,
+ "pages": 1
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysPicture/listPage?pageNo=1&pageSize=10&albumName=banner" \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 2. 查询所有图片
+
+**接口地址**: `GET /coder/sysPicture/list`
+
+**接口描述**: 查询所有系统图片(不分页)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:picture:list`
+
+**请求参数**: 同分页查询(除pageNo、pageSize外)
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "pictureId": 1,
+ "pictureName": "banner_20240705_001.jpg",
+ "originalName": "主页横幅.jpg",
+ "picturePath": "/upload/pictures/banner/2024/07/05/banner_20240705_001.jpg",
+ "pictureUrl": "http://localhost:18099/upload/pictures/banner/2024/07/05/banner_20240705_001.jpg",
+ "thumbnailUrl": "http://localhost:18099/upload/pictures/banner/2024/07/05/thumb_banner_20240705_001.jpg",
+ "albumName": "banner",
+ "pictureSize": 2048000,
+ "pictureType": "jpg",
+ "pictureFormat": "JPEG",
+ "width": 1920,
+ "height": 1080,
+ "pictureStatus": "0",
+ "isWatermark": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "description": "主页横幅图片",
+ "tags": "横幅,主页,展示",
+ "viewCount": 100,
+ "downloadCount": 10,
+ "remark": "网站主页横幅",
+ "createBy": "admin",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-07-05 10:00:00"
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysPicture/list?albumName=banner" \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 3. 根据ID查询图片
+
+**接口地址**: `GET /coder/sysPicture/getById/{id}`
+
+**接口描述**: 根据图片ID查询图片详细信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:picture:list`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 图片ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "pictureId": 1,
+ "pictureName": "banner_20240705_001.jpg",
+ "originalName": "主页横幅.jpg",
+ "picturePath": "/upload/pictures/banner/2024/07/05/banner_20240705_001.jpg",
+ "pictureUrl": "http://localhost:18099/upload/pictures/banner/2024/07/05/banner_20240705_001.jpg",
+ "thumbnailUrl": "http://localhost:18099/upload/pictures/banner/2024/07/05/thumb_banner_20240705_001.jpg",
+ "albumName": "banner",
+ "pictureSize": 2048000,
+ "pictureType": "jpg",
+ "pictureFormat": "JPEG",
+ "width": 1920,
+ "height": 1080,
+ "pictureStatus": "0",
+ "isWatermark": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "description": "主页横幅图片",
+ "tags": "横幅,主页,展示",
+ "viewCount": 100,
+ "downloadCount": 10,
+ "remark": "网站主页横幅",
+ "createBy": "admin",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-07-05 10:00:00"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysPicture/getById/1 \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 4. 新增图片记录
+
+**接口地址**: `POST /coder/sysPicture/add`
+
+**接口描述**: 新增系统图片记录
+
+**是否需要认证**: 是
+
+**权限要求**: `system:picture:upload`
+
+**请求参数**:
+
+```json
+{
+ "pictureName": "product_20240705_001.jpg",
+ "originalName": "产品展示图.jpg",
+ "picturePath": "/upload/pictures/product/2024/07/05/product_20240705_001.jpg",
+ "pictureUrl": "http://localhost:18099/upload/pictures/product/2024/07/05/product_20240705_001.jpg",
+ "thumbnailUrl": "http://localhost:18099/upload/pictures/product/2024/07/05/thumb_product_20240705_001.jpg",
+ "albumName": "product",
+ "pictureSize": 1536000,
+ "pictureType": "jpg",
+ "pictureFormat": "JPEG",
+ "width": 1200,
+ "height": 800,
+ "pictureStatus": "0",
+ "isWatermark": "1",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "description": "产品展示图片",
+ "tags": "产品,展示,商品",
+ "remark": "商品详情页图片"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| pictureName | String | 是 | 存储图片名 | 不能为空 |
+| originalName | String | 是 | 原始图片名 | 不能为空 |
+| picturePath | String | 是 | 图片相对路径 | 不能为空 |
+| pictureUrl | String | 是 | 图片访问URL | 不能为空 |
+| thumbnailUrl | String | 否 | 缩略图URL | 可为空 |
+| albumName | String | 是 | 相册名称 | 不能为空 |
+| pictureSize | Long | 是 | 图片大小 | 必须大于0 |
+| pictureType | String | 是 | 图片类型 | 不能为空 |
+| pictureFormat | String | 是 | 图片格式 | 不能为空 |
+| width | Integer | 否 | 图片宽度 | 大于0 |
+| height | Integer | 否 | 图片高度 | 大于0 |
+| pictureStatus | String | 是 | 图片状态 | 0-正常 1-删除 |
+| isWatermark | String | 否 | 是否水印 | 0-否 1-是 |
+| uploadUserId | Long | 否 | 上传用户ID | 有效的用户ID |
+| uploadUserName | String | 否 | 上传用户名 | 可为空 |
+| description | String | 否 | 图片描述 | 最长500字符 |
+| tags | String | 否 | 图片标签 | 逗号分隔 |
+| remark | String | 否 | 备注信息 | 最长200字符 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "新增成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysPicture/add \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "pictureName": "product_20240705_001.jpg",
+ "originalName": "产品展示图.jpg",
+ "picturePath": "/upload/pictures/product/2024/07/05/product_20240705_001.jpg",
+ "pictureUrl": "http://localhost:18099/upload/pictures/product/2024/07/05/product_20240705_001.jpg",
+ "albumName": "product",
+ "pictureSize": 1536000,
+ "pictureType": "jpg",
+ "pictureFormat": "JPEG",
+ "width": 1200,
+ "height": 800,
+ "pictureStatus": "0",
+ "description": "产品展示图片",
+ "tags": "产品,展示,商品"
+ }'
+```
+
+---
+
+### 5. 修改图片信息
+
+**接口地址**: `POST /coder/sysPicture/update`
+
+**接口描述**: 修改系统图片记录信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:picture:edit`
+
+**请求参数**:
+
+```json
+{
+ "pictureId": 1,
+ "pictureName": "product_20240705_001.jpg",
+ "originalName": "产品展示图.jpg",
+ "picturePath": "/upload/pictures/product/2024/07/05/product_20240705_001.jpg",
+ "pictureUrl": "http://localhost:18099/upload/pictures/product/2024/07/05/product_20240705_001.jpg",
+ "thumbnailUrl": "http://localhost:18099/upload/pictures/product/2024/07/05/thumb_product_20240705_001.jpg",
+ "albumName": "product",
+ "pictureSize": 1536000,
+ "pictureType": "jpg",
+ "pictureFormat": "JPEG",
+ "width": 1200,
+ "height": 800,
+ "pictureStatus": "0",
+ "isWatermark": "1",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "description": "产品展示图片(已更新)",
+ "tags": "产品,展示,商品,新品",
+ "remark": "商品详情页图片(已更新)"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| pictureId | Long | 是 | 图片ID | 必须是有效的图片ID |
+| 其他参数 | - | - | 同新增图片记录 | - |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysPicture/update \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "pictureId": 1,
+ "description": "产品展示图片(已更新)",
+ "tags": "产品,展示,商品,新品"
+ }'
+```
+
+---
+
+### 6. 删除图片
+
+**接口地址**: `POST /coder/sysPicture/deleteById/{id}`
+
+**接口描述**: 根据ID删除图片记录和物理文件
+
+**是否需要认证**: 是
+
+**权限要求**: `system:picture:remove`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 图片ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysPicture/deleteById/1 \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 7. 批量删除图片
+
+**接口地址**: `POST /coder/sysPicture/batchDelete`
+
+**接口描述**: 批量删除图片记录和物理文件
+
+**是否需要认证**: 是
+
+**权限要求**: `system:picture:remove`
+
+**请求参数**:
+
+```json
+{
+ "ids": [1, 2, 3]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| ids | Long[] | 是 | 图片ID数组 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysPicture/batchDelete \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "ids": [1, 2, 3]
+ }'
+```
+
+---
+
+## 图片字段说明
+
+### 基础字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| pictureId | Long | 图片ID | 1 |
+| pictureName | String | 存储图片名 | banner_20240705_001.jpg |
+| originalName | String | 原始图片名 | 主页横幅.jpg |
+| picturePath | String | 图片相对路径 | /upload/pictures/banner/2024/07/05/banner_20240705_001.jpg |
+| pictureUrl | String | 图片访问URL | http://localhost:18099/upload/pictures/banner/2024/07/05/banner_20240705_001.jpg |
+| thumbnailUrl | String | 缩略图URL | http://localhost:18099/upload/pictures/banner/2024/07/05/thumb_banner_20240705_001.jpg |
+
+### 分类字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| albumName | String | 相册名称 | banner |
+| pictureType | String | 图片扩展名 | jpg |
+| pictureFormat | String | 图片格式 | JPEG |
+| pictureSize | Long | 图片大小(字节) | 2048000 |
+
+### 尺寸字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| width | Integer | 图片宽度(像素) | 1920 |
+| height | Integer | 图片高度(像素) | 1080 |
+
+### 状态字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| pictureStatus | String | 图片状态 | 0-正常 1-删除 |
+| isWatermark | String | 是否有水印 | 0-否 1-是 |
+| uploadUserId | Long | 上传用户ID | 1 |
+| uploadUserName | String | 上传用户名 | admin |
+
+### 描述字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| description | String | 图片描述 | 主页横幅图片 |
+| tags | String | 图片标签 | 横幅,主页,展示 |
+| remark | String | 备注信息 | 网站主页横幅 |
+
+### 统计字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| viewCount | Integer | 查看次数 | 100 |
+| downloadCount | Integer | 下载次数 | 10 |
+
+---
+
+## 数据字典
+
+### 图片状态 (pictureStatus)
+
+| 值 | 说明 |
+|----|------|
+| 0 | 正常 |
+| 1 | 已删除 |
+
+### 是否水印 (isWatermark)
+
+| 值 | 说明 |
+|----|------|
+| 0 | 无水印 |
+| 1 | 有水印 |
+
+### 支持的图片格式
+
+| 格式 | 扩展名 | MIME类型 | 说明 |
+|------|--------|----------|------|
+| JPEG | jpg, jpeg | image/jpeg | 有损压缩,适合照片 |
+| PNG | png | image/png | 无损压缩,支持透明 |
+| GIF | gif | image/gif | 支持动画 |
+| WebP | webp | image/webp | 现代格式,压缩率高 |
+| BMP | bmp | image/bmp | 位图格式 |
+| TIFF | tiff, tif | image/tiff | 高质量图像 |
+
+### 相册分类
+
+| 相册名称 | 说明 | 用途 |
+|----------|------|------|
+| banner | 横幅图片 | 网站横幅展示 |
+| product | 产品图片 | 商品展示 |
+| avatar | 头像图片 | 用户头像 |
+| gallery | 图库图片 | 相册展示 |
+| article | 文章图片 | 文章配图 |
+| icon | 图标图片 | 小图标 |
+| background | 背景图片 | 页面背景 |
+
+---
+
+## 图片处理功能
+
+### 1. 图片上传处理
+
+```java
+@Service
+public class PictureUploadService {
+
+ /**
+ * 上传图片
+ */
+ public PictureUploadResult uploadPicture(MultipartFile file, String album) {
+ // 1. 图片校验
+ validatePicture(file);
+
+ // 2. 生成文件名
+ String fileName = generatePictureName(file.getOriginalFilename());
+
+ // 3. 保存原图
+ String originalPath = savePicture(file, album, fileName);
+
+ // 4. 生成缩略图
+ String thumbnailPath = generateThumbnail(originalPath, album, fileName);
+
+ // 5. 添加水印
+ if (needWatermark(album)) {
+ addWatermark(originalPath);
+ }
+
+ // 6. 获取图片信息
+ BufferedImage image = ImageIO.read(new File(originalPath));
+ int width = image.getWidth();
+ int height = image.getHeight();
+
+ // 7. 生成访问URL
+ String pictureUrl = generatePictureUrl(album, fileName);
+ String thumbnailUrl = generateThumbnailUrl(album, fileName);
+
+ // 8. 返回结果
+ return PictureUploadResult.builder()
+ .pictureName(fileName)
+ .originalName(file.getOriginalFilename())
+ .picturePath(originalPath)
+ .pictureUrl(pictureUrl)
+ .thumbnailUrl(thumbnailUrl)
+ .pictureSize(file.getSize())
+ .pictureType(getFileExtension(fileName))
+ .pictureFormat(getImageFormat(file))
+ .width(width)
+ .height(height)
+ .build();
+ }
+
+ /**
+ * 生成缩略图
+ */
+ private String generateThumbnail(String originalPath, String album, String fileName) {
+ try {
+ BufferedImage originalImage = ImageIO.read(new File(originalPath));
+
+ // 计算缩略图尺寸
+ int thumbnailWidth = 200;
+ int thumbnailHeight = 150;
+
+ // 保持宽高比
+ double ratio = Math.min((double) thumbnailWidth / originalImage.getWidth(),
+ (double) thumbnailHeight / originalImage.getHeight());
+ int width = (int) (originalImage.getWidth() * ratio);
+ int height = (int) (originalImage.getHeight() * ratio);
+
+ // 生成缩略图
+ BufferedImage thumbnailImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
+ Graphics2D g2d = thumbnailImage.createGraphics();
+ g2d.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BILINEAR);
+ g2d.drawImage(originalImage, 0, 0, width, height, null);
+ g2d.dispose();
+
+ // 保存缩略图
+ String thumbnailPath = getThumbnailPath(album, fileName);
+ ImageIO.write(thumbnailImage, "jpg", new File(thumbnailPath));
+
+ return thumbnailPath;
+ } catch (IOException e) {
+ throw new BusinessException("生成缩略图失败", e);
+ }
+ }
+
+ /**
+ * 添加水印
+ */
+ private void addWatermark(String imagePath) {
+ try {
+ BufferedImage originalImage = ImageIO.read(new File(imagePath));
+ Graphics2D g2d = originalImage.createGraphics();
+
+ // 设置水印属性
+ g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, 0.5f));
+ g2d.setColor(Color.LIGHT_GRAY);
+ g2d.setFont(new Font("Arial", Font.BOLD, 20));
+
+ // 计算水印位置
+ FontMetrics fm = g2d.getFontMetrics();
+ String watermarkText = "© Your Company";
+ int x = originalImage.getWidth() - fm.stringWidth(watermarkText) - 10;
+ int y = originalImage.getHeight() - fm.getHeight() + fm.getAscent() - 10;
+
+ // 绘制水印
+ g2d.drawString(watermarkText, x, y);
+ g2d.dispose();
+
+ // 保存带水印的图片
+ ImageIO.write(originalImage, "jpg", new File(imagePath));
+ } catch (IOException e) {
+ throw new BusinessException("添加水印失败", e);
+ }
+ }
+}
+```
+
+### 2. 图片压缩
+
+```java
+/**
+ * 图片压缩
+ */
+public void compressPicture(String inputPath, String outputPath, float quality) {
+ try {
+ BufferedImage image = ImageIO.read(new File(inputPath));
+
+ // 创建输出流
+ FileOutputStream fos = new FileOutputStream(outputPath);
+
+ // 获取JPEG写入器
+ Iterator writers = ImageIO.getImageWritersByFormatName("jpg");
+ ImageWriter writer = writers.next();
+
+ // 设置输出
+ ImageOutputStream ios = ImageIO.createImageOutputStream(fos);
+ writer.setOutput(ios);
+
+ // 设置压缩参数
+ ImageWriteParam param = writer.getDefaultWriteParam();
+ param.setCompressionMode(ImageWriteParam.MODE_EXPLICIT);
+ param.setCompressionQuality(quality); // 压缩质量 0.0-1.0
+
+ // 写入图片
+ writer.write(null, new IIOImage(image, null, null), param);
+
+ // 清理资源
+ ios.close();
+ fos.close();
+ writer.dispose();
+ } catch (IOException e) {
+ throw new BusinessException("图片压缩失败", e);
+ }
+}
+```
+
+### 3. 图片格式转换
+
+```java
+/**
+ * 图片格式转换
+ */
+public void convertPictureFormat(String inputPath, String outputPath, String format) {
+ try {
+ BufferedImage image = ImageIO.read(new File(inputPath));
+
+ // 如果转换为JPEG,需要处理透明背景
+ if ("jpg".equalsIgnoreCase(format) || "jpeg".equalsIgnoreCase(format)) {
+ BufferedImage jpegImage = new BufferedImage(image.getWidth(), image.getHeight(), BufferedImage.TYPE_INT_RGB);
+ Graphics2D g2d = jpegImage.createGraphics();
+ g2d.setColor(Color.WHITE);
+ g2d.fillRect(0, 0, image.getWidth(), image.getHeight());
+ g2d.drawImage(image, 0, 0, null);
+ g2d.dispose();
+ image = jpegImage;
+ }
+
+ // 保存转换后的图片
+ ImageIO.write(image, format, new File(outputPath));
+ } catch (IOException e) {
+ throw new BusinessException("图片格式转换失败", e);
+ }
+}
+```
+
+---
+
+## 错误码说明
+
+| 错误码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 图片不能为空 | 上传图片为空 |
+| 400 | 图片大小不能超过{size}MB | 图片大小超过限制 |
+| 400 | 不支持的图片格式 | 图片格式不支持 |
+| 400 | 图片尺寸超过限制 | 图片宽高超过限制 |
+| 400 | 图片名称不能为空 | 图片名称为空 |
+| 400 | 相册名称不能为空 | 相册名称为空 |
+| 400 | 图片不存在 | 图片ID不存在 |
+| 400 | 图片已被删除 | 图片状态为已删除 |
+| 401 | 当前会话未登录 | 未登录或Token无效 |
+| 403 | 权限不足 | 没有相应的操作权限 |
+| 500 | 图片上传失败 | 图片保存到磁盘失败 |
+| 500 | 缩略图生成失败 | 缩略图生成过程失败 |
+| 500 | 水印添加失败 | 水印添加过程失败 |
+| 500 | 图片压缩失败 | 图片压缩过程失败 |
+| 500 | 图片删除失败 | 物理文件删除失败 |
+
+---
+
+## 安全特性
+
+### 1. 图片验证
+
+- **格式验证**: 验证图片格式和MIME类型
+- **尺寸验证**: 验证图片宽高限制
+- **内容验证**: 验证图片内容安全性
+- **病毒扫描**: 对上传图片进行病毒扫描
+
+### 2. 访问控制
+
+- **权限验证**: 验证用户访问权限
+- **防盗链**: 防止图片被盗链
+- **水印保护**: 添加水印保护版权
+- **下载限制**: 限制图片下载次数
+
+### 3. 存储安全
+
+- **路径安全**: 防止路径遍历攻击
+- **文件重命名**: 重命名避免冲突
+- **备份策略**: 重要图片定期备份
+- **权限设置**: 设置合适的文件权限
+
+---
+
+## 性能优化
+
+### 1. 图片处理优化
+
+- **异步处理**: 图片压缩和水印添加异步处理
+- **批量处理**: 支持批量图片处理
+- **缓存策略**: 缓存处理后的图片
+- **CDN加速**: 使用CDN加速图片访问
+
+### 2. 存储优化
+
+- **分目录存储**: 按相册和日期分目录存储
+- **格式优化**: 自动选择最优图片格式
+- **压缩存储**: 自动压缩降低存储空间
+- **重复检测**: 检测重复图片避免冗余
+
+### 3. 访问优化
+
+- **懒加载**: 图片懒加载技术
+- **响应式图片**: 根据设备提供不同尺寸
+- **WebP支持**: 支持WebP格式提高性能
+- **预加载**: 关键图片预加载
+
+---
+
+## 使用建议
+
+### 1. 图片命名规范
+
+- **唯一性**: 确保图片名唯一
+- **时间戳**: 包含时间戳信息
+- **分类标识**: 包含相册分类信息
+- **版本控制**: 支持图片版本管理
+
+### 2. 相册管理
+
+- **分类管理**: 按用途分类管理图片
+- **权限控制**: 不同相册设置不同权限
+- **容量限制**: 设置相册容量限制
+- **定期清理**: 定期清理无用图片
+
+### 3. 图片优化
+
+- **尺寸适配**: 根据用途选择合适尺寸
+- **格式选择**: 根据内容选择最优格式
+- **质量平衡**: 平衡图片质量和文件大小
+- **缓存策略**: 设置合适的缓存时间
+
+---
+
+## 注意事项
+
+1. **图片安全**: 严格验证上传图片格式和内容
+2. **版权保护**: 添加水印保护图片版权
+3. **存储管理**: 定期清理无效和重复图片
+4. **性能考虑**: 大图片上传时注意性能影响
+5. **备份策略**: 重要图片需要备份
+6. **访问控制**: 合理设置图片访问权限
+7. **格式支持**: 根据需要支持不同图片格式
+8. **缩略图**: 为提高加载速度生成缩略图
+9. **压缩处理**: 适当压缩减少存储空间
+10. **监控告警**: 监控存储空间和访问异常
\ No newline at end of file
diff --git a/api/system/文件管理API.md b/api/system/文件管理API.md
new file mode 100644
index 0000000..cde6262
--- /dev/null
+++ b/api/system/文件管理API.md
@@ -0,0 +1,867 @@
+# 文件管理API
+
+## 概述
+
+文件管理模块提供文件上传、下载、存储管理等功能。支持多种文件类型,提供文件记录管理和批量操作功能。分为通用文件上传接口和系统文件记录管理两部分。
+
+## 权限说明
+
+文件管理接口需要相应的权限才能访问:
+
+| 操作 | 权限码 | 说明 |
+|------|--------|------|
+| 查询文件列表 | `system:file:list` | 查看文件列表权限 |
+| 上传文件 | `system:file:upload` | 上传文件权限 |
+| 下载文件 | `system:file:download` | 下载文件权限 |
+| 删除文件 | `system:file:remove` | 删除文件权限 |
+
+## 文件上传接口
+
+### 1. 上传文件(需认证)
+
+**接口地址**: `POST /coder/file/uploadFile/{fileSize}/{folderName}/{fileParam}`
+
+**接口描述**: 上传文件到服务器(需要登录认证)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:upload`
+
+**请求头**:
+```
+Authorization: Bearer your-token-value
+Content-Type: multipart/form-data
+```
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 | 示例 |
+|--------|------|------|------|------|
+| fileSize | Integer | 是 | 文件大小限制(MB) | 10 |
+| folderName | String | 是 | 存储文件夹名称 | avatar |
+| fileParam | String | 是 | 文件参数名 | file |
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| file | File | 是 | 上传的文件 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "fileName": "avatar.jpg",
+ "originalName": "my-avatar.jpg",
+ "filePath": "/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "fileUrl": "http://localhost:18099/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "fileSize": 1024000,
+ "fileType": "jpg",
+ "uploadTime": "2024-07-05 10:00:00"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**响应参数说明**:
+
+| 参数名 | 类型 | 说明 |
+|--------|------|------|
+| fileName | String | 存储文件名 |
+| originalName | String | 原始文件名 |
+| filePath | String | 文件相对路径 |
+| fileUrl | String | 文件访问URL |
+| fileSize | Long | 文件大小(字节) |
+| fileType | String | 文件类型 |
+| uploadTime | String | 上传时间 |
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/file/uploadFile/10/avatar/file \
+ -H "Authorization: Bearer your-token-value" \
+ -F "file=@avatar.jpg"
+```
+
+**前端使用示例**:
+
+```javascript
+// 使用FormData上传文件
+function uploadFile(file, folder, maxSize) {
+ const formData = new FormData();
+ formData.append('file', file);
+
+ fetch(`/coder/file/uploadFile/${maxSize}/${folder}/file`, {
+ method: 'POST',
+ headers: {
+ 'Authorization': 'Bearer ' + localStorage.getItem('token')
+ },
+ body: formData
+ })
+ .then(response => response.json())
+ .then(data => {
+ if (data.status === 200) {
+ console.log('上传成功:', data.data.fileUrl);
+ } else {
+ console.error('上传失败:', data.msg);
+ }
+ });
+}
+```
+
+---
+
+### 2. 匿名上传文件
+
+**接口地址**: `POST /coder/file/uploadAnyFile/{fileSize}/{folderName}/{fileParam}`
+
+**接口描述**: 匿名上传文件到服务器(无需登录认证)
+
+**是否需要认证**: 否
+
+**权限要求**: 无
+
+**路径参数**: 同上传文件接口
+
+**请求参数**: 同上传文件接口
+
+**响应示例**: 同上传文件接口
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/file/uploadAnyFile/5/temp/file \
+ -F "file=@document.pdf"
+```
+
+**使用场景**:
+- 用户注册时上传头像
+- 访客上传临时文件
+- 公开资源上传
+
+---
+
+## 系统文件记录管理接口
+
+### 1. 分页查询文件列表
+
+**接口地址**: `GET /coder/sysFile/listPage`
+
+**接口描述**: 分页查询系统文件记录列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:list`
+
+**请求头**:
+```
+Authorization: Bearer your-token-value
+```
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 | 示例 |
+|--------|------|------|------|------|
+| pageNo | Integer | 否 | 页码 | 1 |
+| pageSize | Integer | 否 | 每页大小 | 10 |
+| fileName | String | 否 | 文件名称 | avatar.jpg |
+| fileType | String | 否 | 文件类型 | jpg |
+| folderName | String | 否 | 文件夹名称 | avatar |
+| beginTime | String | 否 | 开始时间 | 2024-01-01 |
+| endTime | String | 否 | 结束时间 | 2024-12-31 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "records": [
+ {
+ "fileId": 1,
+ "fileName": "avatar_20240705_001.jpg",
+ "originalName": "my-avatar.jpg",
+ "filePath": "/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "fileUrl": "http://localhost:18099/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "folderName": "avatar",
+ "fileSize": 1024000,
+ "fileType": "jpg",
+ "contentType": "image/jpeg",
+ "fileStatus": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "remark": "用户头像",
+ "createBy": "admin",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-07-05 10:00:00"
+ }
+ ],
+ "total": 1,
+ "size": 10,
+ "current": 1,
+ "pages": 1
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysFile/listPage?pageNo=1&pageSize=10&fileName=avatar" \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 2. 查询所有文件
+
+**接口地址**: `GET /coder/sysFile/list`
+
+**接口描述**: 查询所有系统文件记录(不分页)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:list`
+
+**请求参数**: 同分页查询(除pageNo、pageSize外)
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "fileId": 1,
+ "fileName": "avatar_20240705_001.jpg",
+ "originalName": "my-avatar.jpg",
+ "filePath": "/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "fileUrl": "http://localhost:18099/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "folderName": "avatar",
+ "fileSize": 1024000,
+ "fileType": "jpg",
+ "contentType": "image/jpeg",
+ "fileStatus": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "remark": "用户头像",
+ "createBy": "admin",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-07-05 10:00:00"
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysFile/list?fileType=jpg" \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 3. 根据ID查询文件
+
+**接口地址**: `GET /coder/sysFile/getById/{id}`
+
+**接口描述**: 根据文件ID查询文件详细信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:list`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 文件ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "fileId": 1,
+ "fileName": "avatar_20240705_001.jpg",
+ "originalName": "my-avatar.jpg",
+ "filePath": "/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "fileUrl": "http://localhost:18099/upload/avatar/2024/07/05/avatar_20240705_001.jpg",
+ "folderName": "avatar",
+ "fileSize": 1024000,
+ "fileType": "jpg",
+ "contentType": "image/jpeg",
+ "fileStatus": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "remark": "用户头像",
+ "createBy": "admin",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-07-05 10:00:00"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysFile/getById/1 \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 4. 新增文件记录
+
+**接口地址**: `POST /coder/sysFile/add`
+
+**接口描述**: 新增系统文件记录
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:upload`
+
+**请求参数**:
+
+```json
+{
+ "fileName": "document_20240705_001.pdf",
+ "originalName": "重要文档.pdf",
+ "filePath": "/upload/documents/2024/07/05/document_20240705_001.pdf",
+ "fileUrl": "http://localhost:18099/upload/documents/2024/07/05/document_20240705_001.pdf",
+ "folderName": "documents",
+ "fileSize": 2048000,
+ "fileType": "pdf",
+ "contentType": "application/pdf",
+ "fileStatus": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "remark": "重要文档"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| fileName | String | 是 | 存储文件名 | 不能为空 |
+| originalName | String | 是 | 原始文件名 | 不能为空 |
+| filePath | String | 是 | 文件相对路径 | 不能为空 |
+| fileUrl | String | 是 | 文件访问URL | 不能为空 |
+| folderName | String | 是 | 文件夹名称 | 不能为空 |
+| fileSize | Long | 是 | 文件大小 | 必须大于0 |
+| fileType | String | 是 | 文件类型 | 不能为空 |
+| contentType | String | 是 | MIME类型 | 不能为空 |
+| fileStatus | String | 是 | 文件状态 | 0-正常 1-删除 |
+| uploadUserId | Long | 否 | 上传用户ID | 有效的用户ID |
+| uploadUserName | String | 否 | 上传用户名 | 可为空 |
+| remark | String | 否 | 备注信息 | 最长200字符 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "新增成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysFile/add \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "fileName": "document_20240705_001.pdf",
+ "originalName": "重要文档.pdf",
+ "filePath": "/upload/documents/2024/07/05/document_20240705_001.pdf",
+ "fileUrl": "http://localhost:18099/upload/documents/2024/07/05/document_20240705_001.pdf",
+ "folderName": "documents",
+ "fileSize": 2048000,
+ "fileType": "pdf",
+ "contentType": "application/pdf",
+ "fileStatus": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "remark": "重要文档"
+ }'
+```
+
+---
+
+### 5. 修改文件信息
+
+**接口地址**: `POST /coder/sysFile/update`
+
+**接口描述**: 修改系统文件记录信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:edit`
+
+**请求参数**:
+
+```json
+{
+ "fileId": 1,
+ "fileName": "document_20240705_001.pdf",
+ "originalName": "重要文档.pdf",
+ "filePath": "/upload/documents/2024/07/05/document_20240705_001.pdf",
+ "fileUrl": "http://localhost:18099/upload/documents/2024/07/05/document_20240705_001.pdf",
+ "folderName": "documents",
+ "fileSize": 2048000,
+ "fileType": "pdf",
+ "contentType": "application/pdf",
+ "fileStatus": "0",
+ "uploadUserId": 1,
+ "uploadUserName": "admin",
+ "remark": "重要文档(已更新)"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| fileId | Long | 是 | 文件ID | 必须是有效的文件ID |
+| 其他参数 | - | - | 同新增文件记录 | - |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysFile/update \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "fileId": 1,
+ "fileName": "document_20240705_001.pdf",
+ "originalName": "重要文档.pdf",
+ "remark": "重要文档(已更新)"
+ }'
+```
+
+---
+
+### 6. 删除文件
+
+**接口地址**: `POST /coder/sysFile/deleteById/{id}`
+
+**接口描述**: 根据ID删除文件记录和物理文件
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:remove`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 文件ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysFile/deleteById/1 \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 7. 批量删除文件
+
+**接口地址**: `POST /coder/sysFile/batchDelete`
+
+**接口描述**: 批量删除文件记录和物理文件
+
+**是否需要认证**: 是
+
+**权限要求**: `system:file:remove`
+
+**请求参数**:
+
+```json
+{
+ "ids": [1, 2, 3]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| ids | Long[] | 是 | 文件ID数组 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysFile/batchDelete \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "ids": [1, 2, 3]
+ }'
+```
+
+---
+
+## 文件字段说明
+
+### 基础字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| fileId | Long | 文件ID | 1 |
+| fileName | String | 存储文件名 | avatar_20240705_001.jpg |
+| originalName | String | 原始文件名 | my-avatar.jpg |
+| filePath | String | 文件相对路径 | /upload/avatar/2024/07/05/avatar_20240705_001.jpg |
+| fileUrl | String | 文件访问URL | http://localhost:18099/upload/avatar/2024/07/05/avatar_20240705_001.jpg |
+
+### 分类字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| folderName | String | 文件夹名称 | avatar |
+| fileType | String | 文件扩展名 | jpg |
+| contentType | String | MIME类型 | image/jpeg |
+| fileSize | Long | 文件大小(字节) | 1024000 |
+
+### 状态字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| fileStatus | String | 文件状态 | 0-正常 1-删除 |
+| uploadUserId | Long | 上传用户ID | 1 |
+| uploadUserName | String | 上传用户名 | admin |
+| remark | String | 备注信息 | 用户头像 |
+
+---
+
+## 数据字典
+
+### 文件状态 (fileStatus)
+
+| 值 | 说明 |
+|----|------|
+| 0 | 正常 |
+| 1 | 已删除 |
+
+### 常见文件类型
+
+| 文件类型 | 扩展名 | MIME类型 | 说明 |
+|----------|--------|----------|------|
+| 图片 | jpg, jpeg, png, gif, bmp | image/* | 图片文件 |
+| 文档 | pdf, doc, docx, xls, xlsx, ppt, pptx | application/* | 办公文档 |
+| 音频 | mp3, wav, flac, aac | audio/* | 音频文件 |
+| 视频 | mp4, avi, mov, wmv | video/* | 视频文件 |
+| 压缩包 | zip, rar, 7z, tar, gz | application/* | 压缩文件 |
+| 文本 | txt, md, csv | text/* | 文本文件 |
+
+---
+
+## 文件配置
+
+### 1. 文件存储配置
+
+```yaml
+# application-dev.yml
+coder:
+ # 文件存储路径
+ filePath: /data/upload/
+
+ # 文件访问域名
+ domain: http://localhost:18099
+
+ # 文件大小限制(MB)
+ maxFileSize: 100
+
+ # 允许的文件类型
+ allowedTypes:
+ - jpg
+ - jpeg
+ - png
+ - gif
+ - pdf
+ - doc
+ - docx
+ - xls
+ - xlsx
+
+ # 禁止的文件类型
+ forbiddenTypes:
+ - exe
+ - bat
+ - sh
+ - jsp
+ - php
+```
+
+### 2. 文件夹分类
+
+```java
+// 文件夹类型枚举
+public enum FolderType {
+ AVATAR("avatar", "用户头像"),
+ DOCUMENT("document", "文档文件"),
+ IMAGE("image", "图片文件"),
+ TEMP("temp", "临时文件"),
+ EXPORT("export", "导出文件");
+
+ private final String code;
+ private final String name;
+}
+```
+
+### 3. 文件上传工具类
+
+```java
+@Component
+public class FileUploadUtil {
+
+ /**
+ * 上传文件
+ */
+ public FileUploadResult uploadFile(MultipartFile file, String folder, int maxSize) {
+ // 1. 文件校验
+ validateFile(file, maxSize);
+
+ // 2. 生成文件名
+ String fileName = generateFileName(file.getOriginalFilename());
+
+ // 3. 创建目录
+ String datePath = DateUtil.format(new Date(), "yyyy/MM/dd");
+ String dirPath = filePath + folder + "/" + datePath + "/";
+ FileUtil.createDirs(dirPath);
+
+ // 4. 保存文件
+ String filePath = dirPath + fileName;
+ file.transferTo(new File(filePath));
+
+ // 5. 生成访问URL
+ String fileUrl = domain + "/upload/" + folder + "/" + datePath + "/" + fileName;
+
+ // 6. 返回结果
+ return FileUploadResult.builder()
+ .fileName(fileName)
+ .originalName(file.getOriginalFilename())
+ .filePath(filePath)
+ .fileUrl(fileUrl)
+ .fileSize(file.getSize())
+ .fileType(FileUtil.getExtension(fileName))
+ .uploadTime(new Date())
+ .build();
+ }
+
+ /**
+ * 文件校验
+ */
+ private void validateFile(MultipartFile file, int maxSize) {
+ if (file.isEmpty()) {
+ throw new BusinessException("文件不能为空");
+ }
+
+ if (file.getSize() > maxSize * 1024 * 1024) {
+ throw new BusinessException("文件大小不能超过" + maxSize + "MB");
+ }
+
+ String extension = FileUtil.getExtension(file.getOriginalFilename());
+ if (!allowedTypes.contains(extension.toLowerCase())) {
+ throw new BusinessException("不支持的文件类型:" + extension);
+ }
+
+ if (forbiddenTypes.contains(extension.toLowerCase())) {
+ throw new BusinessException("禁止上传的文件类型:" + extension);
+ }
+ }
+
+ /**
+ * 生成文件名
+ */
+ private String generateFileName(String originalName) {
+ String extension = FileUtil.getExtension(originalName);
+ String baseName = FileUtil.getBaseName(originalName);
+ String timestamp = DateUtil.format(new Date(), "yyyyMMdd_HHmmss");
+ String random = RandomUtil.randomString(3);
+ return baseName + "_" + timestamp + "_" + random + "." + extension;
+ }
+}
+```
+
+---
+
+## 错误码说明
+
+| 错误码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 文件不能为空 | 上传文件为空 |
+| 400 | 文件大小不能超过{size}MB | 文件大小超过限制 |
+| 400 | 不支持的文件类型 | 文件类型不在允许列表中 |
+| 400 | 禁止上传的文件类型 | 文件类型在禁止列表中 |
+| 400 | 文件名不能为空 | 文件名为空 |
+| 400 | 文件路径不能为空 | 文件路径为空 |
+| 400 | 文件不存在 | 文件ID不存在 |
+| 400 | 文件已被删除 | 文件状态为已删除 |
+| 401 | 当前会话未登录 | 未登录或Token无效 |
+| 403 | 权限不足 | 没有相应的操作权限 |
+| 500 | 文件上传失败 | 文件保存到磁盘失败 |
+| 500 | 文件删除失败 | 物理文件删除失败 |
+| 500 | 磁盘空间不足 | 服务器磁盘空间不足 |
+
+---
+
+## 安全特性
+
+### 1. 文件类型验证
+
+- **扩展名校验**: 验证文件扩展名
+- **MIME类型校验**: 验证文件MIME类型
+- **文件内容校验**: 验证文件真实类型
+- **黑名单过滤**: 禁止上传危险文件类型
+
+### 2. 文件大小限制
+
+- **单文件大小限制**: 限制单个文件大小
+- **总文件大小限制**: 限制用户总文件大小
+- **磁盘空间检查**: 检查服务器磁盘空间
+
+### 3. 访问控制
+
+- **权限验证**: 验证用户上传和访问权限
+- **路径防护**: 防止路径遍历攻击
+- **防盗链**: 防止文件被盗链
+
+### 4. 病毒扫描
+
+- **文件扫描**: 上传文件病毒扫描
+- **隔离机制**: 可疑文件隔离处理
+- **定期扫描**: 定期扫描存储文件
+
+---
+
+## 性能优化
+
+### 1. 文件存储
+
+- **分目录存储**: 按日期和类型分目录存储
+- **CDN加速**: 使用CDN加速文件访问
+- **压缩存储**: 自动压缩图片文件
+- **缓存策略**: 设置合适的缓存策略
+
+### 2. 上传优化
+
+- **分片上传**: 大文件分片上传
+- **断点续传**: 支持断点续传功能
+- **并发上传**: 支持多文件并发上传
+- **进度显示**: 实时显示上传进度
+
+### 3. 存储优化
+
+- **重复文件检测**: 检测并避免重复文件
+- **文件压缩**: 自动压缩文件
+- **定期清理**: 清理临时文件和无效文件
+- **存储监控**: 监控存储空间使用情况
+
+---
+
+## 使用建议
+
+### 1. 文件命名规范
+
+- **唯一性**: 确保文件名唯一
+- **时间戳**: 包含时间戳信息
+- **随机性**: 添加随机字符
+- **可读性**: 保持一定的可读性
+
+### 2. 文件分类管理
+
+- **按类型分类**: 不同类型文件存储在不同目录
+- **按用户分类**: 不同用户文件分开存储
+- **按时间分类**: 按年月日创建目录结构
+- **按业务分类**: 按业务模块分类存储
+
+### 3. 安全防护
+
+- **输入验证**: 严格验证上传文件
+- **权限控制**: 细粒度的权限控制
+- **日志记录**: 记录所有文件操作
+- **备份策略**: 重要文件定期备份
+
+### 4. 监控告警
+
+- **存储监控**: 监控存储空间使用
+- **性能监控**: 监控上传下载性能
+- **安全监控**: 监控异常文件操作
+- **容量告警**: 存储空间不足告警
+
+---
+
+## 注意事项
+
+1. **文件安全**: 严格验证上传文件类型和内容
+2. **存储路径**: 确保文件存储路径安全
+3. **权限控制**: 合理设置文件访问权限
+4. **磁盘空间**: 定期清理无效文件,避免磁盘满
+5. **备份策略**: 重要文件需要备份
+6. **访问控制**: 防止文件被恶意访问
+7. **性能考虑**: 大文件上传时注意性能影响
+8. **并发控制**: 处理并发上传时的文件冲突
+9. **错误处理**: 完善的错误处理和回滚机制
+10. **日志审计**: 记录所有文件操作日志
\ No newline at end of file
diff --git a/api/system/登录日志API.md b/api/system/登录日志API.md
new file mode 100644
index 0000000..8f5cc29
--- /dev/null
+++ b/api/system/登录日志API.md
@@ -0,0 +1,1017 @@
+# 登录日志API
+
+## 概述
+
+登录日志模块用于记录和管理系统用户的登录行为,包括登录成功、登录失败、退出登录等操作的详细记录。提供日志查询、统计分析和安全监控功能,是系统安全审计的重要组成部分。
+
+## 权限说明
+
+登录日志接口需要相应的权限才能访问:
+
+| 操作 | 权限码 | 说明 |
+|------|--------|------|
+| 查询登录日志 | `system:loginlog:list` | 查看登录日志权限 |
+| 删除登录日志 | `system:loginlog:remove` | 删除登录日志权限 |
+| 导出登录日志 | `system:loginlog:export` | 导出登录日志权限 |
+
+## 接口列表
+
+### 1. 分页查询登录日志
+
+**接口地址**: `GET /coder/sysLoginLog/listPage`
+
+**接口描述**: 分页查询系统登录日志列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:loginlog:list`
+
+**请求头**:
+```
+Authorization: Bearer your-token-value
+```
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 | 示例 |
+|--------|------|------|------|------|
+| pageNo | Integer | 否 | 页码 | 1 |
+| pageSize | Integer | 否 | 每页大小 | 10 |
+| loginName | String | 否 | 登录账号 | admin |
+| loginStatus | String | 否 | 登录状态 | 0 |
+| clientType | String | 否 | 客户端类型 | WEB |
+| loginIp | String | 否 | 登录IP | 127.0.0.1 |
+| loginAddress | String | 否 | 登录地址 | 本地登录 |
+| browser | String | 否 | 浏览器类型 | Chrome |
+| os | String | 否 | 操作系统 | Windows 10 |
+| beginTime | String | 否 | 开始时间 | 2024-01-01 |
+| endTime | String | 否 | 结束时间 | 2024-12-31 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "records": [
+ {
+ "infoId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userId": 1,
+ "loginStatus": "0",
+ "clientType": "WEB",
+ "deviceName": "Windows PC",
+ "loginIp": "127.0.0.1",
+ "loginAddress": "本地登录",
+ "browser": "Chrome 120.0.0.0",
+ "os": "Windows 10",
+ "userAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
+ "loginTime": "2024-07-05 10:00:00",
+ "logoutTime": "2024-07-05 11:00:00",
+ "sessionDuration": 3600,
+ "failureReason": "",
+ "remark": "登录成功",
+ "createBy": "system",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "system",
+ "updateTime": "2024-07-05 11:00:00"
+ }
+ ],
+ "total": 1,
+ "size": 10,
+ "current": 1,
+ "pages": 1
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysLoginLog/listPage?pageNo=1&pageSize=10&loginName=admin&loginStatus=0" \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 2. 查询所有登录日志
+
+**接口地址**: `GET /coder/sysLoginLog/list`
+
+**接口描述**: 查询所有系统登录日志(不分页)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:loginlog:list`
+
+**请求参数**: 同分页查询(除pageNo、pageSize外)
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "infoId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userId": 1,
+ "loginStatus": "0",
+ "clientType": "WEB",
+ "deviceName": "Windows PC",
+ "loginIp": "127.0.0.1",
+ "loginAddress": "本地登录",
+ "browser": "Chrome 120.0.0.0",
+ "os": "Windows 10",
+ "userAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
+ "loginTime": "2024-07-05 10:00:00",
+ "logoutTime": "2024-07-05 11:00:00",
+ "sessionDuration": 3600,
+ "failureReason": "",
+ "remark": "登录成功",
+ "createBy": "system",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "system",
+ "updateTime": "2024-07-05 11:00:00"
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysLoginLog/list?loginStatus=1&beginTime=2024-07-01" \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 3. 根据ID查询登录日志
+
+**接口地址**: `GET /coder/sysLoginLog/getById/{id}`
+
+**接口描述**: 根据日志ID查询登录日志详细信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:loginlog:list`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 日志ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "infoId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userId": 1,
+ "loginStatus": "0",
+ "clientType": "WEB",
+ "deviceName": "Windows PC",
+ "loginIp": "127.0.0.1",
+ "loginAddress": "本地登录",
+ "browser": "Chrome 120.0.0.0",
+ "os": "Windows 10",
+ "userAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
+ "loginTime": "2024-07-05 10:00:00",
+ "logoutTime": "2024-07-05 11:00:00",
+ "sessionDuration": 3600,
+ "failureReason": "",
+ "remark": "登录成功",
+ "createBy": "system",
+ "createTime": "2024-07-05 10:00:00",
+ "updateBy": "system",
+ "updateTime": "2024-07-05 11:00:00"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysLoginLog/getById/1 \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 4. 新增登录日志
+
+**接口地址**: `POST /coder/sysLoginLog/add`
+
+**接口描述**: 新增系统登录日志记录
+
+**是否需要认证**: 是
+
+**权限要求**: `system:loginlog:add`
+
+**请求参数**:
+
+```json
+{
+ "loginName": "admin",
+ "userName": "管理员",
+ "userId": 1,
+ "loginStatus": "0",
+ "clientType": "WEB",
+ "deviceName": "Windows PC",
+ "loginIp": "127.0.0.1",
+ "loginAddress": "本地登录",
+ "browser": "Chrome 120.0.0.0",
+ "os": "Windows 10",
+ "userAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
+ "loginTime": "2024-07-05 10:00:00",
+ "failureReason": "",
+ "remark": "登录成功"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| loginName | String | 是 | 登录账号 | 不能为空 |
+| userName | String | 否 | 用户姓名 | 可为空 |
+| userId | Long | 否 | 用户ID | 有效的用户ID |
+| loginStatus | String | 是 | 登录状态 | 0-成功 1-失败 |
+| clientType | String | 是 | 客户端类型 | WEB, MOBILE, API |
+| deviceName | String | 否 | 设备名称 | 可为空 |
+| loginIp | String | 是 | 登录IP | 有效的IP地址 |
+| loginAddress | String | 否 | 登录地址 | 可为空 |
+| browser | String | 否 | 浏览器信息 | 可为空 |
+| os | String | 否 | 操作系统 | 可为空 |
+| userAgent | String | 否 | 用户代理 | 可为空 |
+| loginTime | String | 是 | 登录时间 | 时间格式 |
+| failureReason | String | 否 | 失败原因 | 登录失败时必填 |
+| remark | String | 否 | 备注信息 | 最长500字符 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "新增成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginLog/add \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "loginName": "admin",
+ "userName": "管理员",
+ "userId": 1,
+ "loginStatus": "0",
+ "clientType": "WEB",
+ "loginIp": "127.0.0.1",
+ "loginTime": "2024-07-05 10:00:00",
+ "remark": "登录成功"
+ }'
+```
+
+---
+
+### 5. 修改登录日志
+
+**接口地址**: `POST /coder/sysLoginLog/update`
+
+**接口描述**: 修改系统登录日志信息(通常用于更新退出时间)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:loginlog:edit`
+
+**请求参数**:
+
+```json
+{
+ "infoId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userId": 1,
+ "loginStatus": "0",
+ "clientType": "WEB",
+ "deviceName": "Windows PC",
+ "loginIp": "127.0.0.1",
+ "loginAddress": "本地登录",
+ "browser": "Chrome 120.0.0.0",
+ "os": "Windows 10",
+ "userAgent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
+ "loginTime": "2024-07-05 10:00:00",
+ "logoutTime": "2024-07-05 11:00:00",
+ "sessionDuration": 3600,
+ "failureReason": "",
+ "remark": "正常退出"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| infoId | Long | 是 | 日志ID | 必须是有效的日志ID |
+| logoutTime | String | 否 | 退出时间 | 时间格式 |
+| sessionDuration | Integer | 否 | 会话时长(秒) | 大于等于0 |
+| 其他参数 | - | - | 同新增日志 | - |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginLog/update \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "infoId": 1,
+ "logoutTime": "2024-07-05 11:00:00",
+ "sessionDuration": 3600,
+ "remark": "正常退出"
+ }'
+```
+
+---
+
+### 6. 删除登录日志
+
+**接口地址**: `POST /coder/sysLoginLog/deleteById/{id}`
+
+**接口描述**: 根据ID删除登录日志
+
+**是否需要认证**: 是
+
+**权限要求**: `system:loginlog:remove`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 日志ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginLog/deleteById/1 \
+ -H "Authorization: Bearer your-token-value"
+```
+
+---
+
+### 7. 批量删除登录日志
+
+**接口地址**: `POST /coder/sysLoginLog/batchDelete`
+
+**接口描述**: 批量删除登录日志
+
+**是否需要认证**: 是
+
+**权限要求**: `system:loginlog:remove`
+
+**请求参数**:
+
+```json
+{
+ "ids": [1, 2, 3]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| ids | Long[] | 是 | 日志ID数组 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginLog/batchDelete \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer your-token-value" \
+ -d '{
+ "ids": [1, 2, 3]
+ }'
+```
+
+---
+
+## 登录日志字段说明
+
+### 基础字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| infoId | Long | 日志ID | 1 |
+| loginName | String | 登录账号 | admin |
+| userName | String | 用户姓名 | 管理员 |
+| userId | Long | 用户ID | 1 |
+
+### 状态字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| loginStatus | String | 登录状态 | 0-成功 1-失败 |
+| clientType | String | 客户端类型 | WEB, MOBILE, API |
+| deviceName | String | 设备名称 | Windows PC |
+
+### 网络字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| loginIp | String | 登录IP地址 | 127.0.0.1 |
+| loginAddress | String | 登录地理位置 | 本地登录 |
+| userAgent | String | 用户代理字符串 | Mozilla/5.0... |
+
+### 环境字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| browser | String | 浏览器信息 | Chrome 120.0.0.0 |
+| os | String | 操作系统 | Windows 10 |
+
+### 时间字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| loginTime | String | 登录时间 | 2024-07-05 10:00:00 |
+| logoutTime | String | 退出时间 | 2024-07-05 11:00:00 |
+| sessionDuration | Integer | 会话时长(秒) | 3600 |
+
+### 其他字段
+
+| 字段名 | 类型 | 说明 | 示例 |
+|--------|------|------|------|
+| failureReason | String | 失败原因 | 密码错误 |
+| remark | String | 备注信息 | 登录成功 |
+
+---
+
+## 数据字典
+
+### 登录状态 (loginStatus)
+
+| 值 | 说明 |
+|----|------|
+| 0 | 登录成功 |
+| 1 | 登录失败 |
+
+### 客户端类型 (clientType)
+
+| 值 | 说明 |
+|----|------|
+| WEB | 网页端 |
+| MOBILE | 移动端 |
+| API | 接口调用 |
+| DESKTOP | 桌面应用 |
+| WECHAT | 微信小程序 |
+
+### 常见失败原因
+
+| 失败原因 | 说明 |
+|----------|------|
+| 用户不存在 | 登录账号不存在 |
+| 密码错误 | 登录密码不正确 |
+| 验证码错误 | 验证码输入错误 |
+| 账号被禁用 | 用户账号被禁用 |
+| 账号被锁定 | 用户账号被锁定 |
+| IP被限制 | 登录IP被限制 |
+| 设备被限制 | 登录设备被限制 |
+| 会话过期 | 用户会话已过期 |
+
+---
+
+## 日志记录机制
+
+### 1. 自动记录
+
+```java
+@Component
+@Slf4j
+public class LoginLogService {
+
+ /**
+ * 记录登录成功日志
+ */
+ public void recordLoginSuccess(String loginName, HttpServletRequest request) {
+ try {
+ SysLoginLog loginLog = new SysLoginLog();
+
+ // 基础信息
+ loginLog.setLoginName(loginName);
+ loginLog.setUserId(getCurrentUserId(loginName));
+ loginLog.setUserName(getCurrentUserName(loginName));
+ loginLog.setLoginStatus("0");
+ loginLog.setLoginTime(LocalDateTime.now());
+
+ // 网络信息
+ loginLog.setLoginIp(getClientIP(request));
+ loginLog.setLoginAddress(getAddressByIP(loginLog.getLoginIp()));
+
+ // 客户端信息
+ loginLog.setClientType(getClientType(request));
+ loginLog.setUserAgent(request.getHeader("User-Agent"));
+ loginLog.setBrowser(getBrowserInfo(request));
+ loginLog.setOs(getOSInfo(request));
+ loginLog.setDeviceName(getDeviceName(request));
+
+ // 其他信息
+ loginLog.setRemark("登录成功");
+ loginLog.setCreateBy("system");
+ loginLog.setCreateTime(LocalDateTime.now());
+
+ // 保存日志
+ sysLoginLogMapper.insert(loginLog);
+
+ } catch (Exception e) {
+ log.error("记录登录成功日志失败", e);
+ }
+ }
+
+ /**
+ * 记录登录失败日志
+ */
+ public void recordLoginFailure(String loginName, String failureReason, HttpServletRequest request) {
+ try {
+ SysLoginLog loginLog = new SysLoginLog();
+
+ // 基础信息
+ loginLog.setLoginName(loginName);
+ loginLog.setLoginStatus("1");
+ loginLog.setLoginTime(LocalDateTime.now());
+ loginLog.setFailureReason(failureReason);
+
+ // 网络信息
+ loginLog.setLoginIp(getClientIP(request));
+ loginLog.setLoginAddress(getAddressByIP(loginLog.getLoginIp()));
+
+ // 客户端信息
+ loginLog.setClientType(getClientType(request));
+ loginLog.setUserAgent(request.getHeader("User-Agent"));
+ loginLog.setBrowser(getBrowserInfo(request));
+ loginLog.setOs(getOSInfo(request));
+ loginLog.setDeviceName(getDeviceName(request));
+
+ // 其他信息
+ loginLog.setRemark("登录失败:" + failureReason);
+ loginLog.setCreateBy("system");
+ loginLog.setCreateTime(LocalDateTime.now());
+
+ // 保存日志
+ sysLoginLogMapper.insert(loginLog);
+
+ } catch (Exception e) {
+ log.error("记录登录失败日志失败", e);
+ }
+ }
+
+ /**
+ * 记录退出登录日志
+ */
+ public void recordLogout(String loginName, LocalDateTime loginTime) {
+ try {
+ // 查找对应的登录日志
+ SysLoginLog loginLog = sysLoginLogMapper.selectOne(
+ new LambdaQueryWrapper()
+ .eq(SysLoginLog::getLoginName, loginName)
+ .eq(SysLoginLog::getLoginTime, loginTime)
+ .eq(SysLoginLog::getLoginStatus, "0")
+ .isNull(SysLoginLog::getLogoutTime)
+ .last("ORDER BY create_time DESC LIMIT 1")
+ );
+
+ if (loginLog != null) {
+ LocalDateTime logoutTime = LocalDateTime.now();
+ long sessionDuration = Duration.between(loginLog.getLoginTime(), logoutTime).getSeconds();
+
+ loginLog.setLogoutTime(logoutTime);
+ loginLog.setSessionDuration((int) sessionDuration);
+ loginLog.setRemark("正常退出");
+ loginLog.setUpdateBy("system");
+ loginLog.setUpdateTime(LocalDateTime.now());
+
+ sysLoginLogMapper.updateById(loginLog);
+ }
+
+ } catch (Exception e) {
+ log.error("记录退出登录日志失败", e);
+ }
+ }
+
+ /**
+ * 获取客户端IP
+ */
+ private String getClientIP(HttpServletRequest request) {
+ String ip = request.getHeader("X-Forwarded-For");
+ if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+ ip = request.getHeader("Proxy-Client-IP");
+ }
+ if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+ ip = request.getHeader("WL-Proxy-Client-IP");
+ }
+ if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+ ip = request.getHeader("HTTP_CLIENT_IP");
+ }
+ if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+ ip = request.getHeader("HTTP_X_FORWARDED_FOR");
+ }
+ if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+ ip = request.getRemoteAddr();
+ }
+ return ip;
+ }
+
+ /**
+ * 解析浏览器信息
+ */
+ private String getBrowserInfo(HttpServletRequest request) {
+ String userAgent = request.getHeader("User-Agent");
+ if (userAgent == null) {
+ return "Unknown";
+ }
+
+ if (userAgent.contains("Chrome")) {
+ return "Chrome";
+ } else if (userAgent.contains("Firefox")) {
+ return "Firefox";
+ } else if (userAgent.contains("Safari")) {
+ return "Safari";
+ } else if (userAgent.contains("Edge")) {
+ return "Edge";
+ } else if (userAgent.contains("IE")) {
+ return "Internet Explorer";
+ } else {
+ return "Other";
+ }
+ }
+
+ /**
+ * 解析操作系统信息
+ */
+ private String getOSInfo(HttpServletRequest request) {
+ String userAgent = request.getHeader("User-Agent");
+ if (userAgent == null) {
+ return "Unknown";
+ }
+
+ if (userAgent.contains("Windows NT 10.0")) {
+ return "Windows 10";
+ } else if (userAgent.contains("Windows NT 6.3")) {
+ return "Windows 8.1";
+ } else if (userAgent.contains("Windows NT 6.2")) {
+ return "Windows 8";
+ } else if (userAgent.contains("Windows NT 6.1")) {
+ return "Windows 7";
+ } else if (userAgent.contains("Mac OS X")) {
+ return "Mac OS";
+ } else if (userAgent.contains("Linux")) {
+ return "Linux";
+ } else if (userAgent.contains("Android")) {
+ return "Android";
+ } else if (userAgent.contains("iPhone") || userAgent.contains("iPad")) {
+ return "iOS";
+ } else {
+ return "Other";
+ }
+ }
+}
+```
+
+### 2. 异步处理
+
+```java
+@Service
+public class AsyncLoginLogService {
+
+ @Async("taskExecutor")
+ public void recordLoginLogAsync(SysLoginLog loginLog) {
+ try {
+ // 异步记录登录日志
+ sysLoginLogMapper.insert(loginLog);
+
+ // 更新用户登录信息
+ updateUserLoginInfo(loginLog);
+
+ // 检查异常登录
+ checkAbnormalLogin(loginLog);
+
+ } catch (Exception e) {
+ log.error("异步记录登录日志失败", e);
+ }
+ }
+
+ /**
+ * 检查异常登录
+ */
+ private void checkAbnormalLogin(SysLoginLog loginLog) {
+ // 检查IP异常
+ if (isAbnormalIP(loginLog.getLoginIp(), loginLog.getLoginName())) {
+ sendSecurityAlert("检测到异常IP登录", loginLog);
+ }
+
+ // 检查设备异常
+ if (isAbnormalDevice(loginLog.getUserAgent(), loginLog.getLoginName())) {
+ sendSecurityAlert("检测到异常设备登录", loginLog);
+ }
+
+ // 检查时间异常
+ if (isAbnormalTime(loginLog.getLoginTime(), loginLog.getLoginName())) {
+ sendSecurityAlert("检测到异常时间登录", loginLog);
+ }
+ }
+}
+```
+
+---
+
+## 统计分析功能
+
+### 1. 登录统计
+
+```java
+/**
+ * 登录统计服务
+ */
+@Service
+public class LoginStatisticsService {
+
+ /**
+ * 获取登录成功率统计
+ */
+ public LoginSuccessRateVO getLoginSuccessRate(String beginTime, String endTime) {
+ // 查询总登录次数
+ long totalCount = sysLoginLogMapper.selectCount(
+ new LambdaQueryWrapper()
+ .between(SysLoginLog::getLoginTime, beginTime, endTime)
+ );
+
+ // 查询成功登录次数
+ long successCount = sysLoginLogMapper.selectCount(
+ new LambdaQueryWrapper()
+ .eq(SysLoginLog::getLoginStatus, "0")
+ .between(SysLoginLog::getLoginTime, beginTime, endTime)
+ );
+
+ // 计算成功率
+ double successRate = totalCount > 0 ? (double) successCount / totalCount * 100 : 0;
+
+ return LoginSuccessRateVO.builder()
+ .totalCount(totalCount)
+ .successCount(successCount)
+ .failureCount(totalCount - successCount)
+ .successRate(successRate)
+ .build();
+ }
+
+ /**
+ * 获取每日登录统计
+ */
+ public List getDailyLoginStat(String beginTime, String endTime) {
+ return sysLoginLogMapper.selectDailyLoginStat(beginTime, endTime);
+ }
+
+ /**
+ * 获取客户端类型统计
+ */
+ public List getClientTypeStat(String beginTime, String endTime) {
+ return sysLoginLogMapper.selectClientTypeStat(beginTime, endTime);
+ }
+}
+```
+
+### 2. 安全监控
+
+```java
+/**
+ * 安全监控服务
+ */
+@Service
+public class SecurityMonitorService {
+
+ /**
+ * 检测暴力破解
+ */
+ public void detectBruteForce() {
+ // 查询5分钟内失败次数超过5次的IP
+ List suspiciousIPs = sysLoginLogMapper.selectSuspiciousIPs();
+
+ for (String ip : suspiciousIPs) {
+ // 加入黑名单
+ addToBlacklist(ip);
+
+ // 发送告警
+ sendSecurityAlert("检测到暴力破解攻击", ip);
+ }
+ }
+
+ /**
+ * 检测异地登录
+ */
+ public void detectRemoteLogin() {
+ // 查询用户最近登录地址
+ List recentLogins = sysLoginLogMapper.selectRecentLogins();
+
+ for (SysLoginLog loginLog : recentLogins) {
+ String lastLoginAddress = getLastLoginAddress(loginLog.getLoginName());
+
+ if (!loginLog.getLoginAddress().equals(lastLoginAddress)) {
+ // 发送异地登录通知
+ sendRemoteLoginNotification(loginLog);
+ }
+ }
+ }
+}
+```
+
+---
+
+## 错误码说明
+
+| 错误码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 登录账号不能为空 | 登录账号为空 |
+| 400 | 登录状态不能为空 | 登录状态为空 |
+| 400 | 登录时间不能为空 | 登录时间为空 |
+| 400 | 登录IP不能为空 | 登录IP为空 |
+| 400 | 日志不存在 | 日志ID不存在 |
+| 400 | 时间格式错误 | 时间格式不正确 |
+| 401 | 当前会话未登录 | 未登录或Token无效 |
+| 403 | 权限不足 | 没有相应的操作权限 |
+| 500 | 日志记录失败 | 日志保存失败 |
+| 500 | 系统异常 | 服务器内部错误 |
+
+---
+
+## 日志清理策略
+
+### 1. 定时清理
+
+```java
+@Component
+public class LoginLogCleanupTask {
+
+ /**
+ * 每天凌晨2点清理30天前的登录日志
+ */
+ @Scheduled(cron = "0 0 2 * * ?")
+ public void cleanupOldLogs() {
+ try {
+ LocalDateTime cutoffTime = LocalDateTime.now().minusDays(30);
+
+ int deletedCount = sysLoginLogMapper.delete(
+ new LambdaQueryWrapper()
+ .lt(SysLoginLog::getCreateTime, cutoffTime)
+ );
+
+ log.info("清理登录日志完成,删除{}条记录", deletedCount);
+
+ } catch (Exception e) {
+ log.error("清理登录日志失败", e);
+ }
+ }
+
+ /**
+ * 归档登录日志
+ */
+ @Scheduled(cron = "0 0 1 1 * ?") // 每月1号凌晨1点执行
+ public void archiveLogs() {
+ try {
+ LocalDateTime lastMonth = LocalDateTime.now().minusMonths(1);
+
+ // 归档上个月的日志到历史表
+ sysLoginLogMapper.archiveLogsToHistory(lastMonth);
+
+ log.info("归档登录日志完成");
+
+ } catch (Exception e) {
+ log.error("归档登录日志失败", e);
+ }
+ }
+}
+```
+
+### 2. 日志配置
+
+```yaml
+# application.yml
+logging:
+ login-log:
+ # 是否启用登录日志
+ enabled: true
+ # 日志保留天数
+ retention-days: 90
+ # 是否记录成功登录
+ log-success: true
+ # 是否记录失败登录
+ log-failure: true
+ # 是否异步记录
+ async: true
+ # 是否启用IP地址解析
+ resolve-address: true
+ # 是否启用安全监控
+ security-monitor: true
+```
+
+---
+
+## 使用建议
+
+### 1. 日志管理
+
+- **合理保留期**: 根据业务需要设置日志保留期
+- **定期清理**: 定期清理过期日志释放存储空间
+- **分表存储**: 大量日志可考虑按月分表存储
+- **归档备份**: 重要日志定期归档备份
+
+### 2. 安全监控
+
+- **实时监控**: 实时监控异常登录行为
+- **告警机制**: 建立完善的安全告警机制
+- **自动响应**: 对可疑行为自动采取防护措施
+- **定期分析**: 定期分析登录日志发现安全趋势
+
+### 3. 性能优化
+
+- **异步记录**: 使用异步方式记录日志避免影响登录性能
+- **批量处理**: 批量处理日志数据提高效率
+- **索引优化**: 为查询字段建立合适索引
+- **分页查询**: 大数据量查询使用分页
+
+### 4. 隐私保护
+
+- **敏感信息**: 不记录密码等敏感信息
+- **数据脱敏**: 对部分信息进行脱敏处理
+- **访问控制**: 严格控制日志访问权限
+- **合规要求**: 遵守相关数据保护法规
+
+---
+
+## 注意事项
+
+1. **日志完整性**: 确保登录日志记录的完整性和准确性
+2. **性能影响**: 日志记录不应影响用户登录体验
+3. **存储空间**: 注意日志存储空间的管理和清理
+4. **安全防护**: 防止日志被恶意篡改或删除
+5. **隐私保护**: 不记录用户密码等敏感信息
+6. **异常处理**: 日志记录失败不应影响正常业务
+7. **监控告警**: 建立异常登录的监控告警机制
+8. **数据备份**: 重要日志数据需要备份
+9. **合规要求**: 遵守相关法律法规对日志的要求
+10. **访问审计**: 对日志访问行为也要进行审计
\ No newline at end of file
diff --git a/api/user/用户管理API.md b/api/user/用户管理API.md
new file mode 100644
index 0000000..efd4fa5
--- /dev/null
+++ b/api/user/用户管理API.md
@@ -0,0 +1,888 @@
+# 用户管理API
+
+## 概述
+
+用户管理模块提供系统用户的增删改查功能,包括用户信息管理、状态控制、密码管理、角色分配等核心功能。
+
+## 权限说明
+
+用户管理接口需要相应的权限才能访问:
+
+| 操作 | 权限码 | 说明 |
+|------|--------|------|
+| 查询用户列表 | `system:user:list` | 查看用户列表权限 |
+| 新增用户 | `system:user:add` | 新增用户权限 |
+| 修改用户 | `system:user:edit` | 修改用户信息权限 |
+| 删除用户 | `system:user:remove` | 删除用户权限 |
+| 重置密码 | `system:user:resetPwd` | 重置用户密码权限 |
+| 修改状态 | `system:user:status` | 修改用户状态权限 |
+| 分配角色 | `system:user:role` | 分配用户角色权限 |
+| 导入用户 | `system:user:import` | 导入用户数据权限 |
+| 导出用户 | `system:user:export` | 导出用户数据权限 |
+
+## 接口列表
+
+### 1. 分页查询用户列表
+
+**接口地址**: `GET /coder/sysLoginUser/listPage`
+
+**接口描述**: 分页查询系统用户列表
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:list`
+
+**请求头**:
+```
+Authorization: your-token-value
+```
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 | 示例 |
+|--------|------|------|------|------|
+| pageNo | Integer | 否 | 页码 | 1 |
+| pageSize | Integer | 否 | 每页大小 | 10 |
+| loginName | String | 否 | 登录账号 | admin |
+| userName | String | 否 | 用户姓名 | 管理员 |
+| userType | String | 否 | 用户类型 | 1 |
+| phone | String | 否 | 手机号码 | 13800138000 |
+| sex | String | 否 | 用户性别 | 1 |
+| userStatus | String | 否 | 用户状态 | 0 |
+| beginTime | String | 否 | 开始时间 | 2024-01-01 |
+| endTime | String | 否 | 结束时间 | 2024-12-31 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "records": [
+ {
+ "userId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userType": "1",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "avatar": "/upload/avatar/admin.jpg",
+ "userStatus": "0",
+ "loginIp": "127.0.0.1",
+ "loginTime": "2024-01-01 10:00:00",
+ "pwdUpdateTime": "2024-01-01 10:00:00",
+ "remark": "系统管理员",
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00",
+ "roleIds": [1, 2]
+ }
+ ],
+ "total": 1,
+ "size": 10,
+ "current": 1,
+ "pages": 1
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysLoginUser/listPage?pageNo=1&pageSize=10&loginName=admin" \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 2. 查询所有用户
+
+**接口地址**: `GET /coder/sysLoginUser/list`
+
+**接口描述**: 查询所有系统用户(不分页)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:list`
+
+**请求参数**: 同分页查询(除pageNo、pageSize外)
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": [
+ {
+ "userId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ // 其他字段...
+ }
+ ],
+ "traceId": "trace-123456"
+}
+```
+
+---
+
+### 3. 根据ID查询用户
+
+**接口地址**: `GET /coder/sysLoginUser/getById/{id}`
+
+**接口描述**: 根据用户ID查询用户详细信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:list`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 用户ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "userId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userType": "1",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "avatar": "/upload/avatar/admin.jpg",
+ "userStatus": "0",
+ "loginIp": "127.0.0.1",
+ "loginTime": "2024-01-01 10:00:00",
+ "pwdUpdateTime": "2024-01-01 10:00:00",
+ "remark": "系统管理员",
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00",
+ "roleIds": [1, 2]
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysLoginUser/getById/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 4. 新增用户
+
+**接口地址**: `POST /coder/sysLoginUser/add`
+
+**接口描述**: 新增系统用户
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:add`
+
+**请求参数**:
+
+```json
+{
+ "loginName": "newuser",
+ "userName": "新用户",
+ "password": "123456",
+ "userType": "1",
+ "email": "newuser@example.com",
+ "phone": "13800138001",
+ "sex": "1",
+ "userStatus": "0",
+ "remark": "新用户备注",
+ "roleIds": [2]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| loginName | String | 是 | 登录账号 | 3-16位字母数字 |
+| userName | String | 是 | 用户姓名 | 不能为空 |
+| password | String | 是 | 登录密码 | 不能为空 |
+| userType | String | 是 | 用户类型 | 1-系统用户 2-注册用户 |
+| email | String | 否 | 邮箱地址 | 邮箱格式验证 |
+| phone | String | 否 | 手机号码 | 手机号格式验证 |
+| sex | String | 否 | 用户性别 | 1-男 2-女 3-未知 |
+| userStatus | String | 是 | 用户状态 | 0-启用 1-停用 |
+| remark | String | 否 | 备注信息 | 最长200字符 |
+| roleIds | Long[] | 否 | 角色ID数组 | 有效的角色ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "新增成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/add \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "loginName": "newuser",
+ "userName": "新用户",
+ "password": "123456",
+ "userType": "1",
+ "email": "newuser@example.com",
+ "phone": "13800138001",
+ "sex": "1",
+ "userStatus": "0",
+ "remark": "新用户备注",
+ "roleIds": [2]
+ }'
+```
+
+---
+
+### 5. 修改用户信息
+
+**接口地址**: `POST /coder/sysLoginUser/update`
+
+**接口描述**: 修改系统用户信息
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:edit`
+
+**请求参数**:
+
+```json
+{
+ "userId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userType": "1",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "userStatus": "0",
+ "remark": "系统管理员",
+ "roleIds": [1, 2]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| userId | Long | 是 | 用户ID | 必须是有效的用户ID |
+| loginName | String | 是 | 登录账号 | 3-16位字母数字 |
+| userName | String | 是 | 用户姓名 | 不能为空 |
+| userType | String | 是 | 用户类型 | 1-系统用户 2-注册用户 |
+| email | String | 否 | 邮箱地址 | 邮箱格式验证 |
+| phone | String | 否 | 手机号码 | 手机号格式验证 |
+| sex | String | 否 | 用户性别 | 1-男 2-女 3-未知 |
+| userStatus | String | 是 | 用户状态 | 0-启用 1-停用 |
+| remark | String | 否 | 备注信息 | 最长200字符 |
+| roleIds | Long[] | 否 | 角色ID数组 | 有效的角色ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/update \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "userId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userType": "1",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "userStatus": "0",
+ "remark": "系统管理员",
+ "roleIds": [1, 2]
+ }'
+```
+
+---
+
+### 6. 删除用户
+
+**接口地址**: `POST /coder/sysLoginUser/deleteById/{id}`
+
+**接口描述**: 根据ID删除用户
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:remove`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 用户ID |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/deleteById/1 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 7. 批量删除用户
+
+**接口地址**: `POST /coder/sysLoginUser/batchDelete`
+
+**接口描述**: 批量删除用户
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:remove`
+
+**请求参数**:
+
+```json
+{
+ "ids": [1, 2, 3]
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| ids | Long[] | 是 | 用户ID数组 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "删除成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/batchDelete \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "ids": [1, 2, 3]
+ }'
+```
+
+---
+
+### 8. 修改用户状态
+
+**接口地址**: `POST /coder/sysLoginUser/updateStatus/{userId}/{userStatus}`
+
+**接口描述**: 修改用户状态(启用/停用)
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:status`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| userId | Long | 是 | 用户ID |
+| userStatus | String | 是 | 用户状态(0-启用 1-停用) |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/updateStatus/1/0 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 9. 获取登录用户信息
+
+**接口地址**: `GET /coder/sysLoginUser/getLoginUserInformation`
+
+**接口描述**: 获取当前登录用户的详细信息
+
+**是否需要认证**: 是
+
+**权限要求**: 无(已登录用户可访问)
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "userId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "userType": "1",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "avatar": "/upload/avatar/admin.jpg",
+ "userStatus": "0",
+ "loginIp": "127.0.0.1",
+ "loginTime": "2024-01-01 10:00:00",
+ "pwdUpdateTime": "2024-01-01 10:00:00",
+ "remark": "系统管理员",
+ "createBy": "admin",
+ "createTime": "2024-01-01 10:00:00",
+ "updateBy": "admin",
+ "updateTime": "2024-01-01 10:00:00",
+ "roleIds": [1, 2],
+ "roles": [
+ {
+ "roleId": 1,
+ "roleName": "超级管理员",
+ "roleCode": "admin"
+ }
+ ],
+ "permissions": [
+ "*:*:*"
+ ]
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysLoginUser/getLoginUserInformation \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 10. 获取个人资料
+
+**接口地址**: `GET /coder/sysLoginUser/getPersonalData`
+
+**接口描述**: 获取当前用户的个人资料
+
+**是否需要认证**: 是
+
+**权限要求**: 无(已登录用户可访问)
+
+**请求参数**: 无
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "userId": 1,
+ "loginName": "admin",
+ "userName": "管理员",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "avatar": "/upload/avatar/admin.jpg",
+ "remark": "系统管理员"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysLoginUser/getPersonalData \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 11. 修改个人资料
+
+**接口地址**: `POST /coder/sysLoginUser/updateBasicData`
+
+**接口描述**: 修改当前用户的个人资料
+
+**是否需要认证**: 是
+
+**权限要求**: 无(已登录用户可访问)
+
+**请求参数**:
+
+```json
+{
+ "userName": "管理员",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "avatar": "/upload/avatar/admin.jpg",
+ "remark": "系统管理员"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| userName | String | 是 | 用户姓名 | 不能为空 |
+| email | String | 否 | 邮箱地址 | 邮箱格式验证 |
+| phone | String | 否 | 手机号码 | 手机号格式验证 |
+| sex | String | 否 | 用户性别 | 1-男 2-女 3-未知 |
+| avatar | String | 否 | 头像地址 | 有效的文件路径 |
+| remark | String | 否 | 备注信息 | 最长200字符 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/updateBasicData \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "userName": "管理员",
+ "email": "admin@example.com",
+ "phone": "13800138000",
+ "sex": "1",
+ "avatar": "/upload/avatar/admin.jpg",
+ "remark": "系统管理员"
+ }'
+```
+
+---
+
+### 12. 修改登录密码
+
+**接口地址**: `POST /coder/sysLoginUser/updateUserPwd`
+
+**接口描述**: 修改当前用户的登录密码
+
+**是否需要认证**: 是
+
+**权限要求**: 无(已登录用户可访问)
+
+**请求参数**:
+
+```json
+{
+ "oldPassword": "123456",
+ "newPassword": "654321",
+ "confirmPassword": "654321"
+}
+```
+
+**请求参数说明**:
+
+| 参数名 | 类型 | 必填 | 说明 | 校验规则 |
+|--------|------|------|------|----------|
+| oldPassword | String | 是 | 原密码 | 不能为空 |
+| newPassword | String | 是 | 新密码 | 不能为空 |
+| confirmPassword | String | 是 | 确认密码 | 必须与新密码一致 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "修改成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/updateUserPwd \
+ -H "Content-Type: application/json" \
+ -H "Authorization: your-token-value" \
+ -d '{
+ "oldPassword": "123456",
+ "newPassword": "654321",
+ "confirmPassword": "654321"
+ }'
+```
+
+---
+
+### 13. 重置用户密码
+
+**接口地址**: `POST /coder/sysLoginUser/resetPwd/{id}/{password}`
+
+**接口描述**: 重置指定用户的密码
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:resetPwd`
+
+**路径参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| id | Long | 是 | 用户ID |
+| password | String | 是 | 新密码 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": "重置成功",
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/resetPwd/1/123456 \
+ -H "Authorization: your-token-value"
+```
+
+---
+
+### 14. 下载用户导入模板
+
+**接口地址**: `GET /coder/sysLoginUser/downloadExcelTemplate`
+
+**接口描述**: 下载用户批量导入的Excel模板
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:import`
+
+**请求参数**: 无
+
+**响应**: Excel文件下载
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ http://localhost:18099/coder/sysLoginUser/downloadExcelTemplate \
+ -H "Authorization: your-token-value" \
+ -o user_template.xlsx
+```
+
+---
+
+### 15. 导出用户数据
+
+**接口地址**: `GET /coder/sysLoginUser/exportExcelData`
+
+**接口描述**: 导出用户数据到Excel
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:export`
+
+**请求参数**: 同查询参数
+
+**响应**: Excel文件下载
+
+**调用示例**:
+
+```bash
+curl -X GET \
+ "http://localhost:18099/coder/sysLoginUser/exportExcelData?userStatus=0" \
+ -H "Authorization: your-token-value" \
+ -o users.xlsx
+```
+
+---
+
+### 16. 导入用户数据
+
+**接口地址**: `POST /coder/sysLoginUser/importExcelData`
+
+**接口描述**: 从Excel文件导入用户数据
+
+**是否需要认证**: 是
+
+**权限要求**: `system:user:import`
+
+**请求参数**: 文件上传
+
+**Content-Type**: `multipart/form-data`
+
+**请求参数**:
+
+| 参数名 | 类型 | 必填 | 说明 |
+|--------|------|------|------|
+| file | File | 是 | Excel文件 |
+| updateSupport | Boolean | 否 | 是否更新已存在的用户 |
+
+**响应示例**:
+
+```json
+{
+ "status": 200,
+ "msg": "SUCCESS",
+ "data": {
+ "total": 100,
+ "success": 95,
+ "failed": 5,
+ "message": "导入成功95条,失败5条"
+ },
+ "traceId": "trace-123456"
+}
+```
+
+**调用示例**:
+
+```bash
+curl -X POST \
+ http://localhost:18099/coder/sysLoginUser/importExcelData \
+ -H "Authorization: your-token-value" \
+ -F "file=@users.xlsx" \
+ -F "updateSupport=true"
+```
+
+---
+
+## 数据字典
+
+### 用户类型 (userType)
+
+| 值 | 说明 |
+|----|------|
+| 1 | 系统用户 |
+| 2 | 注册用户 |
+| 3 | 微信用户 |
+
+### 用户性别 (sex)
+
+| 值 | 说明 |
+|----|------|
+| 1 | 男 |
+| 2 | 女 |
+| 3 | 未知 |
+
+### 用户状态 (userStatus)
+
+| 值 | 说明 |
+|----|------|
+| 0 | 启用 |
+| 1 | 停用 |
+
+---
+
+## 错误码说明
+
+| 错误码 | 错误信息 | 说明 |
+|--------|----------|------|
+| 400 | 账号长度为 3-16 位 | 登录账号长度不符合要求 |
+| 400 | 账号格式为数字以及字母 | 登录账号格式不正确 |
+| 400 | 登录账号不能为空 | 登录账号为空 |
+| 400 | 用户真实姓名不能为空 | 用户姓名为空 |
+| 400 | 用户不存在 | 用户ID不存在 |
+| 400 | 账号已存在 | 登录账号已被使用 |
+| 400 | 手机号已存在 | 手机号已被使用 |
+| 400 | 邮箱已存在 | 邮箱已被使用 |
+| 400 | 不能删除当前用户 | 不能删除自己的账号 |
+| 400 | 不能停用当前用户 | 不能停用自己的账号 |
+| 400 | 原密码错误 | 修改密码时原密码不正确 |
+| 400 | 新密码不能与原密码相同 | 新密码与原密码一致 |
+| 400 | 两次输入的密码不一致 | 确认密码与新密码不一致 |
+| 401 | 当前会话未登录 | 未登录或Token无效 |
+| 403 | 权限不足 | 没有相应的操作权限 |
+| 500 | 系统异常 | 服务器内部错误 |
+
+---
+
+## 注意事项
+
+1. **权限验证**: 所有用户管理操作都需要相应的权限
+2. **数据校验**: 新增和修改用户时会进行数据格式校验
+3. **唯一性约束**: 登录账号、手机号、邮箱必须唯一
+4. **密码安全**: 密码使用盐值加密存储
+5. **操作日志**: 所有用户操作都会记录日志
+6. **批量操作**: 批量删除时会跳过超级管理员账号
+7. **Excel导入**: 支持Excel模板导入用户数据
+8. **角色关联**: 用户删除时会自动删除角色关联关系
+9. **会话管理**: 用户状态变更时会影响其登录会话
+10. **数据完整性**: 删除用户前会检查关联数据
\ No newline at end of file