refactor(monorepo): reorganización lógica + renames + SDDs + split CHANGELOG

Reorganización física de crates/:
- core/ (mezclaba 6 propósitos) se divide en protocol/, init/, runtime/, compat/
- shared/ (3 crates) se redistribuye en protocol/ e init/
- lapaloma (sub-módulo de ui_engine) se promueve a modules/pineal/

Renames de proyectos:
- shipote → shuma (runtime de sandboxes)
- nouser → akasha (explorador de Mónadas)
- yahweh → nahual (motor GPUI, antes ui_engine/)
- lapaloma → pineal (data-viz agnóstica)

Fraccionamiento UI → core agnóstico:
- vista-core (DeckState + snap, 175 LOC, 5 tests verdes)
- barra-core (Task + render_html + sanitize, 90 LOC, 5 tests verdes)
- vista-web y barra-web ahora son thin DOM bindings

Documentación nueva:
- 16 SDDs por subdirectorio (≤80 LOC c/u): protocol/init/runtime/compat
  + 10 módulos + apps/
- docs/STATUS.md con cifras reales por proyecto
- docs/ROADMAP.md con plan a finalización (6 hitos, ~6-8 semanas)
- CHANGELOG.md particionado en docs/changelog/<proyecto>.md (7 buckets)

Automatización:
- scripts/reorg.py — script idempotente que: git mv directorios, renombra
  package names, recomputa path = refs, reescribe imports rust, actualiza
  workspace Cargo.toml. Soporta --dry-run.
- scripts/split-changelog.py — particiona CHANGELOG por componente.

