vector_core/config/
metrics_expiration.rsuse vector_config::configurable_component;
#[configurable_component]
#[derive(Clone, Debug, PartialEq, Default)]
pub struct PerMetricSetExpiration {
#[serde(default, skip_serializing_if = "crate::serde::is_default")]
pub name: Option<MetricNameMatcherConfig>,
#[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>,
#[configurable(metadata(docs::examples = 60.0))]
pub expire_secs: f64,
}
#[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 {
Exact {
value: String,
},
Regex {
pattern: String,
},
}
#[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 {
Exact {
key: String,
value: String,
},
Regex {
key: String,
value_pattern: String,
},
}
#[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 {
Any {
matchers: Vec<MetricLabelMatcher>,
},
All {
matchers: Vec<MetricLabelMatcher>,
},
}
#[cfg(test)]
mod tests {
use vrl::prelude::indoc;
use super::*;
#[test]
fn just_expiration_config() {
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);
}
}