github.com/m3db/m3@v1.5.0/src/metrics/filters/filter_test.go (about)

     1  // Copyright (c) 2017 Uber Technologies, Inc.
     2  //
     3  // Permission is hereby granted, free of charge, to any person obtaining a copy
     4  // of this software and associated documentation files (the "Software"), to deal
     5  // in the Software without restriction, including without limitation the rights
     6  // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
     7  // copies of the Software, and to permit persons to whom the Software is
     8  // furnished to do so, subject to the following conditions:
     9  //
    10  // The above copyright notice and this permission notice shall be included in
    11  // all copies or substantial portions of the Software.
    12  //
    13  // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
    14  // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
    15  // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
    16  // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
    17  // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
    18  // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
    19  // THE SOFTWARE.
    20  
    21  package filters
    22  
    23  import (
    24  	"fmt"
    25  	"testing"
    26  
    27  	"github.com/stretchr/testify/require"
    28  )
    29  
    30  func TestNewFilterFromFilterValueInvalidPattern(t *testing.T) {
    31  	inputs := []string{"ab]c[sdf", "abc[z-a]", "*con[tT]ains*"}
    32  	for _, input := range inputs {
    33  		_, err := NewFilterFromFilterValue(FilterValue{Pattern: input})
    34  		require.Error(t, err)
    35  	}
    36  }
    37  
    38  func TestNewFilterFromFilterValueWithNegation(t *testing.T) {
    39  	inputs := []struct {
    40  		pattern string
    41  		negate  bool
    42  		data    []mockFilterData
    43  	}{
    44  		{
    45  			pattern: "foo",
    46  			negate:  false,
    47  			data: []mockFilterData{
    48  				{val: "foo", match: true},
    49  				{val: "fo", match: false},
    50  			},
    51  		},
    52  		{
    53  			pattern: "foo*bar",
    54  			negate:  false,
    55  			data: []mockFilterData{
    56  				{val: "foobar", match: true},
    57  				{val: "foozapbar", match: true},
    58  				{val: "bazbar", match: false},
    59  			},
    60  		},
    61  		{
    62  			pattern: "ba?[0-9][!a-z]9",
    63  			negate:  false,
    64  			data: []mockFilterData{
    65  				{val: "bar959", match: true},
    66  				{val: "bat449", match: true},
    67  				{val: "bar9", match: false},
    68  			},
    69  		},
    70  		{
    71  			pattern: "{ba,fo,car}*",
    72  			negate:  false,
    73  			data: []mockFilterData{
    74  				{val: "ba", match: true},
    75  				{val: "foo", match: true},
    76  				{val: "car", match: true},
    77  				{val: "ca", match: false},
    78  			},
    79  		},
    80  		{
    81  			pattern: "foo",
    82  			negate:  true,
    83  			data: []mockFilterData{
    84  				{val: "foo", match: false},
    85  				{val: "fo", match: true},
    86  			},
    87  		},
    88  		{
    89  			pattern: "foo*bar",
    90  			negate:  true,
    91  			data: []mockFilterData{
    92  				{val: "foobar", match: false},
    93  				{val: "foozapbar", match: false},
    94  				{val: "bazbar", match: true},
    95  			},
    96  		},
    97  		{
    98  			pattern: "ba?[0-9][!a-z]9",
    99  			negate:  true,
   100  			data: []mockFilterData{
   101  				{val: "bar959", match: false},
   102  				{val: "bat449", match: false},
   103  				{val: "bar9", match: true},
   104  			},
   105  		},
   106  		{
   107  			pattern: "{ba,fo,car}*",
   108  			negate:  true,
   109  			data: []mockFilterData{
   110  				{val: "ba", match: false},
   111  				{val: "foo", match: false},
   112  				{val: "car", match: false},
   113  				{val: "ca", match: true},
   114  			},
   115  		},
   116  	}
   117  
   118  	for _, input := range inputs {
   119  		f, err := NewFilterFromFilterValue(FilterValue{Pattern: input.pattern, Negate: input.negate})
   120  		require.NoError(t, err)
   121  		for _, testcase := range input.data {
   122  			require.Equal(t, testcase.match, f.Matches([]byte(testcase.val)))
   123  		}
   124  	}
   125  }
   126  
   127  func TestFilters(t *testing.T) {
   128  	filters := genAndValidateFilters(t, []testPattern{
   129  		testPattern{pattern: "f[A-z]?*", expectedStr: "StartsWith(Equals(\"f\") then Range(\"A-z\") then AnyChar)"},
   130  		testPattern{pattern: "*ba[a-z]", expectedStr: "EndsWith(Equals(\"ba\") then Range(\"a-z\"))"},
   131  		testPattern{pattern: "fo*?ba[!0-9][0-9]{8,9}", expectedStr: "StartsWith(Equals(\"fo\")) && EndsWith(AnyChar then Equals(\"ba\") then Not(Range(\"0-9\")) then Range(\"0-9\") then Range(\"8,9\"))"},
   132  	})
   133  
   134  	inputs := []testInput{
   135  		newTestInput("foo", true, false, false),
   136  		newTestInput("test", false, false, false),
   137  		newTestInput("bar", false, true, false),
   138  		newTestInput("foobar", true, true, false),
   139  		newTestInput("foobar08", true, false, true),
   140  		newTestInput("footybar09", true, false, true),
   141  	}
   142  
   143  	for _, input := range inputs {
   144  		for i, expectedMatch := range input.matches {
   145  			require.Equal(t, expectedMatch, filters[i].Matches(input.val),
   146  				fmt.Sprintf("input: %s, pattern: %s", input.val, filters[i].String()))
   147  		}
   148  	}
   149  }
   150  
   151  func TestEqualityFilter(t *testing.T) {
   152  	inputs := []mockFilterData{
   153  		{val: "foo", match: true},
   154  		{val: "fo", match: false},
   155  		{val: "foob", match: false},
   156  	}
   157  	f := newEqualityFilter([]byte("foo"))
   158  	for _, input := range inputs {
   159  		require.Equal(t, input.match, f.Matches([]byte(input.val)))
   160  	}
   161  }
   162  
   163  func TestEmptyFilter(t *testing.T) {
   164  	f, err := NewFilter(nil)
   165  	require.NoError(t, err)
   166  	require.True(t, f.Matches([]byte("")))
   167  	require.False(t, f.Matches([]byte(" ")))
   168  	require.False(t, f.Matches([]byte("foo")))
   169  }
   170  
   171  func TestWildcardFilters(t *testing.T) {
   172  	filters := genAndValidateFilters(t, []testPattern{
   173  		testPattern{pattern: "foo", expectedStr: "Equals(\"foo\")"},
   174  		testPattern{pattern: "*bar", expectedStr: "EndsWith(Equals(\"bar\"))"},
   175  		testPattern{pattern: "baz*", expectedStr: "StartsWith(Equals(\"baz\"))"},
   176  		testPattern{pattern: "*cat*", expectedStr: "Contains(\"cat\")"},
   177  		testPattern{pattern: "foo*bar", expectedStr: "StartsWith(Equals(\"foo\")) && EndsWith(Equals(\"bar\"))"},
   178  		testPattern{pattern: "*", expectedStr: "All"},
   179  	})
   180  
   181  	inputs := []testInput{
   182  		newTestInput("foo", true, false, false, false, false, true),
   183  		newTestInput("foobar", false, true, false, false, true, true),
   184  		newTestInput("foozapbar", false, true, false, false, true, true),
   185  		newTestInput("bazbar", false, true, true, false, false, true),
   186  		newTestInput("bazzzbar", false, true, true, false, false, true),
   187  		newTestInput("cat", false, false, false, true, false, true),
   188  		newTestInput("catbar", false, true, false, true, false, true),
   189  		newTestInput("baztestcat", false, false, true, true, false, true),
   190  		newTestInput("foocatbar", false, true, false, true, true, true),
   191  		newTestInput("footestcatbar", false, true, false, true, true, true),
   192  	}
   193  
   194  	for _, input := range inputs {
   195  		for i, expectedMatch := range input.matches {
   196  			require.Equal(t, expectedMatch, filters[i].Matches(input.val),
   197  				fmt.Sprintf("input: %s, pattern: %s", input.val, filters[i].String()))
   198  		}
   199  	}
   200  }
   201  
   202  func TestRangeFilters(t *testing.T) {
   203  	filters := genAndValidateFilters(t, []testPattern{
   204  		testPattern{pattern: "fo[a-zA-Z0-9]", expectedStr: "Equals(\"fo\") then Range(\"a-z || A-Z || 0-9\")"},
   205  		testPattern{pattern: "f[o-o]?", expectedStr: "Equals(\"f\") then Range(\"o-o\") then AnyChar"},
   206  		testPattern{pattern: "???", expectedStr: "AnyChar then AnyChar then AnyChar"},
   207  		testPattern{pattern: "ba?", expectedStr: "Equals(\"ba\") then AnyChar"},
   208  		testPattern{pattern: "[!cC]ar", expectedStr: "Not(Range(\"cC\")) then Equals(\"ar\")"},
   209  		testPattern{pattern: "ba?[0-9][!a-z]9", expectedStr: "Equals(\"ba\") then AnyChar then Range(\"0-9\") then Not(Range(\"a-z\")) then Equals(\"9\")"},
   210  		testPattern{pattern: "{ba,fo,car}*", expectedStr: "StartsWith(Range(\"ba,fo,car\"))"},
   211  		testPattern{pattern: "ba{r,t}*[!a-zA-Z]", expectedStr: "StartsWith(Equals(\"ba\") then Range(\"r,t\")) && EndsWith(Not(Range(\"a-z || A-Z\")))"},
   212  		testPattern{pattern: "*{9}", expectedStr: "EndsWith(Range(\"9\"))"},
   213  	})
   214  
   215  	inputs := []testInput{
   216  		newTestInput("foo", true, true, true, false, false, false, true, false, false),
   217  		newTestInput("fo!", false, true, true, false, false, false, true, false, false),
   218  		newTestInput("boo", false, false, true, false, false, false, false, false, false),
   219  		newTestInput("bar", false, false, true, true, true, false, true, false, false),
   220  		newTestInput("Bar", false, false, true, false, true, false, false, false, false),
   221  		newTestInput("car", false, false, true, false, false, false, true, false, false),
   222  		newTestInput("bar9", false, false, false, false, false, false, true, true, true),
   223  		newTestInput("bar990", false, false, false, false, false, false, true, true, false),
   224  		newTestInput("bar959", false, false, false, false, false, true, true, true, true),
   225  		newTestInput("bar009", false, false, false, false, false, true, true, true, true),
   226  		newTestInput("bat449", false, false, false, false, false, true, true, true, true),
   227  	}
   228  
   229  	for _, input := range inputs {
   230  		for i, expectedMatch := range input.matches {
   231  			require.Equal(t, expectedMatch, filters[i].Matches(input.val),
   232  				fmt.Sprintf("input: %s, pattern: %s", input.val, filters[i].String()))
   233  		}
   234  	}
   235  }
   236  
   237  func TestMultiFilter(t *testing.T) {
   238  	cf, _ := newContainsFilter([]byte("bar"))
   239  	filters := []Filter{
   240  		NewMultiFilter([]Filter{}, Conjunction),
   241  		NewMultiFilter([]Filter{}, Disjunction),
   242  		NewMultiFilter([]Filter{newEqualityFilter([]byte("foo"))}, Conjunction),
   243  		NewMultiFilter([]Filter{newEqualityFilter([]byte("foo"))}, Disjunction),
   244  		NewMultiFilter([]Filter{newEqualityFilter([]byte("foo")), cf}, Conjunction),
   245  		NewMultiFilter([]Filter{newEqualityFilter([]byte("foo")), cf}, Disjunction),
   246  	}
   247  
   248  	inputs := []testInput{
   249  		newTestInput("cat", true, true, false, false, false, false),
   250  		newTestInput("foo", true, true, true, true, false, true),
   251  		newTestInput("foobar", true, true, false, false, false, true),
   252  		newTestInput("bar", true, true, false, false, false, true),
   253  	}
   254  
   255  	for _, input := range inputs {
   256  		for i, expectedMatch := range input.matches {
   257  			require.Equal(t, expectedMatch, filters[i].Matches(input.val),
   258  				fmt.Sprintf("input: %s, pattern: %s", input.val, filters[i].String()))
   259  		}
   260  	}
   261  }
   262  
   263  func TestNegationFilter(t *testing.T) {
   264  	filters := genAndValidateFilters(t, []testPattern{
   265  		testPattern{pattern: "!foo", expectedStr: "Not(Equals(\"foo\"))"},
   266  		testPattern{pattern: "!*bar", expectedStr: "Not(EndsWith(Equals(\"bar\")))"},
   267  		testPattern{pattern: "!baz*", expectedStr: "Not(StartsWith(Equals(\"baz\")))"},
   268  		testPattern{pattern: "!*cat*", expectedStr: "Not(Contains(\"cat\"))"},
   269  		testPattern{pattern: "!foo*bar", expectedStr: "Not(StartsWith(Equals(\"foo\")) && EndsWith(Equals(\"bar\")))"},
   270  		testPattern{pattern: "foo!", expectedStr: "Equals(\"foo!\")"},
   271  	})
   272  
   273  	inputs := []testInput{
   274  		newTestInput("foo", false, true, true, true, true, false),
   275  		newTestInput("foo!", true, true, true, true, true, true),
   276  		newTestInput("foobar", true, false, true, true, false, false),
   277  		newTestInput("bazbar", true, false, false, true, true, false),
   278  		newTestInput("cat", true, true, true, false, true, false),
   279  		newTestInput("catbar", true, false, true, false, true, false),
   280  		newTestInput("baztestcat", true, true, false, false, true, false),
   281  		newTestInput("foocatbar", true, false, true, false, false, false),
   282  		newTestInput("footestcatbar", true, false, true, false, false, false),
   283  	}
   284  
   285  	for _, input := range inputs {
   286  		for i, expectedMatch := range input.matches {
   287  			require.Equal(t, expectedMatch, filters[i].Matches(input.val),
   288  				fmt.Sprintf("input: %s, pattern: %s", input.val, filters[i].String()))
   289  		}
   290  	}
   291  }
   292  
   293  func TestBadPatterns(t *testing.T) {
   294  	patterns := []string{
   295  		"!", // negation of nothing is everything, so user should use *.
   296  		"**",
   297  		"***",
   298  		"*too*many*",
   299  		"*too**many",
   300  		"to*o*many",
   301  		"to*o*ma*ny",
   302  		"abc[sdf",
   303  		"ab]c[sdf",
   304  		"abc[z-a]",
   305  		"*con[tT]ains*",
   306  		"*con{tT}ains*",
   307  		"*con?ains*",
   308  		"abc[a-zA-Z0-]",
   309  		"abc[a-zA-Z0]",
   310  		"abc[a-zZ-A]",
   311  		"ab}c{sdf",
   312  		"ab{}sdf",
   313  		"ab[]sdf",
   314  	}
   315  
   316  	for _, pattern := range patterns {
   317  		_, err := NewFilter([]byte(pattern))
   318  		require.Error(t, err, fmt.Sprintf("pattern: %s", pattern))
   319  	}
   320  }
   321  
   322  func TestMultiCharSequenceFilter(t *testing.T) {
   323  	_, err := newMultiCharSequenceFilter([]byte(""), false)
   324  	require.Error(t, err)
   325  
   326  	f, err := newMultiCharSequenceFilter([]byte("test2,test,tent,book"), false)
   327  	require.NoError(t, err)
   328  	validateLookup(t, f, "", false, "")
   329  	validateLookup(t, f, "t", false, "")
   330  	validateLookup(t, f, "tes", false, "")
   331  	validateLookup(t, f, "tset", false, "")
   332  
   333  	validateLookup(t, f, "test", true, "")
   334  	validateLookup(t, f, "tent", true, "")
   335  	validateLookup(t, f, "test3", true, "3")
   336  	validateLookup(t, f, "test2", true, "")
   337  	validateLookup(t, f, "book123", true, "123")
   338  
   339  	f, err = newMultiCharSequenceFilter([]byte("test2,test,tent,book"), true)
   340  	require.NoError(t, err)
   341  	validateLookup(t, f, "", false, "")
   342  	validateLookup(t, f, "t", false, "")
   343  	validateLookup(t, f, "tes", false, "")
   344  	validateLookup(t, f, "test3", false, "")
   345  
   346  	validateLookup(t, f, "test", true, "")
   347  	validateLookup(t, f, "tent", true, "")
   348  	validateLookup(t, f, "test2", true, "")
   349  	validateLookup(t, f, "12test", true, "12")
   350  	validateLookup(t, f, "12test2", true, "12")
   351  	validateLookup(t, f, "123book", true, "123")
   352  }
   353  
   354  func validateLookup(t *testing.T, f chainFilter, val string, expectedMatch bool, expectedRemainder string) {
   355  	remainder, match := f.matches([]byte(val))
   356  	require.Equal(t, expectedMatch, match)
   357  	require.Equal(t, expectedRemainder, string(remainder))
   358  }
   359  
   360  type mockFilterData struct {
   361  	val   string
   362  	match bool
   363  	err   error
   364  }
   365  
   366  type testPattern struct {
   367  	pattern     string
   368  	expectedStr string
   369  }
   370  
   371  type testInput struct {
   372  	val     []byte
   373  	matches []bool
   374  }
   375  
   376  func newTestInput(val string, matches ...bool) testInput {
   377  	return testInput{val: []byte(val), matches: matches}
   378  }
   379  
   380  func genAndValidateFilters(t *testing.T, patterns []testPattern) []Filter {
   381  	var err error
   382  	filters := make([]Filter, len(patterns))
   383  	for i, pattern := range patterns {
   384  		filters[i], err = NewFilter([]byte(pattern.pattern))
   385  		require.NoError(t, err, fmt.Sprintf("No error expected, but got: %v for pattern: %s", err, pattern.pattern))
   386  		require.Equal(t, pattern.expectedStr, filters[i].String())
   387  	}
   388  
   389  	return filters
   390  }