46 crates, 744 tests green (up from 726 at v0.31.0). ## kei-dna-index (new) — read-only adjacency analysis over kei-ledger Answers "who else touched same files / solved same task / ran nearby in time". Does NOT mutate ledger — parses DNA strings in memory. Respects SSoT (DNA string is the single source; columns NOT duplicated). Public API: - adjacent(target_dna, kind) — 5 kinds: Scope / Body / Role / Temporal / All - cluster_by(scope|body|role) — group DNAs, ≥2 members per cluster - precedent(body_sha, status_filter) — find past successful runs of same task - stats — totals, unique scopes/bodies, avg cluster size CLI: - kei-dna-index adjacent --dna D [--by kind] [--limit N] [--db PATH] - kei-dna-index cluster --by scope|body|role - kei-dna-index precedent --body HEX [--status merged|failed|all] - kei-dna-index stats 18 tests pass (13 integration + 5 parsed unit). Zero sibling deps (no kei-ledger, no kei-agent-runtime path imports — standalone tool). Separation of concerns: kei-ledger stays PURE provenance primitive. Analytical layer lives in kei-dna-index. Can swap implementations (naive scan → cached → embeddings) without touching ledger schema. ## kei-fork v0.31.2 — Option D path convention Moved fork worktree root from `.claude/forks/<id>/` to `_forks/<id>/`. Reasons: - `.claude/` is Anthropic-reserved; kit artefacts shouldn't pollute it - Claude Code sandbox denies Write in `.claude/forks/` for agents - `_forks/` matches existing kit convention (_primitives/, _roles/, _archive/, _blocks/, _capabilities/, _agents/) - Independent namespace — no coupling to Claude Code internals 13 existing kei-fork tests still pass (they use tempfile kit_roots so path convention is transparent). ## Usage enabled by these two - kei-prune can now query "all DNAs in same scope-cluster" → retire dupes - kei-brain-view can cluster-render instead of tree-render - Three-role pipeline (writer/auditor/merger) can use precedent() to find successful past patterns for same body-hash - Agents with worktree isolation can write to _forks/ without sandbox permission issues Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
54 lines
1.6 KiB
Rust
54 lines
1.6 KiB
Rust
//! `rescue(agent_id, kit_root, out_dir)` — copy a fork's files out of
|
|
//! band.
|
|
//!
|
|
//! Resolution order:
|
|
//! 1. `_forks/<id>/` (live) → copy to `out_dir`
|
|
//! 2. `_archive/forks/<date>/<id>/` (archived) → copy to `out_dir`
|
|
//! 3. Neither → `Error::Gone`
|
|
//!
|
|
//! Copy is recursive; the destination may pre-exist (we merge on top).
|
|
//! Returns the number of regular files copied.
|
|
|
|
use crate::error::Error;
|
|
use std::fs;
|
|
use std::path::{Path, PathBuf};
|
|
|
|
pub fn rescue(agent_id: &str, kit_root: &Path, out_dir: &Path) -> Result<usize, Error> {
|
|
let src = locate(agent_id, kit_root).ok_or_else(|| Error::Gone(agent_id.to_string()))?;
|
|
fs::create_dir_all(out_dir)?;
|
|
Ok(copy_tree(&src, out_dir)?)
|
|
}
|
|
|
|
fn locate(agent_id: &str, kit_root: &Path) -> Option<PathBuf> {
|
|
let live = kit_root.join("_forks").join(agent_id);
|
|
if live.is_dir() {
|
|
return Some(live);
|
|
}
|
|
let archive_root = kit_root.join("_archive/forks");
|
|
let dates = fs::read_dir(&archive_root).ok()?;
|
|
for e in dates.flatten() {
|
|
let candidate = e.path().join(agent_id);
|
|
if candidate.is_dir() {
|
|
return Some(candidate);
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
fn copy_tree(src: &Path, dst: &Path) -> std::io::Result<usize> {
|
|
let mut n = 0;
|
|
for entry in fs::read_dir(src)? {
|
|
let entry = entry?;
|
|
let name = entry.file_name();
|
|
let from = entry.path();
|
|
let to = dst.join(&name);
|
|
if from.is_dir() {
|
|
fs::create_dir_all(&to)?;
|
|
n += copy_tree(&from, &to)?;
|
|
} else if from.is_file() {
|
|
fs::copy(&from, &to)?;
|
|
n += 1;
|
|
}
|
|
}
|
|
Ok(n)
|
|
}
|