forked from coinpaprika/ratelimiter
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmap_limit_store.go
65 lines (57 loc) · 1.87 KB
/
map_limit_store.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
package ratelimiter
import (
"fmt"
"sync"
"time"
)
type limitValue struct {
val int64
lastUpdate time.Time
}
// MapLimitStore represents internal limiter data database where data are stored in golang maps
type MapLimitStore struct {
data map[string]limitValue
mutex sync.RWMutex
expirationTime time.Duration
}
// NewMapLimitStore creates new in-memory data store for internal limiter data. Each element of MapLimitStore is set as expired after expirationTime from its last counter increment. Expired elements are removed with a period specified by the flushInterval argument
func NewMapLimitStore(expirationTime time.Duration, flushInterval time.Duration) (m *MapLimitStore) {
m = &MapLimitStore{
data: make(map[string]limitValue),
expirationTime: expirationTime,
}
go func() {
ticker := time.NewTicker(flushInterval)
for range ticker.C {
for key, val := range m.data {
if val.lastUpdate.Before(time.Now().UTC().Add(-m.expirationTime)) {
m.mutex.Lock()
delete(m.data, key)
m.mutex.Unlock()
}
}
}
}()
return m
}
// Inc increments current window limit counter for key
func (m *MapLimitStore) Inc(key string, window time.Time) error {
m.mutex.Lock()
defer m.mutex.Unlock()
data := m.data[mapKey(key, window)]
data.val++
data.lastUpdate = time.Now().UTC()
m.data[mapKey(key, window)] = data
return nil
}
// Get gets value of previous window counter and current window counter for key
func (m *MapLimitStore) Get(key string, previousWindow, currentWindow time.Time) (prevValue int64, currValue int64, err error) {
m.mutex.RLock()
defer m.mutex.RUnlock()
prevValue = m.data[mapKey(key, previousWindow)].val
currValue = m.data[mapKey(key, currentWindow)].val
return prevValue, currValue, nil
}
func mapKey(key string, window time.Time) string {
return fmt.Sprintf("%s_%s", key, window.Format(time.RFC3339))
}