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
12static 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
54pub 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 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, ¤t, &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, ¤t, &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, ¤t, &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, ¤t_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 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}