mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-07 08:32:18 +01:00
145 lines
4.2 KiB
Go
145 lines
4.2 KiB
Go
/*
|
|
* Copyright 2019 gRPC authors.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
// Package cache implements caches to be used in gRPC.
|
|
package cache
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
type cacheEntry struct {
|
|
item interface{}
|
|
// Note that to avoid deadlocks (potentially caused by lock ordering),
|
|
// callback can only be called without holding cache's mutex.
|
|
callback func()
|
|
timer *time.Timer
|
|
// deleted is set to true in Remove() when the call to timer.Stop() fails.
|
|
// This can happen when the timer in the cache entry fires around the same
|
|
// time that timer.stop() is called in Remove().
|
|
deleted bool
|
|
}
|
|
|
|
// TimeoutCache is a cache with items to be deleted after a timeout.
|
|
type TimeoutCache struct {
|
|
mu sync.Mutex
|
|
timeout time.Duration
|
|
cache map[interface{}]*cacheEntry
|
|
}
|
|
|
|
// NewTimeoutCache creates a TimeoutCache with the given timeout.
|
|
func NewTimeoutCache(timeout time.Duration) *TimeoutCache {
|
|
return &TimeoutCache{
|
|
timeout: timeout,
|
|
cache: make(map[interface{}]*cacheEntry),
|
|
}
|
|
}
|
|
|
|
// Add adds an item to the cache, with the specified callback to be called when
|
|
// the item is removed from the cache upon timeout. If the item is removed from
|
|
// the cache using a call to Remove before the timeout expires, the callback
|
|
// will not be called.
|
|
//
|
|
// If the Add was successful, it returns (newly added item, true). If there is
|
|
// an existing entry for the specified key, the cache entry is not be updated
|
|
// with the specified item and it returns (existing item, false).
|
|
func (c *TimeoutCache) Add(key, item interface{}, callback func()) (interface{}, bool) {
|
|
c.mu.Lock()
|
|
defer c.mu.Unlock()
|
|
if e, ok := c.cache[key]; ok {
|
|
return e.item, false
|
|
}
|
|
|
|
entry := &cacheEntry{
|
|
item: item,
|
|
callback: callback,
|
|
}
|
|
entry.timer = time.AfterFunc(c.timeout, func() {
|
|
c.mu.Lock()
|
|
if entry.deleted {
|
|
c.mu.Unlock()
|
|
// Abort the delete since this has been taken care of in Remove().
|
|
return
|
|
}
|
|
delete(c.cache, key)
|
|
c.mu.Unlock()
|
|
entry.callback()
|
|
})
|
|
c.cache[key] = entry
|
|
return item, true
|
|
}
|
|
|
|
// Remove the item with the key from the cache.
|
|
//
|
|
// If the specified key exists in the cache, it returns (item associated with
|
|
// key, true) and the callback associated with the item is guaranteed to be not
|
|
// called. If the given key is not found in the cache, it returns (nil, false)
|
|
func (c *TimeoutCache) Remove(key interface{}) (item interface{}, ok bool) {
|
|
c.mu.Lock()
|
|
defer c.mu.Unlock()
|
|
entry, ok := c.removeInternal(key)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
return entry.item, true
|
|
}
|
|
|
|
// removeInternal removes and returns the item with key.
|
|
//
|
|
// caller must hold c.mu.
|
|
func (c *TimeoutCache) removeInternal(key interface{}) (*cacheEntry, bool) {
|
|
entry, ok := c.cache[key]
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
delete(c.cache, key)
|
|
if !entry.timer.Stop() {
|
|
// If stop was not successful, the timer has fired (this can only happen
|
|
// in a race). But the deleting function is blocked on c.mu because the
|
|
// mutex was held by the caller of this function.
|
|
//
|
|
// Set deleted to true to abort the deleting function. When the lock is
|
|
// released, the delete function will acquire the lock, check the value
|
|
// of deleted and return.
|
|
entry.deleted = true
|
|
}
|
|
return entry, true
|
|
}
|
|
|
|
// Clear removes all entries, and runs the callbacks if runCallback is true.
|
|
func (c *TimeoutCache) Clear(runCallback bool) {
|
|
var entries []*cacheEntry
|
|
c.mu.Lock()
|
|
for key := range c.cache {
|
|
if e, ok := c.removeInternal(key); ok {
|
|
entries = append(entries, e)
|
|
}
|
|
}
|
|
c.mu.Unlock()
|
|
|
|
if !runCallback {
|
|
return
|
|
}
|
|
|
|
// removeInternal removes entries from cache, and also stops the timer, so
|
|
// the callback is guaranteed to be not called. If runCallback is true,
|
|
// manual execute all callbacks.
|
|
for _, entry := range entries {
|
|
entry.callback()
|
|
}
|
|
}
|