grafana/pkg/services/ngalert/notifier/config.go
gotjosh 511dab3b4b
Update grafana/alerting to the latest main (#61810)
* Update `grafana/alerting` to the latest main

Also updates prometheus-alertmanager since we use that one directly for some structs.
2023-01-19 20:44:49 +00:00

160 lines
5.0 KiB
Go

package notifier
import (
"crypto/md5"
"encoding/json"
"fmt"
"os"
"path/filepath"
"github.com/grafana/alerting/alerting"
"github.com/grafana/alerting/alerting/notifier/channels"
"github.com/grafana/grafana/pkg/infra/log"
api "github.com/grafana/grafana/pkg/services/ngalert/api/tooling/definitions"
)
var cfglogger = log.New("notifier.config")
func PersistTemplates(cfg *api.PostableUserConfig, path string) ([]string, bool, error) {
if len(cfg.TemplateFiles) < 1 {
return nil, false, nil
}
var templatesChanged bool
pathSet := map[string]struct{}{}
for name, content := range cfg.TemplateFiles {
if name != filepath.Base(filepath.Clean(name)) {
return nil, false, fmt.Errorf("template file name '%s' is not valid", name)
}
err := os.MkdirAll(path, 0750)
if err != nil {
return nil, false, fmt.Errorf("unable to create template directory %q: %s", path, err)
}
file := filepath.Join(path, name)
pathSet[file] = struct{}{}
// Check if the template file already exists and if it has changed
// We can safely ignore gosec here as we've previously checked the filename is clean
// nolint:gosec
if tmpl, err := os.ReadFile(file); err == nil && string(tmpl) == content {
// Templates file is the same we have, no-op and continue.
continue
} else if err != nil && !os.IsNotExist(err) {
return nil, false, err
}
// We can safely ignore gosec here as we've previously checked the filename is clean
// nolint:gosec
if err := os.WriteFile(file, []byte(content), 0644); err != nil {
return nil, false, fmt.Errorf("unable to create Alertmanager template file %q: %s", file, err)
}
templatesChanged = true
}
// Now that we have the list of _actual_ templates, let's remove the ones that we don't need.
existingFiles, err := os.ReadDir(path)
if err != nil {
cfglogger.Error("unable to read directory for deleting Alertmanager templates", "error", err, "path", path)
}
for _, existingFile := range existingFiles {
p := filepath.Join(path, existingFile.Name())
_, ok := pathSet[p]
if !ok {
templatesChanged = true
err := os.Remove(p)
if err != nil {
cfglogger.Error("unable to delete template", "error", err, "file", p)
}
}
}
paths := make([]string, 0, len(pathSet))
for path := range pathSet {
paths = append(paths, path)
}
return paths, templatesChanged, nil
}
func Load(rawConfig []byte) (*api.PostableUserConfig, error) {
cfg := &api.PostableUserConfig{}
if err := json.Unmarshal(rawConfig, cfg); err != nil {
return nil, fmt.Errorf("unable to parse Alertmanager configuration: %w", err)
}
return cfg, nil
}
// AlertingConfiguration provides configuration for an Alertmanager.
// It implements the alerting.Configuration interface.
type AlertingConfiguration struct {
AlertmanagerConfig api.PostableApiAlertingConfig
RawAlertmanagerConfig []byte
AlertmanagerTemplates *alerting.Template
IntegrationsFunc func(receivers []*api.PostableApiReceiver, templates *alerting.Template) (map[string][]*alerting.Integration, error)
ReceiverIntegrationsFunc func(r *api.PostableGrafanaReceiver, tmpl *alerting.Template) (channels.NotificationChannel, error)
}
func (a AlertingConfiguration) BuildReceiverIntegrationsFunc() func(next *alerting.GrafanaReceiver, tmpl *alerting.Template) (alerting.Notifier, error) {
return func(next *alerting.GrafanaReceiver, tmpl *alerting.Template) (alerting.Notifier, error) {
//TODO: We shouldn't need to do all of this marshalling - there should be no difference between types.
var out api.RawMessage
settingsJSON, err := json.Marshal(next.Settings)
if err != nil {
return nil, fmt.Errorf("unable to marshal settings to JSON: %v", err)
}
err = out.UnmarshalJSON(settingsJSON)
if err != nil {
return nil, fmt.Errorf("unable to marshal JSON to RawMessage: %v", err)
}
gr := &api.PostableGrafanaReceiver{
UID: next.UID,
Name: next.Name,
Type: next.Type,
DisableResolveMessage: next.DisableResolveMessage,
Settings: out,
SecureSettings: next.SecureSettings,
}
return a.ReceiverIntegrationsFunc(gr, tmpl)
}
}
func (a AlertingConfiguration) DispatcherLimits() alerting.DispatcherLimits {
return &nilLimits{}
}
func (a AlertingConfiguration) InhibitRules() []alerting.InhibitRule {
return a.AlertmanagerConfig.InhibitRules
}
func (a AlertingConfiguration) MuteTimeIntervals() []alerting.MuteTimeInterval {
return a.AlertmanagerConfig.MuteTimeIntervals
}
func (a AlertingConfiguration) ReceiverIntegrations() (map[string][]*alerting.Integration, error) {
return a.IntegrationsFunc(a.AlertmanagerConfig.Receivers, a.AlertmanagerTemplates)
}
func (a AlertingConfiguration) RoutingTree() *alerting.Route {
return a.AlertmanagerConfig.Route.AsAMRoute()
}
func (a AlertingConfiguration) Templates() *alerting.Template {
return a.AlertmanagerTemplates
}
func (a AlertingConfiguration) Hash() [16]byte {
return md5.Sum(a.RawAlertmanagerConfig)
}
func (a AlertingConfiguration) Raw() []byte {
return a.RawAlertmanagerConfig
}