mirror of
				https://gitee.com/dromara/mayfly-go
				synced 2025-11-04 08:20:25 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			282 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			282 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package api
 | 
						||
 | 
						||
import (
 | 
						||
	"fmt"
 | 
						||
	"io/ioutil"
 | 
						||
	"mayfly-go/base/biz"
 | 
						||
	"mayfly-go/base/ctx"
 | 
						||
	"mayfly-go/base/ginx"
 | 
						||
	"mayfly-go/base/model"
 | 
						||
	"mayfly-go/base/utils"
 | 
						||
	"mayfly-go/base/ws"
 | 
						||
	"mayfly-go/server/devops/api/form"
 | 
						||
	"mayfly-go/server/devops/api/vo"
 | 
						||
	"mayfly-go/server/devops/application"
 | 
						||
	"mayfly-go/server/devops/domain/entity"
 | 
						||
	sysApplication "mayfly-go/server/sys/application"
 | 
						||
	"strconv"
 | 
						||
	"strings"
 | 
						||
 | 
						||
	"github.com/gin-gonic/gin"
 | 
						||
)
 | 
						||
 | 
						||
type Db struct {
 | 
						||
	DbApp      application.Db
 | 
						||
	MsgApp     sysApplication.Msg
 | 
						||
	ProjectApp application.Project
 | 
						||
}
 | 
						||
 | 
						||
// @router /api/dbs [get]
 | 
						||
func (d *Db) Dbs(rc *ctx.ReqCtx) {
 | 
						||
	g := rc.GinCtx
 | 
						||
	m := &entity.Db{EnvId: uint64(ginx.QueryInt(g, "envId", 0)),
 | 
						||
		ProjectId: uint64(ginx.QueryInt(g, "projectId", 0)),
 | 
						||
		Database:  g.Query("database"),
 | 
						||
	}
 | 
						||
	m.CreatorId = rc.LoginAccount.Id
 | 
						||
	rc.ResData = d.DbApp.GetPageList(m, ginx.GetPageParam(rc.GinCtx), new([]vo.SelectDataDbVO))
 | 
						||
}
 | 
						||
 | 
						||
func (d *Db) Save(rc *ctx.ReqCtx) {
 | 
						||
	form := &form.DbForm{}
 | 
						||
	ginx.BindJsonAndValid(rc.GinCtx, form)
 | 
						||
 | 
						||
	rc.ReqParam = form
 | 
						||
 | 
						||
	db := new(entity.Db)
 | 
						||
	utils.Copy(db, form)
 | 
						||
	db.SetBaseInfo(rc.LoginAccount)
 | 
						||
	d.DbApp.Save(db)
 | 
						||
}
 | 
						||
 | 
						||
func (d *Db) DeleteDb(rc *ctx.ReqCtx) {
 | 
						||
	d.DbApp.Delete(GetDbId(rc.GinCtx))
 | 
						||
}
 | 
						||
 | 
						||
func (d *Db) TableInfos(rc *ctx.ReqCtx) {
 | 
						||
	rc.ResData = d.DbApp.GetDbInstance(GetDbId(rc.GinCtx)).GetTableInfos()
 | 
						||
}
 | 
						||
 | 
						||
func (d *Db) TableIndex(rc *ctx.ReqCtx) {
 | 
						||
	tn := rc.GinCtx.Query("tableName")
 | 
						||
	biz.NotEmpty(tn, "tableName不能为空")
 | 
						||
	rc.ResData = d.DbApp.GetDbInstance(GetDbId(rc.GinCtx)).GetTableIndex(tn)
 | 
						||
}
 | 
						||
 | 
						||
func (d *Db) GetCreateTableDdl(rc *ctx.ReqCtx) {
 | 
						||
	tn := rc.GinCtx.Query("tableName")
 | 
						||
	biz.NotEmpty(tn, "tableName不能为空")
 | 
						||
	rc.ResData = d.DbApp.GetDbInstance(GetDbId(rc.GinCtx)).GetCreateTableDdl(tn)
 | 
						||
}
 | 
						||
 | 
						||
