kaizen-cli 0.1.43

Distributable agent observability: real-time-tailable sessions, agile-style retros, and repo-level improvement (Cursor, Claude Code, Codex). SQLite, redact before any sync you enable.
Documentation
// SPDX-License-Identifier: AGPL-3.0-or-later
//! Background daemon process startup.

use super::lifecycle::{RuntimePaths, runtime_paths, runtime_paths_for, try_status};
use crate::ipc::{DaemonStatus, WebEndpoint};
use anyhow::{Context, Result, anyhow};
use std::path::Path;
use std::process::{Child, Command, Stdio};
use std::time::{Duration, Instant};

const START_WAIT_MS: u64 = 2_000;

#[derive(Debug, Clone)]
pub struct BackgroundStart {
    pub pid: u32,
    pub paths: RuntimePaths,
    pub already_running: bool,
    pub web: Option<WebEndpoint>,
}

pub fn start_background() -> Result<BackgroundStart> {
    start_background_at(runtime_paths()?)
}

pub fn start_background_for(workspace: &Path) -> Result<BackgroundStart> {
    start_background_at(runtime_paths_for(workspace)?)
}

fn start_background_at(paths: RuntimePaths) -> Result<BackgroundStart> {
    if let Some(start) = running_start(&paths) {
        return Ok(start);
    }
    std::fs::create_dir_all(&paths.dir)?;
    let mut child = spawn_background(&paths)?;
    wait_until_ready(paths, &mut child)
}

fn running_start(paths: &RuntimePaths) -> Option<BackgroundStart> {
    try_status()
        .ok()
        .map(|status| background_start(status, paths.clone(), true))
}

fn spawn_background(paths: &RuntimePaths) -> Result<Child> {
    let log = open_log(&paths.log)?;
    let err = log.try_clone()?;
    background_command(log, err)?
        .spawn()
        .context("spawn kaizen daemon")
}

fn open_log(path: &Path) -> Result<std::fs::File> {
    crate::core::safe_fs::append(path)
        .with_context(|| format!("open daemon log: {}", path.display()))
}

fn wait_until_ready(paths: RuntimePaths, child: &mut Child) -> Result<BackgroundStart> {
    let deadline = Instant::now() + Duration::from_millis(START_WAIT_MS);
    while Instant::now() < deadline {
        if let Some(start) = poll_start(child, &paths)? {
            return Ok(start);
        }
        std::thread::sleep(Duration::from_millis(25));
    }
    Err(start_timeout(&paths))
}

fn poll_start(child: &mut Child, paths: &RuntimePaths) -> Result<Option<BackgroundStart>> {
    if let Some(status) = child.try_wait().context("poll daemon child")? {
        return Err(early_exit(status, paths));
    }
    Ok(try_status()
        .ok()
        .map(|status| background_start(status, paths.clone(), false)))
}

fn background_start(
    status: DaemonStatus,
    paths: RuntimePaths,
    already_running: bool,
) -> BackgroundStart {
    BackgroundStart {
        pid: status.pid,
        paths,
        already_running,
        web: status.web,
    }
}

fn early_exit(status: std::process::ExitStatus, paths: &RuntimePaths) -> anyhow::Error {
    anyhow!(
        "daemon exited before ready with status {status}; see {}",
        paths.log.display()
    )
}

fn start_timeout(paths: &RuntimePaths) -> anyhow::Error {
    anyhow!(
        "daemon did not become ready at {}; see {}",
        paths.sock.display(),
        paths.log.display()
    )
}

fn background_command(log: std::fs::File, err: std::fs::File) -> Result<Command> {
    let mut command = Command::new(std::env::current_exe()?);
    command
        .args(["daemon", "start"])
        .stdin(Stdio::null())
        .stdout(Stdio::from(log))
        .stderr(Stdio::from(err));
    detach_background(&mut command);
    Ok(command)
}

#[cfg(unix)]
fn detach_background(command: &mut Command) {
    use std::os::unix::process::CommandExt;
    unsafe {
        command.pre_exec(|| {
            (libc::setsid() != -1)
                .then_some(())
                .ok_or_else(std::io::Error::last_os_error)
        });
    }
}

#[cfg(not(unix))]
fn detach_background(_command: &mut Command) {}