mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 07:19:17 +01:00
lib/storage: properly handle (?i)
in the tag filter regexp
Fixes https://github.com/VictoriaMetrics/VictoriaMetrics/issues/161
This commit is contained in:
parent
0dc0006f34
commit
2fc157ff7a
@ -592,17 +592,20 @@ func extractRegexpPrefix(b []byte) ([]byte, []byte) {
|
|||||||
if re == emptyRegexp {
|
if re == emptyRegexp {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
if re.Op == syntax.OpLiteral {
|
if re.Op == syntax.OpLiteral && re.Flags & syntax.FoldCase == 0 {
|
||||||
return []byte(string(re.Rune)), nil
|
return []byte(string(re.Rune)), nil
|
||||||
}
|
}
|
||||||
var prefix []byte
|
var prefix []byte
|
||||||
if re.Op == syntax.OpConcat && re.Sub[0].Op == syntax.OpLiteral {
|
if re.Op == syntax.OpConcat {
|
||||||
prefix = []byte(string(re.Sub[0].Rune))
|
sub0 := re.Sub[0]
|
||||||
|
if sub0.Op == syntax.OpLiteral && sub0.Flags & syntax.FoldCase == 0 {
|
||||||
|
prefix = []byte(string(sub0.Rune))
|
||||||
re.Sub = re.Sub[1:]
|
re.Sub = re.Sub[1:]
|
||||||
if len(re.Sub) == 0 {
|
if len(re.Sub) == 0 {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if _, err := syntax.Compile(re); err != nil {
|
if _, err := syntax.Compile(re); err != nil {
|
||||||
// Cannot compile the regexp. Return it all as prefix.
|
// Cannot compile the regexp. Return it all as prefix.
|
||||||
return b, nil
|
return b, nil
|
||||||
|
@ -5,6 +5,21 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func TestExtractRegexpPrefix(t *testing.T) {
|
||||||
|
f := func(s string, expectedPrefix, expectedSuffix string) {
|
||||||
|
t.Helper()
|
||||||
|
prefix, suffix := extractRegexpPrefix([]byte(s))
|
||||||
|
if string(prefix) != expectedPrefix {
|
||||||
|
t.Fatalf("unexpected prefix for %q; got %q; want %q", s, prefix, expectedPrefix)
|
||||||
|
}
|
||||||
|
if string(suffix) != expectedSuffix {
|
||||||
|
t.Fatalf("unexpected suffix for %q; got %q; want %q", s, suffix, expectedSuffix)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
f("", "", "")
|
||||||
|
f("foobar", "foobar", "")
|
||||||
|
}
|
||||||
|
|
||||||
func TestGetRegexpFromCache(t *testing.T) {
|
func TestGetRegexpFromCache(t *testing.T) {
|
||||||
f := func(s string, orValuesExpected, expectedMatches, expectedMismatches []string) {
|
f := func(s string, orValuesExpected, expectedMatches, expectedMismatches []string) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
@ -397,49 +412,7 @@ func TestGetOrValues(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestGetRegexpPrefix(t *testing.T) {
|
func TestGetRegexpPrefix(t *testing.T) {
|
||||||
testGetRegexpPrefix(t, "", "", "")
|
f := func(t *testing.T, s, expectedPrefix, expectedSuffix string) {
|
||||||
testGetRegexpPrefix(t, "^", "", "")
|
|
||||||
testGetRegexpPrefix(t, "$", "", "")
|
|
||||||
testGetRegexpPrefix(t, "^()$", "", "")
|
|
||||||
testGetRegexpPrefix(t, "^(?:)$", "", "")
|
|
||||||
testGetRegexpPrefix(t, "foobar", "foobar", "")
|
|
||||||
testGetRegexpPrefix(t, "foo$|^foobar", "foo", "(?:(?:)|bar)")
|
|
||||||
testGetRegexpPrefix(t, "^(foo$|^foobar)$", "foo", "(?:(?:)|bar)")
|
|
||||||
testGetRegexpPrefix(t, "foobar|foobaz", "fooba", "[rz]")
|
|
||||||
testGetRegexpPrefix(t, "(fo|(zar|bazz)|x)", "", "fo|zar|bazz|x")
|
|
||||||
testGetRegexpPrefix(t, "(тестЧЧ|тест)", "тест", "(?:ЧЧ|(?:))")
|
|
||||||
testGetRegexpPrefix(t, "foo(bar|baz|bana)", "fooba", "(?:[rz]|na)")
|
|
||||||
testGetRegexpPrefix(t, "^foobar|foobaz", "fooba", "[rz]")
|
|
||||||
testGetRegexpPrefix(t, "^foobar|^foobaz$", "fooba", "[rz]")
|
|
||||||
testGetRegexpPrefix(t, "foobar|foobaz", "fooba", "[rz]")
|
|
||||||
testGetRegexpPrefix(t, "(?:^foobar|^foobaz)aa.*", "fooba", "[rz]aa(?-s:.)*")
|
|
||||||
testGetRegexpPrefix(t, "foo[bar]+", "foo", "[a-br]+")
|
|
||||||
testGetRegexpPrefix(t, "foo[a-z]+", "foo", "[a-z]+")
|
|
||||||
testGetRegexpPrefix(t, "foo[bar]*", "foo", "[a-br]*")
|
|
||||||
testGetRegexpPrefix(t, "foo[a-z]*", "foo", "[a-z]*")
|
|
||||||
testGetRegexpPrefix(t, "foo[x]+", "foo", "x+")
|
|
||||||
testGetRegexpPrefix(t, "foo[^x]+", "foo", "[^x]+")
|
|
||||||
testGetRegexpPrefix(t, "foo[x]*", "foo", "x*")
|
|
||||||
testGetRegexpPrefix(t, "foo[^x]*", "foo", "[^x]*")
|
|
||||||
testGetRegexpPrefix(t, "foo[x]*bar", "foo", "x*bar")
|
|
||||||
testGetRegexpPrefix(t, "fo\\Bo[x]*bar?", "fo", "\\Box*bar?")
|
|
||||||
|
|
||||||
// test invalid regexps
|
|
||||||
testGetRegexpPrefix(t, "a(", "a(", "")
|
|
||||||
testGetRegexpPrefix(t, "a[", "a[", "")
|
|
||||||
testGetRegexpPrefix(t, "a[]", "a[]", "")
|
|
||||||
testGetRegexpPrefix(t, "a{", "a{", "")
|
|
||||||
testGetRegexpPrefix(t, "a{}", "a{}", "")
|
|
||||||
testGetRegexpPrefix(t, "invalid(regexp", "invalid(regexp", "")
|
|
||||||
|
|
||||||
// The transformed regexp mustn't match aba
|
|
||||||
testGetRegexpPrefix(t, "a?(^ba|c)", "", "a?(?:\\Aba|c)")
|
|
||||||
|
|
||||||
// The transformed regexp mustn't match barx
|
|
||||||
testGetRegexpPrefix(t, "(foo|bar$)x*", "", "(?:foo|bar(?-m:$))x*")
|
|
||||||
}
|
|
||||||
|
|
||||||
func testGetRegexpPrefix(t *testing.T, s, expectedPrefix, expectedSuffix string) {
|
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
|
||||||
prefix, suffix := getRegexpPrefix([]byte(s))
|
prefix, suffix := getRegexpPrefix([]byte(s))
|
||||||
@ -460,6 +433,54 @@ func testGetRegexpPrefix(t *testing.T, s, expectedPrefix, expectedSuffix string)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
f(t, "", "", "")
|
||||||
|
f(t, "^", "", "")
|
||||||
|
f(t, "$", "", "")
|
||||||
|
f(t, "^()$", "", "")
|
||||||
|
f(t, "^(?:)$", "", "")
|
||||||
|
f(t, "foobar", "foobar", "")
|
||||||
|
f(t, "foo$|^foobar", "foo", "(?:(?:)|bar)")
|
||||||
|
f(t, "^(foo$|^foobar)$", "foo", "(?:(?:)|bar)")
|
||||||
|
f(t, "foobar|foobaz", "fooba", "[rz]")
|
||||||
|
f(t, "(fo|(zar|bazz)|x)", "", "fo|zar|bazz|x")
|
||||||
|
f(t, "(тестЧЧ|тест)", "тест", "(?:ЧЧ|(?:))")
|
||||||
|
f(t, "foo(bar|baz|bana)", "fooba", "(?:[rz]|na)")
|
||||||
|
f(t, "^foobar|foobaz", "fooba", "[rz]")
|
||||||
|
f(t, "^foobar|^foobaz$", "fooba", "[rz]")
|
||||||
|
f(t, "foobar|foobaz", "fooba", "[rz]")
|
||||||
|
f(t, "(?:^foobar|^foobaz)aa.*", "fooba", "[rz]aa(?-s:.)*")
|
||||||
|
f(t, "foo[bar]+", "foo", "[a-br]+")
|
||||||
|
f(t, "foo[a-z]+", "foo", "[a-z]+")
|
||||||
|
f(t, "foo[bar]*", "foo", "[a-br]*")
|
||||||
|
f(t, "foo[a-z]*", "foo", "[a-z]*")
|
||||||
|
f(t, "foo[x]+", "foo", "x+")
|
||||||
|
f(t, "foo[^x]+", "foo", "[^x]+")
|
||||||
|
f(t, "foo[x]*", "foo", "x*")
|
||||||
|
f(t, "foo[^x]*", "foo", "[^x]*")
|
||||||
|
f(t, "foo[x]*bar", "foo", "x*bar")
|
||||||
|
f(t, "fo\\Bo[x]*bar?", "fo", "\\Box*bar?")
|
||||||
|
f(t, "foo.+bar", "foo", "(?-s:.)+bar")
|
||||||
|
f(t, "a(b|c.*).+", "a", "(?:b|c(?-s:.)*)(?-s:.)+")
|
||||||
|
f(t, "ab|ac", "a", "[b-c]")
|
||||||
|
f(t, "(?i)xyz", "", "(?i:XYZ)")
|
||||||
|
f(t, "(?i)up.+x", "", "(?i:UP)(?-s:.)+(?i:X)")
|
||||||
|
f(t, "(?smi)xy.*z$", "", "(?i:XY)(?s:.)*(?i:Z)(?m:$)")
|
||||||
|
|
||||||
|
// test invalid regexps
|
||||||
|
f(t, "a(", "a(", "")
|
||||||
|
f(t, "a[", "a[", "")
|
||||||
|
f(t, "a[]", "a[]", "")
|
||||||
|
f(t, "a{", "a{", "")
|
||||||
|
f(t, "a{}", "a{}", "")
|
||||||
|
f(t, "invalid(regexp", "invalid(regexp", "")
|
||||||
|
|
||||||
|
// The transformed regexp mustn't match aba
|
||||||
|
f(t, "a?(^ba|c)", "", "a?(?:\\Aba|c)")
|
||||||
|
|
||||||
|
// The transformed regexp mustn't match barx
|
||||||
|
f(t, "(foo|bar$)x*", "", "(?:foo|bar(?-m:$))x*")
|
||||||
|
}
|
||||||
|
|
||||||
func TestTagFiltersAddEmpty(t *testing.T) {
|
func TestTagFiltersAddEmpty(t *testing.T) {
|
||||||
tfs := NewTagFilters()
|
tfs := NewTagFilters()
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user