forked from caddyserver/certmagic
-
Notifications
You must be signed in to change notification settings - Fork 0
/
ratelimiter.go
160 lines (145 loc) · 4.33 KB
/
ratelimiter.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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
package certmagic
import (
"sync"
"time"
)
// NewRateLimiter returns a rate limiter that allows up to maxEvents
// in a sliding window of size window. If maxEvents and window are
// both 0, or if maxEvents is non-zero and window is 0, rate limiting
// is disabled. If maxEvents is 0 but the window is non-zero, it is
// impossible to make reservations, so Allow() will always return false
// and Wait() will panic (instead of blocking forever). This function
// panics if maxEvents is less than 0.
func NewRateLimiter(maxEvents int, window time.Duration) *RingBufferRateLimiter {
if maxEvents < 0 {
panic("maxEvents cannot be less than zero")
}
return &RingBufferRateLimiter{
window: window,
ring: make([]time.Time, maxEvents),
}
}
// RingBufferRateLimiter uses a ring to enforce rate limits
// consisting of a maximum number of events within a single
// sliding window of a given duration. An empty value is
// not valid; use NewRateLimiter to get one.
type RingBufferRateLimiter struct {
window time.Duration
ring []time.Time // maxEvents == len(ring)
cursor int // always points to the oldest timestamp
mu sync.Mutex
}
// Allow returns true if the event is allowed to
// happen right now. It does not wait. If the event
// is allowed, a reservation is made.
func (r *RingBufferRateLimiter) Allow() bool {
r.mu.Lock()
defer r.mu.Unlock()
if len(r.ring) == 0 {
return r.window == 0
}
if time.Since(r.ring[r.cursor]) >= r.window {
r.reserve(time.Now())
return true
}
return false
}
// Wait makes a reservation then blocks until the
// event is allowed to occur. It panics if maxEvents
// is 0 but the window is non-zero, because Wait
// would only be able to block forever in that case.
func (r *RingBufferRateLimiter) Wait() {
r.mu.Lock()
if len(r.ring) == 0 {
if r.window == 0 {
r.mu.Unlock()
return
}
panic("cannot wait when maxEvents = 0 and window != 0 ")
}
if r.ring[r.cursor].IsZero() {
r.reserve(time.Now())
r.mu.Unlock()
return
}
then := r.ring[r.cursor].Add(r.window)
r.reserve(then)
r.mu.Unlock()
time.Sleep(time.Until(then))
}
// MaxEvents returns the maximum number of events that
// are allowed within the sliding window.
func (r *RingBufferRateLimiter) MaxEvents() int {
r.mu.Lock()
defer r.mu.Unlock()
return len(r.ring)
}
// SetMaxEvents changes the maximum number of events that are
// allowed in the sliding window. If the new limit is lower,
// the oldest events will be forgotten. If the new limit is
// higher, the window will suddenly have capacity for new
// reservations.
func (r *RingBufferRateLimiter) SetMaxEvents(maxEvents int) {
newRing := make([]time.Time, maxEvents)
r.mu.Lock()
defer r.mu.Unlock()
// only make the change if the new limit is different
if maxEvents == len(r.ring) {
return
}
// the new ring may be smaller; fast-forward to the
// oldest timestamp that will be kept in the new
// ring so the oldest ones are forgotten and the
// newest ones will be remembered
sizeDiff := len(r.ring) - maxEvents
for i := 0; i < sizeDiff; i++ {
r.advance()
}
if len(r.ring) > 0 {
// copy timestamps into the new ring until we
// have either copied all of them or have reached
// the capacity of the new ring
startCursor := r.cursor
for i := 0; i < len(newRing); i++ {
newRing[i] = r.ring[r.cursor]
r.advance()
if r.cursor == startCursor {
// new ring is larger than old one;
// "we've come full circle"
break
}
}
}
r.ring = newRing
r.cursor = 0
}
// Window returns the size of the sliding window.
func (r *RingBufferRateLimiter) Window() time.Duration {
r.mu.Lock()
defer r.mu.Unlock()
return r.window
}
// SetWindow changes r's sliding window duration to window.
// Goroutines that are already blocked on a call to Wait()
// will not be affected.
func (r *RingBufferRateLimiter) SetWindow(window time.Duration) {
r.mu.Lock()
r.window = window
r.mu.Unlock()
}
// reserve claims the current spot at the head of
// the window. It is NOT safe for concurrent use,
// so it must be called inside a lock on r.mu.
func (r *RingBufferRateLimiter) reserve(when time.Time) {
r.ring[r.cursor] = when
r.advance()
}
// advance moves the cursor to the next position.
// It is NOT safe for concurrent use, so it must
// be called inside a lock on r.mu.
func (r *RingBufferRateLimiter) advance() {
r.cursor++
if r.cursor >= len(r.ring) {
r.cursor = 0
}
}