mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-03 11:57:39 +03:00
Fix clear lru cache
This commit is contained in:
parent
ad36d3be6d
commit
8c0bf1c05e
2 changed files with 121 additions and 2 deletions
10
common/cache/lrucache.go
vendored
10
common/cache/lrucache.go
vendored
|
@ -261,9 +261,15 @@ func (c *LruCache[K, V]) Delete(key K) {
|
|||
func (c *LruCache[K, V]) Clear() {
|
||||
c.mu.Lock()
|
||||
defer c.mu.Unlock()
|
||||
for element := c.lru.Front(); element != nil; element = element.Next() {
|
||||
c.deleteElement(element)
|
||||
|
||||
if c.onEvict != nil {
|
||||
for le := c.lru.Front(); le != nil; le = le.Next() {
|
||||
c.onEvict(le.Value.key, le.Value.value)
|
||||
}
|
||||
}
|
||||
|
||||
c.lru.Init()
|
||||
c.cache = make(map[K]*list.Element[*entry[K, V]])
|
||||
}
|
||||
|
||||
func (c *LruCache[K, V]) maybeDeleteOldest() {
|
||||
|
|
113
common/cache/lrucache_test.go
vendored
Normal file
113
common/cache/lrucache_test.go
vendored
Normal file
|
@ -0,0 +1,113 @@
|
|||
package cache_test
|
||||
|
||||
import (
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/sagernet/sing/common/cache"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
)
|
||||
|
||||
func TestLRUCache(t *testing.T) {
|
||||
t.Run("basic operations", func(t *testing.T) {
|
||||
t.Parallel()
|
||||
c := cache.New[string, int]()
|
||||
|
||||
c.Store("key1", 1)
|
||||
value, exists := c.Load("key1")
|
||||
require.True(t, exists)
|
||||
require.Equal(t, 1, value)
|
||||
|
||||
value, exists = c.Load("missing")
|
||||
require.False(t, exists)
|
||||
require.Zero(t, value)
|
||||
|
||||
c.Delete("key1")
|
||||
_, exists = c.Load("key1")
|
||||
require.False(t, exists)
|
||||
})
|
||||
|
||||
t.Run("max size", func(t *testing.T) {
|
||||
t.Parallel()
|
||||
c := cache.New[string, int](cache.WithSize[string, int](2))
|
||||
|
||||
c.Store("key1", 1)
|
||||
c.Store("key2", 2)
|
||||
c.Store("key3", 3)
|
||||
|
||||
_, exists := c.Load("key1")
|
||||
require.False(t, exists)
|
||||
|
||||
value, exists := c.Load("key2")
|
||||
require.True(t, exists)
|
||||
require.Equal(t, 2, value)
|
||||
})
|
||||
|
||||
t.Run("expiration", func(t *testing.T) {
|
||||
t.Parallel()
|
||||
c := cache.New[string, int](cache.WithAge[string, int](1))
|
||||
|
||||
c.Store("key1", 1)
|
||||
|
||||
value, exists := c.Load("key1")
|
||||
require.True(t, exists)
|
||||
require.Equal(t, 1, value)
|
||||
|
||||
time.Sleep(time.Second * 2)
|
||||
|
||||
value, exists = c.Load("key1")
|
||||
require.False(t, exists)
|
||||
require.Zero(t, value)
|
||||
})
|
||||
|
||||
t.Run("clear", func(t *testing.T) {
|
||||
t.Parallel()
|
||||
evicted := make(map[string]int)
|
||||
c := cache.New[string, int](
|
||||
cache.WithEvict[string, int](func(key string, value int) {
|
||||
evicted[key] = value
|
||||
}),
|
||||
)
|
||||
|
||||
c.Store("key1", 1)
|
||||
c.Store("key2", 2)
|
||||
|
||||
c.Clear()
|
||||
|
||||
require.Equal(t, map[string]int{"key1": 1, "key2": 2}, evicted)
|
||||
_, exists := c.Load("key1")
|
||||
require.False(t, exists)
|
||||
})
|
||||
|
||||
t.Run("load or store", func(t *testing.T) {
|
||||
t.Parallel()
|
||||
c := cache.New[string, int]()
|
||||
|
||||
value, loaded := c.LoadOrStore("key1", func() int { return 1 })
|
||||
require.False(t, loaded)
|
||||
require.Equal(t, 1, value)
|
||||
|
||||
value, loaded = c.LoadOrStore("key1", func() int { return 2 })
|
||||
require.True(t, loaded)
|
||||
require.Equal(t, 1, value)
|
||||
})
|
||||
|
||||
t.Run("update age on get", func(t *testing.T) {
|
||||
t.Parallel()
|
||||
c := cache.New[string, int](
|
||||
cache.WithAge[string, int](5),
|
||||
cache.WithUpdateAgeOnGet[string, int](),
|
||||
)
|
||||
|
||||
c.Store("key1", 1)
|
||||
|
||||
time.Sleep(time.Second * 3)
|
||||
_, exists := c.Load("key1")
|
||||
require.True(t, exists)
|
||||
|
||||
time.Sleep(time.Second * 3)
|
||||
_, exists = c.Load("key1")
|
||||
require.True(t, exists)
|
||||
})
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue