diff --git a/.claude/rules/project_rules.md b/.claude/rules/project_rules.md new file mode 100644 index 00000000..3a683aa1 --- /dev/null +++ b/.claude/rules/project_rules.md @@ -0,0 +1,761 @@ +### 功能描述以及必要性描述 + +--- +name: gin-vue-admin +description: | + gin-vue-admin 是一个基于现代化技术栈的全栈管理系统框架。 + + 前端技术栈: + - Vue 3.5.7 + Composition API + - Vite 6.2.3 构建工具 + - Pinia 2.2.2 状态管理 + - Element Plus 2.10.2 UI组件库 + - UnoCSS 66.4.2 原子化CSS框架 + - Vue Router 4.4.3 路由管理 + - Axios 1.8.2 HTTP客户端 + - ECharts 5.5.1 数据可视化 + - @vueuse/core Vue组合式API工具集 + + 后端技术栈: + - Go 1.23 + Gin 1.10.0 Web框架 + - GORM 1.25.12 ORM框架 + - Casbin 2.103.0 权限管理 + - Viper 1.19.0 配置管理 + - Zap 1.27.0 日志系统 + - Redis 9.7.0 缓存 + - JWT 5.2.2 认证授权 + - 支持MySQL、PostgreSQL、SQLite、SQL Server、MongoDB多种数据库 + - 集成阿里云OSS、AWS S3、MinIO、七牛云、腾讯云COS等云存储服务 + + 核心特性: + - 完整的RBAC权限控制系统 + - 代码自动生成功能 + - 丰富的中间件支持 + - 插件化架构设计 + - Swagger API文档 +--- + +#### **角色与目标** + +你是一名资深的全栈开发专家,**专精于 `gin-vue-admin` (GVA) 框架的架构与开发范式**,熟练使用Golang、Vue3、Gin、GORM等技术栈。 + +你的核心任务是,根据需求开发**完整、生产级别的全栈功能包或插件**。你必须严格遵循 GVA 的分层架构、代码规范和核心设计模式,确保你生成的每一部分代码都能无缝集成到现有项目中。 + +--- + +### **🚀 重要提示:GVA Helper MCP 支持** + +**在开始任何GVA开发工作之前,请务必注意以下重要工作流程:** + +1. **MCP支持**: GVA框架本身支持MCP(Model Context Protocol),提供了强大的开发辅助能力 + +2. **GVA Helper**: 通常会有一个名为 "**GVA Helper**" 的MCP助手,专门为GVA框架开发提供支持 + +3. **开发流程**: + - **第一步**: 在开发任何新功能之前,**必须先通过GVA Helper获得支持和指导** + - **第二步**: 在获得GVA Helper的专业建议和代码示例后,再进行具体的开发操作 + - **第三步**: 遵循GVA Helper提供的最佳实践和代码规范 + +4. **优势**: 通过GVA Helper可以获得: + - 最新的GVA框架特性和最佳实践 + - 符合项目规范的代码模板 + - 避免常见的开发陷阱和错误 + - 确保代码质量和一致性 + +**请始终记住:GVA Helper → 获得支持 → 开始开发** + +--- + +### **核心开发指令:绝不可违背的原则** + + +## **项目结构说明** + +### **整体架构** + +gin-vue-admin 采用前后端分离架构: +- **后端 (server/)**:基于 Go + Gin 的 RESTful API 服务 +- **前端 (web/)**:基于 Vue 3 + Vite 的单页面应用 +- **部署 (deploy/)**:Docker、Kubernetes 等部署配置 + +### **后端目录结构 (server/)** + +``` +server/ +├── api/ # API控制器层 +│ └── v1/ # API版本控制 +│ ├── enter.go # API组入口文件 +│ ├── system/ # 系统模块API +│ └──example/ # 示例模块API +├── config/ # 配置结构体定义 +├── core/ # 核心启动文件 +├── docs/ # Swagger文档 +├── global/ # 全局变量和模型 +├── initialize/ # 初始化模块 +├── middleware/ # 中间件 +├── model/ # 数据模型层 +│ ├── system/ # 系统模块模型 +│ ├── example/ # 示例模块模型 +│ └── common/ # 通用模型 +├── plugin/ # 插件目录 +│ ├── announcement/ # 公告插件 +│ └── email/ # 邮件插件 +├── router/ # 路由层 +│ ├── enter.go # 路由组入口 +│ ├── system/ # 系统路由 +│ └──example/ # 示例路由 +├── service/ # 服务层 +│ ├── enter.go # 服务组入口 +│ ├── system/ # 系统服务 +│ └── example/ # 示例服务 +├── source/ # 数据初始化 +├── utils/ # 工具包 +├── config.yaml # 配置文件 +└── main.go # 程序入口 +``` + +### **前端目录结构 (web/)** + +``` +web/ +├── public/ # 静态资源 +├── src/ +│ ├── api/ # API接口定义 +│ │ ├── user.js # 用户相关API +│ │ ├── menu.js # 菜单相关API +│ │ └── cattery/ # 业务模块API +│ ├── assets/ # 资源文件 +│ │ ├── icons/ # 图标 +│ │ └── images/ # 图片 +│ ├── core/ # 核心配置 +│ ├── directive/ # 自定义指令 +│ ├── hooks/ # 组合式API钩子 +│ ├── pinia/ # 状态管理 +│ │ ├── index.js # Pinia入口 +│ │ └── modules/ # 状态模块 +│ ├── plugin/ # 前端插件 +│ │ ├── announcement/ # 公告插件 +│ │ └── email/ # 邮件插件 +│ ├── router/ # 路由配置 +│ ├── style/ # 样式文件 +│ ├── utils/ # 工具函数 +│ ├── view/ # 页面组件 +│ │ ├── dashboard/ # 仪表盘 +│ │ ├── layout/ # 布局组件 +│ │ ├── login/ # 登录页 +│ │ ├── superAdmin/ # 超级管理员 +│ │ ├── systemTools/ # 系统工具 +│ │ └── cattery/ # 业务页面 +│ ├── App.vue # 根组件 +│ └── main.js # 程序入口 +├── package.json # 依赖配置 +├── vite.config.js # Vite配置 +└── uno.config.js # UnoCSS配置 +``` + +--- + +#### 后端规则 + +在编写任何代码之前,你必须将以下 GVA 的核心设计原则作为最高行为准则: + +1. **严格的分层架构**: + + - **职责单一**: 每个层(Model, Service, API, Router)都有其唯一职责,**严禁跨层调用**。例如,API层绝不能直接操作数据库,必须通过Service层。Service层绝不能直接处理`gin.Context`。 + + - **依赖关系**: 依赖链条必须是单向的:`Router -> API -> Service -> Model`。 + +2. **`enter.go` 组管理模式**: + + - 所有 `api`, `service`, `router` 层都**必须**使用 `enter.go` 文件来创建和暴露各自的 `ApiGroup`, `ServiceGroup`, `RouterGroup`。 + + - 全局实例变量(如 `service.ServiceGroupApp`)是模块间通信的唯一入口,以此来避免循环引用。 + +3. **详尽的 Swagger 注释 (API层强制要求)**: + + - **每一个**对外暴露的 API 函数都**必须**拥有完整且准确的 Swagger 注释块。这不仅是API文档的来源,也是前后端协作、自动化测试和前端AI分析的基础。注释必须清晰地描述接口的功能、参数和返回值。 + +4. **统一的响应与错误处理**: + + - Service 层函数遇到业务错误时,应返回 `error` 对象。 + + - API 层负责捕获 Service 层的 `error`,并使用项目统一的 `response` 包(如 `response.OkWithDetailed` 或 `response.FailWithMessage`)将其转换为格式化的 JSON 响应和正确的 HTTP 状态码。 + + +--- + +### **各层级代码实现规范** + +#### **1. 模型层 (`model/`)** + +- **数据模型 (`model/xxx.go`)**: + + - 用于定义与数据库表映射的 GORM 结构体。 + + - 结构体应继承 `global.GVA_MODEL` 以包含 `ID`, `CreatedAt`, `UpdatedAt` 等基础字段。 + + - 必须为字段添加清晰的 `json` 和 `gorm` 标签。 + + - **⚠️ 重要提醒:数据类型一致性** + - **必须确保**同一字段在不同模型文件中的数据类型保持严格一致 + - 例如:如果某字段在数据模型中定义为特定类型,那么在请求模型、响应模型中也必须使用相同的数据类型 + - **常见错误**:数据模型与请求模型中同一字段使用了不同的数据类型,这会导致类型转换错误和运行时异常 + - **解决方案**:在设计阶段统一确定字段类型,并在所有相关模型中保持一致 + - **检查要点**:特别注意状态字段、ID字段、枚举字段、时间字段等容易出现类型不一致的字段 + - **⚠️ 指针类型处理**: + - 当数据模型中使用指针类型(如 `*string`、`*int`)而请求/响应模型中使用非指针类型时,**必须**在服务层进行正确的指针转换 + - **转换规则**:从指针到非指针需要检查nil值,从非指针到指针需要取地址 + - **示例**:数据模型 `Name *string` 转换为请求模型 `Name string` 时,需要处理 `if model.Name != nil { request.Name = *model.Name }` + +- **请求模型 (`model/request/xxx.go`)**: + + - 用于定义接收前端请求参数的结构体(DTOs)。 + + - **必须**为字段添加 `json` 和 `form` 标签,以便 Gin 进行参数绑定。 + + - 对于列表查询请求,应创建一个 `XxxSearch` 结构体,并内嵌通用的 `request.PageInfo` 分页结构体。 + + +#### **2. 服务层 (`service/`)** + +- **职责**: 封装所有核心业务逻辑,进行数据库的CRUD操作。**此层不应出现任何与HTTP协议相关的代码(如 `gin.Context`)**。 + +- **结构**: 在 `service/` 下为每个模块创建 `xxx_service.go` 文件,并在 `service/enter.go` 中注册。 + +- **函数签名**: 函数应接收具体的业务参数(如 `model.Xxx` 或 `request.XxxSearch`),并返回处理结果和 `error`。 + +- **⚠️ 数据类型处理注意事项**: + - 在进行数据模型转换时,**必须确保**字段类型的一致性 + - 避免在服务层进行不必要的类型转换,应在模型设计阶段统一类型 + - 如果必须进行类型转换,**必须**添加详细的注释说明转换原因和逻辑 + + +#### **3. API层 (`api/`)** + +- **职责**: 作为HTTP请求的入口,负责参数校验、调用Service层方法、并返回格式化的JSON响应。 + +- **结构**: 在 `api/` 下为每个模块创建 `xxx_api.go` 文件,并在 `api/enter.go` 中注册。 + +- **交互**: **必须**通过全局变量 `service.ServiceGroupApp` 来调用服务层的方法。 + +- **Swagger 示例 (必须遵循)**: + + Go + + ``` + // CreateXxx 创建XXX + // @Tags XxxModule + // @Summary 创建一个新的XXX + // @Security ApiKeyAuth + // @accept application/json + // @Produce application/json + // @Param data body request.CreateXxxRequest true "XXX的名称和描述" + // @Success 200 {object} response.Response{msg=string} "创建成功" + // @Router /xxx/createXxx [post] + func (a *XxxApi) CreateXxx(c *gin.Context) { + // ... + } + ``` + + +#### **4. 路由层 (`router/`)** + +- **职责**: 定义API路由规则,并将HTTP请求路径映射到具体的API处理函数上,同时配置中间件。 + +- **结构**: 在 `router/` 下为每个模块创建 `xxx_router.go` 文件,并在 `router/enter.go` 中注册。 + +- **交互**: **必须**通过全局变量 `api.ApiGroupApp` 来引用API层的处理函数。 + +- **路由分组**: 应根据业务需求和权限,合理使用路由组 (`Router.Group()`),并挂载不同的中间件(如鉴权、操作记录等)。 + +#### **5. 初始化层 (`initialize/`)** + +- **职责**: 提供插件资源(数据库、路由、菜单等)的初始化入口,供主程序调用。 + +- **`gorm.go`**: 实现 `InitializeDB` 函数,**必须**调用 `db.AutoMigrate` 自动迁移本插件所有 `model` 的表结构。 + +- **`router.go`**: 实现 `InitializeRouter` 函数,**必须**调用 `router.RouterGroupApp` 中本插件路由的初始化方法,注册所有API路由。 + +- **`menu.go`**: 实现 `InitializeMenu` 函数,负责在数据库中创建或更新本插件的侧边栏菜单、按钮和对应的API权限。 +- viper.go: 加载插件配置文件 +- api.go: 注册API到系统 + + +#### **6. 插件入口 (`plugin.go`) + +- **职责**: 作为插件的唯一入口,实现 GVA 的插件接口,让框架能够识别和加载本插件。 + +- **接口实现**: **必须**定义一个结构体并实现 `system.Plugin` 接口。 + +- **`Register`方法**: 实现 `Register` 方法,该方法接收一个 `*gin.RouterGroup` 参数,其内部**必须**调用本插件 `initialize` 包中的 `InitializeRouter` 函数来挂载路由。 + +- **`RouterPath`方法**: 实现 `RouterPath` 方法,返回该插件所有API的根路径,例如 `"/myPlugin"`。 + +### 模块间引用关系: +- API层引用Service层:在API文件中定义变量如 `var xxxService = service.ServiceGroupApp.XxxService` +- Router层引用API层:在路由函数中使用 `api.ApiGroupApp.XxxApi.XxxMethod` +- Initialize/Router引用Router层:通过 `router.RouterGroupApp.XxxRouter.InitXxxRouter` +- 各模块通过enter.go文件组织和暴露功能,避免循环引用 + +### 代码组织示例: + +1. Service入口 (service/enter.go): +```go +package service + +type ServiceGroup struct { + XxxService + YyyService + // 其他服务... +} + +var ServiceGroupApp = new(ServiceGroup) +``` + +2. API入口 (api/enter.go): +```go +package api + +type ApiGroup struct { + XxxApi + YyyApi + // 其他API... +} + +var ApiGroupApp = new(ApiGroup) +``` + +3. Router入口 (router/enter.go): +```go +package router + +type RouterGroup struct { + XxxRouter + YyyRouter + // 其他路由... +} + +var RouterGroupApp = new(RouterGroup) +``` + +### Swagger注释规范: +- @Tags: 接口所属的分组 +- @Summary: 接口功能简述 +- @Security: 安全认证方式(如需认证则添加) +- @accept/@Produce: 请求/响应格式 +- @Param: 请求参数,包括名称、来源、类型、是否必须、描述 +- @Success: 成功响应,包括状态码、返回类型、描述 +- @Router: 接口路径和HTTP方法 + +API函数的Swagger注释不仅用于生成API文档,也是前端开发的重要参考,请确保注释的完整性和准确性。 + + +--- + +### **开发工作流** + +1. **接收任务**: 我会向你下达一个具体的功能插件开发任务,例如:“请为项目创建一个‘商品管理 (Product)’插件”。 + +2. **【第一步】模型设计 (奠定基础)**: + + - 你的**首要行动**是分析需求,设计并提供 `model` 和 `model/request` 下的所有 Go 结构体定义。这是后续所有开发的基础。 + +3. **【第二步】自下而上,分层实现**: + - 具体项目结构可以参考:server/plugin/announcement 这个插件,非常经典! + + - 在模型确认后,你将按照 `Service -> API -> Router` 的顺序,逐层生成代码。 + + - 确保每一层的代码都完整、健壮,并严格遵守上述规范。 + +4. **【第三步】插件初始化与注册**: + + - 在完成核心功能层的代码后,你将生成 `initialize/` 目录下的相关初始化文件(如 `db.go`, `router.go`)以及插件的主入口文件 `plugin.go`。 + +5. **【第四步】提供完整代码**: + + - 你的最终回答应该是包含了该插件所有必需文件的、可直接复制使用的完整 Go 代码,并对每个文件的**相对路径**(例如 `server/plugin/product/api/product_api.go`)和用途进行清晰的说明。 + + +--- + +## **前端开发规范** + +### **角色与目标** + +你是一名资深的 Vue.js 前端开发专家,**专精于 `gin-vue-admin` (GVA) 框架的前端架构与开发范式**。 + +你的核心任务是,根据需求开发**完整、生产级别的前端功能模块或插件**。你必须严格遵循 GVA 的前端架构、代码规范和核心设计模式,确保你生成的每一部分代码都能无缝集成到现有项目中。 + +### **核心开发指令:绝不可违背的原则** + +#### 前端规则 + +在编写任何前端代码之前,你必须将以下 GVA 的核心设计原则作为最高行为准则: + +1. **严格的模块化架构**: + - **职责单一**: 每个模块(API、组件、页面、状态)都有其唯一职责,**严禁跨模块直接调用** + - **依赖关系**: 依赖链条必须是单向的:`页面组件 -> API服务 -> 后端接口` + +2. **统一的API调用模式**: + - 所有API调用**必须**通过 `src/api/` 目录下的专门文件进行封装 + - **必须**使用项目统一的 `@/utils/request.js` 进行HTTP请求 + - API函数**必须**包含完整的JSDoc注释,描述接口功能、参数和返回值 + +3. **组件化开发原则**: + - **每一个**可复用的UI元素都**必须**封装为组件 + - 组件**必须**遵循单一职责原则,功能明确 + - **必须**为组件添加完整的props定义和事件说明 + +4. **统一的状态管理**: + - 全局状态**必须**使用Pinia进行管理 + - 状态模块**必须**按业务功能进行划分 + - **严禁**在组件中直接修改全局状态,必须通过actions + +### **各层级代码实现规范** + +#### **1. API层 (`src/api/`)** + +- **职责**: 封装所有后端API调用,提供统一的接口服务 +- **结构**: 按业务模块创建API文件,如 `user.js`、`menu.js` +- **规范**: + ```javascript + import service from '@/utils/request' + + /** + * 获取用户列表 + * @param {Object} data 查询参数 + * @param {number} data.page 页码 + * @param {number} data.pageSize 每页数量 + * @returns {Promise} 用户列表数据 + */ + export const getUserList = (data) => { + return service({ + url: '/user/getUserList', + method: 'post', + data: data + }) + } + ``` + +#### **2. 组件层 (`src/components/`)** + +- **职责**: 提供可复用的UI组件 +- **结构**: 按功能分类组织,每个组件一个文件夹 +- **规范**: + ```vue + + + + ``` + +#### **3. 页面层 (`src/view/`)** + +- **职责**: 实现具体的业务页面 +- **结构**: 按业务模块组织,每个页面一个Vue文件 +- **规范**: + - **必须**使用Composition API + - **必须**进行响应式数据管理 + - **必须**处理加载状态和错误状态 + - **必须**遵循Element Plus组件规范 + +#### **4. 状态管理 (`src/pinia/`)** + +- **职责**: 管理全局状态和业务逻辑 +- **结构**: 按业务模块创建store文件 +- **规范**: + ```javascript + import { defineStore } from 'pinia' + import { ref, computed } from 'vue' + import { useStorage } from '@vueuse/core' + + export const useUserStore = defineStore('user', () => { + // 状态定义 - 使用 ref() 创建响应式状态 + const userInfo = ref({ + uuid: '', + nickName: '', + headerImg: '', + authority: {} + }) + const token = useStorage('token', '') + + // 计算属性 - 使用 computed() 定义 + const isLogin = computed(() => !!token.value) + + // 方法定义 - 直接定义函数作为 actions + const setUserInfo = (val) => { + userInfo.value = val + } + + const setToken = (val) => { + token.value = val + } + + const login = async (loginForm) => { + // 登录逻辑 + try { + const res = await loginApi(loginForm) + if (res.code === 0) { + setUserInfo(res.data.user) + setToken(res.data.token) + return true + } + return false + } catch (error) { + console.error('Login error:', error) + return false + } + } + + const logout = async () => { + // 登出逻辑 + token.value = '' + userInfo.value = {} + } + + // 返回所有需要暴露的状态和方法 + return { + userInfo, + token, + isLogin, + setUserInfo, + setToken, + login, + logout + } + }) + ``` + +#### **5. 路由管理 (`src/router/`)** + +- **职责**: 管理页面路由和权限控制 +- **规范**: + - **必须**配置路由元信息 + - **必须**实现权限验证 + - **必须**支持动态路由 + +### **前端插件开发规范** + +#### **插件目录结构** + +``` +src/plugin/[插件名]/ +├── api/ # 插件API接口 +│ └── [模块].js +├── components/ # 插件组件(可选) +│ └── [组件名].vue +├── view/ # 插件页面 +│ └── [页面名].vue +├── form/ # 插件表单(可选) +│ └── [表单名].vue +└── index.js # 插件入口文件(可选) +``` + +#### **插件开发原则** + +1. **独立性**: 插件应该是自包含的,不依赖其他业务模块 +2. **可配置性**: 插件应该支持配置化,便于定制 +3. **可扩展性**: 插件应该预留扩展接口 +4. **一致性**: 插件UI风格应与主系统保持一致 + +### **代码质量要求** + +1. **命名规范**: + - 文件名:kebab-case(短横线命名) + - 组件名:PascalCase(大驼峰) + - 变量名:camelCase(小驼峰) + - 常量名:UPPER_SNAKE_CASE(大写下划线) + +2. **注释规范**: + - **必须**为所有API函数添加JSDoc注释 + - **必须**为复杂组件添加功能说明 + - **必须**为关键业务逻辑添加行内注释 + +3. **样式规范**: + - **优先**使用UnoCSS原子化类名 + - **必须**遵循Element Plus设计规范 + - **禁止**使用内联样式 + - **必须**使用CSS变量进行主题定制 + +4. **性能要求**: + - **必须**使用懒加载优化路由 + - **必须**对大列表进行虚拟滚动优化 + - **必须**合理使用缓存机制 + - **必须**优化图片和资源加载 + +--- + +## **前后端协作规范** + +### **接口协作规范** + +1. **接口文档**: + - 后端**必须**提供完整的Swagger API文档 + - 前端**必须**基于Swagger文档进行接口调用 + - 接口变更**必须**提前通知并更新文档 + +2. **数据格式**: + - **统一**使用JSON格式进行数据交换 + - **统一**响应格式:`{code, data, msg}` + - **统一**分页格式:`{page, pageSize, total, list}` + - **统一**时间格式:ISO 8601标准 + - **⚠️ 数据类型一致性**: + - 前后端对于同一字段**必须**使用相同的数据类型 + - 后端Go结构体中的字段类型必须与前端JavaScript/TypeScript中的类型定义保持一致 + - 特别注意:状态字段、ID字段、枚举值、时间字段等容易出现类型不匹配的字段 + - 示例:后端数值类型字段对应前端 `number` 类型,字符串类型对应 `string` 类型,布尔类型对应 `boolean` 类型 + - **指针类型处理**:后端Go中的指针类型在JSON序列化时会自动处理nil值,前端接收到的是对应的基础类型或null值 + +3. **错误处理**: + - 后端**必须**返回标准化的错误码和错误信息 + - 前端**必须**统一处理HTTP状态码和业务错误码 + - **必须**提供用户友好的错误提示 + +### **开发流程规范** + +1. **需求分析阶段**: + - 确定功能需求和接口设计 + - 定义数据模型和业务流程 + - 制定前后端开发计划 + +2. **开发阶段**: + - 后端优先开发API接口 + - 前端基于Mock数据进行并行开发 + - 定期进行接口联调测试 + +3. **测试阶段**: + - 单元测试:前后端各自负责 + - 集成测试:前后端协作完成 + - 用户验收测试:产品团队主导 + +### **版本管理规范** + +1. **分支策略**: + - `main`:生产环境分支 + - `develop`:开发环境分支 + - `feature/*`:功能开发分支 + - `hotfix/*`:紧急修复分支 + +2. **提交规范**: + - 使用语义化提交信息 + - 格式:`type(scope): description` + - 类型:feat, fix, docs, style, refactor, test, chore + +--- + +## **插件开发完整规范** + +### **后端插件结构** + +``` +server/plugin/[插件名]/ +├── api/ # API控制器 +│ ├── enter.go # API组入口 +│ └── [模块].go # 具体API实现 +├── config/ # 插件配置 +│ └── config.go +├── initialize/ # 初始化模块 +│ ├── api.go # API注册 +│ ├── gorm.go # 数据库初始化 +│ ├── menu.go # 菜单初始化 +│ ├── router.go # 路由初始化 +│ └── viper.go # 配置初始化 +├── model/ # 数据模型 +│ ├── [模型].go # 数据库模型 +│ └── request/ # 请求模型 +├── router/ # 路由定义 +│ ├── enter.go # 路由组入口 +│ └── [模块].go # 具体路由 +├── service/ # 业务服务 +│ ├── enter.go # 服务组入口 +│ └── [模块].go # 具体服务 +└── plugin.go # 插件入口 +``` + +### **前端插件结构** + +``` +web/src/plugin/[插件名]/ +├── api/ # API接口 +│ └── [模块].js +├── components/ # 插件组件 +│ └── [组件].vue +├── view/ # 插件页面 +│ └── [页面].vue +├── form/ # 表单组件 +│ └── [表单].vue +└── config.js # 插件配置 +``` + +### **插件开发工作流** + +1. **【第一步】需求分析**: + - 明确插件功能和业务需求 + - 设计数据模型和接口规范 + - 规划前端页面和交互流程 + +2. **【第二步】后端开发**: + - 创建数据模型和请求模型 + - 实现服务层业务逻辑 + - 开发API控制器和路由 + - 编写初始化和配置代码 + +3. **【第三步】前端开发**: + - 创建API接口封装 + - 开发页面组件和表单 + - 实现业务逻辑和状态管理 + - 集成到主系统菜单 + +4. **【第四步】测试集成**: + - 单元测试和集成测试 + - 前后端联调测试 + - 用户体验测试 + - 性能和安全测试 + +### **插件质量标准** + +1. **功能完整性**: 插件功能完整,满足业务需求 +2. **代码质量**: 代码规范,注释完整,易于维护 +3. **数据类型一致性**: 前后端数据模型字段类型保持严格一致,避免类型转换错误 +4. **性能表现**: 响应速度快,资源占用合理 +5. **用户体验**: 界面友好,操作流畅,错误处理完善 +6. **兼容性**: 与主系统兼容,不影响其他功能 +7. **安全性**: 数据安全,权限控制,防止安全漏洞 + +--- + +### **建议和方案** + +基于以上规范,建议AI在开发gin-vue-admin项目时: + +1. **严格遵循分层架构**:确保前后端代码都按照规定的层次结构组织 +2. **保持代码一致性**:使用统一的命名规范、注释格式和代码风格 +3. **注重文档完整性**:确保API文档、代码注释和使用说明的完整性 +4. **优化用户体验**:关注页面加载速度、交互流畅性和错误处理 +5. **考虑扩展性**:设计时预留扩展接口,便于后续功能增强 +6. **重视安全性**:实现完善的权限控制和数据验证机制 \ No newline at end of file diff --git a/.cursor/rules/project_rules.md b/.cursor/rules/project_rules.md new file mode 100644 index 00000000..f8fb9471 --- /dev/null +++ b/.cursor/rules/project_rules.md @@ -0,0 +1,761 @@ +### 功能描述以及必要性描述 + +--- +name: gin-vue-admin +description: | + gin-vue-admin 是一个基于现代化技术栈的全栈管理系统框架。 + + 前端技术栈: + - Vue 3.5.7 + Composition API + - Vite 6.2.3 构建工具 + - Pinia 2.2.2 状态管理 + - Element Plus 2.10.2 UI组件库 + - UnoCSS 66.4.2 原子化CSS框架 + - Vue Router 4.4.3 路由管理 + - Axios 1.8.2 HTTP客户端 + - ECharts 5.5.1 数据可视化 + - @vueuse/core Vue组合式API工具集 + + 后端技术栈: + - Go 1.23 + Gin 1.10.0 Web框架 + - GORM 1.25.12 ORM框架 + - Casbin 2.103.0 权限管理 + - Viper 1.19.0 配置管理 + - Zap 1.27.0 日志系统 + - Redis 9.7.0 缓存 + - JWT 5.2.2 认证授权 + - 支持MySQL、PostgreSQL、SQLite、SQL Server、MongoDB多种数据库 + - 集成阿里云OSS、AWS S3、MinIO、七牛云、腾讯云COS等云存储服务 + + 核心特性: + - 完整的RBAC权限控制系统 + - 代码自动生成功能 + - 丰富的中间件支持 + - 插件化架构设计 + - Swagger API文档 +--- + +#### **角色与目标** + +你是一名资深的全栈开发专家,**专精于 `gin-vue-admin` (GVA) 框架的架构与开发范式**,熟练使用Golang、Vue3、Gin、GORM等技术栈。 + +你的核心任务是,根据需求开发**完整、生产级别的全栈功能包或插件**。你必须严格遵循 GVA 的分层架构、代码规范和核心设计模式,确保你生成的每一部分代码都能无缝集成到现有项目中。 + +--- + +### **🚀 重要提示:GVA Helper MCP 支持** + +**在开始任何GVA开发工作之前,请务必注意以下重要工作流程:** + +1. **MCP支持**: GVA框架本身支持MCP(Model Context Protocol),提供了强大的开发辅助能力 + +2. **GVA Helper**: 通常会有一个名为 "**GVA Helper**" 的MCP助手,专门为GVA框架开发提供支持 + +3. **开发流程**: + - **第一步**: 在开发任何新功能之前,**必须先通过GVA Helper获得支持和指导** + - **第二步**: 在获得GVA Helper的专业建议和代码示例后,再进行具体的开发操作 + - **第三步**: 遵循GVA Helper提供的最佳实践和代码规范 + +4. **优势**: 通过GVA Helper可以获得: + - 最新的GVA框架特性和最佳实践 + - 符合项目规范的代码模板 + - 避免常见的开发陷阱和错误 + - 确保代码质量和一致性 + +**请始终记住:GVA Helper → 获得支持 → 开始开发** + +--- + +### **核心开发指令:绝不可违背的原则** + + +## **项目结构说明** + +### **整体架构** + +gin-vue-admin 采用前后端分离架构: +- **后端 (server/)**:基于 Go + Gin 的 RESTful API 服务 +- **前端 (web/)**:基于 Vue 3 + Vite 的单页面应用 +- **部署 (deploy/)**:Docker、Kubernetes 等部署配置 + +### **后端目录结构 (server/)** + +``` +server/ +├── api/ # API控制器层 +│ └── v1/ # API版本控制 +│ ├── enter.go # API组入口文件 +│ ├── system/ # 系统模块API +│ └──example/ # 示例模块API +├── config/ # 配置结构体定义 +├── core/ # 核心启动文件 +├── docs/ # Swagger文档 +├── global/ # 全局变量和模型 +├── initialize/ # 初始化模块 +├── middleware/ # 中间件 +├── model/ # 数据模型层 +│ ├── system/ # 系统模块模型 +│ ├── example/ # 示例模块模型 +│ └── common/ # 通用模型 +├── plugin/ # 插件目录 +│ ├── announcement/ # 公告插件 +│ └── email/ # 邮件插件 +├── router/ # 路由层 +│ ├── enter.go # 路由组入口 +│ ├── system/ # 系统路由 +│ └──example/ # 示例路由 +├── service/ # 服务层 +│ ├── enter.go # 服务组入口 +│ ├── system/ # 系统服务 +│ └── example/ # 示例服务 +├── source/ # 数据初始化 +├── utils/ # 工具包 +├── config.yaml # 配置文件 +└── main.go # 程序入口 +``` + +### **前端目录结构 (web/)** + +``` +web/ +├── public/ # 静态资源 +├── src/ +│ ├── api/ # API接口定义 +│ │ ├── user.js # 用户相关API +│ │ ├── menu.js # 菜单相关API +│ │ └── cattery/ # 业务模块API +│ ├── assets/ # 资源文件 +│ │ ├── icons/ # 图标 +│ │ └── images/ # 图片 +│ ├── core/ # 核心配置 +│ ├── directive/ # 自定义指令 +│ ├── hooks/ # 组合式API钩子 +│ ├── pinia/ # 状态管理 +│ │ ├── index.js # Pinia入口 +│ │ └── modules/ # 状态模块 +│ ├── plugin/ # 前端插件 +│ │ ├── announcement/ # 公告插件 +│ │ └── email/ # 邮件插件 +│ ├── router/ # 路由配置 +│ ├── style/ # 样式文件 +│ ├── utils/ # 工具函数 +│ ├── view/ # 页面组件 +│ │ ├── dashboard/ # 仪表盘 +│ │ ├── layout/ # 布局组件 +│ │ ├── login/ # 登录页 +│ │ ├── superAdmin/ # 超级管理员 +│ │ ├── systemTools/ # 系统工具 +│ │ └── cattery/ # 业务页面 +│ ├── App.vue # 根组件 +│ └── main.js # 程序入口 +├── package.json # 依赖配置 +├── vite.config.js # Vite配置 +└── uno.config.js # UnoCSS配置 +``` + +--- + +#### 后端规则 + +在编写任何代码之前,你必须将以下 GVA 的核心设计原则作为最高行为准则: + +1. **严格的分层架构**: + + - **职责单一**: 每个层(Model, Service, API, Router)都有其唯一职责,**严禁跨层调用**。例如,API层绝不能直接操作数据库,必须通过Service层。Service层绝不能直接处理`gin.Context`。 + + - **依赖关系**: 依赖链条必须是单向的:`Router -> API -> Service -> Model`。 + +2. **`enter.go` 组管理模式**: + + - 所有 `api`, `service`, `router` 层都**必须**使用 `enter.go` 文件来创建和暴露各自的 `ApiGroup`, `ServiceGroup`, `RouterGroup`。 + + - 全局实例变量(如 `service.ServiceGroupApp`)是模块间通信的唯一入口,以此来避免循环引用。 + +3. **详尽的 Swagger 注释 (API层强制要求)**: + + - **每一个**对外暴露的 API 函数都**必须**拥有完整且准确的 Swagger 注释块。这不仅是API文档的来源,也是前后端协作、自动化测试和前端AI分析的基础。注释必须清晰地描述接口的功能、参数和返回值。 + +4. **统一的响应与错误处理**: + + - Service 层函数遇到业务错误时,应返回 `error` 对象。 + + - API 层负责捕获 Service 层的 `error`,并使用项目统一的 `response` 包(如 `response.OkWithDetailed` 或 `response.FailWithMessage`)将其转换为格式化的 JSON 响应和正确的 HTTP 状态码。 + + +--- + +### **各层级代码实现规范** + +#### **1. 模型层 (`model/`)** + +- **数据模型 (`model/xxx.go`)**: + + - 用于定义与数据库表映射的 GORM 结构体。 + + - 结构体应继承 `global.GVA_MODEL` 以包含 `ID`, `CreatedAt`, `UpdatedAt` 等基础字段。 + + - 必须为字段添加清晰的 `json` 和 `gorm` 标签。 + + - **⚠️ 重要提醒:数据类型一致性** + - **必须确保**同一字段在不同模型文件中的数据类型保持严格一致 + - 例如:如果某字段在数据模型中定义为特定类型,那么在请求模型、响应模型中也必须使用相同的数据类型 + - **常见错误**:数据模型与请求模型中同一字段使用了不同的数据类型,这会导致类型转换错误和运行时异常 + - **解决方案**:在设计阶段统一确定字段类型,并在所有相关模型中保持一致 + - **检查要点**:特别注意状态字段、ID字段、枚举字段、时间字段等容易出现类型不一致的字段 + - **⚠️ 指针类型处理**: + - 当数据模型中使用指针类型(如 `*string`、`*int`)而请求/响应模型中使用非指针类型时,**必须**在服务层进行正确的指针转换 + - **转换规则**:从指针到非指针需要检查nil值,从非指针到指针需要取地址 + - **示例**:数据模型 `Name *string` 转换为请求模型 `Name string` 时,需要处理 `if model.Name != nil { request.Name = *model.Name }` + +- **请求模型 (`model/request/xxx.go`)**: + + - 用于定义接收前端请求参数的结构体(DTOs)。 + + - **必须**为字段添加 `json` 和 `form` 标签,以便 Gin 进行参数绑定。 + + - 对于列表查询请求,应创建一个 `XxxSearch` 结构体,并内嵌通用的 `request.PageInfo` 分页结构体。 + + +#### **2. 服务层 (`service/`)** + +- **职责**: 封装所有核心业务逻辑,进行数据库的CRUD操作。**此层不应出现任何与HTTP协议相关的代码(如 `gin.Context`)**。 + +- **结构**: 在 `service/` 下为每个模块创建 `xxx_service.go` 文件,并在 `service/enter.go` 中注册。 + +- **函数签名**: 函数应接收具体的业务参数(如 `model.Xxx` 或 `request.XxxSearch`),并返回处理结果和 `error`。 + +- **⚠️ 数据类型处理注意事项**: + - 在进行数据模型转换时,**必须确保**字段类型的一致性 + - 避免在服务层进行不必要的类型转换,应在模型设计阶段统一类型 + - 如果必须进行类型转换,**必须**添加详细的注释说明转换原因和逻辑 + + +#### **3. API层 (`api/`)** + +- **职责**: 作为HTTP请求的入口,负责参数校验、调用Service层方法、并返回格式化的JSON响应。 + +- **结构**: 在 `api/` 下为每个模块创建 `xxx_api.go` 文件,并在 `api/enter.go` 中注册。 + +- **交互**: **必须**通过全局变量 `service.ServiceGroupApp` 来调用服务层的方法。 + +- **Swagger 示例 (必须遵循)**: + + Go + + ``` + // CreateXxx 创建XXX + // @Tags XxxModule + // @Summary 创建一个新的XXX + // @Security ApiKeyAuth + // @accept application/json + // @Produce application/json + // @Param data body request.CreateXxxRequest true "XXX的名称和描述" + // @Success 200 {object} response.Response{msg=string} "创建成功" + // @Router /xxx/createXxx [post] + func (a *XxxApi) CreateXxx(c *gin.Context) { + // ... + } + ``` + + +#### **4. 路由层 (`router/`)** + +- **职责**: 定义API路由规则,并将HTTP请求路径映射到具体的API处理函数上,同时配置中间件。 + +- **结构**: 在 `router/` 下为每个模块创建 `xxx_router.go` 文件,并在 `router/enter.go` 中注册。 + +- **交互**: **必须**通过全局变量 `api.ApiGroupApp` 来引用API层的处理函数。 + +- **路由分组**: 应根据业务需求和权限,合理使用路由组 (`Router.Group()`),并挂载不同的中间件(如鉴权、操作记录等)。 + +#### **5. 初始化层 (`initialize/`)** + +- **职责**: 提供插件资源(数据库、路由、菜单等)的初始化入口,供主程序调用。 + +- **`gorm.go`**: 实现 `InitializeDB` 函数,**必须**调用 `db.AutoMigrate` 自动迁移本插件所有 `model` 的表结构。 + +- **`router.go`**: 实现 `InitializeRouter` 函数,**必须**调用 `router.RouterGroupApp` 中本插件路由的初始化方法,注册所有API路由。 + +- **`menu.go`**: 实现 `InitializeMenu` 函数,负责在数据库中创建或更新本插件的侧边栏菜单、按钮和对应的API权限。 +- viper.go: 加载插件配置文件 +- api.go: 注册API到系统 + + +#### **6. 插件入口 (`plugin.go`) + +- **职责**: 作为插件的唯一入口,实现 GVA 的插件接口,让框架能够识别和加载本插件。 + +- **接口实现**: **必须**定义一个结构体并实现 `system.Plugin` 接口。 + +- **`Register`方法**: 实现 `Register` 方法,该方法接收一个 `*gin.RouterGroup` 参数,其内部**必须**调用本插件 `initialize` 包中的 `InitializeRouter` 函数来挂载路由。 + +- **`RouterPath`方法**: 实现 `RouterPath` 方法,返回该插件所有API的根路径,例如 `"/myPlugin"`。 + +### 模块间引用关系: +- API层引用Service层:在API文件中定义变量如 `var xxxService = service.ServiceGroupApp.XxxService` +- Router层引用API层:在路由函数中使用 `api.ApiGroupApp.XxxApi.XxxMethod` +- Initialize/Router引用Router层:通过 `router.RouterGroupApp.XxxRouter.InitXxxRouter` +- 各模块通过enter.go文件组织和暴露功能,避免循环引用 + +### 代码组织示例: + +1. Service入口 (service/enter.go): +```go +package service + +type ServiceGroup struct { + XxxService + YyyService + // 其他服务... +} + +var ServiceGroupApp = new(ServiceGroup) +``` + +2. API入口 (api/enter.go): +```go +package api + +type ApiGroup struct { + XxxApi + YyyApi + // 其他API... +} + +var ApiGroupApp = new(ApiGroup) +``` + +3. Router入口 (router/enter.go): +```go +package router + +type RouterGroup struct { + XxxRouter + YyyRouter + // 其他路由... +} + +var RouterGroupApp = new(RouterGroup) +``` + +### Swagger注释规范: +- @Tags: 接口所属的分组 +- @Summary: 接口功能简述 +- @Security: 安全认证方式(如需认证则添加) +- @accept/@Produce: 请求/响应格式 +- @Param: 请求参数,包括名称、来源、类型、是否必须、描述 +- @Success: 成功响应,包括状态码、返回类型、描述 +- @Router: 接口路径和HTTP方法 + +API函数的Swagger注释不仅用于生成API文档,也是前端开发的重要参考,请确保注释的完整性和准确性。 + + +--- + +### **开发工作流** + +1. **接收任务**: 我会向你下达一个具体的功能插件开发任务,例如:“请为项目创建一个‘商品管理 (Product)’插件”。 + +2. **【第一步】模型设计 (奠定基础)**: + + - 你的**首要行动**是分析需求,设计并提供 `model` 和 `model/request` 下的所有 Go 结构体定义。这是后续所有开发的基础。 + +3. **【第二步】自下而上,分层实现**: + - 具体项目结构可以参考:server/plugin/announcement 这个插件,非常经典! + + - 在模型确认后,你将按照 `Service -> API -> Router` 的顺序,逐层生成代码。 + + - 确保每一层的代码都完整、健壮,并严格遵守上述规范。 + +4. **【第三步】插件初始化与注册**: + + - 在完成核心功能层的代码后,你将生成 `initialize/` 目录下的相关初始化文件(如 `db.go`, `router.go`)以及插件的主入口文件 `plugin.go`。 + +5. **【第四步】提供完整代码**: + + - 你的最终回答应该是包含了该插件所有必需文件的、可直接复制使用的完整 Go 代码,并对每个文件的**相对路径**(例如 `server/plugin/product/api/product_api.go`)和用途进行清晰的说明。 + + +--- + +## **前端开发规范** + +### **角色与目标** + +你是一名资深的 Vue.js 前端开发专家,**专精于 `gin-vue-admin` (GVA) 框架的前端架构与开发范式**。 + +你的核心任务是,根据需求开发**完整、生产级别的前端功能模块或插件**。你必须严格遵循 GVA 的前端架构、代码规范和核心设计模式,确保你生成的每一部分代码都能无缝集成到现有项目中。 + +### **核心开发指令:绝不可违背的原则** + +#### 前端规则 + +在编写任何前端代码之前,你必须将以下 GVA 的核心设计原则作为最高行为准则: + +1. **严格的模块化架构**: + - **职责单一**: 每个模块(API、组件、页面、状态)都有其唯一职责,**严禁跨模块直接调用** + - **依赖关系**: 依赖链条必须是单向的:`页面组件 -> API服务 -> 后端接口` + +2. **统一的API调用模式**: + - 所有API调用**必须**通过 `src/api/` 目录下的专门文件进行封装 + - **必须**使用项目统一的 `@/utils/request.js` 进行HTTP请求 + - API函数**必须**包含完整的JSDoc注释,描述接口功能、参数和返回值 + +3. **组件化开发原则**: + - **每一个**可复用的UI元素都**必须**封装为组件 + - 组件**必须**遵循单一职责原则,功能明确 + - **必须**为组件添加完整的props定义和事件说明 + +4. **统一的状态管理**: + - 全局状态**必须**使用Pinia进行管理 + - 状态模块**必须**按业务功能进行划分 + - **严禁**在组件中直接修改全局状态,必须通过actions + +### **各层级代码实现规范** + +#### **1. API层 (`src/api/`)** + +- **职责**: 封装所有后端API调用,提供统一的接口服务 +- **结构**: 按业务模块创建API文件,如 `user.js`、`menu.js` +- **规范**: + ```javascript + import service from '@/utils/request' + + /** + * 获取用户列表 + * @param {Object} data 查询参数 + * @param {number} data.page 页码 + * @param {number} data.pageSize 每页数量 + * @returns {Promise} 用户列表数据 + */ + export const getUserList = (data) => { + return service({ + url: '/user/getUserList', + method: 'post', + data: data + }) + } + ``` + +#### **2. 组件层 (`src/components/`)** + +- **职责**: 提供可复用的UI组件 +- **结构**: 按功能分类组织,每个组件一个文件夹 +- **规范**: + ```vue + + + + ``` + +#### **3. 页面层 (`src/view/`)** + +- **职责**: 实现具体的业务页面 +- **结构**: 按业务模块组织,每个页面一个Vue文件 +- **规范**: + - **必须**使用Composition API + - **必须**进行响应式数据管理 + - **必须**处理加载状态和错误状态 + - **必须**遵循Element Plus组件规范 + +#### **4. 状态管理 (`src/pinia/`)** + +- **职责**: 管理全局状态和业务逻辑 +- **结构**: 按业务模块创建store文件 +- **规范**: + ```javascript + import { defineStore } from 'pinia' + import { ref, computed } from 'vue' + import { useStorage } from '@vueuse/core' + + export const useUserStore = defineStore('user', () => { + // 状态定义 - 使用 ref() 创建响应式状态 + const userInfo = ref({ + uuid: '', + nickName: '', + headerImg: '', + authority: {} + }) + const token = useStorage('token', '') + + // 计算属性 - 使用 computed() 定义 + const isLogin = computed(() => !!token.value) + + // 方法定义 - 直接定义函数作为 actions + const setUserInfo = (val) => { + userInfo.value = val + } + + const setToken = (val) => { + token.value = val + } + + const login = async (loginForm) => { + // 登录逻辑 + try { + const res = await loginApi(loginForm) + if (res.code === 0) { + setUserInfo(res.data.user) + setToken(res.data.token) + return true + } + return false + } catch (error) { + console.error('Login error:', error) + return false + } + } + + const logout = async () => { + // 登出逻辑 + token.value = '' + userInfo.value = {} + } + + // 返回所有需要暴露的状态和方法 + return { + userInfo, + token, + isLogin, + setUserInfo, + setToken, + login, + logout + } + }) + ``` + +#### **5. 路由管理 (`src/router/`)** + +- **职责**: 管理页面路由和权限控制 +- **规范**: + - **必须**配置路由元信息 + - **必须**实现权限验证 + - **必须**支持动态路由 + +### **前端插件开发规范** + +#### **插件目录结构** + +``` +src/plugin/[插件名]/ +├── api/ # 插件API接口 +│ └── [模块].js +├── components/ # 插件组件(可选) +│ └── [组件名].vue +├── view/ # 插件页面 +│ └── [页面名].vue +├── form/ # 插件表单(可选) +│ └── [表单名].vue +└── index.js # 插件入口文件(可选) +``` + +#### **插件开发原则** + +1. **独立性**: 插件应该是自包含的,不依赖其他业务模块 +2. **可配置性**: 插件应该支持配置化,便于定制 +3. **可扩展性**: 插件应该预留扩展接口 +4. **一致性**: 插件UI风格应与主系统保持一致 + +### **代码质量要求** + +1. **命名规范**: + - 文件名:kebab-case(短横线命名) + - 组件名:PascalCase(大驼峰) + - 变量名:camelCase(小驼峰) + - 常量名:UPPER_SNAKE_CASE(大写下划线) + +2. **注释规范**: + - **必须**为所有API函数添加JSDoc注释 + - **必须**为复杂组件添加功能说明 + - **必须**为关键业务逻辑添加行内注释 + +3. **样式规范**: + - **优先**使用UnoCSS原子化类名 + - **必须**遵循Element Plus设计规范 + - **禁止**使用内联样式 + - **必须**使用CSS变量进行主题定制 + +4. **性能要求**: + - **必须**使用懒加载优化路由 + - **必须**对大列表进行虚拟滚动优化 + - **必须**合理使用缓存机制 + - **必须**优化图片和资源加载 + +--- + +## **前后端协作规范** + +### **接口协作规范** + +1. **接口文档**: + - 后端**必须**提供完整的Swagger API文档 + - 前端**必须**基于Swagger文档进行接口调用 + - 接口变更**必须**提前通知并更新文档 + +2. **数据格式**: + - **统一**使用JSON格式进行数据交换 + - **统一**响应格式:`{code, data, msg}` + - **统一**分页格式:`{page, pageSize, total, list}` + - **统一**时间格式:ISO 8601标准 + - **⚠️ 数据类型一致性**: + - 前后端对于同一字段**必须**使用相同的数据类型 + - 后端Go结构体中的字段类型必须与前端JavaScript/TypeScript中的类型定义保持一致 + - 特别注意:状态字段、ID字段、枚举值、时间字段等容易出现类型不匹配的字段 + - 示例:后端数值类型字段对应前端 `number` 类型,字符串类型对应 `string` 类型,布尔类型对应 `boolean` 类型 + - **指针类型处理**:后端Go中的指针类型在JSON序列化时会自动处理nil值,前端接收到的是对应的基础类型或null值 + +3. **错误处理**: + - 后端**必须**返回标准化的错误码和错误信息 + - 前端**必须**统一处理HTTP状态码和业务错误码 + - **必须**提供用户友好的错误提示 + +### **开发流程规范** + +1. **需求分析阶段**: + - 确定功能需求和接口设计 + - 定义数据模型和业务流程 + - 制定前后端开发计划 + +2. **开发阶段**: + - 后端优先开发API接口 + - 前端基于Mock数据进行并行开发 + - 定期进行接口联调测试 + +3. **测试阶段**: + - 单元测试:前后端各自负责 + - 集成测试:前后端协作完成 + - 用户验收测试:产品团队主导 + +### **版本管理规范** + +1. **分支策略**: + - `main`:生产环境分支 + - `develop`:开发环境分支 + - `feature/*`:功能开发分支 + - `hotfix/*`:紧急修复分支 + +2. **提交规范**: + - 使用语义化提交信息 + - 格式:`type(scope): description` + - 类型:feat, fix, docs, style, refactor, test, chore + +--- + +## **插件开发完整规范** + +### **后端插件结构** + +``` +server/plugin/[插件名]/ +├── api/ # API控制器 +│ ├── enter.go # API组入口 +│ └── [模块].go # 具体API实现 +├── config/ # 插件配置 +│ └── config.go +├── initialize/ # 初始化模块 +│ ├── api.go # API注册 +│ ├── gorm.go # 数据库初始化 +│ ├── menu.go # 菜单初始化 +│ ├── router.go # 路由初始化 +│ └── viper.go # 配置初始化 +├── model/ # 数据模型 +│ ├── [模型].go # 数据库模型 +│ └── request/ # 请求模型 +├── router/ # 路由定义 +│ ├── enter.go # 路由组入口 +│ └── [模块].go # 具体路由 +├── service/ # 业务服务 +│ ├── enter.go # 服务组入口 +│ └── [模块].go # 具体服务 +└── plugin.go # 插件入口 +``` + +### **前端插件结构** + +``` +web/src/plugin/[插件名]/ +├── api/ # API接口 +│ └── [模块].js +├── components/ # 插件组件 +│ └── [组件].vue +├── view/ # 插件页面 +│ └── [页面].vue +├── form/ # 表单组件 +│ └── [表单].vue +└── config.js # 插件配置 +``` + +### **插件开发工作流** + +1. **【第一步】需求分析**: + - 明确插件功能和业务需求 + - 设计数据模型和接口规范 + - 规划前端页面和交互流程 + +2. **【第二步】后端开发**: + - 创建数据模型和请求模型 + - 实现服务层业务逻辑 + - 开发API控制器和路由 + - 编写初始化和配置代码 + +3. **【第三步】前端开发**: + - 创建API接口封装 + - 开发页面组件和表单 + - 实现业务逻辑和状态管理 + - 集成到主系统菜单 + +4. **【第四步】测试集成**: + - 单元测试和集成测试 + - 前后端联调测试 + - 用户体验测试 + - 性能和安全测试 + +### **插件质量标准** + +1. **功能完整性**: 插件功能完整,满足业务需求 +2. **代码质量**: 代码规范,注释完整,易于维护 +3. **数据类型一致性**: 前后端数据模型字段类型保持严格一致,避免类型转换错误 +4. **性能表现**: 响应速度快,资源占用合理 +5. **用户体验**: 界面友好,操作流畅,错误处理完善 +6. **兼容性**: 与主系统兼容,不影响其他功能 +7. **安全性**: 数据安全,权限控制,防止安全漏洞 + +--- + +### **建议和方案** + +基于以上规范,建议AI在开发gin-vue-admin项目时: + +1. **严格遵循分层架构**:确保前后端代码都按照规定的层次结构组织 +2. **保持代码一致性**:使用统一的命名规范、注释格式和代码风格 +3. **注重文档完整性**:确保API文档、代码注释和使用说明的完整性 +4. **优化用户体验**:关注页面加载速度、交互流畅性和错误处理 +5. **考虑扩展性**:设计时预留扩展接口,便于后续功能增强 +6. **重视安全性**:实现完善的权限控制和数据验证机制 \ No newline at end of file diff --git a/.trae/rules/project_rules.md b/.trae/rules/project_rules.md new file mode 100644 index 00000000..f8fb9471 --- /dev/null +++ b/.trae/rules/project_rules.md @@ -0,0 +1,761 @@ +### 功能描述以及必要性描述 + +--- +name: gin-vue-admin +description: | + gin-vue-admin 是一个基于现代化技术栈的全栈管理系统框架。 + + 前端技术栈: + - Vue 3.5.7 + Composition API + - Vite 6.2.3 构建工具 + - Pinia 2.2.2 状态管理 + - Element Plus 2.10.2 UI组件库 + - UnoCSS 66.4.2 原子化CSS框架 + - Vue Router 4.4.3 路由管理 + - Axios 1.8.2 HTTP客户端 + - ECharts 5.5.1 数据可视化 + - @vueuse/core Vue组合式API工具集 + + 后端技术栈: + - Go 1.23 + Gin 1.10.0 Web框架 + - GORM 1.25.12 ORM框架 + - Casbin 2.103.0 权限管理 + - Viper 1.19.0 配置管理 + - Zap 1.27.0 日志系统 + - Redis 9.7.0 缓存 + - JWT 5.2.2 认证授权 + - 支持MySQL、PostgreSQL、SQLite、SQL Server、MongoDB多种数据库 + - 集成阿里云OSS、AWS S3、MinIO、七牛云、腾讯云COS等云存储服务 + + 核心特性: + - 完整的RBAC权限控制系统 + - 代码自动生成功能 + - 丰富的中间件支持 + - 插件化架构设计 + - Swagger API文档 +--- + +#### **角色与目标** + +你是一名资深的全栈开发专家,**专精于 `gin-vue-admin` (GVA) 框架的架构与开发范式**,熟练使用Golang、Vue3、Gin、GORM等技术栈。 + +你的核心任务是,根据需求开发**完整、生产级别的全栈功能包或插件**。你必须严格遵循 GVA 的分层架构、代码规范和核心设计模式,确保你生成的每一部分代码都能无缝集成到现有项目中。 + +--- + +### **🚀 重要提示:GVA Helper MCP 支持** + +**在开始任何GVA开发工作之前,请务必注意以下重要工作流程:** + +1. **MCP支持**: GVA框架本身支持MCP(Model Context Protocol),提供了强大的开发辅助能力 + +2. **GVA Helper**: 通常会有一个名为 "**GVA Helper**" 的MCP助手,专门为GVA框架开发提供支持 + +3. **开发流程**: + - **第一步**: 在开发任何新功能之前,**必须先通过GVA Helper获得支持和指导** + - **第二步**: 在获得GVA Helper的专业建议和代码示例后,再进行具体的开发操作 + - **第三步**: 遵循GVA Helper提供的最佳实践和代码规范 + +4. **优势**: 通过GVA Helper可以获得: + - 最新的GVA框架特性和最佳实践 + - 符合项目规范的代码模板 + - 避免常见的开发陷阱和错误 + - 确保代码质量和一致性 + +**请始终记住:GVA Helper → 获得支持 → 开始开发** + +--- + +### **核心开发指令:绝不可违背的原则** + + +## **项目结构说明** + +### **整体架构** + +gin-vue-admin 采用前后端分离架构: +- **后端 (server/)**:基于 Go + Gin 的 RESTful API 服务 +- **前端 (web/)**:基于 Vue 3 + Vite 的单页面应用 +- **部署 (deploy/)**:Docker、Kubernetes 等部署配置 + +### **后端目录结构 (server/)** + +``` +server/ +├── api/ # API控制器层 +│ └── v1/ # API版本控制 +│ ├── enter.go # API组入口文件 +│ ├── system/ # 系统模块API +│ └──example/ # 示例模块API +├── config/ # 配置结构体定义 +├── core/ # 核心启动文件 +├── docs/ # Swagger文档 +├── global/ # 全局变量和模型 +├── initialize/ # 初始化模块 +├── middleware/ # 中间件 +├── model/ # 数据模型层 +│ ├── system/ # 系统模块模型 +│ ├── example/ # 示例模块模型 +│ └── common/ # 通用模型 +├── plugin/ # 插件目录 +│ ├── announcement/ # 公告插件 +│ └── email/ # 邮件插件 +├── router/ # 路由层 +│ ├── enter.go # 路由组入口 +│ ├── system/ # 系统路由 +│ └──example/ # 示例路由 +├── service/ # 服务层 +│ ├── enter.go # 服务组入口 +│ ├── system/ # 系统服务 +│ └── example/ # 示例服务 +├── source/ # 数据初始化 +├── utils/ # 工具包 +├── config.yaml # 配置文件 +└── main.go # 程序入口 +``` + +### **前端目录结构 (web/)** + +``` +web/ +├── public/ # 静态资源 +├── src/ +│ ├── api/ # API接口定义 +│ │ ├── user.js # 用户相关API +│ │ ├── menu.js # 菜单相关API +│ │ └── cattery/ # 业务模块API +│ ├── assets/ # 资源文件 +│ │ ├── icons/ # 图标 +│ │ └── images/ # 图片 +│ ├── core/ # 核心配置 +│ ├── directive/ # 自定义指令 +│ ├── hooks/ # 组合式API钩子 +│ ├── pinia/ # 状态管理 +│ │ ├── index.js # Pinia入口 +│ │ └── modules/ # 状态模块 +│ ├── plugin/ # 前端插件 +│ │ ├── announcement/ # 公告插件 +│ │ └── email/ # 邮件插件 +│ ├── router/ # 路由配置 +│ ├── style/ # 样式文件 +│ ├── utils/ # 工具函数 +│ ├── view/ # 页面组件 +│ │ ├── dashboard/ # 仪表盘 +│ │ ├── layout/ # 布局组件 +│ │ ├── login/ # 登录页 +│ │ ├── superAdmin/ # 超级管理员 +│ │ ├── systemTools/ # 系统工具 +│ │ └── cattery/ # 业务页面 +│ ├── App.vue # 根组件 +│ └── main.js # 程序入口 +├── package.json # 依赖配置 +├── vite.config.js # Vite配置 +└── uno.config.js # UnoCSS配置 +``` + +--- + +#### 后端规则 + +在编写任何代码之前,你必须将以下 GVA 的核心设计原则作为最高行为准则: + +1. **严格的分层架构**: + + - **职责单一**: 每个层(Model, Service, API, Router)都有其唯一职责,**严禁跨层调用**。例如,API层绝不能直接操作数据库,必须通过Service层。Service层绝不能直接处理`gin.Context`。 + + - **依赖关系**: 依赖链条必须是单向的:`Router -> API -> Service -> Model`。 + +2. **`enter.go` 组管理模式**: + + - 所有 `api`, `service`, `router` 层都**必须**使用 `enter.go` 文件来创建和暴露各自的 `ApiGroup`, `ServiceGroup`, `RouterGroup`。 + + - 全局实例变量(如 `service.ServiceGroupApp`)是模块间通信的唯一入口,以此来避免循环引用。 + +3. **详尽的 Swagger 注释 (API层强制要求)**: + + - **每一个**对外暴露的 API 函数都**必须**拥有完整且准确的 Swagger 注释块。这不仅是API文档的来源,也是前后端协作、自动化测试和前端AI分析的基础。注释必须清晰地描述接口的功能、参数和返回值。 + +4. **统一的响应与错误处理**: + + - Service 层函数遇到业务错误时,应返回 `error` 对象。 + + - API 层负责捕获 Service 层的 `error`,并使用项目统一的 `response` 包(如 `response.OkWithDetailed` 或 `response.FailWithMessage`)将其转换为格式化的 JSON 响应和正确的 HTTP 状态码。 + + +--- + +### **各层级代码实现规范** + +#### **1. 模型层 (`model/`)** + +- **数据模型 (`model/xxx.go`)**: + + - 用于定义与数据库表映射的 GORM 结构体。 + + - 结构体应继承 `global.GVA_MODEL` 以包含 `ID`, `CreatedAt`, `UpdatedAt` 等基础字段。 + + - 必须为字段添加清晰的 `json` 和 `gorm` 标签。 + + - **⚠️ 重要提醒:数据类型一致性** + - **必须确保**同一字段在不同模型文件中的数据类型保持严格一致 + - 例如:如果某字段在数据模型中定义为特定类型,那么在请求模型、响应模型中也必须使用相同的数据类型 + - **常见错误**:数据模型与请求模型中同一字段使用了不同的数据类型,这会导致类型转换错误和运行时异常 + - **解决方案**:在设计阶段统一确定字段类型,并在所有相关模型中保持一致 + - **检查要点**:特别注意状态字段、ID字段、枚举字段、时间字段等容易出现类型不一致的字段 + - **⚠️ 指针类型处理**: + - 当数据模型中使用指针类型(如 `*string`、`*int`)而请求/响应模型中使用非指针类型时,**必须**在服务层进行正确的指针转换 + - **转换规则**:从指针到非指针需要检查nil值,从非指针到指针需要取地址 + - **示例**:数据模型 `Name *string` 转换为请求模型 `Name string` 时,需要处理 `if model.Name != nil { request.Name = *model.Name }` + +- **请求模型 (`model/request/xxx.go`)**: + + - 用于定义接收前端请求参数的结构体(DTOs)。 + + - **必须**为字段添加 `json` 和 `form` 标签,以便 Gin 进行参数绑定。 + + - 对于列表查询请求,应创建一个 `XxxSearch` 结构体,并内嵌通用的 `request.PageInfo` 分页结构体。 + + +#### **2. 服务层 (`service/`)** + +- **职责**: 封装所有核心业务逻辑,进行数据库的CRUD操作。**此层不应出现任何与HTTP协议相关的代码(如 `gin.Context`)**。 + +- **结构**: 在 `service/` 下为每个模块创建 `xxx_service.go` 文件,并在 `service/enter.go` 中注册。 + +- **函数签名**: 函数应接收具体的业务参数(如 `model.Xxx` 或 `request.XxxSearch`),并返回处理结果和 `error`。 + +- **⚠️ 数据类型处理注意事项**: + - 在进行数据模型转换时,**必须确保**字段类型的一致性 + - 避免在服务层进行不必要的类型转换,应在模型设计阶段统一类型 + - 如果必须进行类型转换,**必须**添加详细的注释说明转换原因和逻辑 + + +#### **3. API层 (`api/`)** + +- **职责**: 作为HTTP请求的入口,负责参数校验、调用Service层方法、并返回格式化的JSON响应。 + +- **结构**: 在 `api/` 下为每个模块创建 `xxx_api.go` 文件,并在 `api/enter.go` 中注册。 + +- **交互**: **必须**通过全局变量 `service.ServiceGroupApp` 来调用服务层的方法。 + +- **Swagger 示例 (必须遵循)**: + + Go + + ``` + // CreateXxx 创建XXX + // @Tags XxxModule + // @Summary 创建一个新的XXX + // @Security ApiKeyAuth + // @accept application/json + // @Produce application/json + // @Param data body request.CreateXxxRequest true "XXX的名称和描述" + // @Success 200 {object} response.Response{msg=string} "创建成功" + // @Router /xxx/createXxx [post] + func (a *XxxApi) CreateXxx(c *gin.Context) { + // ... + } + ``` + + +#### **4. 路由层 (`router/`)** + +- **职责**: 定义API路由规则,并将HTTP请求路径映射到具体的API处理函数上,同时配置中间件。 + +- **结构**: 在 `router/` 下为每个模块创建 `xxx_router.go` 文件,并在 `router/enter.go` 中注册。 + +- **交互**: **必须**通过全局变量 `api.ApiGroupApp` 来引用API层的处理函数。 + +- **路由分组**: 应根据业务需求和权限,合理使用路由组 (`Router.Group()`),并挂载不同的中间件(如鉴权、操作记录等)。 + +#### **5. 初始化层 (`initialize/`)** + +- **职责**: 提供插件资源(数据库、路由、菜单等)的初始化入口,供主程序调用。 + +- **`gorm.go`**: 实现 `InitializeDB` 函数,**必须**调用 `db.AutoMigrate` 自动迁移本插件所有 `model` 的表结构。 + +- **`router.go`**: 实现 `InitializeRouter` 函数,**必须**调用 `router.RouterGroupApp` 中本插件路由的初始化方法,注册所有API路由。 + +- **`menu.go`**: 实现 `InitializeMenu` 函数,负责在数据库中创建或更新本插件的侧边栏菜单、按钮和对应的API权限。 +- viper.go: 加载插件配置文件 +- api.go: 注册API到系统 + + +#### **6. 插件入口 (`plugin.go`) + +- **职责**: 作为插件的唯一入口,实现 GVA 的插件接口,让框架能够识别和加载本插件。 + +- **接口实现**: **必须**定义一个结构体并实现 `system.Plugin` 接口。 + +- **`Register`方法**: 实现 `Register` 方法,该方法接收一个 `*gin.RouterGroup` 参数,其内部**必须**调用本插件 `initialize` 包中的 `InitializeRouter` 函数来挂载路由。 + +- **`RouterPath`方法**: 实现 `RouterPath` 方法,返回该插件所有API的根路径,例如 `"/myPlugin"`。 + +### 模块间引用关系: +- API层引用Service层:在API文件中定义变量如 `var xxxService = service.ServiceGroupApp.XxxService` +- Router层引用API层:在路由函数中使用 `api.ApiGroupApp.XxxApi.XxxMethod` +- Initialize/Router引用Router层:通过 `router.RouterGroupApp.XxxRouter.InitXxxRouter` +- 各模块通过enter.go文件组织和暴露功能,避免循环引用 + +### 代码组织示例: + +1. Service入口 (service/enter.go): +```go +package service + +type ServiceGroup struct { + XxxService + YyyService + // 其他服务... +} + +var ServiceGroupApp = new(ServiceGroup) +``` + +2. API入口 (api/enter.go): +```go +package api + +type ApiGroup struct { + XxxApi + YyyApi + // 其他API... +} + +var ApiGroupApp = new(ApiGroup) +``` + +3. Router入口 (router/enter.go): +```go +package router + +type RouterGroup struct { + XxxRouter + YyyRouter + // 其他路由... +} + +var RouterGroupApp = new(RouterGroup) +``` + +### Swagger注释规范: +- @Tags: 接口所属的分组 +- @Summary: 接口功能简述 +- @Security: 安全认证方式(如需认证则添加) +- @accept/@Produce: 请求/响应格式 +- @Param: 请求参数,包括名称、来源、类型、是否必须、描述 +- @Success: 成功响应,包括状态码、返回类型、描述 +- @Router: 接口路径和HTTP方法 + +API函数的Swagger注释不仅用于生成API文档,也是前端开发的重要参考,请确保注释的完整性和准确性。 + + +--- + +### **开发工作流** + +1. **接收任务**: 我会向你下达一个具体的功能插件开发任务,例如:“请为项目创建一个‘商品管理 (Product)’插件”。 + +2. **【第一步】模型设计 (奠定基础)**: + + - 你的**首要行动**是分析需求,设计并提供 `model` 和 `model/request` 下的所有 Go 结构体定义。这是后续所有开发的基础。 + +3. **【第二步】自下而上,分层实现**: + - 具体项目结构可以参考:server/plugin/announcement 这个插件,非常经典! + + - 在模型确认后,你将按照 `Service -> API -> Router` 的顺序,逐层生成代码。 + + - 确保每一层的代码都完整、健壮,并严格遵守上述规范。 + +4. **【第三步】插件初始化与注册**: + + - 在完成核心功能层的代码后,你将生成 `initialize/` 目录下的相关初始化文件(如 `db.go`, `router.go`)以及插件的主入口文件 `plugin.go`。 + +5. **【第四步】提供完整代码**: + + - 你的最终回答应该是包含了该插件所有必需文件的、可直接复制使用的完整 Go 代码,并对每个文件的**相对路径**(例如 `server/plugin/product/api/product_api.go`)和用途进行清晰的说明。 + + +--- + +## **前端开发规范** + +### **角色与目标** + +你是一名资深的 Vue.js 前端开发专家,**专精于 `gin-vue-admin` (GVA) 框架的前端架构与开发范式**。 + +你的核心任务是,根据需求开发**完整、生产级别的前端功能模块或插件**。你必须严格遵循 GVA 的前端架构、代码规范和核心设计模式,确保你生成的每一部分代码都能无缝集成到现有项目中。 + +### **核心开发指令:绝不可违背的原则** + +#### 前端规则 + +在编写任何前端代码之前,你必须将以下 GVA 的核心设计原则作为最高行为准则: + +1. **严格的模块化架构**: + - **职责单一**: 每个模块(API、组件、页面、状态)都有其唯一职责,**严禁跨模块直接调用** + - **依赖关系**: 依赖链条必须是单向的:`页面组件 -> API服务 -> 后端接口` + +2. **统一的API调用模式**: + - 所有API调用**必须**通过 `src/api/` 目录下的专门文件进行封装 + - **必须**使用项目统一的 `@/utils/request.js` 进行HTTP请求 + - API函数**必须**包含完整的JSDoc注释,描述接口功能、参数和返回值 + +3. **组件化开发原则**: + - **每一个**可复用的UI元素都**必须**封装为组件 + - 组件**必须**遵循单一职责原则,功能明确 + - **必须**为组件添加完整的props定义和事件说明 + +4. **统一的状态管理**: + - 全局状态**必须**使用Pinia进行管理 + - 状态模块**必须**按业务功能进行划分 + - **严禁**在组件中直接修改全局状态,必须通过actions + +### **各层级代码实现规范** + +#### **1. API层 (`src/api/`)** + +- **职责**: 封装所有后端API调用,提供统一的接口服务 +- **结构**: 按业务模块创建API文件,如 `user.js`、`menu.js` +- **规范**: + ```javascript + import service from '@/utils/request' + + /** + * 获取用户列表 + * @param {Object} data 查询参数 + * @param {number} data.page 页码 + * @param {number} data.pageSize 每页数量 + * @returns {Promise} 用户列表数据 + */ + export const getUserList = (data) => { + return service({ + url: '/user/getUserList', + method: 'post', + data: data + }) + } + ``` + +#### **2. 组件层 (`src/components/`)** + +- **职责**: 提供可复用的UI组件 +- **结构**: 按功能分类组织,每个组件一个文件夹 +- **规范**: + ```vue + + + + ``` + +#### **3. 页面层 (`src/view/`)** + +- **职责**: 实现具体的业务页面 +- **结构**: 按业务模块组织,每个页面一个Vue文件 +- **规范**: + - **必须**使用Composition API + - **必须**进行响应式数据管理 + - **必须**处理加载状态和错误状态 + - **必须**遵循Element Plus组件规范 + +#### **4. 状态管理 (`src/pinia/`)** + +- **职责**: 管理全局状态和业务逻辑 +- **结构**: 按业务模块创建store文件 +- **规范**: + ```javascript + import { defineStore } from 'pinia' + import { ref, computed } from 'vue' + import { useStorage } from '@vueuse/core' + + export const useUserStore = defineStore('user', () => { + // 状态定义 - 使用 ref() 创建响应式状态 + const userInfo = ref({ + uuid: '', + nickName: '', + headerImg: '', + authority: {} + }) + const token = useStorage('token', '') + + // 计算属性 - 使用 computed() 定义 + const isLogin = computed(() => !!token.value) + + // 方法定义 - 直接定义函数作为 actions + const setUserInfo = (val) => { + userInfo.value = val + } + + const setToken = (val) => { + token.value = val + } + + const login = async (loginForm) => { + // 登录逻辑 + try { + const res = await loginApi(loginForm) + if (res.code === 0) { + setUserInfo(res.data.user) + setToken(res.data.token) + return true + } + return false + } catch (error) { + console.error('Login error:', error) + return false + } + } + + const logout = async () => { + // 登出逻辑 + token.value = '' + userInfo.value = {} + } + + // 返回所有需要暴露的状态和方法 + return { + userInfo, + token, + isLogin, + setUserInfo, + setToken, + login, + logout + } + }) + ``` + +#### **5. 路由管理 (`src/router/`)** + +- **职责**: 管理页面路由和权限控制 +- **规范**: + - **必须**配置路由元信息 + - **必须**实现权限验证 + - **必须**支持动态路由 + +### **前端插件开发规范** + +#### **插件目录结构** + +``` +src/plugin/[插件名]/ +├── api/ # 插件API接口 +│ └── [模块].js +├── components/ # 插件组件(可选) +│ └── [组件名].vue +├── view/ # 插件页面 +│ └── [页面名].vue +├── form/ # 插件表单(可选) +│ └── [表单名].vue +└── index.js # 插件入口文件(可选) +``` + +#### **插件开发原则** + +1. **独立性**: 插件应该是自包含的,不依赖其他业务模块 +2. **可配置性**: 插件应该支持配置化,便于定制 +3. **可扩展性**: 插件应该预留扩展接口 +4. **一致性**: 插件UI风格应与主系统保持一致 + +### **代码质量要求** + +1. **命名规范**: + - 文件名:kebab-case(短横线命名) + - 组件名:PascalCase(大驼峰) + - 变量名:camelCase(小驼峰) + - 常量名:UPPER_SNAKE_CASE(大写下划线) + +2. **注释规范**: + - **必须**为所有API函数添加JSDoc注释 + - **必须**为复杂组件添加功能说明 + - **必须**为关键业务逻辑添加行内注释 + +3. **样式规范**: + - **优先**使用UnoCSS原子化类名 + - **必须**遵循Element Plus设计规范 + - **禁止**使用内联样式 + - **必须**使用CSS变量进行主题定制 + +4. **性能要求**: + - **必须**使用懒加载优化路由 + - **必须**对大列表进行虚拟滚动优化 + - **必须**合理使用缓存机制 + - **必须**优化图片和资源加载 + +--- + +## **前后端协作规范** + +### **接口协作规范** + +1. **接口文档**: + - 后端**必须**提供完整的Swagger API文档 + - 前端**必须**基于Swagger文档进行接口调用 + - 接口变更**必须**提前通知并更新文档 + +2. **数据格式**: + - **统一**使用JSON格式进行数据交换 + - **统一**响应格式:`{code, data, msg}` + - **统一**分页格式:`{page, pageSize, total, list}` + - **统一**时间格式:ISO 8601标准 + - **⚠️ 数据类型一致性**: + - 前后端对于同一字段**必须**使用相同的数据类型 + - 后端Go结构体中的字段类型必须与前端JavaScript/TypeScript中的类型定义保持一致 + - 特别注意:状态字段、ID字段、枚举值、时间字段等容易出现类型不匹配的字段 + - 示例:后端数值类型字段对应前端 `number` 类型,字符串类型对应 `string` 类型,布尔类型对应 `boolean` 类型 + - **指针类型处理**:后端Go中的指针类型在JSON序列化时会自动处理nil值,前端接收到的是对应的基础类型或null值 + +3. **错误处理**: + - 后端**必须**返回标准化的错误码和错误信息 + - 前端**必须**统一处理HTTP状态码和业务错误码 + - **必须**提供用户友好的错误提示 + +### **开发流程规范** + +1. **需求分析阶段**: + - 确定功能需求和接口设计 + - 定义数据模型和业务流程 + - 制定前后端开发计划 + +2. **开发阶段**: + - 后端优先开发API接口 + - 前端基于Mock数据进行并行开发 + - 定期进行接口联调测试 + +3. **测试阶段**: + - 单元测试:前后端各自负责 + - 集成测试:前后端协作完成 + - 用户验收测试:产品团队主导 + +### **版本管理规范** + +1. **分支策略**: + - `main`:生产环境分支 + - `develop`:开发环境分支 + - `feature/*`:功能开发分支 + - `hotfix/*`:紧急修复分支 + +2. **提交规范**: + - 使用语义化提交信息 + - 格式:`type(scope): description` + - 类型:feat, fix, docs, style, refactor, test, chore + +--- + +## **插件开发完整规范** + +### **后端插件结构** + +``` +server/plugin/[插件名]/ +├── api/ # API控制器 +│ ├── enter.go # API组入口 +│ └── [模块].go # 具体API实现 +├── config/ # 插件配置 +│ └── config.go +├── initialize/ # 初始化模块 +│ ├── api.go # API注册 +│ ├── gorm.go # 数据库初始化 +│ ├── menu.go # 菜单初始化 +│ ├── router.go # 路由初始化 +│ └── viper.go # 配置初始化 +├── model/ # 数据模型 +│ ├── [模型].go # 数据库模型 +│ └── request/ # 请求模型 +├── router/ # 路由定义 +│ ├── enter.go # 路由组入口 +│ └── [模块].go # 具体路由 +├── service/ # 业务服务 +│ ├── enter.go # 服务组入口 +│ └── [模块].go # 具体服务 +└── plugin.go # 插件入口 +``` + +### **前端插件结构** + +``` +web/src/plugin/[插件名]/ +├── api/ # API接口 +│ └── [模块].js +├── components/ # 插件组件 +│ └── [组件].vue +├── view/ # 插件页面 +│ └── [页面].vue +├── form/ # 表单组件 +│ └── [表单].vue +└── config.js # 插件配置 +``` + +### **插件开发工作流** + +1. **【第一步】需求分析**: + - 明确插件功能和业务需求 + - 设计数据模型和接口规范 + - 规划前端页面和交互流程 + +2. **【第二步】后端开发**: + - 创建数据模型和请求模型 + - 实现服务层业务逻辑 + - 开发API控制器和路由 + - 编写初始化和配置代码 + +3. **【第三步】前端开发**: + - 创建API接口封装 + - 开发页面组件和表单 + - 实现业务逻辑和状态管理 + - 集成到主系统菜单 + +4. **【第四步】测试集成**: + - 单元测试和集成测试 + - 前后端联调测试 + - 用户体验测试 + - 性能和安全测试 + +### **插件质量标准** + +1. **功能完整性**: 插件功能完整,满足业务需求 +2. **代码质量**: 代码规范,注释完整,易于维护 +3. **数据类型一致性**: 前后端数据模型字段类型保持严格一致,避免类型转换错误 +4. **性能表现**: 响应速度快,资源占用合理 +5. **用户体验**: 界面友好,操作流畅,错误处理完善 +6. **兼容性**: 与主系统兼容,不影响其他功能 +7. **安全性**: 数据安全,权限控制,防止安全漏洞 + +--- + +### **建议和方案** + +基于以上规范,建议AI在开发gin-vue-admin项目时: + +1. **严格遵循分层架构**:确保前后端代码都按照规定的层次结构组织 +2. **保持代码一致性**:使用统一的命名规范、注释格式和代码风格 +3. **注重文档完整性**:确保API文档、代码注释和使用说明的完整性 +4. **优化用户体验**:关注页面加载速度、交互流畅性和错误处理 +5. **考虑扩展性**:设计时预留扩展接口,便于后续功能增强 +6. **重视安全性**:实现完善的权限控制和数据验证机制 \ No newline at end of file diff --git a/README.md b/README.md index d5ca7e24..6eea7bbd 100644 --- a/README.md +++ b/README.md @@ -22,7 +22,8 @@ -
+ +

⭐️ 高度适配AI编辑器的MCP

📄 创建基础模板

🤖 AI生成结构

⏰ 生成代码

diff --git a/server/api/v1/system/sys_user.go b/server/api/v1/system/sys_user.go index aae263e3..9169f06c 100644 --- a/server/api/v1/system/sys_user.go +++ b/server/api/v1/system/sys_user.go @@ -27,8 +27,6 @@ import ( func (b *BaseApi) Login(c *gin.Context) { var l systemReq.Login err := c.ShouldBindJSON(&l) - key := c.ClientIP() - if err != nil { response.FailWithMessage(err.Error(), c) return @@ -39,6 +37,7 @@ func (b *BaseApi) Login(c *gin.Context) { return } + key := c.ClientIP() // 判断验证码是否开启 openCaptcha := global.GVA_CONFIG.Captcha.OpenCaptcha // 是否开启防爆次数 openCaptchaTimeOut := global.GVA_CONFIG.Captcha.OpenCaptchaTimeOut // 缓存超时时间 @@ -48,30 +47,30 @@ func (b *BaseApi) Login(c *gin.Context) { } var oc bool = openCaptcha == 0 || openCaptcha < interfaceToInt(v) - - if !oc || (l.CaptchaId != "" && l.Captcha != "" && store.Verify(l.CaptchaId, l.Captcha, true)) { - u := &system.SysUser{Username: l.Username, Password: l.Password} - user, err := userService.Login(u) - if err != nil { - global.GVA_LOG.Error("登陆失败! 用户名不存在或者密码错误!", zap.Error(err)) - // 验证码次数+1 - global.BlackCache.Increment(key, 1) - response.FailWithMessage("用户名不存在或者密码错误", c) - return - } - if user.Enable != 1 { - global.GVA_LOG.Error("登陆失败! 用户被禁止登录!") - // 验证码次数+1 - global.BlackCache.Increment(key, 1) - response.FailWithMessage("用户被禁止登录", c) - return - } - b.TokenNext(c, *user) + if oc && (l.Captcha == "" || l.CaptchaId == "" || !store.Verify(l.CaptchaId, l.Captcha, true)) { + // 验证码次数+1 + global.BlackCache.Increment(key, 1) + response.FailWithMessage("验证码错误", c) return } - // 验证码次数+1 - global.BlackCache.Increment(key, 1) - response.FailWithMessage("验证码错误", c) + + u := &system.SysUser{Username: l.Username, Password: l.Password} + user, err := userService.Login(u) + if err != nil { + global.GVA_LOG.Error("登陆失败! 用户名不存在或者密码错误!", zap.Error(err)) + // 验证码次数+1 + global.BlackCache.Increment(key, 1) + response.FailWithMessage("用户名不存在或者密码错误", c) + return + } + if user.Enable != 1 { + global.GVA_LOG.Error("登陆失败! 用户被禁止登录!") + // 验证码次数+1 + global.BlackCache.Increment(key, 1) + response.FailWithMessage("用户被禁止登录", c) + return + } + b.TokenNext(c, *user) } // TokenNext 登录以后签发jwt diff --git a/server/api/v1/system/sys_version.go b/server/api/v1/system/sys_version.go index 7f441c7c..1e228089 100644 --- a/server/api/v1/system/sys_version.go +++ b/server/api/v1/system/sys_version.go @@ -267,6 +267,17 @@ func (sysVersionApi *SysVersionApi) ExportVersion(c *gin.Context) { } } + // 获取选中的字典数据 + var dictData []system.SysDictionary + if len(req.DictIds) > 0 { + dictData, err = sysVersionService.GetDictionariesByIds(ctx, req.DictIds) + if err != nil { + global.GVA_LOG.Error("获取字典数据失败!", zap.Error(err)) + response.FailWithMessage("获取字典数据失败:"+err.Error(), c) + return + } + } + // 处理菜单数据,构建递归的children结构 processedMenus := buildMenuTree(menuData) @@ -282,6 +293,34 @@ func (sysVersionApi *SysVersionApi) ExportVersion(c *gin.Context) { processedApis = append(processedApis, cleanApi) } + // 处理字典数据,清除ID和时间戳字段,包含字典详情 + processedDicts := make([]system.SysDictionary, 0, len(dictData)) + for _, dict := range dictData { + cleanDict := system.SysDictionary{ + Name: dict.Name, + Type: dict.Type, + Status: dict.Status, + Desc: dict.Desc, + } + + // 处理字典详情数据,清除ID和时间戳字段 + cleanDetails := make([]system.SysDictionaryDetail, 0, len(dict.SysDictionaryDetails)) + for _, detail := range dict.SysDictionaryDetails { + cleanDetail := system.SysDictionaryDetail{ + Label: detail.Label, + Value: detail.Value, + Extend: detail.Extend, + Status: detail.Status, + Sort: detail.Sort, + // 不复制 ID, CreatedAt, UpdatedAt, SysDictionaryID + } + cleanDetails = append(cleanDetails, cleanDetail) + } + cleanDict.SysDictionaryDetails = cleanDetails + + processedDicts = append(processedDicts, cleanDict) + } + // 构建导出数据 exportData := systemRes.ExportVersionResponse{ Version: systemReq.VersionInfo{ @@ -290,8 +329,9 @@ func (sysVersionApi *SysVersionApi) ExportVersion(c *gin.Context) { Description: req.Description, ExportTime: time.Now().Format("2006-01-02 15:04:05"), }, - Menus: processedMenus, - Apis: processedApis, + Menus: processedMenus, + Apis: processedApis, + Dictionaries: processedDicts, } // 转换为JSON @@ -418,6 +458,15 @@ func (sysVersionApi *SysVersionApi) ImportVersion(c *gin.Context) { } } + // 导入字典数据 + if len(importData.ExportDictionary) > 0 { + if err := sysVersionService.ImportDictionaries(importData.ExportDictionary); err != nil { + global.GVA_LOG.Error("导入字典失败!", zap.Error(err)) + response.FailWithMessage("导入字典失败: "+err.Error(), c) + return + } + } + // 创建导入记录 jsonData, _ := json.Marshal(importData) version := system.SysVersion{ diff --git a/server/config.docker.yaml b/server/config.docker.yaml index a7783716..662c5cad 100644 --- a/server/config.docker.yaml +++ b/server/config.docker.yaml @@ -2,107 +2,103 @@ # jwt configuration jwt: - signing-key: qmPlus - expires-time: 7d - buffer-time: 1d - issuer: qmPlus + signing-key: qmPlus + expires-time: 7d + buffer-time: 1d + issuer: qmPlus # zap logger configuration zap: - level: info - format: console - prefix: "[github.com/flipped-aurora/gin-vue-admin/server]" - director: log - show-line: true - encode-level: LowercaseColorLevelEncoder - stacktrace-key: stacktrace - log-in-console: true + level: info + format: console + prefix: "[github.com/flipped-aurora/gin-vue-admin/server]" + director: log + show-line: true + encode-level: LowercaseColorLevelEncoder + stacktrace-key: stacktrace + log-in-console: true + retention-day: -1 # redis configuration redis: - db: 0 - addr: 177.7.0.14:6379 - password: "" + #是否使用redis集群模式 + useCluster: false + #使用集群模式addr和db默认无效 + addr: 177.7.0.14:6379 + password: "" + db: 0 + clusterAddrs: + - "177.7.0.14:7000" + - "177.7.0.15:7001" + - "177.7.0.13:7002" + +# redis-list configuration +redis-list: + - name: cache # 数据库的名称,注意: name 需要在 redis-list 中唯一 + useCluster: false # 是否使用redis集群模式 + addr: 177.7.0.14:6379 # 使用集群模式addr和db默认无效 + password: "" + db: 0 + clusterAddrs: + - "177.7.0.14:7000" + - "177.7.0.15:7001" + - "177.7.0.13:7002" # mongo configuration mongo: - coll: '' - options: '' - database: '' - username: '' - password: '' - min-pool-size: 0 - max-pool-size: 100 - socket-timeout-ms: 0 - connect-timeout-ms: 0 - is-zap: false - hosts: - - host: '' - port: '' + coll: '' + options: '' + database: '' + username: '' + password: '' + auth-source: '' + min-pool-size: 0 + max-pool-size: 100 + socket-timeout-ms: 0 + connect-timeout-ms: 0 + is-zap: false + hosts: + - host: '' + port: '' # email configuration email: - to: xxx@qq.com - port: 465 - from: xxx@163.com - host: smtp.163.com - is-ssl: true - secret: xxx - nickname: test + to: xxx@qq.com + port: 465 + from: xxx@163.com + host: smtp.163.com + is-ssl: true + secret: xxx + nickname: test # system configuration system: - env: public # Change to "develop" to skip authentication for development mode - addr: 8888 - db-type: mysql - oss-type: local # 控制oss选择走本地还是 七牛等其他仓 自行增加其他oss仓可以在 server/utils/upload/upload.go 中 NewOss函数配置 - use-redis: false # 使用redis - use-mongo: false # 使用mongo - use-multipoint: false - # IP限制次数 一个小时15000次 - iplimit-count: 15000 - # IP限制一个小时 - iplimit-time: 3600 + env: local # 修改为public可以关闭路由日志输出 + addr: 8888 + db-type: mysql + oss-type: local # 控制oss选择走本地还是 七牛等其他仓 自行增加其他oss仓可以在 server/utils/upload/upload.go 中 NewOss函数配置 + use-redis: false # 使用redis + use-mongo: false # 使用mongo + use-multipoint: false + # IP限制次数 一个小时15000次 + iplimit-count: 15000 + # IP限制一个小时 + iplimit-time: 3600 + # 路由全局前缀 + router-prefix: "" + # 严格角色模式 打开后权限将会存在上下级关系 + use-strict-auth: false # captcha configuration captcha: - key-long: 6 - img-width: 240 - img-height: 80 - open-captcha: 0 # 0代表一直开启,大于0代表限制次数 - open-captcha-timeout: 3600 # open-captcha大于0时才生效 + key-long: 6 + img-width: 240 + img-height: 80 + open-captcha: 0 # 0代表一直开启,大于0代表限制次数 + open-captcha-timeout: 3600 # open-captcha大于0时才生效 # mysql connect configuration # 未初始化之前请勿手动修改数据库信息!!!如果一定要手动初始化请看(https://gin-vue-admin.com/docs/first_master) mysql: - path: "" - port: "" - config: "" - db-name: "" - username: "" - password: "" - max-idle-conns: 10 - max-open-conns: 100 - log-mode: "" - log-zap: false - -# pgsql connect configuration -# 未初始化之前请勿手动修改数据库信息!!!如果一定要手动初始化请看(https://gin-vue-admin.com/docs/first_master) -pgsql: - path: "" - port: "" - config: "" - db-name: "" - username: "" - password: "" - max-idle-conns: 10 - max-open-conns: 100 - log-mode: "" - log-zap: false - -db-list: - - disable: true # 是否禁用 - type: "" # 数据库的类型,目前支持mysql、pgsql - alias-name: "" # 数据库的名称,注意: alias-name 需要在db-list中唯一 path: "" port: "" config: "" @@ -114,107 +110,174 @@ db-list: log-mode: "" log-zap: false +# pgsql connect configuration +# 未初始化之前请勿手动修改数据库信息!!!如果一定要手动初始化请看(https://gin-vue-admin.com/docs/first_master) +pgsql: + path: "" + port: "" + config: "" + db-name: "" + username: "" + password: "" + max-idle-conns: 10 + max-open-conns: 100 + log-mode: "" + log-zap: false +oracle: + path: "" + port: "" + config: "" + db-name: "" + username: "" + password: "" + max-idle-conns: 10 + max-open-conns: 100 + log-mode: "" + log-zap: false +mssql: + path: "" + port: "" + config: "" + db-name: "" + username: "" + password: "" + max-idle-conns: 10 + max-open-conns: 100 + log-mode: "" + log-zap: false +sqlite: + path: "" + port: "" + config: "" + db-name: "" + username: "" + password: "" + max-idle-conns: 10 + max-open-conns: 100 + log-mode: "" + log-zap: false +db-list: + - disable: true # 是否禁用 + type: "" # 数据库的类型,目前支持mysql、pgsql、mssql、oracle + alias-name: "" # 数据库的名称,注意: alias-name 需要在db-list中唯一 + path: "" + port: "" + config: "" + db-name: "" + username: "" + password: "" + max-idle-conns: 10 + max-open-conns: 100 + log-mode: "" + log-zap: false # local configuration local: - path: uploads/file - store-path: uploads/file + path: uploads/file + store-path: uploads/file # autocode configuration autocode: - transfer-restart: true - # root 自动适配项目根目录 - # 请不要手动配置,他会在项目加载的时候识别出根路径 - root: "" - server: /server - server-plug: /plugin/%s - server-api: /api/v1/%s - server-initialize: /initialize - server-model: /model/%s - server-request: /model/%s/request/ - server-router: /router/%s - server-service: /service/%s - web: /web/src - web-api: /api - web-form: /view - web-table: /view + web: web/src + root: "" # root 自动适配项目根目录, 请不要手动配置,他会在项目加载的时候识别出根路径 + server: server + module: 'github.com/flipped-aurora/gin-vue-admin/server' + ai-path: "" # AI服务路径 # qiniu configuration (请自行七牛申请对应的 公钥 私钥 bucket 和 域名地址) qiniu: - zone: ZoneHuaDong - bucket: "" - img-path: "" - use-https: false - access-key: "" - secret-key: "" - use-cdn-domains: false + zone: ZoneHuaDong + bucket: "" + img-path: "" + use-https: false + access-key: "" + secret-key: "" + use-cdn-domains: false + +# minio oss configuration +minio: + endpoint: yourEndpoint + access-key-id: yourAccessKeyId + access-key-secret: yourAccessKeySecret + bucket-name: yourBucketName + use-ssl: false + base-path: "" + bucket-url: "http://host:9000/yourBucketName" # aliyun oss configuration aliyun-oss: - endpoint: yourEndpoint - access-key-id: yourAccessKeyId - access-key-secret: yourAccessKeySecret - bucket-name: yourBucketName - bucket-url: yourBucketUrl - base-path: yourBasePath + endpoint: yourEndpoint + access-key-id: yourAccessKeyId + access-key-secret: yourAccessKeySecret + bucket-name: yourBucketName + bucket-url: yourBucketUrl + base-path: yourBasePath # tencent cos configuration tencent-cos: - bucket: xxxxx-10005608 - region: ap-shanghai - secret-id: your-secret-id - secret-key: your-secret-key - base-url: https://gin.vue.admin - path-prefix: github.com/flipped-aurora/gin-vue-admin/server + bucket: xxxxx-10005608 + region: ap-shanghai + secret-id: your-secret-id + secret-key: your-secret-key + base-url: https://gin.vue.admin + path-prefix: github.com/flipped-aurora/gin-vue-admin/server # aws s3 configuration (minio compatible) aws-s3: - bucket: xxxxx-10005608 - region: ap-shanghai - endpoint: "" - s3-force-path-style: false - disable-ssl: false - secret-id: your-secret-id - secret-key: your-secret-key - base-url: https://gin.vue.admin - path-prefix: github.com/flipped-aurora/gin-vue-admin/server + bucket: xxxxx-10005608 + region: ap-shanghai + endpoint: "" + s3-force-path-style: false + disable-ssl: false + secret-id: your-secret-id + secret-key: your-secret-key + base-url: https://gin.vue.admin + path-prefix: github.com/flipped-aurora/gin-vue-admin/server + +# cloudflare r2 configuration +cloudflare-r2: + bucket: xxxx0bucket + base-url: https://gin.vue.admin.com + path: uploads + account-id: xxx_account_id + access-key-id: xxx_key_id + secret-access-key: xxx_secret_key # huawei obs configuration hua-wei-obs: - path: you-path - bucket: you-bucket - endpoint: you-endpoint - access-key: you-access-key - secret-key: you-secret-key + path: you-path + bucket: you-bucket + endpoint: you-endpoint + access-key: you-access-key + secret-key: you-secret-key # excel configuration excel: - dir: ./resource/excel/ + dir: ./resource/excel/ -# timer task db clear table -Timer: - start: true - spec: "@daily" # 定时任务详细配置参考 https://pkg.go.dev/github.com/robfig/cron/v3 - detail: - - tableName: sys_operation_records - compareField: created_at - interval: 2160h - - tableName: jwt_blacklists - compareField: created_at - interval: 168h +# disk usage configuration +disk-list: + - mount-point: "/" # 跨域配置 # 需要配合 server/initialize/router.go -> `Router.Use(middleware.CorsByRules())` 使用 cors: - mode: whitelist # 放行模式: allow-all, 放行全部; whitelist, 白名单模式, 来自白名单内域名的请求添加 cors 头; strict-whitelist 严格白名单模式, 白名单外的请求一律拒绝 - whitelist: - - allow-origin: example1.com - allow-headers: content-type - allow-methods: GET, POST - expose-headers: Content-Length, Access-Control-Allow-Origin, Access-Control-Allow-Headers, Content-Type - allow-credentials: true # 布尔值 - - allow-origin: example2.com - allow-headers: content-type - allow-methods: GET, POST - expose-headers: Content-Length, Access-Control-Allow-Origin, Access-Control-Allow-Headers, Content-Type - allow-credentials: true # 布尔值 \ No newline at end of file + mode: strict-whitelist # 放行模式: allow-all, 放行全部; whitelist, 白名单模式, 来自白名单内域名的请求添加 cors 头; strict-whitelist 严格白名单模式, 白名单外的请求一律拒绝 + whitelist: + - allow-origin: example1.com + allow-headers: Content-Type,AccessToken,X-CSRF-Token, Authorization, Token,X-Token,X-User-Id + allow-methods: POST, GET + expose-headers: Content-Length, Access-Control-Allow-Origin, Access-Control-Allow-Headers, Content-Type + + allow-credentials: true # 布尔值 + - allow-origin: example2.com + allow-headers: content-type + allow-methods: GET, POST + expose-headers: Content-Length, Access-Control-Allow-Origin, Access-Control-Allow-Headers, Content-Type + allow-credentials: true # 布尔值 +mcp: + name: GVA_MCP + version: v1.0.0 + sse_path: /sse + message_path: /message + url_prefix: '' \ No newline at end of file diff --git a/server/config/gorm_oracle.go b/server/config/gorm_oracle.go index 1bbeb46a..52cf21c0 100644 --- a/server/config/gorm_oracle.go +++ b/server/config/gorm_oracle.go @@ -1,10 +1,18 @@ package config +import ( + "fmt" + "net" + "net/url" +) + type Oracle struct { GeneralDB `yaml:",inline" mapstructure:",squash"` } func (m *Oracle) Dsn() string { - return "oracle://" + m.Username + ":" + m.Password + "@" + m.Path + ":" + m.Port + "/" + m.Dbname + "?" + m.Config + dsn := fmt.Sprintf("oracle://%s:%s@%s/%s?%s", url.PathEscape(m.Username), url.PathEscape(m.Password), + net.JoinHostPort(m.Path, m.Port), url.PathEscape(m.Dbname), m.Config) + return dsn } diff --git a/server/core/server.go b/server/core/server.go index 6e386e07..f7f9a7f9 100644 --- a/server/core/server.go +++ b/server/core/server.go @@ -35,7 +35,7 @@ func RunServer() { fmt.Printf(` 欢迎使用 gin-vue-admin - 当前版本:v2.8.4 + 当前版本:%s 加群方式:微信号:shouzi_1994 QQ群:470239250 项目地址:https://github.com/flipped-aurora/gin-vue-admin 插件市场:https://plugin.gin-vue-admin.com @@ -48,6 +48,7 @@ func RunServer() { ** 版权所有方:flipped-aurora开源团队 ** ** 版权持有公司:北京翻转极光科技有限责任公司 ** ** 剔除授权标识需购买商用授权:https://gin-vue-admin.com/empower/index.html ** -`, address, address, global.GVA_CONFIG.MCP.SSEPath, address, global.GVA_CONFIG.MCP.MessagePath) + ** 感谢您对Gin-Vue-Admin的支持与关注 合法授权使用更有利于项目的长久发展** +`, global.Version, address, address, global.GVA_CONFIG.MCP.SSEPath, address, global.GVA_CONFIG.MCP.MessagePath) initServer(address, Router, 10*time.Minute, 10*time.Minute) } diff --git a/server/docs/docs.go b/server/docs/docs.go index 81813578..54615785 100644 --- a/server/docs/docs.go +++ b/server/docs/docs.go @@ -1,7 +1,10 @@ // Package docs Code generated by swaggo/swag. DO NOT EDIT package docs -import "github.com/swaggo/swag" +import ( + "github.com/flipped-aurora/gin-vue-admin/server/global" + "github.com/swaggo/swag" +) const docTemplate = `{ "schemes": {{ marshal .Schemes }}, @@ -10411,7 +10414,7 @@ const docTemplate = `{ // SwaggerInfo holds exported Swagger Info so clients can modify it var SwaggerInfo = &swag.Spec{ - Version: "v2.8.4", + Version: global.Version, Host: "", BasePath: "", Schemes: []string{}, diff --git a/server/global/version.go b/server/global/version.go new file mode 100644 index 00000000..68314d1e --- /dev/null +++ b/server/global/version.go @@ -0,0 +1,12 @@ +package global + +// Version 版本信息 +// 目前只有Version正式使用 其余为预留 +const ( + // Version 当前版本号 + Version = "v2.8.5" + // AppName 应用名称 + AppName = "Gin-Vue-Admin" + // Description 应用描述 + Description = "使用gin+vue进行极速开发的全栈开发基础平台" +) \ No newline at end of file diff --git a/server/go.mod b/server/go.mod index d80009ef..b3b8dfe4 100644 --- a/server/go.mod +++ b/server/go.mod @@ -9,6 +9,7 @@ require ( github.com/aws/aws-sdk-go v1.55.6 github.com/casbin/casbin/v2 v2.103.0 github.com/casbin/gorm-adapter/v3 v3.32.0 + github.com/dzwvip/gorm-oracle v0.1.2 github.com/fsnotify/fsnotify v1.8.0 github.com/gin-gonic/gin v1.10.0 github.com/glebarez/sqlite v1.11.0 @@ -30,7 +31,6 @@ require ( github.com/redis/go-redis/v9 v9.7.0 github.com/robfig/cron/v3 v3.0.1 github.com/shirou/gopsutil/v3 v3.24.5 - github.com/silenceper/wechat/v2 v2.1.9 github.com/songzhibin97/gkit v1.2.13 github.com/spf13/viper v1.19.0 github.com/stretchr/testify v1.10.0 @@ -39,7 +39,6 @@ require ( github.com/swaggo/swag v1.16.4 github.com/tencentyun/cos-go-sdk-v5 v0.7.60 github.com/unrolled/secure v1.17.0 - github.com/volcengine/volcengine-go-sdk v1.1.25 github.com/xuri/excelize/v2 v2.9.0 go.mongodb.org/mongo-driver v1.17.2 go.uber.org/automaxprocs v1.6.0 @@ -66,7 +65,6 @@ require ( github.com/bodgit/plumbing v1.3.0 // indirect github.com/bodgit/sevenzip v1.6.0 // indirect github.com/bodgit/windows v1.0.1 // indirect - github.com/bradfitz/gomemcache v0.0.0-20220106215444-fb4bf637b56d // indirect github.com/bytedance/sonic v1.12.7 // indirect github.com/bytedance/sonic/loader v0.2.3 // indirect github.com/casbin/govaluate v1.3.0 // indirect @@ -77,7 +75,7 @@ require ( github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect github.com/dsnet/compress v0.0.2-0.20230904184137-39efe44ab707 // indirect github.com/dustin/go-humanize v1.0.1 // indirect - github.com/fatih/structs v1.1.0 // indirect + github.com/emirpasic/gods v1.12.0 // indirect github.com/gabriel-vasile/mimetype v1.4.8 // indirect github.com/gammazero/toposort v0.1.1 // indirect github.com/gin-contrib/sse v1.0.0 // indirect @@ -91,7 +89,6 @@ require ( github.com/go-playground/locales v0.14.1 // indirect github.com/go-playground/universal-translator v0.18.1 // indirect github.com/go-playground/validator/v10 v10.24.0 // indirect - github.com/go-redis/redis/v8 v8.11.5 // indirect github.com/gofrs/flock v0.12.1 // indirect github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 // indirect github.com/golang-sql/sqlexp v0.1.0 // indirect @@ -142,7 +139,7 @@ require ( github.com/sagikazarmark/locafero v0.7.0 // indirect github.com/sagikazarmark/slog-shim v0.1.0 // indirect github.com/shoenig/go-m1cpu v0.1.6 // indirect - github.com/sirupsen/logrus v1.9.0 // indirect + github.com/sijms/go-ora/v2 v2.7.17 // indirect github.com/sorairolake/lzip-go v0.3.5 // indirect github.com/sourcegraph/conc v0.3.0 // indirect github.com/spf13/afero v1.12.0 // indirect @@ -150,15 +147,12 @@ require ( github.com/spf13/pflag v1.0.5 // indirect github.com/subosito/gotenv v1.6.0 // indirect github.com/therootcompany/xz v1.0.1 // indirect - github.com/tidwall/gjson v1.14.1 // indirect - github.com/tidwall/match v1.1.1 // indirect - github.com/tidwall/pretty v1.2.0 // indirect + github.com/thoas/go-funk v0.7.0 // indirect github.com/tklauser/go-sysconf v0.3.14 // indirect github.com/tklauser/numcpus v0.9.0 // indirect github.com/twitchyliquid64/golang-asm v0.15.1 // indirect github.com/ugorji/go/codec v1.2.12 // indirect github.com/ulikunitz/xz v0.5.12 // indirect - github.com/volcengine/volc-sdk-golang v1.0.23 // indirect github.com/xdg-go/pbkdf2 v1.0.0 // indirect github.com/xdg-go/scram v1.1.2 // indirect github.com/xdg-go/stringprep v1.0.4 // indirect @@ -180,7 +174,6 @@ require ( golang.org/x/tools v0.29.0 // indirect google.golang.org/protobuf v1.36.3 // indirect gopkg.in/ini.v1 v1.67.0 // indirect - gopkg.in/yaml.v2 v2.4.0 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect gorm.io/hints v1.1.2 // indirect gorm.io/plugin/dbresolver v1.5.3 // indirect diff --git a/server/go.sum b/server/go.sum index f94a5605..4b735946 100644 --- a/server/go.sum +++ b/server/go.sum @@ -50,15 +50,10 @@ github.com/STARRY-S/zip v0.2.1 h1:pWBd4tuSGm3wtpoqRZZ2EAwOmcHK6XFf7bU9qcJXyFg= github.com/STARRY-S/zip v0.2.1/go.mod h1:xNvshLODWtC4EJ702g7cTYn13G53o1+X9BWnPFpcWV4= github.com/alex-ant/gomath v0.0.0-20160516115720-89013a210a82 h1:7dONQ3WNZ1zy960TmkxJPuwoolZwL7xKtpcM04MBnt4= github.com/alex-ant/gomath v0.0.0-20160516115720-89013a210a82/go.mod h1:nLnM0KdK1CmygvjpDUO6m1TjSsiQtL61juhNsvV/JVI= -github.com/alicebob/gopher-json v0.0.0-20200520072559-a9ecdc9d1d3a h1:HbKu58rmZpUGpz5+4FfNmIU+FmZg2P3Xaj2v2bfNWmk= -github.com/alicebob/gopher-json v0.0.0-20200520072559-a9ecdc9d1d3a/go.mod h1:SGnFV6hVsYE877CKEZ6tDNTjaSXYUk6QqoIK6PrAtcc= -github.com/alicebob/miniredis/v2 v2.30.0 h1:uA3uhDbCxfO9+DI/DuGeAMr9qI+noVWwGPNTFuKID5M= -github.com/alicebob/miniredis/v2 v2.30.0/go.mod h1:84TWKZlxYkfgMucPBf5SOQBYJceZeQRFIaQgNMiCX6Q= github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible h1:8psS8a+wKfiLt1iVDX79F7Y6wUM49Lcha2FMXt4UM8g= github.com/aliyun/aliyun-oss-go-sdk v3.0.2+incompatible/go.mod h1:T/Aws4fEfogEE9v+HPhhw+CntffsBHJ8nXQCwKr0/g8= github.com/andybalholm/brotli v1.1.1 h1:PR2pgnyFznKEugtsUo0xLdDop5SKXd5Qf5ysW+7XdTA= github.com/andybalholm/brotli v1.1.1/go.mod h1:05ib4cKhjx3OQYUY22hTVd34Bc8upXjOLL2rKwwZBoA= -github.com/avast/retry-go v3.0.0+incompatible/go.mod h1:XtSnn+n/sHqQIpZ10K1qAevBhOOCWBLXXy3hyiqqBrY= github.com/aws/aws-sdk-go v1.55.6 h1:cSg4pvZ3m8dgYcgqB97MrcdjUmZ1BeMYKUxMMB89IPk= github.com/aws/aws-sdk-go v1.55.6/go.mod h1:eRwEWoyTWFMVYVQzKMNHWP5/RV4xIUGMQfXQHfHkpNU= github.com/bmatcuk/doublestar/v4 v4.6.1/go.mod h1:xBQ8jztBU6kakFMg+8WGxn0c6z1fTSPVIjEY1Wr7jzc= @@ -70,8 +65,6 @@ github.com/bodgit/sevenzip v1.6.0 h1:a4R0Wu6/P1o1pP/3VV++aEOcyeBxeO/xE2Y9NSTrr6A github.com/bodgit/sevenzip v1.6.0/go.mod h1:zOBh9nJUof7tcrlqJFv1koWRrhz3LbDbUNngkuZxLMc= github.com/bodgit/windows v1.0.1 h1:tF7K6KOluPYygXa3Z2594zxlkbKPAOvqr97etrGNIz4= github.com/bodgit/windows v1.0.1/go.mod h1:a6JLwrB4KrTR5hBpp8FI9/9W9jJfeQ2h4XDXU74ZCdM= -github.com/bradfitz/gomemcache v0.0.0-20220106215444-fb4bf637b56d h1:pVrfxiGfwelyab6n21ZBkbkmbevaf+WvMIiR7sr97hw= -github.com/bradfitz/gomemcache v0.0.0-20220106215444-fb4bf637b56d/go.mod h1:H0wQNHz2YrLsuXOZozoeDmnHXkNCRmMW0gwFWDfEZDA= github.com/bsm/ginkgo/v2 v2.12.0 h1:Ny8MWAHyOepLGlLKYmXG4IEkioBysk6GpaRTLC8zwWs= github.com/bsm/ginkgo/v2 v2.12.0/go.mod h1:SwYbGRRDovPVboqFv0tPTcG1sN61LM1Z4ARdbAV9g4c= github.com/bsm/gomega v1.27.10 h1:yeMWxP2pV2fG3FgAODIY8EiRE3dy0aeFYt4l7wh6yKA= @@ -88,7 +81,6 @@ github.com/casbin/gorm-adapter/v3 v3.32.0/go.mod h1:Zre/H8p17mpv5U3EaWgPoxLILLdX github.com/casbin/govaluate v1.3.0 h1:VA0eSY0M2lA86dYd5kPPuNZMUD9QkWnOCnavGrw9myc= github.com/casbin/govaluate v1.3.0/go.mod h1:G/UnbIjZk/0uMNaLwZZmFQrR72tYRZWQkO70si/iR7A= github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= -github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= @@ -115,14 +107,14 @@ github.com/dsnet/compress v0.0.2-0.20230904184137-39efe44ab707/go.mod h1:qssHWj6 github.com/dsnet/golib v0.0.0-20171103203638-1ea166775780/go.mod h1:Lj+Z9rebOhdfkVLjJ8T6VcRQv3SXugXy999NBtR9aFY= github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= +github.com/dzwvip/gorm-oracle v0.1.2 h1:811aFDY7oDfKWHc0Z0lHdXzzr89EmKBSwc/jLJ8GU5g= +github.com/dzwvip/gorm-oracle v0.1.2/go.mod h1:TbF7idnO9UgGpJ0qJpDZby1/wGquzP5GYof88ScBITE= +github.com/emirpasic/gods v1.12.0 h1:QAUIPSaCu4G+POclxeqb3F+WPpdKqFGlw36+yOzGlrg= +github.com/emirpasic/gods v1.12.0/go.mod h1:YfzfFFoVP/catgzJb4IKIqXjX78Ha8FMSDh3ymbK86o= github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= -github.com/fatih/structs v1.1.0 h1:Q7juDM0QtcnhCpeyLGQKyg4TOIghuNXrkL32pHAUMxo= -github.com/fatih/structs v1.1.0/go.mod h1:9NiDSp5zOcgEDl+j00MP/WkGVPOlPRLejGD8Ga6PJ7M= github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= -github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= -github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= github.com/fsnotify/fsnotify v1.8.0 h1:dAwr6QBTBZIkG8roQaJjGof0pp0EeF+tNV7YBP3F/8M= github.com/fsnotify/fsnotify v1.8.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= github.com/gabriel-vasile/mimetype v1.4.8 h1:FfZ3gj38NjllZIeJAmMhr+qKL8Wu+nOoI3GqacKw1NM= @@ -169,12 +161,9 @@ github.com/go-playground/validator/v10 v10.4.1/go.mod h1:nlOn6nFhuKACm19sB/8EGNn github.com/go-playground/validator/v10 v10.7.0/go.mod h1:xm76BBt941f7yWdGnI2DVPFFg1UK3YY04qifoXU3lOk= github.com/go-playground/validator/v10 v10.24.0 h1:KHQckvo8G6hlWnrPX4NJJ+aBfWNAE/HH+qdL2cBpCmg= github.com/go-playground/validator/v10 v10.24.0/go.mod h1:GGzBIJMuE98Ic/kJsBXbz1x/7cByt++cQ+YOuDM5wus= -github.com/go-redis/redis/v8 v8.11.5 h1:AcZZR7igkdvfVmQTPnu9WE37LRrO/YrBH5zWyjDC0oI= -github.com/go-redis/redis/v8 v8.11.5/go.mod h1:gREzHqY1hg6oD9ngVRbLStwAWKhA0FEgq8Jd4h5lpwo= github.com/go-sql-driver/mysql v1.7.0/go.mod h1:OXbVy3sEdcQ2Doequ6Z5BW6fXNQTmx+9S1MCJN5yJMI= github.com/go-sql-driver/mysql v1.8.1 h1:LedoTUt/eveggdHS9qUFC1EFSa8bU2+1pZjSRpvNJ1Y= github.com/go-sql-driver/mysql v1.8.1/go.mod h1:wEBSXgmK//2ZFJyE+qWnIsVGmvmEKlqwuVSjsCm7DZg= -github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= github.com/goccy/go-json v0.10.4 h1:JSwxQzIqKfmFX1swYPpUThQZp/Ka4wzJdK0LWVytLPM= github.com/goccy/go-json v0.10.4/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PULtXL6M= github.com/gofrs/flock v0.8.1/go.mod h1:F1TvTiK9OcQqauNUHlbJvyl9Qa1QvF/gOUDKA14jxHU= @@ -205,16 +194,6 @@ github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5y github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= -github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= -github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= -github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= -github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= -github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= -github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= -github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= -github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= -github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= -github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= @@ -223,7 +202,6 @@ github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5a github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= @@ -236,7 +214,6 @@ github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXi github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/pprof v0.0.0-20240409012703-83162a5b38cd h1:gbpYu9NMq8jhDVbvlGkMFWCjLFlqqEZjEmObmhUy6Vo= github.com/google/pprof v0.0.0-20240409012703-83162a5b38cd/go.mod h1:kf6iHlnVGwgKolg33glAes7Yg/8iWP8ukqeldJSO7jw= github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= @@ -253,8 +230,6 @@ github.com/gorilla/mux v1.8.1 h1:TuBL49tXwgrFYWhqrNgrUNEY92u81SPhu7sTdzQEiWY= github.com/gorilla/mux v1.8.1/go.mod h1:AKf9I4AEqPTmMytcMc0KkNouC66V3BtZ4qD5fmWSiMQ= github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4= github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM= -github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542 h1:2VTzZjLZBgl62/EtslCrtky5vbi9dd7HrQPQIx6wqiw= -github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542/go.mod h1:Ow0tF8D4Kplbc8s8sSb3V2oUCygFHVp8gC3Dn6U4MNI= github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I= github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= @@ -268,12 +243,10 @@ github.com/hashicorp/golang-lru/v2 v2.0.7 h1:a+bsQ5rvGLjzHuww6tVxozPZFVghXaHOwFs github.com/hashicorp/golang-lru/v2 v2.0.7/go.mod h1:QeFd9opnmA6QUJc5vARoKUSoFhyfM2/ZepoAG6RGpeM= github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4= github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= -github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= github.com/huaweicloud/huaweicloud-sdk-go-obs v3.24.9+incompatible h1:XQVXdk+WAJ4fSNB6mMRuYNvFWou7BZs6SZB925hPrnk= github.com/huaweicloud/huaweicloud-sdk-go-obs v3.24.9+incompatible/go.mod h1:l7VUhRbTKCzdOacdT4oWCwATKyvZqUOlOqr0Ous3k4s= github.com/iancoleman/strcase v0.3.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= -github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761 h1:iCEnooe7UlwOQYpKFhBabPMi4aNAfoODPEFNiAnClxo= @@ -291,6 +264,7 @@ github.com/jcmturner/rpc/v2 v2.0.3/go.mod h1:VUJYCIDm3PVOEHw8sgt091/20OJjskO/YJk github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= +github.com/jinzhu/now v1.1.4/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/jmespath/go-jmespath v0.4.0 h1:BEgLn5cpjn8UN1mAw4NjwDrS35OdebyEtFe+9YPoQUg= @@ -319,7 +293,6 @@ github.com/klauspost/pgzip v1.2.6 h1:8RXeL5crjEUFnR2/Sn6GJNWtSQ3Dk8pq4CL3jvdDyjU github.com/klauspost/pgzip v1.2.6/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs= github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= @@ -379,25 +352,10 @@ github.com/montanaflynn/stats v0.7.1/go.mod h1:etXPPgVO6n31NxCd9KQUMvCM+ve0ruNzt github.com/mozillazg/go-httpheader v0.2.1/go.mod h1:jJ8xECTlalr6ValeXYdOF8fFUISeBAdw6E61aqQma60= github.com/mozillazg/go-httpheader v0.4.0 h1:aBn6aRXtFzyDLZ4VIRLsZbbJloagQfMnCiYgOq6hK4w= github.com/mozillazg/go-httpheader v0.4.0/go.mod h1:PuT8h0pw6efvp8ZeUec1Rs7dwjK08bt6gKSReGMqtdA= -github.com/nbio/st v0.0.0-20140626010706-e9e8d9816f32/go.mod h1:9wM+0iRr9ahx58uYLpLIr5fm8diHn0JbqRycJi6w0Ms= github.com/ncruces/go-strftime v0.1.9 h1:bY0MQC28UADQmHmaF5dgpLmImcShSi2kHU9XLdhx/f4= github.com/ncruces/go-strftime v0.1.9/go.mod h1:Fwc5htZGVVkseilnfgOVb9mKy6w1naJmn9CehxcKcls= github.com/nwaples/rardecode/v2 v2.1.0 h1:JQl9ZoBPDy+nIZGb1mx8+anfHp/LV3NE2MjMiv0ct/U= github.com/nwaples/rardecode/v2 v2.1.0/go.mod h1:7uz379lSxPe6j9nvzxUZ+n7mnJNgjsRNb6IbvGVHRmw= -github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= -github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= -github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= -github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= -github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= -github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= -github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= -github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= -github.com/onsi/ginkgo/v2 v2.0.0/go.mod h1:vw5CSIxN1JObi/U8gcbwft7ZxR2dgaR70JSE3/PpL4c= -github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= -github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= -github.com/onsi/gomega v1.17.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= -github.com/onsi/gomega v1.18.1 h1:M1GfJqGRrBrrGGsbxzV5dqM2U2ApXefZCQpkukxYRLE= -github.com/onsi/gomega v1.18.1/go.mod h1:0q+aL8jAiMXy9hbwj2mr5GziHiwhAIQpFmmtT5hitRs= github.com/otiai10/copy v1.14.1 h1:5/7E6qsUMBaH5AnQ0sSLzzTg1oTECmcCmT6lvF45Na8= github.com/otiai10/copy v1.14.1/go.mod h1:oQwrEDDOci3IM8dJF0d8+jnbfPDllW6vUjNc3DoZm9I= github.com/otiai10/mint v1.6.3 h1:87qsV/aw1F5as1eH1zS/yqHY85ANKVMgkDrf9rcxbQs= @@ -455,10 +413,8 @@ github.com/shoenig/go-m1cpu v0.1.6 h1:nxdKQNcEB6vzgA2E2bvzKIYRuNj7XNJ4S/aRSwKzFt github.com/shoenig/go-m1cpu v0.1.6/go.mod h1:1JJMcUBvfNwpq05QDQVAnx3gUHr9IYF7GNg9SUEw2VQ= github.com/shoenig/test v0.6.4 h1:kVTaSd7WLz5WZ2IaoM0RSzRsUD+m8wRR+5qvntpn4LU= github.com/shoenig/test v0.6.4/go.mod h1:byHiCGXqrVaflBLAMq/srcZIHynQPQgeyvkvXnjqq0k= -github.com/silenceper/wechat/v2 v2.1.9 h1:wc092gUkGbbBRTdzPxROhQhOH5iE98stnfzKA73mnTo= -github.com/silenceper/wechat/v2 v2.1.9/go.mod h1:7Iu3EhQYVtDUJAj+ZVRy8yom75ga7aDWv8RurLkVm0s= -github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= -github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +github.com/sijms/go-ora/v2 v2.7.17 h1:M/pYIqjaMUeBxyzOWp2oj4ntF6fHSBloJWGNH9vbmsU= +github.com/sijms/go-ora/v2 v2.7.17/go.mod h1:EHxlY6x7y9HAsdfumurRfTd+v8NrEOTR3Xl4FWlH6xk= github.com/songzhibin97/gkit v1.2.13 h1:paY0XJkdRuy9/8k9nTnbdrzo8pC22jIIFldUkOQv5nU= github.com/songzhibin97/gkit v1.2.13/go.mod h1:38CreNR27eTGaG1UMGihrXqI4xc3nGfYxLVKKVx6Ngg= github.com/sorairolake/lzip-go v0.3.5 h1:ms5Xri9o1JBIWvOFAorYtUNik6HI3HgBTkISiqu0Cwg= @@ -467,7 +423,6 @@ github.com/sourcegraph/conc v0.3.0 h1:OQTbbt6P72L20UqAkXXuLOj79LfEanQ+YQFNpLA9yS github.com/sourcegraph/conc v0.3.0/go.mod h1:Sdozi7LEKbFPqYX2/J+iBAM6HpqSLTASQIKqDmF7Mt0= github.com/spf13/afero v1.12.0 h1:UcOPyRBYczmFn6yvphxkn9ZEOY65cpwGKb5mL36mrqs= github.com/spf13/afero v1.12.0/go.mod h1:ZTlWwG4/ahT8W7T0WQ5uYmjI9duaLQGy3Q2OAl4sk/4= -github.com/spf13/cast v1.4.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= github.com/spf13/cast v1.7.1 h1:cuNEagBQEHWN1FnbGEjCXL2szYEXqfJPbP2HNUaca9Y= github.com/spf13/cast v1.7.1/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo= github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= @@ -479,10 +434,8 @@ github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSS github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/objx v0.5.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY= github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA= -github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= -github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= @@ -505,12 +458,8 @@ github.com/tencentyun/cos-go-sdk-v5 v0.7.60 h1:/e/tmvRmfKexr/QQIBzWhOkZWsmY3EK72 github.com/tencentyun/cos-go-sdk-v5 v0.7.60/go.mod h1:8+hG+mQMuRP/OIS9d83syAvXvrMj9HhkND6Q1fLghw0= github.com/therootcompany/xz v1.0.1 h1:CmOtsn1CbtmyYiusbfmhmkpAAETj0wBIH6kCYaX+xzw= github.com/therootcompany/xz v1.0.1/go.mod h1:3K3UH1yCKgBneZYhuQUvJ9HPD19UEXEI0BWbMn8qNMY= -github.com/tidwall/gjson v1.14.1 h1:iymTbGkQBhveq21bEvAQ81I0LEBork8BFe1CUZXdyuo= -github.com/tidwall/gjson v1.14.1/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk= -github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA= -github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM= -github.com/tidwall/pretty v1.2.0 h1:RWIZEg2iJ8/g6fDDYzMpobmaoGh5OLl4AXtGUGPcqCs= -github.com/tidwall/pretty v1.2.0/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU= +github.com/thoas/go-funk v0.7.0 h1:GmirKrs6j6zJbhJIficOsz2aAI7700KsU/5YrdHRM1Y= +github.com/thoas/go-funk v0.7.0/go.mod h1:+IWnUfUmFO1+WVYQWQtIJHeRRdaIyyYglZN7xzUPe4Q= github.com/tklauser/go-sysconf v0.3.14 h1:g5vzr9iPFFz24v2KZXs/pvpvh8/V9Fw6vQK5ZZb78yU= github.com/tklauser/go-sysconf v0.3.14/go.mod h1:1ym4lWMLUOhuBOPGtRcJm7tEGX4SCYNEEEtghGG/8uY= github.com/tklauser/numcpus v0.9.0 h1:lmyCHtANi8aRUgkckBgoDk1nHCux3n2cgkJLXdQGPDo= @@ -524,10 +473,6 @@ github.com/ulikunitz/xz v0.5.12 h1:37Nm15o69RwBkXM0J6A5OlE67RZTfzUxTj8fB3dfcsc= github.com/ulikunitz/xz v0.5.12/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14= github.com/unrolled/secure v1.17.0 h1:Io7ifFgo99Bnh0J7+Q+qcMzWM6kaDPCA5FroFZEdbWU= github.com/unrolled/secure v1.17.0/go.mod h1:BmF5hyM6tXczk3MpQkFf1hpKSRqCyhqcbiQtiAF7+40= -github.com/volcengine/volc-sdk-golang v1.0.23 h1:anOslb2Qp6ywnsbyq9jqR0ljuO63kg9PY+4OehIk5R8= -github.com/volcengine/volc-sdk-golang v1.0.23/go.mod h1:AfG/PZRUkHJ9inETvbjNifTDgut25Wbkm2QoYBTbvyU= -github.com/volcengine/volcengine-go-sdk v1.1.25 h1:wwR2DTJGw2sOZ1wTWaQLn03PGO0O+motGvsoVvAp5Zk= -github.com/volcengine/volcengine-go-sdk v1.1.25/go.mod h1:EyKoi6t6eZxoPNGr2GdFCZti2Skd7MO3eUzx7TtSvNo= github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c= github.com/xdg-go/pbkdf2 v1.0.0/go.mod h1:jrpuAogTd400dnrH08LKmI/xc1MbPOebTwRqcT5RDeI= github.com/xdg-go/scram v1.1.2 h1:FHX5I5B4i4hKRVRBCFRxq1iQRej7WO3hhBuJf+UUySY= @@ -548,10 +493,7 @@ github.com/yosida95/uritemplate/v3 v3.0.2 h1:Ed3Oyj9yrmi9087+NczuL5BwkIc4wvTb5zI github.com/yosida95/uritemplate/v3 v3.0.2/go.mod h1:ILOh0sOhIJR3+L/8afwt/kE++YT040gmv5BQTMR2HP4= github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78 h1:ilQV1hzziu+LLM3zUTJ0trRztfwgjqKnBWNtSRkbmwM= github.com/youmark/pkcs8 v0.0.0-20240726163527-a2c0da244d78/go.mod h1:aL8wCCfTfSfmXjznFBSZNN13rSJjlIOI1fUNAtF7rmI= -github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= -github.com/yuin/gopher-lua v0.0.0-20220504180219-658193537a64 h1:5mLPGnFdSsevFRFc9q3yYbBkB6tsm4aCwwQV/j1JQAQ= -github.com/yuin/gopher-lua v0.0.0-20220504180219-658193537a64/go.mod h1:GBR0iDaNXjAgGg9zfCvksxSRnQx76gclCIb7kdAd1Pw= github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0= github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= go.mongodb.org/mongo-driver v1.17.1/go.mod h1:wwWm/+BuOddhcq3n68LKRmgk2wXzmF6s0SFOa0GINL4= @@ -579,7 +521,6 @@ golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8U golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= -golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58= golang.org/x/crypto v0.11.0/go.mod h1:xgJhtzW8F9jGdVFWZESrid1U1bjeNy4zgy5cRr/CIio= golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw= @@ -623,7 +564,6 @@ golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKG golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY= golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= @@ -633,7 +573,6 @@ golang.org/x/mod v0.22.0 h1:D4nJWe9zXqHOmWqj4VMOJhvzj7bEZg4wEYa759z1pH4= golang.org/x/mod v0.22.0/go.mod h1:6SkKJ3Xj0I0BrPOZoBy3bdMptDDU9oJrpohJ3eWZ1fY= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= @@ -646,11 +585,7 @@ golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLL golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= -golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= -golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= @@ -677,7 +612,6 @@ golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.3.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y= @@ -687,8 +621,6 @@ golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ= golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20190204203706-41f3e6584952/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -697,24 +629,15 @@ golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210616045830-e2b7044e8c71/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -795,7 +718,6 @@ golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapK golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200207183749-b753a1ba74fa/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200212150539-ea181f53ac56/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= -golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= golang.org/x/tools v0.13.0/go.mod h1:HvlwmtVNQAhOuCjW7xxvovg8wbNq7LwfXh/k7wXUl58= @@ -805,7 +727,6 @@ golang.org/x/tools v0.29.0/go.mod h1:KMQVMRsVxU6nHCFXrBPhDB8XncLNLM0lIy/F14RP588 golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= @@ -832,7 +753,6 @@ google.golang.org/genproto v0.0.0-20191115194625-c23dd37a84c9/go.mod h1:n3cpQtvx google.golang.org/genproto v0.0.0-20191216164720-4f79533eabd1/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= google.golang.org/genproto v0.0.0-20200212174721-66ed5ce911ce/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= @@ -840,37 +760,18 @@ google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyac google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= -google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= -google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= -google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= -google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= -google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= -google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= -google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= -google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= -google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= -google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= -google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.36.3 h1:82DV7MYdb8anAVi3qge1wSnMDrnKK7ebr+I0hHRN1BU= google.golang.org/protobuf v1.36.3/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= -gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= -gopkg.in/h2non/gock.v1 v1.1.2 h1:jBbHXgGBK/AoPVfJh5x4r/WxIrElvbLel8TCZkkZJoY= -gopkg.in/h2non/gock.v1 v1.1.2/go.mod h1:n7UGz/ckNChHiK05rDoiC4MYSunEC/lyaUm2WWaDva0= gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA= gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= -gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= -gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= @@ -889,6 +790,7 @@ gorm.io/driver/sqlserver v1.5.4 h1:xA+Y1KDNspv79q43bPyjDMUgHoYHLhXYmdFcYPobg8g= gorm.io/driver/sqlserver v1.5.4/go.mod h1:+frZ/qYmuna11zHPlh5oc2O6ZA/lS88Keb0XSH1Zh/g= gorm.io/gen v0.3.26 h1:sFf1j7vNStimPRRAtH4zz5NiHM+1dr6eA9aaRdplyhY= gorm.io/gen v0.3.26/go.mod h1:a5lq5y3w4g5LMxBcw0wnO6tYUCdNutWODq5LrIt75LE= +gorm.io/gorm v1.24.0/go.mod h1:DVrVomtaYTbqs7gB/x2uVvqnXzv0nqjB396B8cG4dBA= gorm.io/gorm v1.24.7-0.20230306060331-85eaf9eeda11/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= gorm.io/gorm v1.25.0/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= gorm.io/gorm v1.25.7-0.20240204074919-46816ad31dde/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= diff --git a/server/initialize/gorm_mssql.go b/server/initialize/gorm_mssql.go index e1478ea4..530275ab 100644 --- a/server/initialize/gorm_mssql.go +++ b/server/initialize/gorm_mssql.go @@ -28,7 +28,9 @@ func GormMssql() *gorm.DB { DSN: m.Dsn(), // DSN data source name DefaultStringSize: 191, // string 类型字段的默认长度 } - if db, err := gorm.Open(sqlserver.New(mssqlConfig), internal.Gorm.Config(m.Prefix, m.Singular)); err != nil { + // 数据库配置 + general := m.GeneralDB + if db, err := gorm.Open(sqlserver.New(mssqlConfig), internal.Gorm.Config(general)); err != nil { return nil } else { db.InstanceSet("gorm:table_options", "ENGINE="+m.Engine) @@ -48,7 +50,9 @@ func GormMssqlByConfig(m config.Mssql) *gorm.DB { DSN: m.Dsn(), // DSN data source name DefaultStringSize: 191, // string 类型字段的默认长度 } - if db, err := gorm.Open(sqlserver.New(mssqlConfig), internal.Gorm.Config(m.Prefix, m.Singular)); err != nil { + // 数据库配置 + general := m.GeneralDB + if db, err := gorm.Open(sqlserver.New(mssqlConfig), internal.Gorm.Config(general)); err != nil { panic(err) } else { db.InstanceSet("gorm:table_options", "ENGINE=InnoDB") diff --git a/server/initialize/gorm_mysql.go b/server/initialize/gorm_mysql.go index 61e50ba2..cf949aa1 100644 --- a/server/initialize/gorm_mysql.go +++ b/server/initialize/gorm_mysql.go @@ -34,8 +34,9 @@ func initMysqlDatabase(m config.Mysql) *gorm.DB { DefaultStringSize: 191, // string 类型字段的默认长度 SkipInitializeWithVersion: false, // 根据版本自动配置 } - - if db, err := gorm.Open(mysql.New(mysqlConfig), internal.Gorm.Config(m.Prefix, m.Singular)); err != nil { + // 数据库配置 + general := m.GeneralDB + if db, err := gorm.Open(mysql.New(mysqlConfig), internal.Gorm.Config(general)); err != nil { panic(err) } else { db.InstanceSet("gorm:table_options", "ENGINE="+m.Engine) diff --git a/server/initialize/gorm_oracle.go b/server/initialize/gorm_oracle.go index 513359f0..9d3d6635 100644 --- a/server/initialize/gorm_oracle.go +++ b/server/initialize/gorm_oracle.go @@ -1,18 +1,14 @@ package initialize import ( - //"github.com/dzwvip/oracle" + oracle "github.com/dzwvip/gorm-oracle" "github.com/flipped-aurora/gin-vue-admin/server/config" "github.com/flipped-aurora/gin-vue-admin/server/global" "github.com/flipped-aurora/gin-vue-admin/server/initialize/internal" - - //_ "github.com/godror/godror" - "gorm.io/driver/mysql" "gorm.io/gorm" ) // GormOracle 初始化oracle数据库 -// 如果需要Oracle库 放开import里的注释 把下方 mysql.Config 改为 oracle.Config ; mysql.New 改为 oracle.New func GormOracle() *gorm.DB { m := global.GVA_CONFIG.Oracle return initOracleDatabase(m) @@ -28,13 +24,9 @@ func initOracleDatabase(m config.Oracle) *gorm.DB { if m.Dbname == "" { return nil } - - oracleConfig := mysql.Config{ - DSN: m.Dsn(), // DSN data source name - DefaultStringSize: 191, // string 类型字段的默认长度 - } - - if db, err := gorm.Open(mysql.New(oracleConfig), internal.Gorm.Config(m.Prefix, m.Singular)); err != nil { + // 数据库配置 + general := m.GeneralDB + if db, err := gorm.Open(oracle.Open(m.Dsn()), internal.Gorm.Config(general)); err != nil { panic(err) } else { sqlDB, _ := db.DB() diff --git a/server/initialize/gorm_pgsql.go b/server/initialize/gorm_pgsql.go index 6abde589..b5022db2 100644 --- a/server/initialize/gorm_pgsql.go +++ b/server/initialize/gorm_pgsql.go @@ -30,7 +30,9 @@ func initPgSqlDatabase(p config.Pgsql) *gorm.DB { DSN: p.Dsn(), // DSN data source name PreferSimpleProtocol: false, } - if db, err := gorm.Open(postgres.New(pgsqlConfig), internal.Gorm.Config(p.Prefix, p.Singular)); err != nil { + // 数据库配置 + general := p.GeneralDB + if db, err := gorm.Open(postgres.New(pgsqlConfig), internal.Gorm.Config(general)); err != nil { panic(err) } else { sqlDB, _ := db.DB() diff --git a/server/initialize/gorm_sqlite.go b/server/initialize/gorm_sqlite.go index 9d158bf1..33510eea 100644 --- a/server/initialize/gorm_sqlite.go +++ b/server/initialize/gorm_sqlite.go @@ -25,7 +25,9 @@ func initSqliteDatabase(s config.Sqlite) *gorm.DB { return nil } - if db, err := gorm.Open(sqlite.Open(s.Dsn()), internal.Gorm.Config(s.Prefix, s.Singular)); err != nil { + // 数据库配置 + general := s.GeneralDB + if db, err := gorm.Open(sqlite.Open(s.Dsn()), internal.Gorm.Config(general)); err != nil { panic(err) } else { sqlDB, _ := db.DB() diff --git a/server/initialize/internal/gorm.go b/server/initialize/internal/gorm.go index ebf5526c..28f40979 100644 --- a/server/initialize/internal/gorm.go +++ b/server/initialize/internal/gorm.go @@ -1,12 +1,12 @@ package internal import ( + "time" + "github.com/flipped-aurora/gin-vue-admin/server/config" - "github.com/flipped-aurora/gin-vue-admin/server/global" "gorm.io/gorm" "gorm.io/gorm/logger" "gorm.io/gorm/schema" - "time" ) var Gorm = new(_gorm) @@ -15,22 +15,7 @@ type _gorm struct{} // Config gorm 自定义配置 // Author [SliverHorn](https://github.com/SliverHorn) -func (g *_gorm) Config(prefix string, singular bool) *gorm.Config { - var general config.GeneralDB - switch global.GVA_CONFIG.System.DbType { - case "mysql": - general = global.GVA_CONFIG.Mysql.GeneralDB - case "pgsql": - general = global.GVA_CONFIG.Pgsql.GeneralDB - case "oracle": - general = global.GVA_CONFIG.Oracle.GeneralDB - case "sqlite": - general = global.GVA_CONFIG.Sqlite.GeneralDB - case "mssql": - general = global.GVA_CONFIG.Mssql.GeneralDB - default: - general = global.GVA_CONFIG.Mysql.GeneralDB - } +func (g *_gorm) Config(general config.GeneralDB) *gorm.Config { return &gorm.Config{ Logger: logger.New(NewWriter(general), logger.Config{ SlowThreshold: 200 * time.Millisecond, @@ -38,8 +23,8 @@ func (g *_gorm) Config(prefix string, singular bool) *gorm.Config { Colorful: true, }), NamingStrategy: schema.NamingStrategy{ - TablePrefix: prefix, - SingularTable: singular, + TablePrefix: general.Prefix, + SingularTable: general.Singular, }, DisableForeignKeyConstraintWhenMigrating: true, } diff --git a/server/main.go b/server/main.go index f5badf75..1f4c37d5 100644 --- a/server/main.go +++ b/server/main.go @@ -21,7 +21,7 @@ import ( // @Tag.Description 用户 // @title Gin-Vue-Admin Swagger API接口文档 -// @version v2.8.4 +// @version v2.8.5 // @description 使用gin+vue进行极速开发的全栈开发基础平台 // @securityDefinitions.apikey ApiKeyAuth // @in header diff --git a/server/mcp/api_creator.go b/server/mcp/api_creator.go index d2de59b5..1a1503d9 100644 --- a/server/mcp/api_creator.go +++ b/server/mcp/api_creator.go @@ -41,7 +41,11 @@ type ApiCreator struct{} // New 创建API创建工具 func (a *ApiCreator) New() mcp.Tool { return mcp.NewTool("create_api", - mcp.WithDescription("创建后端API记录,用于AI编辑器自动添加API接口时自动创建对应的API权限记录。注意:使用gva_auto_generate创建的包和模块会自动创建API权限,无需调用此工具。仅在AI编辑器自动添加API或router下的文件产生路径变化时使用。"), + mcp.WithDescription(`创建后端API记录,用于AI编辑器自动添加API接口时自动创建对应的API权限记录。 + +**重要限制:** +- 当使用gva_auto_generate工具且needCreatedModules=true时,模块创建会自动生成API权限,不应调用此工具 +- 仅在以下情况使用:1) 单独创建API(不涉及模块创建);2) AI编辑器自动添加API;3) router下的文件产生路径变化时`), mcp.WithString("path", mcp.Required(), mcp.Description("API路径,如:/user/create"), diff --git a/server/mcp/api_lister.go b/server/mcp/api_lister.go new file mode 100644 index 00000000..d8a87d1f --- /dev/null +++ b/server/mcp/api_lister.go @@ -0,0 +1,165 @@ +package mcpTool + +import ( + "context" + "encoding/json" + "fmt" + + "github.com/flipped-aurora/gin-vue-admin/server/global" + "github.com/flipped-aurora/gin-vue-admin/server/model/system" + "github.com/mark3labs/mcp-go/mcp" + "go.uber.org/zap" +) + +// 注册工具 +func init() { + // 注册工具将在enter.go中统一处理 + RegisterTool(&ApiLister{}) +} + +// ApiInfo API信息结构 +type ApiInfo struct { + ID uint `json:"id,omitempty"` // 数据库ID(仅数据库API有) + Path string `json:"path"` // API路径 + Description string `json:"description,omitempty"` // API描述 + ApiGroup string `json:"apiGroup,omitempty"` // API组 + Method string `json:"method"` // HTTP方法 + Source string `json:"source"` // 来源:database 或 gin +} + +// ApiListResponse API列表响应结构 +type ApiListResponse struct { + Success bool `json:"success"` + Message string `json:"message"` + DatabaseApis []ApiInfo `json:"databaseApis"` // 数据库中的API + GinApis []ApiInfo `json:"ginApis"` // gin框架中的API + TotalCount int `json:"totalCount"` // 总数量 +} + +// ApiLister API列表工具 +type ApiLister struct{} + +// New 创建API列表工具 +func (a *ApiLister) New() mcp.Tool { + return mcp.NewTool("list_all_apis", + mcp.WithDescription(`获取系统中所有的API接口,分为两组: + +**功能说明:** +- 返回数据库中已注册的API列表 +- 返回gin框架中实际注册的路由API列表 +- 帮助前端判断是使用现有API还是需要创建新的API,如果api在前端未使用且需要前端调用的时候,请到api文件夹下对应模块的js中添加方法并暴露给当前业务调用 + +**返回数据结构:** +- databaseApis: 数据库中的API记录(包含ID、描述、分组等完整信息) +- ginApis: gin路由中的API(仅包含路径和方法),需要AI根据路径自行揣摩路径的业务含义,例如:/api/user/:id 表示根据用户ID获取用户信息`), + ) +} + +// Handle 处理API列表请求 +func (a *ApiLister) Handle(_ context.Context, _ mcp.CallToolRequest) (*mcp.CallToolResult, error) { + + // 获取数据库中的API + databaseApis, err := a.getDatabaseApis() + if err != nil { + global.GVA_LOG.Error("获取数据库API失败", zap.Error(err)) + errorResponse := ApiListResponse{ + Success: false, + Message: "获取数据库API失败: " + err.Error(), + } + resultJSON, _ := json.Marshal(errorResponse) + return &mcp.CallToolResult{ + Content: []mcp.Content{ + mcp.TextContent{ + Type: "text", + Text: string(resultJSON), + }, + }, + }, nil + } + + // 获取gin路由中的API + ginApis, err := a.getGinApis() + if err != nil { + global.GVA_LOG.Error("获取gin路由API失败", zap.Error(err)) + errorResponse := ApiListResponse{ + Success: false, + Message: "获取gin路由API失败: " + err.Error(), + } + resultJSON, _ := json.Marshal(errorResponse) + return &mcp.CallToolResult{ + Content: []mcp.Content{ + mcp.TextContent{ + Type: "text", + Text: string(resultJSON), + }, + }, + }, nil + } + + // 构建响应 + response := ApiListResponse{ + Success: true, + Message: "获取API列表成功", + DatabaseApis: databaseApis, + GinApis: ginApis, + TotalCount: len(databaseApis) + len(ginApis), + } + + global.GVA_LOG.Info("API列表获取成功", + zap.Int("数据库API数量", len(databaseApis)), + zap.Int("gin路由API数量", len(ginApis)), + zap.Int("总数量", response.TotalCount)) + + resultJSON, err := json.Marshal(response) + if err != nil { + return nil, fmt.Errorf("序列化结果失败: %v", err) + } + + return &mcp.CallToolResult{ + Content: []mcp.Content{ + mcp.TextContent{ + Type: "text", + Text: string(resultJSON), + }, + }, + }, nil +} + +// getDatabaseApis 获取数据库中的所有API +func (a *ApiLister) getDatabaseApis() ([]ApiInfo, error) { + var apis []system.SysApi + err := global.GVA_DB.Model(&system.SysApi{}).Order("api_group ASC, path ASC").Find(&apis).Error + if err != nil { + return nil, err + } + + // 转换为ApiInfo格式 + var result []ApiInfo + for _, api := range apis { + result = append(result, ApiInfo{ + ID: api.ID, + Path: api.Path, + Description: api.Description, + ApiGroup: api.ApiGroup, + Method: api.Method, + Source: "database", + }) + } + + return result, nil +} + +// getGinApis 获取gin路由中的所有API(包含被忽略的API) +func (a *ApiLister) getGinApis() ([]ApiInfo, error) { + // 从gin路由信息中获取所有API + var result []ApiInfo + for _, route := range global.GVA_ROUTERS { + result = append(result, ApiInfo{ + Path: route.Path, + Method: route.Method, + Source: "gin", + }) + } + + return result, nil +} diff --git a/server/mcp/current_time.go b/server/mcp/current_time.go deleted file mode 100644 index 41eaea30..00000000 --- a/server/mcp/current_time.go +++ /dev/null @@ -1,81 +0,0 @@ -package mcpTool - -import ( - "context" - "errors" - "fmt" - "github.com/mark3labs/mcp-go/mcp" - "time" -) - -func init() { - RegisterTool(&CurrentTime{}) -} - -type CurrentTime struct { -} - -// 获取当前系统时间 -func (t *CurrentTime) Handle(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) { - // 获取当前系统时间 - - timezone, ok := request.GetArguments()["timezone"].(string) - - if !ok { - return nil, errors.New("参数错误:timezone 必须是字符串类型") - } - // 根据timezone参数加载对应时区 - loc, err := loadTimeZone(timezone) - if err != nil { - return nil, err - } - - // 获取当前时间并转换为指定时区 - currentTime := time.Now().In(loc).Format("2006-01-02 15:04:05") - //返回 - return &mcp.CallToolResult{ - Content: []mcp.Content{ - mcp.TextContent{ - Type: "text", - Text: fmt.Sprintf("%s 时区的当前时间是:%s", timezone, currentTime), - }, - }, - }, nil -} - -func (t *CurrentTime) New() mcp.Tool { - return mcp.NewTool("currentTime", - mcp.WithDescription("获取当前系统时间"), - mcp.WithString("timezone", - mcp.Required(), - mcp.Description("时区"), - mcp.Enum("UTC", "CST", "PST", "EST", "GMT", "CET", "JST", "MST", "IST", "AST", "HST"), - )) -} - -// 将简写时区转换为IANA标准时区 -func loadTimeZone(timezone string) (*time.Location, error) { - // 时区映射表 - timezoneMap := map[string]string{ - "UTC": "UTC", - "CST": "Asia/Shanghai", // 中国标准时间 - "PST": "America/Los_Angeles", - "EST": "America/New_York", - "GMT": "GMT", - "CET": "Europe/Paris", - "JST": "Asia/Tokyo", - "MST": "America/Denver", - "IST": "Asia/Kolkata", - "AST": "Asia/Riyadh", // 阿拉伯标准时间 - "HST": "Pacific/Honolulu", - } - - // 获取标准时区名称 - tzName, exists := timezoneMap[timezone] - if !exists { - return nil, errors.New("不支持的时区: " + timezone) - } - - // 加载时区 - return time.LoadLocation(tzName) -} diff --git a/server/mcp/execution_plan_schema.md b/server/mcp/execution_plan_schema.md index 36f9bb20..622e83d0 100644 --- a/server/mcp/execution_plan_schema.md +++ b/server/mcp/execution_plan_schema.md @@ -7,13 +7,13 @@ ExecutionPlan 是用于自动化模块创建的执行计划结构体,包含了 ```go type ExecutionPlan struct { -PackageName string `json:"packageName"` // 包名,如:"user", "order", "product" -PackageType string `json:"packageType"` // "plugin" 或 "package" -NeedCreatedPackage bool `json:"needCreatedPackage"` // 是否需要创建包 -NeedCreatedModules bool `json:"needCreatedModules"` // 是否需要创建模块 -PackageInfo *request.SysAutoCodePackageCreate `json:"packageInfo,omitempty"` // 包信息(当NeedCreatedPackage=true时必需) -ModulesInfo []*request.AutoCode `json:"modulesInfo,omitempty"` // 模块信息数组(当NeedCreatedModules=true时必需,支持批量创建) -Paths map[string]string `json:"paths,omitempty"` // 路径信息 + PackageName string `json:"packageName"` // 包名,如:"user", "order", "product" + PackageType string `json:"packageType"` // "plugin" 或 "package" + NeedCreatedPackage bool `json:"needCreatedPackage"` // 是否需要创建包 + NeedCreatedModules bool `json:"needCreatedModules"` // 是否需要创建模块 + PackageInfo *request.SysAutoCodePackageCreate `json:"packageInfo,omitempty"` // 包信息(当NeedCreatedPackage=true时必需) + ModulesInfo []*request.AutoCode `json:"modulesInfo,omitempty"` // 模块信息数组(当NeedCreatedModules=true时必需,支持批量创建) + Paths map[string]string `json:"paths,omitempty"` // 路径信息 } ``` @@ -501,19 +501,19 @@ type DataSource struct { 2. **NeedCreatedPackage**: 当为true时,PackageInfo必须提供 3. **NeedCreatedModules**: 当为true时,ModulesInfo必须提供 4. **字段类型**: FieldType支持的类型包括: - - string(字符串) - - richtext(富文本) - - int(整型) - - bool(布尔值) - - float64(浮点型) - - time.Time(时间) - - enum(枚举) - - picture(单图片,字符串) - - pictures(多图片,json字符串) - - video(视频,字符串) - - file(文件,json字符串) - - json(JSON) - - array(数组) + - string(字符串) + - richtext(富文本) + - int(整型) + - bool(布尔值) + - float64(浮点型) + - time.Time(时间) + - enum(枚举) + - picture(单图片,字符串) + - pictures(多图片,json字符串) + - video(视频,字符串) + - file(文件,json字符串) + - json(JSON) + - array(数组) 5. **搜索类型**: FieldSearchType支持:EQ, NE, GT, GE, LT, LE, LIKE, BETWEEN等 6. **索引类型**: FieldIndexType支持:index, unique等 7. **GvaModel**: 设置为true时会自动包含ID、CreatedAt、UpdatedAt、DeletedAt字段 diff --git a/server/mcp/get_nickname.go b/server/mcp/get_nickname.go deleted file mode 100644 index 6ad55f6a..00000000 --- a/server/mcp/get_nickname.go +++ /dev/null @@ -1,79 +0,0 @@ -package mcpTool - -import ( - "context" - "errors" - "fmt" - "github.com/flipped-aurora/gin-vue-admin/server/global" - "github.com/flipped-aurora/gin-vue-admin/server/model/system" - "github.com/mark3labs/mcp-go/mcp" - "gorm.io/gorm" -) - -func init() { - RegisterTool(&GetNickname{}) -} - -type GetNickname struct{} - -// 根据用户username获取nickname -func (t *GetNickname) New() mcp.Tool { - return mcp.NewTool("getNickname", - mcp.WithDescription("根据用户username获取nickname"), - mcp.WithString("username", - mcp.Required(), - mcp.Description("用户的username"), - )) -} - -// Handle 处理获取昵称的请求 -func (t *GetNickname) Handle(ctx context.Context, request mcp.CallToolRequest) (*mcp.CallToolResult, error) { - // 1. 参数验证 - username, ok := request.GetArguments()["username"].(string) - if !ok { - return nil, errors.New("参数错误:username 必须是字符串类型") - } - - if username == "" { - return nil, errors.New("参数错误:username 不能为空") - } - - // 2. 记录操作日志 - global.GVA_LOG.Info("getNickname 工具被调用") - - // 3. 优化查询,只选择需要的字段 - var user struct { - NickName string - } - - err := global.GVA_DB.Model(&system.SysUser{}). - Select("nick_name"). - Where("username = ?", username). - First(&user).Error - - // 4. 优化错误处理 - if err != nil { - if errors.Is(err, gorm.ErrRecordNotFound) { - return &mcp.CallToolResult{ - Content: []mcp.Content{ - mcp.TextContent{ - Type: "text", - Text: fmt.Sprintf("用户 %s 不存在", username), - }, - }, - }, nil - } - global.GVA_LOG.Error("数据库查询错误") - return nil, errors.New("系统错误,请稍后再试") - } - - // 构造回复信息 - return &mcp.CallToolResult{ - Content: []mcp.Content{ - mcp.TextContent{ - Type: "text", - Text: fmt.Sprintf("用户 %s 的昵称是 %s", username, user.NickName), - }, - }, - }, nil -} diff --git a/server/mcp/gva_auto_generate.go b/server/mcp/gva_auto_generate.go index c8b4c0f5..8fe4ef85 100644 --- a/server/mcp/gva_auto_generate.go +++ b/server/mcp/gva_auto_generate.go @@ -9,6 +9,7 @@ import ( "path/filepath" "strings" "time" + "unicode" "github.com/flipped-aurora/gin-vue-admin/server/global" common "github.com/flipped-aurora/gin-vue-admin/server/model/common/request" @@ -135,6 +136,10 @@ func (t *AutomationModuleAnalyzer) New() mcp.Tool { 2. AI分析需求为1xxx2xxx格式 → gva_auto_generate(执行创建) 3. 创建完成后,根据需要使用其他辅助工具 +**重要限制:** +- 当needCreatedModules=true时,模块创建会自动生成API和菜单,因此不应再调用api_creator和menu_creator工具 +- 只有在单独创建API或菜单(不涉及模块创建)时才使用api_creator和menu_creator工具 + 重要:ExecutionPlan结构体格式要求(支持批量创建): { "packageName": "包名(string)", @@ -169,10 +174,10 @@ func (t *AutomationModuleAnalyzer) New() mcp.Tool { "generateWeb": "是否生成前端(bool)", "generateServer": "是否生成后端(bool)", "fields": [{ - "fieldName": "字段名(string)", + "fieldName": "字段名(string)必须大写开头", "fieldDesc": "字段描述(string)", - "fieldType": "字段类型:string/int/bool/time.Time等(string)", - "fieldJson": "JSON标签(string)", + "fieldType": "字段类型支持:string(字符串),richtext(富文本),int(整型),bool(布尔值),float64(浮点型),time.Time(时间),enum(枚举),picture(单图片,字符串),pictures(多图片,json字符串),video(视频,字符串),file(文件,json字符串),json(JSON),array(数组)", + "fieldJson": "JSON标签(string 必须是小驼峰命名,例:userName)", "dataTypeLong": "数据长度(string)", "comment": "注释(string)", "columnName": "数据库列名(string)", @@ -184,7 +189,7 @@ func (t *AutomationModuleAnalyzer) New() mcp.Tool { "desc": "详情显示(bool)", "excel": "导入导出(bool)", "require": "是否必填(bool)", - "defaultValue": "默认值(string)", + "defaultValue": "默认值(string),JSON类型(array,json,file,pictures)请保持为空他们不可以设置默认值", "errorText": "错误提示(string)", "clearable": "是否可清空(bool)", "sort": "是否排序(bool)", @@ -803,8 +808,8 @@ func (t *AutomationModuleAnalyzer) handleAnalyze(ctx context.Context, request mc "fields": [{ "fieldName": "字段名(必须大写开头)", "fieldDesc": "字段描述", - "fieldType": "GO 语言的数据类型", - "fieldJson": "json标签", + "fieldType": "字段类型支持:string(字符串),richtext(富文本),int(整型),bool(布尔值),float64(浮点型),time.Time(时间),enum(枚举),picture(单图片,字符串),pictures(多图片,json字符串),video(视频,字符串),file(文件,json字符串),json(JSON),array(数组)", + "fieldJson": "json标签(string 必须是小驼峰命名,例:userName)", "dataTypeLong": "长度", "comment": "注释", "columnName": "数据库列名", @@ -1239,6 +1244,7 @@ func (t *AutomationModuleAnalyzer) validateExecutionPlan(plan *ExecutionPlan) er if field.FieldName == "" { return fmt.Errorf("模块 %d 字段 %d 的 fieldName 不能为空", moduleIndex+1, i+1) } + if field.FieldDesc == "" { return fmt.Errorf("模块 %d 字段 %d 的 fieldDesc 不能为空", moduleIndex+1, i+1) } @@ -1252,6 +1258,48 @@ func (t *AutomationModuleAnalyzer) validateExecutionPlan(plan *ExecutionPlan) er return fmt.Errorf("模块 %d 字段 %d 的 columnName 不能为空", moduleIndex+1, i+1) } + // 确保字段名首字母大写 + if len(field.FieldName) > 0 { + firstChar := string(field.FieldName[0]) + if firstChar >= "a" && firstChar <= "z" { + moduleInfo.Fields[i].FieldName = strings.ToUpper(firstChar) + field.FieldName[1:] + } + } + + // 确保FieldJson使用小驼峰命名 + if len(field.FieldJson) > 0 { + // 处理下划线命名转小驼峰 + if strings.Contains(field.FieldJson, "_") { + parts := strings.Split(field.FieldJson, "_") + camelCase := strings.ToLower(parts[0]) + for j := 1; j < len(parts); j++ { + if len(parts[j]) > 0 { + camelCase += strings.ToUpper(string(parts[j][0])) + strings.ToLower(parts[j][1:]) + } + } + moduleInfo.Fields[i].FieldJson = camelCase + } else { + // 处理首字母大写转小写 + firstChar := string(field.FieldJson[0]) + if firstChar >= "A" && firstChar <= "Z" { + moduleInfo.Fields[i].FieldJson = strings.ToLower(firstChar) + field.FieldJson[1:] + } + } + } + + // 确保ColumnName使用下划线命名 + if len(field.ColumnName) > 0 { + // 将驼峰命名转换为下划线命名 + var result strings.Builder + for i, r := range field.ColumnName { + if i > 0 && r >= 'A' && r <= 'Z' { + result.WriteRune('_') + } + result.WriteRune(unicode.ToLower(r)) + } + moduleInfo.Fields[i].ColumnName = result.String() + } + // 验证字段类型 validFieldTypes := []string{"string", "int", "int64", "float64", "bool", "time.Time", "enum", "picture", "video", "file", "pictures", "array", "richtext", "json"} validType := false @@ -1365,6 +1413,13 @@ func (t *AutomationModuleAnalyzer) executeCreation(ctx context.Context, plan *Ex } result.Message += fmt.Sprintf("批量创建完成,共处理 %d 个模块; ", len(plan.ModulesInfo)) + + // 添加重要提醒:不要使用其他MCP工具 + result.Message += "\n\n⚠️ 重要提醒:\n" + result.Message += "模块创建已完成,API和菜单已自动生成。请不要再调用以下MCP工具:\n" + result.Message += "- api_creator:API权限已在模块创建时自动生成\n" + result.Message += "- menu_creator:前端菜单已在模块创建时自动生成\n" + result.Message += "如需修改API或菜单,请直接在系统管理界面中进行配置。\n" } result.Message += "已构建目录结构信息; " diff --git a/server/mcp/menu_creator.go b/server/mcp/menu_creator.go index 4add855c..a5f83545 100644 --- a/server/mcp/menu_creator.go +++ b/server/mcp/menu_creator.go @@ -64,7 +64,11 @@ type MenuCreator struct{} // New 创建菜单创建工具 func (m *MenuCreator) New() mcp.Tool { return mcp.NewTool("create_menu", - mcp.WithDescription("创建前端菜单记录,用于AI编辑器自动添加前端页面时自动创建对应的菜单项。注意:使用gva_auto_generate创建的包和模块会自动创建菜单项,无需调用此工具。仅在AI编辑器自动添加前端页面时使用。"), + mcp.WithDescription(`创建前端菜单记录,用于AI编辑器自动添加前端页面时自动创建对应的菜单项。 + +**重要限制:** +- 当使用gva_auto_generate工具且needCreatedModules=true时,模块创建会自动生成菜单项,不应调用此工具 +- 仅在以下情况使用:1) 单独创建菜单(不涉及模块创建);2) AI编辑器自动添加前端页面时`), mcp.WithNumber("parentId", mcp.Description("父菜单ID,0表示根菜单"), mcp.DefaultNumber(0), diff --git a/server/mcp/menu_lister.go b/server/mcp/menu_lister.go new file mode 100644 index 00000000..c04c83e9 --- /dev/null +++ b/server/mcp/menu_lister.go @@ -0,0 +1,112 @@ +package mcpTool + +import ( + "context" + "encoding/json" + "fmt" + + "github.com/flipped-aurora/gin-vue-admin/server/global" + "github.com/flipped-aurora/gin-vue-admin/server/model/system" + "github.com/mark3labs/mcp-go/mcp" + "go.uber.org/zap" +) + +// 注册工具 +func init() { + // 注册工具将在enter.go中统一处理 + RegisterTool(&MenuLister{}) +} + +// MenuListResponse 菜单列表响应结构 +type MenuListResponse struct { + Success bool `json:"success"` + Message string `json:"message"` + Menus []system.SysBaseMenu `json:"menus"` + TotalCount int `json:"totalCount"` + Description string `json:"description"` +} + +// MenuLister 菜单列表工具 +type MenuLister struct{} + +// New 创建菜单列表工具 +func (m *MenuLister) New() mcp.Tool { + return mcp.NewTool("list_all_menus", + mcp.WithDescription(`获取系统中所有菜单信息,包括菜单树结构、路由信息、组件路径等,用于前端编写vue-router时正确跳转 + +**功能说明:** +- 返回完整的菜单树形结构 +- 包含路由配置信息(path、name、component) +- 包含菜单元数据(title、icon、keepAlive等) +- 包含菜单参数和按钮配置 +- 支持父子菜单关系展示 + +**使用场景:** +- 前端路由配置:获取所有菜单信息用于配置vue-router +- 菜单权限管理:了解系统中所有可用的菜单项 +- 导航组件开发:构建动态导航菜单 +- 系统架构分析:了解系统的菜单结构和页面组织`), + ) +} + +// Handle 处理菜单列表请求 +func (m *MenuLister) Handle(_ context.Context, _ mcp.CallToolRequest) (*mcp.CallToolResult, error) { + // 获取所有基础菜单 + allMenus, err := m.getAllMenus() + if err != nil { + global.GVA_LOG.Error("获取菜单列表失败", zap.Error(err)) + return &mcp.CallToolResult{ + Content: []mcp.Content{ + mcp.TextContent{ + Type: "text", + Text: fmt.Sprintf("获取菜单列表失败: %v", err), + }, + }, + IsError: true, + }, nil + } + + // 构建返回结果 + response := MenuListResponse{ + Success: true, + Message: "获取菜单列表成功", + Menus: allMenus, + TotalCount: len(allMenus), + Description: "系统中所有菜单信息的标准列表,包含路由配置和组件信息", + } + + // 序列化响应 + responseJSON, err := json.MarshalIndent(response, "", " ") + if err != nil { + global.GVA_LOG.Error("序列化菜单响应失败", zap.Error(err)) + return &mcp.CallToolResult{ + Content: []mcp.Content{ + mcp.TextContent{ + Type: "text", + Text: fmt.Sprintf("序列化响应失败: %v", err), + }, + }, + IsError: true, + }, nil + } + + return &mcp.CallToolResult{ + Content: []mcp.Content{ + mcp.TextContent{ + Type: "text", + Text: string(responseJSON), + }, + }, + }, nil +} + +// getAllMenus 获取所有基础菜单 +func (m *MenuLister) getAllMenus() ([]system.SysBaseMenu, error) { + var menus []system.SysBaseMenu + err := global.GVA_DB.Order("sort").Preload("Parameters").Preload("MenuBtn").Find(&menus).Error + if err != nil { + return nil, err + } + return menus, nil +} + diff --git a/server/mcp/requirement_analyzer.go b/server/mcp/requirement_analyzer.go index e5ddd563..541b925b 100644 --- a/server/mcp/requirement_analyzer.go +++ b/server/mcp/requirement_analyzer.go @@ -20,6 +20,8 @@ type RequirementAnalysisRequest struct { UserRequirement string `json:"userRequirement"` } + + // RequirementAnalysisResponse 需求分析响应 type RequirementAnalysisResponse struct { AIPrompt string `json:"aiPrompt"` // 给AI的提示词 diff --git a/server/model/system/request/sys_version.go b/server/model/system/request/sys_version.go index 23dd665b..3d6d7ff2 100644 --- a/server/model/system/request/sys_version.go +++ b/server/model/system/request/sys_version.go @@ -20,13 +20,15 @@ type ExportVersionRequest struct { Description string `json:"description"` // 版本描述 MenuIds []uint `json:"menuIds"` // 选中的菜单ID列表 ApiIds []uint `json:"apiIds"` // 选中的API ID列表 + DictIds []uint `json:"dictIds"` // 选中的字典ID列表 } // ImportVersionRequest 导入版本请求结构体 type ImportVersionRequest struct { - VersionInfo VersionInfo `json:"version" binding:"required"` // 版本信息 - ExportMenu []system.SysBaseMenu `json:"menus"` // 菜单数据,直接复用SysBaseMenu - ExportApi []system.SysApi `json:"apis"` // API数据,直接复用SysApi + VersionInfo VersionInfo `json:"version" binding:"required"` // 版本信息 + ExportMenu []system.SysBaseMenu `json:"menus"` // 菜单数据,直接复用SysBaseMenu + ExportApi []system.SysApi `json:"apis"` // API数据,直接复用SysApi + ExportDictionary []system.SysDictionary `json:"dictionaries"` // 字典数据,直接复用SysDictionary } // VersionInfo 版本信息结构体 diff --git a/server/model/system/response/sys_version.go b/server/model/system/response/sys_version.go index b1509b89..b207ddc8 100644 --- a/server/model/system/response/sys_version.go +++ b/server/model/system/response/sys_version.go @@ -7,7 +7,8 @@ import ( // ExportVersionResponse 导出版本响应结构体 type ExportVersionResponse struct { - Version request.VersionInfo `json:"version"` // 版本信息 - Menus []system.SysBaseMenu `json:"menus"` // 菜单数据,直接复用SysBaseMenu - Apis []system.SysApi `json:"apis"` // API数据,直接复用SysApi -} \ No newline at end of file + Version request.VersionInfo `json:"version"` // 版本信息 + Menus []system.SysBaseMenu `json:"menus"` // 菜单数据,直接复用SysBaseMenu + Apis []system.SysApi `json:"apis"` // API数据,直接复用SysApi + Dictionaries []system.SysDictionary `json:"dictionaries"` // 字典数据,直接复用SysDictionary +} diff --git a/server/model/system/sys_version.go b/server/model/system/sys_version.go index e20cbdf2..e7c0a82c 100644 --- a/server/model/system/sys_version.go +++ b/server/model/system/sys_version.go @@ -8,10 +8,10 @@ import ( // 版本管理 结构体 SysVersion type SysVersion struct { global.GVA_MODEL - VersionName *string `json:"versionName" form:"versionName" gorm:"comment:版本名称;column:version_name;size:255;" binding:"required"` //版本名称 - VersionCode *string `json:"versionCode" form:"versionCode" gorm:"comment:版本号;column:version_code;size:100;" binding:"required"` //版本号 - Description *string `json:"description" form:"description" gorm:"comment:版本描述;column:description;size:500;"` //版本描述 - VersionData *string `json:"versionData" form:"versionData" gorm:"comment:版本数据JSON;column:version_data;type:text;"` //版本数据 + VersionName *string `json:"versionName" form:"versionName" gorm:"comment:版本名称;column:version_name;size:255;" binding:"required"` //版本名称 + VersionCode *string `json:"versionCode" form:"versionCode" gorm:"comment:版本号;column:version_code;size:100;" binding:"required"` //版本号 + Description *string `json:"description" form:"description" gorm:"comment:版本描述;column:description;size:500;"` //版本描述 + VersionData *string `json:"versionData" form:"versionData" gorm:"comment:版本数据JSON;column:version_data;type:text;"` //版本数据 } // TableName 版本管理 SysVersion自定义表名 sys_versions diff --git a/server/resource/package/web/view/table.vue.tpl b/server/resource/package/web/view/table.vue.tpl index 895ad2d4..b2662da8 100644 --- a/server/resource/package/web/view/table.vue.tpl +++ b/server/resource/package/web/view/table.vue.tpl @@ -110,7 +110,7 @@ getDataSourceFunc() {{ if .GvaModel }} - + {{ end }} @@ -393,8 +393,8 @@ const searchInfo = ref({}) // 排序 const sortChange = ({ prop, order }) => { const sortMap = { - CreatedAt:"CreatedAt", - ID:"ID", + CreatedAt:"created_at", + ID:"id", {{- range .Fields}} {{- if .Table}} {{- if and .Sort}} diff --git a/server/resource/plugin/web/view/view.vue.tpl b/server/resource/plugin/web/view/view.vue.tpl index 8363bdbd..98b557a5 100644 --- a/server/resource/plugin/web/view/view.vue.tpl +++ b/server/resource/plugin/web/view/view.vue.tpl @@ -108,7 +108,7 @@ getDataSourceFunc() {{ if .GvaModel }} - + {{ end }} @@ -388,8 +388,8 @@ const searchInfo = ref({}) // 排序 const sortChange = ({ prop, order }) => { const sortMap = { - CreatedAt:"CreatedAt", - ID:"ID", + CreatedAt:"created_at", + ID:"id", {{- range .Fields}} {{- if .Table}} {{- if and .Sort}} diff --git a/server/router/system/sys_user.go b/server/router/system/sys_user.go index 6c14b820..0e076f77 100644 --- a/server/router/system/sys_user.go +++ b/server/router/system/sys_user.go @@ -18,7 +18,7 @@ func (s *UserRouter) InitUserRouter(Router *gin.RouterGroup) { userRouter.PUT("setUserInfo", baseApi.SetUserInfo) // 设置用户信息 userRouter.PUT("setSelfInfo", baseApi.SetSelfInfo) // 设置自身信息 userRouter.POST("setUserAuthorities", baseApi.SetUserAuthorities) // 设置用户权限组 - userRouter.POST("resetPassword", baseApi.ResetPassword) // 设置用户权限组 + userRouter.POST("resetPassword", baseApi.ResetPassword) // 重置用户密码 userRouter.PUT("setSelfSetting", baseApi.SetSelfSetting) // 用户界面配置 } { diff --git a/server/service/system/sys_auto_code_oracle.go b/server/service/system/sys_auto_code_oracle.go index 3cdb362e..fc3c43fb 100644 --- a/server/service/system/sys_auto_code_oracle.go +++ b/server/service/system/sys_auto_code_oracle.go @@ -64,7 +64,7 @@ WHERE lower(a.table_name) = ? AND lower(a.OWNER) = ? ORDER BY - a.COLUMN_ID; + a.COLUMN_ID ` err = global.GVA_DBList[businessDB].Raw(sql, tableName, dbName).Scan(&entities).Error diff --git a/server/service/system/sys_version.go b/server/service/system/sys_version.go index 56894102..68d5ae2d 100644 --- a/server/service/system/sys_version.go +++ b/server/service/system/sys_version.go @@ -86,6 +86,12 @@ func (sysVersionService *SysVersionService) GetApisByIds(ctx context.Context, id return } +// GetDictionariesByIds 根据ID列表获取字典数据 +func (sysVersionService *SysVersionService) GetDictionariesByIds(ctx context.Context, ids []uint) (dictionaries []system.SysDictionary, err error) { + err = global.GVA_DB.Where("id in ?", ids).Preload("SysDictionaryDetails").Find(&dictionaries).Error + return +} + // ImportMenus 导入菜单数据 func (sysVersionService *SysVersionService) ImportMenus(ctx context.Context, menus []system.SysBaseMenu) error { return global.GVA_DB.Transaction(func(tx *gorm.DB) error { @@ -194,3 +200,31 @@ func (sysVersionService *SysVersionService) ImportApis(apis []system.SysApi) err return nil }) } + +// ImportDictionaries 导入字典数据 +func (sysVersionService *SysVersionService) ImportDictionaries(dictionaries []system.SysDictionary) error { + return global.GVA_DB.Transaction(func(tx *gorm.DB) error { + for _, dict := range dictionaries { + // 检查字典是否已存在 + var existingDict system.SysDictionary + if err := tx.Where("type = ?", dict.Type).First(&existingDict).Error; err == nil { + // 字典已存在,跳过 + continue + } + + // 创建新字典 + newDict := system.SysDictionary{ + Name: dict.Name, + Type: dict.Type, + Status: dict.Status, + Desc: dict.Desc, + SysDictionaryDetails: dict.SysDictionaryDetails, + } + + if err := tx.Create(&newDict).Error; err != nil { + return err + } + } + return nil + }) +} diff --git a/server/source/system/dictionary_detail.go b/server/source/system/dictionary_detail.go index 938cbc23..8a2bd2b6 100644 --- a/server/source/system/dictionary_detail.go +++ b/server/source/system/dictionary_detail.go @@ -66,7 +66,7 @@ func (i *initDictDetail) InitializeData(ctx context.Context) (context.Context, e } dicts[2].SysDictionaryDetails = []sysModel.SysDictionaryDetail{ - {Label: "date", Status: &True}, + {Label: "date", Value: "0", Status: &True, Extend: "mysql", Sort: 0}, {Label: "time", Value: "1", Status: &True, Extend: "mysql", Sort: 1}, {Label: "year", Value: "2", Status: &True, Extend: "mysql", Sort: 2}, {Label: "datetime", Value: "3", Status: &True, Extend: "mysql", Sort: 3}, @@ -74,7 +74,7 @@ func (i *initDictDetail) InitializeData(ctx context.Context) (context.Context, e {Label: "timestamptz", Value: "6", Status: &True, Extend: "pgsql", Sort: 5}, } dicts[3].SysDictionaryDetails = []sysModel.SysDictionaryDetail{ - {Label: "float", Status: &True}, + {Label: "float", Value: "0", Status: &True, Extend: "mysql", Sort: 0}, {Label: "double", Value: "1", Status: &True, Extend: "mysql", Sort: 1}, {Label: "decimal", Value: "2", Status: &True, Extend: "mysql", Sort: 2}, {Label: "numeric", Value: "3", Status: &True, Extend: "pgsql", Sort: 3}, @@ -82,7 +82,7 @@ func (i *initDictDetail) InitializeData(ctx context.Context) (context.Context, e } dicts[4].SysDictionaryDetails = []sysModel.SysDictionaryDetail{ - {Label: "char", Status: &True}, + {Label: "char", Value: "0", Status: &True, Extend: "mysql", Sort: 0}, {Label: "varchar", Value: "1", Status: &True, Extend: "mysql", Sort: 1}, {Label: "tinyblob", Value: "2", Status: &True, Extend: "mysql", Sort: 2}, {Label: "tinytext", Value: "3", Status: &True, Extend: "mysql", Sort: 3}, diff --git a/server/source/system/user.go b/server/source/system/user.go index 8f674594..d5c20d59 100644 --- a/server/source/system/user.go +++ b/server/source/system/user.go @@ -70,7 +70,7 @@ func (i *initUser) InitializeData(ctx context.Context) (next context.Context, er Username: "a303176530", Password: password, NickName: "用户1", - HeaderImg: "https:///qmplusimg.henrongyi.top/1572075907logo.png", + HeaderImg: "https://qmplusimg.henrongyi.top/1572075907logo.png", AuthorityId: 9528, Phone: "17611111111", Email: "333333333@qq.com"}, diff --git a/server/utils/autocode/template_funcs.go b/server/utils/autocode/template_funcs.go index 70b29788..40e56134 100644 --- a/server/utils/autocode/template_funcs.go +++ b/server/utils/autocode/template_funcs.go @@ -46,7 +46,8 @@ func GenerateField(field systemReq.AutoCodeField) string { gormTag += "column:" + field.ColumnName + ";" - if field.DataTypeLong != "" && field.FieldType != "enum" { + // 对于int类型,根据DataTypeLong决定具体的Go类型,不使用size标签 + if field.DataTypeLong != "" && field.FieldType != "enum" && field.FieldType != "int" { gormTag += fmt.Sprintf("size:%s;", field.DataTypeLong) } @@ -85,8 +86,27 @@ func GenerateField(field systemReq.AutoCodeField) string { tagContent := fmt.Sprintf(`json:"%s" form:"%s" gorm:"%s"`, field.FieldJson, field.FieldJson, gormTag) + // 对于int类型,根据DataTypeLong决定具体的Go类型 + var fieldType string + if field.FieldType == "int" { + switch field.DataTypeLong { + case "1", "2", "3": + fieldType = "int8" + case "4", "5": + fieldType = "int16" + case "6", "7", "8", "9", "10": + fieldType = "int32" + case "11", "12", "13", "14", "15", "16", "17", "18", "19", "20": + fieldType = "int64" + default: + fieldType = "int64" + } + } else { + fieldType = field.FieldType + } + result = fmt.Sprintf(`%s *%s `+"`"+`%s`+"`"+``, - field.FieldName, field.FieldType, tagContent) + field.FieldName, fieldType, tagContent) } if field.Require { @@ -222,11 +242,11 @@ func GenerateSearchFormItem(field systemReq.AutoCodeField) string { ` } else if field.FieldType == "float64" || field.FieldType == "int" { if field.FieldSearchType == "BETWEEN" || field.FieldSearchType == "NOT BETWEEN" { - result += fmt.Sprintf(` + result += fmt.Sprintf(` `, field.FieldName) result += ` — ` - result += fmt.Sprintf(` + result += fmt.Sprintf(` `, field.FieldName) } else { result += fmt.Sprintf(` @@ -250,7 +270,7 @@ func GenerateSearchFormItem(field systemReq.AutoCodeField) string { ` result += ` ` - result += fmt.Sprintf(``, field.FieldJson) + result += fmt.Sprintf(``, field.FieldJson) } else { result += fmt.Sprintf(``, field.FieldJson) } diff --git a/server/utils/verify.go b/server/utils/verify.go index e81c1f19..cc2cb78e 100644 --- a/server/utils/verify.go +++ b/server/utils/verify.go @@ -5,7 +5,7 @@ var ( ApiVerify = Rules{"Path": {NotEmpty()}, "Description": {NotEmpty()}, "ApiGroup": {NotEmpty()}, "Method": {NotEmpty()}} MenuVerify = Rules{"Path": {NotEmpty()}, "Name": {NotEmpty()}, "Component": {NotEmpty()}, "Sort": {Ge("0")}} MenuMetaVerify = Rules{"Title": {NotEmpty()}} - LoginVerify = Rules{"CaptchaId": {NotEmpty()}, "Captcha": {NotEmpty()}, "Username": {NotEmpty()}, "Password": {NotEmpty()}} + LoginVerify = Rules{"Username": {NotEmpty()}, "Password": {NotEmpty()}} RegisterVerify = Rules{"Username": {NotEmpty()}, "NickName": {NotEmpty()}, "Password": {NotEmpty()}, "AuthorityId": {NotEmpty()}} PageInfoVerify = Rules{"Page": {NotEmpty()}, "PageSize": {NotEmpty()}} CustomerVerify = Rules{"CustomerName": {NotEmpty()}, "CustomerPhoneData": {NotEmpty()}} diff --git a/web/limit.js b/web/limit.js index 6ba9d462..f23fa512 100644 --- a/web/limit.js +++ b/web/limit.js @@ -1,6 +1,6 @@ // 运行项目前通过node执行此脚本 (此脚本与 node_modules 目录同级) -const fs = require('fs') -const path = require('path') +import fs from 'fs' +import path from 'path' const wfPath = path.resolve(__dirname, './node_modules/.bin') fs.readdir(wfPath, (err, files) => { diff --git a/web/openDocument.js b/web/openDocument.js index 1693eeeb..03a886b4 100644 --- a/web/openDocument.js +++ b/web/openDocument.js @@ -3,7 +3,7 @@ 未经授权的商用使用可能会被我们的资产搜索引擎爬取,并可能导致后续索赔。索赔金额将不低于高级授权费的十倍。请您遵守版权法律法规,尊重知识产权。 */ -var child_process = require('child_process') +import child_process from 'child_process' var url = 'https://www.gin-vue-admin.com' var cmd = '' diff --git a/web/package.json b/web/package.json index 3e2bb3e2..ca6ceeb1 100644 --- a/web/package.json +++ b/web/package.json @@ -1,6 +1,6 @@ { "name": "gin-vue-admin", - "version": "2.8.4", + "version": "2.8.5", "private": true, "scripts": { "serve": "node openDocument.js && vite --host --mode development", @@ -9,10 +9,12 @@ "preview": "vite preview", "fix-memory-limit": "cross-env LIMIT=4096 increase-memory-limit" }, + "type": "module", "dependencies": { "@element-plus/icons-vue": "^2.3.1", "@form-create/designer": "^3.2.6", "@form-create/element-ui": "^3.2.10", + "@unocss/transformer-directives": "^66.4.2", "@vue-office/docx": "^1.6.2", "@vue-office/excel": "^1.7.11", "@vue-office/pdf": "^2.0.2", @@ -39,10 +41,9 @@ "screenfull": "^6.0.2", "sortablejs": "^1.15.3", "spark-md5": "^3.0.2", - "tailwindcss": "^3.4.10", "universal-cookie": "^7", "vform3-builds": "^3.0.10", - "vite-auto-import-svg": "^1.6.0", + "vite-auto-import-svg": "^1.9.0", "vue": "^3.5.7", "vue-cropper": "^1.1.4", "vue-echarts": "^7.0.3", @@ -55,6 +56,9 @@ "devDependencies": { "@babel/eslint-parser": "^7.25.1", "@eslint/js": "^8.56.0", + "@unocss/extractor-svelte": "^66.4.2", + "@unocss/preset-wind3": "^66.4.2", + "@unocss/vite": "^66.5.0", "@vitejs/plugin-legacy": "^6.0.0", "@vitejs/plugin-vue": "^5.0.3", "@vue/cli-plugin-babel": "~5.0.8", @@ -69,8 +73,10 @@ "dotenv": "^16.4.5", "eslint": "^8.57.0", "eslint-plugin-vue": "^9.19.2", + "globals": "^16.3.0", "sass": "^1.78.0", "terser": "^5.31.6", + "unocss": "^66.4.2", "vite": "^6.2.3", "vite-plugin-banner": "^0.8.0", "vite-plugin-importer": "^0.2.5", diff --git a/web/postcss.config.js b/web/postcss.config.js deleted file mode 100644 index 85f717cc..00000000 --- a/web/postcss.config.js +++ /dev/null @@ -1,6 +0,0 @@ -module.exports = { - plugins: { - tailwindcss: {}, - autoprefixer: {} - } -} diff --git a/web/src/components/errorPreview/index.vue b/web/src/components/errorPreview/index.vue index dd08402a..d3d99b0f 100644 --- a/web/src/components/errorPreview/index.vue +++ b/web/src/components/errorPreview/index.vue @@ -7,13 +7,13 @@

{{ displayData.title }}

- +
+ +
-
+
错误类型
@@ -45,16 +45,16 @@
- +
diff --git a/web/src/view/init/index.vue b/web/src/view/init/index.vue index 211ec4a8..5d769764 100644 --- a/web/src/view/init/index.vue +++ b/web/src/view/init/index.vue @@ -136,7 +136,7 @@ // @ts-ignore import { initDB } from '@/api/initdb' import { reactive, ref } from 'vue' - import { ElLoading, ElMessage } from 'element-plus' + import { ElLoading, ElMessage, ElMessageBox } from 'element-plus' import { useRouter } from 'vue-router' defineOptions({ @@ -274,7 +274,25 @@ type: 'success', message: res.msg }) - router.push({ name: 'Login' }) + + // 显示AI助手配置提示弹窗 + ElMessageBox.confirm( + '已经完成基础数据库初始化!建议先进行编辑器AI助手配置,以获得更好的开发体验。', + '配置完成', + { + confirmButtonText: '查看AI配置文档', + cancelButtonText: '稍后配置', + type: 'success', + center: true + } + ).then(() => { + // 点击确认按钮,打开AI配置文档 + window.open('https://www.gin-vue-admin.com/guide/server/mcp.html', '_blank') + router.push({ name: 'Login' }) + }).catch(() => { + // 点击取消按钮或关闭弹窗,直接跳转到登录页 + router.push({ name: 'Login' }) + }) } loading.close() } catch (_) { diff --git a/web/src/view/layout/aside/combinationMode.vue b/web/src/view/layout/aside/combinationMode.vue index 3e588ce6..5584e891 100644 --- a/web/src/view/layout/aside/combinationMode.vue +++ b/web/src/view/layout/aside/combinationMode.vue @@ -7,7 +7,7 @@ @@ -34,7 +34,7 @@ :collapse="isCollapse" :collapse-transition="false" :default-active="active" - class="border-r-0 w-full" + class="!border-r-0 w-full" unique-opened @select="(index, _, ele) => selectMenuItem(index, _, ele, false)" > diff --git a/web/src/view/layout/aside/headMode.vue b/web/src/view/layout/aside/headMode.vue index 78318ba4..db48fc9d 100644 --- a/web/src/view/layout/aside/headMode.vue +++ b/web/src/view/layout/aside/headMode.vue @@ -6,7 +6,7 @@ diff --git a/web/src/view/layout/aside/sidebarMode.vue b/web/src/view/layout/aside/sidebarMode.vue index 032a2537..128c655c 100644 --- a/web/src/view/layout/aside/sidebarMode.vue +++ b/web/src/view/layout/aside/sidebarMode.vue @@ -2,7 +2,7 @@
@@ -74,7 +74,7 @@ :collapse="isCollapse" :collapse-transition="false" :default-active="active" - class="border-r-0 w-full" + class="!border-r-0 w-full" unique-opened @select="selectMenuItem" > diff --git a/web/src/view/layout/header/tools.vue b/web/src/view/layout/header/tools.vue index deba8d8c..1176debd 100644 --- a/web/src/view/layout/header/tools.vue +++ b/web/src/view/layout/header/tools.vue @@ -8,7 +8,7 @@ @@ -29,7 +29,7 @@ @@ -37,7 +37,7 @@ @@ -46,7 +46,7 @@ @@ -61,14 +61,14 @@ > diff --git a/web/src/view/layout/index.vue b/web/src/view/layout/index.vue index 43342024..5210a4c8 100644 --- a/web/src/view/layout/index.vue +++ b/web/src/view/layout/index.vue @@ -7,11 +7,11 @@ :font="font" :z-index="9999" :gap="[180, 150]" - class="absolute inset-0 pointer-events-none" + class="!absolute !inset-0 !pointer-events-none" :content="userStore.userInfo.nickName" /> -
+
+
- +
diff --git a/web/src/view/layout/tabs/index.vue b/web/src/view/layout/tabs/index.vue index 4ad54f37..6a6ad8ad 100644 --- a/web/src/view/layout/tabs/index.vue +++ b/web/src/view/layout/tabs/index.vue @@ -8,7 +8,7 @@ v-model="activeValue" :closable="!(historys.length === 1 && $route.name === defaultRouter)" type="card" - class="bg-white text-slate-700 dark:text-slate-500 dark:bg-slate-900" + class="bg-white text-slate-700 dark:text-slate-500 dark:bg-slate-900 pt-1" @contextmenu.prevent="openContextMenu($event)" @tab-click="changeTab" @tab-remove="removeTab" diff --git a/web/src/view/superAdmin/authority/components/menus.vue b/web/src/view/superAdmin/authority/components/menus.vue index 2dd571dc..fd00f8d2 100644 --- a/web/src/view/superAdmin/authority/components/menus.vue +++ b/web/src/view/superAdmin/authority/components/menus.vue @@ -23,7 +23,7 @@