2026-02-26 15:53:09 -08:00
|
|
|
pub mod ipc;
|
|
|
|
|
pub mod messages;
|
|
|
|
|
|
|
|
|
|
use std::io::{BufRead, BufReader, Write};
|
2026-02-26 16:50:14 -08:00
|
|
|
use std::process::{Child, ChildStdin, Command, Stdio};
|
|
|
|
|
use std::sync::{Mutex, OnceLock};
|
2026-02-26 15:53:09 -08:00
|
|
|
|
|
|
|
|
use crate::sidecar::messages::IPCMessage;
|
|
|
|
|
|
2026-02-26 16:50:14 -08:00
|
|
|
/// Get the global sidecar manager singleton.
|
|
|
|
|
pub fn sidecar() -> &'static SidecarManager {
|
|
|
|
|
static INSTANCE: OnceLock<SidecarManager> = OnceLock::new();
|
|
|
|
|
INSTANCE.get_or_init(SidecarManager::new)
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-26 15:53:09 -08:00
|
|
|
/// Manages the Python sidecar process lifecycle.
|
2026-02-26 16:50:14 -08:00
|
|
|
/// Uses separated stdin/stdout ownership to avoid BufReader conflicts.
|
2026-02-26 15:53:09 -08:00
|
|
|
pub struct SidecarManager {
|
|
|
|
|
process: Mutex<Option<Child>>,
|
2026-02-26 16:50:14 -08:00
|
|
|
stdin: Mutex<Option<ChildStdin>>,
|
|
|
|
|
reader: Mutex<Option<BufReader<std::process::ChildStdout>>>,
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl SidecarManager {
|
|
|
|
|
pub fn new() -> Self {
|
|
|
|
|
Self {
|
|
|
|
|
process: Mutex::new(None),
|
2026-02-26 16:50:14 -08:00
|
|
|
stdin: Mutex::new(None),
|
|
|
|
|
reader: Mutex::new(None),
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-26 16:50:14 -08:00
|
|
|
/// Ensure the sidecar is running, starting it if needed.
|
|
|
|
|
pub fn ensure_running(&self) -> Result<(), String> {
|
|
|
|
|
if self.is_running() {
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let python_path = std::env::current_dir()
|
|
|
|
|
.map_err(|e| e.to_string())?
|
|
|
|
|
.join("../python")
|
|
|
|
|
.canonicalize()
|
|
|
|
|
.map_err(|e| format!("Cannot find python directory: {e}"))?;
|
|
|
|
|
|
|
|
|
|
self.start(&python_path.to_string_lossy())
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-26 15:53:09 -08:00
|
|
|
/// Spawn the Python sidecar process.
|
|
|
|
|
pub fn start(&self, python_path: &str) -> Result<(), String> {
|
2026-02-26 16:50:14 -08:00
|
|
|
// Stop existing process if any
|
|
|
|
|
self.stop().ok();
|
|
|
|
|
|
|
|
|
|
let mut child = Command::new("python3")
|
2026-02-26 15:53:09 -08:00
|
|
|
.arg("-m")
|
|
|
|
|
.arg("voice_to_notes.main")
|
|
|
|
|
.current_dir(python_path)
|
|
|
|
|
.env("PYTHONPATH", python_path)
|
|
|
|
|
.stdin(Stdio::piped())
|
|
|
|
|
.stdout(Stdio::piped())
|
2026-02-26 16:50:14 -08:00
|
|
|
.stderr(Stdio::inherit())
|
2026-02-26 15:53:09 -08:00
|
|
|
.spawn()
|
|
|
|
|
.map_err(|e| format!("Failed to start sidecar: {e}"))?;
|
|
|
|
|
|
2026-02-26 16:50:14 -08:00
|
|
|
// Take ownership of stdin and stdout separately
|
|
|
|
|
let stdin = child.stdin.take().ok_or("Failed to get sidecar stdin")?;
|
|
|
|
|
let stdout = child.stdout.take().ok_or("Failed to get sidecar stdout")?;
|
|
|
|
|
let buf_reader = BufReader::new(stdout);
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
let mut proc = self.process.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
*proc = Some(child);
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
let mut s = self.stdin.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
*s = Some(stdin);
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
let mut r = self.reader.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
*r = Some(buf_reader);
|
|
|
|
|
}
|
2026-02-26 15:53:09 -08:00
|
|
|
|
|
|
|
|
// Wait for the "ready" message
|
|
|
|
|
self.wait_for_ready()?;
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Wait for the sidecar to send its ready message.
|
|
|
|
|
fn wait_for_ready(&self) -> Result<(), String> {
|
2026-02-26 16:50:14 -08:00
|
|
|
let mut reader_guard = self.reader.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
if let Some(ref mut reader) = *reader_guard {
|
|
|
|
|
let mut line = String::new();
|
|
|
|
|
loop {
|
|
|
|
|
line.clear();
|
|
|
|
|
let bytes = reader
|
|
|
|
|
.read_line(&mut line)
|
|
|
|
|
.map_err(|e| format!("Read error: {e}"))?;
|
|
|
|
|
if bytes == 0 {
|
|
|
|
|
return Err("Sidecar closed stdout before sending ready".to_string());
|
|
|
|
|
}
|
|
|
|
|
let trimmed = line.trim();
|
|
|
|
|
if trimmed.is_empty() {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if let Ok(msg) = serde_json::from_str::<IPCMessage>(trimmed) {
|
|
|
|
|
if msg.msg_type == "ready" {
|
|
|
|
|
return Ok(());
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
}
|
2026-02-26 16:50:14 -08:00
|
|
|
// Non-ready message: something is wrong
|
|
|
|
|
break;
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err("Sidecar did not send ready message".to_string())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Send a message to the sidecar and read the response.
|
2026-02-26 17:14:25 -08:00
|
|
|
/// This is a blocking call. Progress messages are skipped.
|
2026-02-26 15:53:09 -08:00
|
|
|
pub fn send_and_receive(&self, msg: &IPCMessage) -> Result<IPCMessage, String> {
|
2026-02-26 17:14:25 -08:00
|
|
|
self.send_and_receive_with_progress(msg, |_| {})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Send a message and read the response, calling on_progress for each progress message.
|
|
|
|
|
pub fn send_and_receive_with_progress(
|
|
|
|
|
&self,
|
|
|
|
|
msg: &IPCMessage,
|
|
|
|
|
on_progress: impl Fn(&IPCMessage),
|
|
|
|
|
) -> Result<IPCMessage, String> {
|
2026-02-26 16:50:14 -08:00
|
|
|
// Write to stdin
|
|
|
|
|
{
|
|
|
|
|
let mut stdin_guard = self.stdin.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
if let Some(ref mut stdin) = *stdin_guard {
|
2026-02-26 15:53:09 -08:00
|
|
|
let json = serde_json::to_string(msg).map_err(|e| e.to_string())?;
|
|
|
|
|
stdin
|
|
|
|
|
.write_all(json.as_bytes())
|
|
|
|
|
.map_err(|e| format!("Write error: {e}"))?;
|
|
|
|
|
stdin
|
|
|
|
|
.write_all(b"\n")
|
|
|
|
|
.map_err(|e| format!("Write error: {e}"))?;
|
|
|
|
|
stdin.flush().map_err(|e| format!("Flush error: {e}"))?;
|
|
|
|
|
} else {
|
|
|
|
|
return Err("Sidecar stdin not available".to_string());
|
|
|
|
|
}
|
2026-02-26 16:50:14 -08:00
|
|
|
}
|
2026-02-26 15:53:09 -08:00
|
|
|
|
2026-02-26 16:50:14 -08:00
|
|
|
// Read from stdout
|
|
|
|
|
{
|
|
|
|
|
let mut reader_guard = self.reader.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
if let Some(ref mut reader) = *reader_guard {
|
2026-02-26 15:53:09 -08:00
|
|
|
let mut line = String::new();
|
|
|
|
|
loop {
|
|
|
|
|
line.clear();
|
|
|
|
|
let bytes_read = reader
|
|
|
|
|
.read_line(&mut line)
|
|
|
|
|
.map_err(|e| format!("Read error: {e}"))?;
|
|
|
|
|
if bytes_read == 0 {
|
|
|
|
|
return Err("Sidecar closed stdout".to_string());
|
|
|
|
|
}
|
|
|
|
|
let trimmed = line.trim();
|
|
|
|
|
if trimmed.is_empty() {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2026-02-26 16:50:14 -08:00
|
|
|
let response: IPCMessage = serde_json::from_str(trimmed)
|
|
|
|
|
.map_err(|e| format!("Parse error: {e}"))?;
|
2026-02-26 15:53:09 -08:00
|
|
|
|
2026-02-26 17:14:25 -08:00
|
|
|
if response.msg_type == "progress" {
|
|
|
|
|
on_progress(&response);
|
|
|
|
|
continue;
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
2026-02-26 17:14:25 -08:00
|
|
|
return Ok(response);
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
} else {
|
2026-02-26 16:50:14 -08:00
|
|
|
Err("Sidecar stdout not available".to_string())
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Stop the sidecar process.
|
|
|
|
|
pub fn stop(&self) -> Result<(), String> {
|
2026-02-26 16:50:14 -08:00
|
|
|
// Drop stdin to signal EOF
|
|
|
|
|
{
|
|
|
|
|
let mut stdin_guard = self.stdin.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
*stdin_guard = None;
|
|
|
|
|
}
|
|
|
|
|
// Drop reader
|
|
|
|
|
{
|
|
|
|
|
let mut reader_guard = self.reader.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
*reader_guard = None;
|
|
|
|
|
}
|
|
|
|
|
// Wait for process to exit
|
|
|
|
|
{
|
|
|
|
|
let mut proc = self.process.lock().map_err(|e| e.to_string())?;
|
|
|
|
|
if let Some(ref mut child) = proc.take() {
|
|
|
|
|
match child.wait() {
|
|
|
|
|
Ok(_) => {}
|
|
|
|
|
Err(_) => {
|
|
|
|
|
let _ = child.kill();
|
|
|
|
|
}
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2026-02-26 16:50:14 -08:00
|
|
|
Ok(())
|
2026-02-26 15:53:09 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn is_running(&self) -> bool {
|
|
|
|
|
let proc = self.process.lock().ok();
|
|
|
|
|
proc.map_or(false, |p| p.is_some())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Drop for SidecarManager {
|
|
|
|
|
fn drop(&mut self) {
|
|
|
|
|
let _ = self.stop();
|
|
|
|
|
}
|
|
|
|
|
}
|