2018-04-04 03:46:07 +00:00
|
|
|
// Package cache provides caching features for data from a Consul server.
|
|
|
|
//
|
|
|
|
// While this is similar in some ways to the "agent/ae" package, a key
|
|
|
|
// difference is that with anti-entropy, the agent is the authoritative
|
|
|
|
// source so it resolves differences the server may have. With caching (this
|
|
|
|
// package), the server is the authoritative source and we do our best to
|
|
|
|
// balance performance and correctness, depending on the type of data being
|
|
|
|
// requested.
|
|
|
|
//
|
2018-04-17 23:42:49 +00:00
|
|
|
// The types of data that can be cached is configurable via the Type interface.
|
|
|
|
// This allows specialized behavior for certain types of data. Each type of
|
|
|
|
// Consul data (CA roots, leaf certs, intentions, KV, catalog, etc.) will
|
|
|
|
// have to be manually implemented. This usually is not much work, see
|
|
|
|
// the "agent/cache-types" package.
|
2018-04-04 03:46:07 +00:00
|
|
|
package cache
|
|
|
|
|
|
|
|
import (
|
2018-04-20 00:31:50 +00:00
|
|
|
"container/heap"
|
2018-04-04 03:46:07 +00:00
|
|
|
"fmt"
|
|
|
|
"sync"
|
2018-10-04 10:27:11 +00:00
|
|
|
"sync/atomic"
|
2018-04-04 03:46:07 +00:00
|
|
|
"time"
|
2018-04-17 23:03:13 +00:00
|
|
|
|
|
|
|
"github.com/armon/go-metrics"
|
2019-01-18 17:44:04 +00:00
|
|
|
"github.com/hashicorp/consul/lib"
|
2018-04-04 03:46:07 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
//go:generate mockery -all -inpkg
|
|
|
|
|
2018-05-09 18:04:52 +00:00
|
|
|
// Constants related to refresh backoff. We probably don't ever need to
|
|
|
|
// make these configurable knobs since they primarily exist to lower load.
|
|
|
|
const (
|
|
|
|
CacheRefreshBackoffMin = 3 // 3 attempts before backing off
|
|
|
|
CacheRefreshMaxWait = 1 * time.Minute // maximum backoff wait time
|
|
|
|
)
|
|
|
|
|
2018-04-17 23:42:49 +00:00
|
|
|
// Cache is a agent-local cache of Consul data. Create a Cache using the
|
|
|
|
// New function. A zero-value Cache is not ready for usage and will result
|
|
|
|
// in a panic.
|
|
|
|
//
|
|
|
|
// The types of data to be cached must be registered via RegisterType. Then,
|
|
|
|
// calls to Get specify the type and a Request implementation. The
|
|
|
|
// implementation of Request is usually done directly on the standard RPC
|
|
|
|
// struct in agent/structs. This API makes cache usage a mostly drop-in
|
|
|
|
// replacement for non-cached RPC calls.
|
|
|
|
//
|
|
|
|
// The cache is partitioned by ACL and datacenter. This allows the cache
|
|
|
|
// to be safe for multi-DC queries and for queries where the data is modified
|
|
|
|
// due to ACLs all without the cache having to have any clever logic, at
|
|
|
|
// the slight expense of a less perfect cache.
|
|
|
|
//
|
|
|
|
// The Cache exposes various metrics via go-metrics. Please view the source
|
|
|
|
// searching for "metrics." to see the various metrics exposed. These can be
|
|
|
|
// used to explore the performance of the cache.
|
2018-04-04 03:46:07 +00:00
|
|
|
type Cache struct {
|
2018-04-10 15:05:34 +00:00
|
|
|
// types stores the list of data types that the cache knows how to service.
|
|
|
|
// These can be dynamically registered with RegisterType.
|
2018-04-04 03:46:07 +00:00
|
|
|
typesLock sync.RWMutex
|
|
|
|
types map[string]typeEntry
|
2018-04-10 15:05:34 +00:00
|
|
|
|
2018-04-20 00:31:50 +00:00
|
|
|
// entries contains the actual cache data. Access to entries and
|
|
|
|
// entriesExpiryHeap must be protected by entriesLock.
|
|
|
|
//
|
|
|
|
// entriesExpiryHeap is a heap of *cacheEntry values ordered by
|
|
|
|
// expiry, with the soonest to expire being first in the list (index 0).
|
2018-04-10 15:05:34 +00:00
|
|
|
//
|
|
|
|
// NOTE(mitchellh): The entry map key is currently a string in the format
|
|
|
|
// of "<DC>/<ACL token>/<Request key>" in order to properly partition
|
|
|
|
// requests to different datacenters and ACL tokens. This format has some
|
|
|
|
// big drawbacks: we can't evict by datacenter, ACL token, etc. For an
|
2018-05-09 19:30:43 +00:00
|
|
|
// initial implementation this works and the tests are agnostic to the
|
2018-04-10 15:05:34 +00:00
|
|
|
// internal storage format so changing this should be possible safely.
|
2018-04-20 00:31:50 +00:00
|
|
|
entriesLock sync.RWMutex
|
|
|
|
entries map[string]cacheEntry
|
|
|
|
entriesExpiryHeap *expiryHeap
|
2018-10-04 10:27:11 +00:00
|
|
|
|
|
|
|
// stopped is used as an atomic flag to signal that the Cache has been
|
|
|
|
// discarded so background fetches and expiry processing should stop.
|
|
|
|
stopped uint32
|
|
|
|
// stopCh is closed when Close is called
|
|
|
|
stopCh chan struct{}
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// typeEntry is a single type that is registered with a Cache.
|
|
|
|
type typeEntry struct {
|
|
|
|
Type Type
|
|
|
|
Opts *RegisterOptions
|
|
|
|
}
|
|
|
|
|
2018-06-15 12:13:54 +00:00
|
|
|
// ResultMeta is returned from Get calls along with the value and can be used
|
|
|
|
// to expose information about the cache status for debugging or testing.
|
|
|
|
type ResultMeta struct {
|
2018-09-06 10:34:28 +00:00
|
|
|
// Hit indicates whether or not the request was a cache hit
|
2018-06-15 12:13:54 +00:00
|
|
|
Hit bool
|
2018-09-06 10:34:28 +00:00
|
|
|
|
|
|
|
// Age identifies how "stale" the result is. It's semantics differ based on
|
|
|
|
// whether or not the cache type performs background refresh or not as defined
|
|
|
|
// in https://www.consul.io/api/index.html#agent-caching.
|
|
|
|
//
|
|
|
|
// For background refresh types, Age is 0 unless the background blocking query
|
|
|
|
// is currently in a failed state and so not keeping up with the server's
|
|
|
|
// values. If it is non-zero it represents the time since the first failure to
|
|
|
|
// connect during background refresh, and is reset after a background request
|
|
|
|
// does manage to reconnect and either return successfully, or block for at
|
|
|
|
// least the yamux keepalive timeout of 30 seconds (which indicates the
|
|
|
|
// connection is OK but blocked as expected).
|
|
|
|
//
|
|
|
|
// For simple cache types, Age is the time since the result being returned was
|
|
|
|
// fetched from the servers.
|
|
|
|
Age time.Duration
|
2018-10-02 10:27:10 +00:00
|
|
|
|
|
|
|
// Index is the internal ModifyIndex for the cache entry. Not all types
|
|
|
|
// support blocking and all that do will likely have this in their result type
|
|
|
|
// already but this allows generic code to reason about whether cache values
|
|
|
|
// have changed.
|
|
|
|
Index uint64
|
2018-06-15 12:13:54 +00:00
|
|
|
}
|
|
|
|
|
2018-04-08 13:30:14 +00:00
|
|
|
// Options are options for the Cache.
|
|
|
|
type Options struct {
|
|
|
|
// Nothing currently, reserved.
|
|
|
|
}
|
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
// New creates a new cache with the given RPC client and reasonable defaults.
|
|
|
|
// Further settings can be tweaked on the returned value.
|
2018-04-08 13:30:14 +00:00
|
|
|
func New(*Options) *Cache {
|
2018-04-20 00:31:50 +00:00
|
|
|
// Initialize the heap. The buffer of 1 is really important because
|
|
|
|
// its possible for the expiry loop to trigger the heap to update
|
|
|
|
// itself and it'd block forever otherwise.
|
|
|
|
h := &expiryHeap{NotifyCh: make(chan struct{}, 1)}
|
|
|
|
heap.Init(h)
|
|
|
|
|
|
|
|
c := &Cache{
|
|
|
|
types: make(map[string]typeEntry),
|
|
|
|
entries: make(map[string]cacheEntry),
|
|
|
|
entriesExpiryHeap: h,
|
2018-10-04 10:27:11 +00:00
|
|
|
stopCh: make(chan struct{}),
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
2018-04-20 00:31:50 +00:00
|
|
|
|
|
|
|
// Start the expiry watcher
|
|
|
|
go c.runExpiryLoop()
|
|
|
|
|
|
|
|
return c
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterOptions are options that can be associated with a type being
|
|
|
|
// registered for the cache. This changes the behavior of the cache for
|
|
|
|
// this type.
|
|
|
|
type RegisterOptions struct {
|
2018-04-20 00:31:50 +00:00
|
|
|
// LastGetTTL is the time that the values returned by this type remain
|
|
|
|
// in the cache after the last get operation. If a value isn't accessed
|
|
|
|
// within this duration, the value is purged from the cache and
|
|
|
|
// background refreshing will cease.
|
|
|
|
LastGetTTL time.Duration
|
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
// Refresh configures whether the data is actively refreshed or if
|
|
|
|
// the data is only refreshed on an explicit Get. The default (false)
|
|
|
|
// is to only request data on explicit Get.
|
|
|
|
Refresh bool
|
|
|
|
|
|
|
|
// RefreshTimer is the time between attempting to refresh data.
|
|
|
|
// If this is zero, then data is refreshed immediately when a fetch
|
|
|
|
// is returned.
|
|
|
|
//
|
|
|
|
// RefreshTimeout determines the maximum query time for a refresh
|
|
|
|
// operation. This is specified as part of the query options and is
|
|
|
|
// expected to be implemented by the Type itself.
|
|
|
|
//
|
|
|
|
// Using these values, various "refresh" mechanisms can be implemented:
|
|
|
|
//
|
|
|
|
// * With a high timer duration and a low timeout, a timer-based
|
|
|
|
// refresh can be set that minimizes load on the Consul servers.
|
|
|
|
//
|
|
|
|
// * With a low timer and high timeout duration, a blocking-query-based
|
|
|
|
// refresh can be set so that changes in server data are recognized
|
|
|
|
// within the cache very quickly.
|
|
|
|
//
|
|
|
|
RefreshTimer time.Duration
|
|
|
|
RefreshTimeout time.Duration
|
|
|
|
}
|
|
|
|
|
|
|
|
// RegisterType registers a cacheable type.
|
2018-04-17 23:03:13 +00:00
|
|
|
//
|
|
|
|
// This makes the type available for Get but does not automatically perform
|
|
|
|
// any prefetching. In order to populate the cache, Get must be called.
|
2018-04-04 03:46:07 +00:00
|
|
|
func (c *Cache) RegisterType(n string, typ Type, opts *RegisterOptions) {
|
2018-04-19 18:36:14 +00:00
|
|
|
if opts == nil {
|
|
|
|
opts = &RegisterOptions{}
|
|
|
|
}
|
2018-04-20 00:31:50 +00:00
|
|
|
if opts.LastGetTTL == 0 {
|
|
|
|
opts.LastGetTTL = 72 * time.Hour // reasonable default is days
|
|
|
|
}
|
2018-04-19 18:36:14 +00:00
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
c.typesLock.Lock()
|
|
|
|
defer c.typesLock.Unlock()
|
|
|
|
c.types[n] = typeEntry{Type: typ, Opts: opts}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get loads the data for the given type and request. If data satisfying the
|
|
|
|
// minimum index is present in the cache, it is returned immediately. Otherwise,
|
|
|
|
// this will block until the data is available or the request timeout is
|
|
|
|
// reached.
|
|
|
|
//
|
|
|
|
// Multiple Get calls for the same Request (matching CacheKey value) will
|
|
|
|
// block on a single network request.
|
2018-04-17 23:03:13 +00:00
|
|
|
//
|
|
|
|
// The timeout specified by the Request will be the timeout on the cache
|
|
|
|
// Get, and does not correspond to the timeout of any background data
|
|
|
|
// fetching. If the timeout is reached before data satisfying the minimum
|
|
|
|
// index is retrieved, the last known value (maybe nil) is returned. No
|
|
|
|
// error is returned on timeout. This matches the behavior of Consul blocking
|
|
|
|
// queries.
|
2018-06-15 12:13:54 +00:00
|
|
|
func (c *Cache) Get(t string, r Request) (interface{}, ResultMeta, error) {
|
2018-10-02 10:27:10 +00:00
|
|
|
return c.getWithIndex(t, r, r.CacheInfo().MinIndex)
|
|
|
|
}
|
|
|
|
|
2019-05-07 10:15:49 +00:00
|
|
|
// getEntryLocked retrieves a cache entry and checks if it is ready to be
|
|
|
|
// returned given the other parameters. It reads from entries and the caller
|
|
|
|
// has to issue a read lock if necessary.
|
|
|
|
func (c *Cache) getEntryLocked(tEntry typeEntry, key string, maxAge time.Duration, revalidate bool, minIndex uint64) (bool, bool, cacheEntry) {
|
|
|
|
entry, ok := c.entries[key]
|
|
|
|
cacheHit := false
|
|
|
|
|
|
|
|
if !ok {
|
|
|
|
return ok, cacheHit, entry
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if we have a hit
|
|
|
|
cacheHit = ok && entry.Valid
|
|
|
|
|
|
|
|
supportsBlocking := tEntry.Type.SupportsBlocking()
|
|
|
|
|
|
|
|
// Check index is not specified or lower than value, or the type doesn't
|
|
|
|
// support blocking.
|
|
|
|
if cacheHit && supportsBlocking &&
|
|
|
|
minIndex > 0 && minIndex >= entry.Index {
|
|
|
|
// MinIndex was given and matches or is higher than current value so we
|
|
|
|
// ignore the cache and fallthrough to blocking on a new value below.
|
|
|
|
cacheHit = false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check MaxAge is not exceeded if this is not a background refreshing type
|
|
|
|
// and MaxAge was specified.
|
|
|
|
if cacheHit && !tEntry.Opts.Refresh && maxAge > 0 &&
|
|
|
|
!entry.FetchedAt.IsZero() && maxAge < time.Since(entry.FetchedAt) {
|
|
|
|
cacheHit = false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if we are requested to revalidate. If so the first time round the
|
|
|
|
// loop is not a hit but subsequent ones should be treated normally.
|
|
|
|
if cacheHit && !tEntry.Opts.Refresh && revalidate {
|
|
|
|
cacheHit = false
|
|
|
|
}
|
|
|
|
|
|
|
|
return ok, cacheHit, entry
|
|
|
|
}
|
|
|
|
|
2018-10-02 10:27:10 +00:00
|
|
|
// getWithIndex implements the main Get functionality but allows internal
|
|
|
|
// callers (Watch) to manipulate the blocking index separately from the actual
|
|
|
|
// request object.
|
|
|
|
func (c *Cache) getWithIndex(t string, r Request, minIndex uint64) (interface{}, ResultMeta, error) {
|
2018-04-08 14:08:34 +00:00
|
|
|
info := r.CacheInfo()
|
|
|
|
if info.Key == "" {
|
2018-04-17 23:03:13 +00:00
|
|
|
metrics.IncrCounter([]string{"consul", "cache", "bypass"}, 1)
|
|
|
|
|
2018-04-08 13:30:14 +00:00
|
|
|
// If no key is specified, then we do not cache this request.
|
|
|
|
// Pass directly through to the backend.
|
2018-10-02 10:27:10 +00:00
|
|
|
return c.fetchDirect(t, r, minIndex)
|
2018-04-08 13:30:14 +00:00
|
|
|
}
|
2018-04-04 03:46:07 +00:00
|
|
|
|
2018-04-10 15:05:34 +00:00
|
|
|
// Get the actual key for our entry
|
2018-07-25 19:26:27 +00:00
|
|
|
key := c.entryKey(t, &info)
|
2018-04-10 15:05:34 +00:00
|
|
|
|
2018-04-11 09:18:24 +00:00
|
|
|
// First time through
|
2018-06-03 20:15:09 +00:00
|
|
|
first := true
|
2018-04-11 09:18:24 +00:00
|
|
|
|
2018-04-22 20:52:48 +00:00
|
|
|
// timeoutCh for watching our timeout
|
2018-04-16 10:06:08 +00:00
|
|
|
var timeoutCh <-chan time.Time
|
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
RETRY_GET:
|
2018-09-06 10:34:28 +00:00
|
|
|
// Get the type that we're fetching
|
|
|
|
c.typesLock.RLock()
|
|
|
|
tEntry, ok := c.types[t]
|
|
|
|
c.typesLock.RUnlock()
|
|
|
|
if !ok {
|
|
|
|
// Shouldn't happen given that we successfully fetched this at least
|
|
|
|
// once. But be robust against panics.
|
|
|
|
return nil, ResultMeta{}, fmt.Errorf("unknown type in cache: %s", t)
|
|
|
|
}
|
|
|
|
|
2018-10-02 10:27:10 +00:00
|
|
|
// Get the current value
|
|
|
|
c.entriesLock.RLock()
|
2019-05-07 10:15:49 +00:00
|
|
|
_, cacheHit, entry := c.getEntryLocked(tEntry, key, info.MaxAge, info.MustRevalidate && first, minIndex)
|
2018-10-02 10:27:10 +00:00
|
|
|
c.entriesLock.RUnlock()
|
|
|
|
|
2018-09-06 10:34:28 +00:00
|
|
|
if cacheHit {
|
2018-10-02 10:27:10 +00:00
|
|
|
meta := ResultMeta{Index: entry.Index}
|
2018-09-06 10:34:28 +00:00
|
|
|
if first {
|
|
|
|
metrics.IncrCounter([]string{"consul", "cache", t, "hit"}, 1)
|
|
|
|
meta.Hit = true
|
2018-04-08 13:30:14 +00:00
|
|
|
}
|
2018-09-06 10:34:28 +00:00
|
|
|
|
|
|
|
// If refresh is enabled, calculate age based on whether the background
|
|
|
|
// routine is still connected.
|
|
|
|
if tEntry.Opts.Refresh {
|
|
|
|
meta.Age = time.Duration(0)
|
|
|
|
if !entry.RefreshLostContact.IsZero() {
|
|
|
|
meta.Age = time.Since(entry.RefreshLostContact)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// For non-background refresh types, the age is just how long since we
|
|
|
|
// fetched it last.
|
|
|
|
if !entry.FetchedAt.IsZero() {
|
|
|
|
meta.Age = time.Since(entry.FetchedAt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Touch the expiration and fix the heap.
|
|
|
|
c.entriesLock.Lock()
|
|
|
|
entry.Expiry.Reset()
|
|
|
|
c.entriesExpiryHeap.Fix(entry.Expiry)
|
|
|
|
c.entriesLock.Unlock()
|
|
|
|
|
2019-01-08 10:06:38 +00:00
|
|
|
// We purposely do not return an error here since the cache only works with
|
|
|
|
// fetching values that either have a value or have an error, but not both.
|
|
|
|
// The Error may be non-nil in the entry in the case that an error has
|
|
|
|
// occurred _since_ the last good value, but we still want to return the
|
|
|
|
// good value to clients that are not requesting a specific version. The
|
|
|
|
// effect of this is that blocking clients will all see an error immediately
|
|
|
|
// without waiting a whole timeout to see it, but clients that just look up
|
|
|
|
// cache with an older index than the last valid result will still see the
|
|
|
|
// result and not the error here. I.e. the error is not "cached" without a
|
2019-03-06 17:13:28 +00:00
|
|
|
// new fetch attempt occurring, but the last good value can still be fetched
|
2019-01-08 10:06:38 +00:00
|
|
|
// from cache.
|
2018-09-06 10:34:28 +00:00
|
|
|
return entry.Value, meta, nil
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
|
|
|
|
2019-01-08 10:06:38 +00:00
|
|
|
// If this isn't our first time through and our last value has an error, then
|
|
|
|
// we return the error. This has the behavior that we don't sit in a retry
|
|
|
|
// loop getting the same error for the entire duration of the timeout.
|
|
|
|
// Instead, we make one effort to fetch a new value, and if there was an
|
|
|
|
// error, we return. Note that the invariant is that if both entry.Value AND
|
|
|
|
// entry.Error are non-nil, the error _must_ be more recent than the Value. In
|
|
|
|
// other words valid fetches should reset the error. See
|
|
|
|
// https://github.com/hashicorp/consul/issues/4480.
|
2018-06-03 20:15:09 +00:00
|
|
|
if !first && entry.Error != nil {
|
2018-10-02 10:27:10 +00:00
|
|
|
return entry.Value, ResultMeta{Index: entry.Index}, entry.Error
|
2018-04-19 16:19:55 +00:00
|
|
|
}
|
|
|
|
|
2018-06-03 20:15:09 +00:00
|
|
|
if first {
|
2018-04-17 23:03:13 +00:00
|
|
|
// We increment two different counters for cache misses depending on
|
|
|
|
// whether we're missing because we didn't have the data at all,
|
|
|
|
// or if we're missing because we're blocking on a set index.
|
2018-10-02 10:27:10 +00:00
|
|
|
if minIndex == 0 {
|
2018-04-17 23:03:13 +00:00
|
|
|
metrics.IncrCounter([]string{"consul", "cache", t, "miss_new"}, 1)
|
|
|
|
} else {
|
|
|
|
metrics.IncrCounter([]string{"consul", "cache", t, "miss_block"}, 1)
|
|
|
|
}
|
2018-04-11 09:18:24 +00:00
|
|
|
}
|
|
|
|
|
2018-04-16 10:06:08 +00:00
|
|
|
// Set our timeout channel if we must
|
|
|
|
if info.Timeout > 0 && timeoutCh == nil {
|
|
|
|
timeoutCh = time.After(info.Timeout)
|
|
|
|
}
|
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
// At this point, we know we either don't have a value at all or the
|
|
|
|
// value we have is too old. We need to wait for new data.
|
2019-05-07 10:15:49 +00:00
|
|
|
waiterCh, err := c.fetch(t, key, r, true, 0, minIndex, false, !first)
|
2018-04-04 03:46:07 +00:00
|
|
|
if err != nil {
|
2018-10-02 10:27:10 +00:00
|
|
|
return nil, ResultMeta{Index: entry.Index}, err
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
|
|
|
|
2019-05-07 10:15:49 +00:00
|
|
|
// No longer our first time through
|
|
|
|
first = false
|
|
|
|
|
2018-04-16 10:06:08 +00:00
|
|
|
select {
|
|
|
|
case <-waiterCh:
|
2018-10-02 10:27:10 +00:00
|
|
|
// Our fetch returned, retry the get from the cache.
|
2018-04-16 10:06:08 +00:00
|
|
|
goto RETRY_GET
|
|
|
|
|
|
|
|
case <-timeoutCh:
|
|
|
|
// Timeout on the cache read, just return whatever we have.
|
2018-10-02 10:27:10 +00:00
|
|
|
return entry.Value, ResultMeta{Index: entry.Index}, nil
|
2018-04-16 10:06:08 +00:00
|
|
|
}
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
|
|
|
|
2018-04-10 15:05:34 +00:00
|
|
|
// entryKey returns the key for the entry in the cache. See the note
|
|
|
|
// about the entry key format in the structure docs for Cache.
|
2018-07-25 19:26:27 +00:00
|
|
|
func (c *Cache) entryKey(t string, r *RequestInfo) string {
|
2019-06-27 20:22:07 +00:00
|
|
|
return makeEntryKey(t, r.Datacenter, r.Token, r.Key)
|
|
|
|
}
|
|
|
|
|
|
|
|
func makeEntryKey(t, dc, token, key string) string {
|
|
|
|
return fmt.Sprintf("%s/%s/%s/%s", t, dc, token, key)
|
2018-04-10 15:05:34 +00:00
|
|
|
}
|
|
|
|
|
2018-04-17 23:03:13 +00:00
|
|
|
// fetch triggers a new background fetch for the given Request. If a
|
|
|
|
// background fetch is already running for a matching Request, the waiter
|
|
|
|
// channel for that request is returned. The effect of this is that there
|
|
|
|
// is only ever one blocking query for any matching requests.
|
2018-04-20 00:31:50 +00:00
|
|
|
//
|
|
|
|
// If allowNew is true then the fetch should create the cache entry
|
|
|
|
// if it doesn't exist. If this is false, then fetch will do nothing
|
|
|
|
// if the entry doesn't exist. This latter case is to support refreshing.
|
2019-05-07 10:15:49 +00:00
|
|
|
func (c *Cache) fetch(t, key string, r Request, allowNew bool, attempt uint, minIndex uint64, ignoreExisting bool, ignoreRevalidation bool) (<-chan struct{}, error) {
|
2018-04-04 03:46:07 +00:00
|
|
|
// Get the type that we're fetching
|
|
|
|
c.typesLock.RLock()
|
|
|
|
tEntry, ok := c.types[t]
|
|
|
|
c.typesLock.RUnlock()
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("unknown type in cache: %s", t)
|
|
|
|
}
|
|
|
|
|
2019-05-07 10:15:49 +00:00
|
|
|
info := r.CacheInfo()
|
|
|
|
|
2018-04-17 23:03:13 +00:00
|
|
|
// We acquire a write lock because we may have to set Fetching to true.
|
2018-04-04 03:46:07 +00:00
|
|
|
c.entriesLock.Lock()
|
|
|
|
defer c.entriesLock.Unlock()
|
2019-05-07 10:15:49 +00:00
|
|
|
ok, cacheHit, entry := c.getEntryLocked(tEntry, key, info.MaxAge, info.MustRevalidate && !ignoreRevalidation, minIndex)
|
|
|
|
|
|
|
|
// This handles the case where a fetch succeeded after checking for its existence in
|
|
|
|
// getWithIndex. This ensures that we don't miss updates.
|
|
|
|
if ok && cacheHit && !ignoreExisting {
|
|
|
|
ch := make(chan struct{})
|
|
|
|
close(ch)
|
|
|
|
return ch, nil
|
|
|
|
}
|
2018-04-04 03:46:07 +00:00
|
|
|
|
2018-04-20 00:31:50 +00:00
|
|
|
// If we aren't allowing new values and we don't have an existing value,
|
|
|
|
// return immediately. We return an immediately-closed channel so nothing
|
|
|
|
// blocks.
|
|
|
|
if !ok && !allowNew {
|
|
|
|
ch := make(chan struct{})
|
|
|
|
close(ch)
|
|
|
|
return ch, nil
|
|
|
|
}
|
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
// If we already have an entry and it is actively fetching, then return
|
|
|
|
// the currently active waiter.
|
|
|
|
if ok && entry.Fetching {
|
|
|
|
return entry.Waiter, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we don't have an entry, then create it. The entry must be marked
|
|
|
|
// as invalid so that it isn't returned as a valid value for a zero index.
|
|
|
|
if !ok {
|
|
|
|
entry = cacheEntry{Valid: false, Waiter: make(chan struct{})}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set that we're fetching to true, which makes it so that future
|
|
|
|
// identical calls to fetch will return the same waiter rather than
|
|
|
|
// perform multiple fetches.
|
|
|
|
entry.Fetching = true
|
2018-04-10 15:05:34 +00:00
|
|
|
c.entries[key] = entry
|
2018-04-17 23:03:13 +00:00
|
|
|
metrics.SetGauge([]string{"consul", "cache", "entries_count"}, float32(len(c.entries)))
|
2018-04-04 03:46:07 +00:00
|
|
|
|
|
|
|
// The actual Fetch must be performed in a goroutine.
|
|
|
|
go func() {
|
2018-09-06 10:34:28 +00:00
|
|
|
// If we have background refresh and currently are in "disconnected" state,
|
|
|
|
// waiting for a response might mean we mark our results as stale for up to
|
|
|
|
// 10 minutes (max blocking timeout) after connection is restored. To reduce
|
|
|
|
// that window, we assume that if the fetch takes more than 31 seconds then
|
|
|
|
// they are correctly blocking. We choose 31 seconds because yamux
|
|
|
|
// keepalives are every 30 seconds so the RPC should fail if the packets are
|
|
|
|
// being blackholed for more than 30 seconds.
|
|
|
|
var connectedTimer *time.Timer
|
|
|
|
if tEntry.Opts.Refresh && entry.Index > 0 &&
|
|
|
|
tEntry.Opts.RefreshTimeout > (31*time.Second) {
|
|
|
|
connectedTimer = time.AfterFunc(31*time.Second, func() {
|
|
|
|
c.entriesLock.Lock()
|
|
|
|
defer c.entriesLock.Unlock()
|
|
|
|
entry, ok := c.entries[key]
|
|
|
|
if !ok || entry.RefreshLostContact.IsZero() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
entry.RefreshLostContact = time.Time{}
|
|
|
|
c.entries[key] = entry
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
fOpts := FetchOptions{}
|
|
|
|
if tEntry.Type.SupportsBlocking() {
|
|
|
|
fOpts.MinIndex = entry.Index
|
|
|
|
fOpts.Timeout = tEntry.Opts.RefreshTimeout
|
|
|
|
}
|
2019-01-10 12:46:11 +00:00
|
|
|
if entry.Valid {
|
|
|
|
fOpts.LastResult = &FetchResult{
|
|
|
|
Value: entry.Value,
|
|
|
|
State: entry.State,
|
|
|
|
Index: entry.Index,
|
|
|
|
}
|
|
|
|
}
|
2018-09-06 10:34:28 +00:00
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
// Start building the new entry by blocking on the fetch.
|
2018-09-06 10:34:28 +00:00
|
|
|
result, err := tEntry.Type.Fetch(fOpts, r)
|
|
|
|
if connectedTimer != nil {
|
|
|
|
connectedTimer.Stop()
|
|
|
|
}
|
2018-04-04 03:46:07 +00:00
|
|
|
|
2018-04-20 00:31:50 +00:00
|
|
|
// Copy the existing entry to start.
|
|
|
|
newEntry := entry
|
|
|
|
newEntry.Fetching = false
|
2019-01-08 10:06:38 +00:00
|
|
|
|
|
|
|
// Importantly, always reset the Error. Having both Error and a Value that
|
|
|
|
// are non-nil is allowed in the cache entry but it indicates that the Error
|
|
|
|
// is _newer_ than the last good value. So if the err is nil then we need to
|
|
|
|
// reset to replace any _older_ errors and avoid them bubbling up. If the
|
|
|
|
// error is non-nil then we need to set it anyway and used to do it in the
|
|
|
|
// code below. See https://github.com/hashicorp/consul/issues/4480.
|
|
|
|
newEntry.Error = err
|
|
|
|
|
2018-04-20 00:31:50 +00:00
|
|
|
if result.Value != nil {
|
2018-04-16 10:06:08 +00:00
|
|
|
// A new value was given, so we create a brand new entry.
|
|
|
|
newEntry.Value = result.Value
|
2019-01-10 12:46:11 +00:00
|
|
|
newEntry.State = result.State
|
2018-04-16 10:06:08 +00:00
|
|
|
newEntry.Index = result.Index
|
2018-09-06 10:34:28 +00:00
|
|
|
newEntry.FetchedAt = time.Now()
|
2018-06-15 20:03:19 +00:00
|
|
|
if newEntry.Index < 1 {
|
|
|
|
// Less than one is invalid unless there was an error and in this case
|
|
|
|
// there wasn't since a value was returned. If a badly behaved RPC
|
|
|
|
// returns 0 when it has no data, we might get into a busy loop here. We
|
|
|
|
// set this to minimum of 1 which is safe because no valid user data can
|
|
|
|
// ever be written at raft index 1 due to the bootstrap process for
|
|
|
|
// raft. This insure that any subsequent background refresh request will
|
|
|
|
// always block, but allows the initial request to return immediately
|
|
|
|
// even if there is no data.
|
|
|
|
newEntry.Index = 1
|
|
|
|
}
|
2018-04-16 10:06:08 +00:00
|
|
|
|
|
|
|
// This is a valid entry with a result
|
|
|
|
newEntry.Valid = true
|
2019-01-22 17:19:36 +00:00
|
|
|
} else if result.State != nil && err == nil {
|
|
|
|
// Also set state if it's non-nil but Value is nil. This is important in the
|
|
|
|
// case we are returning nil due to a timeout or a transient error like rate
|
|
|
|
// limiting that we want to mask from the user - there is no result yet but
|
|
|
|
// we want to manage retrying internally before we return an error to user.
|
|
|
|
// The retrying state is in State so we need to still update that in the
|
|
|
|
// entry even if we don't have an actual result yet (e.g. hit a rate limit
|
|
|
|
// on first request for a leaf certificate).
|
|
|
|
newEntry.State = result.State
|
2018-04-16 10:06:08 +00:00
|
|
|
}
|
2018-04-04 03:46:07 +00:00
|
|
|
|
2018-05-09 18:04:52 +00:00
|
|
|
// Error handling
|
|
|
|
if err == nil {
|
|
|
|
metrics.IncrCounter([]string{"consul", "cache", "fetch_success"}, 1)
|
|
|
|
metrics.IncrCounter([]string{"consul", "cache", t, "fetch_success"}, 1)
|
|
|
|
|
2018-06-15 20:03:19 +00:00
|
|
|
if result.Index > 0 {
|
|
|
|
// Reset the attempts counter so we don't have any backoff
|
|
|
|
attempt = 0
|
|
|
|
} else {
|
|
|
|
// Result having a zero index is an implicit error case. There was no
|
|
|
|
// actual error but it implies the RPC found in index (nothing written
|
|
|
|
// yet for that type) but didn't take care to return safe "1" index. We
|
|
|
|
// don't want to actually treat it like an error by setting
|
|
|
|
// newEntry.Error to something non-nil, but we should guard against 100%
|
|
|
|
// CPU burn hot loops caused by that case which will never block but
|
|
|
|
// also won't backoff either. So we treat it as a failed attempt so that
|
|
|
|
// at least the failure backoff will save our CPU while still
|
|
|
|
// periodically refreshing so normal service can resume when the servers
|
|
|
|
// actually have something to return from the RPC. If we get in this
|
|
|
|
// state it can be considered a bug in the RPC implementation (to ever
|
|
|
|
// return a zero index) however since it can happen this is a safety net
|
|
|
|
// for the future.
|
|
|
|
attempt++
|
|
|
|
}
|
2018-09-06 10:34:28 +00:00
|
|
|
|
|
|
|
// If we have refresh active, this successful response means cache is now
|
|
|
|
// "connected" and should not be stale. Reset the lost contact timer.
|
|
|
|
if tEntry.Opts.Refresh {
|
|
|
|
newEntry.RefreshLostContact = time.Time{}
|
|
|
|
}
|
2018-05-09 18:04:52 +00:00
|
|
|
} else {
|
|
|
|
metrics.IncrCounter([]string{"consul", "cache", "fetch_error"}, 1)
|
|
|
|
metrics.IncrCounter([]string{"consul", "cache", t, "fetch_error"}, 1)
|
|
|
|
|
2018-05-09 18:10:17 +00:00
|
|
|
// Increment attempt counter
|
|
|
|
attempt++
|
2018-05-09 18:04:52 +00:00
|
|
|
|
2018-09-06 10:34:28 +00:00
|
|
|
// If we are refreshing and just failed, updated the lost contact time as
|
|
|
|
// our cache will be stale until we get successfully reconnected. We only
|
|
|
|
// set this on the first failure (if it's zero) so we can track how long
|
|
|
|
// it's been since we had a valid connection/up-to-date view of the state.
|
|
|
|
if tEntry.Opts.Refresh && newEntry.RefreshLostContact.IsZero() {
|
|
|
|
newEntry.RefreshLostContact = time.Now()
|
|
|
|
}
|
2018-04-19 16:19:55 +00:00
|
|
|
}
|
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
// Create a new waiter that will be used for the next fetch.
|
|
|
|
newEntry.Waiter = make(chan struct{})
|
|
|
|
|
2018-04-20 00:31:50 +00:00
|
|
|
// Set our entry
|
2018-04-04 03:46:07 +00:00
|
|
|
c.entriesLock.Lock()
|
2018-04-20 01:28:01 +00:00
|
|
|
|
|
|
|
// If this is a new entry (not in the heap yet), then setup the
|
|
|
|
// initial expiry information and insert. If we're already in
|
|
|
|
// the heap we do nothing since we're reusing the same entry.
|
|
|
|
if newEntry.Expiry == nil || newEntry.Expiry.HeapIndex == -1 {
|
|
|
|
newEntry.Expiry = &cacheEntryExpiry{
|
|
|
|
Key: key,
|
|
|
|
TTL: tEntry.Opts.LastGetTTL,
|
|
|
|
}
|
|
|
|
newEntry.Expiry.Reset()
|
|
|
|
heap.Push(c.entriesExpiryHeap, newEntry.Expiry)
|
2018-04-20 00:31:50 +00:00
|
|
|
}
|
2018-04-20 01:28:01 +00:00
|
|
|
|
2018-04-10 15:05:34 +00:00
|
|
|
c.entries[key] = newEntry
|
2018-04-04 03:46:07 +00:00
|
|
|
c.entriesLock.Unlock()
|
|
|
|
|
2018-04-20 00:31:50 +00:00
|
|
|
// Trigger the old waiter
|
2018-04-04 03:46:07 +00:00
|
|
|
close(entry.Waiter)
|
|
|
|
|
|
|
|
// If refresh is enabled, run the refresh in due time. The refresh
|
|
|
|
// below might block, but saves us from spawning another goroutine.
|
2018-06-14 05:26:01 +00:00
|
|
|
if tEntry.Opts.Refresh {
|
2018-05-09 18:10:17 +00:00
|
|
|
c.refresh(tEntry.Opts, attempt, t, key, r)
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return entry.Waiter, nil
|
|
|
|
}
|
|
|
|
|
2018-04-17 23:03:13 +00:00
|
|
|
// fetchDirect fetches the given request with no caching. Because this
|
|
|
|
// bypasses the caching entirely, multiple matching requests will result
|
|
|
|
// in multiple actual RPC calls (unlike fetch).
|
2018-10-02 10:27:10 +00:00
|
|
|
func (c *Cache) fetchDirect(t string, r Request, minIndex uint64) (interface{}, ResultMeta, error) {
|
2018-04-08 13:30:14 +00:00
|
|
|
// Get the type that we're fetching
|
|
|
|
c.typesLock.RLock()
|
|
|
|
tEntry, ok := c.types[t]
|
|
|
|
c.typesLock.RUnlock()
|
|
|
|
if !ok {
|
2018-06-15 12:13:54 +00:00
|
|
|
return nil, ResultMeta{}, fmt.Errorf("unknown type in cache: %s", t)
|
2018-04-08 13:30:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Fetch it with the min index specified directly by the request.
|
|
|
|
result, err := tEntry.Type.Fetch(FetchOptions{
|
2018-10-02 10:27:10 +00:00
|
|
|
MinIndex: minIndex,
|
2018-04-08 13:30:14 +00:00
|
|
|
}, r)
|
|
|
|
if err != nil {
|
2018-06-15 12:13:54 +00:00
|
|
|
return nil, ResultMeta{}, err
|
2018-04-08 13:30:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Return the result and ignore the rest
|
2018-06-15 12:13:54 +00:00
|
|
|
return result.Value, ResultMeta{}, nil
|
2018-04-08 13:30:14 +00:00
|
|
|
}
|
|
|
|
|
2018-10-02 10:27:10 +00:00
|
|
|
func backOffWait(failures uint) time.Duration {
|
|
|
|
if failures > CacheRefreshBackoffMin {
|
|
|
|
shift := failures - CacheRefreshBackoffMin
|
|
|
|
waitTime := CacheRefreshMaxWait
|
|
|
|
if shift < 31 {
|
|
|
|
waitTime = (1 << shift) * time.Second
|
|
|
|
}
|
|
|
|
if waitTime > CacheRefreshMaxWait {
|
|
|
|
waitTime = CacheRefreshMaxWait
|
|
|
|
}
|
2019-01-18 17:44:04 +00:00
|
|
|
return waitTime + lib.RandomStagger(waitTime)
|
2018-10-02 10:27:10 +00:00
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2018-04-17 23:03:13 +00:00
|
|
|
// refresh triggers a fetch for a specific Request according to the
|
|
|
|
// registration options.
|
2018-05-09 18:54:15 +00:00
|
|
|
func (c *Cache) refresh(opts *RegisterOptions, attempt uint, t string, key string, r Request) {
|
2018-04-04 03:46:07 +00:00
|
|
|
// Sanity-check, we should not schedule anything that has refresh disabled
|
|
|
|
if !opts.Refresh {
|
|
|
|
return
|
|
|
|
}
|
2018-10-04 10:27:11 +00:00
|
|
|
// Check if cache was stopped
|
|
|
|
if atomic.LoadUint32(&c.stopped) == 1 {
|
|
|
|
return
|
|
|
|
}
|
2018-04-04 03:46:07 +00:00
|
|
|
|
2018-06-03 20:15:09 +00:00
|
|
|
// If we're over the attempt minimum, start an exponential backoff.
|
2018-10-02 10:27:10 +00:00
|
|
|
if wait := backOffWait(attempt); wait > 0 {
|
|
|
|
time.Sleep(wait)
|
2018-06-03 20:15:09 +00:00
|
|
|
}
|
|
|
|
|
2018-04-04 03:46:07 +00:00
|
|
|
// If we have a timer, wait for it
|
|
|
|
if opts.RefreshTimer > 0 {
|
|
|
|
time.Sleep(opts.RefreshTimer)
|
|
|
|
}
|
|
|
|
|
2018-04-20 00:31:50 +00:00
|
|
|
// Trigger. The "allowNew" field is false because in the time we were
|
|
|
|
// waiting to refresh we may have expired and got evicted. If that
|
|
|
|
// happened, we don't want to create a new entry.
|
2019-05-07 10:15:49 +00:00
|
|
|
c.fetch(t, key, r, false, attempt, 0, true, true)
|
2018-04-20 00:31:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// runExpiryLoop is a blocking function that watches the expiration
|
|
|
|
// heap and invalidates entries that have expired.
|
|
|
|
func (c *Cache) runExpiryLoop() {
|
|
|
|
var expiryTimer *time.Timer
|
|
|
|
for {
|
|
|
|
// If we have a previous timer, stop it.
|
|
|
|
if expiryTimer != nil {
|
|
|
|
expiryTimer.Stop()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the entry expiring soonest
|
2018-04-20 01:28:01 +00:00
|
|
|
var entry *cacheEntryExpiry
|
2018-04-20 00:31:50 +00:00
|
|
|
var expiryCh <-chan time.Time
|
|
|
|
c.entriesLock.RLock()
|
|
|
|
if len(c.entriesExpiryHeap.Entries) > 0 {
|
|
|
|
entry = c.entriesExpiryHeap.Entries[0]
|
2018-04-20 01:28:01 +00:00
|
|
|
expiryTimer = time.NewTimer(entry.Expires.Sub(time.Now()))
|
2018-04-20 00:31:50 +00:00
|
|
|
expiryCh = expiryTimer.C
|
|
|
|
}
|
|
|
|
c.entriesLock.RUnlock()
|
|
|
|
|
|
|
|
select {
|
2018-10-04 10:27:11 +00:00
|
|
|
case <-c.stopCh:
|
|
|
|
return
|
2018-04-20 00:31:50 +00:00
|
|
|
case <-c.entriesExpiryHeap.NotifyCh:
|
|
|
|
// Entries changed, so the heap may have changed. Restart loop.
|
|
|
|
|
|
|
|
case <-expiryCh:
|
|
|
|
c.entriesLock.Lock()
|
2018-04-20 01:28:01 +00:00
|
|
|
|
|
|
|
// Entry expired! Remove it.
|
2018-04-20 00:31:50 +00:00
|
|
|
delete(c.entries, entry.Key)
|
2018-04-20 01:28:01 +00:00
|
|
|
heap.Remove(c.entriesExpiryHeap, entry.HeapIndex)
|
|
|
|
|
|
|
|
// This is subtle but important: if we race and simultaneously
|
|
|
|
// evict and fetch a new value, then we set this to -1 to
|
|
|
|
// have it treated as a new value so that the TTL is extended.
|
|
|
|
entry.HeapIndex = -1
|
|
|
|
|
2018-04-20 01:40:12 +00:00
|
|
|
// Set some metrics
|
2018-04-20 00:31:50 +00:00
|
|
|
metrics.IncrCounter([]string{"consul", "cache", "evict_expired"}, 1)
|
2018-04-20 01:40:12 +00:00
|
|
|
metrics.SetGauge([]string{"consul", "cache", "entries_count"}, float32(len(c.entries)))
|
|
|
|
|
|
|
|
c.entriesLock.Unlock()
|
2018-04-20 00:31:50 +00:00
|
|
|
}
|
|
|
|
}
|
2018-04-04 03:46:07 +00:00
|
|
|
}
|
2018-10-04 10:27:11 +00:00
|
|
|
|
|
|
|
// Close stops any background work and frees all resources for the cache.
|
|
|
|
// Current Fetch requests are allowed to continue to completion and callers may
|
|
|
|
// still access the current cache values so coordination isn't needed with
|
|
|
|
// callers, however no background activity will continue. It's intended to close
|
|
|
|
// the cache at agent shutdown so no further requests should be made, however
|
|
|
|
// concurrent or in-flight ones won't break.
|
|
|
|
func (c *Cache) Close() error {
|
|
|
|
wasStopped := atomic.SwapUint32(&c.stopped, 1)
|
|
|
|
if wasStopped == 0 {
|
|
|
|
// First time only, close stop chan
|
|
|
|
close(c.stopCh)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2019-06-27 20:22:07 +00:00
|
|
|
|
|
|
|
// Prepopulate puts something in the cache manually. This is useful when the
|
|
|
|
// correct initial value is know and the cache shouldn't refetch the same thing
|
|
|
|
// on startup. It is used to set the ConnectRootCA and AgentLeafCert when
|
|
|
|
// AutoEncrypt.TLS is turned on. The cache itself cannot fetch that the first
|
|
|
|
// time because it requires a special RPCType. Subsequent runs are fine though.
|
|
|
|
func (c *Cache) Prepopulate(t string, res FetchResult, dc, token, k string) error {
|
|
|
|
// Check the type that we're prepolulating
|
|
|
|
c.typesLock.RLock()
|
|
|
|
tEntry, ok := c.types[t]
|
|
|
|
c.typesLock.RUnlock()
|
|
|
|
if !ok {
|
|
|
|
return fmt.Errorf("unknown type in cache: %s", t)
|
|
|
|
}
|
|
|
|
key := makeEntryKey(t, dc, token, k)
|
|
|
|
newEntry := cacheEntry{
|
|
|
|
Valid: true, Value: res.Value, State: res.State, Index: res.Index,
|
|
|
|
FetchedAt: time.Now(), Waiter: make(chan struct{}),
|
|
|
|
Expiry: &cacheEntryExpiry{Key: key, TTL: tEntry.Opts.LastGetTTL},
|
|
|
|
}
|
|
|
|
c.entriesLock.Lock()
|
|
|
|
c.entries[key] = newEntry
|
|
|
|
c.entriesLock.Unlock()
|
|
|
|
return nil
|
|
|
|
}
|