// unconscious_screen.rs — F4: memory daemon status // // Fetches status from the poc-memory daemon via socket RPC and // displays graph health gauges, running tasks, and recent completions. use ratatui::{ layout::{Constraint, Layout, Rect}, style::{Color, Modifier, Style}, text::{Line, Span}, widgets::{Block, Borders, Gauge, Paragraph, Wrap}, Frame, }; use super::{App, SCREEN_LEGEND}; use crate::subconscious::daemon::GraphHealth; /// Status fetched from the daemon socket. #[derive(serde::Deserialize, Default)] struct DaemonStatus { #[allow(dead_code)] pid: u32, tasks: Vec, #[serde(default)] graph_health: Option, } fn fetch_status() -> Option { let json = jobkit::daemon::socket::send_rpc(&crate::config::get().data_dir, "")?; serde_json::from_str(&json).ok() } impl App { pub(crate) fn draw_unconscious(&self, frame: &mut Frame, size: Rect) { let block = Block::default() .title_top(Line::from(SCREEN_LEGEND).left_aligned()) .title_top(Line::from(" unconscious ").right_aligned()) .borders(Borders::ALL) .border_style(Style::default().fg(Color::Cyan)); let inner = block.inner(size); frame.render_widget(block, size); let status = fetch_status(); match &status { None => { let dim = Style::default().fg(Color::DarkGray); frame.render_widget( Paragraph::new(Line::styled(" daemon not running", dim)), inner, ); } Some(st) => { // Split into health area and tasks area let has_health = st.graph_health.is_some(); let [health_area, tasks_area] = Layout::vertical([ Constraint::Length(if has_health { 9 } else { 0 }), Constraint::Min(1), ]) .areas(inner); if let Some(ref gh) = st.graph_health { Self::render_health(frame, gh, health_area); } Self::render_tasks(frame, &st.tasks, tasks_area); } } } fn render_health(frame: &mut Frame, gh: &GraphHealth, area: Rect) { let [metrics_area, gauges_area, plan_area] = Layout::vertical([ Constraint::Length(2), Constraint::Length(4), Constraint::Min(1), ]) .areas(area); // Metrics summary let summary = Line::from(format!( " {} nodes {} edges {} communities", gh.nodes, gh.edges, gh.communities )); let ep_line = Line::from(vec![ Span::raw(" episodic: "), Span::styled( format!("{:.0}%", gh.episodic_ratio * 100.0), if gh.episodic_ratio < 0.4 { Style::default().fg(Color::Green) } else { Style::default().fg(Color::Red) }, ), Span::raw(format!(" σ={:.1} interference={}", gh.sigma, gh.interference)), ]); frame.render_widget(Paragraph::new(vec![summary, ep_line]), metrics_area); // Health gauges let [g1, g2, g3] = Layout::horizontal([ Constraint::Ratio(1, 3), Constraint::Ratio(1, 3), Constraint::Ratio(1, 3), ]) .areas(gauges_area); let alpha_color = if gh.alpha >= 2.5 { Color::Green } else { Color::Red }; frame.render_widget( Gauge::default() .block(Block::default().borders(Borders::ALL).title(" α (≥2.5) ")) .gauge_style(Style::default().fg(alpha_color)) .ratio((gh.alpha / 5.0).clamp(0.0, 1.0) as f64) .label(format!("{:.2}", gh.alpha)), g1, ); let gini_color = if gh.gini <= 0.4 { Color::Green } else { Color::Red }; frame.render_widget( Gauge::default() .block(Block::default().borders(Borders::ALL).title(" gini (≤0.4) ")) .gauge_style(Style::default().fg(gini_color)) .ratio(gh.gini.clamp(0.0, 1.0) as f64) .label(format!("{:.3}", gh.gini)), g2, ); let cc_color = if gh.avg_cc >= 0.2 { Color::Green } else { Color::Red }; frame.render_widget( Gauge::default() .block(Block::default().borders(Borders::ALL).title(" cc (≥0.2) ")) .gauge_style(Style::default().fg(cc_color)) .ratio(gh.avg_cc.clamp(0.0, 1.0) as f64) .label(format!("{:.3}", gh.avg_cc)), g3, ); // Plan summary let plan_total: usize = gh.plan_counts.values().sum::() + 1; let mut plan_items: Vec<_> = gh.plan_counts.iter() .filter(|(_, c)| **c > 0) .collect(); plan_items.sort_by(|a, b| a.0.cmp(b.0)); let plan_summary: Vec = plan_items.iter() .map(|(a, c)| format!("{}{}", &a[..1], c)) .collect(); let plan_line = Line::from(vec![ Span::raw(" plan: "), Span::styled( format!("{}", plan_total), Style::default().add_modifier(Modifier::BOLD), ), Span::raw(format!(" agents ({} +health)", plan_summary.join(" "))), ]); frame.render_widget(Paragraph::new(plan_line), plan_area); } fn render_tasks(frame: &mut Frame, tasks: &[jobkit::TaskInfo], area: Rect) { let mut lines: Vec = Vec::new(); let section = Style::default().fg(Color::Yellow); let dim = Style::default().fg(Color::DarkGray); let running: Vec<_> = tasks.iter() .filter(|t| matches!(t.status, jobkit::TaskStatus::Running)) .collect(); let completed: Vec<_> = tasks.iter() .filter(|t| matches!(t.status, jobkit::TaskStatus::Completed)) .collect(); let failed: Vec<_> = tasks.iter() .filter(|t| matches!(t.status, jobkit::TaskStatus::Failed)) .collect(); lines.push(Line::styled("── Tasks ──", section)); lines.push(Line::raw(format!( " Running: {} Completed: {} Failed: {}", running.len(), completed.len(), failed.len() ))); lines.push(Line::raw("")); // Running tasks with elapsed time if !running.is_empty() { for task in &running { let elapsed = task.started_at .map(|s| { let now = std::time::SystemTime::now() .duration_since(std::time::UNIX_EPOCH) .unwrap_or_default() .as_secs_f64(); format!("{}s", (now - s) as u64) }) .unwrap_or_default(); lines.push(Line::from(vec![ Span::raw(" "), Span::styled("●", Style::default().fg(Color::Green)), Span::raw(format!(" {} ({})", task.name, elapsed)), ])); } lines.push(Line::raw("")); } // Recent completed (last 10) if !completed.is_empty() { lines.push(Line::styled(" Recent:", dim)); for task in completed.iter().rev().take(10) { lines.push(Line::from(vec![ Span::raw(" "), Span::styled("✓", Style::default().fg(Color::Green)), Span::raw(format!(" {}", task.name)), ])); } } // Failed tasks if !failed.is_empty() { lines.push(Line::raw("")); lines.push(Line::styled(" Failed:", Style::default().fg(Color::Red))); for task in failed.iter().rev().take(5) { lines.push(Line::from(vec![ Span::raw(" "), Span::styled("✗", Style::default().fg(Color::Red)), Span::raw(format!(" {}", task.name)), ])); } } frame.render_widget( Paragraph::new(lines).wrap(Wrap { trim: false }), area, ); } }