mirror of
https://github.com/grafana/grafana.git
synced 2025-02-20 11:48:34 -06:00
* Allow customizable googlechat message via optional setting * Add optional message field in googlechat contact point configurator * Fix strange error message on send if template fails to fully evaluate * Elevate template evaluation failure logs to Warn level * Extract default.title template embed from all channels to shared constant
126 lines
3.5 KiB
Go
126 lines
3.5 KiB
Go
package channels
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
|
|
"github.com/prometheus/alertmanager/notify"
|
|
"github.com/prometheus/alertmanager/template"
|
|
"github.com/prometheus/alertmanager/types"
|
|
"github.com/prometheus/common/model"
|
|
|
|
"github.com/grafana/grafana/pkg/bus"
|
|
"github.com/grafana/grafana/pkg/components/simplejson"
|
|
"github.com/grafana/grafana/pkg/infra/log"
|
|
"github.com/grafana/grafana/pkg/models"
|
|
)
|
|
|
|
// KafkaNotifier is responsible for sending
|
|
// alert notifications to Kafka.
|
|
type KafkaNotifier struct {
|
|
*Base
|
|
Endpoint string
|
|
Topic string
|
|
log log.Logger
|
|
tmpl *template.Template
|
|
}
|
|
|
|
// NewKafkaNotifier is the constructor function for the Kafka notifier.
|
|
func NewKafkaNotifier(model *NotificationChannelConfig, t *template.Template) (*KafkaNotifier, error) {
|
|
if model.Settings == nil {
|
|
return nil, receiverInitError{Cfg: *model, Reason: "no settings supplied"}
|
|
}
|
|
|
|
endpoint := model.Settings.Get("kafkaRestProxy").MustString()
|
|
if endpoint == "" {
|
|
return nil, receiverInitError{Cfg: *model, Reason: "could not find kafka rest proxy endpoint property in settings"}
|
|
}
|
|
topic := model.Settings.Get("kafkaTopic").MustString()
|
|
if topic == "" {
|
|
return nil, receiverInitError{Cfg: *model, Reason: "could not find kafka topic property in settings"}
|
|
}
|
|
|
|
return &KafkaNotifier{
|
|
Base: NewBase(&models.AlertNotification{
|
|
Uid: model.UID,
|
|
Name: model.Name,
|
|
Type: model.Type,
|
|
DisableResolveMessage: model.DisableResolveMessage,
|
|
Settings: model.Settings,
|
|
}),
|
|
Endpoint: endpoint,
|
|
Topic: topic,
|
|
log: log.New("alerting.notifier.kafka"),
|
|
tmpl: t,
|
|
}, nil
|
|
}
|
|
|
|
// Notify sends the alert notification.
|
|
func (kn *KafkaNotifier) Notify(ctx context.Context, as ...*types.Alert) (bool, error) {
|
|
// We are using the state from 7.x to not break kafka.
|
|
// TODO: should we switch to the new ones?
|
|
alerts := types.Alerts(as...)
|
|
state := models.AlertStateAlerting
|
|
if alerts.Status() == model.AlertResolved {
|
|
state = models.AlertStateOK
|
|
}
|
|
|
|
kn.log.Debug("Notifying Kafka", "alert_state", state)
|
|
|
|
var tmplErr error
|
|
tmpl, _ := TmplText(ctx, kn.tmpl, as, kn.log, &tmplErr)
|
|
|
|
bodyJSON := simplejson.New()
|
|
bodyJSON.Set("alert_state", state)
|
|
bodyJSON.Set("description", tmpl(DefaultMessageTitleEmbed))
|
|
bodyJSON.Set("client", "Grafana")
|
|
bodyJSON.Set("details", tmpl(`{{ template "default.message" . }}`))
|
|
|
|
ruleURL := joinUrlPath(kn.tmpl.ExternalURL.String(), "/alerting/list", kn.log)
|
|
bodyJSON.Set("client_url", ruleURL)
|
|
|
|
groupKey, err := notify.ExtractGroupKey(ctx)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
bodyJSON.Set("incident_key", groupKey.Hash())
|
|
|
|
valueJSON := simplejson.New()
|
|
valueJSON.Set("value", bodyJSON)
|
|
|
|
recordJSON := simplejson.New()
|
|
recordJSON.Set("records", []interface{}{valueJSON})
|
|
|
|
body, err := recordJSON.MarshalJSON()
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
topicURL := strings.TrimRight(kn.Endpoint, "/") + "/topics/" + tmpl(kn.Topic)
|
|
|
|
if tmplErr != nil {
|
|
kn.log.Warn("failed to template Kafka message", "err", tmplErr.Error())
|
|
}
|
|
|
|
cmd := &models.SendWebhookSync{
|
|
Url: topicURL,
|
|
Body: string(body),
|
|
HttpMethod: "POST",
|
|
HttpHeader: map[string]string{
|
|
"Content-Type": "application/vnd.kafka.json.v2+json",
|
|
"Accept": "application/vnd.kafka.v2+json",
|
|
},
|
|
}
|
|
|
|
if err := bus.Dispatch(ctx, cmd); err != nil {
|
|
kn.log.Error("Failed to send notification to Kafka", "error", err, "body", string(body))
|
|
return false, err
|
|
}
|
|
|
|
return true, nil
|
|
}
|
|
|
|
func (kn *KafkaNotifier) SendResolved() bool {
|
|
return !kn.GetDisableResolveMessage()
|
|
}
|