// @router /api/db/:dbId/exec-sql [get]
 | 
						||
func (d *Db) ExecSql(rc *ctx.ReqCtx) {
 | 
						||
	g := rc.GinCtx
 | 
						||
 | 
						||
	dbInstance := d.DbApp.GetDbInstance(GetDbId(g))
 | 
						||
	biz.IsTrue(d.ProjectApp.CanAccess(rc.LoginAccount.Id, dbInstance.ProjectId), "您无权操作该资源")
 | 
						||
 | 
						||
	// 去除前后空格及换行符
 | 
						||
	sql := strings.TrimFunc(g.Query("sql"), func(r rune) bool {
 | 
						||
		s := string(r)
 | 
						||
		return s == " " || s == "\n"
 | 
						||
	})
 | 
						||
	rc.ReqParam = sql
 | 
						||
 | 
						||
	biz.NotEmpty(sql, "sql不能为空")
 | 
						||
	if strings.HasPrefix(sql, "SELECT") || strings.HasPrefix(sql, "select") {
 | 
						||
		colNames, res, err := dbInstance.SelectData(sql)
 | 
						||
		biz.ErrIsNilAppendErr(err, "查询失败: %s")
 | 
						||
		colAndRes := make(map[string]interface{})
 | 
						||
		colAndRes["colNames"] = colNames
 | 
						||
		colAndRes["res"] = res
 | 
						||
		rc.ResData = colAndRes
 | 
						||
	} else {
 | 
						||
		rowsAffected, err := dbInstance.Exec(sql)
 | 
						||
		biz.ErrIsNilAppendErr(err, "执行失败: %s")
 | 
						||
		res := make([]map[string]string, 0)
 | 
						||
		resData := make(map[string]string)
 | 
						||
		resData["影响条数"] = fmt.Sprintf("%d", rowsAffected)
 | 
						||
		res = append(res, resData)
 | 
						||
 | 
						||
		colAndRes := make(map[string]interface{})
 | 
						||
		colAndRes["colNames"] = []string{"影响条数"}
 | 
						||
		colAndRes["res"] = res
 | 
						||
 | 
						||
		rc.ResData = colAndRes
 | 
						||
	}
 | 
						||
}
 | 
						||
 | 
						||
// 执行sql文件
 | 
						||
func (d *Db) ExecSqlFile(rc *ctx.ReqCtx) {
 | 
						||
	g := rc.GinCtx
 | 
						||
	fileheader, err := g.FormFile("file")
 | 
						||
	biz.ErrIsNilAppendErr(err, "读取sql文件失败: %s")
 | 
						||
 | 
						||
	// 读取sql文件并根据;切割sql语句
 | 
						||
	file, _ := fileheader.Open()
 | 
						||
	filename := fileheader.Filename
 | 
						||
	bytes, _ := ioutil.ReadAll(file)
 | 
						||
	sqlContent := string(bytes)
 | 
						||
	sqls := strings.Split(sqlContent, ";")
 | 
						||
	dbId := GetDbId(g)
 | 
						||
 | 
						||
	go func() {
 | 
						||
		db := d.DbApp.GetDbInstance(dbId)
 | 
						||
 | 
						||
		dbEntity := d.DbApp.GetById(dbId)
 | 
						||
		dbInfo := fmt.Sprintf("于%s的%s环境", dbEntity.Name, dbEntity.Env)
 | 
						||
 | 
						||
		defer func() {
 | 
						||
			if err := recover(); err != nil {
 | 
						||
				switch t := err.(type) {
 | 
						||
				case *biz.BizError:
 | 
						||
					d.MsgApp.CreateAndSend(rc.LoginAccount, ws.ErrMsg("sql脚本执行失败", fmt.Sprintf("[%s]%s执行失败: [%s]", filename, dbInfo, t.Error())))
 | 
						||
				}
 | 
						||
			}
 | 
						||
		}()
 | 
						||
 | 
						||
		biz.IsTrue(d.ProjectApp.CanAccess(rc.LoginAccount.Id, db.ProjectId), "您无权操作该资源")
 | 
						||
 | 
						||
		for _, sql := range sqls {
 | 
						||
			sql = strings.Trim(sql, " ")
 | 
						||
			if sql == "" || sql == "\n" {
 | 
						||
				continue
 | 
						||
			}
 | 
						||
			_, err := db.Exec(sql)
 | 
						||
			if err != nil {
 | 
						||
				d.MsgApp.CreateAndSend(rc.LoginAccount, ws.ErrMsg("sql脚本执行失败", fmt.Sprintf("[%s]%s执行失败: [%s]", filename, dbInfo, err.Error())))
 | 
						||
				return
 | 
						||
			}
 | 
						||
		}
 | 
						||
		d.MsgApp.CreateAndSend(rc.LoginAccount, ws.SuccessMsg("sql脚本执行成功", fmt.Sprintf("[%s]%s执行完成", filename, dbInfo)))
 | 
						||
	}()
 | 
						||
}
 | 
						||
 | 
						||
