171 lines
4.1 KiB
Go
171 lines
4.1 KiB
Go
package redis
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/gob"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"management/internal/config"
|
|
|
|
"github.com/redis/go-redis/v9"
|
|
)
|
|
|
|
var ErrRedisKeyNotFound = errors.New("redis key not found")
|
|
|
|
type IRedis interface {
|
|
Encode(a any) ([]byte, error)
|
|
Set(ctx context.Context, key string, value interface{}, expiration time.Duration) error
|
|
Del(ctx context.Context, keys ...string) error
|
|
Get(ctx context.Context, key string) (string, error)
|
|
GetBytes(ctx context.Context, key string) ([]byte, error)
|
|
Scan(ctx context.Context, cursor uint64, match string, count int64) *redis.ScanCmd
|
|
Keys(ctx context.Context, pattern string) ([]string, error)
|
|
ListKeys(ctx context.Context, pattern string, pageID int, pageSize int) ([]string, int, error)
|
|
}
|
|
|
|
type redisCache struct {
|
|
engine *redis.Client
|
|
}
|
|
|
|
var _ IRedis = (*redisCache)(nil)
|
|
|
|
func New(conf config.Redis) (*redisCache, error) {
|
|
rdb := redis.NewClient(&redis.Options{
|
|
Addr: fmt.Sprintf("%s:%d", conf.Host, conf.Port),
|
|
Password: conf.Password,
|
|
DB: conf.DB,
|
|
})
|
|
_, err := rdb.Ping(context.Background()).Result()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &redisCache{
|
|
engine: rdb,
|
|
}, nil
|
|
}
|
|
|
|
func (r *redisCache) Encode(a any) ([]byte, error) {
|
|
var b bytes.Buffer
|
|
if err := gob.NewEncoder(&b).Encode(a); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return b.Bytes(), nil
|
|
}
|
|
|
|
// Set 设置值
|
|
func (r *redisCache) Set(ctx context.Context, key string, value interface{}, expiration time.Duration) error {
|
|
return r.engine.Set(ctx, key, value, expiration).Err()
|
|
}
|
|
|
|
// Del 删除键值
|
|
func (r *redisCache) Del(ctx context.Context, keys ...string) error {
|
|
return r.engine.Del(ctx, keys...).Err()
|
|
}
|
|
|
|
// Get 获取值
|
|
func (r *redisCache) Get(ctx context.Context, key string) (string, error) {
|
|
val, err := r.engine.Get(ctx, key).Result()
|
|
if err == redis.Nil {
|
|
return "", ErrRedisKeyNotFound
|
|
} else if err != nil {
|
|
return "", fmt.Errorf("cannot get value with:[%s]: %v", key, err)
|
|
} else {
|
|
return val, nil
|
|
}
|
|
}
|
|
|
|
// GetBytes 获取值
|
|
func (r *redisCache) GetBytes(ctx context.Context, key string) ([]byte, error) {
|
|
val, err := r.engine.Get(ctx, key).Bytes()
|
|
if err == redis.Nil {
|
|
return nil, ErrRedisKeyNotFound
|
|
} else if err != nil {
|
|
return nil, fmt.Errorf("cannot get value with:[%s]: %v", key, err)
|
|
} else {
|
|
return val, nil
|
|
}
|
|
}
|
|
|
|
func (r *redisCache) Scan(ctx context.Context, cursor uint64, match string, count int64) *redis.ScanCmd {
|
|
return r.engine.Scan(ctx, cursor, match, count)
|
|
}
|
|
|
|
func (r *redisCache) Keys(ctx context.Context, pattern string) ([]string, error) {
|
|
return r.engine.Keys(ctx, pattern).Result()
|
|
}
|
|
|
|
func (r *redisCache) ListKeys(ctx context.Context, pattern string, pageID int, pageSize int) ([]string, int, error) {
|
|
all, err := r.engine.Keys(ctx, pattern).Result()
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
count := len(all)
|
|
if count == 0 {
|
|
return nil, 0, err
|
|
}
|
|
|
|
// 使用SCAN命令分页获取键
|
|
cursor := uint64(0)
|
|
var keys []string
|
|
for {
|
|
var scanResult []string
|
|
var err error
|
|
scanResult, cursor, err = r.engine.Scan(ctx, cursor, pattern, int64(pageSize)).Result()
|
|
if err != nil {
|
|
return nil, count, err
|
|
}
|
|
keys = append(keys, scanResult...)
|
|
if cursor == 0 {
|
|
break
|
|
}
|
|
}
|
|
|
|
startIndex := (pageID - 1) * pageSize
|
|
endIndex := startIndex + pageSize
|
|
if startIndex >= len(keys) {
|
|
return nil, count, nil
|
|
}
|
|
if endIndex > len(keys) {
|
|
endIndex = len(keys)
|
|
}
|
|
return keys[startIndex:endIndex], count, nil
|
|
}
|
|
|
|
// ==========================
|
|
func Encode(a any) ([]byte, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
func Set(ctx context.Context, key string, value interface{}, expiration time.Duration) error {
|
|
return nil
|
|
}
|
|
|
|
func Del(ctx context.Context, keys ...string) error {
|
|
return nil
|
|
}
|
|
|
|
func Get(ctx context.Context, key string) (string, error) {
|
|
return "", nil
|
|
}
|
|
|
|
func GetBytes(ctx context.Context, key string) ([]byte, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
func Scan(ctx context.Context, cursor uint64, match string, count int64) *redis.ScanCmd {
|
|
return nil
|
|
}
|
|
|
|
func Keys(ctx context.Context, pattern string) ([]string, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
func ListKeys(ctx context.Context, pattern string, pageID int, pageSize int) ([]string, int, error) {
|
|
return nil, 0, nil
|
|
}
|