2021-07-28 18:03:19 +08:00
|
|
|
|
package application
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
2023-11-07 21:05:21 +08:00
|
|
|
|
"context"
|
2023-12-05 23:03:51 +08:00
|
|
|
|
"mayfly-go/internal/common/consts"
|
2024-03-02 19:08:19 +08:00
|
|
|
|
flowapp "mayfly-go/internal/flow/application"
|
|
|
|
|
|
flowentity "mayfly-go/internal/flow/domain/entity"
|
2022-09-09 18:26:08 +08:00
|
|
|
|
"mayfly-go/internal/redis/domain/entity"
|
|
|
|
|
|
"mayfly-go/internal/redis/domain/repository"
|
2023-10-27 17:41:45 +08:00
|
|
|
|
"mayfly-go/internal/redis/rdm"
|
2023-12-05 23:03:51 +08:00
|
|
|
|
tagapp "mayfly-go/internal/tag/application"
|
2024-04-13 17:01:12 +08:00
|
|
|
|
tagentity "mayfly-go/internal/tag/domain/entity"
|
2023-10-26 17:15:49 +08:00
|
|
|
|
"mayfly-go/pkg/base"
|
|
|
|
|
|
"mayfly-go/pkg/errorx"
|
2024-03-02 19:08:19 +08:00
|
|
|
|
"mayfly-go/pkg/logx"
|
2022-06-02 17:41:11 +08:00
|
|
|
|
"mayfly-go/pkg/model"
|
2024-03-02 19:08:19 +08:00
|
|
|
|
"mayfly-go/pkg/utils/jsonx"
|
2023-12-05 23:03:51 +08:00
|
|
|
|
"mayfly-go/pkg/utils/stringx"
|
2022-09-29 13:14:50 +08:00
|
|
|
|
"strconv"
|
2022-07-10 12:14:06 +08:00
|
|
|
|
"strings"
|
2024-03-02 19:08:19 +08:00
|
|
|
|
|
2024-04-13 17:01:12 +08:00
|
|
|
|
"github.com/may-fly/cast"
|
2024-03-02 19:08:19 +08:00
|
|
|
|
"github.com/redis/go-redis/v9"
|
2021-07-28 18:03:19 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
2024-04-13 17:01:12 +08:00
|
|
|
|
type SaveRedisParam struct {
|
2024-04-17 21:28:28 +08:00
|
|
|
|
Redis *entity.Redis
|
|
|
|
|
|
AuthCert *tagentity.ResourceAuthCert
|
|
|
|
|
|
TagCodePaths []string
|
2024-04-13 17:01:12 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-03-02 19:08:19 +08:00
|
|
|
|
type RunCmdParam struct {
|
|
|
|
|
|
Id uint64 `json:"id"`
|
|
|
|
|
|
Db int `json:"db"`
|
|
|
|
|
|
Cmd []any `json:"cmd"`
|
|
|
|
|
|
Remark string
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2021-07-28 18:03:19 +08:00
|
|
|
|
type Redis interface {
|
2023-10-26 17:15:49 +08:00
|
|
|
|
base.App[*entity.Redis]
|
2024-03-02 19:08:19 +08:00
|
|
|
|
flowapp.FlowBizHandler
|
2023-10-26 17:15:49 +08:00
|
|
|
|
|
2021-07-28 18:03:19 +08:00
|
|
|
|
// 分页获取机器脚本信息列表
|
2023-10-26 17:15:49 +08:00
|
|
|
|
GetPageList(condition *entity.RedisQuery, pageParam *model.PageParam, toEntity any, orderBy ...string) (*model.PageResult[any], error)
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
2023-11-12 20:14:44 +08:00
|
|
|
|
// 测试连接
|
2024-04-13 17:01:12 +08:00
|
|
|
|
TestConn(re *SaveRedisParam) error
|
2023-11-12 20:14:44 +08:00
|
|
|
|
|
2024-04-13 17:01:12 +08:00
|
|
|
|
SaveRedis(ctx context.Context, param *SaveRedisParam) error
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
|
|
|
|
|
// 删除数据库信息
|
2023-11-07 21:05:21 +08:00
|
|
|
|
Delete(ctx context.Context, id uint64) error
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
|
|
|
|
|
// 获取数据库连接实例
|
2022-09-29 13:14:50 +08:00
|
|
|
|
// id: 数据库实例id
|
|
|
|
|
|
// db: 库号
|
2023-10-27 17:41:45 +08:00
|
|
|
|
GetRedisConn(id uint64, db int) (*rdm.RedisConn, error)
|
2024-03-02 19:08:19 +08:00
|
|
|
|
|
|
|
|
|
|
// 执行redis命令
|
|
|
|
|
|
RunCmd(ctx context.Context, redisConn *rdm.RedisConn, cmdParam *RunCmdParam) (any, error)
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2022-09-09 18:26:08 +08:00
|
|
|
|
type redisAppImpl struct {
|
2023-10-26 17:15:49 +08:00
|
|
|
|
base.AppImpl[*entity.Redis, repository.Redis]
|
2023-12-05 23:03:51 +08:00
|
|
|
|
|
2024-04-13 17:01:12 +08:00
|
|
|
|
tagApp tagapp.TagTree `inject:"TagTreeApp"`
|
2024-05-08 21:04:25 +08:00
|
|
|
|
procdefApp flowapp.Procdef `inject:"ProcdefApp"`
|
2024-04-13 17:01:12 +08:00
|
|
|
|
procinstApp flowapp.Procinst `inject:"ProcinstApp"`
|
|
|
|
|
|
resourceAuthCertApp tagapp.ResourceAuthCert `inject:"ResourceAuthCertApp"`
|
2024-01-21 22:52:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 注入RedisRepo
|
|
|
|
|
|
func (r *redisAppImpl) InjectRedisRepo(repo repository.Redis) {
|
|
|
|
|
|
r.Repo = repo
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-10-27 17:41:45 +08:00
|
|
|
|
// 分页获取redis列表
|
2023-10-26 17:15:49 +08:00
|
|
|
|
func (r *redisAppImpl) GetPageList(condition *entity.RedisQuery, pageParam *model.PageParam, toEntity any, orderBy ...string) (*model.PageResult[any], error) {
|
|
|
|
|
|
return r.GetRepo().GetRedisList(condition, pageParam, toEntity, orderBy...)
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-04-13 17:01:12 +08:00
|
|
|
|
func (r *redisAppImpl) TestConn(param *SaveRedisParam) error {
|
2023-11-12 20:14:44 +08:00
|
|
|
|
db := 0
|
2024-04-13 17:01:12 +08:00
|
|
|
|
re := param.Redis
|
2023-11-12 20:14:44 +08:00
|
|
|
|
if re.Db != "" {
|
2024-04-13 17:01:12 +08:00
|
|
|
|
db = cast.ToInt(strings.Split(re.Db, ",")[0])
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
authCert := param.AuthCert
|
|
|
|
|
|
if authCert.Id != 0 {
|
|
|
|
|
|
// 密文可能被清除,故需要重新获取
|
|
|
|
|
|
authCert, _ = r.resourceAuthCertApp.GetAuthCert(authCert.Name)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if authCert.CiphertextType == tagentity.AuthCertCiphertextTypePublic {
|
|
|
|
|
|
publicAuthCert, err := r.resourceAuthCertApp.GetAuthCert(authCert.Ciphertext)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
authCert = publicAuthCert
|
|
|
|
|
|
}
|
2022-07-18 20:36:31 +08:00
|
|
|
|
}
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
2024-04-13 17:01:12 +08:00
|
|
|
|
rc, err := re.ToRedisInfo(db, authCert).Conn()
|
2023-11-12 20:14:44 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
rc.Close()
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-04-13 17:01:12 +08:00
|
|
|
|
func (r *redisAppImpl) SaveRedis(ctx context.Context, param *SaveRedisParam) error {
|
|
|
|
|
|
re := param.Redis
|
2024-04-17 21:28:28 +08:00
|
|
|
|
tagCodePaths := param.TagCodePaths
|
2021-07-28 18:03:19 +08:00
|
|
|
|
// 查找是否存在该库
|
2023-12-11 01:00:09 +08:00
|
|
|
|
oldRedis := &entity.Redis{
|
|
|
|
|
|
Host: re.Host,
|
|
|
|
|
|
SshTunnelMachineId: re.SshTunnelMachineId,
|
2023-03-17 09:46:41 +08:00
|
|
|
|
}
|
2024-04-28 23:45:57 +08:00
|
|
|
|
err := r.GetByCond(oldRedis)
|
2021-07-28 18:03:19 +08:00
|
|
|
|
|
|
|
|
|
|
if re.Id == 0 {
|
|
|
|
|
|
if err == nil {
|
2023-10-26 17:15:49 +08:00
|
|
|
|
return errorx.NewBiz("该实例已存在")
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
2024-04-06 18:19:17 +08:00
|
|
|
|
if r.CountByCond(&entity.Redis{Code: re.Code}) > 0 {
|
|
|
|
|
|
return errorx.NewBiz("该编码已存在")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-12-05 23:03:51 +08:00
|
|
|
|
return r.Tx(ctx, func(ctx context.Context) error {
|
|
|
|
|
|
return r.Insert(ctx, re)
|
|
|
|
|
|
}, func(ctx context.Context) error {
|
2024-04-12 13:24:20 +08:00
|
|
|
|
return r.tagApp.SaveResourceTag(ctx, &tagapp.SaveResourceTagParam{
|
2024-04-17 21:28:28 +08:00
|
|
|
|
ResourceTag: &tagapp.ResourceTag{
|
|
|
|
|
|
Type: tagentity.TagTypeRedis,
|
|
|
|
|
|
Code: re.Code,
|
|
|
|
|
|
Name: re.Name,
|
|
|
|
|
|
},
|
|
|
|
|
|
ParentTagCodePaths: tagCodePaths,
|
2024-04-06 18:19:17 +08:00
|
|
|
|
})
|
2024-04-13 17:01:12 +08:00
|
|
|
|
}, func(ctx context.Context) error {
|
|
|
|
|
|
return r.resourceAuthCertApp.RelateAuthCert(ctx, &tagapp.RelateAuthCertParam{
|
|
|
|
|
|
ResourceCode: re.Code,
|
|
|
|
|
|
ResourceType: tagentity.TagTypeRedis,
|
|
|
|
|
|
AuthCerts: []*tagentity.ResourceAuthCert{param.AuthCert},
|
|
|
|
|
|
})
|
2023-12-05 23:03:51 +08:00
|
|
|
|
})
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
2023-10-26 17:15:49 +08:00
|
|
|
|
|
|
|
|
|
|
// 如果存在该库,则校验修改的库是否为该库
|
|
|
|
|
|
if err == nil && oldRedis.Id != re.Id {
|
|
|
|
|
|
return errorx.NewBiz("该实例已存在")
|
|
|
|
|
|
}
|
|
|
|
|
|
// 如果修改了redis实例的库信息,则关闭旧库的连接
|
2024-05-08 21:04:25 +08:00
|
|
|
|
if oldRedis.Db != re.Db || oldRedis.SshTunnelMachineId != re.SshTunnelMachineId {
|
2023-10-26 17:15:49 +08:00
|
|
|
|
for _, dbStr := range strings.Split(oldRedis.Db, ",") {
|
|
|
|
|
|
db, _ := strconv.Atoi(dbStr)
|
2023-10-27 17:41:45 +08:00
|
|
|
|
rdm.CloseConn(re.Id, db)
|
2023-10-26 17:15:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2023-12-20 23:01:51 +08:00
|
|
|
|
// 如果调整了ssh等会查不到旧数据,故需要根据id获取旧信息将code赋值给标签进行关联
|
|
|
|
|
|
if oldRedis.Code == "" {
|
2024-05-05 14:53:30 +08:00
|
|
|
|
oldRedis, _ = r.GetById(re.Id)
|
2023-12-20 23:01:51 +08:00
|
|
|
|
}
|
2023-12-05 23:03:51 +08:00
|
|
|
|
|
2024-04-06 18:19:17 +08:00
|
|
|
|
re.Code = ""
|
2023-12-05 23:03:51 +08:00
|
|
|
|
return r.Tx(ctx, func(ctx context.Context) error {
|
|
|
|
|
|
return r.UpdateById(ctx, re)
|
|
|
|
|
|
}, func(ctx context.Context) error {
|
2024-04-17 21:28:28 +08:00
|
|
|
|
if oldRedis.Name != re.Name {
|
|
|
|
|
|
if err := r.tagApp.UpdateTagName(ctx, tagentity.TagTypeMachine, oldRedis.Code, re.Name); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-04-12 13:24:20 +08:00
|
|
|
|
return r.tagApp.SaveResourceTag(ctx, &tagapp.SaveResourceTagParam{
|
2024-04-17 21:28:28 +08:00
|
|
|
|
ResourceTag: &tagapp.ResourceTag{
|
|
|
|
|
|
Type: tagentity.TagTypeRedis,
|
|
|
|
|
|
Code: oldRedis.Code,
|
|
|
|
|
|
Name: re.Name,
|
|
|
|
|
|
},
|
|
|
|
|
|
ParentTagCodePaths: tagCodePaths,
|
2024-04-06 18:19:17 +08:00
|
|
|
|
})
|
2024-04-13 17:01:12 +08:00
|
|
|
|
}, func(ctx context.Context) error {
|
|
|
|
|
|
return r.resourceAuthCertApp.RelateAuthCert(ctx, &tagapp.RelateAuthCertParam{
|
|
|
|
|
|
ResourceCode: oldRedis.Code,
|
|
|
|
|
|
ResourceType: tagentity.TagTypeRedis,
|
|
|
|
|
|
AuthCerts: []*tagentity.ResourceAuthCert{param.AuthCert},
|
|
|
|
|
|
})
|
2023-12-05 23:03:51 +08:00
|
|
|
|
})
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 删除Redis信息
|
2023-11-07 21:05:21 +08:00
|
|
|
|
func (r *redisAppImpl) Delete(ctx context.Context, id uint64) error {
|
2024-05-05 14:53:30 +08:00
|
|
|
|
re, err := r.GetById(id)
|
2023-10-26 17:15:49 +08:00
|
|
|
|
if err != nil {
|
|
|
|
|
|
return errorx.NewBiz("该redis信息不存在")
|
|
|
|
|
|
}
|
2022-09-29 13:14:50 +08:00
|
|
|
|
// 如果存在连接,则关闭所有库连接信息
|
|
|
|
|
|
for _, dbStr := range strings.Split(re.Db, ",") {
|
|
|
|
|
|
db, _ := strconv.Atoi(dbStr)
|
2023-10-27 17:41:45 +08:00
|
|
|
|
rdm.CloseConn(re.Id, db)
|
2022-09-29 13:14:50 +08:00
|
|
|
|
}
|
2023-12-13 14:01:13 +08:00
|
|
|
|
|
|
|
|
|
|
return r.Tx(ctx, func(ctx context.Context) error {
|
|
|
|
|
|
return r.DeleteById(ctx, id)
|
|
|
|
|
|
}, func(ctx context.Context) error {
|
2024-04-12 13:24:20 +08:00
|
|
|
|
return r.tagApp.SaveResourceTag(ctx, &tagapp.SaveResourceTagParam{
|
2024-04-17 21:28:28 +08:00
|
|
|
|
ResourceTag: &tagapp.ResourceTag{
|
|
|
|
|
|
Type: tagentity.TagTypeRedis,
|
|
|
|
|
|
Code: re.Code,
|
|
|
|
|
|
},
|
2024-04-06 18:19:17 +08:00
|
|
|
|
})
|
2024-04-13 17:01:12 +08:00
|
|
|
|
}, func(ctx context.Context) error {
|
|
|
|
|
|
return r.resourceAuthCertApp.RelateAuthCert(ctx, &tagapp.RelateAuthCertParam{
|
|
|
|
|
|
ResourceCode: re.Code,
|
|
|
|
|
|
ResourceType: tagentity.TagTypeRedis,
|
|
|
|
|
|
})
|
2023-12-13 14:01:13 +08:00
|
|
|
|
})
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获取数据库连接实例
|
2023-10-27 17:41:45 +08:00
|
|
|
|
func (r *redisAppImpl) GetRedisConn(id uint64, db int) (*rdm.RedisConn, error) {
|
|
|
|
|
|
return rdm.GetRedisConn(id, db, func() (*rdm.RedisInfo, error) {
|
|
|
|
|
|
// 缓存不存在,则回调获取redis信息
|
2024-05-05 14:53:30 +08:00
|
|
|
|
re, err := r.GetById(id)
|
2023-10-26 17:15:49 +08:00
|
|
|
|
if err != nil {
|
2023-10-27 17:41:45 +08:00
|
|
|
|
return nil, errorx.NewBiz("redis信息不存在")
|
2023-10-26 17:15:49 +08:00
|
|
|
|
}
|
2024-04-13 17:01:12 +08:00
|
|
|
|
authCert, err := r.resourceAuthCertApp.GetResourceAuthCert(tagentity.TagTypeRedis, re.Code)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
2024-01-05 08:55:34 +08:00
|
|
|
|
}
|
2024-04-13 17:01:12 +08:00
|
|
|
|
return re.ToRedisInfo(db, authCert, r.tagApp.ListTagPathByTypeAndCode(consts.ResourceTypeRedis, re.Code)...), nil
|
2021-08-18 17:57:33 +08:00
|
|
|
|
})
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
2024-03-02 19:08:19 +08:00
|
|
|
|
|
|
|
|
|
|
func (r *redisAppImpl) RunCmd(ctx context.Context, redisConn *rdm.RedisConn, cmdParam *RunCmdParam) (any, error) {
|
|
|
|
|
|
if redisConn == nil {
|
|
|
|
|
|
return nil, errorx.NewBiz("redis连接不存在")
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 开启工单流程,并且为写入命令,则开启对应审批流程
|
2024-05-08 21:04:25 +08:00
|
|
|
|
if procdefId := r.procdefApp.GetProcdefIdByCodePath(ctx, redisConn.Info.CodePath...); procdefId != 0 && rdm.IsWriteCmd(cmdParam.Cmd[0]) {
|
|
|
|
|
|
_, err := r.procinstApp.StartProc(ctx, procdefId, &flowapp.StarProcParam{
|
2024-03-02 19:08:19 +08:00
|
|
|
|
BizType: RedisRunWriteCmdFlowBizType,
|
|
|
|
|
|
BizKey: stringx.Rand(24),
|
|
|
|
|
|
BizForm: jsonx.ToStr(cmdParam),
|
|
|
|
|
|
Remark: cmdParam.Remark,
|
|
|
|
|
|
})
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return nil, err
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
res, err := redisConn.RunCmd(ctx, cmdParam.Cmd...)
|
|
|
|
|
|
// 获取的key不存在,不报错
|
|
|
|
|
|
if err == redis.Nil {
|
|
|
|
|
|
return nil, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
return res, err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (r *redisAppImpl) FlowBizHandle(ctx context.Context, bizHandleParam *flowapp.BizHandleParam) error {
|
|
|
|
|
|
bizKey := bizHandleParam.BizKey
|
|
|
|
|
|
procinstStatus := bizHandleParam.ProcinstStatus
|
|
|
|
|
|
|
|
|
|
|
|
logx.Debugf("RedisRunWriteCmd FlowBizHandle -> bizKey: %s, procinstStatus: %s", bizKey, flowentity.ProcinstStatusEnum.GetDesc(procinstStatus))
|
|
|
|
|
|
// 流程非完成状态,不处理
|
|
|
|
|
|
if procinstStatus != flowentity.ProcinstStatusCompleted {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
runCmdParam, err := jsonx.To(bizHandleParam.BizForm, new(RunCmdParam))
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return errorx.NewBiz("业务表单信息解析失败: %s", err.Error())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
redisConn, err := r.GetRedisConn(runCmdParam.Id, runCmdParam.Db)
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_, err = redisConn.RunCmd(ctx, runCmdParam.Cmd...)
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|