mirror of
https://github.com/charlienet/go-mixed.git
synced 2025-07-17 16:12:42 +08:00
95 lines
2.1 KiB
Go
95 lines
2.1 KiB
Go
package redis
|
||
|
||
import (
|
||
"context"
|
||
"net"
|
||
"strings"
|
||
|
||
"github.com/redis/go-redis/v9"
|
||
)
|
||
|
||
var (
|
||
// sequentials = sets.NewHashSet("RENAME", "RENAMENX", "MGET", "BLPOP", "BRPOP", "RPOPLPUSH", "SDIFFSTORE", "SINTER")
|
||
)
|
||
|
||
type renameKey struct {
|
||
prefix string
|
||
separator string
|
||
}
|
||
|
||
func (r renameKey) DialHook(next redis.DialHook) redis.DialHook {
|
||
return func(ctx context.Context, network, addr string) (net.Conn, error) {
|
||
return next(ctx, network, addr)
|
||
}
|
||
}
|
||
|
||
func (r renameKey) ProcessPipelineHook(next redis.ProcessPipelineHook) redis.ProcessPipelineHook {
|
||
return func(ctx context.Context, cmds []redis.Cmder) error {
|
||
|
||
// 对多个KEY进行更名操作
|
||
for i := 0; i < len(cmds); i++ {
|
||
r.renameKey(cmds[i])
|
||
}
|
||
|
||
return next(ctx, cmds)
|
||
}
|
||
}
|
||
|
||
func (r renameKey) ProcessHook(next redis.ProcessHook) redis.ProcessHook {
|
||
return func(ctx context.Context, cmd redis.Cmder) error {
|
||
r.renameKey(cmd)
|
||
next(ctx, cmd)
|
||
|
||
return nil
|
||
}
|
||
}
|
||
|
||
func (r renameKey) renameKey(cmd redis.Cmder) {
|
||
if len(r.prefix) == 0 {
|
||
return
|
||
}
|
||
|
||
args := cmd.Args()
|
||
if len(args) == 1 {
|
||
return
|
||
}
|
||
|
||
switch strings.ToUpper(cmd.Name()) {
|
||
case "SELECT":
|
||
// 无KEY指令
|
||
case "RENAME", "RENAMENX", "MGET", "BLPOP", "BRPOP", "RPOPLPUSH", "SDIFFSTORE", "SINTER", "SINTERSTORE", "SUNIONSTORE":
|
||
// 连续KEY
|
||
r.rename(args, createSepuence(1, len(args), 1)...)
|
||
case "sssss":
|
||
// 除最后一个外连续键
|
||
r.rename(args, createSepuence(1, len(args)-1, 1)...)
|
||
case "MSET", "MSETNX":
|
||
// 间隔KEY,KEY位置规则1,3,5,7
|
||
r.rename(args, createSepuence(1, len(args), 2)...)
|
||
default:
|
||
// 默认第一个参数为键值
|
||
r.rename(args, 1)
|
||
}
|
||
}
|
||
|
||
func (r renameKey) rename(args []any, indexes ...int) {
|
||
for _, i := range indexes {
|
||
if key, ok := args[i].(string); ok {
|
||
var builder strings.Builder
|
||
builder.WriteString(r.prefix)
|
||
builder.WriteString(r.separator)
|
||
builder.WriteString(key)
|
||
|
||
args[i] = builder.String()
|
||
}
|
||
}
|
||
}
|
||
|
||
func createSepuence(start, end, step int) []int {
|
||
ret := make([]int, 0, (end-start)/step+1)
|
||
for i := start; i <= end; i += step {
|
||
ret = append(ret, i)
|
||
}
|
||
return ret
|
||
}
|