use std::iter::FusedIterator;
use crate::{FilterStatKey, GenericFilterConfig, GenericFiltersConfig, GenericFiltersMap};
use relay_protocol::{Getter, RuleCondition};
const MAX_SUPPORTED_VERSION: u16 = 1;
pub fn are_generic_filters_supported(
global_filters_version: Option<u16>,
project_filters_version: u16,
) -> bool {
global_filters_version.map_or(true, |v| v <= MAX_SUPPORTED_VERSION)
&& project_filters_version <= MAX_SUPPORTED_VERSION
}
fn matches<F: Getter>(item: &F, condition: Option<&RuleCondition>) -> bool {
condition.map_or(false, |condition| condition.matches(item))
}
pub(crate) fn should_filter<F: Getter>(
item: &F,
project_filters: &GenericFiltersConfig,
global_filters: Option<&GenericFiltersConfig>,
) -> Result<(), FilterStatKey> {
let filters = merge_generic_filters(
project_filters,
global_filters,
#[cfg(test)]
MAX_SUPPORTED_VERSION,
);
for filter_config in filters {
if filter_config.is_enabled && matches(item, filter_config.condition) {
return Err(FilterStatKey::GenericFilter(filter_config.id.to_owned()));
}
}
Ok(())
}
fn merge_generic_filters<'a>(
project: &'a GenericFiltersConfig,
global: Option<&'a GenericFiltersConfig>,
#[cfg(test)] max_supported_version: u16,
) -> impl Iterator<Item = GenericFilterConfigRef<'a>> {
#[cfg(not(test))]
let max_supported_version = MAX_SUPPORTED_VERSION;
let is_supported = project.version <= max_supported_version
&& global.map_or(true, |gf| gf.version <= max_supported_version);
is_supported
.then(|| {
DynamicGenericFiltersConfigIter::new(&project.filters, global.map(|gc| &gc.filters))
})
.into_iter()
.flatten()
}
struct DynamicGenericFiltersConfigIter<'a> {
project: &'a GenericFiltersMap,
project_index: usize,
global: Option<&'a GenericFiltersMap>,
global_index: usize,
}
impl<'a> DynamicGenericFiltersConfigIter<'a> {
pub fn new(project: &'a GenericFiltersMap, global: Option<&'a GenericFiltersMap>) -> Self {
DynamicGenericFiltersConfigIter {
project,
project_index: 0,
global,
global_index: 0,
}
}
}
impl<'a> Iterator for DynamicGenericFiltersConfigIter<'a> {
type Item = GenericFilterConfigRef<'a>;
fn next(&mut self) -> Option<Self::Item> {
if let Some((id, filter)) = self.project.get_index(self.project_index) {
self.project_index += 1;
let merged = merge_filters(filter, self.global.and_then(|gf| gf.get(id)));
return Some(merged);
}
loop {
let (id, filter) = self.global?.get_index(self.global_index)?;
self.global_index += 1;
if !self.project.contains_key(id) {
return Some(filter.into());
}
}
}
}
impl FusedIterator for DynamicGenericFiltersConfigIter<'_> {}
fn merge_filters<'a>(
primary: &'a GenericFilterConfig,
secondary: Option<&'a GenericFilterConfig>,
) -> GenericFilterConfigRef<'a> {
GenericFilterConfigRef {
id: primary.id.as_str(),
is_enabled: primary.is_enabled,
condition: primary
.condition
.as_ref()
.or(secondary.and_then(|filter| filter.condition.as_ref())),
}
}
#[derive(Debug, Default, PartialEq)]
struct GenericFilterConfigRef<'a> {
id: &'a str,
is_enabled: bool,
condition: Option<&'a RuleCondition>,
}
impl<'a> From<&'a GenericFilterConfig> for GenericFilterConfigRef<'a> {
fn from(value: &'a GenericFilterConfig) -> Self {
GenericFilterConfigRef {
id: value.id.as_str(),
is_enabled: value.is_enabled,
condition: value.condition.as_ref(),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use relay_event_schema::protocol::{Event, LenientString};
use relay_protocol::Annotated;
fn mock_filters() -> GenericFiltersMap {
vec![
GenericFilterConfig {
id: "firstReleases".to_string(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.release", "1.0")),
},
GenericFilterConfig {
id: "helloTransactions".to_string(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.transaction", "/hello")),
},
]
.into()
}
#[test]
fn test_should_filter_match_rules() {
let config = GenericFiltersConfig {
version: 1,
filters: mock_filters(),
};
let event = Event {
release: Annotated::new(LenientString("1.0".to_string())),
..Default::default()
};
assert_eq!(
should_filter(&event, &config, None),
Err(FilterStatKey::GenericFilter("firstReleases".to_string()))
);
let event = Event {
transaction: Annotated::new("/hello".to_string()),
..Default::default()
};
assert_eq!(
should_filter(&event, &config, None),
Err(FilterStatKey::GenericFilter(
"helloTransactions".to_string()
))
);
}
#[test]
fn test_should_filter_fifo_match_rules() {
let config = GenericFiltersConfig {
version: 1,
filters: mock_filters(),
};
let event = Event {
release: Annotated::new(LenientString("1.0".to_string())),
transaction: Annotated::new("/hello".to_string()),
..Default::default()
};
assert_eq!(
should_filter(&event, &config, None),
Err(FilterStatKey::GenericFilter("firstReleases".to_string()))
);
}
#[test]
fn test_should_filter_match_no_rules() {
let config = GenericFiltersConfig {
version: 1,
filters: mock_filters(),
};
let event = Event {
transaction: Annotated::new("/world".to_string()),
..Default::default()
};
assert_eq!(should_filter(&event, &config, None), Ok(()));
}
#[test]
fn test_should_filter_with_higher_config_version() {
let config = GenericFiltersConfig {
version: MAX_SUPPORTED_VERSION + 1,
filters: mock_filters(),
};
let event = Event {
release: Annotated::new(LenientString("1.0".to_string())),
transaction: Annotated::new("/hello".to_string()),
..Default::default()
};
assert_eq!(should_filter(&event, &config, None), Ok(()));
}
#[test]
fn test_should_filter_from_global_filters() {
let project = GenericFiltersConfig {
version: 1,
filters: vec![GenericFilterConfig {
id: "firstReleases".to_string(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.release", "1.0")),
}]
.into(),
};
let global = GenericFiltersConfig {
version: 1,
filters: vec![GenericFilterConfig {
id: "helloTransactions".to_string(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.transaction", "/hello")),
}]
.into(),
};
let event = Event {
transaction: Annotated::new("/hello".to_string()),
..Default::default()
};
assert_eq!(
should_filter(&event, &project, Some(&global)),
Err(FilterStatKey::GenericFilter(
"helloTransactions".to_string()
))
);
}
fn empty_filter() -> GenericFiltersConfig {
GenericFiltersConfig {
version: 1,
filters: GenericFiltersMap::new(),
}
}
fn enabled_filter(id: &str) -> GenericFiltersConfig {
GenericFiltersConfig {
version: 1,
filters: vec![GenericFilterConfig {
id: id.to_owned(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.exceptions", "myError")),
}]
.into(),
}
}
fn enabled_flag_filter(id: &str) -> GenericFiltersConfig {
GenericFiltersConfig {
version: 1,
filters: vec![GenericFilterConfig {
id: id.to_owned(),
is_enabled: true,
condition: None,
}]
.into(),
}
}
fn disabled_filter(id: &str) -> GenericFiltersConfig {
GenericFiltersConfig {
version: 1,
filters: vec![GenericFilterConfig {
id: id.to_owned(),
is_enabled: false,
condition: Some(RuleCondition::eq("event.exceptions", "myError")),
}]
.into(),
}
}
fn disabled_flag_filter(id: &str) -> GenericFiltersConfig {
GenericFiltersConfig {
version: 1,
filters: vec![GenericFilterConfig {
id: id.to_owned(),
is_enabled: false,
condition: None,
}]
.into(),
}
}
#[test]
fn test_no_combined_when_unsupported_project_version() {
let mut project = enabled_filter("unsupported-project");
project.version = 2;
let global = enabled_filter("supported-global");
assert!(merge_generic_filters(&project, Some(&global), 1).eq(None.into_iter()));
}
#[test]
fn test_no_combined_when_unsupported_project_version_no_global() {
let mut project = enabled_filter("unsupported-project");
project.version = 2;
assert!(merge_generic_filters(&project, None, 1).eq(None.into_iter()));
}
#[test]
fn test_no_combined_when_unsupported_global_version() {
let project = enabled_filter("supported-project");
let mut global = enabled_filter("unsupported-global");
global.version = 2;
assert!(merge_generic_filters(&project, Some(&global), 1).eq(None.into_iter()));
}
#[test]
fn test_no_combined_when_unsupported_project_and_global_version() {
let mut project = enabled_filter("unsupported-project");
project.version = 2;
let mut global = enabled_filter("unsupported-global");
global.version = 2;
assert!(merge_generic_filters(&project, Some(&global), 1).eq(None.into_iter()));
}
#[test]
fn test_both_combined_when_supported_project_and_global_version() {
let project = enabled_filter("supported-project");
let global = enabled_filter("supported-global");
assert!(merge_generic_filters(&project, Some(&global), 1).eq([
project.filters.first().unwrap().1.into(),
global.filters.first().unwrap().1.into()
]
.into_iter()));
}
#[test]
fn test_project_combined_when_duplicated_filter_project_and_global() {
let project = enabled_filter("filter");
let global = enabled_filter("filter");
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter())
);
}
#[test]
fn test_no_combined_when_empty_project_and_global() {
let project = empty_filter();
let global = empty_filter();
assert!(merge_generic_filters(&project, Some(&global), 1).eq(None.into_iter()));
}
#[test]
fn test_global_combined_when_empty_project_disabled_global_filter() {
let project = empty_filter();
let global = disabled_filter("disabled-global");
assert!(merge_generic_filters(&project, Some(&global), 1).eq([global
.filters
.first()
.unwrap()
.1
.into()]
.into_iter()));
}
#[test]
fn test_global_combined_when_empty_project_enabled_global_filters() {
let project = empty_filter();
let global = enabled_filter("enabled-global");
assert!(merge_generic_filters(&project, Some(&global), 1).eq([global
.filters
.first()
.unwrap()
.1
.into()]
.into_iter()));
}
#[test]
fn test_global_combined_when_empty_project_enabled_flag_global() {
let project = empty_filter();
let global = enabled_flag_filter("skip");
assert!(merge_generic_filters(&project, Some(&global), 1).eq([global
.filters
.first()
.unwrap()
.1
.into()]));
}
#[test]
fn test_project_combined_when_disabled_project_empty_global() {
let project = disabled_filter("disabled-project");
let global = empty_filter();
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter())
);
}
#[test]
fn test_project_combined_when_disabled_project_missing_global() {
let project = disabled_filter("disabled-project");
assert!(merge_generic_filters(&project, None, 1).eq([project
.filters
.first()
.unwrap()
.1
.into(),]
.into_iter()));
}
#[test]
fn test_both_combined_when_different_disabled_project_and_global() {
let project = disabled_filter("disabled-project");
let global = disabled_filter("disabled-global");
assert!(merge_generic_filters(&project, Some(&global), 1).eq([
project.filters.first().unwrap().1.into(),
global.filters.first().unwrap().1.into()
]));
}
#[test]
fn test_project_combined_duplicated_disabled_project_and_global() {
let project = disabled_filter("filter");
let global = disabled_filter("filter");
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()])
);
}
#[test]
fn test_merged_combined_when_disabled_project_enabled_global() {
let project = disabled_filter("filter");
let global = enabled_filter("filter");
let expected = &GenericFilterConfig {
id: "filter".to_owned(),
is_enabled: false,
condition: global.filters.first().unwrap().1.condition.clone(),
};
assert!(merge_generic_filters(&project, Some(&global), 1).eq([expected.into()].into_iter()));
}
#[test]
fn test_no_combined_when_enabled_flag_project_empty_global() {
let project = enabled_flag_filter("filter");
let global = empty_filter();
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter())
);
}
#[test]
fn test_project_combined_when_enabled_flag_project_missing_global() {
let project = enabled_flag_filter("filter");
assert!(merge_generic_filters(&project, None, 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter()));
}
#[test]
fn test_project_combined_when_disabled_flag_project_empty_global() {
let project = disabled_flag_filter("filter");
let global = empty_filter();
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()])
);
}
#[test]
fn test_project_combined_when_disabled_flag_project_missing_global() {
let project = disabled_flag_filter("filter");
assert!(merge_generic_filters(&project, None, 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]));
}
#[test]
fn test_project_combined_when_enabled_project_empty_global() {
let project = enabled_filter("enabled-project");
let global = empty_filter();
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter())
);
}
#[test]
fn test_project_combined_when_enabled_project_missing_global() {
let project = enabled_filter("enabled-project");
assert!(merge_generic_filters(&project, None, 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter()));
}
#[test]
fn test_merged_combined_when_enabled_flag_project_disabled_global() {
let project = enabled_flag_filter("filter");
let global = disabled_filter("filter");
let expected = &GenericFilterConfig {
id: "filter".to_owned(),
is_enabled: true,
condition: global.filters.first().unwrap().1.condition.clone(),
};
assert!(merge_generic_filters(&project, Some(&global), 1).eq([expected.into()].into_iter()));
}
#[test]
fn test_global_combined_when_disabled_flag_project_disabled_global() {
let project = disabled_flag_filter("filter");
let global = disabled_filter("filter");
assert!(merge_generic_filters(&project, Some(&global), 1).eq([global
.filters
.first()
.unwrap()
.1
.into()]));
}
#[test]
fn test_project_combined_when_enabled_project_disabled_global() {
let project = enabled_filter("filter");
let global = disabled_filter("filter");
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter())
);
}
#[test]
fn test_global_combined_when_enabled_flag_project_enabled_global() {
let project = enabled_flag_filter("filter");
let global = enabled_filter("filter");
assert!(merge_generic_filters(&project, Some(&global), 1).eq([global
.filters
.first()
.unwrap()
.1
.into()]
.into_iter()));
}
#[test]
fn test_merged_combined_when_disabled_flag_project_enabled_global() {
let project = disabled_flag_filter("filter");
let global = enabled_filter("filter");
let expected = &GenericFilterConfig {
id: "filter".to_owned(),
is_enabled: false,
condition: global.filters.first().unwrap().1.condition.clone(),
};
assert!(merge_generic_filters(&project, Some(&global), 1).eq([expected.into()].into_iter()));
}
#[test]
fn test_project_combined_when_enabled_project_enabled_flag_global() {
let project = enabled_filter("filter");
let global = enabled_flag_filter("filter");
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter())
);
}
#[test]
fn test_project_combined_when_enabled_flags_project_and_global() {
let project = enabled_flag_filter("filter");
let global = enabled_flag_filter("filter");
assert!(
merge_generic_filters(&project, Some(&global), 1).eq([project
.filters
.first()
.unwrap()
.1
.into()]
.into_iter())
);
}
#[test]
fn test_multiple_combined_filters() {
let project = GenericFiltersConfig {
version: 1,
filters: vec![
GenericFilterConfig {
id: "0".to_owned(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.exceptions", "myError")),
},
GenericFilterConfig {
id: "1".to_owned(),
is_enabled: true,
condition: None,
},
GenericFilterConfig {
id: "2".to_owned(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.exceptions", "myError")),
},
]
.into(),
};
let global = GenericFiltersConfig {
version: 1,
filters: vec![
GenericFilterConfig {
id: "1".to_owned(),
is_enabled: false,
condition: Some(RuleCondition::eq("event.exceptions", "myOtherError")),
},
GenericFilterConfig {
id: "3".to_owned(),
is_enabled: false,
condition: Some(RuleCondition::eq("event.exceptions", "myLastError")),
},
]
.into(),
};
let expected0 = &project.filters[0];
let expected1 = &GenericFilterConfig {
id: "1".to_owned(),
is_enabled: true,
condition: Some(RuleCondition::eq("event.exceptions", "myOtherError")),
};
let expected2 = &project.filters[2];
let expected3 = &global.filters[1];
assert!(merge_generic_filters(&project, Some(&global), 1).eq([
expected0.into(),
expected1.into(),
expected2.into(),
expected3.into()
]
.into_iter()));
}
}