mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-18 15:54:13 +00:00
661cbc45ae
* add changes * fix tests * change to minute * remove dep * remove * fix tests * add test for period * improve * linter * build files * ci * make it stricter * fix tests * fix * Update beacon-chain/sync/rate_limiter.go Co-authored-by: Preston Van Loon <preston@prysmaticlabs.com> Co-authored-by: terencechain <terence@prysmaticlabs.com> Co-authored-by: Preston Van Loon <preston@prysmaticlabs.com>
203 lines
4.8 KiB
Go
203 lines
4.8 KiB
Go
package leakybucket
|
|
|
|
import (
|
|
"container/heap"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
//TODO: Finer grained locking.
|
|
|
|
type bucketMap map[string]*LeakyBucket
|
|
|
|
// A Collector can keep track of multiple LeakyBucket's. The caller does not
|
|
// directly interact with the buckets, but instead addresses them by a string
|
|
// key (e.g. IP address, hostname, hash, etc.) that is passed to most Collector
|
|
// methods.
|
|
//
|
|
// All Collector methods are goroutine safe.
|
|
type Collector struct {
|
|
buckets bucketMap
|
|
heap priorityQueue
|
|
rate float64
|
|
capacity int64
|
|
period time.Duration
|
|
lock sync.Mutex
|
|
quit chan bool
|
|
}
|
|
|
|
// NewCollector creates a new Collector. When new buckets are created within
|
|
// the Collector, they will be assigned the capacity and rate of the Collector.
|
|
// A Collector does not provide a way to change the rate or capacity of
|
|
// bucket's within it. If different rates or capacities are required, either
|
|
// use multiple Collector's or manage your own LeakyBucket's.
|
|
//
|
|
// If deleteEmptyBuckets is true, a concurrent goroutine will be run that
|
|
// watches for bucket's that become empty and automatically removes them,
|
|
// freeing up memory resources.
|
|
func NewCollector(rate float64, capacity int64, period time.Duration, deleteEmptyBuckets bool) *Collector {
|
|
c := &Collector{
|
|
buckets: make(bucketMap),
|
|
heap: make(priorityQueue, 0, 4096),
|
|
rate: rate,
|
|
capacity: capacity,
|
|
period: period,
|
|
quit: make(chan bool),
|
|
}
|
|
|
|
if deleteEmptyBuckets {
|
|
c.PeriodicPrune()
|
|
}
|
|
|
|
return c
|
|
}
|
|
|
|
// Free releases the collector's resources. If the collector was created with
|
|
// deleteEmptyBuckets = true, then the goroutine looking for empty buckets,
|
|
// will be stopped.
|
|
func (c *Collector) Free() {
|
|
c.Reset()
|
|
close(c.quit)
|
|
}
|
|
|
|
// Reset removes all internal buckets and resets the collector back to as if it
|
|
// was just created.
|
|
func (c *Collector) Reset() {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
|
|
// Let the garbage collector do all the work.
|
|
c.buckets = make(bucketMap)
|
|
c.heap = make(priorityQueue, 0, 4096)
|
|
}
|
|
|
|
// Capacity returns the collector's capacity.
|
|
func (c *Collector) Capacity() int64 {
|
|
return c.capacity
|
|
}
|
|
|
|
// Rate returns the collector's rate.
|
|
func (c *Collector) Rate() float64 {
|
|
return c.rate
|
|
}
|
|
|
|
// Remaining returns the remaining capacity of the internal bucket associated
|
|
// with key. If key is not associated with a bucket internally, it is treated
|
|
// as being empty.
|
|
func (c *Collector) Remaining(key string) int64 {
|
|
return c.capacity - c.Count(key)
|
|
}
|
|
|
|
// Count returns the count of the internal bucket associated with key. If key
|
|
// is not associated with a bucket internally, it is treated as being empty.
|
|
func (c *Collector) Count(key string) int64 {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
|
|
b, ok := c.buckets[key]
|
|
if !ok || b == nil {
|
|
return 0
|
|
}
|
|
|
|
return b.Count()
|
|
}
|
|
|
|
// TillEmpty returns how much time must pass until the internal bucket
|
|
// associated with key is empty. If key is not associated with a bucket
|
|
// internally, it is treated as being empty.
|
|
func (c *Collector) TillEmpty(key string) time.Duration {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
|
|
b, ok := c.buckets[key]
|
|
if !ok || b == nil {
|
|
return 0
|
|
}
|
|
|
|
return b.TillEmpty()
|
|
}
|
|
|
|
// Remove deletes the internal bucket associated with key. If key is not
|
|
// associated with a bucket internally, nothing is done.
|
|
func (c *Collector) Remove(key string) {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
|
|
b, ok := c.buckets[key]
|
|
if !ok || b == nil {
|
|
return
|
|
}
|
|
|
|
delete(c.buckets, b.key)
|
|
heap.Remove(&c.heap, b.index)
|
|
}
|
|
|
|
// Add 'amount' to the internal bucket associated with key, up to it's
|
|
// capacity. Returns how much was added to the bucket. If the return is less
|
|
// than 'amount', then the bucket's capacity was reached.
|
|
//
|
|
// If key is not associated with a bucket internally, a new bucket is created
|
|
// and amount is added to it.
|
|
func (c *Collector) Add(key string, amount int64) int64 {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
|
|
b, ok := c.buckets[key]
|
|
|
|
if !ok || b == nil {
|
|
// Create a new bucket.
|
|
b = &LeakyBucket{
|
|
key: key,
|
|
capacity: c.capacity,
|
|
rate: c.rate,
|
|
period: c.period,
|
|
p: now(),
|
|
}
|
|
c.heap.Push(b)
|
|
c.buckets[key] = b
|
|
}
|
|
|
|
n := b.Add(amount)
|
|
|
|
if n > 0 {
|
|
heap.Fix(&c.heap, b.index)
|
|
}
|
|
|
|
return n
|
|
}
|
|
|
|
// Prune removes all empty buckets in the collector.
|
|
func (c *Collector) Prune() {
|
|
c.lock.Lock()
|
|
for c.heap.Peak() != nil {
|
|
b := c.heap.Peak()
|
|
|
|
if now().Before(b.p) {
|
|
// The bucket isn't empty.
|
|
break
|
|
}
|
|
|
|
// The bucket should be empty.
|
|
delete(c.buckets, b.key)
|
|
heap.Remove(&c.heap, b.index)
|
|
}
|
|
c.lock.Unlock()
|
|
}
|
|
|
|
// PeriodicPrune runs a concurrent goroutine that calls Prune() at the given
|
|
// time interval.
|
|
func (c *Collector) PeriodicPrune() {
|
|
go func() {
|
|
ticker := time.NewTicker(c.period)
|
|
for {
|
|
select {
|
|
case <-ticker.C:
|
|
c.Prune()
|
|
case <-c.quit:
|
|
ticker.Stop()
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|