Implement 6 optional enhancements for the wifi-Mat module: 1. Hardware Integration (csi_receiver.rs + hardware_adapter.rs) - ESP32 CSI support via serial/UDP - Intel 5300 BFEE file parsing - Atheros CSI Tool integration - Live UDP packet streaming - PCAP replay capability 2. CLI Commands (wifi-densepose-cli/src/mat.rs) - `wifi-mat scan` - Run disaster detection scan - `wifi-mat status` - Check event status - `wifi-mat zones` - Manage scan zones - `wifi-mat survivors` - List detected survivors - `wifi-mat alerts` - View and acknowledge alerts - `wifi-mat export` - Export data in various formats 3. REST API (wifi-densepose-mat/src/api/) - Full CRUD for disaster events - Zone management endpoints - Survivor and alert queries - WebSocket streaming for real-time updates - Comprehensive DTOs and error handling 4. WASM Build (wifi-densepose-wasm/src/mat.rs) - Browser-based disaster dashboard - Real-time survivor tracking - Zone visualization - Alert management - JavaScript API bindings 5. Detection Benchmarks (benches/detection_bench.rs) - Single survivor detection - Multi-survivor detection - Full pipeline benchmarks - Signal processing benchmarks - Hardware adapter benchmarks 6. ML Models for Debris Penetration (ml/) - DebrisModel for material analysis - VitalSignsClassifier for triage - FFT-based feature extraction - Bandpass filtering - Monte Carlo dropout for uncertainty All 134 unit tests pass. Compilation verified for: - wifi-densepose-mat - wifi-densepose-cli - wifi-densepose-wasm (with mat feature)
489 lines
15 KiB
Rust
489 lines
15 KiB
Rust
//! # 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 api;
|
|
pub mod detection;
|
|
pub mod domain;
|
|
pub mod integration;
|
|
pub mod localization;
|
|
pub mod ml;
|
|
|
|
// 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,
|
|
};
|
|
|
|
pub use api::{
|
|
create_router, AppState,
|
|
};
|
|
|
|
pub use ml::{
|
|
// Core ML types
|
|
MlError, MlResult, MlDetectionConfig, MlDetectionPipeline, MlDetectionResult,
|
|
// Debris penetration model
|
|
DebrisPenetrationModel, DebrisFeatures, DepthEstimate as MlDepthEstimate,
|
|
DebrisModel, DebrisModelConfig, DebrisFeatureExtractor,
|
|
MaterialType, DebrisClassification, AttenuationPrediction,
|
|
// Vital signs classifier
|
|
VitalSignsClassifier, VitalSignsClassifierConfig,
|
|
BreathingClassification, HeartbeatClassification,
|
|
UncertaintyEstimate, ClassifierOutput,
|
|
};
|
|
|
|
/// 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),
|
|
|
|
/// Machine learning error
|
|
#[error("ML error: {0}")]
|
|
Ml(#[from] ml::MlError),
|
|
}
|
|
|
|
/// 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<()> {
|
|
// Collect detections first to avoid borrowing issues
|
|
let mut detections = Vec::new();
|
|
|
|
{
|
|
let event = self.event.as_ref()
|
|
.ok_or_else(|| MatError::Domain("No active disaster event".into()))?;
|
|
|
|
for zone in event.zones() {
|
|
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);
|
|
|
|
detections.push((zone.id().clone(), vital_signs, location));
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now process detections with mutable access
|
|
let event = self.event.as_mut()
|
|
.ok_or_else(|| MatError::Domain("No active disaster event".into()))?;
|
|
|
|
for (zone_id, vital_signs, location) in detections {
|
|
let survivor = event.record_detection(zone_id, 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,
|
|
// ML types
|
|
MlDetectionConfig, MlDetectionPipeline, MlDetectionResult,
|
|
DebrisModel, MaterialType, DebrisClassification,
|
|
VitalSignsClassifier, UncertaintyEstimate,
|
|
};
|
|
}
|
|
|
|
#[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());
|
|
}
|
|
}
|