支持存储配置到 redis

This commit is contained in:
dragon
2025-07-23 11:24:47 +08:00
parent 12eb21a5cd
commit 1e0fa15c42
7 changed files with 202 additions and 106 deletions

View File

@@ -1,13 +1,13 @@
package utils
import (
"encoding/json"
"context"
"fmt"
"io"
"os"
"sync"
"net/url"
"strconv"
"strings"
"time"
"go.uber.org/zap"
)
const TtlKeep = -1
@@ -19,92 +19,32 @@ type KVConfig interface {
io.Closer
}
// ConfigMemory 一个简单的内存配置归档,仅用于测试
type ConfigMemory struct {
data sync.Map
store string
}
func NewConfigMemory(store string) (KVConfig, error) {
ret := &ConfigMemory{
store: store,
data: sync.Map{},
func NewAutoConfig(src string) (KVConfig, error) {
if src == "" ||
strings.HasPrefix(src, "./") ||
strings.HasPrefix(src, "/") ||
strings.HasPrefix(src, "\\") ||
strings.HasPrefix(src, ".\\") {
return NewConfigMemory(src)
}
if store != "" {
item := make(map[string]ConfigContent)
data, err := os.ReadFile(store)
if err != nil && !os.IsNotExist(err) {
parse, err := url.Parse(src)
if err != nil {
return nil, err
}
switch parse.Scheme {
case "local":
return NewConfigMemory(parse.Path)
case "redis":
query := parse.Query()
addr := query.Get("addr")
pass := query.Get("pass")
db := query.Get("db")
dbi, err := strconv.Atoi(db)
if err != nil {
return nil, err
}
if err == nil {
err = json.Unmarshal(data, &item)
if err != nil {
return nil, err
}
}
for key, content := range item {
if content.Ttl == nil || time.Now().Before(*content.Ttl) {
ret.data.Store(key, content)
}
}
clear(item)
return NewConfigRedis(context.Background(), addr, pass, dbi)
default:
return nil, fmt.Errorf("unsupported scheme: %s", parse.Scheme)
}
return ret, nil
}
type ConfigContent struct {
Data string `json:"data"`
Ttl *time.Time `json:"ttl,omitempty"`
}
func (m *ConfigMemory) Put(key string, value string, ttl time.Duration) error {
d := time.Now().Add(ttl)
td := &d
if ttl == -1 {
td = nil
}
m.data.Store(key, ConfigContent{
Data: value,
Ttl: td,
})
return nil
}
func (m *ConfigMemory) Get(key string) (string, error) {
if value, ok := m.data.Load(key); ok {
content := value.(ConfigContent)
if content.Ttl != nil && time.Now().After(*content.Ttl) {
return "", os.ErrNotExist
}
return content.Data, nil
}
return "", os.ErrNotExist
}
func (m *ConfigMemory) Delete(key string) error {
m.data.Delete(key)
return nil
}
func (m *ConfigMemory) Close() error {
defer m.data.Clear()
if m.store != "" {
item := make(map[string]ConfigContent)
now := time.Now()
m.data.Range(
func(key, value interface{}) bool {
content := value.(ConfigContent)
if content.Ttl == nil || now.Before(*content.Ttl) {
item[key.(string)] = content
}
return true
})
zap.L().Debug("回写内容到本地存储", zap.String("store", m.store), zap.Int("length", len(item)))
saved, err := json.Marshal(item)
if err != nil {
return err
}
return os.WriteFile(m.store, saved, 0o600)
}
return nil
}