pub mod bridge;
pub mod error;
pub mod occupancy;
pub use bridge::{default_socket_path, OccWorldBridge};
pub use error::WorldModelError;
pub use occupancy::worldgraph_to_occupancy;
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct OccupancyGrid3D {
pub width: u32,
pub height: u32,
pub depth: u32,
pub voxels: Vec<u8>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct TrajectoryWaypoint {
pub e: f64,
pub n: f64,
pub u: f64,
pub t_s: f32,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct TrajectoryPrior {
pub track_id: u64,
pub waypoints: Vec<TrajectoryWaypoint>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct SceneBoundsJson {
pub min_e: f64,
pub min_n: f64,
pub max_e: f64,
pub max_n: f64,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct OccupancyWorldModelRequest {
pub past_frames: Vec<OccupancyGrid3D>,
pub voxel_resolution_m: f32,
pub scene_bounds: SceneBoundsJson,
pub prediction_steps: u32,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct OccupancyWorldModelResponse {
pub future_frames: Vec<OccupancyGrid3D>,
pub trajectory_priors: Vec<TrajectoryPrior>,
pub confidence: f32,
pub model_id: String,
pub inference_ms: u64,
}
use wifi_densepose_worldgraph::WorldGraph;
use crate::occupancy::PersonPosition;
pub fn persons_from_worldgraph(graph: &WorldGraph) -> Vec<PersonPosition> {
let bytes = match graph.to_json() {
Ok(b) => b,
Err(e) => {
eprintln!("[worldmodel] WorldGraph::to_json failed: {e}");
return Vec::new();
}
};
let value: serde_json::Value = match serde_json::from_slice(&bytes) {
Ok(v) => v,
Err(e) => {
eprintln!("[worldmodel] failed to parse WorldGraph JSON: {e}");
return Vec::new();
}
};
let nodes = match value.get("nodes").and_then(|n| n.as_array()) {
Some(arr) => arr,
None => return Vec::new(),
};
nodes
.iter()
.filter_map(|node| {
if node.get("kind")?.as_str()? != "person_track" {
return None;
}
let track_id = node.get("track_id")?.as_u64()?;
let pos = node.get("last_position")?;
let east_m = pos.get("east_m")?.as_f64()?;
let north_m = pos.get("north_m")?.as_f64()?;
let up_m = pos.get("up_m")?.as_f64()?;
Some(PersonPosition { track_id, east_m, north_m, up_m })
})
.collect()
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn occupancy_grid_serde_roundtrip() {
let grid = OccupancyGrid3D {
width: 4,
height: 4,
depth: 2,
voxels: vec![17u8; 32],
};
let json = serde_json::to_string(&grid).expect("serialize");
let decoded: OccupancyGrid3D = serde_json::from_str(&json).expect("deserialize");
assert_eq!(decoded.width, grid.width);
assert_eq!(decoded.voxels.len(), grid.voxels.len());
}
#[test]
fn trajectory_prior_serde_roundtrip() {
let prior = TrajectoryPrior {
track_id: 42,
waypoints: vec![
TrajectoryWaypoint { e: 1.0, n: 2.0, u: 0.0, t_s: 0.1 },
TrajectoryWaypoint { e: 1.1, n: 2.1, u: 0.0, t_s: 0.2 },
],
};
let json = serde_json::to_string(&prior).expect("serialize");
let decoded: TrajectoryPrior = serde_json::from_str(&json).expect("deserialize");
assert_eq!(decoded.track_id, 42);
assert_eq!(decoded.waypoints.len(), 2);
}
#[test]
fn request_serde_roundtrip() {
let req = OccupancyWorldModelRequest {
past_frames: vec![OccupancyGrid3D {
width: 200,
height: 200,
depth: 16,
voxels: vec![17u8; 200 * 200 * 16],
}],
voxel_resolution_m: 0.1,
scene_bounds: SceneBoundsJson {
min_e: -10.0,
min_n: -10.0,
max_e: 10.0,
max_n: 10.0,
},
prediction_steps: 15,
};
let json = serde_json::to_string(&req).expect("serialize");
let decoded: OccupancyWorldModelRequest =
serde_json::from_str(&json).expect("deserialize");
assert_eq!(decoded.prediction_steps, 15);
assert_eq!(decoded.past_frames.len(), 1);
}
#[test]
fn response_serde_roundtrip() {
let resp = OccupancyWorldModelResponse {
future_frames: vec![],
trajectory_priors: vec![TrajectoryPrior {
track_id: 1,
waypoints: vec![TrajectoryWaypoint { e: 0.0, n: 0.0, u: 0.0, t_s: 0.0 }],
}],
confidence: 0.82,
model_id: "occworld-dummy-v0".into(),
inference_ms: 375,
};
let json = serde_json::to_string(&resp).expect("serialize");
let decoded: OccupancyWorldModelResponse =
serde_json::from_str(&json).expect("deserialize");
assert_eq!(decoded.inference_ms, 375);
assert!((decoded.confidence - 0.82).abs() < 1e-5);
}
#[test]
fn vram_error_sentinel_roundtrip() {
let resp = OccupancyWorldModelResponse {
future_frames: vec![],
trajectory_priors: vec![],
confidence: 0.0,
model_id: "error:vram:out of memory (CUDA)".into(),
inference_ms: 0,
};
assert!(resp.model_id.starts_with("error:vram:"));
}
}