Files
mayfly-go/server/internal/db/api/db.go

460 lines
13 KiB
Go
Raw Normal View History

2021-09-11 14:04:09 +08:00
package api
2021-01-08 15:37:32 +08:00
import (
"fmt"
"io"
2022-09-09 18:26:08 +08:00
"mayfly-go/internal/db/api/form"
"mayfly-go/internal/db/api/vo"
"mayfly-go/internal/db/application"
"mayfly-go/internal/db/domain/entity"
sysapp "mayfly-go/internal/sys/application"
2022-10-26 20:49:29 +08:00
tagapp "mayfly-go/internal/tag/application"
"mayfly-go/pkg/biz"
"mayfly-go/pkg/ginx"
"mayfly-go/pkg/gormx"
"mayfly-go/pkg/model"
2023-01-14 16:29:52 +08:00
"mayfly-go/pkg/req"
"mayfly-go/pkg/utils"
"mayfly-go/pkg/ws"
2021-01-08 15:37:32 +08:00
"strconv"
2021-04-21 10:22:09 +08:00
"strings"
"time"
2021-01-08 15:37:32 +08:00
2021-04-16 15:10:07 +08:00
"github.com/gin-gonic/gin"
"github.com/xwb1989/sqlparser"
2021-04-16 15:10:07 +08:00
)
2021-01-08 15:37:32 +08:00
type Db struct {
DbApp application.Db
DbSqlExecApp application.DbSqlExec
2022-09-09 18:26:08 +08:00
MsgApp sysapp.Msg
2022-10-26 20:49:29 +08:00
TagApp tagapp.TagTree
}
2023-02-22 15:54:53 +08:00
const DEFAULT_ROW_SIZE = 5000
2021-01-08 15:37:32 +08:00
// @router /api/dbs [get]
2023-01-14 16:29:52 +08:00
func (d *Db) Dbs(rc *req.Ctx) {
2022-10-26 20:49:29 +08:00
condition := new(entity.DbQuery)
condition.TagPathLike = rc.GinCtx.Query("tagPath")
// 不存在可访问标签id即没有可操作数据
tagIds := d.TagApp.ListTagIdByAccountId(rc.LoginAccount.Id)
if len(tagIds) == 0 {
rc.ResData = model.EmptyPageResult[any]()
2022-10-26 20:49:29 +08:00
return
}
2022-10-26 20:49:29 +08:00
condition.TagIds = tagIds
rc.ResData = d.DbApp.GetPageList(condition, ginx.GetPageParam(rc.GinCtx), new([]vo.SelectDataDbVO))
2021-01-08 15:37:32 +08:00
}
2023-01-14 16:29:52 +08:00
func (d *Db) Save(rc *req.Ctx) {
form := &form.DbForm{}
ginx.BindJsonAndValid(rc.GinCtx, form)
db := new(entity.Db)
utils.Copy(db, form)
// 密码解密,并使用解密后的赋值
originPwd, err := utils.DefaultRsaDecrypt(form.Password, true)
biz.ErrIsNilAppendErr(err, "解密密码错误: %s")
db.Password = originPwd
// 密码脱敏记录日志
form.Password = "****"
rc.ReqParam = form
db.SetBaseInfo(rc.LoginAccount)
d.DbApp.Save(db)
}
// 获取数据库实例密码,由于数据库是加密存储,故提供该接口展示原文密码
2023-01-14 16:29:52 +08:00
func (d *Db) GetDbPwd(rc *req.Ctx) {
dbId := GetDbId(rc.GinCtx)
dbEntity := d.DbApp.GetById(dbId, "Password")
dbEntity.PwdDecrypt()
rc.ResData = dbEntity.Password
}
// 获取数据库实例的所有数据库名
2023-01-14 16:29:52 +08:00
func (d *Db) GetDatabaseNames(rc *req.Ctx) {
form := &form.DbForm{}
ginx.BindJsonAndValid(rc.GinCtx, form)
db := new(entity.Db)
utils.Copy(db, form)
// 密码解密,并使用解密后的赋值
originPwd, err := utils.DefaultRsaDecrypt(form.Password, true)
biz.ErrIsNilAppendErr(err, "解密密码错误: %s")
db.Password = originPwd
// 如果id不为空并且密码为空则从数据库查询
if form.Id != 0 && db.Password == "" {
db = d.DbApp.GetById(form.Id)
}
rc.ResData = d.DbApp.GetDatabases(db)
}
2023-01-14 16:29:52 +08:00
func (d *Db) DeleteDb(rc *req.Ctx) {
idsStr := ginx.PathParam(rc.GinCtx, "dbId")
rc.ReqParam = idsStr
ids := strings.Split(idsStr, ",")
for _, v := range ids {
value, err := strconv.Atoi(v)
biz.ErrIsNilAppendErr(err, "string类型转换为int异常: %s")
dbId := uint64(value)
d.DbApp.Delete(dbId)
// 删除该库的sql执行记录
d.DbSqlExecApp.DeleteBy(&entity.DbSqlExec{DbId: dbId})
}
}
2023-01-14 16:29:52 +08:00
func (d *Db) TableInfos(rc *req.Ctx) {
2022-08-10 19:46:17 +08:00
rc.ResData = d.DbApp.GetDbInstance(GetIdAndDb(rc.GinCtx)).GetMeta().GetTableInfos()
}
2023-01-14 16:29:52 +08:00
func (d *Db) TableIndex(rc *req.Ctx) {
tn := rc.GinCtx.Query("tableName")
biz.NotEmpty(tn, "tableName不能为空")
2022-08-10 19:46:17 +08:00
rc.ResData = d.DbApp.GetDbInstance(GetIdAndDb(rc.GinCtx)).GetMeta().GetTableIndex(tn)
}
2023-01-14 16:29:52 +08:00
func (d *Db) GetCreateTableDdl(rc *req.Ctx) {
tn := rc.GinCtx.Query("tableName")
biz.NotEmpty(tn, "tableName不能为空")
2022-08-10 19:46:17 +08:00
rc.ResData = d.DbApp.GetDbInstance(GetIdAndDb(rc.GinCtx)).GetMeta().GetCreateTableDdl(tn)
}
2023-01-14 16:29:52 +08:00
func (d *Db) ExecSql(rc *req.Ctx) {
2021-04-16 15:10:07 +08:00
g := rc.GinCtx
form := &form.DbSqlExecForm{}
ginx.BindJsonAndValid(g, form)
id := GetDbId(g)
db := form.Db
dbInstance := d.DbApp.GetDbInstance(id, db)
2022-11-18 17:52:30 +08:00
biz.ErrIsNilAppendErr(d.TagApp.CanAccess(rc.LoginAccount.Id, dbInstance.Info.TagPath), "%s")
2022-12-17 22:24:21 +08:00
rc.ReqParam = fmt.Sprintf("%s\n-> %s", dbInstance.Info.GetLogDesc(), form.Sql)
2022-11-02 19:27:40 +08:00
biz.NotEmpty(form.Sql, "sql不能为空")
2022-11-02 19:27:40 +08:00
// 去除前后空格及换行符
sql := utils.StrTrimSpaceAndBr(form.Sql)
2022-11-02 19:27:40 +08:00
execReq := &application.DbSqlExecReq{
DbId: id,
Db: db,
Remark: form.Remark,
DbInstance: dbInstance,
LoginAccount: rc.LoginAccount,
2022-11-02 19:27:40 +08:00
}
2023-06-17 08:59:37 +08:00
sqls, err := sqlparser.SplitStatementToPieces(sql)
biz.ErrIsNil(err, "SQL解析错误,请检查您的执行SQL")
2022-11-02 19:27:40 +08:00
isMulti := len(sqls) > 1
var execResAll *application.DbSqlExecRes
for _, s := range sqls {
s = utils.StrTrimSpaceAndBr(s)
// 多条执行,如果有查询语句,则跳过
if isMulti && strings.HasPrefix(strings.ToLower(s), "select") {
continue
}
execReq.Sql = s
execRes, err := d.DbSqlExecApp.Exec(execReq)
2022-12-17 22:24:21 +08:00
if err != nil {
biz.ErrIsNilAppendErr(err, fmt.Sprintf("[%s] -> 执行失败: ", s)+"%s")
}
2022-11-02 19:27:40 +08:00
if execResAll == nil {
execResAll = execRes
} else {
execResAll.Merge(execRes)
}
}
colAndRes := make(map[string]any)
2022-11-02 19:27:40 +08:00
colAndRes["colNames"] = execResAll.ColNames
colAndRes["res"] = execResAll.Res
rc.ResData = colAndRes
2021-01-08 15:37:32 +08:00
}
// 执行sql文件
2023-01-14 16:29:52 +08:00
func (d *Db) ExecSqlFile(rc *req.Ctx) {
g := rc.GinCtx
fileheader, err := g.FormFile("file")
biz.ErrIsNilAppendErr(err, "读取sql文件失败: %s")
file, _ := fileheader.Open()
filename := fileheader.Filename
dbId, db := GetIdAndDb(g)
2022-12-17 22:24:21 +08:00
dbInstance := d.DbApp.GetDbInstance(dbId, db)
biz.ErrIsNilAppendErr(d.TagApp.CanAccess(rc.LoginAccount.Id, dbInstance.Info.TagPath), "%s")
rc.ReqParam = fmt.Sprintf("%s -> filename: %s", dbInstance.Info.GetLogDesc(), filename)
2022-12-17 22:24:21 +08:00
logExecRecord := true
// 如果执行sql文件大于该值则不记录sql执行记录
2023-02-22 15:54:53 +08:00
if fileheader.Size > 50*1024 {
2022-12-17 22:24:21 +08:00
logExecRecord = false
}
2022-12-17 22:24:21 +08:00
go func() {
defer func() {
if err := recover(); err != nil {
switch t := err.(type) {
case *biz.BizError:
2022-12-17 22:24:21 +08:00
d.MsgApp.CreateAndSend(rc.LoginAccount, ws.ErrMsg("sql脚本执行失败", fmt.Sprintf("[%s]%s执行失败: [%s]", filename, dbInstance.Info.GetLogDesc(), t.Error())))
}
}
}()
2022-12-17 22:24:21 +08:00
execReq := &application.DbSqlExecReq{
DbId: dbId,
Db: db,
Remark: fileheader.Filename,
DbInstance: dbInstance,
LoginAccount: rc.LoginAccount,
}
tokens := sqlparser.NewTokenizer(file)
for {
stmt, err := sqlparser.ParseNext(tokens)
if err == io.EOF {
break
}
sql := sqlparser.String(stmt)
2022-12-17 22:24:21 +08:00
execReq.Sql = sql
// 需要记录执行记录
if logExecRecord {
_, err = d.DbSqlExecApp.Exec(execReq)
} else {
_, err = dbInstance.Exec(sql)
}
if err != nil {
2022-12-17 22:24:21 +08:00
d.MsgApp.CreateAndSend(rc.LoginAccount, ws.ErrMsg("sql脚本执行失败", fmt.Sprintf("[%s][%s] -> sql=[%s] 执行失败: [%s]", filename, dbInstance.Info.GetLogDesc(), sql, err.Error())))
return
}
}
2022-12-17 22:24:21 +08:00
d.MsgApp.CreateAndSend(rc.LoginAccount, ws.SuccessMsg("sql脚本执行成功", fmt.Sprintf("[%s]执行完成 -> %s", filename, dbInstance.Info.GetLogDesc())))
}()
}
// 数据库dump
2023-01-14 16:29:52 +08:00
func (d *Db) DumpSql(rc *req.Ctx) {
g := rc.GinCtx
dbId, db := GetIdAndDb(g)
dumpType := g.Query("type")
tablesStr := g.Query("tables")
biz.NotEmpty(tablesStr, "请选择要导出的表")
tables := strings.Split(tablesStr, ",")
// 是否需要导出表结构
needStruct := dumpType == "1" || dumpType == "3"
// 是否需要导出数据
needData := dumpType == "2" || dumpType == "3"
dbInstance := d.DbApp.GetDbInstance(dbId, db)
2022-11-18 17:52:30 +08:00
biz.ErrIsNilAppendErr(d.TagApp.CanAccess(rc.LoginAccount.Id, dbInstance.Info.TagPath), "%s")
now := time.Now()
filename := fmt.Sprintf("%s.%s.sql", db, now.Format("200601021504"))
g.Header("Content-Type", "application/octet-stream")
g.Header("Content-Disposition", "attachment; filename="+filename)
writer := g.Writer
writer.WriteString("-- ----------------------------")
writer.WriteString("\n-- 导出平台: mayfly-go")
writer.WriteString(fmt.Sprintf("\n-- 导出时间: %s ", now.Format("2006-01-02 15:04:05")))
writer.WriteString(fmt.Sprintf("\n-- 导出数据库: %s ", db))
writer.WriteString("\n-- ----------------------------\n")
2022-08-10 19:46:17 +08:00
dbmeta := d.DbApp.GetDbInstance(GetIdAndDb(rc.GinCtx)).GetMeta()
for _, table := range tables {
if needStruct {
writer.WriteString(fmt.Sprintf("\n-- ----------------------------\n-- 表结构: %s \n-- ----------------------------\n", table))
writer.WriteString(fmt.Sprintf("DROP TABLE IF EXISTS `%s`;\n", table))
2023-05-24 12:32:17 +08:00
writer.WriteString(dbmeta.GetCreateTableDdl(table) + ";\n")
}
if !needData {
continue
}
writer.WriteString(fmt.Sprintf("\n-- ----------------------------\n-- 表记录: %s \n-- ----------------------------\n", table))
writer.WriteString("BEGIN;\n")
2022-12-17 22:24:21 +08:00
pageNum := 1
for {
columns, result, _ := dbmeta.GetTableRecord(table, pageNum, DEFAULT_ROW_SIZE)
resultLen := len(result)
if resultLen == 0 {
break
}
insertSql := "INSERT INTO `%s` VALUES (%s);\n"
for _, res := range result {
var values []string
for _, column := range columns {
value := res[column]
if value == nil {
values = append(values, "NULL")
continue
}
strValue, ok := value.(string)
if ok {
values = append(values, fmt.Sprintf("%#v", strValue))
} else {
values = append(values, utils.ToString(value))
}
}
writer.WriteString(fmt.Sprintf(insertSql, table, strings.Join(values, ", ")))
}
2022-12-17 22:24:21 +08:00
if resultLen < DEFAULT_ROW_SIZE {
break
}
pageNum++
}
writer.WriteString("COMMIT;\n")
}
rc.NoRes = true
2022-12-17 22:24:21 +08:00
rc.ReqParam = fmt.Sprintf("%s, tables: %s, dumpType: %s", dbInstance.Info.GetLogDesc(), tablesStr, dumpType)
}
2021-01-08 15:37:32 +08:00
// @router /api/db/:dbId/t-metadata [get]
2023-01-14 16:29:52 +08:00
func (d *Db) TableMA(rc *req.Ctx) {
dbi := d.DbApp.GetDbInstance(GetIdAndDb(rc.GinCtx))
2022-08-10 19:46:17 +08:00
rc.ResData = dbi.GetMeta().GetTables()
2021-01-08 15:37:32 +08:00
}
// @router /api/db/:dbId/c-metadata [get]
2023-01-14 16:29:52 +08:00
func (d *Db) ColumnMA(rc *req.Ctx) {
2021-04-16 15:10:07 +08:00
g := rc.GinCtx
tn := g.Query("tableName")
biz.NotEmpty(tn, "tableName不能为空")
dbi := d.DbApp.GetDbInstance(GetIdAndDb(rc.GinCtx))
2022-08-10 19:46:17 +08:00
rc.ResData = dbi.GetMeta().GetColumns(tn)
2021-01-08 15:37:32 +08:00
}
// @router /api/db/:dbId/hint-tables [get]
2023-01-14 16:29:52 +08:00
func (d *Db) HintTables(rc *req.Ctx) {
dbi := d.DbApp.GetDbInstance(GetIdAndDb(rc.GinCtx))
2022-08-10 19:46:17 +08:00
dm := dbi.GetMeta()
// 获取所有表
tables := dm.GetTables()
tableNames := make([]string, 0)
2021-04-16 15:10:07 +08:00
for _, v := range tables {
2023-05-24 12:32:17 +08:00
tableNames = append(tableNames, v.TableName)
}
// key = 表名value = 列名数组
res := make(map[string][]string)
// 表为空,则直接返回
if len(tableNames) == 0 {
rc.ResData = res
return
}
// 获取所有表下的所有列信息
2022-08-10 19:46:17 +08:00
columnMds := dm.GetColumns(tableNames...)
for _, v := range columnMds {
2023-05-24 12:32:17 +08:00
tName := v.TableName
if res[tName] == nil {
res[tName] = make([]string, 0)
2021-01-08 15:37:32 +08:00
}
2023-05-24 12:32:17 +08:00
columnName := fmt.Sprintf("%s [%s]", v.ColumnName, v.ColumnType)
comment := v.ColumnComment
// 如果字段备注不为空,则加上备注信息
2023-05-24 12:32:17 +08:00
if comment != "" {
columnName = fmt.Sprintf("%s[%s]", columnName, comment)
}
res[tName] = append(res[tName], columnName)
2021-04-16 15:10:07 +08:00
}
rc.ResData = res
2021-01-08 15:37:32 +08:00
}
// @router /api/db/:dbId/sql [post]
2023-01-14 16:29:52 +08:00
func (d *Db) SaveSql(rc *req.Ctx) {
2021-04-16 15:10:07 +08:00
g := rc.GinCtx
account := rc.LoginAccount
dbSqlForm := &form.DbSqlSaveForm{}
ginx.BindJsonAndValid(g, dbSqlForm)
rc.ReqParam = dbSqlForm
2021-01-08 15:37:32 +08:00
2021-04-16 15:10:07 +08:00
dbId := GetDbId(g)
// 判断dbId是否存在
err := gormx.GetById(new(entity.Db), dbId)
biz.ErrIsNil(err, "该数据库信息不存在")
2021-01-08 15:37:32 +08:00
2021-04-16 15:10:07 +08:00
// 获取用于是否有该dbsql的保存记录有则更改否则新增
dbSql := &entity.DbSql{Type: dbSqlForm.Type, DbId: dbId, Name: dbSqlForm.Name, Db: dbSqlForm.Db}
2021-04-16 15:10:07 +08:00
dbSql.CreatorId = account.Id
e := gormx.GetBy(dbSql)
2021-01-08 15:37:32 +08:00
2021-04-16 15:10:07 +08:00
dbSql.SetBaseInfo(account)
// 更新sql信息
dbSql.Sql = dbSqlForm.Sql
if e == nil {
gormx.UpdateById(dbSql)
2021-04-16 15:10:07 +08:00
} else {
gormx.Insert(dbSql)
2021-04-16 15:10:07 +08:00
}
2021-01-08 15:37:32 +08:00
}
// 获取所有保存的sql names
2023-01-14 16:29:52 +08:00
func (d *Db) GetSqlNames(rc *req.Ctx) {
id, db := GetIdAndDb(rc.GinCtx)
// 获取用于是否有该dbsql的保存记录有则更改否则新增
dbSql := &entity.DbSql{Type: 1, DbId: id, Db: db}
dbSql.CreatorId = rc.LoginAccount.Id
var sqls []entity.DbSql
gormx.ListBy(dbSql, &sqls, "id", "name")
rc.ResData = sqls
}
// 删除保存的sql
2023-01-14 16:29:52 +08:00
func (d *Db) DeleteSql(rc *req.Ctx) {
dbSql := &entity.DbSql{Type: 1, DbId: GetDbId(rc.GinCtx)}
dbSql.CreatorId = rc.LoginAccount.Id
dbSql.Name = rc.GinCtx.Query("name")
dbSql.Db = rc.GinCtx.Query("db")
gormx.DeleteByCondition(dbSql)
}
2021-01-08 15:37:32 +08:00
// @router /api/db/:dbId/sql [get]
2023-01-14 16:29:52 +08:00
func (d *Db) GetSql(rc *req.Ctx) {
id, db := GetIdAndDb(rc.GinCtx)
// 根据创建者id 数据库id以及sql模板名称查询保存的sql信息
dbSql := &entity.DbSql{Type: 1, DbId: id, Db: db}
2021-04-16 15:10:07 +08:00
dbSql.CreatorId = rc.LoginAccount.Id
dbSql.Name = rc.GinCtx.Query("name")
e := gormx.GetBy(dbSql)
2021-04-16 15:10:07 +08:00
if e != nil {
return
}
rc.ResData = dbSql
2021-01-08 15:37:32 +08:00
}
2021-04-16 15:10:07 +08:00
func GetDbId(g *gin.Context) uint64 {
dbId, _ := strconv.Atoi(g.Param("dbId"))
2021-03-24 17:18:39 +08:00
biz.IsTrue(dbId > 0, "dbId错误")
2021-01-08 15:37:32 +08:00
return uint64(dbId)
}
func GetIdAndDb(g *gin.Context) (uint64, string) {
db := g.Query("db")
biz.NotEmpty(db, "db不能为空")
return GetDbId(g), db
}