mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-11-21 14:44:00 +00:00
ae4d376e41
…pair `alert_relabel_configs` in [notifier config](https://docs.victoriametrics.com/vmalert/#notifier-configuration-file) can drop alert labels when used to filter different tenant alert message to different notifier. alertmanager would report error like `msg="Failed to validate alerts" err="at least one label pair required"` in this case, but the rest of the alerts inside one request would still be valid in alertmanager, so it's not severe.
174 lines
5.2 KiB
Go
174 lines
5.2 KiB
Go
package notifier
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"strconv"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promauth"
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promrelabel"
|
|
)
|
|
|
|
func TestAlertManager_Addr(t *testing.T) {
|
|
const addr = "http://localhost"
|
|
am, err := NewAlertManager(addr, nil, promauth.HTTPClientConfig{}, nil, 0)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %s", err)
|
|
}
|
|
if am.Addr() != addr {
|
|
t.Fatalf("expected to have %q; got %q", addr, am.Addr())
|
|
}
|
|
}
|
|
|
|
func TestAlertManager_Send(t *testing.T) {
|
|
const baUser, baPass = "foo", "bar"
|
|
const headerKey, headerValue = "TenantID", "foo"
|
|
mux := http.NewServeMux()
|
|
mux.HandleFunc("/", func(_ http.ResponseWriter, _ *http.Request) {
|
|
t.Fatalf("should not be called")
|
|
})
|
|
c := -1
|
|
mux.HandleFunc(alertManagerPath, func(w http.ResponseWriter, r *http.Request) {
|
|
user, pass, ok := r.BasicAuth()
|
|
if !ok {
|
|
t.Fatalf("unauthorized request")
|
|
}
|
|
if user != baUser || pass != baPass {
|
|
t.Fatalf("wrong creds %q:%q; expected %q:%q", user, pass, baUser, baPass)
|
|
}
|
|
c++
|
|
if r.Method != http.MethodPost {
|
|
t.Fatalf("expected POST method got %s", r.Method)
|
|
}
|
|
switch c {
|
|
case 0:
|
|
conn, _, _ := w.(http.Hijacker).Hijack()
|
|
_ = conn.Close()
|
|
case 1:
|
|
if r.Header.Get(headerKey) != headerValue {
|
|
t.Fatalf("expected header %q to be set to %q; got %q instead", headerKey, headerValue, r.Header.Get(headerKey))
|
|
}
|
|
w.WriteHeader(500)
|
|
case 2:
|
|
var a []struct {
|
|
Labels map[string]string `json:"labels"`
|
|
StartsAt time.Time `json:"startsAt"`
|
|
EndAt time.Time `json:"endsAt"`
|
|
Annotations map[string]string `json:"annotations"`
|
|
GeneratorURL string `json:"generatorURL"`
|
|
}
|
|
if err := json.NewDecoder(r.Body).Decode(&a); err != nil {
|
|
t.Fatalf("can not unmarshal data into alert %s", err)
|
|
}
|
|
if len(a) != 1 {
|
|
t.Fatalf("expected 1 alert in array got %d", len(a))
|
|
}
|
|
if a[0].GeneratorURL != "0/0" {
|
|
t.Fatalf("expected 0/0 as generatorURL got %s", a[0].GeneratorURL)
|
|
}
|
|
if a[0].StartsAt.IsZero() {
|
|
t.Fatalf("expected non-zero start time")
|
|
}
|
|
if a[0].EndAt.IsZero() {
|
|
t.Fatalf("expected non-zero end time")
|
|
}
|
|
if len(a[0].Labels) != 1 {
|
|
t.Fatalf("expected 1 labels got %d", len(a[0].Labels))
|
|
}
|
|
if len(a[0].Annotations) != 2 {
|
|
t.Fatalf("expected 2 annotations got %d", len(a[0].Annotations))
|
|
}
|
|
if r.Header.Get(headerKey) != "bar" {
|
|
t.Fatalf("expected header %q to be set to %q; got %q instead", headerKey, headerValue, r.Header.Get(headerKey))
|
|
}
|
|
case 3:
|
|
var a []struct {
|
|
Labels map[string]string `json:"labels"`
|
|
}
|
|
if err := json.NewDecoder(r.Body).Decode(&a); err != nil {
|
|
t.Fatalf("can not unmarshal data into alert %s", err)
|
|
}
|
|
if len(a) != 1 {
|
|
t.Fatalf("expected 1 alert in array got %d", len(a))
|
|
}
|
|
if len(a[0].Labels) != 3 {
|
|
t.Fatalf("expected 3 labels got %d", len(a[0].Labels))
|
|
}
|
|
if a[0].Labels["env"] != "prod" {
|
|
t.Fatalf("expected env label to be prod during relabeling, got %s", a[0].Labels["env"])
|
|
}
|
|
if r.Header.Get(headerKey) != "bar" {
|
|
t.Fatalf("expected header %q to be set to %q; got %q instead", headerKey, "bar", r.Header.Get(headerKey))
|
|
}
|
|
}
|
|
})
|
|
srv := httptest.NewServer(mux)
|
|
defer srv.Close()
|
|
|
|
aCfg := promauth.HTTPClientConfig{
|
|
BasicAuth: &promauth.BasicAuthConfig{
|
|
Username: baUser,
|
|
Password: promauth.NewSecret(baPass),
|
|
},
|
|
Headers: []string{fmt.Sprintf("%s:%s", headerKey, headerValue)},
|
|
}
|
|
parsedConfigs, err := promrelabel.ParseRelabelConfigsData([]byte(`
|
|
- action: drop
|
|
if: '{tenant="0"}'
|
|
regex: ".*"
|
|
- target_label: "env"
|
|
replacement: "prod"
|
|
if: '{tenant="1"}'
|
|
`))
|
|
if err != nil {
|
|
t.Fatalf("unexpected error when parse relabeling config: %s", err)
|
|
}
|
|
am, err := NewAlertManager(srv.URL+alertManagerPath, func(alert Alert) string {
|
|
return strconv.FormatUint(alert.GroupID, 10) + "/" + strconv.FormatUint(alert.ID, 10)
|
|
}, aCfg, parsedConfigs, 0)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %s", err)
|
|
}
|
|
|
|
if err := am.Send(context.Background(), []Alert{{Labels: map[string]string{"a": "b"}}}, nil); err == nil {
|
|
t.Fatalf("expected connection error got nil")
|
|
}
|
|
|
|
if err := am.Send(context.Background(), []Alert{{Labels: map[string]string{"a": "b"}}}, nil); err == nil {
|
|
t.Fatalf("expected wrong http code error got nil")
|
|
}
|
|
|
|
if err := am.Send(context.Background(), []Alert{{
|
|
GroupID: 0,
|
|
Name: "alert0",
|
|
Start: time.Now().UTC(),
|
|
End: time.Now().UTC(),
|
|
Labels: map[string]string{"alertname": "alert0"},
|
|
Annotations: map[string]string{"a": "b", "c": "d"},
|
|
}}, map[string]string{headerKey: "bar"}); err != nil {
|
|
t.Fatalf("unexpected error %s", err)
|
|
}
|
|
|
|
if err := am.Send(context.Background(), []Alert{
|
|
// drop tenant0 alert message during relabeling
|
|
{
|
|
Name: "alert1",
|
|
Labels: map[string]string{"rule": "test", "tenant": "0"},
|
|
},
|
|
{
|
|
Name: "alert2",
|
|
Labels: map[string]string{"rule": "test", "tenant": "1"},
|
|
},
|
|
}, map[string]string{headerKey: "bar"}); err != nil {
|
|
t.Fatalf("unexpected error %s", err)
|
|
}
|
|
|
|
if c != 3 {
|
|
t.Fatalf("expected 3 calls(count from zero) to server got %d", c)
|
|
}
|
|
}
|