Skip to main content

lean_ctx/core/stats/
mod.rs

1mod format;
2mod io;
3mod model;
4
5pub use format::*;
6pub use model::*;
7
8use std::collections::HashMap;
9use std::sync::Mutex;
10use std::time::Instant;
11
12/// (current_state, baseline_from_disk, last_flush_time)
13static STATS_BUFFER: Mutex<Option<(StatsStore, StatsStore, Instant)>> = Mutex::new(None);
14
15const FLUSH_INTERVAL_SECS: u64 = 30;
16
17pub fn load() -> StatsStore {
18    let guard = STATS_BUFFER
19        .lock()
20        .unwrap_or_else(std::sync::PoisonError::into_inner);
21    if let Some((ref current, ref baseline, _)) = *guard {
22        let disk = io::load_from_disk();
23        return io::apply_deltas(&disk, current, baseline);
24    }
25    drop(guard);
26    io::load_from_disk()
27}
28
29pub fn save(store: &StatsStore) {
30    io::locked_write(store);
31}
32
33fn maybe_flush(store: &mut StatsStore, baseline: &mut StatsStore, last_flush: &mut Instant) {
34    if last_flush.elapsed().as_secs() >= FLUSH_INTERVAL_SECS {
35        let merged = io::merge_and_save(store, baseline);
36        *store = merged.clone();
37        *baseline = merged;
38        *last_flush = Instant::now();
39    }
40}
41
42pub fn flush() {
43    let mut guard = STATS_BUFFER
44        .lock()
45        .unwrap_or_else(std::sync::PoisonError::into_inner);
46    if let Some((ref mut store, ref mut baseline, ref mut last_flush)) = *guard {
47        let merged = io::merge_and_save(store, baseline);
48        *store = merged.clone();
49        *baseline = merged;
50        *last_flush = Instant::now();
51    }
52}
53
54/// Adjust saved tokens after post-processing (terse, hints) changed the output size.
55/// Positive delta = savings were over-reported, negative = under-reported.
56pub fn adjust_savings(command: &str, over_report_delta: i64) {
57    let mut guard = STATS_BUFFER
58        .lock()
59        .unwrap_or_else(std::sync::PoisonError::into_inner);
60    let Some((store, _, _)) = guard.as_mut() else {
61        return;
62    };
63    if over_report_delta > 0 {
64        let adj = over_report_delta as u64;
65        store.total_output_tokens = store.total_output_tokens.saturating_add(adj);
66        if let Some(cmd) = store.commands.get_mut(command) {
67            cmd.output_tokens = cmd.output_tokens.saturating_add(adj);
68        }
69    } else {
70        let adj = over_report_delta.unsigned_abs();
71        store.total_output_tokens = store.total_output_tokens.saturating_sub(adj);
72        if let Some(cmd) = store.commands.get_mut(command) {
73            cmd.output_tokens = cmd.output_tokens.saturating_sub(adj);
74        }
75    }
76}
77
78pub fn record(command: &str, input_tokens: usize, output_tokens: usize) {
79    let mut guard = STATS_BUFFER
80        .lock()
81        .unwrap_or_else(std::sync::PoisonError::into_inner);
82    if guard.is_none() {
83        let disk = io::load_from_disk();
84        *guard = Some((disk.clone(), disk, Instant::now()));
85    }
86    let Some((store, baseline, last_flush)) = guard.as_mut() else {
87        return;
88    };
89
90    let is_first_command = store.total_commands == baseline.total_commands;
91    let now = chrono::Local::now();
92    let today = now.format("%Y-%m-%d").to_string();
93    let timestamp = now.to_rfc3339();
94
95    store.total_commands = store.total_commands.saturating_add(1);
96    store.total_input_tokens = store.total_input_tokens.saturating_add(input_tokens as u64);
97    store.total_output_tokens = store
98        .total_output_tokens
99        .saturating_add(output_tokens as u64);
100
101    if store.first_use.is_none() {
102        store.first_use = Some(timestamp.clone());
103    }
104    store.last_use = Some(timestamp);
105
106    let cmd_key = format::normalize_command(command);
107    let entry = store.commands.entry(cmd_key).or_default();
108    entry.count = entry.count.saturating_add(1);
109    entry.input_tokens = entry.input_tokens.saturating_add(input_tokens as u64);
110    entry.output_tokens = entry.output_tokens.saturating_add(output_tokens as u64);
111
112    let current_version = env!("CARGO_PKG_VERSION").to_string();
113    if let Some(day) = store.daily.last_mut() {
114        if day.date == today {
115            day.commands = day.commands.saturating_add(1);
116            day.input_tokens = day.input_tokens.saturating_add(input_tokens as u64);
117            day.output_tokens = day.output_tokens.saturating_add(output_tokens as u64);
118            // Stamp the running version so a mid-day update attributes the day
119            // to the release in use for its latest activity (#307).
120            day.version = current_version;
121        } else {
122            store.daily.push(DayStats {
123                date: today,
124                commands: 1,
125                input_tokens: input_tokens as u64,
126                output_tokens: output_tokens as u64,
127                version: current_version,
128            });
129        }
130    } else {
131        store.daily.push(DayStats {
132            date: today,
133            commands: 1,
134            input_tokens: input_tokens as u64,
135            output_tokens: output_tokens as u64,
136            version: current_version,
137        });
138    }
139
140    if store.daily.len() > 90 {
141        store.daily.drain(..store.daily.len() - 90);
142    }
143
144    if is_first_command {
145        let merged = io::merge_and_save(store, baseline);
146        *store = merged.clone();
147        *baseline = merged;
148        *last_flush = Instant::now();
149    } else {
150        maybe_flush(store, baseline, last_flush);
151    }
152}
153
154pub fn reset_cep() {
155    let mut guard = STATS_BUFFER
156        .lock()
157        .unwrap_or_else(std::sync::PoisonError::into_inner);
158    let mut store = io::load_from_disk();
159    store.cep = CepStats::default();
160    io::locked_write(&store);
161    *guard = Some((store.clone(), store, Instant::now()));
162}
163
164pub fn reset_all() {
165    let mut guard = STATS_BUFFER
166        .lock()
167        .unwrap_or_else(std::sync::PoisonError::into_inner);
168    let store = StatsStore::default();
169    io::locked_write(&store);
170    *guard = Some((store.clone(), store, Instant::now()));
171    crate::core::heatmap::reset();
172}
173
174pub fn load_stats() -> GainSummary {
175    let store = load();
176    let input_saved = store
177        .total_input_tokens
178        .saturating_sub(store.total_output_tokens);
179    GainSummary {
180        total_saved: input_saved,
181        total_calls: store.total_commands,
182    }
183}
184
185#[allow(clippy::too_many_arguments)]
186pub fn record_cep_session(
187    score: u32,
188    cache_hits: u64,
189    cache_reads: u64,
190    tokens_original: u64,
191    tokens_compressed: u64,
192    modes: &HashMap<String, u64>,
193    tool_calls: u64,
194    complexity: &str,
195) {
196    let mut guard = STATS_BUFFER
197        .lock()
198        .unwrap_or_else(std::sync::PoisonError::into_inner);
199    if guard.is_none() {
200        let disk = io::load_from_disk();
201        *guard = Some((disk.clone(), disk, Instant::now()));
202    }
203    let Some((store, baseline, last_flush)) = guard.as_mut() else {
204        return;
205    };
206
207    let cep = &mut store.cep;
208
209    let pid = std::process::id();
210    let prev_original = cep.last_session_original.unwrap_or(0);
211    let prev_compressed = cep.last_session_compressed.unwrap_or(0);
212    let is_same_session = cep.last_session_pid == Some(pid);
213
214    if is_same_session {
215        let delta_original = tokens_original.saturating_sub(prev_original);
216        let delta_compressed = tokens_compressed.saturating_sub(prev_compressed);
217        cep.total_tokens_original += delta_original;
218        cep.total_tokens_compressed += delta_compressed;
219    } else {
220        cep.sessions += 1;
221        cep.total_cache_hits += cache_hits;
222        cep.total_cache_reads += cache_reads;
223        cep.total_tokens_original += tokens_original;
224        cep.total_tokens_compressed += tokens_compressed;
225
226        for (mode, count) in modes {
227            *cep.modes.entry(mode.clone()).or_insert(0) += count;
228        }
229    }
230
231    cep.last_session_pid = Some(pid);
232    cep.last_session_original = Some(tokens_original);
233    cep.last_session_compressed = Some(tokens_compressed);
234
235    let cache_hit_rate = if cache_reads > 0 {
236        (cache_hits as f64 / cache_reads as f64 * 100.0).round() as u32
237    } else {
238        0
239    };
240
241    let compression_rate = if tokens_original > 0 {
242        ((tokens_original - tokens_compressed) as f64 / tokens_original as f64 * 100.0).round()
243            as u32
244    } else {
245        0
246    };
247
248    let total_modes = 6u32;
249    let mode_diversity =
250        ((modes.len() as f64 / total_modes as f64).min(1.0) * 100.0).round() as u32;
251
252    let tokens_saved = tokens_original.saturating_sub(tokens_compressed);
253
254    cep.scores.push(CepSessionSnapshot {
255        timestamp: chrono::Local::now().to_rfc3339(),
256        score,
257        cache_hit_rate,
258        mode_diversity,
259        compression_rate,
260        tool_calls,
261        tokens_saved,
262        complexity: complexity.to_string(),
263    });
264
265    if cep.scores.len() > 100 {
266        cep.scores.drain(..cep.scores.len() - 100);
267    }
268
269    maybe_flush(store, baseline, last_flush);
270}
271
272#[cfg(test)]
273mod tests {
274    use super::*;
275
276    fn make_store(commands: u64, input: u64, output: u64) -> StatsStore {
277        StatsStore {
278            total_commands: commands,
279            total_input_tokens: input,
280            total_output_tokens: output,
281            ..Default::default()
282        }
283    }
284
285    #[test]
286    fn apply_deltas_merges_mcp_and_shell() {
287        let baseline = make_store(0, 0, 0);
288        let mut current = make_store(0, 0, 0);
289        current.total_commands = 5;
290        current.total_input_tokens = 1000;
291        current.total_output_tokens = 200;
292        current.commands.insert(
293            "ctx_read".to_string(),
294            CommandStats {
295                count: 5,
296                input_tokens: 1000,
297                output_tokens: 200,
298            },
299        );
300
301        let mut disk = make_store(20, 500, 490);
302        disk.commands.insert(
303            "echo".to_string(),
304            CommandStats {
305                count: 20,
306                input_tokens: 500,
307                output_tokens: 490,
308            },
309        );
310
311        let merged = io::apply_deltas(&disk, &current, &baseline);
312
313        assert_eq!(merged.total_commands, 25);
314        assert_eq!(merged.total_input_tokens, 1500);
315        assert_eq!(merged.total_output_tokens, 690);
316        assert_eq!(merged.commands["ctx_read"].count, 5);
317        assert_eq!(merged.commands["echo"].count, 20);
318    }
319
320    #[test]
321    fn apply_deltas_incremental_flush() {
322        let baseline = make_store(10, 200, 100);
323        let current = make_store(15, 700, 300);
324
325        let disk = make_store(30, 600, 500);
326
327        let merged = io::apply_deltas(&disk, &current, &baseline);
328
329        assert_eq!(merged.total_commands, 35);
330        assert_eq!(merged.total_input_tokens, 1100);
331        assert_eq!(merged.total_output_tokens, 700);
332    }
333
334    #[test]
335    fn apply_deltas_preserves_disk_commands() {
336        let baseline = make_store(0, 0, 0);
337        let mut current = make_store(2, 100, 50);
338        current.commands.insert(
339            "ctx_read".to_string(),
340            CommandStats {
341                count: 2,
342                input_tokens: 100,
343                output_tokens: 50,
344            },
345        );
346
347        let mut disk = make_store(10, 300, 280);
348        disk.commands.insert(
349            "echo".to_string(),
350            CommandStats {
351                count: 8,
352                input_tokens: 200,
353                output_tokens: 200,
354            },
355        );
356        disk.commands.insert(
357            "ctx_read".to_string(),
358            CommandStats {
359                count: 3,
360                input_tokens: 150,
361                output_tokens: 80,
362            },
363        );
364
365        let merged = io::apply_deltas(&disk, &current, &baseline);
366
367        assert_eq!(merged.commands["echo"].count, 8);
368        assert_eq!(merged.commands["ctx_read"].count, 5);
369        assert_eq!(merged.commands["ctx_read"].input_tokens, 250);
370    }
371
372    #[test]
373    fn merge_daily_combines_same_date() {
374        let baseline_daily = vec![];
375        let current_daily = vec![DayStats {
376            date: "2026-04-18".to_string(),
377            commands: 5,
378            input_tokens: 1000,
379            output_tokens: 200,
380            version: "3.7.0".to_string(),
381        }];
382        let mut merged_daily = vec![DayStats {
383            date: "2026-04-18".to_string(),
384            commands: 20,
385            input_tokens: 500,
386            output_tokens: 490,
387            version: String::new(),
388        }];
389
390        io::merge_daily(&mut merged_daily, &current_daily, &baseline_daily);
391
392        assert_eq!(merged_daily.len(), 1);
393        assert_eq!(merged_daily[0].commands, 25);
394        assert_eq!(merged_daily[0].input_tokens, 1500);
395        // #307: the most recent known version is carried into the merge.
396        assert_eq!(merged_daily[0].version, "3.7.0");
397    }
398
399    #[test]
400    fn format_pct_1dp_normal() {
401        assert_eq!(format::format_pct_1dp(50.0), "50.0%");
402        assert_eq!(format::format_pct_1dp(100.0), "100.0%");
403        assert_eq!(format::format_pct_1dp(33.333), "33.3%");
404    }
405
406    #[test]
407    fn format_pct_1dp_small_values() {
408        assert_eq!(format::format_pct_1dp(0.0), "0.0%");
409        assert_eq!(format::format_pct_1dp(0.05), "<0.1%");
410        assert_eq!(format::format_pct_1dp(0.09), "<0.1%");
411        assert_eq!(format::format_pct_1dp(0.1), "0.1%");
412        assert_eq!(format::format_pct_1dp(0.5), "0.5%");
413    }
414}