-
Notifications
You must be signed in to change notification settings - Fork 906
/
blacklist.go
78 lines (66 loc) · 1.32 KB
/
blacklist.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
package main
import (
"container/list"
"sync"
"time"
)
type entry struct {
addr string
ctime time.Time
}
type blackList struct {
mu sync.Mutex
ll *list.List
cache map[string]*list.Element
expiredAfter time.Duration
limit int
}
func newBlackList(expiredAfter time.Duration, limit int) *blackList {
return &blackList{
ll: list.New(),
cache: make(map[string]*list.Element),
expiredAfter: expiredAfter,
limit: limit,
}
}
func (b *blackList) add(addr string) {
b.mu.Lock()
defer b.mu.Unlock()
b.removeExpiredLocked()
if b.ll.Len() >= b.limit {
return
}
if _, ok := b.cache[addr]; !ok {
e := b.ll.PushBack(&entry{
addr: addr,
ctime: time.Now(),
})
b.cache[addr] = e
}
}
func (b *blackList) has(addr string) bool {
b.mu.Lock()
defer b.mu.Unlock()
if elem := b.cache[addr]; elem != nil {
e := elem.Value.(*entry)
if time.Now().Sub(e.ctime) < b.expiredAfter {
return true
}
b.ll.Remove(elem)
delete(b.cache, e.addr)
}
return false
}
func (b *blackList) removeExpiredLocked() {
now := time.Now()
var next *list.Element
for elem := b.ll.Front(); elem != nil; elem = next {
next = elem.Next()
e := elem.Value.(*entry)
if now.Sub(e.ctime) < b.expiredAfter {
break
}
b.ll.Remove(elem)
delete(b.cache, e.addr)
}
}