status-go/vendor/github.com/hashicorp/golang-lru/lru.go

117 lines
2.7 KiB
Go
Raw Normal View History

2016-06-20 14:47:10 +00:00
package lru
import (
"sync"
"github.com/hashicorp/golang-lru/simplelru"
)
// Cache is a thread-safe fixed size LRU cache.
type Cache struct {
2019-06-09 07:24:20 +00:00
lru simplelru.LRUCache
2016-06-20 14:47:10 +00:00
lock sync.RWMutex
}
2019-06-09 07:24:20 +00:00
// New creates an LRU of the given size.
2016-06-20 14:47:10 +00:00
func New(size int) (*Cache, error) {
return NewWithEvict(size, nil)
}
// NewWithEvict constructs a fixed size cache with the given eviction
// callback.
func NewWithEvict(size int, onEvicted func(key interface{}, value interface{})) (*Cache, error) {
lru, err := simplelru.NewLRU(size, simplelru.EvictCallback(onEvicted))
if err != nil {
return nil, err
}
c := &Cache{
lru: lru,
}
return c, nil
}
2019-06-09 07:24:20 +00:00
// Purge is used to completely clear the cache.
2016-06-20 14:47:10 +00:00
func (c *Cache) Purge() {
c.lock.Lock()
c.lru.Purge()
c.lock.Unlock()
}
// Add adds a value to the cache. Returns true if an eviction occurred.
2019-06-09 07:24:20 +00:00
func (c *Cache) Add(key, value interface{}) (evicted bool) {
2016-06-20 14:47:10 +00:00
c.lock.Lock()
2019-06-09 07:24:20 +00:00
evicted = c.lru.Add(key, value)
c.lock.Unlock()
return evicted
2016-06-20 14:47:10 +00:00
}
// Get looks up a key's value from the cache.
2019-06-09 07:24:20 +00:00
func (c *Cache) Get(key interface{}) (value interface{}, ok bool) {
2016-06-20 14:47:10 +00:00
c.lock.Lock()
2019-06-09 07:24:20 +00:00
value, ok = c.lru.Get(key)
c.lock.Unlock()
return value, ok
2016-06-20 14:47:10 +00:00
}
2019-06-09 07:24:20 +00:00
// Contains checks if a key is in the cache, without updating the
// recent-ness or deleting it for being stale.
2016-06-20 14:47:10 +00:00
func (c *Cache) Contains(key interface{}) bool {
c.lock.RLock()
2019-06-09 07:24:20 +00:00
containKey := c.lru.Contains(key)
c.lock.RUnlock()
return containKey
2016-06-20 14:47:10 +00:00
}
2019-06-09 07:24:20 +00:00
// Peek returns the key value (or undefined if not found) without updating
2016-06-20 14:47:10 +00:00
// the "recently used"-ness of the key.
2019-06-09 07:24:20 +00:00
func (c *Cache) Peek(key interface{}) (value interface{}, ok bool) {
2016-06-20 14:47:10 +00:00
c.lock.RLock()
2019-06-09 07:24:20 +00:00
value, ok = c.lru.Peek(key)
c.lock.RUnlock()
return value, ok
2016-06-20 14:47:10 +00:00
}
// ContainsOrAdd checks if a key is in the cache without updating the
// recent-ness or deleting it for being stale, and if not, adds the value.
// Returns whether found and whether an eviction occurred.
2019-06-09 07:24:20 +00:00
func (c *Cache) ContainsOrAdd(key, value interface{}) (ok, evicted bool) {
2016-06-20 14:47:10 +00:00
c.lock.Lock()
defer c.lock.Unlock()
if c.lru.Contains(key) {
return true, false
}
2019-06-09 07:24:20 +00:00
evicted = c.lru.Add(key, value)
return false, evicted
2016-06-20 14:47:10 +00:00
}
// Remove removes the provided key from the cache.
func (c *Cache) Remove(key interface{}) {
c.lock.Lock()
c.lru.Remove(key)
c.lock.Unlock()
}
// RemoveOldest removes the oldest item from the cache.
func (c *Cache) RemoveOldest() {
c.lock.Lock()
c.lru.RemoveOldest()
c.lock.Unlock()
}
// Keys returns a slice of the keys in the cache, from oldest to newest.
func (c *Cache) Keys() []interface{} {
c.lock.RLock()
2019-06-09 07:24:20 +00:00
keys := c.lru.Keys()
c.lock.RUnlock()
return keys
2016-06-20 14:47:10 +00:00
}
// Len returns the number of items in the cache.
func (c *Cache) Len() int {
c.lock.RLock()
2019-06-09 07:24:20 +00:00
length := c.lru.Len()
c.lock.RUnlock()
return length
2016-06-20 14:47:10 +00:00
}