Skip to main content

fallow_engine/
vital_signs.rs

1//! Vital signs computation and snapshot persistence.
2//!
3//! Vital signs are a fixed set of project-wide metrics computed from available
4//! health data. They are always shown as a summary in the health report and can
5//! be persisted to `.fallow/snapshots/` for Phase 2b trend tracking.
6
7use std::path::{Path, PathBuf};
8
9use crate::git_env::clear_ambient_git_env;
10
11/// Number of seconds in one day.
12const SECS_PER_DAY: u64 = 86_400;
13
14use fallow_output::{
15    DEFAULT_CYCLOMATIC_CRITICAL, FileHealthScore, HEALTH_SCORE_FORMULA_VERSION,
16    HOTSPOT_SCORE_THRESHOLD, HealthScore, HealthScorePenalties, HealthTrend, HotspotEntry,
17    RiskProfile, SNAPSHOT_SCHEMA_VERSION, TrendCount, TrendDirection, TrendMetric, TrendPoint,
18    VitalSigns, VitalSignsCounts, VitalSignsSnapshot, letter_grade,
19};
20
21/// Data sources for computing vital signs.
22///
23/// Fields are `Option` because not all pipelines run in every health invocation.
24pub struct VitalSignsInput<'a> {
25    /// All parsed modules (always available).
26    pub modules: &'a [crate::source::ModuleInfo],
27    /// Optional file-id allowlist used to restrict per-module aggregates
28    /// (cyclomatic distribution, total LOC, unit profiles) to a subset.
29    /// Used by `--workspace` and `--group-by` to scope project-wide metrics
30    /// to a single workspace package without re-parsing.
31    /// `None` includes every module in `modules`.
32    pub module_filter: Option<&'a rustc_hash::FxHashSet<crate::discover::FileId>>,
33    /// File health scores (available when file_scores/hotspots/targets are computed).
34    pub file_scores: Option<&'a [FileHealthScore]>,
35    /// Hotspot entries (available when hotspots are computed).
36    pub hotspots: Option<&'a [HotspotEntry]>,
37    /// Total discovered files (already scoped to the workspace when `--workspace` is set).
38    pub total_files: usize,
39    /// Analysis results (available when file_scores pipeline ran). When a
40    /// `module_filter` is also set, callers should pass workspace-scoped
41    /// counts here so `dead_*_pct` denominators line up with the rest of the
42    /// metrics.
43    pub analysis_counts: Option<AnalysisCounts>,
44}
45
46impl<'a> VitalSignsInput<'a> {
47    /// Iterate the modules selected by `module_filter`.
48    fn selected_modules(&self) -> impl Iterator<Item = &'a crate::source::ModuleInfo> + '_ {
49        let filter = self.module_filter;
50        self.modules
51            .iter()
52            .filter(move |m| filter.is_none_or(|set| set.contains(&m.file_id)))
53    }
54}
55
56/// Aggregate counts from the analysis pipeline.
57#[derive(Clone, Copy)]
58pub struct AnalysisCounts {
59    pub total_exports: usize,
60    pub dead_files: usize,
61    pub dead_exports: usize,
62    pub unused_deps: usize,
63    pub circular_deps: usize,
64    pub total_deps: usize,
65}
66
67fn collect_sorted_cyclomatic(input: &VitalSignsInput<'_>) -> Vec<u16> {
68    let mut values: Vec<u16> = input
69        .selected_modules()
70        .flat_map(|m| m.complexity.iter().map(|c| c.cyclomatic))
71        .collect();
72    values.sort_unstable();
73    values
74}
75
76fn average_cyclomatic(all_cyclomatic: &[u16]) -> f64 {
77    if all_cyclomatic.is_empty() {
78        return 0.0;
79    }
80
81    let sum: u64 = all_cyclomatic.iter().map(|&c| u64::from(c)).sum();
82    (sum as f64 / all_cyclomatic.len() as f64 * 10.0).round() / 10.0
83}
84
85fn critical_complexity_pct(all_cyclomatic: &[u16]) -> Option<f64> {
86    if all_cyclomatic.is_empty() {
87        return None;
88    }
89
90    let critical_count = all_cyclomatic
91        .iter()
92        .filter(|&&c| c >= DEFAULT_CYCLOMATIC_CRITICAL)
93        .count();
94    Some((critical_count as f64 / all_cyclomatic.len() as f64 * 1000.0).round() / 10.0)
95}
96
97#[expect(
98    clippy::cast_possible_truncation,
99    reason = "percentile index is bounded by the cyclomatic collection length"
100)]
101fn p90_cyclomatic(all_cyclomatic: &[u16]) -> u32 {
102    if all_cyclomatic.is_empty() {
103        return 0;
104    }
105
106    let idx = (all_cyclomatic.len() as f64 * 0.9).ceil() as usize;
107    let idx = idx.min(all_cyclomatic.len()) - 1;
108    u32::from(all_cyclomatic[idx])
109}
110
111#[expect(
112    clippy::cast_possible_truncation,
113    reason = "analysis counts are bounded by project size and emitted as compact u32 metrics"
114)]
115fn analysis_count_vitals(
116    counts: Option<&AnalysisCounts>,
117    total_files: usize,
118) -> (Option<f64>, Option<f64>, Option<u32>, Option<u32>) {
119    let Some(counts) = counts else {
120        return (None, None, None, None);
121    };
122
123    let dead_file_pct = if total_files > 0 {
124        Some((counts.dead_files as f64 / total_files as f64 * 1000.0).round() / 10.0)
125    } else {
126        Some(0.0)
127    };
128    let dead_export_pct = if counts.total_exports > 0 {
129        Some((counts.dead_exports as f64 / counts.total_exports as f64 * 1000.0).round() / 10.0)
130    } else {
131        Some(0.0)
132    };
133
134    (
135        dead_file_pct,
136        dead_export_pct,
137        Some(counts.unused_deps as u32),
138        Some(counts.circular_deps as u32),
139    )
140}
141
142struct SelectedModuleMetrics {
143    total_loc: u64,
144    line_counts: Vec<u32>,
145    param_counts: Vec<u8>,
146}
147
148fn selected_module_metrics(input: &VitalSignsInput<'_>) -> SelectedModuleMetrics {
149    let mut total_loc = 0;
150    let mut line_counts = Vec::new();
151    let mut param_counts = Vec::new();
152
153    for module in input.selected_modules() {
154        total_loc += module.line_offsets.len() as u64;
155        line_counts.extend(module.complexity.iter().map(|c| c.line_count));
156        param_counts.extend(module.complexity.iter().map(|c| c.param_count));
157    }
158
159    SelectedModuleMetrics {
160        total_loc,
161        line_counts,
162        param_counts,
163    }
164}
165
166fn vital_sign_counts(input: &VitalSignsInput<'_>, total_loc: u64) -> Option<VitalSignsCounts> {
167    input.analysis_counts.as_ref().map(|ac| VitalSignsCounts {
168        total_files: input.total_files,
169        total_exports: ac.total_exports,
170        dead_files: ac.dead_files,
171        dead_exports: ac.dead_exports,
172        duplicated_lines: None,
173        total_lines: Some(total_loc as usize),
174        files_scored: input.file_scores.map(<[_]>::len),
175        total_deps: ac.total_deps,
176    })
177}
178
179/// Compute vital signs from available health data.
180pub fn compute_vital_signs(input: &VitalSignsInput<'_>) -> VitalSigns {
181    let all_cyclomatic = collect_sorted_cyclomatic(input);
182    let avg_cyclomatic = average_cyclomatic(&all_cyclomatic);
183    let critical_complexity_pct = critical_complexity_pct(&all_cyclomatic);
184    let p90_cyclomatic = p90_cyclomatic(&all_cyclomatic);
185
186    let (dead_file_pct, dead_export_pct, unused_dep_count, circular_dep_count) =
187        analysis_count_vitals(input.analysis_counts.as_ref(), input.total_files);
188    let unused_deps_per_k_files =
189        unused_dep_count.map(|count| per_k_files(count, input.total_files));
190    let circular_deps_per_k_files =
191        circular_dep_count.map(|count| per_k_files(count, input.total_files));
192
193    let (maintainability_avg, maintainability_low_pct) = maintainability_vitals(input.file_scores);
194
195    let (hotspot_count, hotspot_top_pct_count) = hotspot_vitals(input.hotspots, input.total_files);
196
197    let module_metrics = selected_module_metrics(input);
198    let counts = vital_sign_counts(input, module_metrics.total_loc);
199    let functions_over_60_loc_per_k = functions_over_60_loc_per_k(&module_metrics.line_counts);
200    let unit_size_profile = unit_size_profile(&module_metrics.line_counts);
201
202    let unit_interfacing_profile =
203        unit_interfacing_profile(&module_metrics.param_counts, &all_cyclomatic);
204
205    let (p95_fan_in, coupling_high_pct) = if let Some(scores) = input.file_scores {
206        compute_coupling_concentration(scores)
207    } else {
208        (None, None)
209    };
210
211    VitalSigns {
212        dead_file_pct,
213        dead_export_pct,
214        avg_cyclomatic,
215        critical_complexity_pct,
216        p90_cyclomatic,
217        duplication_pct: None, // Lazy: only set if duplication pipeline was run
218        hotspot_count,
219        hotspot_top_pct_count,
220        maintainability_avg,
221        maintainability_low_pct,
222        unused_dep_count,
223        unused_deps_per_k_files,
224        circular_dep_count,
225        circular_deps_per_k_files,
226        counts,
227        unit_size_profile,
228        functions_over_60_loc_per_k,
229        unit_interfacing_profile,
230        p95_fan_in,
231        coupling_high_pct,
232        // Set post-construction from the whole-project analysis results (only
233        // when the opt-in prop-drilling rule is enabled); see health/mod.rs.
234        prop_drilling_chain_count: None,
235        prop_drilling_max_depth: None,
236        // Set post-construction from the whole-project render fan-in metric
237        // (whenever React is declared, the core metric carries the aggregates);
238        // see health/mod.rs.
239        p95_render_fan_in: None,
240        render_fan_in_high_pct: None,
241        max_render_fan_in: None,
242        top_render_fan_in: Vec::new(),
243        total_loc: module_metrics.total_loc,
244    }
245}
246
247fn per_k_files(count: u32, total_files: usize) -> f64 {
248    if total_files == 0 {
249        0.0
250    } else {
251        (f64::from(count) / total_files as f64 * 10_000.0).round() / 10.0
252    }
253}
254
255fn maintainability_vitals(scores: Option<&[FileHealthScore]>) -> (Option<f64>, Option<f64>) {
256    let Some(scores) = scores.filter(|scores| !scores.is_empty()) else {
257        return (None, None);
258    };
259    let sum: f64 = scores.iter().map(|s| s.maintainability_index).sum();
260    let low_count = scores
261        .iter()
262        .filter(|s| s.maintainability_index < 70.0)
263        .count();
264    (
265        Some((sum / scores.len() as f64 * 10.0).round() / 10.0),
266        Some((low_count as f64 / scores.len() as f64 * 1000.0).round() / 10.0),
267    )
268}
269
270fn hotspot_vitals(
271    hotspots: Option<&[HotspotEntry]>,
272    total_files: usize,
273) -> (Option<u32>, Option<u32>) {
274    let hotspot_count = hotspots.map(|entries| {
275        entries
276            .iter()
277            .filter(|e| e.score >= HOTSPOT_SCORE_THRESHOLD)
278            .count() as u32
279    });
280    let hotspot_top_pct_count = hotspots.map(|entries| {
281        if total_files == 0 || entries.is_empty() {
282            return 0;
283        }
284        let top_count = (total_files as f64 * 0.01).ceil() as usize;
285        entries
286            .iter()
287            .take(top_count.max(1))
288            .filter(|entry| entry.score > 0.0)
289            .count() as u32
290    });
291    (hotspot_count, hotspot_top_pct_count)
292}
293
294fn functions_over_60_loc_per_k(line_counts: &[u32]) -> Option<f64> {
295    if line_counts.is_empty() {
296        return None;
297    }
298    let over_60 = line_counts
299        .iter()
300        .filter(|&&line_count| line_count > 60)
301        .count();
302    Some((over_60 as f64 / line_counts.len() as f64 * 10_000.0).round() / 10.0)
303}
304
305fn unit_size_profile(line_counts: &[u32]) -> Option<RiskProfile> {
306    (!line_counts.is_empty()).then(|| compute_size_risk_profile(line_counts))
307}
308
309fn unit_interfacing_profile(param_counts: &[u8], all_cyclomatic: &[u16]) -> Option<RiskProfile> {
310    if all_cyclomatic.is_empty() {
311        return None;
312    }
313    Some(compute_interfacing_risk_profile(param_counts))
314}
315
316/// Compute unit size risk profile from function line counts.
317///
318/// Bins: low risk (1-15 LOC), medium risk (16-30), high risk (31-60), very high risk (>60).
319fn compute_size_risk_profile(line_counts: &[u32]) -> RiskProfile {
320    if line_counts.is_empty() {
321        return RiskProfile {
322            low_risk: 0.0,
323            medium_risk: 0.0,
324            high_risk: 0.0,
325            very_high_risk: 0.0,
326        };
327    }
328    let total = line_counts.len() as f64;
329    let low = line_counts.iter().filter(|&&lc| lc <= 15).count() as f64;
330    let medium = line_counts
331        .iter()
332        .filter(|&&lc| (16..=30).contains(&lc))
333        .count() as f64;
334    let high = line_counts
335        .iter()
336        .filter(|&&lc| (31..=60).contains(&lc))
337        .count() as f64;
338    let very_high = line_counts.iter().filter(|&&lc| lc > 60).count() as f64;
339    RiskProfile {
340        low_risk: (low / total * 1000.0).round() / 10.0,
341        medium_risk: (medium / total * 1000.0).round() / 10.0,
342        high_risk: (high / total * 1000.0).round() / 10.0,
343        very_high_risk: (very_high / total * 1000.0).round() / 10.0,
344    }
345}
346
347/// Compute unit interfacing risk profile from function parameter counts.
348///
349/// Bins: low risk (0-2 params), medium risk (3-4), high risk (5-6), very high risk (>=7).
350fn compute_interfacing_risk_profile(param_counts: &[u8]) -> RiskProfile {
351    if param_counts.is_empty() {
352        return RiskProfile {
353            low_risk: 0.0,
354            medium_risk: 0.0,
355            high_risk: 0.0,
356            very_high_risk: 0.0,
357        };
358    }
359    let total = param_counts.len() as f64;
360    let low = param_counts.iter().filter(|&&pc| pc <= 2).count() as f64;
361    let medium = param_counts
362        .iter()
363        .filter(|&&pc| (3..=4).contains(&pc))
364        .count() as f64;
365    let high = param_counts
366        .iter()
367        .filter(|&&pc| (5..=6).contains(&pc))
368        .count() as f64;
369    let very_high = param_counts.iter().filter(|&&pc| pc >= 7).count() as f64;
370    RiskProfile {
371        low_risk: (low / total * 1000.0).round() / 10.0,
372        medium_risk: (medium / total * 1000.0).round() / 10.0,
373        high_risk: (high / total * 1000.0).round() / 10.0,
374        very_high_risk: (very_high / total * 1000.0).round() / 10.0,
375    }
376}
377
378/// Compute coupling concentration from file health scores.
379///
380/// Returns (p95_fan_in, coupling_high_pct) where coupling_high_pct is the
381/// percentage of files with fan-in above the effective threshold (max(p95_fan_in, 10)).
382///
383/// The component-graph analogue (render fan-in concentration:
384/// `p95_render_fan_in` / `render_fan_in_high_pct` / `max_render_fan_in`) is
385/// computed in core (`crate::render_fan_in`), which has the
386/// resolved-module graph the CLI lacks. It mirrors this helper verbatim (p95 +
387/// `high_pct` over the per-component distinct-parents distribution, reusing the
388/// same `max(p95, 10)` floor) and is assigned onto `VitalSigns` in
389/// `health/mod.rs::prepare_health_vital_data`.
390#[expect(
391    clippy::cast_possible_truncation,
392    reason = "fan-in values are bounded by project size"
393)]
394fn compute_coupling_concentration(scores: &[FileHealthScore]) -> (Option<u32>, Option<f64>) {
395    if scores.is_empty() {
396        return (None, None);
397    }
398    let mut fan_ins: Vec<usize> = scores.iter().map(|s| s.fan_in).collect();
399    fan_ins.sort_unstable();
400    let idx = (fan_ins.len() as f64 * 0.95).ceil() as usize;
401    let idx = idx.min(fan_ins.len()) - 1;
402    let p95 = fan_ins[idx] as u32;
403
404    let threshold = (p95 as usize).max(10);
405    let high_count = fan_ins.iter().filter(|&&fi| fi > threshold).count();
406    let high_pct = (high_count as f64 / fan_ins.len() as f64 * 1000.0).round() / 10.0;
407
408    (Some(p95), Some(high_pct))
409}
410
411/// Compute a project-level health score from vital signs.
412///
413/// The score starts at 100 and subtracts penalties for each metric.
414/// Missing metrics (from pipelines that didn't run) don't penalize.
415/// `total_files` is used to normalize the hotspot count penalty.
416pub fn compute_health_score(vs: &VitalSigns, total_files: usize) -> HealthScore {
417    let penalties = compute_health_score_penalties(vs, total_files);
418    let score = apply_health_score_penalties(&penalties);
419    let grade = letter_grade(score);
420
421    HealthScore {
422        formula_version: HEALTH_SCORE_FORMULA_VERSION,
423        score,
424        grade,
425        penalties,
426    }
427}
428
429fn compute_health_score_penalties(vs: &VitalSigns, total_files: usize) -> HealthScorePenalties {
430    HealthScorePenalties {
431        dead_files: vs.dead_file_pct.map(|pct| round1((pct * 0.2).min(15.0))),
432        dead_exports: vs.dead_export_pct.map(|pct| round1((pct * 0.2).min(15.0))),
433        complexity: complexity_penalty(vs),
434        p90_complexity: p90_complexity_penalty(vs),
435        maintainability: maintainability_penalty(vs),
436        hotspots: hotspot_penalty(vs, total_files),
437        unused_deps: dependency_count_penalty(
438            vs.unused_deps_per_k_files,
439            vs.unused_dep_count,
440            25.0,
441            10.0,
442        ),
443        circular_deps: dependency_count_penalty(
444            vs.circular_deps_per_k_files,
445            vs.circular_dep_count,
446            25.0,
447            10.0,
448        ),
449        unit_size: unit_size_penalty(vs),
450        coupling: coupling_penalty(vs),
451        duplication: vs
452            .duplication_pct
453            .map(|dp| round1((dp - 5.0).clamp(0.0, 10.0))),
454        prop_drilling: prop_drilling_penalty(vs),
455    }
456}
457
458fn apply_health_score_penalties(penalties: &HealthScorePenalties) -> f64 {
459    let mut score = 100.0_f64;
460
461    subtract_optional_penalty(&mut score, penalties.dead_files);
462    subtract_optional_penalty(&mut score, penalties.dead_exports);
463    score -= penalties.complexity;
464    score -= penalties.p90_complexity;
465    subtract_optional_penalty(&mut score, penalties.maintainability);
466    subtract_optional_penalty(&mut score, penalties.hotspots);
467    subtract_optional_penalty(&mut score, penalties.unused_deps);
468    subtract_optional_penalty(&mut score, penalties.circular_deps);
469    subtract_optional_penalty(&mut score, penalties.unit_size);
470    subtract_optional_penalty(&mut score, penalties.coupling);
471    subtract_optional_penalty(&mut score, penalties.duplication);
472    subtract_optional_penalty(&mut score, penalties.prop_drilling);
473
474    round1(score).clamp(0.0, 100.0)
475}
476
477/// Small capped penalty for prop-drilling chains, sized like the coupling
478/// penalty (~5pt cap). Each located chain costs 1pt up to the cap; a deeper
479/// chain does not cost more (depth is descriptive, not a tunable threshold).
480/// `None` (no penalty) unless the opt-in `prop-drilling` rule populated the
481/// count, so the score is unchanged by default.
482fn prop_drilling_penalty(vs: &VitalSigns) -> Option<f64> {
483    vs.prop_drilling_chain_count
484        .map(|count| round1((f64::from(count) * 1.0).min(5.0)))
485}
486
487fn round1(value: f64) -> f64 {
488    (value * 10.0).round() / 10.0
489}
490
491fn subtract_optional_penalty(score: &mut f64, penalty: Option<f64>) {
492    if let Some(penalty) = penalty {
493        *score -= penalty;
494    }
495}
496
497fn complexity_penalty(vs: &VitalSigns) -> f64 {
498    if let Some(critical_pct) = vs.critical_complexity_pct {
499        round1((critical_pct * 4.0).min(20.0))
500    } else {
501        round1(((vs.avg_cyclomatic - 1.5).max(0.0) * 5.0).min(20.0))
502    }
503}
504
505fn p90_complexity_penalty(vs: &VitalSigns) -> f64 {
506    if vs.critical_complexity_pct.is_some() {
507        0.0
508    } else {
509        round1((f64::from(vs.p90_cyclomatic) - 10.0).clamp(0.0, 10.0))
510    }
511}
512
513fn maintainability_penalty(vs: &VitalSigns) -> Option<f64> {
514    if let Some(low_pct) = vs.maintainability_low_pct {
515        Some(round1((low_pct * 1.5).min(15.0)))
516    } else {
517        vs.maintainability_avg
518            .map(|mi| round1(((70.0 - mi).max(0.0) * 0.5).min(15.0)))
519    }
520}
521
522fn hotspot_penalty(vs: &VitalSigns, total_files: usize) -> Option<f64> {
523    if let Some(top_pct_count) = vs.hotspot_top_pct_count {
524        return Some(if total_files > 0 {
525            let top_pct_bucket = (total_files as f64 * 0.01).ceil().max(1.0);
526            round1((f64::from(top_pct_count) / top_pct_bucket * 10.0).min(10.0))
527        } else {
528            0.0
529        });
530    }
531
532    vs.hotspot_count.map(|hc| {
533        if total_files > 0 {
534            round1((f64::from(hc) / total_files as f64 * 200.0).min(10.0))
535        } else {
536            0.0
537        }
538    })
539}
540
541fn dependency_count_penalty(
542    per_k: Option<f64>,
543    count: Option<u32>,
544    per_k_cap: f64,
545    count_cap: f64,
546) -> Option<f64> {
547    if let Some(per_k) = per_k {
548        Some(round1((per_k * 0.5).min(per_k_cap)))
549    } else {
550        count.map(|count| round1(f64::from(count).min(count_cap)))
551    }
552}
553
554fn unit_size_penalty(vs: &VitalSigns) -> Option<f64> {
555    if let Some(per_k) = vs.functions_over_60_loc_per_k {
556        Some(round1((per_k * 0.5).min(10.0)))
557    } else {
558        vs.unit_size_profile
559            .as_ref()
560            .map(|profile| round1(((profile.very_high_risk - 5.0).max(0.0) * 0.5).min(10.0)))
561    }
562}
563
564fn coupling_penalty(vs: &VitalSigns) -> Option<f64> {
565    if let Some(high_pct) = vs.coupling_high_pct {
566        Some(round1((high_pct * 0.5).min(5.0)))
567    } else {
568        vs.p95_fan_in
569            .map(|p95| round1(((f64::from(p95) - 30.0).max(0.0) * 0.25).min(5.0)))
570    }
571}
572
573/// Build the raw counts for a snapshot.
574pub fn build_counts(input: &VitalSignsInput<'_>) -> VitalSignsCounts {
575    let (total_exports, dead_files, dead_exports, total_deps) = input
576        .analysis_counts
577        .as_ref()
578        .map_or((0, 0, 0, 0), |counts| {
579            (
580                counts.total_exports,
581                counts.dead_files,
582                counts.dead_exports,
583                counts.total_deps,
584            )
585        });
586
587    let total_lines: usize = input.selected_modules().map(|m| m.line_offsets.len()).sum();
588
589    VitalSignsCounts {
590        total_files: input.total_files,
591        total_exports,
592        dead_files,
593        dead_exports,
594        duplicated_lines: None,
595        total_lines: Some(total_lines),
596        files_scored: input.file_scores.map(<[_]>::len),
597        total_deps,
598    }
599}
600
601/// Get the current git SHA (short form).
602#[expect(
603    clippy::disallowed_methods,
604    reason = "trusted git spawn with ambient repo-state env stripped, matching the core git spawn policy"
605)]
606fn git_sha(root: &Path) -> Option<String> {
607    let mut command = std::process::Command::new("git");
608    command
609        .args(["rev-parse", "--short", "HEAD"])
610        .current_dir(root);
611    clear_ambient_git_env(&mut command);
612    command
613        .output()
614        .ok()
615        .filter(|o| o.status.success())
616        .map(|o| String::from_utf8_lossy(&o.stdout).trim().to_string())
617}
618
619/// Get the current git branch name.
620#[expect(
621    clippy::disallowed_methods,
622    reason = "trusted git spawn with ambient repo-state env stripped, matching the core git spawn policy"
623)]
624fn git_branch(root: &Path) -> Option<String> {
625    let mut command = std::process::Command::new("git");
626    command
627        .args(["rev-parse", "--abbrev-ref", "HEAD"])
628        .current_dir(root);
629    clear_ambient_git_env(&mut command);
630    command
631        .output()
632        .ok()
633        .filter(|o| o.status.success())
634        .and_then(|o| {
635            let name = String::from_utf8_lossy(&o.stdout).trim().to_string();
636            if name == "HEAD" { None } else { Some(name) }
637        })
638}
639
640/// Build a snapshot from vital signs and input data.
641pub fn build_snapshot(
642    vital_signs: VitalSigns,
643    counts: VitalSignsCounts,
644    root: &Path,
645    shallow_clone: bool,
646    health_score: Option<&HealthScore>,
647    coverage_model: Option<fallow_output::CoverageModel>,
648) -> VitalSignsSnapshot {
649    let now = chrono_timestamp();
650
651    VitalSignsSnapshot {
652        snapshot_schema_version: SNAPSHOT_SCHEMA_VERSION,
653        version: env!("CARGO_PKG_VERSION").to_string(),
654        timestamp: now,
655        git_sha: git_sha(root),
656        git_branch: git_branch(root),
657        shallow_clone,
658        vital_signs,
659        counts,
660        score: health_score.map(|s| s.score),
661        grade: health_score.map(|s| s.grade.to_string()),
662        coverage_model,
663    }
664}
665
666/// ISO 8601 UTC timestamp without external chrono dependency.
667pub fn chrono_timestamp() -> String {
668    use std::time::SystemTime;
669    let now = SystemTime::now()
670        .duration_since(SystemTime::UNIX_EPOCH)
671        .unwrap_or_default();
672    let secs = now.as_secs();
673
674    let days = secs / SECS_PER_DAY;
675    let time_secs = secs % SECS_PER_DAY;
676    let hours = time_secs / 3600;
677    let minutes = (time_secs % 3600) / 60;
678    let seconds = time_secs % 60;
679
680    let (year, month, day) = days_to_ymd(days);
681
682    format!("{year:04}-{month:02}-{day:02}T{hours:02}:{minutes:02}:{seconds:02}Z")
683}
684
685/// Convert days since Unix epoch to (year, month, day).
686const fn days_to_ymd(days: u64) -> (u64, u64, u64) {
687    let z = days + 719_468;
688    let era = z / 146_097;
689    let doe = z - era * 146_097;
690    let yoe = (doe - doe / 1460 + doe / 36524 - doe / 146_096) / 365;
691    let y = yoe + era * 400;
692    let doy = doe - (365 * yoe + yoe / 4 - yoe / 100);
693    let mp = (5 * doy + 2) / 153;
694    let d = doy - (153 * mp + 2) / 5 + 1;
695    let m = if mp < 10 { mp + 3 } else { mp - 9 };
696    let y = if m <= 2 { y + 1 } else { y };
697    (y, m, d)
698}
699
700/// Save a snapshot to disk.
701///
702/// If `path` is `None`, writes to `.fallow/snapshots/{timestamp}.json`.
703/// Creates parent directories as needed.
704pub fn save_snapshot(
705    snapshot: &VitalSignsSnapshot,
706    root: &Path,
707    explicit_path: Option<&Path>,
708) -> Result<PathBuf, String> {
709    let path = explicit_path.map_or_else(
710        || {
711            let dir = root.join(".fallow").join("snapshots");
712            let filename = snapshot.timestamp.replace(':', "-");
713            dir.join(format!("{filename}.json"))
714        },
715        Path::to_path_buf,
716    );
717
718    if let Some(parent) = path.parent() {
719        std::fs::create_dir_all(parent)
720            .map_err(|e| format!("failed to create snapshot directory: {e}"))?;
721    }
722
723    let json =
724        serde_json::to_string_pretty(snapshot).map_err(|e| format!("failed to serialize: {e}"))?;
725    std::fs::write(&path, json).map_err(|e| format!("failed to write snapshot: {e}"))?;
726
727    Ok(path)
728}
729
730/// Load all snapshots from the default snapshot directory, sorted by timestamp ascending.
731///
732/// Corrupt or unreadable files are skipped with a warning to stderr.
733/// Returns an empty vec if the directory does not exist.
734#[expect(
735    clippy::print_stderr,
736    reason = "corrupt-snapshot warnings to stderr, preserved verbatim from the CLI health path"
737)]
738pub fn load_snapshots(root: &Path) -> Vec<VitalSignsSnapshot> {
739    let dir = root.join(".fallow").join("snapshots");
740    let Ok(entries) = std::fs::read_dir(&dir) else {
741        return Vec::new();
742    };
743
744    let mut snapshots = Vec::new();
745    for entry in entries {
746        let Ok(entry) = entry else { continue };
747        let path = entry.path();
748        if path.extension().is_some_and(|ext| ext == "json") {
749            match std::fs::read_to_string(&path) {
750                Ok(content) => match serde_json::from_str::<VitalSignsSnapshot>(&content) {
751                    Ok(snap) => snapshots.push(snap),
752                    Err(e) => {
753                        eprintln!("warning: skipping corrupt snapshot {}: {e}", path.display());
754                    }
755                },
756                Err(e) => {
757                    eprintln!("warning: could not read snapshot {}: {e}", path.display());
758                }
759            }
760        }
761    }
762
763    snapshots.sort_by(|a, b| a.timestamp.cmp(&b.timestamp));
764    snapshots
765}
766
767/// Tolerance for treating a metric delta as "stable" rather than improving/declining.
768const TREND_TOLERANCE: f64 = 0.5;
769
770fn trend_point_from_snapshot(prev: &VitalSignsSnapshot) -> TrendPoint {
771    TrendPoint {
772        timestamp: prev.timestamp.clone(),
773        git_sha: prev.git_sha.clone(),
774        score: prev.score,
775        grade: prev.grade.clone(),
776        coverage_model: prev.coverage_model.clone(),
777        snapshot_schema_version: Some(prev.snapshot_schema_version),
778    }
779}
780
781fn overall_trend_direction(metrics: &[TrendMetric]) -> TrendDirection {
782    let (improving, declining) = metrics.iter().fold((0usize, 0usize), |(imp, dec), metric| {
783        match metric.direction {
784            TrendDirection::Improving => (imp + 1, dec),
785            TrendDirection::Declining => (imp, dec + 1),
786            TrendDirection::Stable => (imp, dec),
787        }
788    });
789
790    match improving.cmp(&declining) {
791        std::cmp::Ordering::Greater => TrendDirection::Improving,
792        std::cmp::Ordering::Less => TrendDirection::Declining,
793        std::cmp::Ordering::Equal => TrendDirection::Stable,
794    }
795}
796
797/// Compute a trend comparison between the current run and the most recent snapshot.
798///
799/// Uses the stored `score` field from the snapshot (never re-derives it).
800/// Returns `None` if no snapshots are available.
801pub fn compute_trend(
802    current_vs: &VitalSigns,
803    current_counts: &VitalSignsCounts,
804    current_score: Option<f64>,
805    snapshots: &[VitalSignsSnapshot],
806) -> Option<HealthTrend> {
807    let prev = snapshots.last()?;
808
809    let compared_to = trend_point_from_snapshot(prev);
810
811    let metrics = TrendBuilder::new(prev, current_vs, current_counts, current_score).build();
812
813    let overall_direction = overall_trend_direction(&metrics);
814
815    Some(HealthTrend {
816        compared_to,
817        metrics,
818        snapshots_loaded: snapshots.len(),
819        overall_direction,
820    })
821}
822
823struct TrendBuilder<'a> {
824    prev: &'a VitalSignsSnapshot,
825    current_vs: &'a VitalSigns,
826    current_counts: &'a VitalSignsCounts,
827    current_score: Option<f64>,
828    metrics: Vec<TrendMetric>,
829}
830
831impl TrendBuilder<'_> {
832    fn new<'a>(
833        prev: &'a VitalSignsSnapshot,
834        current_vs: &'a VitalSigns,
835        current_counts: &'a VitalSignsCounts,
836        current_score: Option<f64>,
837    ) -> TrendBuilder<'a> {
838        TrendBuilder {
839            prev,
840            current_vs,
841            current_counts,
842            current_score,
843            metrics: Vec::new(),
844        }
845    }
846
847    fn build(mut self) -> Vec<TrendMetric> {
848        self.add_score_metric();
849        self.add_dead_code_metrics();
850        self.add_complexity_metrics();
851        self.add_dependency_metrics();
852        self.add_structure_metrics();
853        self.metrics
854    }
855
856    fn push(&mut self, input: TrendMetricInput) {
857        self.metrics.push(make_metric(input));
858    }
859
860    fn add_score_metric(&mut self) {
861        if let (Some(prev_score), Some(cur_score)) = (self.prev.score, self.current_score) {
862            self.push(TrendMetricInput {
863                name: "score",
864                label: "Health Score",
865                previous: prev_score,
866                current: cur_score,
867                unit: "",
868                higher_is_better: true,
869                previous_count: None,
870                current_count: None,
871            });
872        }
873    }
874
875    fn add_dead_code_metrics(&mut self) {
876        if let (Some(prev_val), Some(cur_val)) = (
877            self.prev.vital_signs.dead_file_pct,
878            self.current_vs.dead_file_pct,
879        ) {
880            self.push(TrendMetricInput {
881                name: "dead_file_pct",
882                label: "Dead Files",
883                previous: prev_val,
884                current: cur_val,
885                unit: "%",
886                higher_is_better: false,
887                previous_count: Some(TrendCount {
888                    value: self.prev.counts.dead_files,
889                    total: self.prev.counts.total_files,
890                }),
891                current_count: Some(TrendCount {
892                    value: self.current_counts.dead_files,
893                    total: self.current_counts.total_files,
894                }),
895            });
896        }
897
898        if let (Some(prev_val), Some(cur_val)) = (
899            self.prev.vital_signs.dead_export_pct,
900            self.current_vs.dead_export_pct,
901        ) {
902            self.push(TrendMetricInput {
903                name: "dead_export_pct",
904                label: "Dead Exports",
905                previous: prev_val,
906                current: cur_val,
907                unit: "%",
908                higher_is_better: false,
909                previous_count: Some(TrendCount {
910                    value: self.prev.counts.dead_exports,
911                    total: self.prev.counts.total_exports,
912                }),
913                current_count: Some(TrendCount {
914                    value: self.current_counts.dead_exports,
915                    total: self.current_counts.total_exports,
916                }),
917            });
918        }
919    }
920
921    fn add_complexity_metrics(&mut self) {
922        self.push(TrendMetricInput {
923            name: "avg_cyclomatic",
924            label: "Avg Cyclomatic",
925            previous: self.prev.vital_signs.avg_cyclomatic,
926            current: self.current_vs.avg_cyclomatic,
927            unit: "",
928            higher_is_better: false,
929            previous_count: None,
930            current_count: None,
931        });
932
933        if let (Some(prev_val), Some(cur_val)) = (
934            self.prev.vital_signs.maintainability_avg,
935            self.current_vs.maintainability_avg,
936        ) {
937            self.push(TrendMetricInput {
938                name: "maintainability_avg",
939                label: "Maintainability",
940                previous: prev_val,
941                current: cur_val,
942                unit: "",
943                higher_is_better: true,
944                previous_count: None,
945                current_count: None,
946            });
947        }
948
949        if let (Some(prev_profile), Some(cur_profile)) = (
950            &self.prev.vital_signs.unit_size_profile,
951            &self.current_vs.unit_size_profile,
952        ) {
953            self.push(TrendMetricInput {
954                name: "unit_size_very_high_pct",
955                label: "Oversized Fns",
956                previous: prev_profile.very_high_risk,
957                current: cur_profile.very_high_risk,
958                unit: "%",
959                higher_is_better: false,
960                previous_count: None,
961                current_count: None,
962            });
963        }
964
965        self.add_duplication_metric();
966    }
967
968    fn add_duplication_metric(&mut self) {
969        if let (Some(prev_val), Some(cur_val)) = (
970            self.prev.vital_signs.duplication_pct,
971            self.current_vs.duplication_pct,
972        ) {
973            self.push(TrendMetricInput {
974                name: "duplication_pct",
975                label: "Duplication",
976                previous: prev_val,
977                current: cur_val,
978                unit: "%",
979                higher_is_better: false,
980                previous_count: self
981                    .prev
982                    .counts
983                    .duplicated_lines
984                    .zip(self.prev.counts.total_lines)
985                    .map(|(d, t)| TrendCount { value: d, total: t }),
986                current_count: self
987                    .current_counts
988                    .duplicated_lines
989                    .zip(self.current_counts.total_lines)
990                    .map(|(d, t)| TrendCount { value: d, total: t }),
991            });
992        }
993    }
994
995    fn add_dependency_metrics(&mut self) {
996        if let (Some(prev_val), Some(cur_val)) = (
997            self.prev.vital_signs.unused_dep_count,
998            self.current_vs.unused_dep_count,
999        ) {
1000            self.push(TrendMetricInput {
1001                name: "unused_dep_count",
1002                label: "Unused Deps",
1003                previous: f64::from(prev_val),
1004                current: f64::from(cur_val),
1005                unit: "",
1006                higher_is_better: false,
1007                previous_count: None,
1008                current_count: None,
1009            });
1010        }
1011    }
1012
1013    fn add_structure_metrics(&mut self) {
1014        if let (Some(prev_val), Some(cur_val)) = (
1015            self.prev.vital_signs.circular_dep_count,
1016            self.current_vs.circular_dep_count,
1017        ) {
1018            self.push(TrendMetricInput {
1019                name: "circular_dep_count",
1020                label: "Circular Deps",
1021                previous: f64::from(prev_val),
1022                current: f64::from(cur_val),
1023                unit: "",
1024                higher_is_better: false,
1025                previous_count: None,
1026                current_count: None,
1027            });
1028        }
1029
1030        if let (Some(prev_val), Some(cur_val)) = (
1031            self.prev.vital_signs.hotspot_count,
1032            self.current_vs.hotspot_count,
1033        ) {
1034            self.push(TrendMetricInput {
1035                name: "hotspot_count",
1036                label: "Hotspots",
1037                previous: f64::from(prev_val),
1038                current: f64::from(cur_val),
1039                unit: "",
1040                higher_is_better: false,
1041                previous_count: None,
1042                current_count: None,
1043            });
1044        }
1045
1046        if let (Some(prev_val), Some(cur_val)) =
1047            (self.prev.vital_signs.p95_fan_in, self.current_vs.p95_fan_in)
1048        {
1049            self.push(TrendMetricInput {
1050                name: "p95_fan_in",
1051                label: "P95 Fan-in",
1052                previous: f64::from(prev_val),
1053                current: f64::from(cur_val),
1054                unit: "",
1055                higher_is_better: false,
1056                previous_count: None,
1057                current_count: None,
1058            });
1059        }
1060    }
1061}
1062
1063/// Build a single trend metric.
1064struct TrendMetricInput {
1065    name: &'static str,
1066    label: &'static str,
1067    previous: f64,
1068    current: f64,
1069    unit: &'static str,
1070    higher_is_better: bool,
1071    previous_count: Option<TrendCount>,
1072    current_count: Option<TrendCount>,
1073}
1074
1075fn make_metric(input: TrendMetricInput) -> TrendMetric {
1076    let TrendMetricInput {
1077        name,
1078        label,
1079        previous,
1080        current,
1081        unit,
1082        higher_is_better,
1083        previous_count,
1084        current_count,
1085    } = input;
1086    let delta = (current - previous).round_to(1);
1087    let direction = if delta.abs() < TREND_TOLERANCE {
1088        TrendDirection::Stable
1089    } else if (higher_is_better && delta > 0.0) || (!higher_is_better && delta < 0.0) {
1090        TrendDirection::Improving
1091    } else {
1092        TrendDirection::Declining
1093    };
1094
1095    TrendMetric {
1096        name,
1097        label,
1098        previous,
1099        current,
1100        delta,
1101        direction,
1102        unit,
1103        previous_count,
1104        current_count,
1105    }
1106}
1107
1108/// Extension trait for rounding floats to N decimal places.
1109trait RoundTo {
1110    fn round_to(self, decimals: u32) -> Self;
1111}
1112
1113impl RoundTo for f64 {
1114    fn round_to(self, decimals: u32) -> Self {
1115        let factor = 10_f64.powi(decimals as i32);
1116        (self * factor).round() / factor
1117    }
1118}
1119
1120#[cfg(test)]
1121mod tests {
1122    use super::*;
1123
1124    fn make_module(id: u32, cyclomatic: u16) -> crate::source::ModuleInfo {
1125        crate::source::ModuleInfo {
1126            file_id: crate::discover::FileId(id),
1127            exports: Vec::new(),
1128            imports: Vec::new(),
1129            re_exports: Vec::new(),
1130            dynamic_imports: Vec::new(),
1131            dynamic_import_patterns: Vec::new(),
1132            require_calls: Vec::new(),
1133            package_path_references: Box::default(),
1134            member_accesses: Vec::new(),
1135            semantic_facts: Box::default(),
1136            whole_object_uses: Box::default(),
1137            has_cjs_exports: false,
1138            has_angular_component_template_url: false,
1139            content_hash: 0,
1140            suppressions: Vec::new(),
1141            unknown_suppression_kinds: Vec::new(),
1142            unused_import_bindings: Vec::new(),
1143            type_referenced_import_bindings: vec![],
1144            value_referenced_import_bindings: vec![],
1145            line_offsets: Vec::new(),
1146            flag_uses: Vec::new(),
1147            class_heritage: Vec::new(),
1148            exported_factory_returns: Box::default(),
1149            injection_tokens: Vec::new(),
1150            local_type_declarations: Vec::new(),
1151            public_signature_type_references: Vec::new(),
1152            namespace_object_aliases: Vec::new(),
1153            iconify_prefixes: Vec::new(),
1154            iconify_icon_names: Vec::new(),
1155            auto_import_candidates: Vec::new(),
1156            directives: Vec::new(),
1157            client_only_dynamic_import_spans: Vec::new(),
1158            security_sinks: Vec::new(),
1159            security_sinks_skipped: 0,
1160            security_unresolved_callee_sites: Vec::new(),
1161            tainted_bindings: Vec::new(),
1162            sanitized_sink_args: Vec::new(),
1163            security_control_sites: Vec::new(),
1164            callee_uses: Vec::new(),
1165            misplaced_directives: Vec::new(),
1166            inline_server_action_exports: Vec::new(),
1167            di_key_sites: Vec::new(),
1168            has_dynamic_provide: false,
1169            referenced_import_bindings: Vec::new(),
1170            component_props: Vec::new(),
1171            has_props_attrs_fallthrough: false,
1172            has_define_expose: false,
1173            has_define_model: false,
1174            has_unharvestable_props: false,
1175            component_emits: Vec::new(),
1176            angular_inputs: Vec::new(),
1177            angular_outputs: Vec::new(),
1178            has_unharvestable_emits: false,
1179            has_dynamic_emit: false,
1180            has_emit_whole_object_use: false,
1181            load_return_keys: Vec::new(),
1182            has_unharvestable_load: false,
1183            has_load_data_whole_use: false,
1184            has_page_data_store_whole_use: false,
1185            component_functions: Vec::new(),
1186            react_props: Vec::new(),
1187            hook_uses: Vec::new(),
1188            render_edges: Vec::new(),
1189            svelte_dispatched_events: Vec::new(),
1190            svelte_listened_events: Vec::new(),
1191            angular_component_selectors: Vec::new(),
1192            registered_custom_elements: Vec::new(),
1193            used_custom_element_tags: Vec::new(),
1194            angular_used_selectors: Vec::new(),
1195            angular_entry_component_refs: Vec::new(),
1196            has_dynamic_component_render: false,
1197            has_dynamic_dispatch: false,
1198            complexity: vec![fallow_types::extract::FunctionComplexity {
1199                name: format!("fn_{id}"),
1200                line: id + 1,
1201                col: 0,
1202                cyclomatic,
1203                cognitive: 0,
1204                line_count: 10,
1205                param_count: 0,
1206                react_hook_count: 0,
1207                react_jsx_max_depth: 0,
1208                react_prop_count: 0,
1209                source_hash: None,
1210                contributions: Vec::new(),
1211            }],
1212        }
1213    }
1214
1215    #[expect(
1216        clippy::cast_possible_truncation,
1217        reason = "test values are trivially small"
1218    )]
1219    fn make_modules() -> Vec<crate::source::ModuleInfo> {
1220        (0..10)
1221            .map(|i| make_module(i, (i as u16 + 1) * 2))
1222            .collect()
1223    }
1224
1225    fn assert_close(actual: f64, expected: f64) {
1226        assert!(
1227            (actual - expected).abs() < f64::EPSILON,
1228            "expected {expected}, got {actual}"
1229        );
1230    }
1231
1232    fn assert_some_close(actual: Option<f64>, expected: f64) {
1233        assert_close(actual.expect("expected metric to be present"), expected);
1234    }
1235
1236    #[test]
1237    fn compute_cyclomatic_stats() {
1238        let modules = make_modules();
1239        let input = VitalSignsInput {
1240            modules: &modules,
1241            module_filter: None,
1242            file_scores: None,
1243            hotspots: None,
1244            total_files: 10,
1245            analysis_counts: None,
1246        };
1247        let vs = compute_vital_signs(&input);
1248        assert!((vs.avg_cyclomatic - 11.0).abs() < f64::EPSILON);
1249        assert_eq!(vs.p90_cyclomatic, 18);
1250    }
1251
1252    #[test]
1253    fn compute_with_analysis_counts() {
1254        let modules = make_modules();
1255        let input = VitalSignsInput {
1256            modules: &modules,
1257            module_filter: None,
1258            file_scores: None,
1259            hotspots: None,
1260            total_files: 100,
1261            analysis_counts: Some(AnalysisCounts {
1262                total_exports: 500,
1263                dead_files: 5,
1264                dead_exports: 50,
1265                unused_deps: 3,
1266                circular_deps: 2,
1267                total_deps: 40,
1268            }),
1269        };
1270        let vs = compute_vital_signs(&input);
1271        assert_eq!(vs.dead_file_pct, Some(5.0)); // 5/100 * 100
1272        assert_eq!(vs.dead_export_pct, Some(10.0)); // 50/500 * 100
1273        assert_eq!(vs.unused_dep_count, Some(3));
1274        assert_eq!(vs.circular_dep_count, Some(2));
1275    }
1276
1277    #[test]
1278    fn compute_hotspot_count_with_threshold() {
1279        let hotspots = vec![
1280            HotspotEntry {
1281                path: PathBuf::from("a.ts"),
1282                score: 80.0,
1283                commits: 10,
1284                weighted_commits: 8.0,
1285                lines_added: 100,
1286                lines_deleted: 50,
1287                complexity_density: 0.5,
1288                fan_in: 5,
1289                trend: crate::churn::ChurnTrend::Stable,
1290                ownership: None,
1291                is_test_path: false,
1292            },
1293            HotspotEntry {
1294                path: PathBuf::from("b.ts"),
1295                score: 30.0, // Below threshold
1296                commits: 5,
1297                weighted_commits: 3.0,
1298                lines_added: 40,
1299                lines_deleted: 20,
1300                complexity_density: 0.2,
1301                fan_in: 2,
1302                trend: crate::churn::ChurnTrend::Cooling,
1303                ownership: None,
1304                is_test_path: false,
1305            },
1306            HotspotEntry {
1307                path: PathBuf::from("c.ts"),
1308                score: 50.0, // At threshold
1309                commits: 8,
1310                weighted_commits: 6.0,
1311                lines_added: 80,
1312                lines_deleted: 30,
1313                complexity_density: 0.4,
1314                fan_in: 3,
1315                trend: crate::churn::ChurnTrend::Accelerating,
1316                ownership: None,
1317                is_test_path: false,
1318            },
1319        ];
1320        let modules = Vec::new();
1321        let input = VitalSignsInput {
1322            modules: &modules,
1323            module_filter: None,
1324            file_scores: None,
1325            hotspots: Some(&hotspots),
1326            total_files: 10,
1327            analysis_counts: None,
1328        };
1329        let vs = compute_vital_signs(&input);
1330        assert_eq!(vs.hotspot_count, Some(2)); // 80.0 and 50.0 meet threshold
1331        assert_eq!(vs.hotspot_top_pct_count, Some(1)); // top 1% bucket rounds up to one file
1332    }
1333
1334    #[test]
1335    fn compute_without_hotspots_gives_none() {
1336        let modules = Vec::new();
1337        let input = VitalSignsInput {
1338            modules: &modules,
1339            module_filter: None,
1340            file_scores: None,
1341            hotspots: None,
1342            total_files: 0,
1343            analysis_counts: None,
1344        };
1345        let vs = compute_vital_signs(&input);
1346        assert!(vs.hotspot_count.is_none());
1347    }
1348
1349    #[test]
1350    fn snapshot_save_and_load() {
1351        let dir = tempfile::tempdir().unwrap();
1352        let root = dir.path();
1353        let vs = VitalSigns {
1354            dead_file_pct: Some(3.2),
1355            dead_export_pct: Some(8.1),
1356            avg_cyclomatic: 4.7,
1357            p90_cyclomatic: 12,
1358            hotspot_count: Some(5),
1359            maintainability_avg: Some(72.4),
1360            unused_dep_count: Some(4),
1361            circular_dep_count: Some(2),
1362            ..Default::default()
1363        };
1364        let counts = VitalSignsCounts {
1365            total_files: 1200,
1366            total_exports: 5400,
1367            dead_files: 38,
1368            dead_exports: 437,
1369            files_scored: Some(1150),
1370            total_deps: 42,
1371            ..Default::default()
1372        };
1373        let health_score = compute_health_score(&vs, 1200);
1374        let snapshot = build_snapshot(vs, counts, root, false, Some(&health_score), None);
1375        let saved_path = save_snapshot(&snapshot, root, None).unwrap();
1376
1377        assert!(saved_path.exists());
1378        assert!(saved_path.starts_with(root.join(".fallow/snapshots")));
1379
1380        let content = std::fs::read_to_string(&saved_path).unwrap();
1381        let loaded: VitalSignsSnapshot = serde_json::from_str(&content).unwrap();
1382        assert_eq!(loaded.snapshot_schema_version, SNAPSHOT_SCHEMA_VERSION);
1383        assert!((loaded.vital_signs.avg_cyclomatic - 4.7).abs() < f64::EPSILON);
1384        assert_eq!(loaded.counts.total_files, 1200);
1385        assert!(loaded.score.is_some());
1386        assert!(loaded.grade.is_some());
1387    }
1388
1389    #[test]
1390    fn snapshot_save_explicit_path() {
1391        let dir = tempfile::tempdir().unwrap();
1392        let root = dir.path();
1393        let explicit = root.join("my-snapshot.json");
1394        let vs = VitalSigns {
1395            avg_cyclomatic: 1.0,
1396            p90_cyclomatic: 2,
1397            ..Default::default()
1398        };
1399        let counts = VitalSignsCounts::default();
1400        let snapshot = build_snapshot(vs, counts, root, false, None, None);
1401        let saved = save_snapshot(&snapshot, root, Some(&explicit)).unwrap();
1402        assert_eq!(saved, explicit);
1403        assert!(explicit.exists());
1404    }
1405
1406    #[test]
1407    fn snapshot_save_creates_nested_dirs() {
1408        let dir = tempfile::tempdir().unwrap();
1409        let root = dir.path();
1410        let nested = root.join("a/b/c/snapshot.json");
1411        let vs = VitalSigns {
1412            avg_cyclomatic: 1.0,
1413            p90_cyclomatic: 2,
1414            ..Default::default()
1415        };
1416        let counts = VitalSignsCounts::default();
1417        let snapshot = build_snapshot(vs, counts, root, false, None, None);
1418        let saved = save_snapshot(&snapshot, root, Some(&nested)).unwrap();
1419        assert_eq!(saved, nested);
1420        assert!(nested.exists());
1421    }
1422
1423    #[test]
1424    fn days_to_ymd_epoch() {
1425        assert_eq!(days_to_ymd(0), (1970, 1, 1));
1426    }
1427
1428    #[test]
1429    fn days_to_ymd_known_date() {
1430        assert_eq!(days_to_ymd(20_537), (2026, 3, 25));
1431    }
1432
1433    #[test]
1434    fn health_score_perfect() {
1435        let vs = VitalSigns {
1436            dead_file_pct: Some(0.0),
1437            dead_export_pct: Some(0.0),
1438            avg_cyclomatic: 1.0,
1439            p90_cyclomatic: 2,
1440            hotspot_count: Some(0),
1441            maintainability_avg: Some(90.0),
1442            unused_dep_count: Some(0),
1443            circular_dep_count: Some(0),
1444            ..Default::default()
1445        };
1446        let score = compute_health_score(&vs, 100);
1447        assert!((score.score - 100.0).abs() < f64::EPSILON);
1448        assert_eq!(score.grade, "A");
1449    }
1450
1451    #[test]
1452    fn health_score_no_optional_metrics() {
1453        let vs = VitalSigns {
1454            avg_cyclomatic: 1.0,
1455            p90_cyclomatic: 2,
1456            ..Default::default()
1457        };
1458        let score = compute_health_score(&vs, 0);
1459        assert!((score.score - 100.0).abs() < f64::EPSILON);
1460        assert_eq!(score.grade, "A");
1461        assert!(score.penalties.dead_files.is_none());
1462        assert!(score.penalties.unused_deps.is_none());
1463        assert!(score.penalties.duplication.is_none());
1464    }
1465
1466    #[test]
1467    fn health_score_dead_code_penalty() {
1468        let vs = VitalSigns {
1469            dead_file_pct: Some(50.0),
1470            dead_export_pct: Some(30.0),
1471            avg_cyclomatic: 1.0,
1472            p90_cyclomatic: 2,
1473            ..Default::default()
1474        };
1475        let score = compute_health_score(&vs, 100);
1476        assert!((score.score - 84.0).abs() < 0.1);
1477        assert_eq!(score.grade, "B");
1478    }
1479
1480    #[test]
1481    fn health_score_complexity_penalty() {
1482        let vs = VitalSigns {
1483            avg_cyclomatic: 5.5,
1484            p90_cyclomatic: 15,
1485            ..Default::default()
1486        };
1487        let score = compute_health_score(&vs, 100);
1488        assert!((score.score - 75.0).abs() < 0.1);
1489        assert_eq!(score.grade, "B");
1490    }
1491
1492    #[test]
1493    fn health_score_prop_drilling_penalty_opt_in() {
1494        let base = || VitalSigns {
1495            avg_cyclomatic: 1.0,
1496            p90_cyclomatic: 2,
1497            ..Default::default()
1498        };
1499        let base_score = compute_health_score(&base(), 100).score;
1500
1501        // Each located chain costs 1pt (depth is descriptive, not a multiplier).
1502        let three = VitalSigns {
1503            prop_drilling_chain_count: Some(3),
1504            prop_drilling_max_depth: Some(5),
1505            ..base()
1506        };
1507        assert!((base_score - compute_health_score(&three, 100).score - 3.0).abs() < 0.1);
1508
1509        // Capped at 5pt regardless of chain count.
1510        let many = VitalSigns {
1511            prop_drilling_chain_count: Some(20),
1512            ..base()
1513        };
1514        assert!((base_score - compute_health_score(&many, 100).score - 5.0).abs() < 0.1);
1515
1516        // Dormant by default: the opt-in rule is off, so the count is `None` and
1517        // the score is unchanged.
1518        let off = VitalSigns {
1519            prop_drilling_chain_count: None,
1520            ..base()
1521        };
1522        assert!((base_score - compute_health_score(&off, 100).score).abs() < f64::EPSILON);
1523    }
1524
1525    #[test]
1526    fn health_score_clamped_at_zero() {
1527        let vs = VitalSigns {
1528            dead_file_pct: Some(100.0),
1529            dead_export_pct: Some(100.0),
1530            avg_cyclomatic: 10.0,
1531            p90_cyclomatic: 30,
1532            hotspot_count: Some(50),
1533            maintainability_avg: Some(20.0),
1534            unused_dep_count: Some(100),
1535            circular_dep_count: Some(50),
1536            ..Default::default()
1537        };
1538        let score = compute_health_score(&vs, 100);
1539        assert!((score.score).abs() < f64::EPSILON);
1540        assert_eq!(score.grade, "F");
1541    }
1542
1543    #[test]
1544    fn health_score_hotspot_normalized_by_files() {
1545        let vs = VitalSigns {
1546            avg_cyclomatic: 1.0,
1547            p90_cyclomatic: 2,
1548            hotspot_count: Some(5),
1549            ..Default::default()
1550        };
1551        let score_100 = compute_health_score(&vs, 100);
1552        let score_1000 = compute_health_score(&vs, 1000);
1553        assert!(score_1000.score > score_100.score);
1554    }
1555
1556    #[test]
1557    fn health_score_hotspot_top_pct_can_use_full_budget() {
1558        let vs = VitalSigns {
1559            avg_cyclomatic: 1.0,
1560            p90_cyclomatic: 2,
1561            hotspot_count: Some(0),
1562            hotspot_top_pct_count: Some(250),
1563            ..Default::default()
1564        };
1565
1566        let score = compute_health_score(&vs, 25_000);
1567
1568        assert_some_close(score.penalties.hotspots, 10.0);
1569        assert_close(score.score, 90.0);
1570    }
1571
1572    #[test]
1573    fn health_score_duplication_penalty() {
1574        let vs = VitalSigns {
1575            dead_file_pct: None,
1576            dead_export_pct: None,
1577            avg_cyclomatic: 1.0,
1578            critical_complexity_pct: None,
1579            p90_cyclomatic: 2,
1580            duplication_pct: Some(10.0), // 10% - 5% = 5 points
1581            hotspot_count: None,
1582            hotspot_top_pct_count: None,
1583            maintainability_avg: None,
1584            maintainability_low_pct: None,
1585            unused_dep_count: None,
1586            unused_deps_per_k_files: None,
1587            circular_dep_count: None,
1588            circular_deps_per_k_files: None,
1589            counts: None,
1590            unit_size_profile: None,
1591            functions_over_60_loc_per_k: None,
1592            unit_interfacing_profile: None,
1593            p95_fan_in: None,
1594            coupling_high_pct: None,
1595            prop_drilling_chain_count: None,
1596            prop_drilling_max_depth: None,
1597            p95_render_fan_in: None,
1598            render_fan_in_high_pct: None,
1599            max_render_fan_in: None,
1600            top_render_fan_in: Vec::new(),
1601            total_loc: 0,
1602        };
1603        let score = compute_health_score(&vs, 100);
1604        assert_eq!(score.penalties.duplication, Some(5.0));
1605
1606        let vs_low = VitalSigns {
1607            duplication_pct: Some(4.0),
1608            ..vs.clone()
1609        };
1610        let score_low = compute_health_score(&vs_low, 100);
1611        assert_eq!(score_low.penalties.duplication, Some(0.0));
1612
1613        let vs_high = VitalSigns {
1614            duplication_pct: Some(20.0),
1615            ..vs
1616        };
1617        let score_high = compute_health_score(&vs_high, 100);
1618        assert_eq!(score_high.penalties.duplication, Some(10.0));
1619    }
1620
1621    #[test]
1622    fn health_score_uses_scale_invariant_monorepo_signals() {
1623        let vs = VitalSigns {
1624            dead_file_pct: Some(4.0),
1625            dead_export_pct: Some(9.0),
1626            avg_cyclomatic: 2.3,
1627            critical_complexity_pct: Some(2.3),
1628            p90_cyclomatic: 4,
1629            duplication_pct: Some(6.0),
1630            hotspot_count: Some(0),
1631            hotspot_top_pct_count: Some(250),
1632            maintainability_avg: Some(91.0),
1633            maintainability_low_pct: Some(8.0),
1634            unused_dep_count: Some(180),
1635            unused_deps_per_k_files: Some(7.2),
1636            circular_dep_count: Some(450),
1637            circular_deps_per_k_files: Some(18.0),
1638            unit_size_profile: Some(RiskProfile {
1639                low_risk: 80.0,
1640                medium_risk: 12.7,
1641                high_risk: 5.0,
1642                very_high_risk: 2.3,
1643            }),
1644            functions_over_60_loc_per_k: Some(23.0),
1645            p95_fan_in: Some(7),
1646            coupling_high_pct: Some(4.0),
1647            ..Default::default()
1648        };
1649        let score = compute_health_score(&vs, 25_000);
1650        let penalties = &score.penalties;
1651
1652        assert_some_close(penalties.dead_files, 0.8);
1653        assert_some_close(penalties.dead_exports, 1.8);
1654        assert_close(penalties.complexity, 9.2);
1655        assert!((penalties.p90_complexity).abs() < f64::EPSILON);
1656        assert_some_close(penalties.maintainability, 12.0);
1657        assert_some_close(penalties.hotspots, 10.0);
1658        assert_some_close(penalties.unused_deps, 3.6);
1659        assert_some_close(penalties.circular_deps, 9.0);
1660        assert_some_close(penalties.unit_size, 10.0);
1661        assert_some_close(penalties.coupling, 2.0);
1662        assert_some_close(penalties.duplication, 1.0);
1663        assert_close(score.score, 40.6);
1664        assert_eq!(score.grade, "D");
1665    }
1666
1667    #[test]
1668    fn load_snapshots_empty_dir() {
1669        let dir = tempfile::tempdir().unwrap();
1670        let snaps = load_snapshots(dir.path());
1671        assert!(snaps.is_empty());
1672    }
1673
1674    #[test]
1675    fn load_snapshots_returns_sorted() {
1676        let dir = tempfile::tempdir().unwrap();
1677        let root = dir.path();
1678        let snap_dir = root.join(".fallow/snapshots");
1679        std::fs::create_dir_all(&snap_dir).unwrap();
1680
1681        let older = make_test_snapshot("2026-01-01T00:00:00Z", Some(72.0));
1682        let newer = make_test_snapshot("2026-03-01T00:00:00Z", Some(78.0));
1683
1684        std::fs::write(
1685            snap_dir.join("2026-03-01T00-00-00Z.json"),
1686            serde_json::to_string(&newer).unwrap(),
1687        )
1688        .unwrap();
1689        std::fs::write(
1690            snap_dir.join("2026-01-01T00-00-00Z.json"),
1691            serde_json::to_string(&older).unwrap(),
1692        )
1693        .unwrap();
1694
1695        let loaded = load_snapshots(root);
1696        assert_eq!(loaded.len(), 2);
1697        assert_eq!(loaded[0].timestamp, "2026-01-01T00:00:00Z");
1698        assert_eq!(loaded[1].timestamp, "2026-03-01T00:00:00Z");
1699    }
1700
1701    #[test]
1702    fn load_snapshots_skips_corrupt_files() {
1703        let dir = tempfile::tempdir().unwrap();
1704        let root = dir.path();
1705        let snap_dir = root.join(".fallow/snapshots");
1706        std::fs::create_dir_all(&snap_dir).unwrap();
1707
1708        std::fs::write(snap_dir.join("corrupt.json"), "not valid json").unwrap();
1709        let good = make_test_snapshot("2026-02-01T00:00:00Z", Some(80.0));
1710        std::fs::write(
1711            snap_dir.join("good.json"),
1712            serde_json::to_string(&good).unwrap(),
1713        )
1714        .unwrap();
1715
1716        let loaded = load_snapshots(root);
1717        assert_eq!(loaded.len(), 1);
1718        assert_eq!(loaded[0].timestamp, "2026-02-01T00:00:00Z");
1719    }
1720
1721    #[test]
1722    fn load_snapshots_ignores_non_json() {
1723        let dir = tempfile::tempdir().unwrap();
1724        let root = dir.path();
1725        let snap_dir = root.join(".fallow/snapshots");
1726        std::fs::create_dir_all(&snap_dir).unwrap();
1727
1728        std::fs::write(snap_dir.join("readme.txt"), "not a snapshot").unwrap();
1729
1730        let loaded = load_snapshots(root);
1731        assert!(loaded.is_empty());
1732    }
1733
1734    #[test]
1735    fn compute_trend_no_snapshots() {
1736        let vs = make_test_vital_signs();
1737        let counts = make_test_counts();
1738        assert!(compute_trend(&vs, &counts, Some(78.0), &[]).is_none());
1739    }
1740
1741    #[test]
1742    fn compute_trend_improving() {
1743        let prev = make_test_snapshot("2026-01-01T00:00:00Z", Some(72.0));
1744        let vs = VitalSigns {
1745            dead_file_pct: Some(2.8),
1746            dead_export_pct: Some(7.5),
1747            avg_cyclomatic: 4.1,
1748            p90_cyclomatic: 12,
1749            hotspot_count: Some(3),
1750            maintainability_avg: Some(75.0),
1751            unused_dep_count: Some(3),
1752            circular_dep_count: Some(1),
1753            ..Default::default()
1754        };
1755        let counts = VitalSignsCounts {
1756            total_files: 100,
1757            total_exports: 500,
1758            dead_files: 3,
1759            dead_exports: 38,
1760            files_scored: Some(95),
1761            total_deps: 40,
1762            ..Default::default()
1763        };
1764
1765        let trend = compute_trend(&vs, &counts, Some(78.0), &[prev]).unwrap();
1766        assert_eq!(trend.compared_to.timestamp, "2026-01-01T00:00:00Z");
1767        assert_eq!(trend.snapshots_loaded, 1);
1768        assert_eq!(trend.overall_direction, TrendDirection::Improving);
1769
1770        let score_metric = trend.metrics.iter().find(|m| m.name == "score").unwrap();
1771        assert_eq!(score_metric.direction, TrendDirection::Improving);
1772        assert!((score_metric.delta - 6.0).abs() < f64::EPSILON);
1773    }
1774
1775    #[test]
1776    fn compute_trend_stable_within_tolerance() {
1777        let prev = make_test_snapshot("2026-01-01T00:00:00Z", Some(78.0));
1778        let vs = make_test_vital_signs();
1779        let counts = make_test_counts();
1780
1781        let trend = compute_trend(&vs, &counts, Some(78.3), &[prev]).unwrap();
1782        let score_metric = trend.metrics.iter().find(|m| m.name == "score").unwrap();
1783        assert_eq!(score_metric.direction, TrendDirection::Stable);
1784    }
1785
1786    #[test]
1787    fn compute_trend_uses_most_recent_snapshot() {
1788        let older = make_test_snapshot("2026-01-01T00:00:00Z", Some(60.0));
1789        let newer = make_test_snapshot("2026-03-01T00:00:00Z", Some(72.0));
1790        let vs = make_test_vital_signs();
1791        let counts = make_test_counts();
1792
1793        let trend = compute_trend(&vs, &counts, Some(78.0), &[older, newer]).unwrap();
1794        assert_eq!(trend.compared_to.score, Some(72.0));
1795        assert_eq!(trend.snapshots_loaded, 2);
1796    }
1797
1798    #[test]
1799    fn compute_trend_includes_raw_counts() {
1800        let prev = make_test_snapshot("2026-01-01T00:00:00Z", Some(72.0));
1801        let vs = make_test_vital_signs();
1802        let counts = make_test_counts();
1803
1804        let trend = compute_trend(&vs, &counts, Some(78.0), &[prev]).unwrap();
1805        let dead_files = trend
1806            .metrics
1807            .iter()
1808            .find(|m| m.name == "dead_file_pct")
1809            .unwrap();
1810        assert!(dead_files.previous_count.is_some());
1811        assert!(dead_files.current_count.is_some());
1812    }
1813
1814    fn make_test_vital_signs() -> VitalSigns {
1815        VitalSigns {
1816            dead_file_pct: Some(3.2),
1817            dead_export_pct: Some(8.1),
1818            avg_cyclomatic: 4.2,
1819            p90_cyclomatic: 12,
1820            hotspot_count: Some(5),
1821            maintainability_avg: Some(72.4),
1822            unused_dep_count: Some(4),
1823            circular_dep_count: Some(2),
1824            ..Default::default()
1825        }
1826    }
1827
1828    fn make_test_counts() -> VitalSignsCounts {
1829        VitalSignsCounts {
1830            total_files: 100,
1831            total_exports: 500,
1832            dead_files: 3,
1833            dead_exports: 40,
1834            files_scored: Some(95),
1835            total_deps: 42,
1836            ..Default::default()
1837        }
1838    }
1839
1840    fn make_test_snapshot(timestamp: &str, score: Option<f64>) -> VitalSignsSnapshot {
1841        VitalSignsSnapshot {
1842            snapshot_schema_version: SNAPSHOT_SCHEMA_VERSION,
1843            version: "2.5.5".into(),
1844            timestamp: timestamp.into(),
1845            git_sha: Some("abc1234".into()),
1846            git_branch: Some("main".into()),
1847            shallow_clone: false,
1848            vital_signs: VitalSigns {
1849                dead_file_pct: Some(3.2),
1850                dead_export_pct: Some(8.1),
1851                avg_cyclomatic: 4.7,
1852                p90_cyclomatic: 12,
1853                hotspot_count: Some(5),
1854                maintainability_avg: Some(72.4),
1855                unused_dep_count: Some(4),
1856                circular_dep_count: Some(2),
1857                ..Default::default()
1858            },
1859            counts: VitalSignsCounts {
1860                total_files: 100,
1861                total_exports: 500,
1862                dead_files: 3,
1863                dead_exports: 40,
1864                files_scored: Some(95),
1865                total_deps: 42,
1866                ..Default::default()
1867            },
1868            score,
1869            grade: score.map(|s| letter_grade(s).to_string()),
1870            coverage_model: None,
1871        }
1872    }
1873}