You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
175 lines
4.8 KiB
175 lines
4.8 KiB
package util
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"github.com/karlseguin/ccache/v2"
|
|
"github.com/seaweedfs/seaweedfs/weed/glog"
|
|
)
|
|
|
|
// CacheableStore defines the interface for stores that can be cached
|
|
type CacheableStore[T any] interface {
|
|
Get(ctx context.Context, filerAddress string, key string) (T, error)
|
|
Store(ctx context.Context, filerAddress string, key string, value T) error
|
|
Delete(ctx context.Context, filerAddress string, key string) error
|
|
List(ctx context.Context, filerAddress string) ([]string, error)
|
|
}
|
|
|
|
// CopyFunction defines how to deep copy cached values
|
|
type CopyFunction[T any] func(T) T
|
|
|
|
// CachedStore provides generic TTL caching for any store type
|
|
type CachedStore[T any] struct {
|
|
baseStore CacheableStore[T]
|
|
cache *ccache.Cache
|
|
listCache *ccache.Cache
|
|
copyFunc CopyFunction[T]
|
|
ttl time.Duration
|
|
listTTL time.Duration
|
|
}
|
|
|
|
// CachedStoreConfig holds configuration for the generic cached store
|
|
type CachedStoreConfig struct {
|
|
TTL time.Duration
|
|
ListTTL time.Duration
|
|
MaxCacheSize int64
|
|
}
|
|
|
|
// NewCachedStore creates a new generic cached store
|
|
func NewCachedStore[T any](
|
|
baseStore CacheableStore[T],
|
|
copyFunc CopyFunction[T],
|
|
config CachedStoreConfig,
|
|
) *CachedStore[T] {
|
|
// Apply defaults
|
|
if config.TTL == 0 {
|
|
config.TTL = 5 * time.Minute
|
|
}
|
|
if config.ListTTL == 0 {
|
|
config.ListTTL = 1 * time.Minute
|
|
}
|
|
if config.MaxCacheSize == 0 {
|
|
config.MaxCacheSize = 1000
|
|
}
|
|
|
|
// Create ccache instances
|
|
pruneCount := config.MaxCacheSize >> 3
|
|
if pruneCount <= 0 {
|
|
pruneCount = 100
|
|
}
|
|
|
|
return &CachedStore[T]{
|
|
baseStore: baseStore,
|
|
cache: ccache.New(ccache.Configure().MaxSize(config.MaxCacheSize).ItemsToPrune(uint32(pruneCount))),
|
|
listCache: ccache.New(ccache.Configure().MaxSize(100).ItemsToPrune(10)),
|
|
copyFunc: copyFunc,
|
|
ttl: config.TTL,
|
|
listTTL: config.ListTTL,
|
|
}
|
|
}
|
|
|
|
// Get retrieves an item with caching
|
|
func (c *CachedStore[T]) Get(ctx context.Context, filerAddress string, key string) (T, error) {
|
|
// Try cache first
|
|
item := c.cache.Get(key)
|
|
if item != nil {
|
|
// Cache hit - return cached item (DO NOT extend TTL)
|
|
value := item.Value().(T)
|
|
glog.V(4).Infof("Cache hit for key %s", key)
|
|
return c.copyFunc(value), nil
|
|
}
|
|
|
|
// Cache miss - fetch from base store
|
|
glog.V(4).Infof("Cache miss for key %s, fetching from store", key)
|
|
value, err := c.baseStore.Get(ctx, filerAddress, key)
|
|
if err != nil {
|
|
var zero T
|
|
return zero, err
|
|
}
|
|
|
|
// Cache the result with TTL
|
|
c.cache.Set(key, c.copyFunc(value), c.ttl)
|
|
glog.V(3).Infof("Cached key %s with TTL %v", key, c.ttl)
|
|
return value, nil
|
|
}
|
|
|
|
// Store stores an item and invalidates cache
|
|
func (c *CachedStore[T]) Store(ctx context.Context, filerAddress string, key string, value T) error {
|
|
// Store in base store
|
|
err := c.baseStore.Store(ctx, filerAddress, key, value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Invalidate cache entries
|
|
c.cache.Delete(key)
|
|
c.listCache.Clear() // Invalidate list cache
|
|
|
|
glog.V(3).Infof("Stored and invalidated cache for key %s", key)
|
|
return nil
|
|
}
|
|
|
|
// Delete deletes an item and invalidates cache
|
|
func (c *CachedStore[T]) Delete(ctx context.Context, filerAddress string, key string) error {
|
|
// Delete from base store
|
|
err := c.baseStore.Delete(ctx, filerAddress, key)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Invalidate cache entries
|
|
c.cache.Delete(key)
|
|
c.listCache.Clear() // Invalidate list cache
|
|
|
|
glog.V(3).Infof("Deleted and invalidated cache for key %s", key)
|
|
return nil
|
|
}
|
|
|
|
// List lists all items with caching
|
|
func (c *CachedStore[T]) List(ctx context.Context, filerAddress string) ([]string, error) {
|
|
const listCacheKey = "item_list"
|
|
|
|
// Try list cache first
|
|
item := c.listCache.Get(listCacheKey)
|
|
if item != nil {
|
|
// Cache hit - return cached list (DO NOT extend TTL)
|
|
items := item.Value().([]string)
|
|
glog.V(4).Infof("List cache hit, returning %d items", len(items))
|
|
return append([]string(nil), items...), nil // Return a copy
|
|
}
|
|
|
|
// Cache miss - fetch from base store
|
|
glog.V(4).Infof("List cache miss, fetching from store")
|
|
items, err := c.baseStore.List(ctx, filerAddress)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Cache the result with TTL (store a copy)
|
|
itemsCopy := append([]string(nil), items...)
|
|
c.listCache.Set(listCacheKey, itemsCopy, c.listTTL)
|
|
glog.V(3).Infof("Cached list with %d entries, TTL %v", len(items), c.listTTL)
|
|
return items, nil
|
|
}
|
|
|
|
// ClearCache clears all cached entries
|
|
func (c *CachedStore[T]) ClearCache() {
|
|
c.cache.Clear()
|
|
c.listCache.Clear()
|
|
glog.V(2).Infof("Cleared all cache entries")
|
|
}
|
|
|
|
// GetCacheStats returns cache statistics
|
|
func (c *CachedStore[T]) GetCacheStats() map[string]interface{} {
|
|
return map[string]interface{}{
|
|
"itemCache": map[string]interface{}{
|
|
"size": c.cache.ItemCount(),
|
|
"ttl": c.ttl.String(),
|
|
},
|
|
"listCache": map[string]interface{}{
|
|
"size": c.listCache.ItemCount(),
|
|
"ttl": c.listTTL.String(),
|
|
},
|
|
}
|
|
}
|