mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-07 18:21:20 +00:00
103 lines
3.1 KiB
Go
103 lines
3.1 KiB
Go
package cache
|
|
|
|
import (
|
|
"errors"
|
|
"strconv"
|
|
"sync"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
|
"github.com/prysmaticlabs/prysm/shared/featureconfig"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"k8s.io/client-go/tools/cache"
|
|
)
|
|
|
|
var (
|
|
// ErrNotActiveCountInfo will be returned when a cache object is not a pointer to
|
|
// a ActiveCountByEpoch struct.
|
|
ErrNotActiveCountInfo = errors.New("object is not a active count obj")
|
|
|
|
// maxActiveCountListSize defines the max number of active count can cache.
|
|
maxActiveCountListSize = 1000
|
|
|
|
// Metrics.
|
|
activeCountCacheMiss = promauto.NewCounter(prometheus.CounterOpts{
|
|
Name: "active_validator_count_cache_miss",
|
|
Help: "The number of active validator count requests that aren't present in the cache.",
|
|
})
|
|
activeCountCacheHit = promauto.NewCounter(prometheus.CounterOpts{
|
|
Name: "active_validator_count_cache_hit",
|
|
Help: "The number of active validator count requests that are present in the cache.",
|
|
})
|
|
)
|
|
|
|
// ActiveCountByEpoch defines the active validator count per epoch.
|
|
type ActiveCountByEpoch struct {
|
|
Epoch uint64
|
|
ActiveCount uint64
|
|
}
|
|
|
|
// ActiveCountCache is a struct with 1 queue for looking up active count by epoch.
|
|
type ActiveCountCache struct {
|
|
activeCountCache *cache.FIFO
|
|
lock sync.RWMutex
|
|
}
|
|
|
|
// activeCountKeyFn takes the epoch as the key for the active count of a given epoch.
|
|
func activeCountKeyFn(obj interface{}) (string, error) {
|
|
aInfo, ok := obj.(*ActiveCountByEpoch)
|
|
if !ok {
|
|
return "", ErrNotActiveCountInfo
|
|
}
|
|
|
|
return strconv.Itoa(int(aInfo.Epoch)), nil
|
|
}
|
|
|
|
// NewActiveCountCache creates a new active count cache for storing/accessing active validator count.
|
|
func NewActiveCountCache() *ActiveCountCache {
|
|
return &ActiveCountCache{
|
|
activeCountCache: cache.NewFIFO(activeCountKeyFn),
|
|
}
|
|
}
|
|
|
|
// ActiveCountInEpoch fetches ActiveCountByEpoch by epoch. Returns true with a
|
|
// reference to the ActiveCountInEpoch info, if exists. Otherwise returns false, nil.
|
|
func (c *ActiveCountCache) ActiveCountInEpoch(epoch uint64) (uint64, error) {
|
|
if !featureconfig.Get().EnableActiveCountCache {
|
|
return params.BeaconConfig().FarFutureEpoch, nil
|
|
}
|
|
c.lock.RLock()
|
|
defer c.lock.RUnlock()
|
|
obj, exists, err := c.activeCountCache.GetByKey(strconv.Itoa(int(epoch)))
|
|
if err != nil {
|
|
return params.BeaconConfig().FarFutureEpoch, err
|
|
}
|
|
|
|
if exists {
|
|
activeCountCacheHit.Inc()
|
|
} else {
|
|
activeCountCacheMiss.Inc()
|
|
return params.BeaconConfig().FarFutureEpoch, nil
|
|
}
|
|
|
|
aInfo, ok := obj.(*ActiveCountByEpoch)
|
|
if !ok {
|
|
return params.BeaconConfig().FarFutureEpoch, ErrNotActiveCountInfo
|
|
}
|
|
|
|
return aInfo.ActiveCount, nil
|
|
}
|
|
|
|
// AddActiveCount adds ActiveCountByEpoch object to the cache. This method also trims the least
|
|
// recently added ActiveCountByEpoch object if the cache size has ready the max cache size limit.
|
|
func (c *ActiveCountCache) AddActiveCount(activeCount *ActiveCountByEpoch) error {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
if err := c.activeCountCache.AddIfNotPresent(activeCount); err != nil {
|
|
return err
|
|
}
|
|
|
|
trim(c.activeCountCache, maxActiveCountListSize)
|
|
return nil
|
|
}
|