use std::collections::BTreeMap;
use ahash::HashSet;
use nohash_hasher::{IntMap, IntSet};
use re_chunk::RowId;
use re_chunk_store::{ChunkStoreDiffKind, ChunkStoreEvent, ChunkStoreSubscriber};
use re_log_types::{EntityPath, EntityPathHash, EntityPathPart, TimeInt, Timeline};
use re_query::StorageEngineReadGuard;
use re_types_core::ComponentName;
#[derive(Debug)]
pub struct EntityTree {
pub path: EntityPath,
pub children: BTreeMap<EntityPathPart, EntityTree>,
}
impl ChunkStoreSubscriber for EntityTree {
fn name(&self) -> String {
"rerun.store_subscribers.EntityTree".into()
}
fn as_any(&self) -> &dyn std::any::Any {
self
}
fn as_any_mut(&mut self) -> &mut dyn std::any::Any {
self
}
#[allow(clippy::unimplemented)]
fn on_events(&mut self, _events: &[ChunkStoreEvent]) {
unimplemented!(
r"EntityTree view is maintained manually, see `EntityTree::on_store_{{additions|deletions}}`"
);
}
}
#[derive(Default)]
pub struct CompactedStoreEvents {
pub row_ids: HashSet<RowId>,
pub temporal: IntMap<EntityPathHash, IntMap<Timeline, IntMap<ComponentName, Vec<TimeInt>>>>,
pub timeless: IntMap<EntityPathHash, IntMap<ComponentName, u64>>,
}
impl CompactedStoreEvents {
pub fn new(store_events: &[&ChunkStoreEvent]) -> Self {
let mut this = Self {
row_ids: store_events
.iter()
.flat_map(|event| event.chunk.row_ids())
.collect(),
temporal: Default::default(),
timeless: Default::default(),
};
for event in store_events {
if event.is_static() {
let per_component = this
.timeless
.entry(event.chunk.entity_path().hash())
.or_default();
for component_name in event.chunk.component_names() {
*per_component.entry(component_name).or_default() +=
event.delta().unsigned_abs();
}
} else {
for (&timeline, time_column) in event.chunk.timelines() {
let per_timeline = this
.temporal
.entry(event.chunk.entity_path().hash())
.or_default();
for &time in time_column.times_raw() {
let per_component = per_timeline.entry(timeline).or_default();
for component_name in event.chunk.component_names() {
per_component
.entry(component_name)
.or_default()
.push(TimeInt::new_temporal(time));
}
}
}
}
}
this
}
}
impl EntityTree {
pub fn root() -> Self {
Self::new(EntityPath::root())
}
pub fn new(path: EntityPath) -> Self {
Self {
path,
children: Default::default(),
}
}
pub fn is_leaf(&self) -> bool {
self.children.is_empty()
}
pub fn check_is_empty(&self, engine: &StorageEngineReadGuard<'_>) -> bool {
self.children.is_empty() && !engine.store().entity_has_data(&self.path)
}
pub fn on_store_additions(&mut self, events: &[ChunkStoreEvent]) {
re_tracing::profile_function!();
for event in events
.iter()
.filter(|e| e.kind == ChunkStoreDiffKind::Addition)
{
self.on_store_addition(event);
}
}
fn on_store_addition(&mut self, event: &ChunkStoreEvent) {
re_tracing::profile_function!();
let entity_path = event.chunk.entity_path();
let mut tree = self;
for (i, part) in entity_path.iter().enumerate() {
tree = tree
.children
.entry(part.clone())
.or_insert_with(|| Self::new(entity_path.as_slice()[..=i].into()));
}
}
pub fn on_store_deletions(
&mut self,
engine: &StorageEngineReadGuard<'_>,
entity_paths_with_deletions: &IntSet<EntityPath>,
events: &[ChunkStoreEvent],
) {
let _ = events;
self.children.retain(|_, entity| {
entity.on_store_deletions(engine, entity_paths_with_deletions, events);
let has_children = || !entity.children.is_empty();
let has_recursive_deletion_events = || {
entity_paths_with_deletions
.iter()
.any(|removed_entity_path| removed_entity_path.starts_with(&entity.path))
};
let has_data = || engine.store().entity_has_data(&entity.path);
let should_be_removed =
!has_children() && (has_recursive_deletion_events() && !has_data());
!should_be_removed
});
}
pub fn subtree(&self, path: &EntityPath) -> Option<&Self> {
fn subtree_recursive<'tree>(
this: &'tree EntityTree,
path: &[EntityPathPart],
) -> Option<&'tree EntityTree> {
match path {
[] => Some(this),
[first, rest @ ..] => {
let child = this.children.get(first)?;
subtree_recursive(child, rest)
}
}
}
subtree_recursive(self, path.as_slice())
}
pub fn visit_children_recursively(&self, mut visitor: impl FnMut(&EntityPath)) {
fn visit(this: &EntityTree, visitor: &mut impl FnMut(&EntityPath)) {
visitor(&this.path);
for child in this.children.values() {
visit(child, visitor);
}
}
visit(self, &mut visitor);
}
pub fn find_first_child_recursive(
&self,
mut predicate: impl FnMut(&EntityPath) -> bool,
) -> Option<&Self> {
fn visit<'a>(
this: &'a EntityTree,
predicate: &mut impl FnMut(&EntityPath) -> bool,
) -> Option<&'a EntityTree> {
if predicate(&this.path) {
return Some(this);
};
for child in this.children.values() {
if let Some(subtree) = visit(child, predicate) {
return Some(subtree);
};
}
None
}
visit(self, &mut predicate)
}
}
#[cfg(test)]
mod tests {
use std::sync::Arc;
use re_chunk::{Chunk, RowId};
use re_log_types::{example_components::MyPoint, EntityPath, StoreId, TimePoint, Timeline};
use crate::EntityDb;
#[test]
fn deleting_descendants() -> anyhow::Result<()> {
re_log::setup_logging();
let mut db = EntityDb::new(StoreId::random(re_log_types::StoreKind::Recording));
let timeline_frame = Timeline::new_sequence("frame");
let entity_path_parent: EntityPath = "parent".into();
let entity_path_child: EntityPath = "parent/child1".into();
let entity_path_grandchild: EntityPath = "parent/child1/grandchild".into();
assert!(db.tree().check_is_empty(&db.storage_engine()));
{
let row_id = RowId::new();
let timepoint = TimePoint::from_iter([(timeline_frame, 10)]);
let point = MyPoint::new(1.0, 2.0);
let chunk = Chunk::builder(entity_path_grandchild.clone())
.with_component_batches(row_id, timepoint, [&[point] as _])
.build()?;
db.add_chunk(&Arc::new(chunk))?;
}
{
let parent = db
.tree()
.find_first_child_recursive(|entity_path| *entity_path == entity_path_parent)
.unwrap();
let child = db
.tree()
.find_first_child_recursive(|entity_path| *entity_path == entity_path_child)
.unwrap();
let grandchild = db
.tree()
.find_first_child_recursive(|entity_path| *entity_path == entity_path_grandchild)
.unwrap();
assert_eq!(1, parent.children.len());
assert_eq!(1, child.children.len());
assert_eq!(0, grandchild.children.len());
assert!(!db.tree().check_is_empty(&db.storage_engine()));
assert!(!parent.check_is_empty(&db.storage_engine()));
assert!(!child.check_is_empty(&db.storage_engine()));
assert!(!grandchild.check_is_empty(&db.storage_engine()));
}
let _store_events = db.gc(&re_chunk_store::GarbageCollectionOptions::gc_everything());
{
let parent = db
.tree()
.find_first_child_recursive(|entity_path| *entity_path == entity_path_parent);
let child = db
.tree()
.find_first_child_recursive(|entity_path| *entity_path == entity_path_child);
let grandchild = db
.tree()
.find_first_child_recursive(|entity_path| *entity_path == entity_path_grandchild);
assert!(db.tree().check_is_empty(&db.storage_engine()));
assert!(parent.is_none());
assert!(child.is_none());
assert!(grandchild.is_none());
}
Ok(())
}
}