2015-10-11 15:24:49 +00:00
|
|
|
// Copyright 2015 Prometheus Team
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2015-10-11 11:32:24 +00:00
|
|
|
package template
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2015-11-26 17:19:46 +00:00
|
|
|
"net/url"
|
2015-11-25 14:49:26 +00:00
|
|
|
"sort"
|
|
|
|
"strings"
|
2015-10-11 11:32:24 +00:00
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
tmplhtml "html/template"
|
|
|
|
tmpltext "text/template"
|
2015-11-25 14:49:26 +00:00
|
|
|
|
|
|
|
"github.com/prometheus/common/model"
|
|
|
|
|
|
|
|
"github.com/prometheus/alertmanager/types"
|
2015-10-11 11:32:24 +00:00
|
|
|
)
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// Template bundles a text and a html template instance.
|
2015-10-11 11:32:24 +00:00
|
|
|
type Template struct {
|
2015-11-20 14:10:38 +00:00
|
|
|
text *tmpltext.Template
|
|
|
|
html *tmplhtml.Template
|
2015-11-26 17:19:46 +00:00
|
|
|
|
|
|
|
ExternalURL *url.URL
|
2015-10-11 11:32:24 +00:00
|
|
|
}
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// FromGlobs calls ParseGlob on all path globs provided and returns the
|
|
|
|
// resulting Template.
|
2015-10-11 14:54:39 +00:00
|
|
|
func FromGlobs(paths ...string) (*Template, error) {
|
|
|
|
t := &Template{
|
2015-11-20 14:10:38 +00:00
|
|
|
text: tmpltext.New("").Option("missingkey=zero"),
|
|
|
|
html: tmplhtml.New("").Option("missingkey=zero"),
|
2015-10-11 14:54:39 +00:00
|
|
|
}
|
|
|
|
var err error
|
|
|
|
|
2015-11-25 14:49:26 +00:00
|
|
|
t.text = t.text.Funcs(tmpltext.FuncMap(DefaultFuncs))
|
|
|
|
t.html = t.html.Funcs(tmplhtml.FuncMap(DefaultFuncs))
|
|
|
|
|
2015-10-11 14:54:39 +00:00
|
|
|
for _, tp := range paths {
|
|
|
|
if t.text, err = t.text.ParseGlob(tp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if t.html, err = t.html.ParseGlob(tp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// ExecuteTextString needs a meaningful doc comment (TODO(fabxc)).
|
2015-10-29 13:34:24 +00:00
|
|
|
func (t *Template) ExecuteTextString(text string, data interface{}) (string, error) {
|
|
|
|
tmpl, err := t.text.Clone()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
tmpl, err = tmpl.New("").Parse(text)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2015-10-11 11:32:24 +00:00
|
|
|
var buf bytes.Buffer
|
2015-10-29 13:34:24 +00:00
|
|
|
err = tmpl.Execute(&buf, data)
|
2015-10-11 11:32:24 +00:00
|
|
|
return buf.String(), err
|
|
|
|
}
|
|
|
|
|
2015-11-20 14:10:38 +00:00
|
|
|
// ExecuteHTMLString needs a meaningful doc comment (TODO(fabxc)).
|
2015-10-29 13:34:24 +00:00
|
|
|
func (t *Template) ExecuteHTMLString(html string, data interface{}) (string, error) {
|
|
|
|
tmpl, err := t.html.Clone()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
tmpl, err = tmpl.New("").Parse(html)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2015-10-11 11:32:24 +00:00
|
|
|
var buf bytes.Buffer
|
2015-10-29 13:34:24 +00:00
|
|
|
err = tmpl.Execute(&buf, data)
|
2015-10-11 11:32:24 +00:00
|
|
|
return buf.String(), err
|
|
|
|
}
|
2015-11-25 14:49:26 +00:00
|
|
|
|
|
|
|
type FuncMap map[string]interface{}
|
|
|
|
|
|
|
|
var DefaultFuncs = FuncMap{
|
|
|
|
"toUpper": strings.ToUpper,
|
|
|
|
"toLower": strings.ToLower,
|
2015-11-28 10:16:13 +00:00
|
|
|
"title": strings.Title,
|
2015-11-30 17:00:38 +00:00
|
|
|
// join is equal to strings.Join but inverts the argument order
|
|
|
|
// for easier pipelining in templates.
|
|
|
|
"join": func(sep string, s []string) string {
|
|
|
|
return strings.Join(s, sep)
|
2015-11-25 14:49:26 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pair is a key/value string pair.
|
|
|
|
type Pair struct {
|
|
|
|
Name, Value string
|
|
|
|
}
|
|
|
|
|
2015-11-30 17:00:38 +00:00
|
|
|
// Pairs is a list of key/value string pairs.
|
|
|
|
type Pairs []Pair
|
|
|
|
|
|
|
|
// Names returns a list of names of the pairs.
|
|
|
|
func (ps Pairs) Names() []string {
|
|
|
|
ns := make([]string, 0, len(ps))
|
|
|
|
for _, p := range ps {
|
|
|
|
ns = append(ns, p.Name)
|
|
|
|
}
|
|
|
|
return ns
|
|
|
|
}
|
|
|
|
|
|
|
|
// Values returns a list of values of the pairs.
|
|
|
|
func (ps Pairs) Values() []string {
|
|
|
|
vs := make([]string, 0, len(ps))
|
|
|
|
for _, p := range ps {
|
|
|
|
vs = append(vs, p.Value)
|
|
|
|
}
|
|
|
|
return vs
|
|
|
|
}
|
|
|
|
|
|
|
|
// KV is a set of key/value string pairs.
|
|
|
|
type KV map[string]string
|
|
|
|
|
|
|
|
// SortedPairs returns a sorted list of key/value pairs.
|
|
|
|
func (kv KV) SortedPairs() Pairs {
|
|
|
|
var (
|
|
|
|
pairs = make([]Pair, 0, len(kv))
|
|
|
|
keys = make([]string, 0, len(kv))
|
|
|
|
sortStart = 0
|
|
|
|
)
|
|
|
|
for k := range kv {
|
|
|
|
if k == string(model.AlertNameLabel) {
|
|
|
|
keys = append([]string{k}, keys...)
|
|
|
|
sortStart = 1
|
|
|
|
} else {
|
|
|
|
keys = append(keys, k)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sort.Strings(keys[sortStart:])
|
|
|
|
|
|
|
|
for _, k := range keys {
|
|
|
|
pairs = append(pairs, Pair{k, kv[k]})
|
|
|
|
}
|
|
|
|
return pairs
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove returns a copy of the key/value set without the given keys.
|
|
|
|
func (kv KV) Remove(keys []string) KV {
|
|
|
|
keySet := make(map[string]struct{}, len(keys))
|
|
|
|
for _, k := range keys {
|
|
|
|
keySet[k] = struct{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
res := KV{}
|
|
|
|
for k, v := range kv {
|
|
|
|
if _, ok := keySet[k]; !ok {
|
|
|
|
res[k] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
// Names returns the names of the label names in the LabelSet.
|
|
|
|
func (kv KV) Names() []string {
|
|
|
|
return kv.SortedPairs().Names()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Values returns a list of the values in the LabelSet.
|
|
|
|
func (kv KV) Values() []string {
|
|
|
|
return kv.SortedPairs().Values()
|
|
|
|
}
|
|
|
|
|
2015-11-25 14:49:26 +00:00
|
|
|
// Data is the data passed to notification templates.
|
|
|
|
// End-users should not be exposed to Go's type system,
|
|
|
|
// as this will confuse them and prevent simple things like
|
|
|
|
// simple equality checks to fail. Map everything to float64/string.
|
|
|
|
type Data struct {
|
2015-11-26 17:19:46 +00:00
|
|
|
Receiver string
|
|
|
|
Status string
|
2015-11-30 17:00:38 +00:00
|
|
|
Alerts Alerts
|
2015-11-25 14:49:26 +00:00
|
|
|
|
2015-11-30 17:00:38 +00:00
|
|
|
GroupLabels KV
|
|
|
|
CommonLabels KV
|
|
|
|
CommonAnnotations KV
|
2015-11-25 14:49:26 +00:00
|
|
|
|
|
|
|
ExternalURL string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Alert holds one alert for notification templates.
|
|
|
|
type Alert struct {
|
|
|
|
Status string
|
2015-11-30 17:00:38 +00:00
|
|
|
Labels KV
|
|
|
|
Annotations KV
|
|
|
|
}
|
|
|
|
|
|
|
|
// Alerts is a list of Alert objects.
|
|
|
|
type Alerts []Alert
|
|
|
|
|
|
|
|
// Firing returns the subset of alerts that are firing.
|
|
|
|
func (as Alerts) Firing() []Alert {
|
|
|
|
res := []Alert{}
|
|
|
|
for _, a := range as {
|
|
|
|
if a.Status == string(model.AlertFiring) {
|
|
|
|
res = append(res, a)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resolved returns the subset of alerts that are resolved.
|
|
|
|
func (as Alerts) Resolved() []Alert {
|
|
|
|
res := []Alert{}
|
|
|
|
for _, a := range as {
|
|
|
|
if a.Status == string(model.AlertResolved) {
|
|
|
|
res = append(res, a)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res
|
2015-11-25 14:49:26 +00:00
|
|
|
}
|
|
|
|
|
2015-11-26 17:19:46 +00:00
|
|
|
// Data assembles data for template expansion.
|
|
|
|
func (t *Template) Data(recv string, groupLabels model.LabelSet, as ...*types.Alert) *Data {
|
2015-11-25 14:49:26 +00:00
|
|
|
alerts := types.Alerts(as...)
|
|
|
|
|
|
|
|
data := &Data{
|
2015-11-26 17:19:46 +00:00
|
|
|
Receiver: strings.SplitN(recv, "/", 2)[0],
|
2015-11-25 14:49:26 +00:00
|
|
|
Status: string(alerts.Status()),
|
2015-11-30 17:00:38 +00:00
|
|
|
Alerts: make(Alerts, 0, len(alerts)),
|
|
|
|
GroupLabels: KV{},
|
|
|
|
CommonLabels: KV{},
|
|
|
|
CommonAnnotations: KV{},
|
2015-11-26 17:19:46 +00:00
|
|
|
ExternalURL: t.ExternalURL.String(),
|
2015-11-25 14:49:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, a := range alerts {
|
|
|
|
alert := Alert{
|
|
|
|
Status: string(a.Status()),
|
2015-11-30 17:00:38 +00:00
|
|
|
Labels: make(KV, len(a.Labels)),
|
|
|
|
Annotations: make(KV, len(a.Annotations)),
|
2015-11-25 14:49:26 +00:00
|
|
|
}
|
|
|
|
for k, v := range a.Labels {
|
|
|
|
alert.Labels[string(k)] = string(v)
|
|
|
|
}
|
|
|
|
for k, v := range a.Annotations {
|
|
|
|
alert.Annotations[string(k)] = string(v)
|
|
|
|
}
|
|
|
|
data.Alerts = append(data.Alerts, alert)
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range groupLabels {
|
|
|
|
data.GroupLabels[string(k)] = string(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(alerts) >= 1 {
|
|
|
|
var (
|
|
|
|
commonLabels = alerts[0].Labels.Clone()
|
|
|
|
commonAnnotations = alerts[0].Annotations.Clone()
|
|
|
|
)
|
|
|
|
for _, a := range alerts[1:] {
|
|
|
|
for ln, lv := range commonLabels {
|
|
|
|
if a.Labels[ln] != lv {
|
|
|
|
delete(commonLabels, ln)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for an, av := range commonAnnotations {
|
|
|
|
if a.Annotations[an] != av {
|
|
|
|
delete(commonAnnotations, an)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for k, v := range commonLabels {
|
|
|
|
data.CommonLabels[string(k)] = string(v)
|
|
|
|
}
|
|
|
|
for k, v := range commonAnnotations {
|
|
|
|
data.CommonAnnotations[string(k)] = string(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return data
|
|
|
|
}
|