Skip to content

Commit

Permalink
Enforce the rule: heuristics should never add a new view that would b…
Browse files Browse the repository at this point in the history
…e completely covered by an existing view (#5164)

### What
- Resolves: #5152

This evaluates the queries of newly added space-views and doesn't add
them if there already exists a space-view that would contain the same
contents. This eliminates most of the worst offenders of
unneeded-space-view creation, though re-loading recordings can still
cause spurious root-space-views since those wouldn't technically be
redundant.

Because this check operates on the query expressions it's independent of
the actual state of the entity-tree.

### Checklist
* [x] I have read and agree to [Contributor
Guide](https://github.com/rerun-io/rerun/blob/main/CONTRIBUTING.md) and
the [Code of
Conduct](https://github.com/rerun-io/rerun/blob/main/CODE_OF_CONDUCT.md)
* [x] I've included a screenshot or gif (if applicable)
* [x] I have tested the web demo (if applicable):
* Using newly built examples:
[app.rerun.io](https://app.rerun.io/pr/5164/index.html)
* Using examples from latest `main` build:
[app.rerun.io](https://app.rerun.io/pr/5164/index.html?manifest_url=https://app.rerun.io/version/main/examples_manifest.json)
* Using full set of examples from `nightly` build:
[app.rerun.io](https://app.rerun.io/pr/5164/index.html?manifest_url=https://app.rerun.io/version/nightly/examples_manifest.json)
* [x] The PR title and labels are set such as to maximize their
usefulness for the next release's CHANGELOG
* [x] If applicable, add a new check to the [release
checklist](https://github.com/rerun-io/rerun/blob/main/tests/python/release_checklist)!

- [PR Build Summary](https://build.rerun.io/pr/5164)
- [Docs
preview](https://rerun.io/preview/e35bd2e974540f0b3b2b9c2759f64dbfa635d515/docs)
<!--DOCS-PREVIEW-->
- [Examples
preview](https://rerun.io/preview/e35bd2e974540f0b3b2b9c2759f64dbfa635d515/examples)
<!--EXAMPLES-PREVIEW-->
- [Recent benchmark results](https://build.rerun.io/graphs/crates.html)
- [Wasm size tracking](https://build.rerun.io/graphs/sizes.html)

---------

Co-authored-by: Emil Ernerfeldt <[email protected]>
Co-authored-by: Andreas Reich <[email protected]>
  • Loading branch information
3 people committed Feb 12, 2024
1 parent fce62bb commit 40b43ad
Show file tree
Hide file tree
Showing 4 changed files with 280 additions and 90 deletions.
320 changes: 239 additions & 81 deletions crates/re_log_types/src/path/entity_path_filter.rs
Original file line number Diff line number Diff line change
Expand Up @@ -259,6 +259,74 @@ impl EntityPathFilter {

false // no matching rule and we are exclude-by-default.
}

/// Checks whether this results of this filter "fully contain" the results of another filter.
///
/// If this returns `true` there should not exist any [`EntityPath`] for which [`Self::is_included`]
/// would return `true` and the other filter would return `false` for this filter.
///
/// This check operates purely on the rule expressions and not the actual entity tree,
/// and will thus not reason about entities included in an actual recording:
/// different queries that are *not* a superset of each other may still query the same entities
/// in given recording.
///
/// This is a conservative estimate, and may return `false` in situations where the
/// query does in fact cover the other query. However, it should never return `true`
/// in a case where the other query would not be fully covered.
pub fn is_superset_of(&self, other: &Self) -> bool {
// First check that we include everything included by other
for (other_rule, other_effect) in &other.rules {
match other_effect {
RuleEffect::Include => {
if let Some((self_rule, self_effect)) = self
.rules
.iter()
.rev()
.find(|(r, _)| r.matches(&other_rule.path))
{
match self_effect {
RuleEffect::Include => {
// If the other rule includes the subtree, but the matching
// rule doesn't, then we don't fully contain the other rule.
if other_rule.include_subtree && !self_rule.include_subtree {
return false;
}
}
RuleEffect::Exclude => return false,
}
} else {
// No matching rule means this path isn't included
return false;
}
}
RuleEffect::Exclude => {}
}
}
// Next check that the other rule hasn't included something that we've excluded
for (self_rule, self_effect) in &self.rules {
match self_effect {
RuleEffect::Include => {}
RuleEffect::Exclude => {
if let Some((_, other_effect)) = other
.rules
.iter()
.rev()
.find(|(r, _)| r.matches(&self_rule.path))
{
match other_effect {
RuleEffect::Include => {
return false;
}
RuleEffect::Exclude => {}
}
}
}
}
}
// If we got here, we checked every inclusion rule in `other` and they all had a more-inclusive
// inclusion rule and didn't hit an exclusion rule.
true
}
}

impl EntityPathRule {
Expand Down Expand Up @@ -325,110 +393,200 @@ impl std::cmp::PartialOrd for EntityPathRule {
}
}

#[test]
fn test_rule_order() {
use std::cmp::Ordering;
#[cfg(test)]
mod tests {
use crate::{EntityPath, EntityPathFilter, EntityPathRule, RuleEffect};

#[test]
fn test_rule_order() {
use std::cmp::Ordering;

fn check_total_order(rules: &[EntityPathRule]) {
fn ordering_str(ord: Ordering) -> &'static str {
match ord {
Ordering::Greater => ">",
Ordering::Equal => "=",
Ordering::Less => "<",
fn check_total_order(rules: &[EntityPathRule]) {
fn ordering_str(ord: Ordering) -> &'static str {
match ord {
Ordering::Greater => ">",
Ordering::Equal => "=",
Ordering::Less => "<",
}
}
}

for (i, x) in rules.iter().enumerate() {
for (j, y) in rules.iter().enumerate() {
let actual_ordering = x.cmp(y);
let expected_ordering = i.cmp(&j);
assert!(
actual_ordering == expected_ordering,
"Got {x:?} {} {y:?}; expected {x:?} {} {y:?}",
ordering_str(actual_ordering),
ordering_str(expected_ordering),
);
for (i, x) in rules.iter().enumerate() {
for (j, y) in rules.iter().enumerate() {
let actual_ordering = x.cmp(y);
let expected_ordering = i.cmp(&j);
assert!(
actual_ordering == expected_ordering,
"Got {x:?} {} {y:?}; expected {x:?} {} {y:?}",
ordering_str(actual_ordering),
ordering_str(expected_ordering),
);
}
}
}
}

let rules = [
"/**",
"/apa",
"/world/**",
"/world/",
"/world/car",
"/world/car/driver",
"/x/y/z",
];
let rules = rules.map(EntityPathRule::parse_forgiving);
check_total_order(&rules);
}
let rules = [
"/**",
"/apa",
"/world/**",
"/world/",
"/world/car",
"/world/car/driver",
"/x/y/z",
];
let rules = rules.map(EntityPathRule::parse_forgiving);
check_total_order(&rules);
}

#[test]
fn test_entity_path_filter() {
let filter = EntityPathFilter::parse_forgiving(
r#"
#[test]
fn test_entity_path_filter() {
let filter = EntityPathFilter::parse_forgiving(
r#"
+ /world/**
- /world/
- /world/car/**
+ /world/car/driver
"#,
);

for (path, expected_effect) in [
("/unworldly", None),
("/world", Some(RuleEffect::Exclude)),
("/world/house", Some(RuleEffect::Include)),
("/world/car", Some(RuleEffect::Exclude)),
("/world/car/hood", Some(RuleEffect::Exclude)),
("/world/car/driver", Some(RuleEffect::Include)),
("/world/car/driver/head", Some(RuleEffect::Exclude)),
] {
);

for (path, expected_effect) in [
("/unworldly", None),
("/world", Some(RuleEffect::Exclude)),
("/world/house", Some(RuleEffect::Include)),
("/world/car", Some(RuleEffect::Exclude)),
("/world/car/hood", Some(RuleEffect::Exclude)),
("/world/car/driver", Some(RuleEffect::Include)),
("/world/car/driver/head", Some(RuleEffect::Exclude)),
] {
assert_eq!(
filter.most_specific_match(&EntityPath::from(path)),
expected_effect,
"path: {path:?}",
);
}

assert_eq!(
filter.most_specific_match(&EntityPath::from(path)),
expected_effect,
"path: {path:?}",
EntityPathFilter::parse_forgiving("/**").formatted(),
"+ /**"
);
}

assert_eq!(
EntityPathFilter::parse_forgiving("/**").formatted(),
"+ /**"
);
}

#[test]
fn test_entity_path_filter_subtree() {
let filter = EntityPathFilter::parse_forgiving(
r#"
#[test]
fn test_entity_path_filter_subtree() {
let filter = EntityPathFilter::parse_forgiving(
r#"
+ /world/**
- /world/car/**
+ /world/car/driver
- /world/car/driver/head/**
- /world/city
- /world/houses/**
"#,
);

for (path, expected) in [
("/2d", false),
("/2d/image", false),
("/world", true),
("/world/car", true),
("/world/car/driver", true),
("/world/car/driver/head", false),
("/world/car/driver/head/ear", false),
("/world/city", true),
("/world/city/block", true),
("/world/houses", false),
("/world/houses/1", false),
("/world/houses/1/roof", false),
] {
assert_eq!(
filter.is_anything_in_subtree_included(&EntityPath::from(path)),
expected,
"path: {path:?}",
);

for (path, expected) in [
("/2d", false),
("/2d/image", false),
("/world", true),
("/world/car", true),
("/world/car/driver", true),
("/world/car/driver/head", false),
("/world/car/driver/head/ear", false),
("/world/city", true),
("/world/city/block", true),
("/world/houses", false),
("/world/houses/1", false),
("/world/houses/1/roof", false),
] {
assert_eq!(
filter.is_anything_in_subtree_included(&EntityPath::from(path)),
expected,
"path: {path:?}",
);
}
}

#[test]
fn test_is_superset_of() {
struct TestCase {
filter: &'static str,
contains: Vec<&'static str>,
not_contains: Vec<&'static str>,
}

let cases = [
TestCase {
filter: "+ /**",
contains: [
"",
"+ /**",
"+ /a/**",
r#"+ /a/**
+ /b/c
+ /b/d
"#,
]
.into(),
not_contains: [].into(),
},
TestCase {
filter: "+ /a/**",
contains: ["+ /a/**", "+ /a", "+ /a/b/**"].into(),
not_contains: [
"+ /**",
"+ /b/**",
"+ /b",
r#"+ /a/b
+ /b
"#,
]
.into(),
},
TestCase {
filter: r#"
+ /a
+ /b/c
"#,
contains: ["+ /a", "+ /b/c"].into(),
not_contains: ["+ /a/**", "+ /b/**", "+ /b/c/d"].into(),
},
TestCase {
filter: r#"
+ /**
- /b/c
"#,
contains: ["+ /a", "+ /a/**", "+ /b", "+ /b/c/d"].into(),
not_contains: ["+ /b/**", "+ /b/c"].into(),
},
TestCase {
filter: r#"
+ /**
- /b/c/**
"#,
contains: ["+ /a", "+ /a/**", "+ /b"].into(),
not_contains: ["+ /b/**", "+ /b/c", "+ /b/c/d"].into(),
},
];

for case in &cases {
let filter = EntityPathFilter::parse_forgiving(case.filter);
for contains in &case.contains {
let contains_filter = EntityPathFilter::parse_forgiving(contains);
assert!(
filter.is_superset_of(&contains_filter),
"Expected {:?} to fully contain {:?}, but it didn't",
filter.formatted(),
contains_filter.formatted(),
);
}
for not_contains in &case.not_contains {
let not_contains_filter = EntityPathFilter::parse_forgiving(not_contains);
assert!(
!filter.is_superset_of(&not_contains_filter),
"Expected {:?} to NOT fully contain {:?}, but it did",
filter.formatted(),
not_contains_filter.formatted(),
);
}
}
}
}
19 changes: 19 additions & 0 deletions crates/re_space_view/src/data_query_blueprint.rs
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,25 @@ impl DataQueryBlueprint {
.eq(&other.space_view_class_identifier)
&& self.entity_path_filter.eq(&other.entity_path_filter)
}

/// Checks whether the results of this query "fully contains" the results of another query.
///
/// If this returns `true` then the [`DataQueryResult`] returned by this query should always
/// contain any [`EntityPath`] that would be included in the results of the other query.
///
/// This is a conservative estimate, and may return `false` in situations where the
/// query does in fact cover the other query. However, it should never return `true`
/// in a case where the other query would not be fully covered.
pub fn entity_path_filter_is_superset_of(&self, other: &DataQueryBlueprint) -> bool {
// A query can't fully contain another if their space-view classes don't match
if self.space_view_class_identifier != other.space_view_class_identifier {
return false;
}

// Anything included by the other query is also included by this query
self.entity_path_filter
.is_superset_of(&other.entity_path_filter)
}
}

impl DataQueryBlueprint {
Expand Down
Loading

0 comments on commit 40b43ad

Please sign in to comment.