mirror of https://github.com/prometheus/prometheus
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
189 lines
5.1 KiB
189 lines
5.1 KiB
// Copyright 2017 The Prometheus Authors
|
|
// 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.
|
|
|
|
package rulefmt
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prometheus/common/model"
|
|
"github.com/prometheus/prometheus/promql"
|
|
yaml "gopkg.in/yaml.v2"
|
|
)
|
|
|
|
// Error represents semantical errors on parsing rule groups.
|
|
type Error struct {
|
|
Group string
|
|
Rule int
|
|
RuleName string
|
|
Err error
|
|
}
|
|
|
|
func (err *Error) Error() string {
|
|
return errors.Wrapf(err.Err, "group %q, rule %d, %q", err.Group, err.Rule, err.RuleName).Error()
|
|
}
|
|
|
|
// RuleGroups is a set of rule groups that are typically exposed in a file.
|
|
type RuleGroups struct {
|
|
Groups []RuleGroup `yaml:"groups"`
|
|
|
|
// Catches all undefined fields and must be empty after parsing.
|
|
XXX map[string]interface{} `yaml:",inline"`
|
|
}
|
|
|
|
// Validate validates all rules in the rule groups.
|
|
func (g *RuleGroups) Validate() (errs []error) {
|
|
set := map[string]struct{}{}
|
|
|
|
for _, g := range g.Groups {
|
|
if g.Name == "" {
|
|
errs = append(errs, errors.Errorf("Groupname should not be empty"))
|
|
}
|
|
|
|
if _, ok := set[g.Name]; ok {
|
|
errs = append(
|
|
errs,
|
|
errors.Errorf("groupname: \"%s\" is repeated in the same file", g.Name),
|
|
)
|
|
}
|
|
|
|
if err := checkOverflow(g.XXX, "rule_group"); err != nil {
|
|
errs = append(errs, errors.Wrapf(err, "Group: %s", g.Name))
|
|
}
|
|
|
|
set[g.Name] = struct{}{}
|
|
|
|
for i, r := range g.Rules {
|
|
for _, err := range r.Validate() {
|
|
var ruleName string
|
|
if r.Alert != "" {
|
|
ruleName = r.Alert
|
|
} else {
|
|
ruleName = r.Record
|
|
}
|
|
errs = append(errs, &Error{
|
|
Group: g.Name,
|
|
Rule: i,
|
|
RuleName: ruleName,
|
|
Err: err,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
if err := checkOverflow(g.XXX, "config_file"); err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
|
|
return errs
|
|
}
|
|
|
|
// RuleGroup is a list of sequentially evaluated recording and alerting rules.
|
|
type RuleGroup struct {
|
|
Name string `yaml:"name"`
|
|
Interval model.Duration `yaml:"interval,omitempty"`
|
|
Rules []Rule `yaml:"rules"`
|
|
|
|
// Catches all undefined fields and must be empty after parsing.
|
|
XXX map[string]interface{} `yaml:",inline"`
|
|
}
|
|
|
|
// Rule describes an alerting or recording rule.
|
|
type Rule struct {
|
|
Record string `yaml:"record,omitempty"`
|
|
Alert string `yaml:"alert,omitempty"`
|
|
Expr string `yaml:"expr"`
|
|
For model.Duration `yaml:"for,omitempty"`
|
|
Labels map[string]string `yaml:"labels,omitempty"`
|
|
Annotations map[string]string `yaml:"annotations,omitempty"`
|
|
|
|
// Catches all undefined fields and must be empty after parsing.
|
|
XXX map[string]interface{} `yaml:",inline"`
|
|
}
|
|
|
|
// Validate the rule and return a list of encountered errors.
|
|
func (r *Rule) Validate() (errs []error) {
|
|
if r.Record != "" && r.Alert != "" {
|
|
errs = append(errs, errors.Errorf("only one of 'record' and 'alert' must be set"))
|
|
}
|
|
if r.Record == "" && r.Alert == "" {
|
|
errs = append(errs, errors.Errorf("one of 'record' or 'alert' must be set"))
|
|
}
|
|
|
|
if r.Expr == "" {
|
|
errs = append(errs, errors.Errorf("field 'expr' must be set in rule"))
|
|
} else if _, err := promql.ParseExpr(r.Expr); err != nil {
|
|
errs = append(errs, errors.Errorf("could not parse expression: %s", err))
|
|
}
|
|
if r.Record != "" {
|
|
if len(r.Annotations) > 0 {
|
|
errs = append(errs, errors.Errorf("invalid field 'annotations' in recording rule"))
|
|
}
|
|
if r.For != 0 {
|
|
errs = append(errs, errors.Errorf("invalid field 'for' in recording rule"))
|
|
}
|
|
if !model.IsValidMetricName(model.LabelValue(r.Record)) {
|
|
errs = append(errs, errors.Errorf("invalid recording rule name: %s", r.Record))
|
|
}
|
|
}
|
|
|
|
for k, v := range r.Labels {
|
|
if !model.LabelName(k).IsValid() {
|
|
errs = append(errs, errors.Errorf("invalid label name: %s", k))
|
|
}
|
|
|
|
if !model.LabelValue(v).IsValid() {
|
|
errs = append(errs, errors.Errorf("invalid label value: %s", v))
|
|
}
|
|
}
|
|
|
|
for k := range r.Annotations {
|
|
if !model.LabelName(k).IsValid() {
|
|
errs = append(errs, errors.Errorf("invalid annotation name: %s", k))
|
|
}
|
|
}
|
|
|
|
if err := checkOverflow(r.XXX, "rule"); err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
|
|
return errs
|
|
}
|
|
|
|
func checkOverflow(m map[string]interface{}, ctx string) error {
|
|
if len(m) > 0 {
|
|
var keys []string
|
|
for k := range m {
|
|
keys = append(keys, k)
|
|
}
|
|
return fmt.Errorf("unknown fields in %s: %s", ctx, strings.Join(keys, ", "))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ParseFile parses the rule file and validates it.
|
|
func ParseFile(file string) (*RuleGroups, []error) {
|
|
b, err := ioutil.ReadFile(file)
|
|
if err != nil {
|
|
return nil, []error{err}
|
|
}
|
|
var groups RuleGroups
|
|
if err := yaml.Unmarshal(b, &groups); err != nil {
|
|
return nil, []error{err}
|
|
}
|
|
return &groups, groups.Validate()
|
|
}
|