Validación:
- cargo check --workspace pasa (124 crates + 2 nuevos cores).
- 10 tests adicionales (5 en vista-core + 5 en barra-core) verdes.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
sergio
2026-05-19 14:48:34 +00:00
parent 86fb6ae20b
commit 550c98f275
375 changed files with 8512 additions and 7155 deletions
+21
View File
@@ -0,0 +1,21 @@
[package]
name = "shuma-gateway"
version.workspace = true
edition.workspace = true
rust-version.workspace = true
license.workspace = true
authors.workspace = true
publish.workspace = true
description = "HTTP/JSON gateway para shipote — traduce JSON ↔ postcard contra el admin socket."
[[bin]]
name = "shuma-gateway"
path = "src/main.rs"
[dependencies]
shuma-protocol = { path = "../../modules/shuma/shuma-protocol" }
anyhow = { workspace = true }
serde_json = { workspace = true }
tokio = { workspace = true }
tracing = { workspace = true }
tracing-subscriber = { workspace = true }
+168
View File
@@ -0,0 +1,168 @@
//! `shuma-gateway` — HTTP/JSON adapter para el daemon.
//!
//! Acepta `POST /rpc` con body JSON serializado como `shuma_protocol::Request`,
//! hace round-trip al admin socket via postcard, devuelve `Response` como JSON.
//!
//! Diseñado para clients no-Rust (curl, Python, web app) que no pueden
//! hablar postcard directo. NO es un proxy completo — sólo translation
//! layer del protocolo.
//!
//! Sin dep de axum/hyper: HTTP parser ad-hoc, suficiente para 1 endpoint.
use shuma_protocol::{default_socket_path, read_frame, write_frame, Request, Response};
use std::sync::Arc;
use tokio::io::{AsyncReadExt, AsyncWriteExt};
use tokio::net::{TcpListener, TcpStream, UnixStream};
use tracing::{info, warn};
const DEFAULT_LISTEN: &str = "127.0.0.1:7378";
#[tokio::main]
async fn main() -> anyhow::Result<()> {
init_tracing();
let listen = std::env::var("SHIPOTE_GATEWAY_LISTEN").unwrap_or_else(|_| DEFAULT_LISTEN.into());
let daemon_sock = Arc::new(default_socket_path());
let listener = TcpListener::bind(&listen).await?;
info!(listen = %listen, daemon = %daemon_sock.display(), "shuma-gateway listening");
loop {
match listener.accept().await {
Ok((stream, peer)) => {
let sock = daemon_sock.clone();
tokio::spawn(async move {
if let Err(e) = handle_http(stream, sock).await {
warn!(?e, ?peer, "request error");
}
});
}
Err(e) => {
warn!(?e, "accept failed");
tokio::time::sleep(std::time::Duration::from_millis(100)).await;
}
}
}
}
async fn handle_http(mut stream: TcpStream, daemon_sock: Arc<std::path::PathBuf>) -> anyhow::Result<()> {
// Parser HTTP mínimo: read hasta `\r\n\r\n`, parsear request line +
// Content-Length, después leer body exacto.
let mut buf = Vec::with_capacity(4096);
let mut tmp = [0u8; 4096];
let header_end;
loop {
let n = stream.read(&mut tmp).await?;
if n == 0 {
return Ok(()); // closed
}
buf.extend_from_slice(&tmp[..n]);
if let Some(pos) = find_double_crlf(&buf) {
header_end = pos + 4;
break;
}
if buf.len() > 64 * 1024 {
return write_error(&mut stream, 413, "headers too large").await;
}
}
let header_str = std::str::from_utf8(&buf[..header_end - 4])?;
let mut lines = header_str.lines();
let request_line = lines.next().unwrap_or("");
let mut parts = request_line.split_whitespace();
let method = parts.next().unwrap_or("");
let path = parts.next().unwrap_or("");
let mut content_length: usize = 0;
for line in lines {
if let Some(v) = line.strip_prefix("Content-Length:").or_else(|| line.strip_prefix("content-length:")) {
content_length = v.trim().parse().unwrap_or(0);
}
}
// Rutas:
if method == "GET" && (path == "/" || path == "/health") {
return write_text(&mut stream, 200, "shuma-gateway ok\n").await;
}
if method != "POST" || path != "/rpc" {
return write_error(&mut stream, 404, "use POST /rpc").await;
}
// Leer body.
let mut body = buf[header_end..].to_vec();
while body.len() < content_length {
let n = stream.read(&mut tmp).await?;
if n == 0 {
break;
}
body.extend_from_slice(&tmp[..n]);
}
body.truncate(content_length);
// Parsear JSON → Request.
let req: Request = match serde_json::from_slice(&body) {
Ok(r) => r,
Err(e) => return write_error(&mut stream, 400, &format!("bad json: {e}")).await,
};
// Round-trip al daemon.
let resp = match round_trip_daemon(&daemon_sock, &req).await {
Ok(r) => r,
Err(e) => return write_error(&mut stream, 502, &format!("daemon: {e}")).await,
};
// Serializar Response → JSON.
let body_json = serde_json::to_vec(&resp)?;
write_response(&mut stream, 200, "application/json", &body_json).await
}
async fn round_trip_daemon(sock: &std::path::Path, req: &Request) -> anyhow::Result<Response> {
let mut stream = UnixStream::connect(sock).await?;
write_frame(&mut stream, req).await?;
let resp: Response = read_frame(&mut stream).await?;
Ok(resp)
}
fn find_double_crlf(buf: &[u8]) -> Option<usize> {
buf.windows(4).position(|w| w == b"\r\n\r\n")
}
async fn write_response(
stream: &mut TcpStream,
code: u16,
content_type: &str,
body: &[u8],
) -> anyhow::Result<()> {
let status = match code {
200 => "OK",
400 => "Bad Request",
404 => "Not Found",
413 => "Payload Too Large",
502 => "Bad Gateway",
_ => "Unknown",
};
let head = format!(
"HTTP/1.1 {code} {status}\r\n\
Content-Type: {content_type}\r\n\
Content-Length: {}\r\n\
Connection: close\r\n\
\r\n",
body.len()
);
stream.write_all(head.as_bytes()).await?;
stream.write_all(body).await?;
stream.flush().await?;
Ok(())
}
async fn write_text(stream: &mut TcpStream, code: u16, body: &str) -> anyhow::Result<()> {
write_response(stream, code, "text/plain", body.as_bytes()).await
}
async fn write_error(stream: &mut TcpStream, code: u16, msg: &str) -> anyhow::Result<()> {
let body = serde_json::json!({ "error": msg }).to_string();
write_response(stream, code, "application/json", body.as_bytes()).await
}
fn init_tracing() {
use tracing_subscriber::{fmt, EnvFilter};
let filter = EnvFilter::try_from_env("SHIPOTE_GATEWAY_LOG").unwrap_or_else(|_| EnvFilter::new("info"));
fmt().with_env_filter(filter).init();
}