From e077d174f5b27d73445ecd5cdcc29ca751b8a933 Mon Sep 17 00:00:00 2001 From: Joel Takvorian Date: Tue, 20 Feb 2024 16:05:26 +0100 Subject: [PATCH] NETOBSERV-1425: enhance metrics filters - New filters: exact_not, regex_not - Variable interpolation: e.g. "api.MetricsFilter{Key: "src-ns", Value: "$(dst-ns)"}" to filter for intra-namespace traffic --- docs/api.md | 24 ++++++--- pkg/api/api.go | 2 + pkg/api/encode_prom.go | 6 ++- pkg/pipeline/encode/encode_prom_metric.go | 53 ++++++++++++++++-- pkg/pipeline/encode/encode_prom_test.go | 66 +++++++++++++++++++++++ 5 files changed, 138 insertions(+), 13 deletions(-) diff --git a/docs/api.md b/docs/api.md index d45a2f5e7..69d4e87d8 100644 --- a/docs/api.md +++ b/docs/api.md @@ -20,19 +20,23 @@ Following is the supported API format for prometheus encode: filter: an optional criterion to filter entries by. Deprecated: use filters instead. key: the key to match and filter by value: the value to match and filter by - type: (enum) the type of filter match: exact (default), presence, absence or regex - exact: match exactly the provided fitler value + type: (enum) the type of filter match: exact (default), exact_not, presence, absence, regex or regex_not + exact: match exactly the provided filter value + exact_not: the value must be different from the provided filter presence: filter key must be present (filter value is ignored) absence: filter key must be absent (filter value is ignored) regex: match filter value as a regular expression + regex_not: the filter value must not match the provided regular expression filters: a list of criteria to filter entries by key: the key to match and filter by value: the value to match and filter by - type: (enum) the type of filter match: exact (default), presence, absence or regex - exact: match exactly the provided fitler value + type: (enum) the type of filter match: exact (default), exact_not, presence, absence, regex or regex_not + exact: match exactly the provided filter value + exact_not: the value must be different from the provided filter presence: filter key must be present (filter value is ignored) absence: filter key must be absent (filter value is ignored) regex: match filter value as a regular expression + regex_not: the filter value must not match the provided regular expression valueKey: entry key from which to resolve metric value labels: labels to be associated with the metric buckets: histogram buckets @@ -353,19 +357,23 @@ Following is the supported API format for writing metrics to an OpenTelemetry co filter: an optional criterion to filter entries by. Deprecated: use filters instead. key: the key to match and filter by value: the value to match and filter by - type: (enum) the type of filter match: exact (default), presence, absence or regex - exact: match exactly the provided fitler value + type: (enum) the type of filter match: exact (default), exact_not, presence, absence, regex or regex_not + exact: match exactly the provided filter value + exact_not: the value must be different from the provided filter presence: filter key must be present (filter value is ignored) absence: filter key must be absent (filter value is ignored) regex: match filter value as a regular expression + regex_not: the filter value must not match the provided regular expression filters: a list of criteria to filter entries by key: the key to match and filter by value: the value to match and filter by - type: (enum) the type of filter match: exact (default), presence, absence or regex - exact: match exactly the provided fitler value + type: (enum) the type of filter match: exact (default), exact_not, presence, absence, regex or regex_not + exact: match exactly the provided filter value + exact_not: the value must be different from the provided filter presence: filter key must be present (filter value is ignored) absence: filter key must be absent (filter value is ignored) regex: match filter value as a regular expression + regex_not: the filter value must not match the provided regular expression valueKey: entry key from which to resolve metric value labels: labels to be associated with the metric buckets: histogram buckets diff --git a/pkg/api/api.go b/pkg/api/api.go index 0bbc0d725..73eaea282 100644 --- a/pkg/api/api.go +++ b/pkg/api/api.go @@ -51,9 +51,11 @@ const ( AddKubernetesInfraRuleType = "add_kubernetes_infra" ReinterpretDirectionRuleType = "reinterpret_direction" PromFilterExact = "exact" + PromFilterExactNot = "exact_not" PromFilterPresence = "presence" PromFilterAbsence = "absence" PromFilterRegex = "regex" + PromFilterRegexNot = "regex_not" TagYaml = "yaml" TagDoc = "doc" diff --git a/pkg/api/encode_prom.go b/pkg/api/encode_prom.go index 07213d613..c2a46c2f2 100644 --- a/pkg/api/encode_prom.go +++ b/pkg/api/encode_prom.go @@ -70,14 +70,16 @@ type MetricsItems []MetricsItem type MetricsFilter struct { Key string `yaml:"key" json:"key" doc:"the key to match and filter by"` Value string `yaml:"value" json:"value" doc:"the value to match and filter by"` - Type string `yaml:"type" json:"type" enum:"MetricEncodeFilterTypeEnum" doc:"the type of filter match: exact (default), presence, absence or regex"` + Type string `yaml:"type" json:"type" enum:"MetricEncodeFilterTypeEnum" doc:"the type of filter match: exact (default), exact_not, presence, absence, regex or regex_not"` } type MetricEncodeFilterTypeEnum struct { - Exact string `yaml:"exact" json:"exact" doc:"match exactly the provided fitler value"` + Exact string `yaml:"exact" json:"exact" doc:"match exactly the provided filter value"` + ExactNot string `yaml:"exact_not" json:"exact_not" doc:"the value must be different from the provided filter"` Presence string `yaml:"presence" json:"presence" doc:"filter key must be present (filter value is ignored)"` Absence string `yaml:"absence" json:"absence" doc:"filter key must be absent (filter value is ignored)"` Regex string `yaml:"regex" json:"regex" doc:"match filter value as a regular expression"` + RegexNot string `yaml:"regex_not" json:"regex_not" doc:"the filter value must not match the provided regular expression"` } func MetricEncodeFilterTypeName(t string) string { diff --git a/pkg/pipeline/encode/encode_prom_metric.go b/pkg/pipeline/encode/encode_prom_metric.go index 15fe7854e..2a9eb230e 100644 --- a/pkg/pipeline/encode/encode_prom_metric.go +++ b/pkg/pipeline/encode/encode_prom_metric.go @@ -3,6 +3,7 @@ package encode import ( "fmt" "regexp" + "strings" "github.com/netobserv/flowlogs-pipeline/pkg/api" "github.com/netobserv/flowlogs-pipeline/pkg/config" @@ -10,6 +11,8 @@ import ( type Predicate func(flow config.GenericMap) bool +var variableExtractor, _ = regexp.Compile(`\$\(([^\)]+)\)`) + type MetricInfo struct { api.MetricsItem FilterPredicates []Predicate @@ -30,19 +33,29 @@ func Absence(filter api.MetricsFilter) Predicate { } func Exact(filter api.MetricsFilter) Predicate { + varLookups := extractVarLookups(filter.Value) return func(flow config.GenericMap) bool { if val, found := flow[filter.Key]; found { sVal, ok := val.(string) if !ok { sVal = fmt.Sprint(val) } - return sVal == filter.Value + value := filter.Value + if len(varLookups) > 0 { + value = injectVars(flow, value, varLookups) + } + return sVal == value } return false } } -func regex(filter api.MetricsFilter) Predicate { +func ExactNot(filter api.MetricsFilter) Predicate { + pred := Exact(filter) + return func(flow config.GenericMap) bool { return !pred(flow) } +} + +func Regex(filter api.MetricsFilter) Predicate { r, _ := regexp.Compile(filter.Value) return func(flow config.GenericMap) bool { if val, found := flow[filter.Key]; found { @@ -56,21 +69,55 @@ func regex(filter api.MetricsFilter) Predicate { } } +func RegexNot(filter api.MetricsFilter) Predicate { + pred := Regex(filter) + return func(flow config.GenericMap) bool { return !pred(flow) } +} + func filterToPredicate(filter api.MetricsFilter) Predicate { switch filter.Type { case api.PromFilterExact: return Exact(filter) + case api.PromFilterExactNot: + return ExactNot(filter) case api.PromFilterPresence: return Presence(filter) case api.PromFilterAbsence: return Absence(filter) case api.PromFilterRegex: - return regex(filter) + return Regex(filter) + case api.PromFilterRegexNot: + return RegexNot(filter) } // Default = Exact return Exact(filter) } +func extractVarLookups(value string) [][]string { + // Extract list of variables to lookup + // E.g: filter "$(SrcAddr):$(SrcPort)" would return [SrcAddr,SrcPort] + if len(value) > 0 { + return variableExtractor.FindAllStringSubmatch(value, -1) + } + return nil +} + +func injectVars(flow config.GenericMap, filterValue string, varLookups [][]string) string { + injected := filterValue + for _, matchGroup := range varLookups { + var value string + if rawVal, found := flow[matchGroup[1]]; found { + if sVal, ok := rawVal.(string); ok { + value = sVal + } else { + value = fmt.Sprint(rawVal) + } + } + injected = strings.ReplaceAll(injected, matchGroup[0], value) + } + return injected +} + func CreateMetricInfo(def api.MetricsItem) *MetricInfo { mi := MetricInfo{ MetricsItem: def, diff --git a/pkg/pipeline/encode/encode_prom_test.go b/pkg/pipeline/encode/encode_prom_test.go index 6a2f81a83..2798347f5 100644 --- a/pkg/pipeline/encode/encode_prom_test.go +++ b/pkg/pipeline/encode/encode_prom_test.go @@ -380,6 +380,63 @@ func Test_FilterDirection(t *testing.T) { require.Contains(t, exposed, `test_ingress_or_inner_packets_total 1010`) } +func Test_FilterSameOrDifferentNamespace(t *testing.T) { + metrics := []config.GenericMap{ + { + "src-ns": "a", + "dst-ns": "b", + "packets": 10, + }, + { + "src-ns": "b", + "dst-ns": "a", + "packets": 100, + }, + { + "src-ns": "a", + "dst-ns": "a", + "packets": 1000, + }, + { + "src-ns": "b", + "dst-ns": "b", + "packets": 10000, + }, + } + params := api.PromEncode{ + Prefix: "test_", + ExpiryTime: api.Duration{ + Duration: time.Duration(60 * time.Second), + }, + Metrics: []api.MetricsItem{ + { + Name: "packets_same_namespace_total", + Type: "counter", + ValueKey: "packets", + Filters: []api.MetricsFilter{{Key: "src-ns", Value: "$(dst-ns)"}}, + }, + { + Name: "packets_different_namespace_total", + Type: "counter", + ValueKey: "packets", + Filters: []api.MetricsFilter{{Key: "src-ns", Type: "exact_not", Value: "$(dst-ns)"}}, + }, + }, + } + + encodeProm, err := initProm(¶ms) + require.NoError(t, err) + for _, metric := range metrics { + encodeProm.Encode(metric) + } + time.Sleep(100 * time.Millisecond) + + exposed := test.ReadExposedMetrics(t) + + require.Contains(t, exposed, `test_packets_same_namespace_total 11000`) + require.Contains(t, exposed, `test_packets_different_namespace_total 110`) +} + func Test_ValueScale(t *testing.T) { metrics := []config.GenericMap{{"rtt": 15_000_000} /*15ms*/, {"rtt": 110_000_000} /*110ms*/} params := api.PromEncode{ @@ -650,3 +707,12 @@ func Test_MultipleProm(t *testing.T) { // TODO: Add test for different addresses, but need to deal with StartPromServer (ListenAndServe) } + +func Test_Filters_extractVarLookups(t *testing.T) { + variables := extractVarLookups("$(abc)--$(def)") + + require.Equal(t, [][]string{{"$(abc)", "abc"}, {"$(def)", "def"}}, variables) + + variables = extractVarLookups("") + require.Empty(t, variables) +}