2022-06-15 11:50:17 +08:00
|
|
|
// Package redis ...
|
|
|
|
//
|
|
|
|
// Description : redis 客户端
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 2021-02-27 4:49 下午
|
|
|
|
package redis
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2022-06-15 16:40:39 +08:00
|
|
|
"path/filepath"
|
2022-06-15 11:50:17 +08:00
|
|
|
"strings"
|
2022-06-15 15:02:45 +08:00
|
|
|
"sync"
|
2022-06-15 11:50:17 +08:00
|
|
|
"time"
|
|
|
|
|
2022-06-15 15:02:45 +08:00
|
|
|
"git.zhangdeman.cn/zhangdeman/util"
|
|
|
|
|
2022-06-15 11:50:17 +08:00
|
|
|
"git.zhangdeman.cn/zhangdeman/logger"
|
|
|
|
redisInstance "github.com/go-redis/redis/v8"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
2022-06-15 15:02:45 +08:00
|
|
|
var (
|
|
|
|
// Client 连接实例
|
|
|
|
Client ClientInterface
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
Client = &OwnClient{
|
|
|
|
lock: &sync.RWMutex{},
|
|
|
|
instanceTable: make(map[string]*RealClient),
|
2022-06-15 16:40:39 +08:00
|
|
|
confTable: make(map[string]*FullConfig),
|
2022-06-15 15:02:45 +08:00
|
|
|
parseErrorFunc: defaultParseError,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-15 11:50:17 +08:00
|
|
|
// defaultParseError ...
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 10:59 下午 2021/2/27
|
|
|
|
func defaultParseError(err error) error {
|
|
|
|
if nil == err {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
errMsg := err.Error()
|
|
|
|
if errMsg == "nil" || errMsg == "<nil>" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
strArr := strings.Split(errMsg, ":")
|
|
|
|
if len(strArr) != 2 {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
msg := strings.ToLower(strings.TrimSpace(strArr[1]))
|
|
|
|
if msg == "nil" || msg == "<nil>" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Options 连接选项,百分之百兼容第三方包的选项
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 4:57 下午 2021/2/27
|
|
|
|
type Options struct {
|
|
|
|
Conf *redisInstance.Options // 第三方包的选项
|
|
|
|
Logger *LoggerConfig // 日志的配置
|
|
|
|
LoggerFieldConfig *LogFieldConfig // 日志字段的配置
|
|
|
|
}
|
|
|
|
|
|
|
|
// RealClient 包装好的 redis client
|
|
|
|
type RealClient struct {
|
|
|
|
Flag string // redis 标识
|
|
|
|
Instance *redisInstance.Client // redis 实例
|
|
|
|
Logger *zap.Logger // 日志实例
|
|
|
|
LoggerFieldConfig *LogFieldConfig // 日志字段的配置
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewClient 获取redis client实例
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 5:05 下午 2021/2/27
|
2022-06-15 16:40:39 +08:00
|
|
|
func NewClient(config map[string]*FullConfig, parseErrorFunc func(err error) error) (ClientInterface, error) {
|
2022-06-15 15:02:45 +08:00
|
|
|
c := &OwnClient{
|
2022-06-15 11:50:17 +08:00
|
|
|
instanceTable: make(map[string]*RealClient),
|
2022-06-15 15:02:45 +08:00
|
|
|
loggerTable: make(map[string]*zap.Logger),
|
2022-06-15 11:50:17 +08:00
|
|
|
confTable: config,
|
|
|
|
parseErrorFunc: parseErrorFunc,
|
|
|
|
}
|
|
|
|
if nil == c.parseErrorFunc {
|
|
|
|
c.parseErrorFunc = defaultParseError
|
|
|
|
}
|
|
|
|
return c, c.init()
|
|
|
|
}
|
|
|
|
|
2022-06-15 15:02:45 +08:00
|
|
|
// OwnClient 包装的redis client
|
2022-06-15 11:50:17 +08:00
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 4:52 下午 2021/2/27
|
2022-06-15 15:02:45 +08:00
|
|
|
type OwnClient struct {
|
|
|
|
lock *sync.RWMutex // 锁
|
|
|
|
loggerTable map[string]*zap.Logger // 日志实例
|
2022-06-15 11:50:17 +08:00
|
|
|
instanceTable map[string]*RealClient // redis 实例
|
2022-06-15 16:40:39 +08:00
|
|
|
confTable map[string]*FullConfig // redis 配置
|
2022-06-15 11:50:17 +08:00
|
|
|
parseErrorFunc func(err error) error // 解析err的function,解析执行结果是否为失败,有的场景,执行成功,返回 redis:nil / redis:<nil>
|
|
|
|
}
|
|
|
|
|
2022-06-15 15:02:45 +08:00
|
|
|
// loadConfig 载入配置文件
|
2022-06-15 11:50:17 +08:00
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
2022-06-15 15:02:45 +08:00
|
|
|
// Date : 14:31 2022/6/15
|
2022-06-15 16:47:12 +08:00
|
|
|
func (c *OwnClient) loadConfig(cfgPath string) error {
|
2022-06-15 11:50:17 +08:00
|
|
|
var (
|
|
|
|
err error
|
2022-06-15 15:02:45 +08:00
|
|
|
cfg FullConfig
|
2022-06-15 11:50:17 +08:00
|
|
|
)
|
|
|
|
|
2022-06-15 16:40:39 +08:00
|
|
|
filePathArr := strings.Split(cfgPath, string(filepath.Separator))
|
|
|
|
if len(filePathArr) == 0 {
|
2022-06-15 16:47:12 +08:00
|
|
|
return CfgFilePathError()
|
2022-06-15 15:02:45 +08:00
|
|
|
}
|
2022-06-15 16:40:39 +08:00
|
|
|
fileName := strings.ToLower(filePathArr[len(filePathArr)-1])
|
|
|
|
fileArr := strings.Split(fileName, ".")
|
|
|
|
if len(filePathArr) < 2 {
|
2022-06-15 16:47:12 +08:00
|
|
|
return CfgFileFormatErr("未知")
|
2022-06-15 16:40:39 +08:00
|
|
|
}
|
|
|
|
flag := strings.Trim(
|
|
|
|
strings.Trim(
|
|
|
|
strings.Trim(fileName, ".json"),
|
|
|
|
".yaml"), ".yml")
|
|
|
|
switch strings.ToLower(fileArr[len(fileArr)-1]) {
|
|
|
|
case "json":
|
|
|
|
if err = util.File.ReadJSONContent(cfgPath, &cfg); nil != err {
|
2022-06-15 16:47:12 +08:00
|
|
|
return err
|
2022-06-15 16:40:39 +08:00
|
|
|
}
|
|
|
|
flag = strings.Join(fileArr[0:len(fileArr)-1], ".")
|
|
|
|
case "yml":
|
|
|
|
fallthrough
|
|
|
|
case "yaml":
|
|
|
|
if err = util.File.ReadYmlContent(cfgPath, &cfg); nil != err {
|
2022-06-15 16:47:12 +08:00
|
|
|
return err
|
2022-06-15 16:40:39 +08:00
|
|
|
}
|
|
|
|
flag = strings.Join(fileArr[0:len(fileArr)-1], ".")
|
|
|
|
default:
|
2022-06-15 16:47:12 +08:00
|
|
|
return CfgFileFormatErr(fileArr[len(fileArr)-1])
|
2022-06-15 16:40:39 +08:00
|
|
|
}
|
|
|
|
|
2022-06-15 16:47:12 +08:00
|
|
|
c.lock.Lock()
|
2022-06-15 16:40:39 +08:00
|
|
|
c.confTable[flag] = &cfg
|
2022-06-15 16:47:12 +08:00
|
|
|
c.lock.Unlock()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// batchLoadConfig 批量载入配置
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 16:44 2022/6/15
|
|
|
|
func (c *OwnClient) batchLoadConfig(cfgDir string) error {
|
|
|
|
filepathNames, _ := filepath.Glob(filepath.Join(cfgDir, "*"))
|
|
|
|
for i := range filepathNames {
|
|
|
|
if err := c.loadConfig(filepathNames[i]); nil != err {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2022-06-15 15:02:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// init 初始化redis连接
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 5:31 下午 2021/2/27
|
|
|
|
func (c *OwnClient) init() error {
|
|
|
|
|
2022-06-15 16:40:39 +08:00
|
|
|
for flag, _ := range c.confTable {
|
2022-06-15 11:50:17 +08:00
|
|
|
c.instanceTable[flag] = &RealClient{
|
2022-06-15 16:40:39 +08:00
|
|
|
Flag: flag,
|
|
|
|
//Instance: redisInstance.NewClient(conf.Conf),
|
|
|
|
Logger: nil,
|
|
|
|
//LoggerFieldConfig: conf.LoggerFieldConfig,
|
2022-06-15 11:50:17 +08:00
|
|
|
}
|
|
|
|
if nil == c.instanceTable[flag].LoggerFieldConfig {
|
|
|
|
c.instanceTable[flag].LoggerFieldConfig = &LogFieldConfig{
|
|
|
|
Message: "",
|
|
|
|
UsedTimeField: "",
|
|
|
|
CommandField: "",
|
|
|
|
FlagField: "",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(c.instanceTable[flag].LoggerFieldConfig.Message) == 0 {
|
|
|
|
c.instanceTable[flag].LoggerFieldConfig.Message = defaultMessage
|
|
|
|
}
|
|
|
|
if len(c.instanceTable[flag].LoggerFieldConfig.CommandField) == 0 {
|
|
|
|
c.instanceTable[flag].LoggerFieldConfig.CommandField = defaultCommandField
|
|
|
|
}
|
|
|
|
if len(c.instanceTable[flag].LoggerFieldConfig.UsedTimeField) == 0 {
|
|
|
|
c.instanceTable[flag].LoggerFieldConfig.UsedTimeField = defaultUsedTimeField
|
|
|
|
}
|
|
|
|
if len(c.instanceTable[flag].LoggerFieldConfig.FlagField) == 0 {
|
|
|
|
c.instanceTable[flag].LoggerFieldConfig.FlagField = defaultFlagField
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-06-15 15:02:45 +08:00
|
|
|
// initLogger 初始化日志
|
2022-06-15 11:50:17 +08:00
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 7:07 下午 2021/2/27
|
2022-06-15 15:02:45 +08:00
|
|
|
func (c *OwnClient) initLogger(flag string, conf *logger.InputLogConfig) error {
|
|
|
|
var err error
|
|
|
|
c.loggerTable[flag], err = logger.GetLogInstanceFromInputConfig(conf)
|
|
|
|
return LoggerInitFail(flag, err)
|
2022-06-15 11:50:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetRedisClient 获取redis实例
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 5:16 下午 2021/2/27
|
2022-06-15 15:02:45 +08:00
|
|
|
func (c *OwnClient) GetRedisClient(flag string) (*RealClient, error) {
|
2022-06-15 11:50:17 +08:00
|
|
|
redisClient, exist := c.instanceTable[flag]
|
|
|
|
if !exist {
|
|
|
|
return nil, FlagNotFound(flag)
|
|
|
|
}
|
|
|
|
return redisClient, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// log 记录redis请求日志
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 8:52 下午 2021/2/27
|
2022-06-15 15:02:45 +08:00
|
|
|
func (c *OwnClient) log(ctx *Context, realClient *RealClient, cmdResult redisInstance.Cmder, startTime int64, finishTime int64) {
|
2022-06-15 11:50:17 +08:00
|
|
|
if nil == realClient || nil == realClient.Logger {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
realClient.Logger.Info(
|
|
|
|
"执行redis命令日志记录",
|
|
|
|
zap.Any(ctx.RequestIDField, ctx.RequestID), // 上下文串联的requestID
|
|
|
|
zap.String(realClient.LoggerFieldConfig.CommandField, cmdResult.String()), // 执行的命令
|
|
|
|
zap.Float64(realClient.LoggerFieldConfig.UsedTimeField, float64(finishTime-startTime)/1e6), // 耗时,单位: ms
|
|
|
|
zap.Error(cmdResult.Err()), // 异常信息
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// CommandProxy 执行命令的代理
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 9:41 下午 2021/2/27
|
2022-06-15 15:02:45 +08:00
|
|
|
func (c *OwnClient) CommandProxy(ctx *Context, flag string, cmd string, param ...interface{}) (string, error) {
|
2022-06-15 11:50:17 +08:00
|
|
|
var (
|
|
|
|
realClient *RealClient
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
if len(cmd) == 0 {
|
|
|
|
return "", EmptyCmd()
|
|
|
|
}
|
|
|
|
|
|
|
|
if nil == ctx {
|
|
|
|
ctx = NewContext(flag)
|
|
|
|
}
|
|
|
|
if realClient, err = c.GetRedisClient(ctx.Flag); nil != err {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
redisCmd := append([]interface{}{cmd}, param...)
|
|
|
|
startTime := time.Now().Unix()
|
|
|
|
cmdResult := realClient.Instance.Do(ctx.Ctx, redisCmd...)
|
|
|
|
go c.log(ctx, realClient, cmdResult, startTime, time.Now().UnixNano())
|
|
|
|
return fmt.Sprintf("%v", cmdResult.Val()), c.parseErrorFunc(cmdResult.Err())
|
|
|
|
}
|
|
|
|
|
|
|
|
// CommandProxyWithReceiver 执行命令,并解析结果
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 10:00 下午 2021/2/27
|
2022-06-15 15:02:45 +08:00
|
|
|
func (c *OwnClient) CommandProxyWithReceiver(ctx *Context, flag string, receiver interface{}, cmd string, param ...interface{}) error {
|
2022-06-15 11:50:17 +08:00
|
|
|
if nil == receiver {
|
|
|
|
return ReceiverISNIL()
|
|
|
|
}
|
|
|
|
var (
|
|
|
|
err error
|
|
|
|
result string
|
|
|
|
)
|
|
|
|
|
|
|
|
if result, err = c.CommandProxy(ctx, flag, cmd, param...); nil != err {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ResultConvertFail(json.Unmarshal([]byte(result), receiver))
|
|
|
|
}
|
|
|
|
|
|
|
|
// ClientInterface 定义redis client的接口实现,方便单元测试数据mock
|
|
|
|
//
|
|
|
|
// Author : go_developer@163.com<白茶清欢>
|
|
|
|
//
|
|
|
|
// Date : 10:49 下午 2021/2/27
|
|
|
|
type ClientInterface interface {
|
|
|
|
GetRedisClient(flag string) (*RealClient, error)
|
|
|
|
CommandProxy(ctx *Context, flag string, cmd string, param ...interface{}) (string, error)
|
|
|
|
CommandProxyWithReceiver(ctx *Context, flag string, receiver interface{}, cmd string, param ...interface{}) error
|
|
|
|
}
|