package freelru import ( "sync" "time" ) type SyncedLRU[K comparable, V comparable] struct { mu sync.RWMutex lru *LRU[K, V] } var _ Cache[int, int] = (*SyncedLRU[int, int])(nil) // SetLifetime sets the default lifetime of LRU elements. // Lifetime 0 means "forever". func (lru *SyncedLRU[K, V]) SetLifetime(lifetime time.Duration) { lru.mu.Lock() lru.lru.SetLifetime(lifetime) lru.mu.Unlock() } // SetOnEvict sets the OnEvict callback function. // The onEvict function is called for each evicted lru entry. func (lru *SyncedLRU[K, V]) SetOnEvict(onEvict OnEvictCallback[K, V]) { lru.mu.Lock() lru.lru.SetOnEvict(onEvict) lru.mu.Unlock() } func (lru *SyncedLRU[K, V]) SetHealthCheck(healthCheck HealthCheckCallback[K, V]) { lru.mu.Lock() lru.lru.SetHealthCheck(healthCheck) lru.mu.Unlock() } // NewSynced creates a new thread-safe LRU hashmap with the given capacity. func NewSynced[K comparable, V comparable](capacity uint32, hash HashKeyCallback[K]) (*SyncedLRU[K, V], error, ) { return NewSyncedWithSize[K, V](capacity, capacity, hash) } func NewSyncedWithSize[K comparable, V comparable](capacity, size uint32, hash HashKeyCallback[K], ) (*SyncedLRU[K, V], error) { lru, err := NewWithSize[K, V](capacity, size, hash) if err != nil { return nil, err } return &SyncedLRU[K, V]{lru: lru}, nil } // Len returns the number of elements stored in the cache. func (lru *SyncedLRU[K, V]) Len() (length int) { lru.mu.RLock() length = lru.lru.Len() lru.mu.RUnlock() return } // AddWithLifetime adds a key:value to the cache with a lifetime. // Returns true, true if key was updated and eviction occurred. func (lru *SyncedLRU[K, V]) AddWithLifetime(key K, value V, lifetime time.Duration) (evicted bool) { hash := lru.lru.hash(key) lru.mu.Lock() evicted = lru.lru.addWithLifetime(hash, key, value, lifetime) lru.mu.Unlock() return } // Add adds a key:value to the cache. // Returns true, true if key was updated and eviction occurred. func (lru *SyncedLRU[K, V]) Add(key K, value V) (evicted bool) { hash := lru.lru.hash(key) lru.mu.Lock() evicted = lru.lru.add(hash, key, value) lru.mu.Unlock() return } // Get returns the value associated with the key, setting it as the most // recently used item. // If the found cache item is already expired, the evict function is called // and the return value indicates that the key was not found. func (lru *SyncedLRU[K, V]) Get(key K) (value V, ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() value, ok = lru.lru.get(hash, key) lru.mu.Unlock() return } func (lru *SyncedLRU[K, V]) GetWithLifetime(key K) (value V, lifetime time.Time, ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() value, lifetime, ok = lru.lru.getWithLifetime(hash, key) lru.mu.Unlock() return } func (lru *SyncedLRU[K, V]) GetWithLifetimeNoExpire(key K) (value V, lifetime time.Time, ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() value, lifetime, ok = lru.lru.getWithLifetimeNoExpire(hash, key) lru.mu.Unlock() return } // GetAndRefresh returns the value associated with the key, setting it as the most // recently used item. // The lifetime of the found cache item is refreshed, even if it was already expired. func (lru *SyncedLRU[K, V]) GetAndRefresh(key K) (value V, ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() value, ok = lru.lru.getAndRefresh(hash, key) lru.mu.Unlock() return } func (lru *SyncedLRU[K, V]) GetAndRefreshOrAdd(key K, constructor func() (V, bool)) (value V, updated bool, ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() value, updated, ok = lru.lru.getAndRefreshOrAdd(hash, key, constructor) if !updated && ok { lru.lru.PurgeExpired() } lru.mu.Unlock() return } // Peek looks up a key's value from the cache, without changing its recent-ness. // If the found entry is already expired, the evict function is called. func (lru *SyncedLRU[K, V]) Peek(key K) (value V, ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() value, ok = lru.lru.peek(hash, key) lru.mu.Unlock() return } func (lru *SyncedLRU[K, V]) PeekWithLifetime(key K) (value V, lifetime time.Time, ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() value, lifetime, ok = lru.lru.peekWithLifetime(hash, key) lru.mu.Unlock() return } func (lru *SyncedLRU[K, V]) UpdateLifetime(key K, value V, lifetime time.Duration) (ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() ok = lru.lru.updateLifetime(hash, key, value, lifetime) lru.mu.Unlock() return } // Contains checks for the existence of a key, without changing its recent-ness. // If the found entry is already expired, the evict function is called. func (lru *SyncedLRU[K, V]) Contains(key K) (ok bool) { hash := lru.lru.hash(key) lru.mu.Lock() ok = lru.lru.contains(hash, key) lru.mu.Unlock() return } // Remove removes the key from the cache. // The return value indicates whether the key existed or not. // The evict function is being called if the key existed. func (lru *SyncedLRU[K, V]) Remove(key K) (removed bool) { hash := lru.lru.hash(key) lru.mu.Lock() removed = lru.lru.remove(hash, key) lru.mu.Unlock() return } // RemoveOldest removes the oldest entry from the cache. // Key, value and an indicator of whether the entry has been removed is returned. // The evict function is called for the removed entry. func (lru *SyncedLRU[K, V]) RemoveOldest() (key K, value V, removed bool) { lru.mu.Lock() key, value, removed = lru.lru.RemoveOldest() lru.mu.Unlock() return } // Keys returns a slice of the keys in the cache, from oldest to newest. // Expired entries are not included. // The evict function is called for each expired item. func (lru *SyncedLRU[K, V]) Keys() (keys []K) { lru.mu.Lock() keys = lru.lru.Keys() lru.mu.Unlock() return } // Purge purges all data (key and value) from the LRU. // The evict function is called for each expired item. // The LRU metrics are reset. func (lru *SyncedLRU[K, V]) Purge() { lru.mu.Lock() lru.lru.Purge() lru.mu.Unlock() } // PurgeExpired purges all expired items from the LRU. // The evict function is called for each expired item. func (lru *SyncedLRU[K, V]) PurgeExpired() { lru.mu.Lock() lru.lru.PurgeExpired() lru.mu.Unlock() } // Metrics returns the metrics of the cache. func (lru *SyncedLRU[K, V]) Metrics() Metrics { lru.mu.Lock() metrics := lru.lru.Metrics() lru.mu.Unlock() return metrics } // ResetMetrics resets the metrics of the cache and returns the previous state. func (lru *SyncedLRU[K, V]) ResetMetrics() Metrics { lru.mu.Lock() metrics := lru.lru.ResetMetrics() lru.mu.Unlock() return metrics } // just used for debugging func (lru *SyncedLRU[K, V]) dump() { lru.mu.RLock() lru.lru.dump() lru.mu.RUnlock() } func (lru *SyncedLRU[K, V]) PrintStats() { lru.mu.RLock() lru.lru.PrintStats() lru.mu.RUnlock() }