mirror of
https://github.com/1f349/cache.git
synced 2025-01-21 06:26:33 +00:00
Add initial cache source
This commit is contained in:
parent
23174db9a8
commit
a5f190e2fb
244
cache.go
Normal file
244
cache.go
Normal file
@ -0,0 +1,244 @@
|
||||
package cache
|
||||
|
||||
import (
|
||||
"github.com/MrMelon54/rescheduler"
|
||||
"sync"
|
||||
"time"
|
||||
)
|
||||
|
||||
// timeNow is an alias used for easier testing
|
||||
var timeNow = time.Now
|
||||
|
||||
// timeUntil is the same as time.Until but uses the test specific timeNow
|
||||
func timeUntil(t time.Time) time.Duration { return t.Sub(timeNow()) }
|
||||
|
||||
// Cache manages items in a sync.Map, items are allowed to be permanent or have
|
||||
// an expiry time. The keys which have an expiry time are stored in chain.
|
||||
// chainAdd and chainDel are used to add/remove items to/from chain. These
|
||||
// actions are completed in the cleaner goroutine to be concurrency safe.
|
||||
type Cache[K comparable, V any] struct {
|
||||
items sync.Map
|
||||
chain *keyed[K] // linked list of to-expire keys
|
||||
sched *rescheduler.Rescheduler // TODO: add a mode to have the goroutine running only once after multiple Run calls
|
||||
close chan struct{}
|
||||
chainAdd chan keyed[K]
|
||||
chainDel chan K
|
||||
}
|
||||
|
||||
// keyed is the same as item but uses the key as the data for the chain based,
|
||||
// item removal scheduler. It also contains the atomic pointer to the next item
|
||||
// in the chain.
|
||||
type keyed[K any] struct {
|
||||
item[K]
|
||||
next *keyed[K]
|
||||
}
|
||||
|
||||
// item is a piece of data stored as a value in items or as a key in chain. The
|
||||
// data is stored along with the expiry date for quick access.
|
||||
type item[T any] struct {
|
||||
data T
|
||||
expires time.Time
|
||||
}
|
||||
|
||||
// HasExpired returns true if the expiry time is non-zero and is before the value
|
||||
// of timeNow.
|
||||
func (c item[T]) HasExpired() bool {
|
||||
return !c.expires.IsZero() && c.expires.Before(timeNow())
|
||||
}
|
||||
|
||||
// New creates a *Cache[K, V] ready to be used by the caller
|
||||
func New[K comparable, V any]() *Cache[K, V] {
|
||||
c := &Cache[K, V]{}
|
||||
c.sched = rescheduler.NewRescheduler(c.cleaner)
|
||||
return c
|
||||
}
|
||||
|
||||
// Close simply sends a signal to stop the cleaner goroutine. The cache is left
|
||||
// in its current state and can still be read. The only difference being that
|
||||
// items which have expired will not be garbage collected.
|
||||
func (c *Cache[K, V]) Close() {
|
||||
close(c.close)
|
||||
}
|
||||
|
||||
// cleaner handles removing expired keys. The chainAdd and chainDel channels are
|
||||
// handled here to prevent race conditions. This ensures the expiry timer can be
|
||||
// stopped before modifying the chain.
|
||||
//
|
||||
// The cleaner is stopped whenever the chain is empty due to there being no chain
|
||||
// to manage.
|
||||
func (c *Cache[K, V]) cleaner() {
|
||||
var t *time.Timer
|
||||
if c.chain == nil {
|
||||
// fake timer as this select isn't used for an empty chain
|
||||
t = &time.Timer{C: make(chan time.Time)}
|
||||
} else {
|
||||
t = time.NewTimer(timeUntil(c.chain.expires))
|
||||
}
|
||||
|
||||
for {
|
||||
select {
|
||||
case <-c.close:
|
||||
// exit the cleaner goroutine
|
||||
return
|
||||
case node := <-c.chainAdd:
|
||||
// stop the timer safely
|
||||
if !t.Stop() {
|
||||
<-t.C
|
||||
}
|
||||
// the chain will not be empty after this insert so no check is required
|
||||
c.chainInsert(node)
|
||||
|
||||
t.Reset(timeUntil(c.chain.expires))
|
||||
case key := <-c.chainDel:
|
||||
// stop the timer safely
|
||||
if !t.Stop() {
|
||||
<-t.C
|
||||
}
|
||||
c.chainSplice(key)
|
||||
|
||||
// if there is no chain then kill the expiry scheduler
|
||||
if c.chain == nil {
|
||||
return
|
||||
}
|
||||
|
||||
t.Reset(timeUntil(c.chain.expires))
|
||||
case <-t.C:
|
||||
// if there is no chain then kill the expiry scheduler
|
||||
if c.chain == nil {
|
||||
return
|
||||
}
|
||||
|
||||
// remove all expired entries
|
||||
for c.chain.HasExpired() {
|
||||
c.items.Delete(c.chain.data)
|
||||
c.chain = c.chain.next
|
||||
}
|
||||
|
||||
// if there is no chain then kill the expiry scheduler
|
||||
if c.chain == nil {
|
||||
return
|
||||
}
|
||||
|
||||
t.Reset(timeUntil(c.chain.expires))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (c *Cache[K, V]) chainInsert(node keyed[K]) {
|
||||
// quick path for an empty chain
|
||||
if c.chain == nil {
|
||||
c.chain = &node
|
||||
return
|
||||
}
|
||||
|
||||
// loop through the chain to add an item
|
||||
ring := c.chain
|
||||
for {
|
||||
if ring.next == nil {
|
||||
// add as last item
|
||||
ring.next = &node
|
||||
break
|
||||
}
|
||||
if ring.expires.After(node.expires) {
|
||||
// add between two nodes
|
||||
node.next = ring.next
|
||||
ring.next = &node
|
||||
break
|
||||
}
|
||||
// move to the next ring in the chain
|
||||
ring = ring.next
|
||||
}
|
||||
}
|
||||
|
||||
func (c *Cache[K, V]) chainSplice(key K) {
|
||||
// quick path if the first node matches
|
||||
if c.chain.data == key {
|
||||
node := c.chain
|
||||
c.chain = node.next
|
||||
node.next = nil
|
||||
return
|
||||
}
|
||||
|
||||
// loop through the chain to find an item
|
||||
ring := c.chain
|
||||
for {
|
||||
// if the node is nil then the end has been reached
|
||||
node := ring.next
|
||||
if node == nil {
|
||||
break
|
||||
}
|
||||
// if the node is found then snip it out
|
||||
if node.data == key {
|
||||
ring.next = node.next
|
||||
node.next = nil
|
||||
break
|
||||
}
|
||||
// move to the next ring in the chain
|
||||
ring = node
|
||||
}
|
||||
}
|
||||
|
||||
// GetExpires returns (value, expiry, true) for any existing key. Or (V{},
|
||||
// time.Time{}, false) for any unknown key.
|
||||
func (c *Cache[K, V]) GetExpires(key K) (V, time.Time, bool) {
|
||||
var v V // empty value
|
||||
|
||||
obj, exists := c.items.Load(key)
|
||||
if !exists {
|
||||
return v, time.Time{}, false
|
||||
}
|
||||
|
||||
i := obj.(*item[V])
|
||||
if i.HasExpired() {
|
||||
return v, time.Time{}, false
|
||||
}
|
||||
|
||||
return i.data, i.expires, true
|
||||
}
|
||||
|
||||
// Get returns (value, true) for any existing key. Or (V{}, false) for any
|
||||
// unknown key. This is equivalent to calling GetExpires and ignoring the expiry
|
||||
// time return value.
|
||||
func (c *Cache[K, V]) Get(key K) (V, bool) {
|
||||
value, _, found := c.GetExpires(key)
|
||||
return value, found
|
||||
}
|
||||
|
||||
// SetPermanent adds an item to the cache without an expiry date.
|
||||
//
|
||||
// If an item is added with the same key then this item with be overwritten.
|
||||
func (c *Cache[K, V]) SetPermanent(key K, value V) {
|
||||
i := &item[V]{data: value} // expires is not set here
|
||||
c.items.Store(key, i)
|
||||
}
|
||||
|
||||
// Set adds an item to the cache with an expiry date.
|
||||
//
|
||||
// If an item is added with the same key then this item with be overwritten.
|
||||
func (c *Cache[K, V]) Set(key K, value V, expires time.Time) {
|
||||
if expires.Before(timeNow()) {
|
||||
return
|
||||
}
|
||||
|
||||
i := &item[V]{data: value, expires: expires}
|
||||
c.items.Store(key, i)
|
||||
c.chainAdd <- keyed[K]{item: item[K]{data: key, expires: expires}}
|
||||
c.sched.Run()
|
||||
}
|
||||
|
||||
// Range calls f with every key-value pair, which has not expired, currently
|
||||
// stored in the map.
|
||||
//
|
||||
// If f returns false, range stops the iteration.
|
||||
//
|
||||
// See sync/Map.Range for implementation specific details.
|
||||
func (c *Cache[K, V]) Range(f func(key K, value V) bool) {
|
||||
c.items.Range(func(key, value any) bool {
|
||||
item := value.(*item[V])
|
||||
if item.HasExpired() {
|
||||
return true
|
||||
}
|
||||
|
||||
return f(key.(K), value.(V))
|
||||
})
|
||||
}
|
16
cache_test.go
Normal file
16
cache_test.go
Normal file
@ -0,0 +1,16 @@
|
||||
package cache
|
||||
|
||||
import (
|
||||
"github.com/stretchr/testify/assert"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func TestItem_HasExpired(t *testing.T) {
|
||||
n := time.Now()
|
||||
a := item[string]{expires: n}
|
||||
timeNow = func() time.Time { return n.Add(time.Second) }
|
||||
assert.True(t, a.HasExpired())
|
||||
a = item[string]{expires: n.Add(time.Second * 2)}
|
||||
assert.False(t, a.HasExpired())
|
||||
}
|
Loading…
Reference in New Issue
Block a user