2020-04-06 13:44:03 +02:00
|
|
|
package notifier
|
2020-02-16 19:59:02 +01:00
|
|
|
|
2020-02-21 22:15:05 +01:00
|
|
|
import (
|
|
|
|
"bytes"
|
2020-05-10 18:58:17 +02:00
|
|
|
"context"
|
2020-02-21 22:15:05 +01:00
|
|
|
"fmt"
|
2020-04-06 13:44:03 +02:00
|
|
|
"io/ioutil"
|
2020-02-21 22:15:05 +01:00
|
|
|
"net/http"
|
2022-02-02 13:11:41 +01:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/utils"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promauth"
|
2020-02-21 22:15:05 +01:00
|
|
|
)
|
|
|
|
|
2020-02-16 19:59:02 +01:00
|
|
|
// AlertManager represents integration provider with Prometheus alert manager
|
2020-04-06 13:44:03 +02:00
|
|
|
// https://github.com/prometheus/alertmanager
|
2020-02-21 22:15:05 +01:00
|
|
|
type AlertManager struct {
|
2022-02-02 13:11:41 +01:00
|
|
|
addr string
|
|
|
|
argFunc AlertURLGenerator
|
|
|
|
client *http.Client
|
|
|
|
timeout time.Duration
|
|
|
|
|
|
|
|
authCfg *promauth.Config
|
|
|
|
|
|
|
|
metrics *metrics
|
|
|
|
}
|
|
|
|
|
|
|
|
type metrics struct {
|
|
|
|
alertsSent *utils.Counter
|
|
|
|
alertsSendErrors *utils.Counter
|
|
|
|
}
|
|
|
|
|
|
|
|
func newMetrics(addr string) *metrics {
|
|
|
|
return &metrics{
|
|
|
|
alertsSent: utils.GetOrCreateCounter(fmt.Sprintf("vmalert_alerts_sent_total{addr=%q}", addr)),
|
|
|
|
alertsSendErrors: utils.GetOrCreateCounter(fmt.Sprintf("vmalert_alerts_send_errors_total{addr=%q}", addr)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close is a destructor method for AlertManager
|
|
|
|
func (am *AlertManager) Close() {
|
|
|
|
am.metrics.alertsSent.Unregister()
|
|
|
|
am.metrics.alertsSendErrors.Unregister()
|
2020-02-21 22:15:05 +01:00
|
|
|
}
|
|
|
|
|
2021-08-31 11:28:02 +02:00
|
|
|
// Addr returns address where alerts are sent.
|
|
|
|
func (am AlertManager) Addr() string { return am.addr }
|
|
|
|
|
2020-02-16 19:59:02 +01:00
|
|
|
// Send an alert or resolve message
|
2020-05-10 18:58:17 +02:00
|
|
|
func (am *AlertManager) Send(ctx context.Context, alerts []Alert) error {
|
2022-02-02 13:11:41 +01:00
|
|
|
am.metrics.alertsSent.Add(len(alerts))
|
|
|
|
err := am.send(ctx, alerts)
|
|
|
|
if err != nil {
|
|
|
|
am.metrics.alertsSendErrors.Add(len(alerts))
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (am *AlertManager) send(ctx context.Context, alerts []Alert) error {
|
2020-04-06 13:44:03 +02:00
|
|
|
b := &bytes.Buffer{}
|
2020-03-13 11:19:31 +01:00
|
|
|
writeamRequest(b, alerts, am.argFunc)
|
2020-05-10 18:58:17 +02:00
|
|
|
|
2022-02-02 13:11:41 +01:00
|
|
|
req, err := http.NewRequest("POST", am.addr, b)
|
2020-02-21 22:15:05 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-11-09 17:03:50 +01:00
|
|
|
req.Header.Set("Content-Type", "application/json")
|
2022-02-02 13:11:41 +01:00
|
|
|
|
|
|
|
if am.timeout > 0 {
|
|
|
|
var cancel context.CancelFunc
|
|
|
|
ctx, cancel = context.WithTimeout(ctx, am.timeout)
|
|
|
|
defer cancel()
|
|
|
|
}
|
|
|
|
|
2020-05-10 18:58:17 +02:00
|
|
|
req = req.WithContext(ctx)
|
2022-02-02 13:11:41 +01:00
|
|
|
|
|
|
|
if am.authCfg != nil {
|
|
|
|
if auth := am.authCfg.GetAuthHeader(); auth != "" {
|
|
|
|
req.Header.Set("Authorization", auth)
|
|
|
|
}
|
2020-06-29 21:21:03 +02:00
|
|
|
}
|
2020-05-10 18:58:17 +02:00
|
|
|
resp, err := am.client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-02-21 22:15:05 +01:00
|
|
|
defer func() { _ = resp.Body.Close() }()
|
2020-04-06 13:44:03 +02:00
|
|
|
|
2020-02-21 22:15:05 +01:00
|
|
|
if resp.StatusCode != http.StatusOK {
|
2020-04-06 13:44:03 +02:00
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
2022-02-02 13:11:41 +01:00
|
|
|
return fmt.Errorf("failed to read response from %q: %w", am.addr, err)
|
2020-02-21 22:15:05 +01:00
|
|
|
}
|
2022-02-02 13:11:41 +01:00
|
|
|
return fmt.Errorf("invalid SC %d from %q; response body: %s", resp.StatusCode, am.addr, string(body))
|
2020-02-21 22:15:05 +01:00
|
|
|
}
|
2020-02-16 19:59:02 +01:00
|
|
|
return nil
|
|
|
|
}
|
2020-04-06 13:44:03 +02:00
|
|
|
|
|
|
|
// AlertURLGenerator returns URL to single alert by given name
|
2020-06-21 12:32:46 +02:00
|
|
|
type AlertURLGenerator func(Alert) string
|
2020-04-06 13:44:03 +02:00
|
|
|
|
|
|
|
const alertManagerPath = "/api/v2/alerts"
|
|
|
|
|
|
|
|
// NewAlertManager is a constructor for AlertManager
|
2022-02-02 13:11:41 +01:00
|
|
|
func NewAlertManager(alertManagerURL string, fn AlertURLGenerator, authCfg promauth.HTTPClientConfig, timeout time.Duration) (*AlertManager, error) {
|
|
|
|
tls := &promauth.TLSConfig{}
|
|
|
|
if authCfg.TLSConfig != nil {
|
|
|
|
tls = authCfg.TLSConfig
|
|
|
|
}
|
|
|
|
tr, err := utils.Transport(alertManagerURL, tls.CertFile, tls.KeyFile, tls.CAFile, tls.ServerName, tls.InsecureSkipVerify)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to create transport: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ba := &promauth.BasicAuthConfig{}
|
|
|
|
if authCfg.BasicAuth != nil {
|
|
|
|
ba = authCfg.BasicAuth
|
2020-04-06 13:44:03 +02:00
|
|
|
}
|
2022-02-02 13:11:41 +01:00
|
|
|
aCfg, err := utils.AuthConfig(ba.Username, ba.Password.String(), ba.PasswordFile, authCfg.BearerToken.String(), authCfg.BearerTokenFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to configure auth: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &AlertManager{
|
|
|
|
addr: alertManagerURL,
|
|
|
|
argFunc: fn,
|
|
|
|
authCfg: aCfg,
|
|
|
|
client: &http.Client{Transport: tr},
|
|
|
|
timeout: timeout,
|
|
|
|
metrics: newMetrics(alertManagerURL),
|
|
|
|
}, nil
|
2020-04-06 13:44:03 +02:00
|
|
|
}
|