mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-13 06:47:28 -08:00
Merge pull request #14068 from colega/quote-label-name-in-matchers-when-needed
Bugfix: quote label name in matchers when needed
This commit is contained in:
commit
e6be4240be
|
@ -14,7 +14,8 @@
|
||||||
package labels
|
package labels
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"bytes"
|
||||||
|
"strconv"
|
||||||
)
|
)
|
||||||
|
|
||||||
// MatchType is an enum for label matching types.
|
// MatchType is an enum for label matching types.
|
||||||
|
@ -78,7 +79,29 @@ func MustNewMatcher(mt MatchType, name, val string) *Matcher {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *Matcher) String() string {
|
func (m *Matcher) String() string {
|
||||||
return fmt.Sprintf("%s%s%q", m.Name, m.Type, m.Value)
|
// Start a buffer with a pre-allocated size on stack to cover most needs.
|
||||||
|
var bytea [1024]byte
|
||||||
|
b := bytes.NewBuffer(bytea[:0])
|
||||||
|
|
||||||
|
if m.shouldQuoteName() {
|
||||||
|
b.Write(strconv.AppendQuote(b.AvailableBuffer(), m.Name))
|
||||||
|
} else {
|
||||||
|
b.WriteString(m.Name)
|
||||||
|
}
|
||||||
|
b.WriteString(m.Type.String())
|
||||||
|
b.Write(strconv.AppendQuote(b.AvailableBuffer(), m.Value))
|
||||||
|
|
||||||
|
return b.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Matcher) shouldQuoteName() bool {
|
||||||
|
for i, c := range m.Name {
|
||||||
|
if c == '_' || (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (i > 0 && c >= '0' && c <= '9') {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// Matches returns whether the matcher matches the given string value.
|
// Matches returns whether the matcher matches the given string value.
|
||||||
|
|
|
@ -15,6 +15,7 @@ package labels
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math/rand"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
@ -225,3 +226,128 @@ func BenchmarkNewMatcher(b *testing.B) {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func BenchmarkMatcher_String(b *testing.B) {
|
||||||
|
type benchCase struct {
|
||||||
|
name string
|
||||||
|
matchers []*Matcher
|
||||||
|
}
|
||||||
|
cases := []benchCase{
|
||||||
|
{
|
||||||
|
name: "short name equal",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchEqual, "foo", "bar"),
|
||||||
|
MustNewMatcher(MatchEqual, "bar", "baz"),
|
||||||
|
MustNewMatcher(MatchEqual, "abc", "def"),
|
||||||
|
MustNewMatcher(MatchEqual, "ghi", "klm"),
|
||||||
|
MustNewMatcher(MatchEqual, "nop", "qrs"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "short quoted name not equal",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchEqual, "f.o", "bar"),
|
||||||
|
MustNewMatcher(MatchEqual, "b.r", "baz"),
|
||||||
|
MustNewMatcher(MatchEqual, "a.c", "def"),
|
||||||
|
MustNewMatcher(MatchEqual, "g.i", "klm"),
|
||||||
|
MustNewMatcher(MatchEqual, "n.p", "qrs"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "short quoted name with quotes not equal",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchEqual, `"foo"`, "bar"),
|
||||||
|
MustNewMatcher(MatchEqual, `"foo"`, "baz"),
|
||||||
|
MustNewMatcher(MatchEqual, `"foo"`, "def"),
|
||||||
|
MustNewMatcher(MatchEqual, `"foo"`, "klm"),
|
||||||
|
MustNewMatcher(MatchEqual, `"foo"`, "qrs"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "short name value with quotes equal",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchEqual, "foo", `"bar"`),
|
||||||
|
MustNewMatcher(MatchEqual, "bar", `"baz"`),
|
||||||
|
MustNewMatcher(MatchEqual, "abc", `"def"`),
|
||||||
|
MustNewMatcher(MatchEqual, "ghi", `"klm"`),
|
||||||
|
MustNewMatcher(MatchEqual, "nop", `"qrs"`),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "short name and long value regexp",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchRegexp, "foo", "five_six_seven_eight_nine_ten_one_two_three_four"),
|
||||||
|
MustNewMatcher(MatchRegexp, "bar", "one_two_three_four_five_six_seven_eight_nine_ten"),
|
||||||
|
MustNewMatcher(MatchRegexp, "abc", "two_three_four_five_six_seven_eight_nine_ten_one"),
|
||||||
|
MustNewMatcher(MatchRegexp, "ghi", "three_four_five_six_seven_eight_nine_ten_one_two"),
|
||||||
|
MustNewMatcher(MatchRegexp, "nop", "four_five_six_seven_eight_nine_ten_one_two_three"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "short name and long value with quotes equal",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchEqual, "foo", `five_six_seven_eight_nine_ten_"one"_two_three_four`),
|
||||||
|
MustNewMatcher(MatchEqual, "bar", `one_two_three_four_five_six_"seven"_eight_nine_ten`),
|
||||||
|
MustNewMatcher(MatchEqual, "abc", `two_three_four_five_six_seven_"eight"_nine_ten_one`),
|
||||||
|
MustNewMatcher(MatchEqual, "ghi", `three_four_five_six_seven_eight_"nine"_ten_one_two`),
|
||||||
|
MustNewMatcher(MatchEqual, "nop", `four_five_six_seven_eight_nine_"ten"_one_two_three`),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "long name regexp",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchRegexp, "one_two_three_four_five_six_seven_eight_nine_ten", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "two_three_four_five_six_seven_eight_nine_ten_one", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "three_four_five_six_seven_eight_nine_ten_one_two", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "four_five_six_seven_eight_nine_ten_one_two_three", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "five_six_seven_eight_nine_ten_one_two_three_four", "val"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "long quoted name regexp",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchRegexp, "one.two.three.four.five.six.seven.eight.nine.ten", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "two.three.four.five.six.seven.eight.nine.ten.one", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "three.four.five.six.seven.eight.nine.ten.one.two", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "four.five.six.seven.eight.nine.ten.one.two.three", "val"),
|
||||||
|
MustNewMatcher(MatchRegexp, "five.six.seven.eight.nine.ten.one.two.three.four", "val"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "long name and long value regexp",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchRegexp, "one_two_three_four_five_six_seven_eight_nine_ten", "five_six_seven_eight_nine_ten_one_two_three_four"),
|
||||||
|
MustNewMatcher(MatchRegexp, "two_three_four_five_six_seven_eight_nine_ten_one", "one_two_three_four_five_six_seven_eight_nine_ten"),
|
||||||
|
MustNewMatcher(MatchRegexp, "three_four_five_six_seven_eight_nine_ten_one_two", "two_three_four_five_six_seven_eight_nine_ten_one"),
|
||||||
|
MustNewMatcher(MatchRegexp, "four_five_six_seven_eight_nine_ten_one_two_three", "three_four_five_six_seven_eight_nine_ten_one_two"),
|
||||||
|
MustNewMatcher(MatchRegexp, "five_six_seven_eight_nine_ten_one_two_three_four", "four_five_six_seven_eight_nine_ten_one_two_three"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "long quoted name and long value regexp",
|
||||||
|
matchers: []*Matcher{
|
||||||
|
MustNewMatcher(MatchRegexp, "one.two.three.four.five.six.seven.eight.nine.ten", "five.six.seven.eight.nine.ten.one.two.three.four"),
|
||||||
|
MustNewMatcher(MatchRegexp, "two.three.four.five.six.seven.eight.nine.ten.one", "one.two.three.four.five.six.seven.eight.nine.ten"),
|
||||||
|
MustNewMatcher(MatchRegexp, "three.four.five.six.seven.eight.nine.ten.one.two", "two.three.four.five.six.seven.eight.nine.ten.one"),
|
||||||
|
MustNewMatcher(MatchRegexp, "four.five.six.seven.eight.nine.ten.one.two.three", "three.four.five.six.seven.eight.nine.ten.one.two"),
|
||||||
|
MustNewMatcher(MatchRegexp, "five.six.seven.eight.nine.ten.one.two.three.four", "four.five.six.seven.eight.nine.ten.one.two.three"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var mixed []*Matcher
|
||||||
|
for _, bc := range cases {
|
||||||
|
mixed = append(mixed, bc.matchers...)
|
||||||
|
}
|
||||||
|
rand.Shuffle(len(mixed), func(i, j int) { mixed[i], mixed[j] = mixed[j], mixed[i] })
|
||||||
|
cases = append(cases, benchCase{name: "mixed", matchers: mixed})
|
||||||
|
|
||||||
|
for _, bc := range cases {
|
||||||
|
b.Run(bc.name, func(b *testing.B) {
|
||||||
|
for i := 0; i <= b.N; i++ {
|
||||||
|
m := bc.matchers[i%len(bc.matchers)]
|
||||||
|
_ = m.String()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -138,6 +138,16 @@ func TestExprString(t *testing.T) {
|
||||||
{
|
{
|
||||||
in: `{__name__="",a="x"}`,
|
in: `{__name__="",a="x"}`,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
in: `{"a.b"="c"}`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
in: `{"0"="1"}`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
in: `{"_0"="1"}`,
|
||||||
|
out: `{_0="1"}`,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, test := range inputs {
|
for _, test := range inputs {
|
||||||
|
|
Loading…
Reference in a new issue