lean_ctx/core/
consolidation_engine.rs1use 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 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 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), ];
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}