mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-11 13:57:36 -08:00
avoid stopping rule groups if new rule groups are as same as old rule… (#6450)
* avoid stopping rule groups if new rule groups are as same as old rule groups Signed-off-by: alburthoffman <alburthoffman@gmail.com>
This commit is contained in:
parent
db1258f2a5
commit
156dcb8cca
|
@ -743,6 +743,33 @@ func (g *Group) RestoreForState(ts time.Time) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Equals return if two groups are the same
|
||||||
|
func (g *Group) Equals(ng *Group) bool {
|
||||||
|
if g.name != ng.name {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if g.file != ng.file {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if g.interval != ng.interval {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(g.rules) != len(ng.rules) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, gr := range g.rules {
|
||||||
|
if gr.String() != ng.rules[i].String() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
// The Manager manages recording and alerting rules.
|
// The Manager manages recording and alerting rules.
|
||||||
type Manager struct {
|
type Manager struct {
|
||||||
opts *ManagerOptions
|
opts *ManagerOptions
|
||||||
|
@ -836,16 +863,21 @@ func (m *Manager) Update(interval time.Duration, files []string, externalLabels
|
||||||
m.restored = true
|
m.restored = true
|
||||||
|
|
||||||
var wg sync.WaitGroup
|
var wg sync.WaitGroup
|
||||||
|
|
||||||
for _, newg := range groups {
|
for _, newg := range groups {
|
||||||
wg.Add(1)
|
// If there is an old group with the same identifier,
|
||||||
|
// check if new group equals with the old group, if yes then skip it.
|
||||||
// If there is an old group with the same identifier, stop it and wait for
|
// If not equals, stop it and wait for it to finish the current iteration.
|
||||||
// it to finish the current iteration. Then copy it into the new group.
|
// Then copy it into the new group.
|
||||||
gn := groupKey(newg.name, newg.file)
|
gn := groupKey(newg.name, newg.file)
|
||||||
oldg, ok := m.groups[gn]
|
oldg, ok := m.groups[gn]
|
||||||
delete(m.groups, gn)
|
delete(m.groups, gn)
|
||||||
|
|
||||||
|
if ok && oldg.Equals(newg) {
|
||||||
|
groups[gn] = oldg
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
wg.Add(1)
|
||||||
go func(newg *Group) {
|
go func(newg *Group) {
|
||||||
if ok {
|
if ok {
|
||||||
oldg.stop()
|
oldg.stop()
|
||||||
|
|
|
@ -15,15 +15,20 @@ package rules
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
"math"
|
"math"
|
||||||
|
"os"
|
||||||
"sort"
|
"sort"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/go-kit/kit/log"
|
"github.com/go-kit/kit/log"
|
||||||
"github.com/prometheus/common/model"
|
"github.com/prometheus/common/model"
|
||||||
|
yaml "gopkg.in/yaml.v2"
|
||||||
|
|
||||||
"github.com/prometheus/prometheus/pkg/labels"
|
"github.com/prometheus/prometheus/pkg/labels"
|
||||||
|
"github.com/prometheus/prometheus/pkg/rulefmt"
|
||||||
"github.com/prometheus/prometheus/pkg/timestamp"
|
"github.com/prometheus/prometheus/pkg/timestamp"
|
||||||
"github.com/prometheus/prometheus/pkg/value"
|
"github.com/prometheus/prometheus/pkg/value"
|
||||||
"github.com/prometheus/prometheus/promql"
|
"github.com/prometheus/prometheus/promql"
|
||||||
|
@ -703,18 +708,73 @@ func TestUpdate(t *testing.T) {
|
||||||
err := ruleManager.Update(10*time.Second, files, nil)
|
err := ruleManager.Update(10*time.Second, files, nil)
|
||||||
testutil.Ok(t, err)
|
testutil.Ok(t, err)
|
||||||
testutil.Assert(t, len(ruleManager.groups) > 0, "expected non-empty rule groups")
|
testutil.Assert(t, len(ruleManager.groups) > 0, "expected non-empty rule groups")
|
||||||
for _, g := range ruleManager.groups {
|
ogs := map[string]*Group{}
|
||||||
|
for h, g := range ruleManager.groups {
|
||||||
g.seriesInPreviousEval = []map[string]labels.Labels{
|
g.seriesInPreviousEval = []map[string]labels.Labels{
|
||||||
expected,
|
expected,
|
||||||
}
|
}
|
||||||
|
ogs[h] = g
|
||||||
}
|
}
|
||||||
|
|
||||||
err = ruleManager.Update(10*time.Second, files, nil)
|
err = ruleManager.Update(10*time.Second, files, nil)
|
||||||
testutil.Ok(t, err)
|
testutil.Ok(t, err)
|
||||||
for _, g := range ruleManager.groups {
|
for h, g := range ruleManager.groups {
|
||||||
for _, actual := range g.seriesInPreviousEval {
|
for _, actual := range g.seriesInPreviousEval {
|
||||||
testutil.Equals(t, expected, actual)
|
testutil.Equals(t, expected, actual)
|
||||||
}
|
}
|
||||||
|
// Groups are the same because of no updates.
|
||||||
|
testutil.Equals(t, ogs[h], g)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Groups will be recreated if updated.
|
||||||
|
rgs, errs := rulefmt.ParseFile("fixtures/rules.yaml")
|
||||||
|
testutil.Assert(t, len(errs) == 0, "file parsing failures")
|
||||||
|
|
||||||
|
tmpFile, err := ioutil.TempFile("", "rules.test.*.yaml")
|
||||||
|
testutil.Ok(t, err)
|
||||||
|
defer os.Remove(tmpFile.Name())
|
||||||
|
defer tmpFile.Close()
|
||||||
|
|
||||||
|
err = ruleManager.Update(10*time.Second, []string{tmpFile.Name()}, nil)
|
||||||
|
testutil.Ok(t, err)
|
||||||
|
|
||||||
|
for h, g := range ruleManager.groups {
|
||||||
|
ogs[h] = g
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update interval and reload
|
||||||
|
for i, g := range rgs.Groups {
|
||||||
|
if g.Interval != 0 {
|
||||||
|
rgs.Groups[i].Interval = g.Interval * 2
|
||||||
|
} else {
|
||||||
|
rgs.Groups[i].Interval = model.Duration(10)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
reloadAndValidate(rgs, t, tmpFile, ruleManager, expected, ogs)
|
||||||
|
|
||||||
|
// Change group rules and reload
|
||||||
|
for i, g := range rgs.Groups {
|
||||||
|
for j, r := range g.Rules {
|
||||||
|
rgs.Groups[i].Rules[j].Expr = fmt.Sprintf("%s * 0", r.Expr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
reloadAndValidate(rgs, t, tmpFile, ruleManager, expected, ogs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func reloadAndValidate(rgs *rulefmt.RuleGroups, t *testing.T, tmpFile *os.File, ruleManager *Manager, expected map[string]labels.Labels, ogs map[string]*Group) {
|
||||||
|
bs, err := yaml.Marshal(rgs)
|
||||||
|
testutil.Ok(t, err)
|
||||||
|
tmpFile.Seek(0, 0)
|
||||||
|
_, err = tmpFile.Write(bs)
|
||||||
|
testutil.Ok(t, err)
|
||||||
|
err = ruleManager.Update(10*time.Second, []string{tmpFile.Name()}, nil)
|
||||||
|
testutil.Ok(t, err)
|
||||||
|
for h, g := range ruleManager.groups {
|
||||||
|
if ogs[h] == g {
|
||||||
|
t.Fail()
|
||||||
|
}
|
||||||
|
ogs[h] = g
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue