feat: Add wifi-densepose-mat disaster detection module
Implements WiFi-Mat (Mass Casualty Assessment Tool) for detecting and localizing survivors trapped in rubble, earthquakes, and natural disasters. Architecture: - Domain-Driven Design with bounded contexts (Detection, Localization, Alerting) - Modular Rust crate integrating with existing wifi-densepose-* crates - Event-driven architecture for audit trails and distributed deployments Features: - Breathing pattern detection from CSI amplitude variations - Heartbeat detection using micro-Doppler analysis - Movement classification (gross, fine, tremor, periodic) - START protocol-compatible triage classification - 3D position estimation via triangulation and depth estimation - Real-time alert generation with priority escalation Documentation: - ADR-001: Architecture Decision Record for wifi-Mat - DDD domain model specification
This commit is contained in:
454
rust-port/wifi-densepose-rs/crates/wifi-densepose-mat/src/lib.rs
Normal file
454
rust-port/wifi-densepose-rs/crates/wifi-densepose-mat/src/lib.rs
Normal file
@@ -0,0 +1,454 @@
|
||||
//! # WiFi-DensePose MAT (Mass Casualty Assessment Tool)
|
||||
//!
|
||||
//! A modular extension for WiFi-based disaster survivor detection and localization.
|
||||
//!
|
||||
//! This crate provides capabilities for detecting human survivors trapped in rubble,
|
||||
//! debris, or collapsed structures using WiFi Channel State Information (CSI) analysis.
|
||||
//!
|
||||
//! ## Features
|
||||
//!
|
||||
//! - **Vital Signs Detection**: Breathing patterns, heartbeat signatures, and movement
|
||||
//! - **Survivor Localization**: 3D position estimation through debris
|
||||
//! - **Triage Classification**: Automatic START protocol-compatible triage
|
||||
//! - **Real-time Alerting**: Priority-based alert generation and dispatch
|
||||
//!
|
||||
//! ## Use Cases
|
||||
//!
|
||||
//! - Earthquake search and rescue
|
||||
//! - Building collapse response
|
||||
//! - Avalanche victim location
|
||||
//! - Flood rescue operations
|
||||
//! - Mine collapse detection
|
||||
//!
|
||||
//! ## Architecture
|
||||
//!
|
||||
//! The crate follows Domain-Driven Design (DDD) principles with clear bounded contexts:
|
||||
//!
|
||||
//! ```text
|
||||
//! ┌─────────────────────────────────────────────────────────┐
|
||||
//! │ wifi-densepose-mat │
|
||||
//! ├─────────────────────────────────────────────────────────┤
|
||||
//! │ ┌───────────┐ ┌─────────────┐ ┌─────────────────┐ │
|
||||
//! │ │ Detection │ │Localization │ │ Alerting │ │
|
||||
//! │ │ Context │ │ Context │ │ Context │ │
|
||||
//! │ └─────┬─────┘ └──────┬──────┘ └────────┬────────┘ │
|
||||
//! │ └───────────────┼──────────────────┘ │
|
||||
//! │ │ │
|
||||
//! │ ┌─────────▼─────────┐ │
|
||||
//! │ │ Integration │ │
|
||||
//! │ │ Layer │ │
|
||||
//! │ └───────────────────┘ │
|
||||
//! └─────────────────────────────────────────────────────────┘
|
||||
//! ```
|
||||
//!
|
||||
//! ## Example
|
||||
//!
|
||||
//! ```rust,no_run
|
||||
//! use wifi_densepose_mat::{
|
||||
//! DisasterResponse, DisasterConfig, DisasterType,
|
||||
//! ScanZone, ZoneBounds,
|
||||
//! };
|
||||
//!
|
||||
//! #[tokio::main]
|
||||
//! async fn main() -> anyhow::Result<()> {
|
||||
//! // Initialize disaster response system
|
||||
//! let config = DisasterConfig::builder()
|
||||
//! .disaster_type(DisasterType::Earthquake)
|
||||
//! .sensitivity(0.8)
|
||||
//! .build();
|
||||
//!
|
||||
//! let mut response = DisasterResponse::new(config);
|
||||
//!
|
||||
//! // Define scan zone
|
||||
//! let zone = ScanZone::new(
|
||||
//! "Building A - North Wing",
|
||||
//! ZoneBounds::rectangle(0.0, 0.0, 50.0, 30.0),
|
||||
//! );
|
||||
//! response.add_zone(zone)?;
|
||||
//!
|
||||
//! // Start scanning
|
||||
//! response.start_scanning().await?;
|
||||
//!
|
||||
//! Ok(())
|
||||
//! }
|
||||
//! ```
|
||||
|
||||
#![cfg_attr(docsrs, feature(doc_cfg))]
|
||||
#![warn(missing_docs)]
|
||||
#![warn(rustdoc::missing_crate_level_docs)]
|
||||
|
||||
pub mod alerting;
|
||||
pub mod detection;
|
||||
pub mod domain;
|
||||
pub mod integration;
|
||||
pub mod localization;
|
||||
|
||||
// Re-export main types
|
||||
pub use domain::{
|
||||
survivor::{Survivor, SurvivorId, SurvivorMetadata, SurvivorStatus},
|
||||
disaster_event::{DisasterEvent, DisasterEventId, DisasterType, EventStatus},
|
||||
scan_zone::{ScanZone, ScanZoneId, ZoneBounds, ZoneStatus, ScanParameters},
|
||||
alert::{Alert, AlertId, AlertPayload, Priority},
|
||||
vital_signs::{
|
||||
VitalSignsReading, BreathingPattern, BreathingType,
|
||||
HeartbeatSignature, MovementProfile, MovementType,
|
||||
},
|
||||
triage::{TriageStatus, TriageCalculator},
|
||||
coordinates::{Coordinates3D, LocationUncertainty, DepthEstimate},
|
||||
events::{DetectionEvent, AlertEvent, DomainEvent},
|
||||
};
|
||||
|
||||
pub use detection::{
|
||||
BreathingDetector, BreathingDetectorConfig,
|
||||
HeartbeatDetector, HeartbeatDetectorConfig,
|
||||
MovementClassifier, MovementClassifierConfig,
|
||||
VitalSignsDetector, DetectionPipeline, DetectionConfig,
|
||||
};
|
||||
|
||||
pub use localization::{
|
||||
Triangulator, TriangulationConfig,
|
||||
DepthEstimator, DepthEstimatorConfig,
|
||||
PositionFuser, LocalizationService,
|
||||
};
|
||||
|
||||
pub use alerting::{
|
||||
AlertGenerator, AlertDispatcher, AlertConfig,
|
||||
TriageService, PriorityCalculator,
|
||||
};
|
||||
|
||||
pub use integration::{
|
||||
SignalAdapter, NeuralAdapter, HardwareAdapter,
|
||||
AdapterError, IntegrationConfig,
|
||||
};
|
||||
|
||||
/// Library version
|
||||
pub const VERSION: &str = env!("CARGO_PKG_VERSION");
|
||||
|
||||
/// Common result type for MAT operations
|
||||
pub type Result<T> = std::result::Result<T, MatError>;
|
||||
|
||||
/// Unified error type for MAT operations
|
||||
#[derive(Debug, thiserror::Error)]
|
||||
pub enum MatError {
|
||||
/// Detection error
|
||||
#[error("Detection error: {0}")]
|
||||
Detection(String),
|
||||
|
||||
/// Localization error
|
||||
#[error("Localization error: {0}")]
|
||||
Localization(String),
|
||||
|
||||
/// Alerting error
|
||||
#[error("Alerting error: {0}")]
|
||||
Alerting(String),
|
||||
|
||||
/// Integration error
|
||||
#[error("Integration error: {0}")]
|
||||
Integration(#[from] AdapterError),
|
||||
|
||||
/// Configuration error
|
||||
#[error("Configuration error: {0}")]
|
||||
Config(String),
|
||||
|
||||
/// Domain invariant violation
|
||||
#[error("Domain error: {0}")]
|
||||
Domain(String),
|
||||
|
||||
/// Repository error
|
||||
#[error("Repository error: {0}")]
|
||||
Repository(String),
|
||||
|
||||
/// Signal processing error
|
||||
#[error("Signal processing error: {0}")]
|
||||
Signal(#[from] wifi_densepose_signal::SignalError),
|
||||
|
||||
/// I/O error
|
||||
#[error("I/O error: {0}")]
|
||||
Io(#[from] std::io::Error),
|
||||
}
|
||||
|
||||
/// Configuration for the disaster response system
|
||||
#[derive(Debug, Clone)]
|
||||
pub struct DisasterConfig {
|
||||
/// Type of disaster event
|
||||
pub disaster_type: DisasterType,
|
||||
/// Detection sensitivity (0.0-1.0)
|
||||
pub sensitivity: f64,
|
||||
/// Minimum confidence threshold for survivor detection
|
||||
pub confidence_threshold: f64,
|
||||
/// Maximum depth to scan (meters)
|
||||
pub max_depth: f64,
|
||||
/// Scan interval in milliseconds
|
||||
pub scan_interval_ms: u64,
|
||||
/// Enable continuous monitoring
|
||||
pub continuous_monitoring: bool,
|
||||
/// Alert configuration
|
||||
pub alert_config: AlertConfig,
|
||||
}
|
||||
|
||||
impl Default for DisasterConfig {
|
||||
fn default() -> Self {
|
||||
Self {
|
||||
disaster_type: DisasterType::Unknown,
|
||||
sensitivity: 0.8,
|
||||
confidence_threshold: 0.5,
|
||||
max_depth: 5.0,
|
||||
scan_interval_ms: 500,
|
||||
continuous_monitoring: true,
|
||||
alert_config: AlertConfig::default(),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl DisasterConfig {
|
||||
/// Create a new configuration builder
|
||||
pub fn builder() -> DisasterConfigBuilder {
|
||||
DisasterConfigBuilder::default()
|
||||
}
|
||||
}
|
||||
|
||||
/// Builder for DisasterConfig
|
||||
#[derive(Debug, Default)]
|
||||
pub struct DisasterConfigBuilder {
|
||||
config: DisasterConfig,
|
||||
}
|
||||
|
||||
impl DisasterConfigBuilder {
|
||||
/// Set disaster type
|
||||
pub fn disaster_type(mut self, disaster_type: DisasterType) -> Self {
|
||||
self.config.disaster_type = disaster_type;
|
||||
self
|
||||
}
|
||||
|
||||
/// Set detection sensitivity
|
||||
pub fn sensitivity(mut self, sensitivity: f64) -> Self {
|
||||
self.config.sensitivity = sensitivity.clamp(0.0, 1.0);
|
||||
self
|
||||
}
|
||||
|
||||
/// Set confidence threshold
|
||||
pub fn confidence_threshold(mut self, threshold: f64) -> Self {
|
||||
self.config.confidence_threshold = threshold.clamp(0.0, 1.0);
|
||||
self
|
||||
}
|
||||
|
||||
/// Set maximum scan depth
|
||||
pub fn max_depth(mut self, depth: f64) -> Self {
|
||||
self.config.max_depth = depth.max(0.0);
|
||||
self
|
||||
}
|
||||
|
||||
/// Set scan interval
|
||||
pub fn scan_interval_ms(mut self, interval: u64) -> Self {
|
||||
self.config.scan_interval_ms = interval.max(100);
|
||||
self
|
||||
}
|
||||
|
||||
/// Enable/disable continuous monitoring
|
||||
pub fn continuous_monitoring(mut self, enabled: bool) -> Self {
|
||||
self.config.continuous_monitoring = enabled;
|
||||
self
|
||||
}
|
||||
|
||||
/// Build the configuration
|
||||
pub fn build(self) -> DisasterConfig {
|
||||
self.config
|
||||
}
|
||||
}
|
||||
|
||||
/// Main disaster response coordinator
|
||||
pub struct DisasterResponse {
|
||||
config: DisasterConfig,
|
||||
event: Option<DisasterEvent>,
|
||||
detection_pipeline: DetectionPipeline,
|
||||
localization_service: LocalizationService,
|
||||
alert_dispatcher: AlertDispatcher,
|
||||
running: std::sync::atomic::AtomicBool,
|
||||
}
|
||||
|
||||
impl DisasterResponse {
|
||||
/// Create a new disaster response system
|
||||
pub fn new(config: DisasterConfig) -> Self {
|
||||
let detection_config = DetectionConfig::from_disaster_config(&config);
|
||||
let detection_pipeline = DetectionPipeline::new(detection_config);
|
||||
|
||||
let localization_service = LocalizationService::new();
|
||||
let alert_dispatcher = AlertDispatcher::new(config.alert_config.clone());
|
||||
|
||||
Self {
|
||||
config,
|
||||
event: None,
|
||||
detection_pipeline,
|
||||
localization_service,
|
||||
alert_dispatcher,
|
||||
running: std::sync::atomic::AtomicBool::new(false),
|
||||
}
|
||||
}
|
||||
|
||||
/// Initialize a new disaster event
|
||||
pub fn initialize_event(
|
||||
&mut self,
|
||||
location: geo::Point<f64>,
|
||||
description: &str,
|
||||
) -> Result<&DisasterEvent> {
|
||||
let event = DisasterEvent::new(
|
||||
self.config.disaster_type.clone(),
|
||||
location,
|
||||
description,
|
||||
);
|
||||
self.event = Some(event);
|
||||
self.event.as_ref().ok_or_else(|| MatError::Domain("Failed to create event".into()))
|
||||
}
|
||||
|
||||
/// Add a scan zone to the current event
|
||||
pub fn add_zone(&mut self, zone: ScanZone) -> Result<()> {
|
||||
let event = self.event.as_mut()
|
||||
.ok_or_else(|| MatError::Domain("No active disaster event".into()))?;
|
||||
event.add_zone(zone);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Start the scanning process
|
||||
pub async fn start_scanning(&mut self) -> Result<()> {
|
||||
use std::sync::atomic::Ordering;
|
||||
|
||||
self.running.store(true, Ordering::SeqCst);
|
||||
|
||||
while self.running.load(Ordering::SeqCst) {
|
||||
self.scan_cycle().await?;
|
||||
|
||||
if !self.config.continuous_monitoring {
|
||||
break;
|
||||
}
|
||||
|
||||
tokio::time::sleep(
|
||||
std::time::Duration::from_millis(self.config.scan_interval_ms)
|
||||
).await;
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Stop the scanning process
|
||||
pub fn stop_scanning(&self) {
|
||||
use std::sync::atomic::Ordering;
|
||||
self.running.store(false, Ordering::SeqCst);
|
||||
}
|
||||
|
||||
/// Execute a single scan cycle
|
||||
async fn scan_cycle(&mut self) -> Result<()> {
|
||||
let event = self.event.as_mut()
|
||||
.ok_or_else(|| MatError::Domain("No active disaster event".into()))?;
|
||||
|
||||
for zone in event.zones_mut() {
|
||||
if zone.status() != &ZoneStatus::Active {
|
||||
continue;
|
||||
}
|
||||
|
||||
// This would integrate with actual hardware in production
|
||||
// For now, we process any available CSI data
|
||||
let detection_result = self.detection_pipeline.process_zone(zone).await?;
|
||||
|
||||
if let Some(vital_signs) = detection_result {
|
||||
// Attempt localization
|
||||
let location = self.localization_service
|
||||
.estimate_position(&vital_signs, zone)
|
||||
.ok();
|
||||
|
||||
// Create or update survivor
|
||||
let survivor = event.record_detection(
|
||||
zone.id().clone(),
|
||||
vital_signs,
|
||||
location,
|
||||
)?;
|
||||
|
||||
// Generate alert if needed
|
||||
if survivor.should_alert() {
|
||||
let alert = self.alert_dispatcher.generate_alert(&survivor)?;
|
||||
self.alert_dispatcher.dispatch(alert).await?;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
/// Get the current disaster event
|
||||
pub fn event(&self) -> Option<&DisasterEvent> {
|
||||
self.event.as_ref()
|
||||
}
|
||||
|
||||
/// Get all detected survivors
|
||||
pub fn survivors(&self) -> Vec<&Survivor> {
|
||||
self.event.as_ref()
|
||||
.map(|e| e.survivors())
|
||||
.unwrap_or_default()
|
||||
}
|
||||
|
||||
/// Get survivors by triage status
|
||||
pub fn survivors_by_triage(&self, status: TriageStatus) -> Vec<&Survivor> {
|
||||
self.survivors()
|
||||
.into_iter()
|
||||
.filter(|s| s.triage_status() == &status)
|
||||
.collect()
|
||||
}
|
||||
}
|
||||
|
||||
/// Prelude module for convenient imports
|
||||
pub mod prelude {
|
||||
pub use crate::{
|
||||
DisasterConfig, DisasterConfigBuilder, DisasterResponse,
|
||||
MatError, Result,
|
||||
// Domain types
|
||||
Survivor, SurvivorId, DisasterEvent, DisasterType,
|
||||
ScanZone, ZoneBounds, TriageStatus,
|
||||
VitalSignsReading, BreathingPattern, HeartbeatSignature,
|
||||
Coordinates3D, Alert, Priority,
|
||||
// Detection
|
||||
DetectionPipeline, VitalSignsDetector,
|
||||
// Localization
|
||||
LocalizationService,
|
||||
// Alerting
|
||||
AlertDispatcher,
|
||||
};
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn test_config_builder() {
|
||||
let config = DisasterConfig::builder()
|
||||
.disaster_type(DisasterType::Earthquake)
|
||||
.sensitivity(0.9)
|
||||
.confidence_threshold(0.6)
|
||||
.max_depth(10.0)
|
||||
.build();
|
||||
|
||||
assert!(matches!(config.disaster_type, DisasterType::Earthquake));
|
||||
assert!((config.sensitivity - 0.9).abs() < f64::EPSILON);
|
||||
assert!((config.confidence_threshold - 0.6).abs() < f64::EPSILON);
|
||||
assert!((config.max_depth - 10.0).abs() < f64::EPSILON);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_sensitivity_clamping() {
|
||||
let config = DisasterConfig::builder()
|
||||
.sensitivity(1.5)
|
||||
.build();
|
||||
|
||||
assert!((config.sensitivity - 1.0).abs() < f64::EPSILON);
|
||||
|
||||
let config = DisasterConfig::builder()
|
||||
.sensitivity(-0.5)
|
||||
.build();
|
||||
|
||||
assert!(config.sensitivity.abs() < f64::EPSILON);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_version() {
|
||||
assert!(!VERSION.is_empty());
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user