mirror of
https://github.com/steveiliop56/tinyauth.git
synced 2026-06-02 17:40:14 +00:00
198 lines
4.1 KiB
Go
198 lines
4.1 KiB
Go
package service
|
|
|
|
import (
|
|
"slices"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
type CacheStoreActions[T any] struct {
|
|
Set func(key string, value T, ttl time.Duration)
|
|
Get func(key string) (T, bool)
|
|
Delete func(key string)
|
|
Update func(key string, value T, ttl time.Duration) bool
|
|
}
|
|
|
|
type cacheEntry[T any] struct {
|
|
value T
|
|
expiresAt *time.Time
|
|
}
|
|
|
|
type CacheStore[T any] struct {
|
|
cache map[string]cacheEntry[T]
|
|
order []string
|
|
mu sync.RWMutex
|
|
maxSize int
|
|
}
|
|
|
|
func NewCacheStore[T any](maxSize int) *CacheStore[T] {
|
|
return &CacheStore[T]{
|
|
cache: make(map[string]cacheEntry[T]),
|
|
order: make([]string, 0),
|
|
maxSize: maxSize,
|
|
}
|
|
}
|
|
|
|
// With lock allows performing multiple operations on the cache store atomically.
|
|
// The provided mutate function receives a set of actions (Set, Get, Delete) that
|
|
// can be used to manipulate the cache store within the locked context.
|
|
func (cs *CacheStore[T]) WithLock(mutate func(actions CacheStoreActions[T])) {
|
|
cs.mu.Lock()
|
|
defer cs.mu.Unlock()
|
|
actions := CacheStoreActions[T]{
|
|
Set: cs.setCallback,
|
|
Get: cs.getCallback,
|
|
Delete: cs.deleteCallback,
|
|
Update: cs.updateCallback,
|
|
}
|
|
mutate(actions)
|
|
}
|
|
|
|
func (cs *CacheStore[T]) updateCallback(key string, value T, ttl time.Duration) bool {
|
|
if currentEntry, exists := cs.cache[key]; exists {
|
|
if currentEntry.expiresAt != nil && time.Now().After(*currentEntry.expiresAt) {
|
|
return false
|
|
}
|
|
|
|
entry := cacheEntry[T]{
|
|
value: value,
|
|
expiresAt: currentEntry.expiresAt,
|
|
}
|
|
|
|
if ttl > 0 {
|
|
expiration := time.Now().Add(ttl)
|
|
entry.expiresAt = &expiration
|
|
}
|
|
|
|
cs.cache[key] = entry
|
|
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func (cs *CacheStore[T]) Update(key string, value T, ttl time.Duration) bool {
|
|
cs.mu.Lock()
|
|
defer cs.mu.Unlock()
|
|
return cs.updateCallback(key, value, ttl)
|
|
}
|
|
|
|
func (cs *CacheStore[T]) setCallback(key string, value T, ttl time.Duration) {
|
|
if cs.maxSize > 0 {
|
|
if _, exists := cs.cache[key]; !exists && len(cs.cache) >= cs.maxSize {
|
|
cs.evictOne()
|
|
}
|
|
}
|
|
|
|
var expiresAt *time.Time
|
|
|
|
if ttl > 0 {
|
|
expiration := time.Now().Add(ttl)
|
|
expiresAt = &expiration
|
|
}
|
|
|
|
cs.cache[key] = cacheEntry[T]{
|
|
value: value,
|
|
expiresAt: expiresAt,
|
|
}
|
|
|
|
if !slices.Contains(cs.order, key) {
|
|
cs.order = append(cs.order, key)
|
|
}
|
|
}
|
|
|
|
func (cs *CacheStore[T]) Set(key string, value T, ttl time.Duration) {
|
|
cs.mu.Lock()
|
|
defer cs.mu.Unlock()
|
|
cs.setCallback(key, value, ttl)
|
|
}
|
|
|
|
func (cs *CacheStore[T]) getCallback(key string) (T, bool) {
|
|
entry, exists := cs.cache[key]
|
|
|
|
if !exists {
|
|
var zero T
|
|
return zero, false
|
|
}
|
|
|
|
if entry.expiresAt != nil && time.Now().After(*entry.expiresAt) {
|
|
var zero T
|
|
return zero, false
|
|
}
|
|
|
|
return entry.value, true
|
|
}
|
|
|
|
func (cs *CacheStore[T]) Get(key string) (T, bool) {
|
|
cs.mu.RLock()
|
|
defer cs.mu.RUnlock()
|
|
return cs.getCallback(key)
|
|
}
|
|
|
|
func (cs *CacheStore[T]) deleteCallback(key string) {
|
|
delete(cs.cache, key)
|
|
keyIdx := slices.Index(cs.order, key)
|
|
if keyIdx != -1 {
|
|
cs.order = append(cs.order[:keyIdx], cs.order[keyIdx+1:]...)
|
|
}
|
|
}
|
|
|
|
func (cs *CacheStore[T]) Delete(key string) {
|
|
cs.mu.Lock()
|
|
defer cs.mu.Unlock()
|
|
cs.deleteCallback(key)
|
|
}
|
|
|
|
func (cs *CacheStore[T]) Sweep() {
|
|
cs.mu.Lock()
|
|
for key, entry := range cs.cache {
|
|
if entry.expiresAt != nil && time.Now().After(*entry.expiresAt) {
|
|
cs.deleteCallback(key)
|
|
}
|
|
}
|
|
cs.mu.Unlock()
|
|
}
|
|
|
|
func (cs *CacheStore[T]) evictOne() bool {
|
|
now := time.Now()
|
|
var oldestKey string
|
|
var oldestExp *time.Time
|
|
|
|
for k, e := range cs.cache {
|
|
if e.expiresAt != nil && now.After(*e.expiresAt) {
|
|
cs.deleteCallback(k)
|
|
return true
|
|
}
|
|
if e.expiresAt != nil && (oldestExp == nil || e.expiresAt.Before(*oldestExp)) {
|
|
oldestKey, oldestExp = k, e.expiresAt
|
|
}
|
|
}
|
|
|
|
// If we found an oldest key, evict it else we delete the first key in the order list
|
|
if oldestKey != "" {
|
|
cs.deleteCallback(oldestKey)
|
|
return true
|
|
} else {
|
|
if len(cs.order) > 0 {
|
|
cs.deleteCallback(cs.order[0])
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func (cs *CacheStore[T]) Size() int {
|
|
cs.mu.RLock()
|
|
defer cs.mu.RUnlock()
|
|
return len(cs.cache)
|
|
}
|
|
|
|
func (cs *CacheStore[T]) Clear() {
|
|
cs.mu.Lock()
|
|
defer cs.mu.Unlock()
|
|
cs.cache = make(map[string]cacheEntry[T])
|
|
cs.order = make([]string, 0)
|
|
}
|