1use std::path::Path;
2use std::process::ExitCode;
3
4use fallow_config::OutputFormat;
5use fallow_core::git_env::clear_ambient_git_env;
6use fallow_core::results::AnalysisResults;
7
8use super::counts::{CheckCounts, DupesCounts, REGRESSION_SCHEMA_VERSION, RegressionBaseline};
9use super::outcome::RegressionOutcome;
10use super::tolerance::Tolerance;
11
12use crate::error::emit_error;
13
14const SECS_PER_DAY: u64 = 86_400;
16
17#[derive(Clone, Copy)]
19pub enum SaveRegressionTarget<'a> {
20 None,
22 Config,
24 File(&'a Path),
26}
27
28#[derive(Clone, Copy)]
30pub struct RegressionOpts<'a> {
31 pub fail_on_regression: bool,
32 pub tolerance: Tolerance,
33 pub regression_baseline_file: Option<&'a Path>,
35 pub save_target: SaveRegressionTarget<'a>,
37 pub scoped: bool,
39 pub quiet: bool,
40 pub output: OutputFormat,
43}
44
45fn is_likely_gitignored(path: &Path, root: &Path) -> bool {
48 let mut command = std::process::Command::new("git");
49 command
50 .args(["check-ignore", "-q"])
51 .arg(path)
52 .current_dir(root);
53 clear_ambient_git_env(&mut command);
54 command.output().ok().is_some_and(|o| o.status.success())
55}
56
57fn current_git_sha(root: &Path) -> Option<String> {
59 let mut command = std::process::Command::new("git");
60 command.args(["rev-parse", "HEAD"]).current_dir(root);
61 clear_ambient_git_env(&mut command);
62 command
63 .output()
64 .ok()
65 .filter(|o| o.status.success())
66 .map(|o| String::from_utf8_lossy(&o.stdout).trim().to_string())
67}
68
69pub fn save_regression_baseline(
75 path: &Path,
76 root: &Path,
77 check_counts: Option<&CheckCounts>,
78 dupes_counts: Option<&DupesCounts>,
79 output: OutputFormat,
80) -> Result<(), ExitCode> {
81 let baseline = RegressionBaseline {
82 schema_version: REGRESSION_SCHEMA_VERSION,
83 fallow_version: env!("CARGO_PKG_VERSION").to_string(),
84 timestamp: chrono_now(),
85 git_sha: current_git_sha(root),
86 check: check_counts.cloned(),
87 dupes: dupes_counts.cloned(),
88 };
89 let json = serde_json::to_string_pretty(&baseline).map_err(|e| {
90 emit_error(
91 &format!("failed to serialize regression baseline: {e}"),
92 2,
93 output,
94 )
95 })?;
96 if let Some(parent) = path.parent() {
97 let _ = std::fs::create_dir_all(parent);
98 }
99 std::fs::write(path, json).map_err(|e| {
100 emit_error(
101 &format!("failed to save regression baseline: {e}"),
102 2,
103 output,
104 )
105 })?;
106 eprintln!("Regression baseline saved to {}", path.display());
107 if is_likely_gitignored(path, root) {
108 eprintln!(
109 "Warning: '{}' may be gitignored. Commit this file so CI can compare against it.",
110 path.display()
111 );
112 }
113 Ok(())
114}
115
116pub fn save_baseline_to_config(
126 config_path: &Path,
127 counts: &CheckCounts,
128 output: OutputFormat,
129) -> Result<(), ExitCode> {
130 let content = match std::fs::read_to_string(config_path) {
131 Ok(c) => c,
132 Err(e) if e.kind() == std::io::ErrorKind::NotFound => {
133 let is_toml = config_path.extension().is_some_and(|ext| ext == "toml");
134 if is_toml {
135 String::new()
136 } else {
137 "{}".to_string()
138 }
139 }
140 Err(e) => {
141 return Err(emit_error(
142 &format!(
143 "failed to read config file '{}': {e}",
144 config_path.display()
145 ),
146 2,
147 output,
148 ));
149 }
150 };
151
152 let baseline = counts.to_config_baseline();
153 let is_toml = config_path.extension().is_some_and(|ext| ext == "toml");
154
155 let updated = if is_toml {
156 Ok(update_toml_regression(&content, &baseline))
157 } else {
158 update_json_regression(&content, &baseline)
159 }
160 .map_err(|e| {
161 emit_error(
162 &format!(
163 "failed to update config file '{}': {e}",
164 config_path.display()
165 ),
166 2,
167 output,
168 )
169 })?;
170
171 std::fs::write(config_path, updated).map_err(|e| {
172 emit_error(
173 &format!(
174 "failed to write config file '{}': {e}",
175 config_path.display()
176 ),
177 2,
178 output,
179 )
180 })?;
181
182 eprintln!(
183 "Regression baseline saved to {} (regression.baseline section)",
184 config_path.display()
185 );
186 Ok(())
187}
188
189fn find_json_key(content: &str, key: &str) -> Option<usize> {
193 let needle = format!("\"{key}\"");
194 let mut search_from = 0;
195 while let Some(pos) = content[search_from..].find(&needle) {
196 let abs_pos = search_from + pos;
197 let line_start = content[..abs_pos].rfind('\n').map_or(0, |i| i + 1);
198 let line_prefix = content[line_start..abs_pos].trim_start();
199 if line_prefix.starts_with("//") {
200 search_from = abs_pos + needle.len();
201 continue;
202 }
203 let before = &content[..abs_pos];
204 let last_open = before.rfind("/*");
205 let last_close = before.rfind("*/");
206 if let Some(open_pos) = last_open
207 && last_close.is_none_or(|close_pos| close_pos < open_pos)
208 {
209 search_from = abs_pos + needle.len();
210 continue;
211 }
212 return Some(abs_pos);
213 }
214 None
215}
216
217fn update_json_regression(
218 content: &str,
219 baseline: &fallow_config::RegressionBaseline,
220) -> Result<String, String> {
221 let baseline_json =
222 serde_json::to_string_pretty(baseline).map_err(|e| format!("serialization error: {e}"))?;
223
224 let indented: String = baseline_json
225 .lines()
226 .enumerate()
227 .map(|(i, line)| {
228 if i == 0 {
229 format!(" {line}")
230 } else {
231 format!("\n {line}")
232 }
233 })
234 .collect();
235
236 let regression_block = format!(" \"regression\": {{\n \"baseline\": {indented}\n }}");
237
238 if let Some(start) = find_json_key(content, "regression") {
239 let after_key = &content[start..];
240 if let Some(brace_start) = after_key.find('{') {
241 let abs_brace = start + brace_start;
242 let mut depth = 0;
243 let mut end = abs_brace;
244 let mut found_close = false;
245 for (i, ch) in content[abs_brace..].char_indices() {
246 match ch {
247 '{' => depth += 1,
248 '}' => {
249 depth -= 1;
250 if depth == 0 {
251 end = abs_brace + i + 1;
252 found_close = true;
253 break;
254 }
255 }
256 _ => {}
257 }
258 }
259 if !found_close {
260 return Err("malformed JSON: unmatched brace in regression object".to_string());
261 }
262 let mut result = String::new();
263 result.push_str(&content[..start]);
264 result.push_str(®ression_block[2..]); result.push_str(&content[end..]);
266 return Ok(result);
267 }
268 }
269
270 if let Some(last_brace) = content.rfind('}') {
271 let before_brace = content[..last_brace].trim_end();
272 let needs_comma = !before_brace.ends_with('{') && !before_brace.ends_with(',');
273
274 let mut result = String::new();
275 result.push_str(before_brace);
276 if needs_comma {
277 result.push(',');
278 }
279 result.push('\n');
280 result.push_str(®ression_block);
281 result.push('\n');
282 result.push_str(&content[last_brace..]);
283 Ok(result)
284 } else {
285 Err("config file has no closing brace".to_string())
286 }
287}
288
289fn update_toml_regression(content: &str, baseline: &fallow_config::RegressionBaseline) -> String {
291 use std::fmt::Write;
292 let mut section = String::from("[regression.baseline]\n");
293 let _ = writeln!(section, "totalIssues = {}", baseline.total_issues);
294 let _ = writeln!(section, "unusedFiles = {}", baseline.unused_files);
295 let _ = writeln!(section, "unusedExports = {}", baseline.unused_exports);
296 let _ = writeln!(section, "unusedTypes = {}", baseline.unused_types);
297 let _ = writeln!(
298 section,
299 "unusedDependencies = {}",
300 baseline.unused_dependencies
301 );
302 let _ = writeln!(
303 section,
304 "unusedDevDependencies = {}",
305 baseline.unused_dev_dependencies
306 );
307 let _ = writeln!(
308 section,
309 "unusedOptionalDependencies = {}",
310 baseline.unused_optional_dependencies
311 );
312 let _ = writeln!(
313 section,
314 "unusedEnumMembers = {}",
315 baseline.unused_enum_members
316 );
317 let _ = writeln!(
318 section,
319 "unusedClassMembers = {}",
320 baseline.unused_class_members
321 );
322 let _ = writeln!(
323 section,
324 "unresolvedImports = {}",
325 baseline.unresolved_imports
326 );
327 let _ = writeln!(
328 section,
329 "unlistedDependencies = {}",
330 baseline.unlisted_dependencies
331 );
332 let _ = writeln!(section, "duplicateExports = {}", baseline.duplicate_exports);
333 let _ = writeln!(
334 section,
335 "circularDependencies = {}",
336 baseline.circular_dependencies
337 );
338 let _ = writeln!(
339 section,
340 "typeOnlyDependencies = {}",
341 baseline.type_only_dependencies
342 );
343 let _ = writeln!(
344 section,
345 "testOnlyDependencies = {}",
346 baseline.test_only_dependencies
347 );
348
349 if let Some(start) = content.find("[regression.baseline]") {
350 let after = &content[start + "[regression.baseline]".len()..];
351 let end_offset = after.find("\n[").map_or(content.len(), |i| {
352 start + "[regression.baseline]".len() + i + 1
353 });
354
355 let mut result = String::new();
356 result.push_str(&content[..start]);
357 result.push_str(§ion);
358 if end_offset < content.len() {
359 result.push_str(&content[end_offset..]);
360 }
361 result
362 } else {
363 let mut result = content.to_string();
364 if !result.ends_with('\n') {
365 result.push('\n');
366 }
367 result.push('\n');
368 result.push_str(§ion);
369 result
370 }
371}
372
373fn format_schema_mismatch_error(
376 path: &Path,
377 expected: u32,
378 actual: u32,
379 writer_version: &str,
380) -> String {
381 let path_display = path.display();
382 if actual == 0 {
383 format!(
384 "regression baseline '{path_display}' appears to predate schema versioning \
385 (schema_version is 0; this fallow build expects {expected}).\n\
386 The baseline was written by fallow {writer_version}.\n\
387 Regenerate it by running: fallow dead-code --save-regression-baseline {path_display}"
388 )
389 } else {
390 format!(
391 "regression baseline '{path_display}' has schema_version {actual} but this fallow build expects {expected}.\n\
392 The baseline was written by fallow {writer_version}.\n\
393 Regenerate it by running: fallow dead-code --save-regression-baseline {path_display}"
394 )
395 }
396}
397
398fn format_missing_schema_version_error(path: &Path) -> String {
402 let path_display = path.display();
403 let expected = REGRESSION_SCHEMA_VERSION;
404 format!(
405 "regression baseline '{path_display}' is missing the schema_version field; \
406 this fallow build expects schema_version {expected}.\n\
407 The baseline likely predates schema versioning or was hand-edited.\n\
408 Regenerate it by running: fallow dead-code --save-regression-baseline {path_display}"
409 )
410}
411
412pub fn load_regression_baseline(
425 path: &Path,
426 output: OutputFormat,
427) -> Result<RegressionBaseline, ExitCode> {
428 let content = std::fs::read_to_string(path).map_err(|e| {
429 if e.kind() == std::io::ErrorKind::NotFound {
430 emit_error(
431 &format!(
432 "no regression baseline found at '{}'.\n\
433 Run with --save-regression-baseline on your main branch to create one.",
434 path.display()
435 ),
436 2,
437 output,
438 )
439 } else {
440 emit_error(
441 &format!(
442 "failed to read regression baseline '{}': {e}",
443 path.display()
444 ),
445 2,
446 output,
447 )
448 }
449 })?;
450 let baseline: RegressionBaseline = serde_json::from_str(&content).map_err(|e| {
451 let message = if e.to_string().contains("missing field `schema_version`") {
452 format_missing_schema_version_error(path)
453 } else {
454 format!(
455 "failed to parse regression baseline '{}': {e}",
456 path.display()
457 )
458 };
459 emit_error(&message, 2, output)
460 })?;
461 if baseline.schema_version != REGRESSION_SCHEMA_VERSION {
462 let message = format_schema_mismatch_error(
463 path,
464 REGRESSION_SCHEMA_VERSION,
465 baseline.schema_version,
466 &baseline.fallow_version,
467 );
468 return Err(emit_error(&message, 2, output));
469 }
470 Ok(baseline)
471}
472
473pub fn compare_check_regression(
485 results: &AnalysisResults,
486 opts: &RegressionOpts<'_>,
487 config_baseline: Option<&fallow_config::RegressionBaseline>,
488) -> Result<Option<RegressionOutcome>, ExitCode> {
489 if !opts.fail_on_regression {
490 return Ok(None);
491 }
492
493 if opts.scoped {
494 let reason = "--changed-since or --workspace is active; regression check skipped \
495 (counts not comparable to full-project baseline)";
496 if !opts.quiet {
497 eprintln!("Warning: {reason}");
498 }
499 return Ok(Some(RegressionOutcome::Skipped { reason }));
500 }
501
502 let baseline_counts: CheckCounts = if let Some(baseline_path) = opts.regression_baseline_file {
503 let baseline = load_regression_baseline(baseline_path, opts.output)?;
504 let Some(counts) = baseline.check else {
505 return Err(emit_error(
506 &format!(
507 "regression baseline '{}' has no check data",
508 baseline_path.display()
509 ),
510 2,
511 opts.output,
512 ));
513 };
514 counts
515 } else if let Some(config_baseline) = config_baseline {
516 CheckCounts::from_config_baseline(config_baseline)
517 } else {
518 return Err(emit_error(
519 "no regression baseline found.\n\
520 Either add a `regression.baseline` section to your config file\n\
521 (run with --save-regression-baseline to generate it),\n\
522 or provide an explicit file via --regression-baseline <PATH>.",
523 2,
524 opts.output,
525 ));
526 };
527
528 let current_total = results.total_issues();
529 let baseline_total = baseline_counts.total_issues;
530
531 if opts.tolerance.exceeded(baseline_total, current_total) {
532 let current_counts = CheckCounts::from_results(results);
533 let type_deltas = baseline_counts.deltas(¤t_counts);
534 Ok(Some(RegressionOutcome::Exceeded {
535 baseline_total,
536 current_total,
537 tolerance: opts.tolerance,
538 type_deltas,
539 }))
540 } else {
541 Ok(Some(RegressionOutcome::Pass {
542 baseline_total,
543 current_total,
544 }))
545 }
546}
547
548fn chrono_now() -> String {
550 let duration = std::time::SystemTime::now()
551 .duration_since(std::time::UNIX_EPOCH)
552 .unwrap_or_default();
553 let secs = duration.as_secs();
554 let days = secs / SECS_PER_DAY;
555 let time_secs = secs % SECS_PER_DAY;
556 let hours = time_secs / 3600;
557 let minutes = (time_secs % 3600) / 60;
558 let seconds = time_secs % 60;
559 let z = days + 719_468;
560 let era = z / 146_097;
561 let doe = z - era * 146_097;
562 let yoe = (doe - doe / 1460 + doe / 36524 - doe / 146_096) / 365;
563 let y = yoe + era * 400;
564 let doy = doe - (365 * yoe + yoe / 4 - yoe / 100);
565 let mp = (5 * doy + 2) / 153;
566 let d = doy - (153 * mp + 2) / 5 + 1;
567 let m = if mp < 10 { mp + 3 } else { mp - 9 };
568 let y = if m <= 2 { y + 1 } else { y };
569 format!("{y:04}-{m:02}-{d:02}T{hours:02}:{minutes:02}:{seconds:02}Z")
570}
571
572#[cfg(test)]
573mod tests {
574 use super::*;
575 use fallow_core::results::*;
576 use std::path::PathBuf;
577
578 fn sample_baseline() -> fallow_config::RegressionBaseline {
579 fallow_config::RegressionBaseline {
580 total_issues: 5,
581 unused_files: 2,
582 ..Default::default()
583 }
584 }
585
586 #[test]
587 fn json_insert_into_empty_object() {
588 let result = update_json_regression("{}", &sample_baseline()).unwrap();
589 assert!(result.contains("\"regression\""));
590 assert!(result.contains("\"totalIssues\": 5"));
591 serde_json::from_str::<serde_json::Value>(&result).unwrap();
592 }
593
594 #[test]
595 fn json_insert_into_existing_config() {
596 let config = r#"{
597 "entry": ["src/main.ts"],
598 "production": true
599}"#;
600 let result = update_json_regression(config, &sample_baseline()).unwrap();
601 assert!(result.contains("\"regression\""));
602 assert!(result.contains("\"entry\""));
603 serde_json::from_str::<serde_json::Value>(&result).unwrap();
604 }
605
606 #[test]
607 fn json_replace_existing_regression() {
608 let config = r#"{
609 "entry": ["src/main.ts"],
610 "regression": {
611 "baseline": {
612 "totalIssues": 99
613 }
614 }
615}"#;
616 let result = update_json_regression(config, &sample_baseline()).unwrap();
617 assert!(!result.contains("99"));
618 assert!(result.contains("\"totalIssues\": 5"));
619 serde_json::from_str::<serde_json::Value>(&result).unwrap();
620 }
621
622 #[test]
623 fn json_skips_regression_in_comment() {
624 let config = "{\n // See \"regression\" docs\n \"entry\": []\n}";
625 let result = update_json_regression(config, &sample_baseline()).unwrap();
626 assert!(result.contains("\"regression\":"));
627 assert!(result.contains("\"entry\""));
628 }
629
630 #[test]
631 fn json_malformed_brace_returns_error() {
632 let config = r#"{ "regression": { "baseline": { "totalIssues": 1 }"#;
633 let result = update_json_regression(config, &sample_baseline());
634 assert!(result.is_err());
635 }
636
637 #[test]
638 fn toml_insert_into_empty() {
639 let result = update_toml_regression("", &sample_baseline());
640 assert!(result.contains("[regression.baseline]"));
641 assert!(result.contains("totalIssues = 5"));
642 }
643
644 #[test]
645 fn toml_insert_after_existing_content() {
646 let config = "[rules]\nunused-files = \"warn\"\n";
647 let result = update_toml_regression(config, &sample_baseline());
648 assert!(result.contains("[rules]"));
649 assert!(result.contains("[regression.baseline]"));
650 assert!(result.contains("totalIssues = 5"));
651 }
652
653 #[test]
654 fn toml_replace_existing_section() {
655 let config =
656 "[regression.baseline]\ntotalIssues = 99\n\n[rules]\nunused-files = \"warn\"\n";
657 let result = update_toml_regression(config, &sample_baseline());
658 assert!(!result.contains("99"));
659 assert!(result.contains("totalIssues = 5"));
660 assert!(result.contains("[rules]"));
661 }
662
663 #[test]
664 fn find_json_key_basic() {
665 assert_eq!(find_json_key(r#"{"foo": 1}"#, "foo"), Some(1));
666 }
667
668 #[test]
669 fn find_json_key_skips_comment() {
670 let content = "{\n // \"foo\" is important\n \"bar\": 1\n}";
671 assert_eq!(find_json_key(content, "foo"), None);
672 assert!(find_json_key(content, "bar").is_some());
673 }
674
675 #[test]
676 fn find_json_key_not_found() {
677 assert_eq!(find_json_key("{}", "missing"), None);
678 }
679
680 #[test]
681 fn find_json_key_skips_block_comment() {
682 let content = "{\n /* \"foo\": old value */\n \"foo\": 1\n}";
683 let pos = find_json_key(content, "foo").unwrap();
684 assert!(content[pos..].starts_with("\"foo\": 1"));
685 }
686
687 #[test]
688 fn chrono_now_format() {
689 let ts = chrono_now();
690 assert_eq!(ts.len(), 20);
691 assert!(ts.ends_with('Z'));
692 assert_eq!(&ts[4..5], "-");
693 assert_eq!(&ts[7..8], "-");
694 assert_eq!(&ts[10..11], "T");
695 assert_eq!(&ts[13..14], ":");
696 assert_eq!(&ts[16..17], ":");
697 }
698
699 #[test]
700 fn save_load_roundtrip() {
701 let dir = tempfile::tempdir().unwrap();
702 let path = dir.path().join("regression-baseline.json");
703 let counts = CheckCounts {
704 total_issues: 15,
705 unused_files: 3,
706 unused_exports: 5,
707 unused_types: 2,
708 unused_dependencies: 1,
709 unused_dev_dependencies: 1,
710 unused_optional_dependencies: 0,
711 unused_enum_members: 1,
712 unused_class_members: 0,
713 unresolved_imports: 1,
714 unlisted_dependencies: 0,
715 duplicate_exports: 1,
716 circular_dependencies: 0,
717 re_export_cycles: 0,
718 type_only_dependencies: 0,
719 test_only_dependencies: 0,
720 boundary_violations: 0,
721 boundary_coverage_violations: 0,
722 boundary_call_violations: 0,
723 policy_violations: 0,
724 };
725 let dupes = DupesCounts {
726 clone_groups: 4,
727 duplication_percentage: 2.5,
728 };
729
730 save_regression_baseline(
731 &path,
732 dir.path(),
733 Some(&counts),
734 Some(&dupes),
735 OutputFormat::Human,
736 )
737 .unwrap();
738 let loaded = load_regression_baseline(&path, OutputFormat::Human).unwrap();
739
740 assert_eq!(loaded.schema_version, REGRESSION_SCHEMA_VERSION);
741 let check = loaded.check.unwrap();
742 assert_eq!(check.total_issues, 15);
743 assert_eq!(check.unused_files, 3);
744 assert_eq!(check.unused_exports, 5);
745 assert_eq!(check.unused_types, 2);
746 assert_eq!(check.unused_dependencies, 1);
747 assert_eq!(check.unresolved_imports, 1);
748 assert_eq!(check.duplicate_exports, 1);
749 let dupes = loaded.dupes.unwrap();
750 assert_eq!(dupes.clone_groups, 4);
751 assert!((dupes.duplication_percentage - 2.5).abs() < f64::EPSILON);
752 }
753
754 #[test]
755 fn save_load_roundtrip_check_only() {
756 let dir = tempfile::tempdir().unwrap();
757 let path = dir.path().join("regression-baseline.json");
758 let counts = CheckCounts {
759 total_issues: 5,
760 unused_files: 5,
761 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
762 };
763
764 save_regression_baseline(&path, dir.path(), Some(&counts), None, OutputFormat::Human)
765 .unwrap();
766 let loaded = load_regression_baseline(&path, OutputFormat::Human).unwrap();
767
768 assert!(loaded.check.is_some());
769 assert!(loaded.dupes.is_none());
770 assert_eq!(loaded.check.unwrap().unused_files, 5);
771 }
772
773 #[test]
774 fn save_creates_parent_directories() {
775 let dir = tempfile::tempdir().unwrap();
776 let path = dir.path().join("nested").join("dir").join("baseline.json");
777 let counts = CheckCounts {
778 total_issues: 1,
779 unused_files: 1,
780 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
781 };
782
783 save_regression_baseline(&path, dir.path(), Some(&counts), None, OutputFormat::Human)
784 .unwrap();
785 assert!(path.exists());
786 }
787
788 #[test]
789 fn load_nonexistent_file_returns_error() {
790 let result = load_regression_baseline(
791 Path::new("/tmp/nonexistent-baseline-12345.json"),
792 OutputFormat::Human,
793 );
794 assert!(result.is_err());
795 }
796
797 #[test]
798 fn load_invalid_json_returns_error() {
799 let dir = tempfile::tempdir().unwrap();
800 let path = dir.path().join("bad.json");
801 std::fs::write(&path, "not valid json {{{").unwrap();
802 let result = load_regression_baseline(&path, OutputFormat::Human);
803 assert!(result.is_err());
804 }
805
806 #[test]
807 fn save_baseline_to_json_config() {
808 let dir = tempfile::tempdir().unwrap();
809 let config_path = dir.path().join(".fallowrc.json");
810 std::fs::write(&config_path, r#"{"entry": ["src/main.ts"]}"#).unwrap();
811
812 let counts = CheckCounts {
813 total_issues: 7,
814 unused_files: 3,
815 unused_exports: 4,
816 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
817 };
818 save_baseline_to_config(&config_path, &counts, OutputFormat::Human).unwrap();
819
820 let content = std::fs::read_to_string(&config_path).unwrap();
821 assert!(content.contains("\"regression\""));
822 assert!(content.contains("\"totalIssues\": 7"));
823 serde_json::from_str::<serde_json::Value>(&content).unwrap();
824 }
825
826 #[test]
827 fn save_baseline_to_toml_config() {
828 let dir = tempfile::tempdir().unwrap();
829 let config_path = dir.path().join("fallow.toml");
830 std::fs::write(&config_path, "[rules]\nunused-files = \"warn\"\n").unwrap();
831
832 let counts = CheckCounts {
833 total_issues: 7,
834 unused_files: 3,
835 unused_exports: 4,
836 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
837 };
838 save_baseline_to_config(&config_path, &counts, OutputFormat::Human).unwrap();
839
840 let content = std::fs::read_to_string(&config_path).unwrap();
841 assert!(content.contains("[regression.baseline]"));
842 assert!(content.contains("totalIssues = 7"));
843 assert!(content.contains("[rules]"));
844 }
845
846 #[test]
847 fn save_baseline_to_nonexistent_json_config() {
848 let dir = tempfile::tempdir().unwrap();
849 let config_path = dir.path().join(".fallowrc.json");
850
851 let counts = CheckCounts {
852 total_issues: 1,
853 unused_files: 1,
854 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
855 };
856 save_baseline_to_config(&config_path, &counts, OutputFormat::Human).unwrap();
857
858 let content = std::fs::read_to_string(&config_path).unwrap();
859 assert!(content.contains("\"regression\""));
860 serde_json::from_str::<serde_json::Value>(&content).unwrap();
861 }
862
863 #[test]
864 fn save_baseline_to_nonexistent_toml_config() {
865 let dir = tempfile::tempdir().unwrap();
866 let config_path = dir.path().join("fallow.toml");
867
868 let counts = CheckCounts {
869 total_issues: 0,
870 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
871 };
872 save_baseline_to_config(&config_path, &counts, OutputFormat::Human).unwrap();
873
874 let content = std::fs::read_to_string(&config_path).unwrap();
875 assert!(content.contains("[regression.baseline]"));
876 assert!(content.contains("totalIssues = 0"));
877 }
878
879 #[test]
880 fn json_insert_with_trailing_comma() {
881 let config = r#"{
882 "entry": ["src/main.ts"],
883}"#;
884 let result = update_json_regression(config, &sample_baseline()).unwrap();
885 assert!(result.contains("\"regression\""));
886 }
887
888 #[test]
889 fn json_no_closing_brace_returns_error() {
890 let result = update_json_regression("", &sample_baseline());
891 assert!(result.is_err());
892 }
893
894 #[test]
895 fn json_nested_regression_object_replaced_correctly() {
896 let config = r#"{
897 "regression": {
898 "baseline": {
899 "totalIssues": 99,
900 "unusedFiles": 10
901 },
902 "tolerance": "5%"
903 },
904 "entry": ["src/main.ts"]
905}"#;
906 let result = update_json_regression(config, &sample_baseline()).unwrap();
907 assert!(!result.contains("99"));
908 assert!(result.contains("\"totalIssues\": 5"));
909 assert!(result.contains("\"entry\""));
910 }
911
912 #[test]
913 fn toml_content_without_trailing_newline() {
914 let config = "[rules]\nunused-files = \"warn\"";
915 let result = update_toml_regression(config, &sample_baseline());
916 assert!(result.contains("[regression.baseline]"));
917 assert!(result.contains("[rules]"));
918 }
919
920 #[test]
921 fn toml_replace_section_not_at_end() {
922 let config = "[regression.baseline]\ntotalIssues = 99\nunusedFiles = 10\n\n[rules]\nunused-files = \"warn\"\n";
923 let result = update_toml_regression(config, &sample_baseline());
924 assert!(!result.contains("99"));
925 assert!(result.contains("totalIssues = 5"));
926 assert!(result.contains("[rules]"));
927 assert!(result.contains("unused-files = \"warn\""));
928 }
929
930 #[test]
931 fn toml_replace_section_at_end() {
932 let config =
933 "[rules]\nunused-files = \"warn\"\n\n[regression.baseline]\ntotalIssues = 99\n";
934 let result = update_toml_regression(config, &sample_baseline());
935 assert!(!result.contains("99"));
936 assert!(result.contains("totalIssues = 5"));
937 assert!(result.contains("[rules]"));
938 }
939
940 #[test]
941 fn find_json_key_multiple_same_keys() {
942 let content = r#"{"foo": 1, "bar": {"foo": 2}}"#;
943 let pos = find_json_key(content, "foo").unwrap();
944 assert_eq!(pos, 1);
945 }
946
947 #[test]
948 fn find_json_key_in_nested_comment_then_real() {
949 let content = "{\n // \"entry\": old\n /* \"entry\": also old */\n \"entry\": []\n}";
950 let pos = find_json_key(content, "entry").unwrap();
951 assert!(content[pos..].starts_with("\"entry\": []"));
952 }
953
954 fn make_opts(
955 fail: bool,
956 tolerance: Tolerance,
957 scoped: bool,
958 baseline_file: Option<&Path>,
959 ) -> RegressionOpts<'_> {
960 RegressionOpts {
961 fail_on_regression: fail,
962 tolerance,
963 regression_baseline_file: baseline_file,
964 save_target: SaveRegressionTarget::None,
965 scoped,
966 quiet: true,
967 output: OutputFormat::Human,
968 }
969 }
970
971 #[test]
972 fn compare_returns_none_when_disabled() {
973 let results = AnalysisResults::default();
974 let opts = make_opts(false, Tolerance::Absolute(0), false, None);
975 let config_baseline = fallow_config::RegressionBaseline {
976 total_issues: 5,
977 ..Default::default()
978 };
979 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
980 assert!(outcome.is_none());
981 }
982
983 #[test]
984 fn compare_returns_skipped_when_scoped() {
985 let results = AnalysisResults::default();
986 let opts = make_opts(true, Tolerance::Absolute(0), true, None);
987 let config_baseline = fallow_config::RegressionBaseline {
988 total_issues: 5,
989 ..Default::default()
990 };
991 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
992 assert!(matches!(outcome, Some(RegressionOutcome::Skipped { .. })));
993 }
994
995 #[test]
996 fn compare_pass_with_config_baseline() {
997 let results = AnalysisResults::default(); let opts = make_opts(true, Tolerance::Absolute(0), false, None);
999 let config_baseline = fallow_config::RegressionBaseline {
1000 total_issues: 0,
1001 ..Default::default()
1002 };
1003 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
1004 match outcome {
1005 Some(RegressionOutcome::Pass {
1006 baseline_total,
1007 current_total,
1008 }) => {
1009 assert_eq!(baseline_total, 0);
1010 assert_eq!(current_total, 0);
1011 }
1012 other => panic!("expected Pass, got {other:?}"),
1013 }
1014 }
1015
1016 #[test]
1017 fn compare_exceeded_with_config_baseline() {
1018 let mut results = AnalysisResults::default();
1019 results
1020 .unused_files
1021 .push(UnusedFileFinding::with_actions(UnusedFile {
1022 path: PathBuf::from("a.ts"),
1023 }));
1024 results
1025 .unused_files
1026 .push(UnusedFileFinding::with_actions(UnusedFile {
1027 path: PathBuf::from("b.ts"),
1028 }));
1029 let opts = make_opts(true, Tolerance::Absolute(0), false, None);
1030 let config_baseline = fallow_config::RegressionBaseline {
1031 total_issues: 0,
1032 ..Default::default()
1033 };
1034 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
1035 match outcome {
1036 Some(RegressionOutcome::Exceeded {
1037 baseline_total,
1038 current_total,
1039 ..
1040 }) => {
1041 assert_eq!(baseline_total, 0);
1042 assert_eq!(current_total, 2);
1043 }
1044 other => panic!("expected Exceeded, got {other:?}"),
1045 }
1046 }
1047
1048 #[test]
1049 fn compare_pass_within_tolerance() {
1050 let mut results = AnalysisResults::default();
1051 results
1052 .unused_files
1053 .push(UnusedFileFinding::with_actions(UnusedFile {
1054 path: PathBuf::from("a.ts"),
1055 }));
1056 let opts = make_opts(true, Tolerance::Absolute(5), false, None);
1057 let config_baseline = fallow_config::RegressionBaseline {
1058 total_issues: 0,
1059 ..Default::default()
1060 };
1061 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
1062 assert!(matches!(outcome, Some(RegressionOutcome::Pass { .. })));
1063 }
1064
1065 #[test]
1066 fn compare_improvement_is_pass() {
1067 let results = AnalysisResults::default(); let opts = make_opts(true, Tolerance::Absolute(0), false, None);
1069 let config_baseline = fallow_config::RegressionBaseline {
1070 total_issues: 10,
1071 unused_files: 5,
1072 unused_exports: 5,
1073 ..Default::default()
1074 };
1075 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
1076 match outcome {
1077 Some(RegressionOutcome::Pass {
1078 baseline_total,
1079 current_total,
1080 }) => {
1081 assert_eq!(baseline_total, 10);
1082 assert_eq!(current_total, 0);
1083 }
1084 other => panic!("expected Pass, got {other:?}"),
1085 }
1086 }
1087
1088 #[test]
1089 fn compare_with_file_baseline() {
1090 let dir = tempfile::tempdir().unwrap();
1091 let baseline_path = dir.path().join("baseline.json");
1092
1093 let counts = CheckCounts {
1094 total_issues: 5,
1095 unused_files: 5,
1096 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
1097 };
1098 save_regression_baseline(
1099 &baseline_path,
1100 dir.path(),
1101 Some(&counts),
1102 None,
1103 OutputFormat::Human,
1104 )
1105 .unwrap();
1106
1107 let results = AnalysisResults::default();
1108 let opts = make_opts(true, Tolerance::Absolute(0), false, Some(&baseline_path));
1109 let outcome = compare_check_regression(&results, &opts, None).unwrap();
1110 assert!(matches!(outcome, Some(RegressionOutcome::Pass { .. })));
1111 }
1112
1113 #[test]
1114 fn compare_file_baseline_missing_check_data_returns_error() {
1115 let dir = tempfile::tempdir().unwrap();
1116 let baseline_path = dir.path().join("baseline.json");
1117
1118 save_regression_baseline(
1119 &baseline_path,
1120 dir.path(),
1121 None,
1122 Some(&DupesCounts {
1123 clone_groups: 1,
1124 duplication_percentage: 1.0,
1125 }),
1126 OutputFormat::Human,
1127 )
1128 .unwrap();
1129
1130 let results = AnalysisResults::default();
1131 let opts = make_opts(true, Tolerance::Absolute(0), false, Some(&baseline_path));
1132 let outcome = compare_check_regression(&results, &opts, None);
1133 assert!(outcome.is_err());
1134 }
1135
1136 #[test]
1137 fn compare_no_baseline_source_returns_error() {
1138 let results = AnalysisResults::default();
1139 let opts = make_opts(true, Tolerance::Absolute(0), false, None);
1140 let outcome = compare_check_regression(&results, &opts, None);
1141 assert!(outcome.is_err());
1142 }
1143
1144 #[test]
1145 fn compare_exceeded_includes_type_deltas() {
1146 let mut results = AnalysisResults::default();
1147 results
1148 .unused_files
1149 .push(UnusedFileFinding::with_actions(UnusedFile {
1150 path: PathBuf::from("a.ts"),
1151 }));
1152 results
1153 .unused_files
1154 .push(UnusedFileFinding::with_actions(UnusedFile {
1155 path: PathBuf::from("b.ts"),
1156 }));
1157 results
1158 .unused_exports
1159 .push(UnusedExportFinding::with_actions(UnusedExport {
1160 path: PathBuf::from("c.ts"),
1161 export_name: "foo".into(),
1162 is_type_only: false,
1163 line: 1,
1164 col: 0,
1165 span_start: 0,
1166 is_re_export: false,
1167 }));
1168
1169 let opts = make_opts(true, Tolerance::Absolute(0), false, None);
1170 let config_baseline = fallow_config::RegressionBaseline {
1171 total_issues: 0,
1172 ..Default::default()
1173 };
1174 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
1175
1176 match outcome {
1177 Some(RegressionOutcome::Exceeded { type_deltas, .. }) => {
1178 assert!(type_deltas.contains(&("unused_files", 2)));
1179 assert!(type_deltas.contains(&("unused_exports", 1)));
1180 }
1181 other => panic!("expected Exceeded, got {other:?}"),
1182 }
1183 }
1184
1185 #[test]
1186 fn compare_with_percentage_tolerance() {
1187 let mut results = AnalysisResults::default();
1188 results
1189 .unused_files
1190 .push(UnusedFileFinding::with_actions(UnusedFile {
1191 path: PathBuf::from("a.ts"),
1192 }));
1193
1194 let opts = make_opts(true, Tolerance::Percentage(50.0), false, None);
1195 let config_baseline = fallow_config::RegressionBaseline {
1196 total_issues: 10,
1197 unused_files: 10,
1198 ..Default::default()
1199 };
1200 let outcome = compare_check_regression(&results, &opts, Some(&config_baseline)).unwrap();
1201 assert!(matches!(outcome, Some(RegressionOutcome::Pass { .. })));
1202 }
1203
1204 fn write_baseline_with_schema_version(dir: &Path, version: u32) -> PathBuf {
1205 let path = dir.join("baseline.json");
1206 let body = format!(
1207 r#"{{
1208 "schema_version": {version},
1209 "fallow_version": "3.0.0",
1210 "timestamp": "2026-05-21T00:00:00Z",
1211 "check": {{
1212 "total_issues": 0,
1213 "unused_files": 0
1214 }}
1215}}"#
1216 );
1217 std::fs::write(&path, body).unwrap();
1218 path
1219 }
1220
1221 #[test]
1222 fn load_rejects_schema_version_too_high() {
1223 let dir = tempfile::tempdir().unwrap();
1224 let path = write_baseline_with_schema_version(dir.path(), REGRESSION_SCHEMA_VERSION + 1);
1225 let result = load_regression_baseline(&path, OutputFormat::Human);
1226 assert!(result.is_err());
1227 }
1228
1229 #[test]
1230 fn load_rejects_schema_version_zero_predates_versioning() {
1231 let dir = tempfile::tempdir().unwrap();
1232 let path = write_baseline_with_schema_version(dir.path(), 0);
1233 let result = load_regression_baseline(&path, OutputFormat::Human);
1234 assert!(result.is_err());
1235 }
1236
1237 #[test]
1238 fn load_accepts_current_schema_version() {
1239 let dir = tempfile::tempdir().unwrap();
1240 let path = write_baseline_with_schema_version(dir.path(), REGRESSION_SCHEMA_VERSION);
1241 let loaded = load_regression_baseline(&path, OutputFormat::Human).unwrap();
1242 assert_eq!(loaded.schema_version, REGRESSION_SCHEMA_VERSION);
1243 }
1244
1245 #[test]
1246 fn load_rewrites_missing_schema_version_field_error() {
1247 let dir = tempfile::tempdir().unwrap();
1248 let path = dir.path().join("baseline.json");
1249 std::fs::write(
1250 &path,
1251 r#"{
1252 "fallow_version": "1.0.0",
1253 "timestamp": "2026-05-21T00:00:00Z",
1254 "check": {}
1255}"#,
1256 )
1257 .unwrap();
1258 let result = load_regression_baseline(&path, OutputFormat::Human);
1259 assert!(result.is_err());
1260 }
1261
1262 #[test]
1263 fn format_schema_mismatch_error_too_high() {
1264 let msg =
1265 format_schema_mismatch_error(Path::new("/repo/.fallow-baseline.json"), 1, 99, "3.0.0");
1266 assert!(msg.contains("schema_version 99"));
1267 assert!(msg.contains("expects 1"));
1268 assert!(msg.contains("fallow 3.0.0"));
1269 assert!(
1270 msg.contains("fallow dead-code --save-regression-baseline /repo/.fallow-baseline.json")
1271 );
1272 assert!(!msg.to_lowercase().contains("refresh"));
1273 assert!(msg.contains("schema_version"));
1274 }
1275
1276 #[test]
1277 fn format_schema_mismatch_error_actual_zero_special_case() {
1278 let msg =
1279 format_schema_mismatch_error(Path::new("/repo/.fallow-baseline.json"), 1, 0, "2.0.0");
1280 assert!(msg.contains("predate"));
1281 assert!(msg.contains("fallow 2.0.0"));
1282 assert!(
1283 msg.contains("fallow dead-code --save-regression-baseline /repo/.fallow-baseline.json")
1284 );
1285 }
1286
1287 #[test]
1288 fn format_missing_schema_version_error_includes_regenerate_command() {
1289 let msg = format_missing_schema_version_error(Path::new("/repo/baseline.json"));
1290 assert!(msg.contains("missing the schema_version field"));
1291 assert!(msg.contains("fallow dead-code --save-regression-baseline /repo/baseline.json"));
1292 }
1293
1294 #[test]
1295 fn save_load_preserves_schema_version() {
1296 let dir = tempfile::tempdir().unwrap();
1297 let path = dir.path().join("baseline.json");
1298 let counts = CheckCounts {
1299 total_issues: 1,
1300 unused_files: 1,
1301 ..CheckCounts::from_config_baseline(&fallow_config::RegressionBaseline::default())
1302 };
1303 save_regression_baseline(&path, dir.path(), Some(&counts), None, OutputFormat::Human)
1304 .unwrap();
1305 let loaded = load_regression_baseline(&path, OutputFormat::Human).unwrap();
1306 assert_eq!(loaded.schema_version, REGRESSION_SCHEMA_VERSION);
1307 }
1308}