diff --git a/config/config.go b/config/config.go index 890592ca90..7fe312278a 100644 --- a/config/config.go +++ b/config/config.go @@ -942,7 +942,11 @@ type InhibitRule struct { TargetMatchers Matchers `yaml:"target_matchers,omitempty" json:"target_matchers,omitempty"` // A set of labels that must be equal between the source and target alert // for them to be a match. - Equal model.LabelNames `yaml:"equal,omitempty" json:"equal,omitempty"` + Equal model.LabelNames `yaml:"-" json:"-"` + // EqualStr allows us to validate the label depending on whether UTF-8 is + // enabled or disabled. It should be removed when Alertmanager is updated + // to use the validation modes in recent versions of prometheus/common. + EqualStr []string `yaml:"equal,omitempty" json:"equal,omitempty"` } // UnmarshalYAML implements the yaml.Unmarshaler interface for InhibitRule. @@ -964,6 +968,14 @@ func (r *InhibitRule) UnmarshalYAML(unmarshal func(interface{}) error) error { } } + for _, l := range r.EqualStr { + labelName := model.LabelName(l) + if !compat.IsValidLabelName(labelName) { + return fmt.Errorf("invalid label name %q in equal list", l) + } + r.Equal = append(r.Equal, labelName) + } + return nil } diff --git a/config/config_test.go b/config/config_test.go index 15edfeed2d..2efbef2164 100644 --- a/config/config_test.go +++ b/config/config_test.go @@ -25,8 +25,12 @@ import ( commoncfg "github.com/prometheus/common/config" "github.com/prometheus/common/model" + "github.com/prometheus/common/promslog" "github.com/stretchr/testify/require" "gopkg.in/yaml.v2" + + "github.com/prometheus/alertmanager/featurecontrol" + "github.com/prometheus/alertmanager/matcher/compat" ) func TestLoadEmptyString(t *testing.T) { @@ -1387,3 +1391,45 @@ func TestNilRegexp(t *testing.T) { }) } } + +func TestInhibitRuleEqual(t *testing.T) { + c, err := LoadFile("testdata/conf.inhibit-equal.yml") + require.NoError(t, err) + + // Check that Equal has the expected labels. + require.Len(t, c.InhibitRules, 1) + r := c.InhibitRules[0] + require.Equal(t, model.LabelNames{"qux", "corge"}, r.Equal) + + // Should not be able to load configuration with UTF-8 in equals + c, err = LoadFile("testdata/conf.inhibit-equal-utf8.yml") + require.Error(t, err) + require.Equal(t, "invalid label name \"qux🙂\" in equal list", err.Error()) + + // Change the mode to UTF-8 mode. + ff, err := featurecontrol.NewFlags(promslog.NewNopLogger(), featurecontrol.FeatureUTF8StrictMode) + require.NoError(t, err) + compat.InitFromFlags(promslog.NewNopLogger(), ff) + + // Restore the mode to classic at the end of the test. + ff, err = featurecontrol.NewFlags(promslog.NewNopLogger(), featurecontrol.FeatureClassicMode) + require.NoError(t, err) + defer compat.InitFromFlags(promslog.NewNopLogger(), ff) + + c, err = LoadFile("testdata/conf.inhibit-equal.yml") + require.NoError(t, err) + + // Check that Equal has the expected labels. + require.Len(t, c.InhibitRules, 1) + r = c.InhibitRules[0] + require.Equal(t, model.LabelNames{"qux", "corge"}, r.Equal) + + // Should also be able to load configuration with UTF-8 in equals + c, err = LoadFile("testdata/conf.inhibit-equal-utf8.yml") + require.NoError(t, err) + + // Check that Equal has the expected labels. + require.Len(t, c.InhibitRules, 1) + r = c.InhibitRules[0] + require.Equal(t, model.LabelNames{"qux🙂", "corge"}, r.Equal) +} diff --git a/config/testdata/conf.inhibit-equal-utf8.yml b/config/testdata/conf.inhibit-equal-utf8.yml new file mode 100644 index 0000000000..292c405a0b --- /dev/null +++ b/config/testdata/conf.inhibit-equal-utf8.yml @@ -0,0 +1,10 @@ +route: + receiver: test +receivers: + - name: test +inhibit_rules: + - source_matchers: + - foo=bar + target_matchers: + - bar=baz + equal: ['qux🙂', 'corge'] \ No newline at end of file diff --git a/config/testdata/conf.inhibit-equal.yml b/config/testdata/conf.inhibit-equal.yml new file mode 100644 index 0000000000..08620decc2 --- /dev/null +++ b/config/testdata/conf.inhibit-equal.yml @@ -0,0 +1,10 @@ +route: + receiver: test +receivers: + - name: test +inhibit_rules: + - source_matchers: + - foo=bar + target_matchers: + - bar=baz + equal: ['qux', 'corge'] \ No newline at end of file