
* fix(style): 修复 border 额外的 reset 导致 tailwind border 属性生效异常的问题 * feat: 添加错误预览组件并优化请求错误处理逻辑 * optimize: select and update necessary fields in `ChangePassword` method - Simplify `ChangePassword` method signature by removing unnecessary return type. - Use `Select()` to fetch only the necessary fields (`id` and `password`) from the database. - Replace `Save()` with `Update()` for more efficient password update operation. Note: use `Save(&user)` to update the whole user record, which will cover other unchanged fields as well, causing data inconsistency when data race conditions. * feat(menu): 版本更新为2.8.4,给菜单增加按钮和参数的预制打包 * feat(menu): 恢复空白的配置文件 * Remove unused `SideMode` field from `ChangeUserInfo` struct Remove unused and deprecated `SideMode` field from user request model. * feat(automation): 增加可以自动生成CURD和续写方法的MCP * fix(mcp): 确保始终返回目录结构信息 * fix(mcp): 当不需要创建模块时提前返回目录结构信息 * feat(automation): 增加可以自动生成CURD和续写方法的MCP * feat(mcp): 添加GAG工具用户确认流程和自动字典创建功能 实现三步工作流程:分析、确认、执行 新增自动字典创建功能,当字段使用字典类型时自动检查并创建字典 添加用户确认机制,确保创建操作前获得用户明确确认 * feat(version): 新增版本管理功能,支持创建、导入、导出和下载版本数据 新增版本管理模块,包含以下功能: 1. 版本数据的增删改查 2. 版本创建功能,可选择关联菜单和API 3. 版本导入导出功能 4. 版本JSON数据下载 5. 相关前端页面和接口实现 * refactor(version): 简化版本管理删除逻辑并移除无用字段 移除版本管理中的状态、创建者、更新者和删除者字段 简化删除和批量删除方法的实现,去除事务和用户ID参数 更新自动生成配置的默认值说明 * feat(版本管理): 新增版本管理功能模块 * fix(menu): 修复递归创建菜单时关联数据未正确处理的问题 * feat(mcp): 添加预设计模块扫描功能以支持代码自动生成 在自动化模块分析器中添加对预设计模块的扫描功能,包括: - 新增PredesignedModuleInfo结构体存储模块信息 - 实现scanPredesignedModules方法扫描plugin和model目录 - 在分析响应中添加predesignedModules字段 - 更新帮助文档说明预设计模块的使用方式 这些修改使系统能够识别并利用现有的预设计模块,提高代码生成效率并减少重复工作。 * feat(mcp): 新增API、菜单和字典生成工具并优化自动生成模块 * docs(mcp): 更新菜单和API创建工具的描述信息 * feat(mcp): 添加字典查询工具用于AI生成逻辑时了解可用字典选项 * feat: 在创建菜单/API/模块结果中添加权限分配提醒 为菜单创建、API创建和模块创建的结果消息添加权限分配提醒,帮助用户了解后续需要进行的权限配置步骤 * refactor(mcp): 统一使用WithBoolean替换WithBool并优化错误处理 * docs(mcp): 更新API创建工具的说明和错误处理日志 * feat(mcp): 添加插件意图检测功能并增强验证逻辑 --------- Co-authored-by: Azir <2075125282@qq.com> Co-authored-by: Feng.YJ <jxfengyijie@gmail.com> Co-authored-by: piexlMax(奇淼 <qimiaojiangjizhao@gmail.com>
252 lines
6.9 KiB
Go
252 lines
6.9 KiB
Go
package system
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"github.com/flipped-aurora/gin-vue-admin/server/global"
|
|
"github.com/flipped-aurora/gin-vue-admin/server/model/system"
|
|
"github.com/flipped-aurora/gin-vue-admin/server/model/system/request"
|
|
"github.com/flipped-aurora/gin-vue-admin/server/utils"
|
|
"github.com/flipped-aurora/gin-vue-admin/server/utils/ast"
|
|
"github.com/mholt/archives"
|
|
cp "github.com/otiai10/copy"
|
|
"github.com/pkg/errors"
|
|
"go.uber.org/zap"
|
|
"go/parser"
|
|
"go/printer"
|
|
"go/token"
|
|
"io"
|
|
"mime/multipart"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
)
|
|
|
|
var AutoCodePlugin = new(autoCodePlugin)
|
|
|
|
type autoCodePlugin struct{}
|
|
|
|
// Install 插件安装
|
|
func (s *autoCodePlugin) Install(file *multipart.FileHeader) (web, server int, err error) {
|
|
const GVAPLUGPINATH = "./gva-plug-temp/"
|
|
defer os.RemoveAll(GVAPLUGPINATH)
|
|
_, err = os.Stat(GVAPLUGPINATH)
|
|
if os.IsNotExist(err) {
|
|
os.Mkdir(GVAPLUGPINATH, os.ModePerm)
|
|
}
|
|
|
|
src, err := file.Open()
|
|
if err != nil {
|
|
return -1, -1, err
|
|
}
|
|
defer src.Close()
|
|
|
|
out, err := os.Create(GVAPLUGPINATH + file.Filename)
|
|
if err != nil {
|
|
return -1, -1, err
|
|
}
|
|
defer out.Close()
|
|
|
|
_, err = io.Copy(out, src)
|
|
|
|
paths, err := utils.Unzip(GVAPLUGPINATH+file.Filename, GVAPLUGPINATH)
|
|
paths = filterFile(paths)
|
|
var webIndex = -1
|
|
var serverIndex = -1
|
|
webPlugin := ""
|
|
serverPlugin := ""
|
|
|
|
for i := range paths {
|
|
paths[i] = filepath.ToSlash(paths[i])
|
|
pathArr := strings.Split(paths[i], "/")
|
|
ln := len(pathArr)
|
|
|
|
if ln < 4 {
|
|
continue
|
|
}
|
|
if pathArr[2]+"/"+pathArr[3] == `server/plugin` && len(serverPlugin) == 0 {
|
|
serverPlugin = filepath.Join(pathArr[0], pathArr[1], pathArr[2], pathArr[3])
|
|
}
|
|
if pathArr[2]+"/"+pathArr[3] == `web/plugin` && len(webPlugin) == 0 {
|
|
webPlugin = filepath.Join(pathArr[0], pathArr[1], pathArr[2], pathArr[3])
|
|
}
|
|
}
|
|
if len(serverPlugin) == 0 && len(webPlugin) == 0 {
|
|
zap.L().Error("非标准插件,请按照文档自动迁移使用")
|
|
return webIndex, serverIndex, errors.New("非标准插件,请按照文档自动迁移使用")
|
|
}
|
|
|
|
if len(serverPlugin) != 0 {
|
|
err = installation(serverPlugin, global.GVA_CONFIG.AutoCode.Server, global.GVA_CONFIG.AutoCode.Server)
|
|
if err != nil {
|
|
return webIndex, serverIndex, err
|
|
}
|
|
}
|
|
|
|
if len(webPlugin) != 0 {
|
|
err = installation(webPlugin, global.GVA_CONFIG.AutoCode.Server, global.GVA_CONFIG.AutoCode.Web)
|
|
if err != nil {
|
|
return webIndex, serverIndex, err
|
|
}
|
|
}
|
|
|
|
return 1, 1, err
|
|
}
|
|
|
|
func installation(path string, formPath string, toPath string) error {
|
|
arr := strings.Split(filepath.ToSlash(path), "/")
|
|
ln := len(arr)
|
|
if ln < 3 {
|
|
return errors.New("arr")
|
|
}
|
|
name := arr[ln-3]
|
|
|
|
var form = filepath.Join(global.GVA_CONFIG.AutoCode.Root, formPath, path)
|
|
var to = filepath.Join(global.GVA_CONFIG.AutoCode.Root, toPath, "plugin")
|
|
_, err := os.Stat(to + name)
|
|
if err == nil {
|
|
zap.L().Error("autoPath 已存在同名插件,请自行手动安装", zap.String("to", to))
|
|
return errors.New(toPath + "已存在同名插件,请自行手动安装")
|
|
}
|
|
return cp.Copy(form, to, cp.Options{Skip: skipMacSpecialDocument})
|
|
}
|
|
|
|
func filterFile(paths []string) []string {
|
|
np := make([]string, 0, len(paths))
|
|
for _, path := range paths {
|
|
if ok, _ := skipMacSpecialDocument(nil, path, ""); ok {
|
|
continue
|
|
}
|
|
np = append(np, path)
|
|
}
|
|
return np
|
|
}
|
|
|
|
func skipMacSpecialDocument(_ os.FileInfo, src, _ string) (bool, error) {
|
|
if strings.Contains(src, ".DS_Store") || strings.Contains(src, "__MACOSX") {
|
|
return true, nil
|
|
}
|
|
return false, nil
|
|
}
|
|
|
|
func (s *autoCodePlugin) PubPlug(plugName string) (zipPath string, err error) {
|
|
if plugName == "" {
|
|
return "", errors.New("插件名称不能为空")
|
|
}
|
|
|
|
// 防止路径穿越
|
|
plugName = filepath.Clean(plugName)
|
|
|
|
webPath := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Web, "plugin", plugName)
|
|
serverPath := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, "plugin", plugName)
|
|
// 创建一个新的zip文件
|
|
|
|
// 判断目录是否存在
|
|
_, err = os.Stat(webPath)
|
|
if err != nil {
|
|
return "", errors.New("web路径不存在")
|
|
}
|
|
_, err = os.Stat(serverPath)
|
|
if err != nil {
|
|
return "", errors.New("server路径不存在")
|
|
}
|
|
|
|
fileName := plugName + ".zip"
|
|
// 创建一个新的zip文件
|
|
files, err := archives.FilesFromDisk(context.Background(), nil, map[string]string{
|
|
webPath: plugName + "/web/plugin/" + plugName,
|
|
serverPath: plugName + "/server/plugin/" + plugName,
|
|
})
|
|
|
|
// create the output file we'll write to
|
|
out, err := os.Create(fileName)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer out.Close()
|
|
|
|
// we can use the CompressedArchive type to gzip a tarball
|
|
// (compression is not required; you could use Tar directly)
|
|
format := archives.CompressedArchive{
|
|
//Compression: archives.Gz{},
|
|
Archival: archives.Zip{},
|
|
}
|
|
|
|
// create the archive
|
|
err = format.Archive(context.Background(), out, files)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
return filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, fileName), nil
|
|
}
|
|
|
|
func (s *autoCodePlugin) InitMenu(menuInfo request.InitMenu) (err error) {
|
|
menuPath := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, "plugin", menuInfo.PlugName, "initialize", "menu.go")
|
|
src, err := os.ReadFile(menuPath)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
}
|
|
fileSet := token.NewFileSet()
|
|
astFile, err := parser.ParseFile(fileSet, "", src, 0)
|
|
arrayAst := ast.FindArray(astFile, "model", "SysBaseMenu")
|
|
var menus []system.SysBaseMenu
|
|
|
|
parentMenu := []system.SysBaseMenu{
|
|
{
|
|
ParentId: 0,
|
|
Path: menuInfo.PlugName + "Menu",
|
|
Name: menuInfo.PlugName + "Menu",
|
|
Hidden: false,
|
|
Component: "view/routerHolder.vue",
|
|
Sort: 0,
|
|
Meta: system.Meta{
|
|
Title: menuInfo.ParentMenu,
|
|
Icon: "school",
|
|
},
|
|
},
|
|
}
|
|
|
|
// 查询菜单及其关联的参数和按钮
|
|
err = global.GVA_DB.Preload("Parameters").Preload("MenuBtn").Find(&menus, "id in (?)", menuInfo.Menus).Error
|
|
if err != nil {
|
|
return err
|
|
}
|
|
menus = append(parentMenu, menus...)
|
|
menuExpr := ast.CreateMenuStructAst(menus)
|
|
arrayAst.Elts = *menuExpr
|
|
|
|
var out []byte
|
|
bf := bytes.NewBuffer(out)
|
|
printer.Fprint(bf, fileSet, astFile)
|
|
|
|
os.WriteFile(menuPath, bf.Bytes(), 0666)
|
|
return nil
|
|
}
|
|
|
|
func (s *autoCodePlugin) InitAPI(apiInfo request.InitApi) (err error) {
|
|
apiPath := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, "plugin", apiInfo.PlugName, "initialize", "api.go")
|
|
src, err := os.ReadFile(apiPath)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
}
|
|
fileSet := token.NewFileSet()
|
|
astFile, err := parser.ParseFile(fileSet, "", src, 0)
|
|
arrayAst := ast.FindArray(astFile, "model", "SysApi")
|
|
var apis []system.SysApi
|
|
err = global.GVA_DB.Find(&apis, "id in (?)", apiInfo.APIs).Error
|
|
if err != nil {
|
|
return err
|
|
}
|
|
apisExpr := ast.CreateApiStructAst(apis)
|
|
arrayAst.Elts = *apisExpr
|
|
|
|
var out []byte
|
|
bf := bytes.NewBuffer(out)
|
|
printer.Fprint(bf, fileSet, astFile)
|
|
|
|
os.WriteFile(apiPath, bf.Bytes(), 0666)
|
|
return nil
|
|
}
|