kaizen-cli 0.1.35

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
use kaizen::experiment::types::{State as ExpLifecycle, transition};
use quint_connect::*;
use serde::Deserialize;

#[derive(Debug, Eq, PartialEq, Deserialize)]
#[serde(tag = "tag")]
enum SpecState {
    Draft,
    Running,
    Concluded,
    Archived,
}

#[derive(Debug, Eq, PartialEq, Deserialize)]
struct ExpState {
    state: SpecState,
    #[serde(with = "itf::de::As::<itf::de::Integer>")]
    step_count: i64,
}

impl State<ExpDriver> for ExpState {
    fn from_driver(d: &ExpDriver) -> Result<Self> {
        Ok(ExpState {
            state: match d.state {
                ExpLifecycle::Draft => SpecState::Draft,
                ExpLifecycle::Running => SpecState::Running,
                ExpLifecycle::Concluded => SpecState::Concluded,
                ExpLifecycle::Archived => SpecState::Archived,
            },
            step_count: d.step_count,
        })
    }
}

#[derive(Debug)]
struct ExpDriver {
    state: ExpLifecycle,
    step_count: i64,
}

impl Default for ExpDriver {
    fn default() -> Self {
        Self {
            state: ExpLifecycle::Draft,
            step_count: 0,
        }
    }
}

impl Driver for ExpDriver {
    type State = ExpState;

    fn step(&mut self, step: &Step) -> Result {
        switch!(step {
            init => {
                self.state = ExpLifecycle::Draft;
                self.step_count = 0;
            },
            // See `specs/session-lifecycle` driver: outer `any{}` can emit a `step` before the
            // concrete action; keep the same baseline as that pattern.
            step => {
                self.state = ExpLifecycle::Draft;
                self.step_count = 0;
            },
            start => {
                self.state = transition(self.state, "start")
                    .ok_or_else(|| anyhow::anyhow!("start not enabled"))?;
                self.step_count += 1;
            },
            conclude => {
                self.state = transition(self.state, "conclude")
                    .ok_or_else(|| anyhow::anyhow!("conclude not enabled"))?;
                self.step_count += 1;
            },
            archive => {
                self.state = transition(self.state, "archive")
                    .ok_or_else(|| anyhow::anyhow!("archive not enabled"))?;
                self.step_count += 1;
            }
        })
    }
}

#[quint_run(
    spec = "specs/experiment-lifecycle.qnt",
    max_samples = 20,
    max_steps = 6,
    seed = "0x1"
)]
fn experiment_lifecycle_run() -> impl Driver {
    ExpDriver::default()
}