forked from rcrowley/go-metrics
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathewma.go
80 lines (69 loc) · 1.9 KB
/
ewma.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
package metrics
import (
"math"
"sync"
"sync/atomic"
)
// EWMAs continuously calculate an exponentially-weighted moving average
// based on an outside source of clock ticks.
//
// This is an interface so as to encourage other structs to implement
// the EWMA API as appropriate.
type EWMA interface {
Rate() float64
Tick()
Update(int64)
}
// The standard implementation of an EWMA tracks the number of uncounted
// events and processes them on each tick. It uses the sync/atomic package
// to manage uncounted events.
type StandardEWMA struct {
alpha float64
init bool
mutex *sync.Mutex
rate float64
uncounted int64
}
// Force the compiler to check that StandardEWMA implements EWMA.
var _ EWMA = &StandardEWMA{}
// Create a new EWMA with the given alpha.
func NewEWMA(alpha float64) *StandardEWMA {
return &StandardEWMA{
alpha: alpha,
mutex: &sync.Mutex{},
}
}
// Create a new EWMA with alpha set for a one-minute moving average.
func NewEWMA1() *StandardEWMA {
return NewEWMA(1 - math.Exp(-5.0/60.0/1))
}
// Create a new EWMA with alpha set for a five-minute moving average.
func NewEWMA5() *StandardEWMA {
return NewEWMA(1 - math.Exp(-5.0/60.0/5))
}
// Create a new EWMA with alpha set for a fifteen-minute moving average.
func NewEWMA15() *StandardEWMA {
return NewEWMA(1 - math.Exp(-5.0/60.0/15))
}
// Return the moving average rate of events per second.
func (a *StandardEWMA) Rate() float64 {
return a.rate * float64(1e9)
}
// Tick the clock to update the moving average.
func (a *StandardEWMA) Tick() {
count := atomic.LoadInt64(&a.uncounted)
atomic.AddInt64(&a.uncounted, -count)
instantRate := float64(count) / float64(5e9)
a.mutex.Lock()
defer a.mutex.Unlock()
if a.init {
a.rate += a.alpha * (instantRate - a.rate)
} else {
a.init = true
a.rate = instantRate
}
}
// Add n uncounted events.
func (a *StandardEWMA) Update(n int64) {
atomic.AddInt64(&a.uncounted, n)
}