provider/mesh: add notification garbage collection

This commit is contained in:
Fabian Reinartz 2016-06-10 12:25:56 +02:00
parent 06a8700472
commit c1ee634c16
2 changed files with 65 additions and 3 deletions

View File

@ -21,20 +21,53 @@ type notificationEntry struct {
type notificationState struct {
mtx sync.RWMutex
set map[string]notificationEntry
stopc chan struct{}
now func() time.Time // test injection hook
}
const gcInterval = 10 * time.Minute
func newNotificationState() *notificationState {
return &notificationState{
set: map[string]notificationEntry{},
stopc: make(chan struct{}),
now: time.Now,
}
}
func (s *notificationState) run(retention time.Duration) {
for {
select {
case <-s.stopc:
return
case <-time.After(gcInterval):
s.gc(retention)
}
}
}
func (s *notificationState) stop() {
close(s.stopc)
}
func decodeNotificationSet(b []byte) (map[string]notificationEntry, error) {
var v map[string]notificationEntry
err := gob.NewDecoder(bytes.NewReader(b)).Decode(&v)
return v, err
}
func (s *notificationState) gc(retention time.Duration) {
s.mtx.Lock()
defer s.mtx.Unlock()
t := s.now().Add(-retention)
for k, v := range s.set {
if v.Timestamp.Before(t) {
delete(s.set, k)
}
}
}
// copy returns a deep copy of the notification state.
func (s *notificationState) copy() *notificationState {
s.mtx.RLock()

View File

@ -12,6 +12,35 @@ import (
"github.com/satori/go.uuid"
)
func TestNotificationStateGC(t *testing.T) {
now := time.Now()
initial := map[string]notificationEntry{
"1": {true, now},
"2": {true, now.Add(30 * time.Minute)},
"3": {true, now.Add(-30 * time.Minute)},
"4": {true, now.Add(-60 * time.Minute)},
"5": {true, now.Add(-61 * time.Minute)},
"6": {true, now.Add(-100 * time.Hour)},
}
final := map[string]notificationEntry{
"1": {true, now},
"2": {true, now.Add(30 * time.Minute)},
"3": {true, now.Add(-30 * time.Minute)},
"4": {true, now.Add(-60 * time.Minute)},
}
st := newNotificationState()
st.now = func() time.Time { return now }
st.set = initial
st.gc(time.Hour)
if !reflect.DeepEqual(st.set, final) {
t.Errorf("Unexpected state after GC")
t.Errorf("%s", pretty.Compare(st.set, final))
}
}
func TestSilenceStateSet(t *testing.T) {
var (
now = time.Now()