Skip to main content

mnml_bridge/
lib.rs

1//! # mnml-bridge — Mount protocol for mnml sibling tools
2//!
3//! Bridge / Mount is the integration layer that lets sibling tools
4//! (`mnml-db-postgres`, `mnml-forge-bitbucket`, …) render their UI as a
5//! first-class pane inside mnml — owning the activity-bar icon, the
6//! rail content, and the editor body — instead of running as a
7//! plain `Pty` pane.
8//!
9//! ## The four tiers
10//!
11//! 1. **Env vars** — every Pty mnml spawns sees `MNML_WORKSPACE`,
12//!    `MNML_THEME`, and `MNML_IPC_DIR`. Zero protocol; just read on
13//!    startup. (Available today for any sibling.)
14//! 2. **JSONL sibling → host** — sibling writes JSONL commands to
15//!    `$MNML_IPC_DIR/command`; mnml ingests them. `toast`,
16//!    `open-pty`, `open` (file), more coming. One-way.
17//! 3. **mnml-bridge SDK** — this crate. Typed Rust API around tiers
18//!    1 + 2, plus the Mount protocol below.
19//! 4. **Mount** — sibling connects to a Unix-socket-per-mount,
20//!    streams cell+style frames back, receives input events. Owns
21//!    rail + body areas of an activity-bar section.
22//!
23//! ## Wire shape
24//!
25//! Length-prefixed JSON. Every message is a `Frame` or `Input`. The
26//! 4-byte little-endian length precedes the JSON body so framing is
27//! trivial (no streaming JSON parser needed).
28//!
29//! Host → Sibling:
30//!   - `MountHello { cols, rows }` first
31//!   - `Resize { cols, rows }` on terminal resize
32//!   - `Input { event }` on every routed key / mouse event
33//!
34//! Sibling → Host:
35//!   - `Frame { cells: Vec<Vec<Cell>> }` whenever the sibling has a
36//!     new screen state. Cell-perfect; the host stamps these into
37//!     its own ratatui frame.
38//!
39//! V1 keeps it simple: full frames, no diffing. A ~24x80 panel is
40//! ~2 KB of JSON; serialization cost is negligible vs ratatui's
41//! own draw cycle.
42
43use serde::{Deserialize, Serialize};
44
45#[cfg(feature = "client")]
46pub mod client;
47
48#[cfg(feature = "client")]
49pub use client::Mount;
50
51pub mod install;
52pub mod ipc;
53pub use install::{
54    ChipSpec, CommandSpec, ContextMenuEntry, IntegrationSpec, MenuBarEntry, NotificationsSpec,
55    OsNotifyPolicy, Requires, SettingsPage, StatuslineSpec, install_integration,
56    integration_manifest_path, list_installed_integrations, uninstall_integration,
57};
58pub use ipc::{
59    NotifyOpts, ProgressStatus, SegmentSide, ToastLevel, notify, progress_end, progress_start,
60    progress_update, register_command, set_activity_badge, statusline_clear_segment,
61    statusline_set_segment, toast, toast_dismiss, toast_error, toast_info, toast_persistent,
62    toast_warn,
63};
64
65/// A single terminal cell — one grapheme + style. Mirrors
66/// ratatui's `buffer::Cell` shape but with serde derived.
67#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
68pub struct Cell {
69    /// The grapheme cluster painted in this cell. Multi-codepoint
70    /// (e.g. flag emoji) is fine; mnml stamps the whole thing into
71    /// a single buffer cell.
72    pub symbol: String,
73    #[serde(default, skip_serializing_if = "Option::is_none")]
74    pub fg: Option<RgbOrIndex>,
75    #[serde(default, skip_serializing_if = "Option::is_none")]
76    pub bg: Option<RgbOrIndex>,
77    /// Bitfield of [`Modifier`] flags. Stored as u16 for compact wire shape.
78    #[serde(default, skip_serializing_if = "is_zero_u16")]
79    pub modifiers: u16,
80}
81
82fn is_zero_u16(v: &u16) -> bool {
83    *v == 0
84}
85
86/// Either a true-color RGB triple or a 256-color palette index.
87#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq)]
88#[serde(untagged)]
89pub enum RgbOrIndex {
90    /// `[r, g, b]` 24-bit color.
91    Rgb([u8; 3]),
92    /// 0-255 palette index (terminal default semantics: 0-7 ANSI,
93    /// 8-15 bright, 16-231 6×6×6 cube, 232-255 grayscale).
94    Index(u8),
95}
96
97/// Bitflags for [`Cell::modifiers`]. Mirrors ratatui's `Modifier`
98/// constants so a sibling can reuse its existing styling.
99pub mod modifier {
100    pub const BOLD: u16 = 1 << 0;
101    pub const DIM: u16 = 1 << 1;
102    pub const ITALIC: u16 = 1 << 2;
103    pub const UNDERLINED: u16 = 1 << 3;
104    pub const SLOW_BLINK: u16 = 1 << 4;
105    pub const RAPID_BLINK: u16 = 1 << 5;
106    pub const REVERSED: u16 = 1 << 6;
107    pub const HIDDEN: u16 = 1 << 7;
108    pub const CROSSED_OUT: u16 = 1 << 8;
109}
110
111/// Sent by the host once on connection, then on every terminal resize.
112#[derive(Debug, Clone, Copy, Serialize, Deserialize)]
113pub struct Geometry {
114    pub cols: u16,
115    pub rows: u16,
116}
117
118/// Routed input event from the host. Key / mouse events that
119/// happened inside the mount's area are forwarded as-is.
120#[derive(Debug, Clone, Serialize, Deserialize)]
121#[serde(tag = "kind", rename_all = "snake_case")]
122pub enum InputEvent {
123    /// A single keypress (key spec, e.g. `"down"`, `"ctrl+c"`).
124    Key { spec: String },
125    /// Mouse click. `button` is `"left" | "middle" | "right"`.
126    Click { col: u16, row: u16, button: String },
127    /// Mouse wheel. Positive `dy` ⇒ scroll up.
128    Scroll { col: u16, row: u16, dy: i16 },
129    /// Mouse hover (cursor moved over the mount).
130    Hover { col: u16, row: u16 },
131}
132
133/// Host → sibling messages.
134#[derive(Debug, Clone, Serialize, Deserialize)]
135#[serde(tag = "kind", rename_all = "snake_case")]
136pub enum HostMessage {
137    /// First message after connect — tells the sibling the initial
138    /// area size.
139    Hello { geometry: Geometry, theme: String },
140    /// Sent on terminal / pane resize.
141    Resize { geometry: Geometry },
142    /// Forwarded user input.
143    Input { event: InputEvent },
144    /// Host is going away (mnml quitting, mount being unmounted).
145    Goodbye,
146}
147
148/// Sibling → host messages.
149#[derive(Debug, Clone, Serialize, Deserialize)]
150#[serde(tag = "kind", rename_all = "snake_case")]
151pub enum SiblingMessage {
152    /// A full screen of cells. `cells.len()` rows × `cells[i].len()`
153    /// cols — must match the most recent `Hello`/`Resize` geometry.
154    /// Rows shorter than the advertised `cols` are right-padded
155    /// with default cells by the host.
156    Frame { cells: Vec<Vec<Cell>> },
157    /// Sibling is voluntarily exiting (clean shutdown).
158    Bye,
159}
160
161/// Read a length-prefixed JSON message from a stream.
162///
163/// Wire format: `[u8; 4]` little-endian length, then `length` bytes
164/// of UTF-8 JSON. Returns `Ok(None)` on clean EOF, `Err` on truncated
165/// reads or malformed JSON.
166pub fn read_message<R, T>(r: &mut R) -> std::io::Result<Option<T>>
167where
168    R: std::io::Read,
169    T: serde::de::DeserializeOwned,
170{
171    let mut len_buf = [0u8; 4];
172    match r.read_exact(&mut len_buf) {
173        Ok(()) => {}
174        Err(e) if e.kind() == std::io::ErrorKind::UnexpectedEof => return Ok(None),
175        Err(e) => return Err(e),
176    }
177    let len = u32::from_le_bytes(len_buf) as usize;
178    if len > 16 * 1024 * 1024 {
179        return Err(std::io::Error::new(
180            std::io::ErrorKind::InvalidData,
181            format!("bridge message too large: {len} bytes"),
182        ));
183    }
184    let mut body = vec![0u8; len];
185    r.read_exact(&mut body)?;
186    let parsed: T = serde_json::from_slice(&body).map_err(|e| {
187        std::io::Error::new(
188            std::io::ErrorKind::InvalidData,
189            format!("bridge JSON parse: {e}"),
190        )
191    })?;
192    Ok(Some(parsed))
193}
194
195/// Write a length-prefixed JSON message to a stream.
196pub fn write_message<W, T>(w: &mut W, msg: &T) -> std::io::Result<()>
197where
198    W: std::io::Write,
199    T: Serialize,
200{
201    let body = serde_json::to_vec(msg).map_err(|e| {
202        std::io::Error::new(
203            std::io::ErrorKind::InvalidData,
204            format!("bridge JSON serialize: {e}"),
205        )
206    })?;
207    let len = body.len() as u32;
208    w.write_all(&len.to_le_bytes())?;
209    w.write_all(&body)?;
210    Ok(())
211}
212
213#[cfg(test)]
214mod tests {
215    use super::*;
216
217    #[test]
218    fn frame_roundtrip() {
219        let frame = SiblingMessage::Frame {
220            cells: vec![vec![Cell {
221                symbol: "x".to_string(),
222                fg: Some(RgbOrIndex::Rgb([255, 0, 0])),
223                bg: None,
224                modifiers: modifier::BOLD,
225            }]],
226        };
227        let mut buf = Vec::new();
228        write_message(&mut buf, &frame).unwrap();
229        let mut cursor = std::io::Cursor::new(&buf);
230        let back: SiblingMessage = read_message(&mut cursor).unwrap().unwrap();
231        match back {
232            SiblingMessage::Frame { cells } => {
233                assert_eq!(cells.len(), 1);
234                assert_eq!(cells[0][0].symbol, "x");
235                assert_eq!(cells[0][0].fg, Some(RgbOrIndex::Rgb([255, 0, 0])));
236                assert_eq!(cells[0][0].modifiers, modifier::BOLD);
237            }
238            _ => panic!("wrong variant"),
239        }
240    }
241
242    #[test]
243    fn host_hello_roundtrip() {
244        let hello = HostMessage::Hello {
245            geometry: Geometry { cols: 80, rows: 24 },
246            theme: "cyberdream".to_string(),
247        };
248        let mut buf = Vec::new();
249        write_message(&mut buf, &hello).unwrap();
250        let mut cursor = std::io::Cursor::new(&buf);
251        let back: HostMessage = read_message(&mut cursor).unwrap().unwrap();
252        match back {
253            HostMessage::Hello { geometry, theme } => {
254                assert_eq!(geometry.cols, 80);
255                assert_eq!(geometry.rows, 24);
256                assert_eq!(theme, "cyberdream");
257            }
258            _ => panic!("wrong variant"),
259        }
260    }
261
262    #[test]
263    fn eof_returns_none() {
264        let mut empty = std::io::Cursor::new(Vec::<u8>::new());
265        let res: Option<HostMessage> = read_message(&mut empty).unwrap();
266        assert!(res.is_none());
267    }
268
269    #[test]
270    fn rejects_oversize_length() {
271        // 4-byte length = 100 MB; should be rejected before allocation.
272        let mut buf = (100u32 * 1024 * 1024).to_le_bytes().to_vec();
273        buf.extend_from_slice(b"junk");
274        let mut cursor = std::io::Cursor::new(&buf);
275        let res: std::io::Result<Option<HostMessage>> = read_message(&mut cursor);
276        assert!(res.is_err());
277    }
278}