// @router /api/db/:dbId/t-metadata [get]
 | 
						||
func (d *Db) TableMA(rc *ctx.ReqCtx) {
 | 
						||
	dbi := d.DbApp.GetDbInstance(GetDbId(rc.GinCtx))
 | 
						||
	biz.IsTrue(d.ProjectApp.CanAccess(rc.LoginAccount.Id, dbi.ProjectId), "您无权操作该资源")
 | 
						||
	rc.ResData = dbi.GetTableMetedatas()
 | 
						||
}
 | 
						||
 | 
						||
// @router /api/db/:dbId/c-metadata [get]
 | 
						||
func (d *Db) ColumnMA(rc *ctx.ReqCtx) {
 | 
						||
	g := rc.GinCtx
 | 
						||
	tn := g.Query("tableName")
 | 
						||
	biz.NotEmpty(tn, "tableName不能为空")
 | 
						||
 | 
						||
	dbi := d.DbApp.GetDbInstance(GetDbId(rc.GinCtx))
 | 
						||
	biz.IsTrue(d.ProjectApp.CanAccess(rc.LoginAccount.Id, dbi.ProjectId), "您无权操作该资源")
 | 
						||
	rc.ResData = dbi.GetColumnMetadatas(tn)
 | 
						||
}
 | 
						||
 | 
						||
// @router /api/db/:dbId/hint-tables [get]
 | 
						||
func (d *Db) HintTables(rc *ctx.ReqCtx) {
 | 
						||
	dbi := d.DbApp.GetDbInstance(GetDbId(rc.GinCtx))
 | 
						||
	biz.IsTrue(d.ProjectApp.CanAccess(rc.LoginAccount.Id, dbi.ProjectId), "您无权操作该资源")
 | 
						||
	// 获取所有表
 | 
						||
	tables := dbi.GetTableMetedatas()
 | 
						||
 | 
						||
	tableNames := make([]string, 0)
 | 
						||
	for _, v := range tables {
 | 
						||
		tableNames = append(tableNames, v["tableName"])
 | 
						||
	}
 | 
						||
	// key = 表名,value = 列名数组
 | 
						||
	res := make(map[string][]string)
 | 
						||
 | 
						||
	// 表为空,则直接返回
 | 
						||
	if len(tableNames) == 0 {
 | 
						||
		rc.ResData = res
 | 
						||
		return
 | 
						||
	}
 | 
						||
 | 
						||
	// 获取所有表下的所有列信息
 | 
						||
	columnMds := dbi.GetColumnMetadatas(tableNames...)
 | 
						||
	for _, v := range columnMds {
 | 
						||
		tName := v["tableName"]
 | 
						||
		if res[tName] == nil {
 | 
						||
			res[tName] = make([]string, 0)
 | 
						||
		}
 | 
						||
 | 
						||
		columnName := fmt.Sprintf("%s  [%s]", v["columnName"], v["columnType"])
 | 
						||
		comment := v["columnComment"]
 | 
						||
		// 如果字段备注不为空,则加上备注信息
 | 
						||
		if comment != "" {
 | 
						||
			columnName = fmt.Sprintf("%s[%s]", columnName, comment)
 | 
						||
		}
 | 
						||
 | 
						||
		res[tName] = append(res[tName], columnName)
 | 
						||
	}
 | 
						||
	rc.ResData = res
 | 
						||
}
 | 
						||
 | 
						||
