48 crates, 859 tests green (+58 kei-pet tests, was 801 at v0.35.0).
Full substrate pipeline test: all 8 agents launched via kei-agent-runtime
prepare → composed capability-fragment prompts → Agent tool invocations.
Zero file conflicts across disjoint scopes. Every agent self-verified
and landed files direct to main.
## A. memory (4 tests) — persistent conversations
- src/memory.rs — (user_id, pet_name)-scoped conversation log
- SQLite via rusqlite, index (user_id, pet_name, ts DESC)
- record_interaction / recent / search with LIKE-escape
## B. evolution (3 tests) — version diff + fork chain
- src/evolution.rs — PersonaVersion { version, parent_version, manifest }
- diff(old, new) → Vec<Change> (tone / directness / initiative / forbidden / humor)
- fork_version increments + links parent
## C. wizard (5 markdown phases) — /pet-init skill
- skills/pet-init/SKILL.md + 4 phases (identity / voice / edge / emit)
- AskUserQuestion-driven, no TOML editing for end users
- Writes ~/.claude/pet/<user_id>.toml + calls kei-pet keygen if needed
## D. templates (3 tests + 5 presets) — role-based personas
- templates/{friend,tutor,coach,therapist-companion,productivity-partner}.toml
- src/templates.rs — PetTemplate enum + load_template + list_templates
- Schema-enum mapping documented (dry→engineering-meta, etc) — schema.rs
expansion is future work
## E. bridge (3 tests) — /spawn-agent pet overlay
- src/bridge.rs — compose_prompt_with_pet(base + persona overlay + task)
- skills/spawn-agent/phase-3-pet-overlay.md — interactive pet selector
## F. recall (4 tests) — "have we discussed this before?"
- src/recall.rs — wraps kei_dna_index::precedent with body_sha8()
- SHA-256 first 4 bytes → 8 hex lowercase (matches kei_shared width)
- Fetches started_ts per hit for honest sort-by-recency
## G. reflect (7 tests) — self-reflection threshold proposals
- src/reflect.rs — CorrectionSignal + ProposedChange
- Thresholds: 3× too_verbose → SetDirectness, 2× forbidden_topic → AddForbidden, etc
- Idempotent: no-op if manifest already in desired state
## H. fleet (6 tests) — multi-pet per user
- src/fleet.rs — PetFleet { user_id, pets, active_pet }
- add_pet / switch_active / load_fleet with toml persistence
- shared_memory_key vs per_pet_memory_key — one user scopes multiple pets
## Known follow-ups (not blockers)
- Phase-4-emit of /spawn-agent should read PET_MANIFEST_PATH from new
phase-3-pet-overlay and pass to kei-spawn (wiring next wave)
- SKILL.md for spawn-agent should list new pet-overlay phase
- Schema enum expansion: humor_style "dry/witty", directness "direct/
gentle/blunt", initiative "proactive/nudge" as first-class variants
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
61 lines
1.7 KiB
Rust
61 lines
1.7 KiB
Rust
//! Integration tests for `kei_pet::evolution` — diff detection + fork
|
|
//! chain linking. Uses `examples/minimal.toml` as the baseline and mutates
|
|
//! clones to exercise each `Change` variant.
|
|
|
|
use kei_pet::evolution::{diff, fork_version, Change, PersonaVersion};
|
|
use kei_pet::parse;
|
|
use kei_pet::schema::Tone;
|
|
|
|
const MINIMAL: &str = include_str!("../examples/minimal.toml");
|
|
|
|
fn base() -> kei_pet::PetManifest {
|
|
parse(MINIMAL).expect("minimal.toml must validate")
|
|
}
|
|
|
|
#[test]
|
|
fn diff_detects_tone_change() {
|
|
let old = base();
|
|
let mut new = old.clone();
|
|
new.voice.tone_primary = Tone::Warm;
|
|
|
|
let changes = diff(&old, &new);
|
|
assert_eq!(changes.len(), 1, "expected exactly one change, got {changes:?}");
|
|
assert_eq!(
|
|
changes[0],
|
|
Change::VoiceTonePrimaryChanged {
|
|
from: "neutral".to_string(),
|
|
to: "warm".to_string(),
|
|
}
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn diff_detects_forbidden_added() {
|
|
let old = base();
|
|
let mut new = old.clone();
|
|
new.forbidden.topics.push("diagnosis".to_string());
|
|
|
|
let changes = diff(&old, &new);
|
|
assert_eq!(changes.len(), 1, "expected exactly one change, got {changes:?}");
|
|
assert_eq!(
|
|
changes[0],
|
|
Change::ForbiddenTopicAdded("diagnosis".to_string())
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn fork_version_increments_and_links() {
|
|
let manifest = base();
|
|
let v1 = PersonaVersion {
|
|
version: 1,
|
|
parent_version: None,
|
|
manifest: manifest.clone(),
|
|
created_at: 1_700_000_000,
|
|
reason: "initial".to_string(),
|
|
};
|
|
|
|
let v2 = fork_version(&v1, "tune tone", manifest);
|
|
assert_eq!(v2.version, 2);
|
|
assert_eq!(v2.parent_version, Some(1));
|
|
assert_eq!(v2.reason, "tune tone");
|
|
}
|