198 lines
4.8 KiB
Go
198 lines
4.8 KiB
Go
// Copyright 2018 Prometheus Team
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package store
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/prometheus/common/model"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/prometheus/alertmanager/types"
|
|
)
|
|
|
|
func TestSetGet(t *testing.T) {
|
|
a := NewAlerts()
|
|
alert := &types.Alert{
|
|
UpdatedAt: time.Now(),
|
|
}
|
|
require.NoError(t, a.Set(alert))
|
|
want := alert.Fingerprint()
|
|
got, err := a.Get(want)
|
|
|
|
require.NoError(t, err)
|
|
require.Equal(t, want, got.Fingerprint())
|
|
}
|
|
|
|
func TestDeleteIfNotModified(t *testing.T) {
|
|
t.Run("unmodified alert should be deleted", func(t *testing.T) {
|
|
a := NewAlerts()
|
|
a1 := &types.Alert{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
UpdatedAt: time.Now().Add(-time.Second),
|
|
}
|
|
require.NoError(t, a.Set(a1))
|
|
|
|
// a1 should be deleted as it has not been modified.
|
|
a.DeleteIfNotModified(types.AlertSlice{a1})
|
|
got, err := a.Get(a1.Fingerprint())
|
|
require.Equal(t, ErrNotFound, err)
|
|
require.Nil(t, got)
|
|
})
|
|
|
|
t.Run("modified alert should not be deleted", func(t *testing.T) {
|
|
a := NewAlerts()
|
|
a1 := &types.Alert{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
UpdatedAt: time.Now(),
|
|
}
|
|
require.NoError(t, a.Set(a1))
|
|
|
|
// Make a copy of a1 that is older, but do not put it.
|
|
// We want to make sure a1 is not deleted.
|
|
a2 := &types.Alert{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
UpdatedAt: time.Now().Add(-time.Second),
|
|
}
|
|
require.True(t, a2.UpdatedAt.Before(a1.UpdatedAt))
|
|
a.DeleteIfNotModified(types.AlertSlice{a2})
|
|
// a1 should not be deleted.
|
|
got, err := a.Get(a1.Fingerprint())
|
|
require.NoError(t, err)
|
|
require.Equal(t, a1, got)
|
|
|
|
// Make another copy of a1 that is older, but do not put it.
|
|
// We want to make sure a2 is not deleted here either.
|
|
a3 := &types.Alert{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
UpdatedAt: time.Now().Add(time.Second),
|
|
}
|
|
require.True(t, a3.UpdatedAt.After(a1.UpdatedAt))
|
|
a.DeleteIfNotModified(types.AlertSlice{a3})
|
|
// a1 should not be deleted.
|
|
got, err = a.Get(a1.Fingerprint())
|
|
require.NoError(t, err)
|
|
require.Equal(t, a1, got)
|
|
})
|
|
|
|
t.Run("should not delete other alerts", func(t *testing.T) {
|
|
a := NewAlerts()
|
|
a1 := &types.Alert{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{
|
|
"foo": "bar",
|
|
},
|
|
},
|
|
UpdatedAt: time.Now(),
|
|
}
|
|
a2 := &types.Alert{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{
|
|
"bar": "baz",
|
|
},
|
|
},
|
|
UpdatedAt: time.Now(),
|
|
}
|
|
require.NoError(t, a.Set(a1))
|
|
require.NoError(t, a.Set(a2))
|
|
|
|
// Deleting a1 should not delete a2.
|
|
require.NoError(t, a.DeleteIfNotModified(types.AlertSlice{a1}))
|
|
// a1 should be deleted.
|
|
got, err := a.Get(a1.Fingerprint())
|
|
require.Equal(t, ErrNotFound, err)
|
|
require.Nil(t, got)
|
|
// a2 should not be deleted.
|
|
got, err = a.Get(a2.Fingerprint())
|
|
require.NoError(t, err)
|
|
require.Equal(t, a2, got)
|
|
})
|
|
}
|
|
|
|
func TestGC(t *testing.T) {
|
|
now := time.Now()
|
|
newAlert := func(key string, start, end time.Duration) *types.Alert {
|
|
return &types.Alert{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{model.LabelName(key): "b"},
|
|
StartsAt: now.Add(start * time.Minute),
|
|
EndsAt: now.Add(end * time.Minute),
|
|
},
|
|
}
|
|
}
|
|
active := []*types.Alert{
|
|
newAlert("b", 10, 20),
|
|
newAlert("c", -10, 10),
|
|
}
|
|
resolved := []*types.Alert{
|
|
newAlert("a", -10, -5),
|
|
newAlert("d", -10, -1),
|
|
}
|
|
s := NewAlerts()
|
|
var (
|
|
n int
|
|
done = make(chan struct{})
|
|
ctx, cancel = context.WithCancel(context.Background())
|
|
)
|
|
s.SetGCCallback(func(a []types.Alert) {
|
|
n += len(a)
|
|
if n >= len(resolved) {
|
|
cancel()
|
|
}
|
|
})
|
|
for _, alert := range append(active, resolved...) {
|
|
require.NoError(t, s.Set(alert))
|
|
}
|
|
go func() {
|
|
s.Run(ctx, 10*time.Millisecond)
|
|
close(done)
|
|
}()
|
|
select {
|
|
case <-done:
|
|
break
|
|
case <-time.After(1 * time.Second):
|
|
t.Fatal("garbage collection didn't complete in time")
|
|
}
|
|
|
|
for _, alert := range active {
|
|
if _, err := s.Get(alert.Fingerprint()); err != nil {
|
|
t.Errorf("alert %v should not have been gc'd", alert)
|
|
}
|
|
}
|
|
for _, alert := range resolved {
|
|
if _, err := s.Get(alert.Fingerprint()); err == nil {
|
|
t.Errorf("alert %v should have been gc'd", alert)
|
|
}
|
|
}
|
|
require.Len(t, resolved, n)
|
|
}
|