2023-10-13 13:54:33 +02:00
|
|
|
package rule
|
2020-06-01 12:46:37 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
2021-05-15 12:25:57 +02:00
|
|
|
"strings"
|
2020-06-01 12:46:37 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/config"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/datasource"
|
2022-02-02 13:11:41 +01:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/utils"
|
2024-11-08 15:47:14 +01:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logger"
|
2024-10-29 16:30:39 +01:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logstorage"
|
2020-06-01 12:46:37 +02:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal"
|
|
|
|
)
|
|
|
|
|
|
|
|
// RecordingRule is a Rule that supposed
|
|
|
|
// to evaluate configured Expression and
|
|
|
|
// return TimeSeries as result.
|
|
|
|
type RecordingRule struct {
|
2023-12-04 16:40:33 +01:00
|
|
|
Type config.Type
|
|
|
|
RuleID uint64
|
|
|
|
Name string
|
|
|
|
Expr string
|
|
|
|
Labels map[string]string
|
|
|
|
GroupID uint64
|
|
|
|
GroupName string
|
|
|
|
File string
|
2020-06-01 12:46:37 +02:00
|
|
|
|
2021-04-28 22:41:15 +02:00
|
|
|
q datasource.Querier
|
|
|
|
|
2022-09-14 14:04:24 +02:00
|
|
|
// state stores recent state changes
|
|
|
|
// during evaluations
|
|
|
|
state *ruleState
|
app/vmalert: extend metrics set exported by `vmalert` #573 (#654)
* app/vmalert: extend metrics set exported by `vmalert` #573
New metrics were added to improve observability:
+ vmalert_alerts_pending{alertname, group} - number of pending alerts per group
per alert;
+ vmalert_alerts_acitve{alertname, group} - number of active alerts per group
per alert;
+ vmalert_alerts_error{alertname, group} - is 1 if alertname ended up with error
during prev execution, is 0 if no errors happened;
+ vmalert_recording_rules_error{recording, group} - is 1 if recording rule
ended up with error during prev execution, is 0 if no errors happened;
* vmalert_iteration_total{group, file} - now contains group and file name labels.
This should improve control over specific groups;
* vmalert_iteration_duration_seconds{group, file} - now contains group and file name labels. This should improve control over specific groups;
Some collisions for alerts and recording rules are possible, because neither
group name nor alert/recording rule name are unique for compatibility reasons.
Commit contains list of TODOs for Unregistering metrics since groups and rules
are ephemeral and could be removed without application restart. In order to
unlock Unregistering feature corresponding PR was filed - https://github.com/VictoriaMetrics/metrics/pull/13
* app/vmalert: extend metrics set exported by `vmalert` #573
The changes are following:
* add an ID label to rules metrics, since `name` collisions within one group is
a common case - see the k8s example alerts;
* supports metrics unregistering on rule updates. Consider the case when one rule
was added or removed from the group, or the whole group was added or removed.
The change depends on https://github.com/VictoriaMetrics/metrics/pull/16
where race condition for Unregister method was fixed.
2020-08-09 08:41:29 +02:00
|
|
|
|
|
|
|
metrics *recordingRuleMetrics
|
|
|
|
}
|
|
|
|
|
|
|
|
type recordingRuleMetrics struct {
|
2023-12-06 19:39:35 +01:00
|
|
|
errors *utils.Counter
|
2022-02-02 13:11:41 +01:00
|
|
|
samples *utils.Gauge
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// String implements Stringer interface
|
|
|
|
func (rr *RecordingRule) String() string {
|
|
|
|
return rr.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
// ID returns unique Rule ID
|
|
|
|
// within the parent Group.
|
|
|
|
func (rr *RecordingRule) ID() uint64 {
|
2020-06-15 21:15:47 +02:00
|
|
|
return rr.RuleID
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
// NewRecordingRule creates a new RecordingRule
|
|
|
|
func NewRecordingRule(qb datasource.QuerierBuilder, group *Group, cfg config.Rule) *RecordingRule {
|
app/vmalert: extend metrics set exported by `vmalert` #573 (#654)
* app/vmalert: extend metrics set exported by `vmalert` #573
New metrics were added to improve observability:
+ vmalert_alerts_pending{alertname, group} - number of pending alerts per group
per alert;
+ vmalert_alerts_acitve{alertname, group} - number of active alerts per group
per alert;
+ vmalert_alerts_error{alertname, group} - is 1 if alertname ended up with error
during prev execution, is 0 if no errors happened;
+ vmalert_recording_rules_error{recording, group} - is 1 if recording rule
ended up with error during prev execution, is 0 if no errors happened;
* vmalert_iteration_total{group, file} - now contains group and file name labels.
This should improve control over specific groups;
* vmalert_iteration_duration_seconds{group, file} - now contains group and file name labels. This should improve control over specific groups;
Some collisions for alerts and recording rules are possible, because neither
group name nor alert/recording rule name are unique for compatibility reasons.
Commit contains list of TODOs for Unregistering metrics since groups and rules
are ephemeral and could be removed without application restart. In order to
unlock Unregistering feature corresponding PR was filed - https://github.com/VictoriaMetrics/metrics/pull/13
* app/vmalert: extend metrics set exported by `vmalert` #573
The changes are following:
* add an ID label to rules metrics, since `name` collisions within one group is
a common case - see the k8s example alerts;
* supports metrics unregistering on rule updates. Consider the case when one rule
was added or removed from the group, or the whole group was added or removed.
The change depends on https://github.com/VictoriaMetrics/metrics/pull/16
where race condition for Unregister method was fixed.
2020-08-09 08:41:29 +02:00
|
|
|
rr := &RecordingRule{
|
2023-12-04 16:40:33 +01:00
|
|
|
Type: group.Type,
|
|
|
|
RuleID: cfg.ID,
|
|
|
|
Name: cfg.Record,
|
|
|
|
Expr: cfg.Expr,
|
|
|
|
Labels: cfg.Labels,
|
|
|
|
GroupID: group.ID(),
|
|
|
|
GroupName: group.Name,
|
|
|
|
File: group.File,
|
|
|
|
metrics: &recordingRuleMetrics{},
|
2021-04-30 08:46:03 +02:00
|
|
|
q: qb.BuildWithParams(datasource.QuerierParams{
|
2024-10-29 16:30:39 +01:00
|
|
|
DataSourceType: group.Type.String(),
|
|
|
|
ApplyIntervalAsTimeFilter: setIntervalAsTimeFilter(group.Type.String(), cfg.Expr),
|
|
|
|
EvaluationInterval: group.Interval,
|
|
|
|
QueryParams: group.Params,
|
|
|
|
Headers: group.Headers,
|
2021-04-30 08:46:03 +02:00
|
|
|
}),
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
2021-02-01 14:02:44 +01:00
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
entrySize := *ruleUpdateEntriesLimit
|
2022-12-29 12:36:44 +01:00
|
|
|
if cfg.UpdateEntriesLimit != nil {
|
2023-10-13 13:54:33 +02:00
|
|
|
entrySize = *cfg.UpdateEntriesLimit
|
|
|
|
}
|
|
|
|
if entrySize < 1 {
|
|
|
|
entrySize = 1
|
|
|
|
}
|
|
|
|
rr.state = &ruleState{
|
|
|
|
entries: make([]StateEntry, entrySize),
|
2022-12-29 12:36:44 +01:00
|
|
|
}
|
|
|
|
|
2023-11-02 16:01:31 +01:00
|
|
|
labels := fmt.Sprintf(`recording=%q, group=%q, file=%q, id="%d"`, rr.Name, group.Name, group.File, rr.ID())
|
2023-12-06 19:39:35 +01:00
|
|
|
rr.metrics.errors = utils.GetOrCreateCounter(fmt.Sprintf(`vmalert_recording_rules_errors_total{%s}`, labels))
|
2022-02-02 13:11:41 +01:00
|
|
|
rr.metrics.samples = utils.GetOrCreateGauge(fmt.Sprintf(`vmalert_recording_rules_last_evaluation_samples{%s}`, labels),
|
2021-08-05 08:59:46 +02:00
|
|
|
func() float64 {
|
2022-09-14 14:04:24 +02:00
|
|
|
e := rr.state.getLast()
|
2023-10-13 13:54:33 +02:00
|
|
|
return float64(e.Samples)
|
2021-08-05 08:59:46 +02:00
|
|
|
})
|
app/vmalert: extend metrics set exported by `vmalert` #573 (#654)
* app/vmalert: extend metrics set exported by `vmalert` #573
New metrics were added to improve observability:
+ vmalert_alerts_pending{alertname, group} - number of pending alerts per group
per alert;
+ vmalert_alerts_acitve{alertname, group} - number of active alerts per group
per alert;
+ vmalert_alerts_error{alertname, group} - is 1 if alertname ended up with error
during prev execution, is 0 if no errors happened;
+ vmalert_recording_rules_error{recording, group} - is 1 if recording rule
ended up with error during prev execution, is 0 if no errors happened;
* vmalert_iteration_total{group, file} - now contains group and file name labels.
This should improve control over specific groups;
* vmalert_iteration_duration_seconds{group, file} - now contains group and file name labels. This should improve control over specific groups;
Some collisions for alerts and recording rules are possible, because neither
group name nor alert/recording rule name are unique for compatibility reasons.
Commit contains list of TODOs for Unregistering metrics since groups and rules
are ephemeral and could be removed without application restart. In order to
unlock Unregistering feature corresponding PR was filed - https://github.com/VictoriaMetrics/metrics/pull/13
* app/vmalert: extend metrics set exported by `vmalert` #573
The changes are following:
* add an ID label to rules metrics, since `name` collisions within one group is
a common case - see the k8s example alerts;
* supports metrics unregistering on rule updates. Consider the case when one rule
was added or removed from the group, or the whole group was added or removed.
The change depends on https://github.com/VictoriaMetrics/metrics/pull/16
where race condition for Unregister method was fixed.
2020-08-09 08:41:29 +02:00
|
|
|
return rr
|
|
|
|
}
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
// close unregisters rule metrics
|
|
|
|
func (rr *RecordingRule) close() {
|
2022-02-02 13:11:41 +01:00
|
|
|
rr.metrics.errors.Unregister()
|
|
|
|
rr.metrics.samples.Unregister()
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
// execRange executes recording rule on the given time range similarly to Exec.
|
2021-06-09 11:20:38 +02:00
|
|
|
// It doesn't update internal states of the Rule and meant to be used just
|
|
|
|
// to get time series for backfilling.
|
2023-10-13 13:54:33 +02:00
|
|
|
func (rr *RecordingRule) execRange(ctx context.Context, start, end time.Time) ([]prompbmarshal.TimeSeries, error) {
|
2023-05-08 09:36:39 +02:00
|
|
|
res, err := rr.q.QueryRange(ctx, rr.Expr, start, end)
|
2021-06-09 11:20:38 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-05-08 09:36:39 +02:00
|
|
|
duplicates := make(map[string]struct{}, len(res.Data))
|
2021-06-09 11:20:38 +02:00
|
|
|
var tss []prompbmarshal.TimeSeries
|
2023-05-08 09:36:39 +02:00
|
|
|
for _, s := range res.Data {
|
2021-06-09 11:20:38 +02:00
|
|
|
ts := rr.toTimeSeries(s)
|
|
|
|
key := stringifyLabels(ts)
|
|
|
|
if _, ok := duplicates[key]; ok {
|
|
|
|
return nil, fmt.Errorf("original metric %v; resulting labels %q: %w", s.Labels, key, errDuplicate)
|
|
|
|
}
|
|
|
|
duplicates[key] = struct{}{}
|
|
|
|
tss = append(tss, ts)
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
2021-06-09 11:20:38 +02:00
|
|
|
return tss, nil
|
|
|
|
}
|
2020-06-01 12:46:37 +02:00
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
// exec executes RecordingRule expression via the given Querier.
|
|
|
|
func (rr *RecordingRule) exec(ctx context.Context, ts time.Time, limit int) ([]prompbmarshal.TimeSeries, error) {
|
2022-09-14 14:04:24 +02:00
|
|
|
start := time.Now()
|
2023-05-08 09:36:39 +02:00
|
|
|
res, req, err := rr.q.Query(ctx, rr.Expr, ts)
|
2023-10-13 13:54:33 +02:00
|
|
|
curState := StateEntry{
|
|
|
|
Time: start,
|
|
|
|
At: ts,
|
|
|
|
Duration: time.Since(start),
|
|
|
|
Samples: len(res.Data),
|
|
|
|
SeriesFetched: res.SeriesFetched,
|
|
|
|
Curl: requestToCurl(req),
|
2022-09-14 14:04:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
rr.state.add(curState)
|
2023-12-06 19:39:35 +01:00
|
|
|
if curState.Err != nil {
|
|
|
|
rr.metrics.errors.Inc()
|
|
|
|
}
|
2022-09-14 14:04:24 +02:00
|
|
|
}()
|
2020-06-01 12:46:37 +02:00
|
|
|
|
|
|
|
if err != nil {
|
2023-10-13 13:54:33 +02:00
|
|
|
curState.Err = fmt.Errorf("failed to execute query %q: %w", rr.Expr, err)
|
|
|
|
return nil, curState.Err
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
|
|
|
|
2023-05-08 09:36:39 +02:00
|
|
|
qMetrics := res.Data
|
2022-06-09 08:21:30 +02:00
|
|
|
numSeries := len(qMetrics)
|
|
|
|
if limit > 0 && numSeries > limit {
|
2023-10-13 13:54:33 +02:00
|
|
|
curState.Err = fmt.Errorf("exec exceeded limit of %d with %d series", limit, numSeries)
|
|
|
|
return nil, curState.Err
|
2022-06-09 08:21:30 +02:00
|
|
|
}
|
|
|
|
|
2021-05-15 12:25:57 +02:00
|
|
|
duplicates := make(map[string]struct{}, len(qMetrics))
|
2020-06-01 12:46:37 +02:00
|
|
|
var tss []prompbmarshal.TimeSeries
|
|
|
|
for _, r := range qMetrics {
|
2021-06-09 11:20:38 +02:00
|
|
|
ts := rr.toTimeSeries(r)
|
2021-05-15 12:25:57 +02:00
|
|
|
key := stringifyLabels(ts)
|
|
|
|
if _, ok := duplicates[key]; ok {
|
2023-10-13 13:54:33 +02:00
|
|
|
curState.Err = fmt.Errorf("original metric %v; resulting labels %q: %w", r, key, errDuplicate)
|
|
|
|
return nil, curState.Err
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
2021-05-15 12:25:57 +02:00
|
|
|
duplicates[key] = struct{}{}
|
2020-06-01 12:46:37 +02:00
|
|
|
tss = append(tss, ts)
|
|
|
|
}
|
|
|
|
return tss, nil
|
|
|
|
}
|
|
|
|
|
2021-05-15 12:25:57 +02:00
|
|
|
func stringifyLabels(ts prompbmarshal.TimeSeries) string {
|
2020-06-01 12:46:37 +02:00
|
|
|
labels := ts.Labels
|
2021-05-15 12:25:57 +02:00
|
|
|
if len(labels) > 1 {
|
|
|
|
sort.Slice(labels, func(i, j int) bool {
|
|
|
|
return labels[i].Name < labels[j].Name
|
|
|
|
})
|
|
|
|
}
|
|
|
|
b := strings.Builder{}
|
|
|
|
for i, l := range labels {
|
|
|
|
b.WriteString(l.Name)
|
|
|
|
b.WriteString("=")
|
|
|
|
b.WriteString(l.Value)
|
|
|
|
if i != len(labels)-1 {
|
|
|
|
b.WriteString(",")
|
|
|
|
}
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
2021-05-15 12:25:57 +02:00
|
|
|
return b.String()
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
|
|
|
|
2021-06-09 11:20:38 +02:00
|
|
|
func (rr *RecordingRule) toTimeSeries(m datasource.Metric) prompbmarshal.TimeSeries {
|
2020-06-01 12:46:37 +02:00
|
|
|
labels := make(map[string]string)
|
|
|
|
for _, l := range m.Labels {
|
|
|
|
labels[l.Name] = l.Value
|
|
|
|
}
|
|
|
|
labels["__name__"] = rr.Name
|
|
|
|
// override existing labels with configured ones
|
|
|
|
for k, v := range rr.Labels {
|
2023-12-22 16:07:47 +01:00
|
|
|
if _, ok := labels[k]; ok && labels[k] != v {
|
|
|
|
labels[fmt.Sprintf("exported_%s", k)] = labels[k]
|
|
|
|
}
|
2020-06-01 12:46:37 +02:00
|
|
|
labels[k] = v
|
|
|
|
}
|
2021-06-09 11:20:38 +02:00
|
|
|
return newTimeSeries(m.Values, m.Timestamps, labels)
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
// updateWith copies all significant fields.
|
|
|
|
func (rr *RecordingRule) updateWith(r Rule) error {
|
2020-06-01 12:46:37 +02:00
|
|
|
nr, ok := r.(*RecordingRule)
|
|
|
|
if !ok {
|
2023-12-22 16:07:47 +01:00
|
|
|
return fmt.Errorf("BUG: attempt to update recording rule with wrong type %#v", r)
|
2020-06-01 12:46:37 +02:00
|
|
|
}
|
|
|
|
rr.Expr = nr.Expr
|
|
|
|
rr.Labels = nr.Labels
|
2021-05-22 23:26:01 +02:00
|
|
|
rr.q = nr.q
|
2020-06-01 12:46:37 +02:00
|
|
|
return nil
|
|
|
|
}
|
2024-10-29 16:30:39 +01:00
|
|
|
|
|
|
|
// setIntervalAsTimeFilter returns true if given LogsQL has a time filter.
|
|
|
|
func setIntervalAsTimeFilter(dType, expr string) bool {
|
|
|
|
if dType != "vlogs" {
|
|
|
|
return false
|
|
|
|
}
|
2024-11-08 15:47:14 +01:00
|
|
|
q, err := logstorage.ParseStatsQuery(expr, 0)
|
|
|
|
if err != nil {
|
|
|
|
logger.Panicf("BUG: the LogsQL query must be valid here; got error: %s; query=[%s]", err, expr)
|
|
|
|
}
|
|
|
|
return !q.HasGlobalTimeFilter()
|
2024-10-29 16:30:39 +01:00
|
|
|
}
|