alertmanager/route.go

167 lines
4.0 KiB
Go
Raw Normal View History

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-09-25 16:14:46 +00:00
package main
2015-07-01 15:56:53 +00:00
import (
"fmt"
"time"
"github.com/prometheus/common/model"
2015-09-25 16:14:46 +00:00
"github.com/prometheus/alertmanager/config"
"github.com/prometheus/alertmanager/types"
2015-07-01 15:56:53 +00:00
)
2015-07-04 10:51:47 +00:00
var DefaultRouteOpts = RouteOpts{
GroupWait: 20 * time.Second,
GroupInterval: 5 * time.Minute,
RepeatInterval: 1 * time.Hour,
SendResolved: true,
2015-07-04 10:51:47 +00:00
}
type Routes []*Route
func (rs Routes) Match(lset model.LabelSet) []*RouteOpts {
fakeParent := &Route{
2015-07-04 10:51:47 +00:00
Routes: rs,
RouteOpts: DefaultRouteOpts,
}
return fakeParent.Match(lset)
}
2015-07-01 15:56:53 +00:00
// A Route is a node that contains definitions of how to handle alerts.
type Route struct {
// The configuration parameters for matches of this route.
RouteOpts RouteOpts
// Equality or regex matchers an alert has to fulfill to match
// this route.
2015-09-25 16:14:46 +00:00
Matchers types.Matchers
2015-07-01 15:56:53 +00:00
// If true, an alert matches further routes on the same level.
Continue bool
// Children routes of this route.
Routes Routes
2015-07-01 15:56:53 +00:00
}
func NewRoute(cr *config.Route, parent *RouteOpts) *Route {
2015-09-25 16:14:46 +00:00
groupBy := map[model.LabelName]struct{}{}
for _, ln := range cr.GroupBy {
groupBy[ln] = struct{}{}
}
// Create default and overwrite with configured settings.
opts := *parent
opts.GroupBy = groupBy
if cr.SendTo != "" {
opts.SendTo = cr.SendTo
2015-09-25 16:14:46 +00:00
}
if cr.GroupWait != nil {
2015-09-25 16:14:46 +00:00
opts.GroupWait = time.Duration(*cr.GroupWait)
}
if cr.GroupInterval != nil {
opts.GroupInterval = time.Duration(*cr.GroupInterval)
2015-09-25 16:14:46 +00:00
}
if cr.RepeatInterval != nil {
opts.RepeatInterval = time.Duration(*cr.RepeatInterval)
}
if cr.SendResolved != nil {
opts.SendResolved = *cr.SendResolved
}
2015-09-25 16:14:46 +00:00
// Build matchers.
2015-09-25 16:14:46 +00:00
var matchers types.Matchers
for ln, lv := range cr.Match {
matchers = append(matchers, types.NewMatcher(model.LabelName(ln), lv))
}
for ln, lv := range cr.MatchRE {
m, err := types.NewRegexMatcher(model.LabelName(ln), lv.String())
if err != nil {
// Must have been sanitized during config validation.
panic(err)
}
matchers = append(matchers, m)
}
return &Route{
RouteOpts: opts,
Matchers: matchers,
Continue: cr.Continue,
Routes: NewRoutes(cr.Routes, &opts),
2015-09-25 16:14:46 +00:00
}
}
func NewRoutes(croutes []*config.Route, parent *RouteOpts) Routes {
if parent == nil {
parent = &DefaultRouteOpts
}
2015-09-25 16:14:46 +00:00
res := Routes{}
for _, cr := range croutes {
res = append(res, NewRoute(cr, parent))
2015-09-25 16:14:46 +00:00
}
return res
}
2015-07-01 15:56:53 +00:00
// Match does a depth-first left-to-right search through the route tree
// and returns the flattened configuration for the reached node.
func (r *Route) Match(lset model.LabelSet) []*RouteOpts {
if !r.Matchers.Match(lset) {
2015-07-01 15:56:53 +00:00
return nil
}
var all []*RouteOpts
2015-07-01 15:56:53 +00:00
for _, cr := range r.Routes {
matches := cr.Match(lset)
all = append(all, matches...)
2015-07-01 15:56:53 +00:00
2015-07-01 16:24:08 +00:00
if matches != nil && !cr.Continue {
2015-07-01 15:56:53 +00:00
break
}
}
if len(all) == 0 {
all = append(all, &r.RouteOpts)
2015-07-01 15:56:53 +00:00
}
return all
2015-07-01 15:56:53 +00:00
}
type RouteOpts struct {
// The identifier of the associated notification configuration
SendTo string
SendResolved bool
2015-07-01 15:56:53 +00:00
// What labels to group alerts by for notifications.
GroupBy map[model.LabelName]struct{}
// How long to wait to group matching alerts before sending
// a notificaiton
GroupWait time.Duration
GroupInterval time.Duration
RepeatInterval time.Duration
2015-07-01 15:56:53 +00:00
}
func (ro *RouteOpts) String() string {
var labels []model.LabelName
for ln := range ro.GroupBy {
labels = append(labels, ln)
}
return fmt.Sprintf("<RouteOpts send_to:%q group_by:%q timers:%q|%q>", ro.SendTo, labels, ro.GroupWait, ro.GroupInterval)
2015-07-01 15:56:53 +00:00
}