513 lines
14 KiB
Go
513 lines
14 KiB
Go
package system
|
||
|
||
import (
|
||
"bytes"
|
||
"context"
|
||
"fmt"
|
||
goast "go/ast"
|
||
"go/parser"
|
||
"go/printer"
|
||
"go/token"
|
||
"io"
|
||
"mime/multipart"
|
||
"os"
|
||
"path/filepath"
|
||
"strings"
|
||
|
||
"git.echol.cn/loser/st/server/global"
|
||
"git.echol.cn/loser/st/server/model/system"
|
||
"git.echol.cn/loser/st/server/model/system/request"
|
||
pluginUtils "git.echol.cn/loser/st/server/plugin/plugin-tool/utils"
|
||
"git.echol.cn/loser/st/server/utils"
|
||
ast "git.echol.cn/loser/st/server/utils/ast"
|
||
"github.com/mholt/archives"
|
||
cp "github.com/otiai10/copy"
|
||
"github.com/pkg/errors"
|
||
"go.uber.org/zap"
|
||
)
|
||
|
||
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
|
||
}
|
||
|
||
// 将上传的文件内容复制到临时文件
|
||
// 使用io.Copy的好处:高效处理大文件,自动管理缓冲区,避免内存溢出
|
||
_, err = io.Copy(out, src)
|
||
if err != nil {
|
||
out.Close()
|
||
return -1, -1, err
|
||
}
|
||
|
||
// 立即关闭文件,确保数据写入磁盘并释放文件句柄
|
||
// 必须在解压前关闭,否则在Windows系统上会导致文件被占用无法解压
|
||
err = out.Close()
|
||
if err != nil {
|
||
return -1, -1, err
|
||
}
|
||
|
||
paths, err := utils.Unzip(GVAPLUGPINATH+file.Filename, GVAPLUGPINATH)
|
||
paths = filterFile(paths)
|
||
var webIndex = -1
|
||
var serverIndex = -1
|
||
webPlugin := ""
|
||
serverPlugin := ""
|
||
serverPackage := ""
|
||
serverRootName := ""
|
||
|
||
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` {
|
||
if len(serverPlugin) == 0 {
|
||
serverPlugin = filepath.Join(pathArr[0], pathArr[1], pathArr[2], pathArr[3])
|
||
}
|
||
if serverRootName == "" && ln > 1 && pathArr[1] != "" {
|
||
serverRootName = pathArr[1]
|
||
}
|
||
if ln > 4 && serverPackage == "" && pathArr[4] != "" {
|
||
serverPackage = pathArr[4]
|
||
}
|
||
}
|
||
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 {
|
||
if serverPackage == "" {
|
||
serverPackage = serverRootName
|
||
}
|
||
err = installation(serverPlugin, global.GVA_CONFIG.AutoCode.Server, global.GVA_CONFIG.AutoCode.Server)
|
||
if err != nil {
|
||
return webIndex, serverIndex, err
|
||
}
|
||
err = ensurePluginRegisterImport(serverPackage)
|
||
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 ensurePluginRegisterImport(packageName string) error {
|
||
module := strings.TrimSpace(global.GVA_CONFIG.AutoCode.Module)
|
||
if module == "" {
|
||
return errors.New("autocode module is empty")
|
||
}
|
||
if packageName == "" {
|
||
return errors.New("plugin package is empty")
|
||
}
|
||
|
||
registerPath := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, "plugin", "register.go")
|
||
src, err := os.ReadFile(registerPath)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
fileSet := token.NewFileSet()
|
||
astFile, err := parser.ParseFile(fileSet, registerPath, src, parser.ParseComments)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
importPath := fmt.Sprintf("%s/plugin/%s", module, packageName)
|
||
if ast.CheckImport(astFile, importPath) {
|
||
return nil
|
||
}
|
||
|
||
importSpec := &goast.ImportSpec{
|
||
Name: goast.NewIdent("_"),
|
||
Path: &goast.BasicLit{Kind: token.STRING, Value: fmt.Sprintf("%q", importPath)},
|
||
}
|
||
var importDecl *goast.GenDecl
|
||
for _, decl := range astFile.Decls {
|
||
genDecl, ok := decl.(*goast.GenDecl)
|
||
if !ok {
|
||
continue
|
||
}
|
||
if genDecl.Tok == token.IMPORT {
|
||
importDecl = genDecl
|
||
break
|
||
}
|
||
}
|
||
if importDecl == nil {
|
||
astFile.Decls = append([]goast.Decl{
|
||
&goast.GenDecl{
|
||
Tok: token.IMPORT,
|
||
Specs: []goast.Spec{importSpec},
|
||
},
|
||
}, astFile.Decls...)
|
||
} else {
|
||
importDecl.Specs = append(importDecl.Specs, importSpec)
|
||
}
|
||
|
||
var out []byte
|
||
bf := bytes.NewBuffer(out)
|
||
printer.Fprint(bf, fileSet, astFile)
|
||
|
||
return os.WriteFile(registerPath, bf.Bytes(), 0666)
|
||
}
|
||
|
||
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
|
||
}
|
||
|
||
func (s *autoCodePlugin) InitDictionary(dictInfo request.InitDictionary) (err error) {
|
||
dictPath := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, "plugin", dictInfo.PlugName, "initialize", "dictionary.go")
|
||
src, err := os.ReadFile(dictPath)
|
||
if err != nil {
|
||
fmt.Println(err)
|
||
}
|
||
fileSet := token.NewFileSet()
|
||
astFile, err := parser.ParseFile(fileSet, "", src, 0)
|
||
arrayAst := ast.FindArray(astFile, "model", "SysDictionary")
|
||
var dictionaries []system.SysDictionary
|
||
err = global.GVA_DB.Preload("SysDictionaryDetails").Find(&dictionaries, "id in (?)", dictInfo.Dictionaries).Error
|
||
if err != nil {
|
||
return err
|
||
}
|
||
dictExpr := ast.CreateDictionaryStructAst(dictionaries)
|
||
arrayAst.Elts = *dictExpr
|
||
|
||
var out []byte
|
||
bf := bytes.NewBuffer(out)
|
||
printer.Fprint(bf, fileSet, astFile)
|
||
|
||
os.WriteFile(dictPath, bf.Bytes(), 0666)
|
||
return nil
|
||
}
|
||
|
||
func (s *autoCodePlugin) Remove(pluginName string, pluginType string) (err error) {
|
||
// 1. 删除前端代码
|
||
if pluginType == "web" || pluginType == "full" {
|
||
webDir := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Web, "plugin", pluginName)
|
||
err = os.RemoveAll(webDir)
|
||
if err != nil {
|
||
return errors.Wrap(err, "删除前端插件目录失败")
|
||
}
|
||
}
|
||
|
||
// 2. 删除后端代码
|
||
if pluginType == "server" || pluginType == "full" {
|
||
serverDir := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, "plugin", pluginName)
|
||
err = os.RemoveAll(serverDir)
|
||
if err != nil {
|
||
return errors.Wrap(err, "删除后端插件目录失败")
|
||
}
|
||
|
||
// 移除注册
|
||
removePluginRegisterImport(pluginName)
|
||
}
|
||
|
||
// 通过utils 获取 api 菜单 字典
|
||
apis, menus, dicts := pluginUtils.GetPluginData(pluginName)
|
||
|
||
// 3. 删除菜单 (递归删除)
|
||
if len(menus) > 0 {
|
||
for _, menu := range menus {
|
||
var dbMenu system.SysBaseMenu
|
||
if err := global.GVA_DB.Where("name = ?", menu.Name).First(&dbMenu).Error; err == nil {
|
||
// 获取该菜单及其所有子菜单的ID
|
||
var menuIds []int
|
||
GetMenuIds(dbMenu, &menuIds)
|
||
// 逆序删除,先删除子菜单
|
||
for i := len(menuIds) - 1; i >= 0; i-- {
|
||
err := BaseMenuServiceApp.DeleteBaseMenu(menuIds[i])
|
||
if err != nil {
|
||
zap.L().Error("删除菜单失败", zap.Int("id", menuIds[i]), zap.Error(err))
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// 4. 删除API
|
||
if len(apis) > 0 {
|
||
for _, api := range apis {
|
||
var dbApi system.SysApi
|
||
if err := global.GVA_DB.Where("path = ? AND method = ?", api.Path, api.Method).First(&dbApi).Error; err == nil {
|
||
err := ApiServiceApp.DeleteApi(dbApi)
|
||
if err != nil {
|
||
zap.L().Error("删除API失败", zap.String("path", api.Path), zap.Error(err))
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// 5. 删除字典
|
||
if len(dicts) > 0 {
|
||
for _, dict := range dicts {
|
||
var dbDict system.SysDictionary
|
||
if err := global.GVA_DB.Where("type = ?", dict.Type).First(&dbDict).Error; err == nil {
|
||
err := DictionaryServiceApp.DeleteSysDictionary(dbDict)
|
||
if err != nil {
|
||
zap.L().Error("删除字典失败", zap.String("type", dict.Type), zap.Error(err))
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return nil
|
||
}
|
||
|
||
func GetMenuIds(menu system.SysBaseMenu, ids *[]int) {
|
||
*ids = append(*ids, int(menu.ID))
|
||
var children []system.SysBaseMenu
|
||
global.GVA_DB.Where("parent_id = ?", menu.ID).Find(&children)
|
||
for _, child := range children {
|
||
// 先递归收集子菜单
|
||
GetMenuIds(child, ids)
|
||
}
|
||
}
|
||
|
||
func removePluginRegisterImport(packageName string) error {
|
||
module := strings.TrimSpace(global.GVA_CONFIG.AutoCode.Module)
|
||
if module == "" {
|
||
return errors.New("autocode module is empty")
|
||
}
|
||
if packageName == "" {
|
||
return errors.New("plugin package is empty")
|
||
}
|
||
|
||
registerPath := filepath.Join(global.GVA_CONFIG.AutoCode.Root, global.GVA_CONFIG.AutoCode.Server, "plugin", "register.go")
|
||
src, err := os.ReadFile(registerPath)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
fileSet := token.NewFileSet()
|
||
astFile, err := parser.ParseFile(fileSet, registerPath, src, parser.ParseComments)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
importPath := fmt.Sprintf("%s/plugin/%s", module, packageName)
|
||
importLit := fmt.Sprintf("%q", importPath)
|
||
|
||
// 移除 import
|
||
var newDecls []goast.Decl
|
||
for _, decl := range astFile.Decls {
|
||
genDecl, ok := decl.(*goast.GenDecl)
|
||
if !ok {
|
||
newDecls = append(newDecls, decl)
|
||
continue
|
||
}
|
||
if genDecl.Tok == token.IMPORT {
|
||
var newSpecs []goast.Spec
|
||
for _, spec := range genDecl.Specs {
|
||
importSpec, ok := spec.(*goast.ImportSpec)
|
||
if !ok {
|
||
newSpecs = append(newSpecs, spec)
|
||
continue
|
||
}
|
||
if importSpec.Path.Value != importLit {
|
||
newSpecs = append(newSpecs, spec)
|
||
}
|
||
}
|
||
// 如果还有其他import,保留该 decl
|
||
if len(newSpecs) > 0 {
|
||
genDecl.Specs = newSpecs
|
||
newDecls = append(newDecls, genDecl)
|
||
}
|
||
} else {
|
||
newDecls = append(newDecls, decl)
|
||
}
|
||
}
|
||
astFile.Decls = newDecls
|
||
|
||
var out []byte
|
||
bf := bytes.NewBuffer(out)
|
||
printer.Fprint(bf, fileSet, astFile)
|
||
|
||
return os.WriteFile(registerPath, bf.Bytes(), 0666)
|
||
}
|