// @router /api/db/:dbId/sql [post]
 | 
						||
func (d *Db) SaveSql(rc *ctx.ReqCtx) {
 | 
						||
	g := rc.GinCtx
 | 
						||
	account := rc.LoginAccount
 | 
						||
	dbSqlForm := &form.DbSqlSaveForm{}
 | 
						||
	ginx.BindJsonAndValid(g, dbSqlForm)
 | 
						||
	rc.ReqParam = dbSqlForm
 | 
						||
 | 
						||
	dbId := GetDbId(g)
 | 
						||
	// 判断dbId是否存在
 | 
						||
	err := model.GetById(new(entity.Db), dbId)
 | 
						||
	biz.ErrIsNil(err, "该数据库信息不存在")
 | 
						||
 | 
						||
	// 获取用于是否有该dbsql的保存记录,有则更改,否则新增
 | 
						||
	dbSql := &entity.DbSql{Type: dbSqlForm.Type, DbId: dbId, Name: dbSqlForm.Name}
 | 
						||
	dbSql.CreatorId = account.Id
 | 
						||
	e := model.GetBy(dbSql)
 | 
						||
 | 
						||
	dbSql.SetBaseInfo(account)
 | 
						||
	// 更新sql信息
 | 
						||
	dbSql.Sql = dbSqlForm.Sql
 | 
						||
	if e == nil {
 | 
						||
		model.UpdateById(dbSql)
 | 
						||
	} else {
 | 
						||
		model.Insert(dbSql)
 | 
						||
	}
 | 
						||
}
 | 
						||
 | 
						||
// 获取所有保存的sql names
 | 
						||
func (d *Db) GetSqlNames(rc *ctx.ReqCtx) {
 | 
						||
	// 获取用于是否有该dbsql的保存记录,有则更改,否则新增
 | 
						||
	dbSql := &entity.DbSql{Type: 1, DbId: GetDbId(rc.GinCtx)}
 | 
						||
	dbSql.CreatorId = rc.LoginAccount.Id
 | 
						||
	var sqls []entity.DbSql
 | 
						||
	model.ListBy(dbSql, &sqls, "id", "name")
 | 
						||
 | 
						||
	rc.ResData = sqls
 | 
						||
}
 | 
						||
 | 
						||
// 删除保存的sql
 | 
						||
func (d *Db) DeleteSql(rc *ctx.ReqCtx) {
 | 
						||
	dbSql := &entity.DbSql{Type: 1, DbId: GetDbId(rc.GinCtx)}
 | 
						||
	dbSql.CreatorId = rc.LoginAccount.Id
 | 
						||
	dbSql.Name = rc.GinCtx.Query("name")
 | 
						||
 | 
						||
	model.DeleteByCondition(dbSql)
 | 
						||
 | 
						||
}
 | 
						||
 | 
						||
// @router /api/db/:dbId/sql [get]
 | 
						||
func (d *Db) GetSql(rc *ctx.ReqCtx) {
 | 
						||
	// 根据创建者id, 数据库id,以及sql模板名称查询保存的sql信息
 | 
						||
	dbSql := &entity.DbSql{Type: 1, DbId: GetDbId(rc.GinCtx)}
 | 
						||
	dbSql.CreatorId = rc.LoginAccount.Id
 | 
						||
	dbSql.Name = rc.GinCtx.Query("name")
 | 
						||
 | 
						||
	e := model.GetBy(dbSql)
 | 
						||
	if e != nil {
 | 
						||
		return
 | 
						||
	}
 | 
						||
	rc.ResData = dbSql
 | 
						||
}
 | 
						||
 | 
						||
func GetDbId(g *gin.Context) uint64 {
 | 
						||
	dbId, _ := strconv.Atoi(g.Param("dbId"))
 | 
						||
	biz.IsTrue(dbId > 0, "dbId错误")
 | 
						||
	return uint64(dbId)
 | 
						||
}
 |