vector_core/config/
metrics_expiration.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
use vector_config::configurable_component;

/// Per metric set expiration options.
#[configurable_component]
#[derive(Clone, Debug, PartialEq, Default)]
pub struct PerMetricSetExpiration {
    /// Metric name to apply this expiration to. Ignores metric name if not defined.
    #[serde(default, skip_serializing_if = "crate::serde::is_default")]
    pub name: Option<MetricNameMatcherConfig>,
    /// Labels to apply this expiration to. Ignores labels if not defined.
    #[serde(default, skip_serializing_if = "crate::serde::is_default")]
    #[configurable(metadata(
        docs::enum_tag_field = "type",
        docs::enum_tagging = "internal",
        docs::enum_tag_description = "Metric label matcher type."
    ))]
    pub labels: Option<MetricLabelMatcherConfig>,
    /// The amount of time, in seconds, that internal metrics will persist after having not been
    /// updated before they expire and are removed.
    ///
    /// Set this to a value larger than your `internal_metrics` scrape interval (default 5 minutes)
    /// so that metrics live long enough to be emitted and captured.
    #[configurable(metadata(docs::examples = 60.0))]
    pub expire_secs: f64,
}

/// Configuration for metric name matcher.
#[configurable_component]
#[derive(Clone, Debug, PartialEq)]
#[serde(tag = "type", rename_all = "snake_case")]
#[configurable(metadata(docs::enum_tag_description = "Metric name matcher type."))]
pub enum MetricNameMatcherConfig {
    /// Only considers exact name matches.
    Exact {
        /// The exact metric name.
        value: String,
    },
    /// Compares metric name to the provided pattern.
    Regex {
        /// Pattern to compare to.
        pattern: String,
    },
}

/// Configuration for metric labels matcher.
#[configurable_component]
#[derive(Clone, Debug, PartialEq)]
#[serde(tag = "type", rename_all = "snake_case")]
#[configurable(metadata(docs::enum_tag_description = "Metric label matcher type."))]
pub enum MetricLabelMatcher {
    /// Looks for an exact match of one label key value pair.
    Exact {
        /// Metric key to look for.
        key: String,
        /// The exact metric label value.
        value: String,
    },
    /// Compares label value with given key to the provided pattern.
    Regex {
        /// Metric key to look for.
        key: String,
        /// Pattern to compare metric label value to.
        value_pattern: String,
    },
}

/// Configuration for metric labels matcher group.
#[configurable_component]
#[derive(Clone, Debug, PartialEq)]
#[serde(tag = "type", rename_all = "snake_case")]
#[configurable(metadata(docs::enum_tag_description = "Metric label group matcher type."))]
pub enum MetricLabelMatcherConfig {
    /// Checks that any of the provided matchers can be applied to given metric.
    Any {
        /// List of matchers to check.
        matchers: Vec<MetricLabelMatcher>,
    },
    /// Checks that all of the provided matchers can be applied to given metric.
    All {
        /// List of matchers to check.
        matchers: Vec<MetricLabelMatcher>,
    },
}

/// Tests to confirm complex examples configuration
#[cfg(test)]
mod tests {
    use vrl::prelude::indoc;

    use super::*;

    #[test]
    fn just_expiration_config() {
        // This configuration should maybe be treated as invalid - because it turns into a global
        // configuration, matching every metric
        let config = serde_yaml::from_str::<PerMetricSetExpiration>(indoc! {r"
            expire_secs: 10.0
            "})
        .unwrap();

        assert!(config.name.is_none());
        assert!(config.labels.is_none());
        assert_eq!(10.0, config.expire_secs);
    }

    #[test]
    fn simple_name_config() {
        let config = serde_yaml::from_str::<PerMetricSetExpiration>(indoc! {r#"
            name:
                type: "exact"
                value: "test_metric"
            expire_secs: 1.0
            "#})
        .unwrap();

        if let Some(MetricNameMatcherConfig::Exact { value }) = config.name {
            assert_eq!("test_metric", value);
        } else {
            panic!("Expected exact name matcher");
        }
        assert!(config.labels.is_none());
        assert_eq!(1.0, config.expire_secs);
    }

    #[test]
    fn simple_labels_config() {
        let config = serde_yaml::from_str::<PerMetricSetExpiration>(indoc! {r#"
            labels:
                type: "all"
                matchers :
                    - type: "exact"
                      key: "test_metric_label"
                      value: "test_value"
            expire_secs: 1.0
            "#})
        .unwrap();

        if let Some(MetricLabelMatcherConfig::All { matchers }) = config.labels {
            if let MetricLabelMatcher::Exact { key, value } = &matchers[0] {
                assert_eq!("test_metric_label", key);
                assert_eq!("test_value", value);
            } else {
                panic!("Expected exact metric matcher");
            }
        } else {
            panic!("Expected all matcher");
        }
        assert!(config.name.is_none());
        assert_eq!(1.0, config.expire_secs);
    }

    #[test]
    fn complex_config() {
        let config = serde_yaml::from_str::<PerMetricSetExpiration>(indoc! {r#"
            name:
                type: "regex"
                pattern: "test_metric.*"
            labels:
                type: "all"
                matchers:
                  - type: "exact"
                    key: "component_kind"
                    value: "sink"
                  - type: "exact"
                    key: "component_kind"
                    value: "source"
            expire_secs: 1.0
            "#})
        .unwrap();

        if let Some(MetricNameMatcherConfig::Regex { ref pattern }) = config.name {
            assert_eq!("test_metric.*", pattern);
        } else {
            panic!("Expected regex name matcher");
        }

        let Some(MetricLabelMatcherConfig::All {
            matchers: all_matchers,
        }) = config.labels
        else {
            panic!("Expected all label matcher");
        };
        assert_eq!(2, all_matchers.len());

        let MetricLabelMatcher::Exact { key, value } = &all_matchers[0] else {
            panic!("Expected first label matcher to be exact matcher");
        };
        assert_eq!("component_kind", key);
        assert_eq!("sink", value);
        let MetricLabelMatcher::Exact { key, value } = &all_matchers[1] else {
            panic!("Expected second label matcher to be exact matcher");
        };
        assert_eq!("component_kind", key);
        assert_eq!("source", value);
    }
}