use ifc_lite_core::{EntityDecoder, EntityScanner, IfcType};
use rustc_hash::FxHashMap;
pub fn propagate_voids_to_parts(
void_index: &mut FxHashMap<u32, Vec<u32>>,
content: &str,
decoder: &mut EntityDecoder,
) {
let mut scanner = EntityScanner::new(content);
let mut propagations: Vec<(u32, Vec<u32>)> = Vec::new();
while let Some((id, type_name, start, end)) = scanner.next_entity() {
if type_name != "IFCRELAGGREGATES" {
continue;
}
let entity = match decoder.decode_at_with_id(id, start, end) {
Ok(e) => e,
Err(_) => continue,
};
let parent_id = match entity.get_ref(4) {
Some(id) => id,
None => continue,
};
if !void_index.contains_key(&parent_id) {
continue;
}
let children_attr = match entity.get(5) {
Some(attr) => attr,
None => continue,
};
let children: Vec<u32> = match children_attr.as_list() {
Some(list) => list.iter().filter_map(|item| item.as_entity_ref()).collect(),
None => continue,
};
let mut eligible_children = Vec::new();
for child_id in children {
if let Ok(child) = decoder.decode_by_id(child_id) {
if child.ifc_type == IfcType::IfcBuildingElementPart {
let has_repr = child.get(6).map(|a| !a.is_null()).unwrap_or(false);
if has_repr {
eligible_children.push(child_id);
}
}
}
}
if !eligible_children.is_empty() {
propagations.push((parent_id, eligible_children));
}
}
for (parent_id, children) in propagations {
let parent_voids = match void_index.get(&parent_id) {
Some(v) => v.clone(),
None => continue,
};
for child_id in children {
void_index
.entry(child_id)
.or_default()
.extend(parent_voids.iter().copied());
}
}
}
#[derive(Debug, Clone)]
pub struct VoidIndex {
host_to_voids: FxHashMap<u32, Vec<u32>>,
void_to_host: FxHashMap<u32, u32>,
relationship_count: usize,
}
impl VoidIndex {
pub fn new() -> Self {
Self {
host_to_voids: FxHashMap::default(),
void_to_host: FxHashMap::default(),
relationship_count: 0,
}
}
pub fn from_content(content: &str, decoder: &mut EntityDecoder) -> Self {
let mut index = Self::new();
let mut scanner = EntityScanner::new(content);
while let Some((_id, type_name, start, end)) = scanner.next_entity() {
if type_name == "IFCRELVOIDSELEMENT" {
if let Ok(entity) = decoder.decode_at(start, end) {
if let (Some(host_id), Some(void_id)) = (entity.get_ref(4), entity.get_ref(5)) {
index.add_relationship(host_id, void_id);
}
}
}
}
index
}
pub fn add_relationship(&mut self, host_id: u32, void_id: u32) {
self.host_to_voids.entry(host_id).or_default().push(void_id);
self.void_to_host.insert(void_id, host_id);
self.relationship_count += 1;
}
pub fn get_voids(&self, host_id: u32) -> &[u32] {
self.host_to_voids
.get(&host_id)
.map(|v| v.as_slice())
.unwrap_or(&[])
}
pub fn get_host(&self, void_id: u32) -> Option<u32> {
self.void_to_host.get(&void_id).copied()
}
pub fn has_voids(&self, host_id: u32) -> bool {
self.host_to_voids
.get(&host_id)
.map(|v| !v.is_empty())
.unwrap_or(false)
}
pub fn void_count(&self, host_id: u32) -> usize {
self.host_to_voids
.get(&host_id)
.map(|v| v.len())
.unwrap_or(0)
}
pub fn host_count(&self) -> usize {
self.host_to_voids.len()
}
pub fn total_relationships(&self) -> usize {
self.relationship_count
}
pub fn iter(&self) -> impl Iterator<Item = (u32, &[u32])> {
self.host_to_voids.iter().map(|(k, v)| (*k, v.as_slice()))
}
pub fn hosts_with_voids(&self) -> Vec<u32> {
self.host_to_voids.keys().copied().collect()
}
pub fn is_void(&self, entity_id: u32) -> bool {
self.void_to_host.contains_key(&entity_id)
}
pub fn is_host_with_voids(&self, entity_id: u32) -> bool {
self.host_to_voids.contains_key(&entity_id)
}
}
impl Default for VoidIndex {
fn default() -> Self {
Self::new()
}
}
#[derive(Debug, Clone)]
pub struct VoidStatistics {
pub hosts_with_voids: usize,
pub total_voids: usize,
pub max_voids_per_host: usize,
pub avg_voids_per_host: f64,
pub hosts_with_many_voids: usize,
}
impl VoidStatistics {
pub fn from_index(index: &VoidIndex) -> Self {
let hosts_with_voids = index.host_count();
let total_voids = index.total_relationships();
let max_voids_per_host = index
.host_to_voids
.values()
.map(|v| v.len())
.max()
.unwrap_or(0);
let avg_voids_per_host = if hosts_with_voids > 0 {
total_voids as f64 / hosts_with_voids as f64
} else {
0.0
};
let hosts_with_many_voids = index.host_to_voids.values().filter(|v| v.len() > 10).count();
Self {
hosts_with_voids,
total_voids,
max_voids_per_host,
avg_voids_per_host,
hosts_with_many_voids,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_void_index_basic() {
let mut index = VoidIndex::new();
index.add_relationship(100, 200);
index.add_relationship(100, 201);
index.add_relationship(101, 202);
assert_eq!(index.get_voids(100), &[200, 201]);
assert_eq!(index.get_voids(101), &[202]);
assert!(index.get_voids(999).is_empty());
assert_eq!(index.get_host(200), Some(100));
assert_eq!(index.get_host(202), Some(101));
assert_eq!(index.get_host(999), None);
assert_eq!(index.void_count(100), 2);
assert_eq!(index.void_count(101), 1);
assert_eq!(index.host_count(), 2);
assert_eq!(index.total_relationships(), 3);
}
#[test]
fn test_void_index_has_voids() {
let mut index = VoidIndex::new();
index.add_relationship(100, 200);
assert!(index.has_voids(100));
assert!(!index.has_voids(999));
}
#[test]
fn test_void_index_is_void() {
let mut index = VoidIndex::new();
index.add_relationship(100, 200);
assert!(index.is_void(200));
assert!(!index.is_void(100));
assert!(!index.is_void(999));
}
#[test]
fn test_void_index_hosts_with_voids() {
let mut index = VoidIndex::new();
index.add_relationship(100, 200);
index.add_relationship(101, 201);
index.add_relationship(102, 202);
let hosts = index.hosts_with_voids();
assert_eq!(hosts.len(), 3);
assert!(hosts.contains(&100));
assert!(hosts.contains(&101));
assert!(hosts.contains(&102));
}
#[test]
fn test_void_statistics() {
let mut index = VoidIndex::new();
index.add_relationship(100, 200);
index.add_relationship(100, 201);
index.add_relationship(100, 202);
index.add_relationship(101, 203);
let stats = VoidStatistics::from_index(&index);
assert_eq!(stats.hosts_with_voids, 2);
assert_eq!(stats.total_voids, 4);
assert_eq!(stats.max_voids_per_host, 3);
assert!((stats.avg_voids_per_host - 2.0).abs() < 0.01);
assert_eq!(stats.hosts_with_many_voids, 0);
}
#[test]
fn test_void_statistics_many_voids() {
let mut index = VoidIndex::new();
for i in 0..15 {
index.add_relationship(100, 200 + i);
}
let stats = VoidStatistics::from_index(&index);
assert_eq!(stats.hosts_with_many_voids, 1);
}
}