Skip to content

Commit

Permalink
feat: add new expressionDoesNotUseMetrics validator
Browse files Browse the repository at this point in the history
Signed-off-by: Martin Chodur <[email protected]>
  • Loading branch information
FUSAKLA committed Dec 4, 2023
1 parent b5ad520 commit b2e0d43
Show file tree
Hide file tree
Showing 5 changed files with 116 additions and 37 deletions.
81 changes: 45 additions & 36 deletions docs/validations.md
Original file line number Diff line number Diff line change
@@ -1,37 +1,36 @@
# Supported validations

- [Labels](#labels)
- [hasLabels](./validations.md#haslabels)
- [hasAnyOfLabels](./validations.md#hasanyoflabels)
- [doesNotHaveLabels](./validations.md#doesnothavelabels)
- [labelMatchesRegexp](./validations.md#labelmatchesregexp)
- [labelHasAllowedValue](./validations.md#labelhasallowedvalue)
- [expressionDoesNotUseLabels](./validations.md#expressiondoesnotuselabels)
- [nonEmptyLabels](./validations.md#nonemptylabels)
- [exclusiveLabels](./validations.md#exclusivelabels)
-
- [Annotations](#annotations)
- [hasAnnotations](./validations.md#hasannotations)
- [doesNotHaveAnnotations](./validations.md#doesnothaveannotations)
- [hasAnyOfAnnotations](./validations.md#hasanyofannotations)
- [annotationMatchesRegexp](./validations.md#annotationmatchesregexp)
- [annotationHasAllowedValue](./validations.md#annotationhasallowedvalue)
- [annotationIsValidURL](./validations.md#annotationisvalidurl)
- [annotationIsValidPromQL](./validations.md#annotationisvalidpromql)
- [validateAnnotationTemplates](./validations.md#validateannotationtemplates)
-
- [PromQL expression](#promql-expression)
- [expressionDoesNotUseOlderDataThan](./validations.md#expressiondoesnotuseolderdatathan)
- [expressionDoesNotUseRangeShorterThan](./validations.md#expressiondoesnotuserangeshorterthan)
- [expressionDoesNotUseIrate](./validations.md#expressiondoesnotuseirate)
- [validFunctionsOnCounters](./validations.md#validfunctionsoncounters)
- [rateBeforeAggregation](./validations.md#ratebeforeaggregation)
- [expressionCanBeEvaluated](./validations.md#expressioncanbeevaluated)
- [expressionUsesExistingLabels](./validations.md#expressionusesexistinglabels)
- [expressionSelectorsMatchesAnything](./validations.md#expressionselectorsmatchesanything)
-
- [Alert](#alert)
- [forIsNotLongerThan](./validations.md#forisnotlongerthan)
- [Supported validations](#supported-validations)
- [Labels](#labels)
- [`hasLabels`](#haslabels)
- [`doesNotHaveLabels`](#doesnothavelabels)
- [`hasAnyOfLabels`](#hasanyoflabels)
- [`labelMatchesRegexp`](#labelmatchesregexp)
- [`labelHasAllowedValue`](#labelhasallowedvalue)
- [`nonEmptyLabels`](#nonemptylabels)
- [`exclusiveLabels`](#exclusivelabels)
- [Annotations](#annotations)
- [`hasAnnotations`](#hasannotations)
- [`doesNotHaveAnnotations`](#doesnothaveannotations)
- [`hasAnyOfAnnotations`](#hasanyofannotations)
- [`annotationMatchesRegexp`](#annotationmatchesregexp)
- [`annotationHasAllowedValue`](#annotationhasallowedvalue)
- [`annotationIsValidURL`](#annotationisvalidurl)
- [`annotationIsValidPromQL`](#annotationisvalidpromql)
- [`validateAnnotationTemplates`](#validateannotationtemplates)
- [PromQL expression](#promql-expression)
- [`expressionDoesNotUseMetrics`](#expressiondoesnotusemetrics)
- [`expressionDoesNotUseLabels`](#expressiondoesnotuselabels)
- [`expressionDoesNotUseOlderDataThan`](#expressiondoesnotuseolderdatathan)
- [`expressionDoesNotUseRangeShorterThan`](#expressiondoesnotuserangeshorterthan)
- [`expressionDoesNotUseIrate`](#expressiondoesnotuseirate)
- [`validFunctionsOnCounters`](#validfunctionsoncounters)
- [`rateBeforeAggregation`](#ratebeforeaggregation)
- [`expressionCanBeEvaluated`](#expressioncanbeevaluated)
- [`expressionUsesExistingLabels`](#expressionusesexistinglabels)
- [`expressionSelectorsMatchesAnything`](#expressionselectorsmatchesanything)
- [Alert](#alert)
- [`forIsNotLongerThan`](#forisnotlongerthan)

## Labels

Expand Down Expand Up @@ -110,7 +109,7 @@ params:
firstLabel: "severity"
firstLabelValue: "critical" # Optional, if not set, only presence of the label excludes the second label
secondLabel: "page"
secondLabelValue: "true" # Optional, if set, fails only if also the second label value matches
secondLabelValue: "true" # Optional, if set, fails only if also the second label value matches
```

## Annotations
Expand Down Expand Up @@ -169,7 +168,7 @@ params:

Fails if annotation value is not a valid URL. If `resolveURL` is enabled, tries to make an HTTP request to the specified
URL and fails if the request does not succeed or returns 404 HTTP status code.
> It's common practise to link a playbook with guide how to solve the alert in the alert itself.
> It's common practice to link a playbook with guide how to solve the alert in the alert itself.
> This way you can verify it's a working URL and possibly if it really exists.

```yaml
Expand All @@ -193,6 +192,16 @@ Fails if the annotation contains invalid Go template.

## PromQL expression

### `expressionDoesNotUseMetrics`

Fails if the rule expression uses metrics matching any of the metric name fully anchored(will be surrounded by `^...$`) regexps.
> If you want to avoid using some metrics in the rules, you can use this validation to make sure it won't happen.

```yaml
params:
metricNameRegexps: [ "foo_bar.*", "foo_baz" ]
```

### `expressionDoesNotUseLabels`

Fails if the rule uses any of specified labels in its `expr` label matchers, aggregations or joins.
Expand All @@ -206,7 +215,7 @@ params:

### `expressionDoesNotUseOlderDataThan`

Fails if the rule `expr` uses older data than specified limit in Prometheus duration syntax. Checks even in subqueries
Fails if the rule `expr` uses older data than specified limit in Prometheus duration syntax. Checks even in sub-queries
and offsets.
> Useful to avoid writing queries which expects longer data retention than the Prometheus actually has.

Expand Down Expand Up @@ -243,7 +252,7 @@ Fails if aggregation function is used before the `rate` or `increase` functions.
> Queries live prometheus instance, requires the `prometheus` config to be set.

This validation runs the expression against the actual Prometheus instance and checks if it ends with error.
Possibly you can set maximum allowed query execution time and maximum number of resulting timeseries.
Possibly you can set maximum allowed query execution time and maximum number of resulting time series.

```yaml
params:
Expand Down
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,6 @@ require (
github.com/creasty/defaults v1.7.0
github.com/prometheus/client_golang v1.17.0
github.com/prometheus/common v0.45.0
github.com/prometheus/prometheus v0.48.0
github.com/sirupsen/logrus v1.9.3
gopkg.in/alecthomas/kingpin.v2 v2.2.6
gopkg.in/yaml.v3 v3.0.1
Expand Down Expand Up @@ -55,6 +54,7 @@ require (
github.com/prometheus/client_model v0.5.0 // indirect
github.com/prometheus/common/sigv4 v0.1.0 // indirect
github.com/prometheus/procfs v0.12.0 // indirect
github.com/prometheus/prometheus v0.48.0
github.com/stretchr/testify v1.8.4 // indirect
go.opentelemetry.io/otel v1.21.0 // indirect
go.opentelemetry.io/otel/metric v1.21.0 // indirect
Expand Down
1 change: 1 addition & 0 deletions pkg/validator/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ var registeredValidators = map[string]validatorCreator{
"expressionDoesNotUseLabels": newExpressionDoesNotUseLabels,
"expressionDoesNotUseOlderDataThan": newExpressionDoesNotUseOlderDataThan,
"expressionDoesNotUseRangeShorterThan": newExpressionDoesNotUseRangeShorterThan,
"expressionDoesNotUseMetrics": newExpressionDoesNotUseMetrics,
"annotationIsValidPromQL": newAnnotationIsValidPromQL,
"validateAnnotationTemplates": newValidateAnnotationTemplates,
"forIsNotLongerThan": newForIsNotLongerThan,
Expand Down
61 changes: 61 additions & 0 deletions pkg/validator/expression.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import (

"github.com/fusakla/promruval/v2/pkg/prometheus"
"github.com/prometheus/common/model"
"github.com/prometheus/prometheus/model/labels"
"github.com/prometheus/prometheus/model/rulefmt"
"github.com/prometheus/prometheus/promql/parser"
log "github.com/sirupsen/logrus"
Expand Down Expand Up @@ -386,3 +387,63 @@ func (h expressionSelectorsMatchesAnything) Validate(rule rulefmt.Rule, promethe
}
return errs
}

func newExpressionDoesNotUseMetrics(paramsConfig yaml.Node) (Validator, error) {
params := struct {
MetricNameRegexps []string `yaml:"metricNameRegexps"`
}{}
if err := paramsConfig.Decode(&params); err != nil {
return nil, err
}
v := expressionDoesNotUseMetrics{}
for _, r := range params.MetricNameRegexps {
compiled, err := regexp.Compile("^" + r + "$")
if err != nil {
return nil, err
}
v.metricNameRegexps = append(v.metricNameRegexps, compiled)
}
return &v, nil
}

type expressionDoesNotUseMetrics struct {
metricNameRegexps []*regexp.Regexp
}

func (h expressionDoesNotUseMetrics) String() string {
return "expression does not use any of these metrics regexps: " + strings.Join(func() []string {
var res []string
for _, r := range h.metricNameRegexps {
res = append(res, r.String())
}
return res
}(), ",")
}

func (h expressionDoesNotUseMetrics) Validate(rule rulefmt.Rule, _ *prometheus.Client) []error {
expr, err := parser.ParseExpr(rule.Expr)
if err != nil {
return []error{fmt.Errorf("failed to parse expression `%s`: %s", rule.Expr, err)}
}
var errs []error
parser.Inspect(expr, func(n parser.Node, ns []parser.Node) error {
switch v := n.(type) {
case *parser.VectorSelector:
// Do not check the VectorSelector.Name since it is automatically parsed into the __name__ LabelMatcher
for _, l := range v.LabelMatchers {
// We care just for the special __name__ label containing name of the metric
// and since we cannot match regexp on regexp.. lets just check for equality
if l.Name != "__name__" || l.Type != labels.MatchEqual {
continue
}
for _, r := range h.metricNameRegexps {
if r.MatchString(l.Value) {
errs = append(errs, fmt.Errorf("expression uses metric `%s` which is forbidden", l.Value))
}
}
}
}
return nil
})
return errs
}
8 changes: 8 additions & 0 deletions pkg/validator/validator_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,14 @@ var testCases = []struct {
// expressionUsesExistingLabels
{name: "labelsExists", validator: expressionUsesExistingLabels{}, promClient: prometheus.NewClientMock([]string{"__name__", "foo"}, 0, false, false), rule: rulefmt.Rule{Expr: `up{foo="bar"}`}, expectedErrors: 0},
{name: "labelsDoesNotExist", validator: expressionUsesExistingLabels{}, promClient: prometheus.NewClientMock([]string{"__name__"}, 0, false, false), rule: rulefmt.Rule{Expr: `up{foo="bar"}`}, expectedErrors: 1},

// expressionDoesNotUseMetrics
{name: "emptyList", validator: expressionDoesNotUseMetrics{metricNameRegexps: []*regexp.Regexp{}}, rule: rulefmt.Rule{Expr: `up{foo="bar"}`}, expectedErrors: 0},
{name: "doesNotUseForbiddenMetric", validator: expressionDoesNotUseMetrics{metricNameRegexps: []*regexp.Regexp{regexp.MustCompile(`foo_bar`)}}, rule: rulefmt.Rule{Expr: `up{foo="bar"}`}, expectedErrors: 0},
{name: "usesForbiddenMetric", validator: expressionDoesNotUseMetrics{metricNameRegexps: []*regexp.Regexp{regexp.MustCompile(`foo_bar`)}}, rule: rulefmt.Rule{Expr: `foo_bar{foo="bar"}`}, expectedErrors: 1},
{name: "usesTwoOfThreeForbiddenMetrics", validator: expressionDoesNotUseMetrics{metricNameRegexps: []*regexp.Regexp{regexp.MustCompile(`foo_bar`), regexp.MustCompile(`foo_baz`), regexp.MustCompile(`^foo$`)}}, rule: rulefmt.Rule{Expr: `foo_baz{foo="bar"} and foo_bar`}, expectedErrors: 2},
{name: "usesMetricMatchingRegexp", validator: expressionDoesNotUseMetrics{metricNameRegexps: []*regexp.Regexp{regexp.MustCompile(`foo_bar.*`)}}, rule: rulefmt.Rule{Expr: `foo_baz_baz{foo="bar"} and foo_bar`}, expectedErrors: 1},
{name: "regexpIsFullyAnchored", validator: expressionDoesNotUseMetrics{metricNameRegexps: []*regexp.Regexp{regexp.MustCompile(`^foo_bar$`)}}, rule: rulefmt.Rule{Expr: `foo_baz_baz{foo="bar"} and foo_bar`}, expectedErrors: 1},
}

func Test(t *testing.T) {
Expand Down

0 comments on commit b2e0d43

Please sign in to comment.