//! Integration tests for kei-ledger. //! //! Constructor Pattern: each test = one scenario, one assertion target. //! Uses tempfile for per-test isolated sqlite file. Loads source modules //! via `#[path]` so we don't need to expose a library crate surface. #[path = "../src/schema.rs"] mod schema; #[path = "../src/error.rs"] mod error; #[path = "../src/ledger.rs"] mod ledger; use rusqlite::Connection; use std::fs; use std::path::{Path, PathBuf}; use tempfile::TempDir; fn open_tmp() -> (TempDir, Connection) { let dir = tempfile::tempdir().unwrap(); let db = dir.path().join("ledger.sqlite"); let conn = ledger::open(&db).unwrap(); (dir, conn) } fn write_artefacts(root: &Path, agent_id: &str, which: &[&str]) -> PathBuf { let base = root.join(".claude/agents").join(agent_id); fs::create_dir_all(&base).unwrap(); for f in which { fs::write(base.join(f), b"x").unwrap(); } base } #[test] fn fork_then_done_marks_terminal() { let (_d, conn) = open_tmp(); ledger::fork(&conn, "a1", "agent/a1", None, "deadbeef", None, None).unwrap(); let running = ledger::list(&conn, Some("running")).unwrap(); assert_eq!(running.len(), 1); assert_eq!(running[0].id, "a1"); let updated = ledger::done(&conn, "a1", "shipped").unwrap(); assert_eq!(updated, 1); let done = ledger::list(&conn, Some("done")).unwrap(); assert_eq!(done.len(), 1); assert_eq!(done[0].summary.as_deref(), Some("shipped")); } #[test] fn fail_flow_sets_reason_and_finished_ts() { let (_d, conn) = open_tmp(); ledger::fork(&conn, "b1", "agent/b1", Some("main"), "cafebabe", None, None).unwrap(); let updated = ledger::fail(&conn, "b1", "cargo build failed").unwrap(); assert_eq!(updated, 1); let failed = ledger::list(&conn, Some("failed")).unwrap(); assert_eq!(failed.len(), 1); assert!(failed[0].finished_ts.is_some()); assert_eq!(failed[0].summary.as_deref(), Some("cargo build failed")); } #[test] fn tree_walks_parent_child_chain() { let (_d, conn) = open_tmp(); ledger::fork(&conn, "root", "agent/root", Some("main"), "aa", None, None).unwrap(); ledger::fork(&conn, "c1", "agent/c1", Some("agent/root"), "bb", None, None).unwrap(); ledger::fork(&conn, "c2", "agent/c2", Some("agent/root"), "cc", None, None).unwrap(); ledger::fork(&conn, "g1", "agent/g1", Some("agent/c1"), "dd", None, None).unwrap(); let t = ledger::tree(&conn, "root").unwrap(); let ids: Vec<_> = t.iter().map(|a| a.id.as_str()).collect(); assert!(ids.contains(&"root")); assert!(ids.contains(&"c1")); assert!(ids.contains(&"c2")); assert!(ids.contains(&"g1")); assert_eq!(ids[0], "root"); assert_eq!(ids.len(), 4); } #[test] fn list_filter_status_excludes_others() { let (_d, conn) = open_tmp(); ledger::fork(&conn, "r1", "br-r1", None, "s1", None, None).unwrap(); ledger::fork(&conn, "r2", "br-r2", None, "s2", None, None).unwrap(); ledger::done(&conn, "r1", "ok").unwrap(); let running = ledger::list(&conn, Some("running")).unwrap(); assert_eq!(running.len(), 1); assert_eq!(running[0].id, "r2"); let all = ledger::list(&conn, None).unwrap(); assert_eq!(all.len(), 2); } #[test] fn validate_detects_missing_artefacts() { let (d, _conn) = open_tmp(); write_artefacts(d.path(), "v1", &["spec.md", "plan.md"]); let missing = ledger::validate(d.path(), "v1"); assert_eq!(missing.len(), 4); assert!(missing.contains(&"progress.json".to_string())); assert!(missing.contains(&"review.md".to_string())); } #[test] fn validate_ok_when_all_six_present() { let (d, _conn) = open_tmp(); write_artefacts( d.path(), "v2", &[ "spec.md", "plan.md", "progress.json", "chatlog.md", "handoffs.md", "review.md", ], ); let missing = ledger::validate(d.path(), "v2"); assert!(missing.is_empty(), "got missing {missing:?}"); } #[test] fn duplicate_fork_id_rejected() { let (_d, conn) = open_tmp(); ledger::fork(&conn, "dup", "br1", None, "x", None, None).unwrap(); let err = ledger::fork(&conn, "dup", "br2", None, "y", None, None); assert!(err.is_err(), "duplicate id must fail"); } #[test] fn done_on_already_done_agent_is_noop() { let (_d, conn) = open_tmp(); ledger::fork(&conn, "n1", "br-n1", None, "h", None, None).unwrap(); assert_eq!(ledger::done(&conn, "n1", "first").unwrap(), 1); assert_eq!(ledger::done(&conn, "n1", "second").unwrap(), 0); let row = &ledger::list(&conn, None).unwrap()[0]; assert_eq!(row.summary.as_deref(), Some("first")); } #[test] fn fork_with_dna_roundtrips_through_list() { let (_d, conn) = open_tmp(); let dna = "edit-local::NG-FW-FD-CP-CG-TG-ND-RF::A7B2::C9F1-xa7c"; ledger::fork(&conn, "dna1", "agent/dna1", None, "spec", None, Some(dna)).unwrap(); let rows = ledger::list(&conn, None).unwrap(); assert_eq!(rows.len(), 1); assert_eq!(rows[0].dna.as_deref(), Some(dna)); ledger::fork(&conn, "legacy1", "agent/legacy1", None, "spec2", None, None).unwrap(); let rows = ledger::list(&conn, None).unwrap(); let legacy = rows.iter().find(|r| r.id == "legacy1").unwrap(); assert!(legacy.dna.is_none(), "legacy fork should leave dna NULL"); } #[test] fn merged_after_done_transitions_status() { let (_d, conn) = open_tmp(); ledger::fork(&conn, "m1", "br-m1", None, "h", None, None).unwrap(); ledger::done(&conn, "m1", "ready").unwrap(); assert_eq!(ledger::merged(&conn, "m1").unwrap(), 1); let merged = ledger::list(&conn, Some("merged")).unwrap(); assert_eq!(merged.len(), 1); assert_eq!(merged[0].summary.as_deref(), Some("ready")); } // --- audit fixes (2026-04-23) ------------------------------------------ /// Fix S2 — cycle in parent_branch must not hang `tree()`. Synthetic cycle /// br-x→br-y→br-x is injected by disabling the check trigger temporarily /// via raw INSERT (bypassing `ledger::fork`'s length guard is not needed; /// the cycle itself is the payload). The walk must terminate with either /// `MaxDepthExceeded` OR cleanly (visited-set short-circuit), never loop. #[test] fn tree_handles_cycle_without_infinite_loop() { let (_d, conn) = open_tmp(); // Two rows whose parent_branch point at each other. ledger::fork(&conn, "cx", "br-x", Some("br-y"), "sha-x", None, None).unwrap(); ledger::fork(&conn, "cy", "br-y", Some("br-x"), "sha-y", None, None).unwrap(); // tree() should either return bounded rows (visited-set kills the loop) // or MaxDepthExceeded. Must not hang / OOM. let out = ledger::tree(&conn, "cx"); match out { Ok(rows) => { // visited-set: cx root, plus cy as child of br-y's... actually cx's // branch is br-x, children of br-x = cy; cy's branch is br-y, // already visited (root chained via frontier pop). <= 2 rows max. assert!(rows.len() <= 2, "got unbounded rows {}", rows.len()); } Err(ledger::LedgerError::MaxDepthExceeded) => { // Acceptable: circuit breaker fired. } Err(e) => panic!("unexpected error: {e}"), } } /// Fix M2 — migration is idempotent: calling `open` twice on the same file /// does not explode with "duplicate column" or leave user_version stale. /// This implicitly exercises the transaction wrapper (v1, v2, v3 must all /// commit cleanly across two opens). #[test] fn migrate_is_idempotent_across_reopens() { let dir = tempfile::tempdir().unwrap(); let db = dir.path().join("ledger.sqlite"); { let conn = ledger::open(&db).unwrap(); ledger::fork(&conn, "pre", "br-pre", None, "h", None, None).unwrap(); } // Second open re-enters migrate(); must be a no-op, not a duplicate // column / trigger error. let conn = ledger::open(&db).unwrap(); let version: i64 = conn .query_row("PRAGMA user_version", [], |r| r.get(0)) .unwrap(); assert_eq!(version, schema::MIGRATIONS.len() as i64); // Row from first session must survive. let rows = ledger::list(&conn, None).unwrap(); assert_eq!(rows.len(), 1); assert_eq!(rows[0].id, "pre"); } /// Fix L1 — branch longer than MAX_BRANCH_LEN must be rejected at the /// library boundary with `LedgerError::BranchTooLong` (clap `value_parser` /// provides the same guard at the CLI boundary). #[test] fn fork_rejects_overlong_branch() { let (_d, conn) = open_tmp(); let long = "x".repeat(schema::MAX_BRANCH_LEN + 1); let res = ledger::fork(&conn, "too-long", &long, None, "h", None, None); match res { Err(ledger::LedgerError::BranchTooLong { field, len }) => { assert_eq!(field, "branch"); assert_eq!(len, schema::MAX_BRANCH_LEN + 1); } other => panic!("expected BranchTooLong, got {other:?}"), } // Parent side same cap. let res2 = ledger::fork(&conn, "ok-br", "fine", Some(&long), "h", None, None); assert!( matches!( res2, Err(ledger::LedgerError::BranchTooLong { field: "parent_branch", .. }) ), "expected parent_branch rejection" ); // Length at the cap is accepted. let at_cap = "y".repeat(schema::MAX_BRANCH_LEN); ledger::fork(&conn, "at-cap", &at_cap, None, "h", None, None).unwrap(); }