Skip to main content

lean_ctx/core/
consolidation_engine.rs

1use chrono::Utc;
2
3use crate::core::knowledge::ProjectKnowledge;
4use crate::core::session::SessionState;
5
6#[derive(Debug, Clone, Copy)]
7pub struct ConsolidationBudgets {
8    pub max_decisions: usize,
9    pub max_findings: usize,
10}
11
12impl Default for ConsolidationBudgets {
13    fn default() -> Self {
14        Self {
15            max_decisions: 5,
16            max_findings: 8,
17        }
18    }
19}
20
21#[derive(Debug, Clone)]
22pub struct ConsolidationOutcome {
23    pub promoted: u32,
24    pub promoted_decisions: u32,
25    pub promoted_findings: u32,
26    pub lifecycle_archived: usize,
27    pub lifecycle_remaining: usize,
28}
29
30pub fn consolidate_latest(
31    project_root: &str,
32    budgets: ConsolidationBudgets,
33) -> Result<ConsolidationOutcome, String> {
34    // Consolidate the session for the explicitly given project root rather than
35    // whatever the process cwd resolves to. This is both correct (the caller
36    // already knows the project) and required after session loads became
37    // strictly project-scoped (#2362): load_latest() is cwd-bound and would miss
38    // a session whose root differs from cwd.
39    let session = SessionState::load_latest_for_project_root(project_root)
40        .ok_or_else(|| "no active session".to_string())?;
41    let policy = crate::core::config::Config::load()
42        .memory_policy_effective()
43        .map_err(|e| format!("invalid memory policy: {e}"))?;
44
45    let mut knowledge = ProjectKnowledge::load_or_create(project_root);
46
47    let mut promoted_decisions = 0u32;
48    let mut promoted_findings = 0u32;
49
50    let mut decisions = session.decisions.clone();
51    decisions.sort_by_key(|x| std::cmp::Reverse(x.timestamp));
52    decisions.truncate(budgets.max_decisions);
53    for d in &decisions {
54        let key = slug_key(&d.summary, 50);
55        knowledge.remember("decision", &key, &d.summary, &session.id, 0.9, &policy);
56        promoted_decisions += 1;
57    }
58
59    let mut findings = session.findings.clone();
60    findings.sort_by_key(|x| std::cmp::Reverse(x.timestamp));
61    let mut kept = Vec::new();
62    for f in &findings {
63        if kept.len() >= budgets.max_findings {
64            break;
65        }
66        if finding_salience(&f.summary) < 45 {
67            continue;
68        }
69        kept.push(f.clone());
70    }
71
72    for f in &kept {
73        let key = if let Some(ref file) = f.file {
74            if let Some(line) = f.line {
75                format!("{file}:{line}")
76            } else {
77                file.clone()
78            }
79        } else {
80            format!("finding-{}", slug_key(&f.summary, 36))
81        };
82        knowledge.remember("finding", &key, &f.summary, &session.id, 0.75, &policy);
83        promoted_findings += 1;
84    }
85
86    // One compact history entry (no prose output to user; stored for auditability).
87    let task_desc = session
88        .task
89        .as_ref()
90        .map_or_else(|| "(no task)".into(), |t| t.description.clone());
91    let summary = format!(
92        "consolidate@{} session={} task=\"{}\" decisions={} findings={}",
93        Utc::now().format("%Y-%m-%d"),
94        session.id,
95        task_desc,
96        promoted_decisions,
97        promoted_findings
98    );
99    knowledge.consolidate(&summary, vec![session.id.clone()], &policy);
100
101    let lifecycle = knowledge.run_memory_lifecycle(&policy);
102    knowledge.save()?;
103
104    crate::core::events::emit(crate::core::events::EventKind::KnowledgeUpdate {
105        category: "memory".to_string(),
106        key: "consolidation".to_string(),
107        action: "run".to_string(),
108    });
109
110    Ok(ConsolidationOutcome {
111        promoted: promoted_decisions + promoted_findings,
112        promoted_decisions,
113        promoted_findings,
114        lifecycle_archived: lifecycle.archived_count,
115        lifecycle_remaining: lifecycle.remaining_facts,
116    })
117}
118
119fn slug_key(s: &str, max: usize) -> String {
120    let mut out = String::new();
121    for ch in s.chars() {
122        if out.len() >= max {
123            break;
124        }
125        if ch.is_ascii_alphanumeric() {
126            out.push(ch.to_ascii_lowercase());
127        } else if (ch.is_whitespace() || ch == '-' || ch == '_')
128            && !out.ends_with('-')
129            && !out.is_empty()
130        {
131            out.push('-');
132        }
133    }
134    out.trim_matches('-').to_string()
135}
136
137fn finding_salience(summary: &str) -> u32 {
138    let s = summary.to_lowercase();
139    let mut score = 20u32;
140
141    let boosts = [
142        ("error", 25),
143        ("failed", 25),
144        ("panic", 30),
145        ("assert", 20),
146        ("forbidden", 25),
147        ("timeout", 20),
148        ("deadlock", 25),
149        ("security", 25),
150        ("vuln", 25),
151        ("e0", 15), // rust error codes often start with E0xxx
152    ];
153
154    for (pat, b) in boosts {
155        if s.contains(pat) {
156            score = score.saturating_add(b);
157        }
158    }
159
160    score
161}
162
163#[cfg(test)]
164mod tests {
165    use super::*;
166
167    #[test]
168    fn consolidate_promotes_decisions_and_salient_findings_only() {
169        let _lock = crate::core::data_dir::test_env_lock();
170        let tmp = tempfile::tempdir().expect("tempdir");
171        std::env::set_var(
172            "LEAN_CTX_DATA_DIR",
173            tmp.path().to_string_lossy().to_string(),
174        );
175
176        let project_root = tmp.path().join("proj");
177        std::fs::create_dir_all(&project_root).expect("mkdir");
178        let project_root_str = project_root.to_string_lossy().to_string();
179
180        let mut session = SessionState::new();
181        session.project_root = Some(project_root_str.clone());
182        session.add_decision("Use archive-only memory lifecycle", None);
183        session.add_finding(None, None, "panic: index out of bounds");
184        session.add_finding(None, None, "just a note");
185        session.save().expect("save session");
186
187        let out = consolidate_latest(
188            &project_root_str,
189            ConsolidationBudgets {
190                max_decisions: 5,
191                max_findings: 5,
192            },
193        )
194        .expect("consolidate");
195        assert!(out.promoted_decisions >= 1);
196        assert!(out.promoted_findings >= 1);
197
198        let k = ProjectKnowledge::load(&project_root_str).expect("knowledge saved");
199        let active = k.facts.iter().filter(|f| f.is_current()).count();
200        assert!(active >= 2, "expected promoted facts");
201
202        std::env::remove_var("LEAN_CTX_DATA_DIR");
203    }
204}