mirror of
https://github.com/prometheus/prometheus.git
synced 2025-01-03 09:57:26 -08:00
e9f0f0ebed
Signed-off-by: Cyril Tovena <cyril.tovena@gmail.com>
283 lines
6.8 KiB
Go
283 lines
6.8 KiB
Go
// Copyright 2020 The Prometheus Authors
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package labels
|
|
|
|
import (
|
|
"regexp"
|
|
"regexp/syntax"
|
|
"strings"
|
|
)
|
|
|
|
const maxSetMatches = 256
|
|
|
|
type FastRegexMatcher struct {
|
|
re *regexp.Regexp
|
|
|
|
setMatches []string
|
|
prefix string
|
|
suffix string
|
|
contains string
|
|
}
|
|
|
|
func NewFastRegexMatcher(v string) (*FastRegexMatcher, error) {
|
|
parsed, err := syntax.Parse(v, syntax.Perl)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// Simplify the syntax tree to run faster.
|
|
parsed = parsed.Simplify()
|
|
re, err := regexp.Compile("^(?:" + parsed.String() + ")$")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
m := &FastRegexMatcher{
|
|
re: re,
|
|
setMatches: findSetMatches(parsed, ""),
|
|
}
|
|
|
|
if parsed.Op == syntax.OpConcat {
|
|
m.prefix, m.suffix, m.contains = optimizeConcatRegex(parsed)
|
|
}
|
|
|
|
return m, nil
|
|
}
|
|
|
|
// findSetMatches extract equality matches from a regexp.
|
|
// Returns nil if we can't replace the regexp by only equality matchers.
|
|
func findSetMatches(re *syntax.Regexp, base string) []string {
|
|
// Matches are not case sensitive, if we find a case insensitive regexp.
|
|
// We have to abort.
|
|
if isCaseInsensitive(re) {
|
|
return nil
|
|
}
|
|
switch re.Op {
|
|
case syntax.OpLiteral:
|
|
return []string{base + string(re.Rune)}
|
|
case syntax.OpEmptyMatch:
|
|
if base != "" {
|
|
return []string{base}
|
|
}
|
|
case syntax.OpAlternate:
|
|
found := findSetMatchesFromAlternate(re, base)
|
|
if found != nil {
|
|
return found
|
|
}
|
|
case syntax.OpCapture:
|
|
clearCapture(re)
|
|
return findSetMatches(re, base)
|
|
case syntax.OpConcat:
|
|
found := findSetMatchesFromConcat(re, base)
|
|
if found != nil {
|
|
return found
|
|
}
|
|
case syntax.OpCharClass:
|
|
if len(re.Rune) == 1 {
|
|
return []string{base + string(re.Rune)}
|
|
}
|
|
var matches []string
|
|
var totalSet int
|
|
for i := 0; i < len(re.Rune); i = i + 2 {
|
|
totalSet += int(re.Rune[i+1]-re.Rune[i]) + 1
|
|
}
|
|
// limits the total characters that can be used to create matches.
|
|
// In some case like negation [^0-9] a lot of possibilities exists and that
|
|
// can create thousands of possible matches at which points we're better off using regexp.
|
|
if totalSet > maxSetMatches {
|
|
return nil
|
|
}
|
|
for i := 0; i < len(re.Rune); i = i + 2 {
|
|
lo, hi := re.Rune[i], re.Rune[i+1]
|
|
for c := lo; c <= hi; c++ {
|
|
matches = append(matches, base+string(c))
|
|
}
|
|
|
|
}
|
|
return matches
|
|
default:
|
|
return nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func findSetMatchesFromConcat(re *syntax.Regexp, base string) []string {
|
|
if isCaseInsensitive(re) {
|
|
return nil
|
|
}
|
|
if len(re.Sub) == 0 {
|
|
return nil
|
|
}
|
|
clearCapture(re.Sub...)
|
|
matches := findSetMatches(re.Sub[0], base)
|
|
if matches == nil {
|
|
return nil
|
|
}
|
|
|
|
for i := 1; i < len(re.Sub); i++ {
|
|
var newMatches []string
|
|
for _, b := range matches {
|
|
m := findSetMatches(re.Sub[i], b)
|
|
if m == nil {
|
|
return nil
|
|
}
|
|
if tooManyMatches(newMatches, m...) {
|
|
return nil
|
|
}
|
|
newMatches = append(newMatches, m...)
|
|
}
|
|
matches = newMatches
|
|
}
|
|
|
|
return matches
|
|
}
|
|
|
|
func findSetMatchesFromAlternate(re *syntax.Regexp, base string) []string {
|
|
var setMatches []string
|
|
for _, sub := range re.Sub {
|
|
found := findSetMatches(sub, base)
|
|
if found == nil {
|
|
return nil
|
|
}
|
|
if tooManyMatches(setMatches, found...) {
|
|
return nil
|
|
}
|
|
setMatches = append(setMatches, found...)
|
|
}
|
|
return setMatches
|
|
}
|
|
|
|
// clearCapture removes capture operation as they are not used for matching.
|
|
func clearCapture(regs ...*syntax.Regexp) {
|
|
for _, r := range regs {
|
|
if r.Op == syntax.OpCapture {
|
|
*r = *r.Sub[0]
|
|
}
|
|
}
|
|
}
|
|
|
|
// isCaseInsensitive tells if a regexp is case insensitive.
|
|
// The flag should be check at each level of the syntax tree.
|
|
func isCaseInsensitive(reg *syntax.Regexp) bool {
|
|
return (reg.Flags & syntax.FoldCase) != 0
|
|
}
|
|
|
|
// tooManyMatches guards against creating too many set matches
|
|
func tooManyMatches(matches []string, new ...string) bool {
|
|
return len(matches)+len(new) > maxSetMatches
|
|
}
|
|
|
|
func (m *FastRegexMatcher) MatchString(s string) bool {
|
|
if len(m.setMatches) != 0 {
|
|
for _, match := range m.setMatches {
|
|
if match == s {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
if m.prefix != "" && !strings.HasPrefix(s, m.prefix) {
|
|
return false
|
|
}
|
|
if m.suffix != "" && !strings.HasSuffix(s, m.suffix) {
|
|
return false
|
|
}
|
|
if m.contains != "" && !strings.Contains(s, m.contains) {
|
|
return false
|
|
}
|
|
return m.re.MatchString(s)
|
|
}
|
|
|
|
func (m *FastRegexMatcher) SetMatches() []string {
|
|
return m.setMatches
|
|
}
|
|
|
|
// optimizeConcatRegex returns literal prefix/suffix text that can be safely
|
|
// checked against the label value before running the regexp matcher.
|
|
func optimizeConcatRegex(r *syntax.Regexp) (prefix, suffix, contains string) {
|
|
sub := r.Sub
|
|
|
|
// We can safely remove begin and end text matchers respectively
|
|
// at the beginning and end of the regexp.
|
|
if len(sub) > 0 && sub[0].Op == syntax.OpBeginText {
|
|
sub = sub[1:]
|
|
}
|
|
if len(sub) > 0 && sub[len(sub)-1].Op == syntax.OpEndText {
|
|
sub = sub[:len(sub)-1]
|
|
}
|
|
|
|
if len(sub) == 0 {
|
|
return
|
|
}
|
|
|
|
// Given Prometheus regex matchers are always anchored to the begin/end
|
|
// of the text, if the first/last operations are literals, we can safely
|
|
// treat them as prefix/suffix.
|
|
if sub[0].Op == syntax.OpLiteral && (sub[0].Flags&syntax.FoldCase) == 0 {
|
|
prefix = string(sub[0].Rune)
|
|
}
|
|
if last := len(sub) - 1; sub[last].Op == syntax.OpLiteral && (sub[last].Flags&syntax.FoldCase) == 0 {
|
|
suffix = string(sub[last].Rune)
|
|
}
|
|
|
|
// If contains any literal which is not a prefix/suffix, we keep the
|
|
// 1st one. We do not keep the whole list of literals to simplify the
|
|
// fast path.
|
|
for i := 1; i < len(sub)-1; i++ {
|
|
if sub[i].Op == syntax.OpLiteral && (sub[i].Flags&syntax.FoldCase) == 0 {
|
|
contains = string(sub[i].Rune)
|
|
break
|
|
}
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// todo remove anchors ^foo$
|
|
// remove captures.
|
|
// .+ | .* | .*POD.* | test-.* | .*-test | test-.+ | .+-test
|
|
|
|
// regexp prefix fn ???? can we use this ?
|
|
|
|
// benchmark them
|
|
|
|
type StringMatcher interface {
|
|
Matches(s string) bool
|
|
}
|
|
|
|
func contains(s, substr string) bool {
|
|
pos := strings.Index(s, substr)
|
|
if pos < 0 {
|
|
return false
|
|
}
|
|
return matchesAnyZeroOrMoreNotNL(s[:pos]) && matchesAnyZeroOrMoreNotNL(s[pos+len(substr):])
|
|
}
|
|
|
|
func matchesAnyZeroOrMoreNotNL(s string) bool {
|
|
// strings.Contains(s string, substr string) faster ??
|
|
for _, r := range s {
|
|
if r == '\n' {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func matchesOneZeroOrMoreNotNL(s string) bool {
|
|
for _, r := range s {
|
|
if r == '\n' {
|
|
return false
|
|
}
|
|
}
|
|
return len(s) > 0
|
|
}
|