chore: monorepo inicial con arje + minga + yahweh absorbidos
Workspace en 4 ejes (core/modules/apps/shared):
- core/: 24 crates de arje (Init systemd-compatible: ente-card, ente-zero,
ente-kernel, ente-bus, ente-cas, ente-soma, ente-wasm, ente-snapshot,
ente-brain, ente-echo, ente-policy-provider, + 12 crates *-compat)
- modules/semantic_dht/: 5 crates de minga (minga-core con AST/CAS/MST,
minga-p2p con libp2p Kad, minga-store, minga-vfs, minga-cli)
- modules/ui_engine/: 11 crates de yahweh (libs/{core,theme,bus,providers},
widgets/{tree,splitter,tabs,tiled,container_core,text_input})
- apps/: 5 crates de yahweh (file_explorer, database_explorer, text_viewer,
image_viewer, yahweh-shell)
- shared_wit/protocol.wit: handshake/lifecycle inicial
Cargo.toml unificado: thiserror bumped a 2 (transparente para arje), tokio
"full", paths intra-workspace de yahweh redirigidos a su nueva ubicación.
cargo check --workspace: 0 errores, 17 warnings (dead code preexistente).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,22 @@
|
||||
[package]
|
||||
name = "ente-bus"
|
||||
version = "0.0.1"
|
||||
edition.workspace = true
|
||||
license.workspace = true
|
||||
publish.workspace = true
|
||||
|
||||
[dependencies]
|
||||
ente-card = { path = "../ente-card" }
|
||||
serde = { workspace = true }
|
||||
postcard = { workspace = true }
|
||||
ulid = { workspace = true }
|
||||
tokio = { workspace = true }
|
||||
anyhow = { workspace = true }
|
||||
tracing = { workspace = true }
|
||||
|
||||
[dev-dependencies]
|
||||
ente-echo = { path = "../ente-echo" }
|
||||
|
||||
[[example]]
|
||||
name = "busctl"
|
||||
path = "examples/busctl.rs"
|
||||
@@ -0,0 +1,52 @@
|
||||
//! busctl: cliente CLI para el bus interno del fractal.
|
||||
//!
|
||||
//! Uso:
|
||||
//! cargo run --example busctl -- list-entes
|
||||
//! cargo run --example busctl -- announce
|
||||
//! cargo run --example busctl -- power-off
|
||||
//!
|
||||
//! Si `ENTE_BUS_SOCK` no está en el entorno, cae al path dev por defecto.
|
||||
|
||||
use ente_bus::{BusClient, BusRequest};
|
||||
use std::env;
|
||||
|
||||
#[tokio::main(flavor = "current_thread")]
|
||||
async fn main() -> anyhow::Result<()> {
|
||||
let args: Vec<String> = env::args().collect();
|
||||
let cmd = args.get(1).map(|s| s.as_str()).unwrap_or("list-entes");
|
||||
|
||||
let mut client = match BusClient::from_env().await {
|
||||
Ok(c) => c,
|
||||
Err(_) => {
|
||||
let user = env::var("USER").unwrap_or_else(|_| "ente".into());
|
||||
let runtime = env::var("XDG_RUNTIME_DIR")
|
||||
.unwrap_or_else(|_| env::var("TMPDIR").unwrap_or_else(|_| "/tmp".into()));
|
||||
let path = format!("{runtime}/ente-bus-{user}.sock");
|
||||
eprintln!("ENTE_BUS_SOCK no definido, intentando {path}");
|
||||
BusClient::connect(&path).await?
|
||||
}
|
||||
};
|
||||
|
||||
let req = match cmd {
|
||||
"list-entes" => BusRequest::ListEntes,
|
||||
"announce" => BusRequest::Announce { capabilities: vec![] },
|
||||
"power-off" => BusRequest::PowerOff { interactive: false },
|
||||
"reboot" => BusRequest::Reboot { interactive: false },
|
||||
"suspend" => BusRequest::Suspend { interactive: false },
|
||||
"invoke-echo" => {
|
||||
let msg = args.get(2).map(|s| s.as_str()).unwrap_or("hola");
|
||||
BusRequest::Invoke {
|
||||
cap: ente_echo::echo_capability(),
|
||||
blob: msg.as_bytes().to_vec(),
|
||||
}
|
||||
}
|
||||
other => {
|
||||
eprintln!("subcomando desconocido: {other}");
|
||||
eprintln!("válidos: list-entes, announce, power-off, reboot, suspend, invoke-echo <text>");
|
||||
std::process::exit(2);
|
||||
}
|
||||
};
|
||||
let resp = client.call(req).await?;
|
||||
println!("{resp:#?}");
|
||||
Ok(())
|
||||
}
|
||||
@@ -0,0 +1,228 @@
|
||||
//! ente-bus: bus de capacidades interno del fractal.
|
||||
//!
|
||||
//! Wire format: Unix SOCK_STREAM con framing length-prefijo (u32 BE) + payload
|
||||
//! postcard. Bidireccional pero por ahora request-response síncrono.
|
||||
//!
|
||||
//! Identidad: cada Ente hijo recibe `ENTE_BUS_SOCK` y `ENTE_ID` en su entorno.
|
||||
//! El cliente lee ambos vía `BusClient::from_env`.
|
||||
|
||||
use ente_card::Capability;
|
||||
use serde::{Deserialize, Serialize};
|
||||
use std::path::Path;
|
||||
use std::str::FromStr;
|
||||
use tokio::io::{AsyncReadExt, AsyncWriteExt};
|
||||
use tokio::net::UnixStream;
|
||||
use ulid::Ulid;
|
||||
|
||||
pub const ENV_BUS_SOCK: &str = "ENTE_BUS_SOCK";
|
||||
pub const ENV_ENTE_ID: &str = "ENTE_ID";
|
||||
pub const MAX_FRAME: usize = 1 << 20; // 1 MiB — protección contra OOM
|
||||
|
||||
/// Interface UUID para decisiones de policy. Un Ente independiente
|
||||
/// (separado de polkit-compat) se anuncia como proveedor de
|
||||
/// `Capability::Endpoint { interface: POLKIT_DECISION_IFACE, version: 1 }`
|
||||
/// para arbitrar autorizaciones. Recibe blob:
|
||||
/// `pid_be | uid_be | action_id_utf8` → responde 1 byte: 1=allow, 0=deny.
|
||||
pub const POLKIT_DECISION_IFACE: ente_card::InterfaceId =
|
||||
ente_card::InterfaceId([0xb0; 16]);
|
||||
|
||||
/// Interface UUID auto-anunciado por compat-polkit. Diferente al de
|
||||
/// decisión para evitar recursión (polkit-compat invoca DECISION pero
|
||||
/// no es proveedor de DECISION; se anuncia como SERVICE).
|
||||
pub const POLKIT_SERVICE_IFACE: ente_card::InterfaceId =
|
||||
ente_card::InterfaceId([0xa4; 16]);
|
||||
|
||||
/// Credenciales del peer extraídas vía SO_PEERCRED al accept del bus.
|
||||
/// Imposibles de falsear desde el cliente — el kernel las inyecta.
|
||||
/// Definidas aquí (no en ente-zero) porque conceptualmente son atributo
|
||||
/// del protocolo del bus, no del init.
|
||||
#[derive(Debug, Clone, Copy)]
|
||||
pub struct PeerCreds {
|
||||
pub pid: i32,
|
||||
pub uid: u32,
|
||||
pub gid: u32,
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub struct BusMessage {
|
||||
pub from: Option<Ulid>,
|
||||
pub seq: u64,
|
||||
pub payload: BusPayload,
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub enum BusPayload {
|
||||
Request(BusRequest),
|
||||
Response(BusResponse),
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub enum BusRequest {
|
||||
/// Saludo. El Ente anuncia que está vivo y declara sus capacidades.
|
||||
/// El Init usa esto para saber que el child arrancó correctamente,
|
||||
/// independientemente de su exit code.
|
||||
Announce { capabilities: Vec<Capability> },
|
||||
|
||||
/// Listar Entes vivos. Útil para debugging y para Entes-supervisor.
|
||||
ListEntes,
|
||||
|
||||
/// Control de estado del fractal. Traducido desde D-Bus por compat-logind.
|
||||
PowerOff { interactive: bool },
|
||||
Reboot { interactive: bool },
|
||||
Suspend { interactive: bool },
|
||||
Hibernate { interactive: bool },
|
||||
|
||||
/// Invocación genérica de capacidad. `cap` debe estar provista por algún
|
||||
/// Ente del grafo; el blob es el argumento opaco que el proveedor parsea.
|
||||
Invoke { cap: Capability, blob: Vec<u8> },
|
||||
|
||||
/// Actualización dinámica del set de capacidades del Ente que llama.
|
||||
/// Sólo aplicable al `from_authenticated` — un Ente sólo puede modificar
|
||||
/// sus propias caps. La Card original (immutable) no se toca; la mutación
|
||||
/// va al `dynamic_provides` del Incarnated.
|
||||
UpdateCapabilities {
|
||||
adds: Vec<Capability>,
|
||||
removes: Vec<Capability>,
|
||||
},
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub enum BusResponse {
|
||||
Ok,
|
||||
Error(String),
|
||||
Entes(Vec<EnteInfo>),
|
||||
Invoked { result: Vec<u8> },
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||
pub struct EnteInfo {
|
||||
pub id: Ulid,
|
||||
pub label: String,
|
||||
pub provides: Vec<Capability>,
|
||||
pub pid: Option<i32>,
|
||||
}
|
||||
|
||||
pub async fn write_frame<W: AsyncWriteExt + Unpin>(w: &mut W, msg: &BusMessage) -> anyhow::Result<()> {
|
||||
let bytes = postcard::to_stdvec(msg)?;
|
||||
if bytes.len() > MAX_FRAME {
|
||||
anyhow::bail!("frame too large: {} > {}", bytes.len(), MAX_FRAME);
|
||||
}
|
||||
w.write_u32(bytes.len() as u32).await?;
|
||||
w.write_all(&bytes).await?;
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub async fn read_frame<R: AsyncReadExt + Unpin>(r: &mut R) -> anyhow::Result<BusMessage> {
|
||||
let len = r.read_u32().await? as usize;
|
||||
if len > MAX_FRAME {
|
||||
anyhow::bail!("frame oversize: {len}");
|
||||
}
|
||||
let mut buf = vec![0u8; len];
|
||||
r.read_exact(&mut buf).await?;
|
||||
Ok(postcard::from_bytes(&buf)?)
|
||||
}
|
||||
|
||||
pub struct BusClient {
|
||||
stream: UnixStream,
|
||||
seq: u64,
|
||||
self_id: Option<Ulid>,
|
||||
}
|
||||
|
||||
/// Trait que un Ente proveedor implementa para servir invokes que el bus le
|
||||
/// forwarda. Sync por simplicidad — un handler async se cubre con
|
||||
/// `tokio::task::block_in_place` o un canal hacia un task externo.
|
||||
pub trait InvokeHandler {
|
||||
fn handle(&mut self, cap: ente_card::Capability, blob: Vec<u8>) -> BusResponse;
|
||||
}
|
||||
|
||||
/// Conexión long-lived para Entes que proveen capacidades. A diferencia de
|
||||
/// `BusClient` (request-response y desconecta), `BusServer`:
|
||||
/// 1. Anuncia su identidad al bus
|
||||
/// 2. Mantiene la conexión abierta
|
||||
/// 3. Atiende invokes forwardeados por el bus en bucle
|
||||
pub struct BusServer {
|
||||
stream: UnixStream,
|
||||
self_id: Ulid,
|
||||
}
|
||||
|
||||
impl BusServer {
|
||||
pub async fn from_env() -> anyhow::Result<Self> {
|
||||
let path = std::env::var(ENV_BUS_SOCK)
|
||||
.map_err(|_| anyhow::anyhow!("{} no definido", ENV_BUS_SOCK))?;
|
||||
let id_s = std::env::var(ENV_ENTE_ID)
|
||||
.map_err(|_| anyhow::anyhow!("{} no definido", ENV_ENTE_ID))?;
|
||||
let self_id = Ulid::from_str(&id_s)
|
||||
.map_err(|_| anyhow::anyhow!("{} no es un Ulid válido: {id_s}", ENV_ENTE_ID))?;
|
||||
let stream = UnixStream::connect(&path).await?;
|
||||
Ok(Self { stream, self_id })
|
||||
}
|
||||
|
||||
pub async fn announce(&mut self, capabilities: Vec<ente_card::Capability>) -> anyhow::Result<()> {
|
||||
let req = BusMessage {
|
||||
from: Some(self.self_id),
|
||||
seq: 1,
|
||||
payload: BusPayload::Request(BusRequest::Announce { capabilities }),
|
||||
};
|
||||
write_frame(&mut self.stream, &req).await?;
|
||||
let resp = read_frame(&mut self.stream).await?;
|
||||
match resp.payload {
|
||||
BusPayload::Response(BusResponse::Ok) => Ok(()),
|
||||
BusPayload::Response(other) => anyhow::bail!("Announce rechazado: {other:?}"),
|
||||
BusPayload::Request(_) => anyhow::bail!("expected Response, got Request"),
|
||||
}
|
||||
}
|
||||
|
||||
/// Bucle principal del proveedor. Atiende invokes hasta que la conexión
|
||||
/// se cierra (el bus muere o el Ente recibe SIGTERM y nosotros dropeamos).
|
||||
pub async fn serve<H: InvokeHandler>(mut self, mut handler: H) -> anyhow::Result<()> {
|
||||
loop {
|
||||
let msg = read_frame(&mut self.stream).await?;
|
||||
let resp = match msg.payload {
|
||||
BusPayload::Request(BusRequest::Invoke { cap, blob }) => {
|
||||
handler.handle(cap, blob)
|
||||
}
|
||||
BusPayload::Request(other) => {
|
||||
BusResponse::Error(format!("BusServer no maneja {other:?}"))
|
||||
}
|
||||
BusPayload::Response(_) => continue,
|
||||
};
|
||||
let out = BusMessage {
|
||||
from: Some(self.self_id),
|
||||
seq: msg.seq,
|
||||
payload: BusPayload::Response(resp),
|
||||
};
|
||||
write_frame(&mut self.stream, &out).await?;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl BusClient {
|
||||
pub async fn connect(path: impl AsRef<Path>) -> anyhow::Result<Self> {
|
||||
let stream = UnixStream::connect(path).await?;
|
||||
let self_id = std::env::var(ENV_ENTE_ID)
|
||||
.ok()
|
||||
.and_then(|s| Ulid::from_str(&s).ok());
|
||||
Ok(Self { stream, seq: 0, self_id })
|
||||
}
|
||||
|
||||
pub async fn from_env() -> anyhow::Result<Self> {
|
||||
let path = std::env::var(ENV_BUS_SOCK)
|
||||
.map_err(|_| anyhow::anyhow!("{} no definido", ENV_BUS_SOCK))?;
|
||||
Self::connect(&path).await
|
||||
}
|
||||
|
||||
pub async fn call(&mut self, req: BusRequest) -> anyhow::Result<BusResponse> {
|
||||
self.seq = self.seq.wrapping_add(1);
|
||||
let msg = BusMessage {
|
||||
from: self.self_id,
|
||||
seq: self.seq,
|
||||
payload: BusPayload::Request(req),
|
||||
};
|
||||
write_frame(&mut self.stream, &msg).await?;
|
||||
let resp = read_frame(&mut self.stream).await?;
|
||||
match resp.payload {
|
||||
BusPayload::Response(r) => Ok(r),
|
||||
BusPayload::Request(_) => anyhow::bail!("expected response, got request"),
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user