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

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(),
},
}
}