mirror of
https://github.com/grafana/grafana.git
synced 2025-07-31 10:02:33 +08:00

This commit renames "Message templates" to "Notification templates" in the user interface as it suggests that these templates cannot be used to template anything other than the message. However, message templates are much more general and can be used to template other fields too such as the subject of an email, or the title of a Slack message.
145 lines
3.6 KiB
Go
145 lines
3.6 KiB
Go
package definitions
|
|
|
|
import (
|
|
"fmt"
|
|
tmplhtml "html/template"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/prometheus/alertmanager/template"
|
|
"github.com/prometheus/common/model"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
// Validate normalizes a possibly nested Route r, and returns errors if r is invalid.
|
|
func (r *Route) validateChild() error {
|
|
r.GroupBy = nil
|
|
r.GroupByAll = false
|
|
for _, l := range r.GroupByStr {
|
|
if l == "..." {
|
|
r.GroupByAll = true
|
|
} else {
|
|
r.GroupBy = append(r.GroupBy, model.LabelName(l))
|
|
}
|
|
}
|
|
|
|
if len(r.GroupBy) > 0 && r.GroupByAll {
|
|
return fmt.Errorf("cannot have wildcard group_by (`...`) and other other labels at the same time")
|
|
}
|
|
|
|
groupBy := map[model.LabelName]struct{}{}
|
|
|
|
for _, ln := range r.GroupBy {
|
|
if _, ok := groupBy[ln]; ok {
|
|
return fmt.Errorf("duplicated label %q in group_by, %s %s", ln, r.Receiver, r.GroupBy)
|
|
}
|
|
groupBy[ln] = struct{}{}
|
|
}
|
|
|
|
if r.GroupInterval != nil && time.Duration(*r.GroupInterval) == time.Duration(0) {
|
|
return fmt.Errorf("group_interval cannot be zero")
|
|
}
|
|
if r.RepeatInterval != nil && time.Duration(*r.RepeatInterval) == time.Duration(0) {
|
|
return fmt.Errorf("repeat_interval cannot be zero")
|
|
}
|
|
|
|
// Routes are a self-referential structure.
|
|
if r.Routes != nil {
|
|
for _, child := range r.Routes {
|
|
err := child.validateChild()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (t *NotificationTemplate) Validate() error {
|
|
if t.Name == "" {
|
|
return fmt.Errorf("template must have a name")
|
|
}
|
|
if t.Template == "" {
|
|
return fmt.Errorf("template must have content")
|
|
}
|
|
|
|
tmpl := tmplhtml.New("").Option("missingkey=zero")
|
|
tmpl.Funcs(tmplhtml.FuncMap(template.DefaultFuncs))
|
|
_, err := tmpl.Parse(t.Template)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid template: %w", err)
|
|
}
|
|
|
|
content := strings.TrimSpace(t.Template)
|
|
found, err := regexp.MatchString(`\{\{\s*define`, content)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to match regex: %w", err)
|
|
}
|
|
if !found {
|
|
lines := strings.Split(content, "\n")
|
|
for i, s := range lines {
|
|
lines[i] = " " + s
|
|
}
|
|
content = strings.Join(lines, "\n")
|
|
content = fmt.Sprintf("{{ define \"%s\" }}\n%s\n{{ end }}", t.Name, content)
|
|
}
|
|
t.Template = content
|
|
|
|
return nil
|
|
}
|
|
|
|
// Validate normalizes a Route r, and returns errors if r is an invalid root route. Root routes must satisfy a few additional conditions.
|
|
func (r *Route) Validate() error {
|
|
if len(r.Receiver) == 0 {
|
|
return fmt.Errorf("root route must specify a default receiver")
|
|
}
|
|
if len(r.Match) > 0 || len(r.MatchRE) > 0 {
|
|
return fmt.Errorf("root route must not have any matchers")
|
|
}
|
|
if len(r.MuteTimeIntervals) > 0 {
|
|
return fmt.Errorf("root route must not have any mute time intervals")
|
|
}
|
|
return r.validateChild()
|
|
}
|
|
|
|
func (r *Route) ValidateReceivers(receivers map[string]struct{}) error {
|
|
if _, exists := receivers[r.Receiver]; !exists {
|
|
return fmt.Errorf("receiver '%s' does not exist", r.Receiver)
|
|
}
|
|
for _, children := range r.Routes {
|
|
err := children.ValidateReceivers(receivers)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *Route) ValidateMuteTimes(muteTimes map[string]struct{}) error {
|
|
for _, name := range r.MuteTimeIntervals {
|
|
if _, exists := muteTimes[name]; !exists {
|
|
return fmt.Errorf("mute time interval '%s' does not exist", name)
|
|
}
|
|
}
|
|
for _, child := range r.Routes {
|
|
err := child.ValidateMuteTimes(muteTimes)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (mt *MuteTimeInterval) Validate() error {
|
|
s, err := yaml.Marshal(mt.MuteTimeInterval)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err = yaml.Unmarshal(s, &(mt.MuteTimeInterval)); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|