feat(brahman-broker-explorer): nueva app probe del broker brahman
Iter 14. Cierra otro frente: visibilidad del broker handshake. La app probe cada 5s vía await_provider_blocking y reporta 4 estados claros (Pending / Down / UpNoProvider / UpWithProvider). crates/apps/brahman-broker-explorer/: - Deps: brahman-handshake + brahman-sidecar + stack yahweh themed. - ProbeState enum con 4 variants. - Polling cx.spawn cada 5s; el probe blocking se ejecuta en cx.background_executor().spawn para no congelar el main thread. - Configuración via env: BRAHMAN_INIT_SOCKET, BRAHMAN_BROKER_PROBE_FLOW (default broker-health), BRAHMAN_BROKER_PROBE_TYPE (default ping). - UI: header probe info + theme switcher; banner permanente Error/Warning/Success según estado; stat card con accent color. - 2 tests sanity. Smoke run verificado: bootstrap OK, panic esperado sin display. Apps GUI themed: 5 (nakui-ui + 3 explorers + ahora broker-explorer). Limitaciones: observer Card se registra/desregistra en cada probe; no muestra lista global de Cards (handshake no expone API). Para timeline real de MatchEvents hace falta mantener el Client vivo entre probes — scope futuro. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -6,6 +6,70 @@ ratio/diff ver `git show <sha>`.
|
|||||||
|
|
||||||
## 2026-05-10
|
## 2026-05-10
|
||||||
|
|
||||||
|
### feat(brahman-broker-explorer): nueva app probe del broker brahman
|
||||||
|
Iter 14. Cierra otro frente: visibilidad del broker brahman (el
|
||||||
|
broker handshake que matchea Cards consumer/producer). Hasta ahora
|
||||||
|
no había forma de "ver" si el broker estaba up sin invocar otro
|
||||||
|
binario CLI. Ahora hay una app GUI que probe cada 5s y reporta 3
|
||||||
|
estados claros.
|
||||||
|
|
||||||
|
Crate nuevo `crates/apps/brahman-broker-explorer/`:
|
||||||
|
- **Deps**: `brahman-handshake`, `brahman-sidecar` + el stack
|
||||||
|
yahweh themed (theme + 3 widgets). Consume el mismo
|
||||||
|
`await_provider_blocking` que usa `nouser-explorer`.
|
||||||
|
- **`ProbeState` enum** con 4 variants:
|
||||||
|
- `Pending` (estado inicial al boot, antes del primer probe).
|
||||||
|
- `Down { reason }` — connect failed, broker no escucha.
|
||||||
|
- `UpNoProvider { flow }` — broker reachable + sin productor
|
||||||
|
para el flow probado dentro del timeout.
|
||||||
|
- `UpWithProvider { flow, producer_socket }` — broker reachable
|
||||||
|
+ matcheó algo, devuelve el socket del provider.
|
||||||
|
- **Polling loop** en `cx.spawn` cada 5s; el probe (que es
|
||||||
|
bloqueante porque internamente usa tokio runtime) se ejecuta en
|
||||||
|
`cx.background_executor().spawn(...)` para no congelar el main
|
||||||
|
thread del UI.
|
||||||
|
- **Configuración via env**:
|
||||||
|
- `BRAHMAN_INIT_SOCKET` — path del broker (default resuelto por
|
||||||
|
`brahman_handshake::transport`).
|
||||||
|
- `BRAHMAN_BROKER_PROBE_FLOW` — flow del Card observer
|
||||||
|
(default `broker-health`).
|
||||||
|
- `BRAHMAN_BROKER_PROBE_TYPE` — type name (default `ping`).
|
||||||
|
- **UI**: header con probe info + theme switcher; banner permanente
|
||||||
|
(Error/Warning/Success/none según estado) debajo del header;
|
||||||
|
stat card con accent color por estado y descripción.
|
||||||
|
- 2 tests sanity (default state es Pending; constants coherentes:
|
||||||
|
PROBE_TIMEOUT < POLL_INTERVAL >= 2s).
|
||||||
|
|
||||||
|
Smoke run del binario verificado: bootstrap completo OK, panic
|
||||||
|
esperado en open_window por falta de display.
|
||||||
|
|
||||||
|
Beneficio operativo:
|
||||||
|
- Si tenés un broker corriendo en `~/.local/share/brahman/init.sock`,
|
||||||
|
el explorer lo detecta + reporta estado verde con su socket.
|
||||||
|
- Si no hay broker, banner rojo + msg claro indicando el path
|
||||||
|
probado.
|
||||||
|
- Si hay broker pero ningún Card produce el flow probado, banner
|
||||||
|
amber — útil para distinguir "broker down" de "broker up,
|
||||||
|
no productor del tipo X".
|
||||||
|
- Apuntando el flow/type via env, podés monitor productores
|
||||||
|
específicos: ej. `BRAHMAN_BROKER_PROBE_FLOW=monad-list
|
||||||
|
BRAHMAN_BROKER_PROBE_TYPE=json` para ver si nouser está sirviendo.
|
||||||
|
|
||||||
|
Apps GUI integradas al stack themed: **5** (nakui-ui, nakui-explorer,
|
||||||
|
nouser-explorer, minga-explorer, brahman-broker-explorer).
|
||||||
|
|
||||||
|
Limitaciones documentadas:
|
||||||
|
- El observer registra una Card temporal en cada probe (cada 5s).
|
||||||
|
Eso ensucia un poco las estadísticas del broker (Cards
|
||||||
|
registradas/desregistradas). No impacta funcionalidad pero
|
||||||
|
inflama el log si el broker tiene observability habilitada.
|
||||||
|
- No muestra la **lista global** de Cards registradas en el broker
|
||||||
|
— el protocolo handshake actual no expone esa API. Para eso
|
||||||
|
habría que agregar un endpoint `ListSessions` al broker server.
|
||||||
|
- No mantiene un buffer de MatchEvents. Cada probe es independiente.
|
||||||
|
Para timeline de matches, hace falta mantener el Client vivo
|
||||||
|
entre probes — scope futuro.
|
||||||
|
|
||||||
### feat(yahweh-theme): persistencia de la preferencia de theme entre runs
|
### feat(yahweh-theme): persistencia de la preferencia de theme entre runs
|
||||||
Iter 13. El theme switcher ya cambiaba el chrome en runtime, pero
|
Iter 13. El theme switcher ya cambiaba el chrome en runtime, pero
|
||||||
al cerrar y reabrir la app el theme volvía a Nebula default. Ahora
|
al cerrar y reabrir la app el theme volvía a Nebula default. Ahora
|
||||||
|
|||||||
Generated
+13
@@ -1235,6 +1235,19 @@ dependencies = [
|
|||||||
"ulid",
|
"ulid",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "brahman-broker-explorer"
|
||||||
|
version = "0.1.0"
|
||||||
|
dependencies = [
|
||||||
|
"brahman-handshake",
|
||||||
|
"brahman-sidecar",
|
||||||
|
"gpui",
|
||||||
|
"yahweh-theme",
|
||||||
|
"yahweh-widget-banner",
|
||||||
|
"yahweh-widget-card",
|
||||||
|
"yahweh-widget-theme-switcher",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "brahman-card"
|
name = "brahman-card"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
|
|||||||
@@ -93,6 +93,7 @@ members = [
|
|||||||
"crates/apps/nakui-explorer",
|
"crates/apps/nakui-explorer",
|
||||||
"crates/apps/nakui-ui",
|
"crates/apps/nakui-ui",
|
||||||
"crates/apps/minga-explorer",
|
"crates/apps/minga-explorer",
|
||||||
|
"crates/apps/brahman-broker-explorer",
|
||||||
]
|
]
|
||||||
|
|
||||||
[workspace.package]
|
[workspace.package]
|
||||||
|
|||||||
@@ -0,0 +1,19 @@
|
|||||||
|
[package]
|
||||||
|
name = "brahman-broker-explorer"
|
||||||
|
version.workspace = true
|
||||||
|
edition.workspace = true
|
||||||
|
license.workspace = true
|
||||||
|
description = "Probe GUI del broker brahman: conecta cada N segundos vía await_provider_blocking con un Card observer agnóstico, reporta 3 estados (down / up sin provider / up con provider)."
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
brahman-handshake = { path = "../../core/brahman-handshake" }
|
||||||
|
brahman-sidecar = { path = "../../shared/brahman-sidecar" }
|
||||||
|
yahweh-theme = { path = "../../modules/ui_engine/libs/theme" }
|
||||||
|
yahweh-widget-banner = { path = "../../modules/ui_engine/widgets/banner" }
|
||||||
|
yahweh-widget-card = { path = "../../modules/ui_engine/widgets/card" }
|
||||||
|
yahweh-widget-theme-switcher = { path = "../../modules/ui_engine/widgets/theme-switcher" }
|
||||||
|
gpui = { workspace = true }
|
||||||
|
|
||||||
|
[[bin]]
|
||||||
|
name = "brahman-broker-explorer"
|
||||||
|
path = "src/main.rs"
|
||||||
@@ -0,0 +1,316 @@
|
|||||||
|
//! `brahman-broker-explorer` — probe GUI del broker brahman.
|
||||||
|
//!
|
||||||
|
//! Cada [`POLL_INTERVAL`] arma un Card observer agnóstico y lo
|
||||||
|
//! manda al broker via `brahman_sidecar::await_provider_blocking`
|
||||||
|
//! (que internamente abre tokio runtime + Unix socket + handshake).
|
||||||
|
//! Reporta 3 estados:
|
||||||
|
//!
|
||||||
|
//! - **Down**: connect failed (broker no escucha en el socket).
|
||||||
|
//! - **Up sin provider**: connect OK, pero el broker no encontró
|
||||||
|
//! productor para el flow probado dentro del timeout.
|
||||||
|
//! - **Up con provider**: connect OK + el broker matcheó algo →
|
||||||
|
//! muestra el `producer_service_socket` recibido.
|
||||||
|
//!
|
||||||
|
//! Configuración via env:
|
||||||
|
//! - `BRAHMAN_INIT_SOCKET` — path del socket del broker (default
|
||||||
|
//! resuelto por `brahman_handshake::transport`).
|
||||||
|
//! - `BRAHMAN_BROKER_PROBE_FLOW` — nombre del flow probe (default
|
||||||
|
//! `broker-health`).
|
||||||
|
//! - `BRAHMAN_BROKER_PROBE_TYPE` — type name del flow probe
|
||||||
|
//! (default `ping`).
|
||||||
|
//!
|
||||||
|
//! Usá un type name probable (ej. `monad-list:json`,
|
||||||
|
//! `event-log:tail`) para detectar productores específicos del
|
||||||
|
//! ecosistema.
|
||||||
|
|
||||||
|
use std::path::PathBuf;
|
||||||
|
use std::time::{Duration, Instant};
|
||||||
|
|
||||||
|
use brahman_handshake::transport;
|
||||||
|
use brahman_sidecar::{await_provider_blocking, build_consumer_card, ConsumerError};
|
||||||
|
use gpui::{
|
||||||
|
div, prelude::*, px, App, Application, Bounds, Context, IntoElement, Render, SharedString,
|
||||||
|
Window, WindowBounds, WindowOptions,
|
||||||
|
};
|
||||||
|
use yahweh_theme::Theme;
|
||||||
|
use yahweh_widget_banner::{banner_themed, Banner};
|
||||||
|
use yahweh_widget_card::card_themed;
|
||||||
|
use yahweh_widget_theme_switcher::theme_switcher;
|
||||||
|
|
||||||
|
const POLL_INTERVAL: Duration = Duration::from_secs(5);
|
||||||
|
const PROBE_TIMEOUT: Duration = Duration::from_secs(1);
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
Application::new().run(|cx: &mut App| {
|
||||||
|
Theme::install_default(cx);
|
||||||
|
let bounds = Bounds::centered(None, gpui::size(px(720.), px(480.)), cx);
|
||||||
|
cx.open_window(
|
||||||
|
WindowOptions {
|
||||||
|
window_bounds: Some(WindowBounds::Windowed(bounds)),
|
||||||
|
titlebar: Some(gpui::TitlebarOptions {
|
||||||
|
title: Some(SharedString::from("Brahman Broker — Probe")),
|
||||||
|
..Default::default()
|
||||||
|
}),
|
||||||
|
..Default::default()
|
||||||
|
},
|
||||||
|
|_w, cx| cx.new(Explorer::new),
|
||||||
|
)
|
||||||
|
.expect("open window");
|
||||||
|
cx.activate(true);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Snapshot de un probe.
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
enum ProbeState {
|
||||||
|
/// Aún no probó (boot, primer ciclo).
|
||||||
|
Pending,
|
||||||
|
/// Connect failed → broker no responde en el path.
|
||||||
|
Down { reason: String },
|
||||||
|
/// Connect OK, sin matching producer dentro del timeout.
|
||||||
|
UpNoProvider { flow: String },
|
||||||
|
/// Connect OK, broker matcheó al menos un producer.
|
||||||
|
UpWithProvider {
|
||||||
|
flow: String,
|
||||||
|
producer_socket: PathBuf,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Explorer {
|
||||||
|
socket_path: PathBuf,
|
||||||
|
flow: String,
|
||||||
|
type_name: String,
|
||||||
|
state: ProbeState,
|
||||||
|
last_probe_ms: u64,
|
||||||
|
last_probe_at: Option<Instant>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Explorer {
|
||||||
|
fn new(cx: &mut Context<Self>) -> Self {
|
||||||
|
let socket_path = transport::default_socket_path();
|
||||||
|
let flow = std::env::var("BRAHMAN_BROKER_PROBE_FLOW")
|
||||||
|
.unwrap_or_else(|_| "broker-health".to_string());
|
||||||
|
let type_name = std::env::var("BRAHMAN_BROKER_PROBE_TYPE")
|
||||||
|
.unwrap_or_else(|_| "ping".to_string());
|
||||||
|
|
||||||
|
let flow_for_loop = flow.clone();
|
||||||
|
let type_for_loop = type_name.clone();
|
||||||
|
cx.spawn(async move |this, cx| {
|
||||||
|
let timer = cx.background_executor().clone();
|
||||||
|
let bg = cx.background_executor().clone();
|
||||||
|
loop {
|
||||||
|
let card = build_consumer_card(
|
||||||
|
"brahman-broker-explorer",
|
||||||
|
flow_for_loop.clone(),
|
||||||
|
type_for_loop.clone(),
|
||||||
|
);
|
||||||
|
let started = Instant::now();
|
||||||
|
// El probe es bloqueante (interno tokio runtime); va
|
||||||
|
// al background executor para no congelar el main.
|
||||||
|
let probe_flow = flow_for_loop.clone();
|
||||||
|
let result = bg
|
||||||
|
.spawn(async move { await_provider_blocking(card, PROBE_TIMEOUT) })
|
||||||
|
.await;
|
||||||
|
let elapsed = started.elapsed().as_millis() as u64;
|
||||||
|
|
||||||
|
let new_state = match result {
|
||||||
|
Ok(socket) => ProbeState::UpWithProvider {
|
||||||
|
flow: probe_flow.clone(),
|
||||||
|
producer_socket: socket,
|
||||||
|
},
|
||||||
|
Err(ConsumerError::NoProvider { .. }) => ProbeState::UpNoProvider {
|
||||||
|
flow: probe_flow.clone(),
|
||||||
|
},
|
||||||
|
Err(e) => ProbeState::Down {
|
||||||
|
reason: e.to_string(),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
let _ = this.update(cx, |me, cx| {
|
||||||
|
me.state = new_state;
|
||||||
|
me.last_probe_ms = elapsed;
|
||||||
|
me.last_probe_at = Some(Instant::now());
|
||||||
|
cx.notify();
|
||||||
|
});
|
||||||
|
|
||||||
|
timer.timer(POLL_INTERVAL).await;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.detach();
|
||||||
|
|
||||||
|
Self {
|
||||||
|
socket_path,
|
||||||
|
flow,
|
||||||
|
type_name,
|
||||||
|
state: ProbeState::Pending,
|
||||||
|
last_probe_ms: 0,
|
||||||
|
last_probe_at: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Render for Explorer {
|
||||||
|
fn render(&mut self, _w: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
||||||
|
let theme = Theme::global(cx).clone();
|
||||||
|
let bg = theme.bg_app.clone();
|
||||||
|
let text = theme.fg_text;
|
||||||
|
let text_dim = theme.fg_muted;
|
||||||
|
// Acentos por estado: verde = arriba, amber = arriba sin
|
||||||
|
// provider, rojo = abajo, gris = pending.
|
||||||
|
let accent_up = gpui::rgb(0xa3be8c);
|
||||||
|
let accent_partial = gpui::rgb(0xebcb8b);
|
||||||
|
let accent_down = gpui::rgb(0xbf616a);
|
||||||
|
let accent_pending = gpui::rgb(0x6a7280);
|
||||||
|
|
||||||
|
let header_text = format!(
|
||||||
|
"Probe: {} · flow: {}/{} · reload {} ms",
|
||||||
|
self.socket_path.display(),
|
||||||
|
self.flow,
|
||||||
|
self.type_name,
|
||||||
|
self.last_probe_ms,
|
||||||
|
);
|
||||||
|
|
||||||
|
let header = div()
|
||||||
|
.flex()
|
||||||
|
.flex_row()
|
||||||
|
.items_center()
|
||||||
|
.px(px(16.))
|
||||||
|
.py(px(12.))
|
||||||
|
.bg(theme.bg_panel.clone())
|
||||||
|
.border_b_1()
|
||||||
|
.border_color(theme.border)
|
||||||
|
.text_color(text)
|
||||||
|
.text_size(px(14.))
|
||||||
|
.child(div().flex_grow().child(header_text))
|
||||||
|
.child(theme_switcher(cx));
|
||||||
|
|
||||||
|
// Banner permanente debajo del header con el estado actual.
|
||||||
|
// Severidad acorde al kind.
|
||||||
|
let status_banner = match &self.state {
|
||||||
|
ProbeState::Pending => None,
|
||||||
|
ProbeState::Down { reason } => Some(banner_themed(
|
||||||
|
cx,
|
||||||
|
Banner::Error,
|
||||||
|
SharedString::from(format!("Broker DOWN — {reason}")),
|
||||||
|
)),
|
||||||
|
ProbeState::UpNoProvider { .. } => Some(banner_themed(
|
||||||
|
cx,
|
||||||
|
Banner::Warning,
|
||||||
|
SharedString::from("Broker UP, sin provider para el flow"),
|
||||||
|
)),
|
||||||
|
ProbeState::UpWithProvider { .. } => Some(banner_themed(
|
||||||
|
cx,
|
||||||
|
Banner::Success,
|
||||||
|
SharedString::from("Broker UP, provider matcheado"),
|
||||||
|
)),
|
||||||
|
};
|
||||||
|
|
||||||
|
let body = div()
|
||||||
|
.flex()
|
||||||
|
.flex_col()
|
||||||
|
.gap(px(8.))
|
||||||
|
.px(px(16.))
|
||||||
|
.py(px(16.))
|
||||||
|
.child(state_card(cx, &self.state, text, text_dim, accent_up,
|
||||||
|
accent_partial, accent_down, accent_pending));
|
||||||
|
|
||||||
|
div()
|
||||||
|
.flex()
|
||||||
|
.flex_col()
|
||||||
|
.size_full()
|
||||||
|
.bg(bg)
|
||||||
|
.child(header)
|
||||||
|
.when_some(status_banner, |d, b| d.child(b))
|
||||||
|
.child(body)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
fn state_card(
|
||||||
|
cx: &mut Context<Explorer>,
|
||||||
|
state: &ProbeState,
|
||||||
|
text: gpui::Hsla,
|
||||||
|
text_dim: gpui::Hsla,
|
||||||
|
accent_up: gpui::Rgba,
|
||||||
|
accent_partial: gpui::Rgba,
|
||||||
|
accent_down: gpui::Rgba,
|
||||||
|
accent_pending: gpui::Rgba,
|
||||||
|
) -> impl IntoElement {
|
||||||
|
let (label, accent, value, description): (&str, gpui::Rgba, String, String) = match state {
|
||||||
|
ProbeState::Pending => (
|
||||||
|
"Estado",
|
||||||
|
accent_pending,
|
||||||
|
"PENDING".into(),
|
||||||
|
"esperando primer probe…".into(),
|
||||||
|
),
|
||||||
|
ProbeState::Down { reason } => (
|
||||||
|
"Estado",
|
||||||
|
accent_down,
|
||||||
|
"DOWN".into(),
|
||||||
|
format!("connect failed: {reason}"),
|
||||||
|
),
|
||||||
|
ProbeState::UpNoProvider { flow } => (
|
||||||
|
"Estado",
|
||||||
|
accent_partial,
|
||||||
|
"UP / NO PROVIDER".into(),
|
||||||
|
format!("broker reachable; sin productor para flow `{flow}`"),
|
||||||
|
),
|
||||||
|
ProbeState::UpWithProvider {
|
||||||
|
flow,
|
||||||
|
producer_socket,
|
||||||
|
} => (
|
||||||
|
"Estado",
|
||||||
|
accent_up,
|
||||||
|
"UP / PROVIDER".into(),
|
||||||
|
format!(
|
||||||
|
"flow `{flow}` matcheado en producer socket: {}",
|
||||||
|
producer_socket.display()
|
||||||
|
),
|
||||||
|
),
|
||||||
|
};
|
||||||
|
|
||||||
|
card_themed(cx)
|
||||||
|
.border_l_4()
|
||||||
|
.border_color(accent)
|
||||||
|
.child(
|
||||||
|
div()
|
||||||
|
.text_color(accent)
|
||||||
|
.text_size(px(11.))
|
||||||
|
.child(SharedString::from(label.to_string())),
|
||||||
|
)
|
||||||
|
.child(
|
||||||
|
div()
|
||||||
|
.text_color(text)
|
||||||
|
.text_size(px(28.))
|
||||||
|
.child(SharedString::from(value)),
|
||||||
|
)
|
||||||
|
.child(
|
||||||
|
div()
|
||||||
|
.text_color(text_dim)
|
||||||
|
.text_size(px(11.))
|
||||||
|
.child(SharedString::from(description)),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn pending_is_default_state_at_boot() {
|
||||||
|
// El estado inicial DEBE ser Pending — sino el banner del
|
||||||
|
// header arrancaría con un Error o Warning sin haber probado
|
||||||
|
// (mensaje engañoso).
|
||||||
|
let s = ProbeState::Pending;
|
||||||
|
assert!(matches!(s, ProbeState::Pending));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn poll_and_probe_constants_are_sane() {
|
||||||
|
// Sanity: el timeout del probe DEBE ser menor que el
|
||||||
|
// intervalo de polling, sino los probes se solapan.
|
||||||
|
assert!(PROBE_TIMEOUT < POLL_INTERVAL);
|
||||||
|
// El intervalo no debería ser tan corto que sature al broker.
|
||||||
|
assert!(POLL_INTERVAL >= Duration::from_secs(2));
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user