1mod badge;
2pub mod ci;
3mod codeclimate;
4mod compact;
5pub mod dupes_grouping;
6pub mod grouping;
7mod human;
8mod json;
9mod markdown;
10mod sarif;
11mod shared;
12pub mod sink;
13pub mod suggestions;
14#[cfg(test)]
15pub mod test_helpers;
16
17use std::path::Path;
18use std::process::ExitCode;
19use std::time::Duration;
20
21use fallow_config::{OutputFormat, RulesConfig, Severity};
22use fallow_core::duplicates::DuplicationReport;
23use fallow_core::results::AnalysisResults;
24use fallow_core::trace::{CloneTrace, DependencyTrace, ExportTrace, FileTrace, PipelineTimings};
25
26use crate::report::sink::outln;
27
28pub use grouping::OwnershipResolver;
29pub use human::health::{render_health_score, render_health_trend};
30#[allow(
31 unused_imports,
32 reason = "used by binary crate modules (combined.rs, audit.rs)"
33)]
34pub use json::strip_root_prefix;
35
36pub struct ReportContext<'a> {
41 pub root: &'a Path,
42 pub rules: &'a RulesConfig,
43 pub elapsed: Duration,
44 pub quiet: bool,
45 pub explain: bool,
46 pub group_by: Option<OwnershipResolver>,
48 pub top: Option<usize>,
50 pub summary: bool,
52 pub summary_heading: bool,
56 pub show_explain_tip: bool,
58 pub baseline_matched: Option<(usize, usize)>,
60 pub config_fixable: bool,
65 pub skip_score_and_trend: bool,
70}
71
72#[must_use]
74pub fn relative_path<'a>(path: &'a Path, root: &Path) -> &'a Path {
75 path.strip_prefix(root).unwrap_or(path)
76}
77
78#[must_use]
88pub fn format_display_path(path: &Path, root: &Path) -> String {
89 relative_path(path, root)
90 .display()
91 .to_string()
92 .replace('\\', "/")
93}
94
95#[must_use]
98pub fn split_dir_filename(path: &str) -> (&str, &str) {
99 path.rfind('/')
100 .map_or(("", path), |pos| (&path[..=pos], &path[pos + 1..]))
101}
102
103#[must_use]
105pub const fn plural(n: usize) -> &'static str {
106 if n == 1 { "" } else { "s" }
107}
108
109#[must_use]
114pub fn emit_json(value: &serde_json::Value, kind: &str) -> ExitCode {
115 match serde_json::to_string_pretty(value) {
116 Ok(json) => {
117 outln!("{json}");
118 ExitCode::SUCCESS
119 }
120 Err(e) => {
121 eprintln!("Error: failed to serialize {kind} output: {e}");
122 ExitCode::from(2)
123 }
124 }
125}
126
127#[must_use]
133pub fn elide_common_prefix<'a>(base: &str, target: &'a str) -> &'a str {
134 let mut last_sep = 0;
135 for (i, (a, b)) in base.bytes().zip(target.bytes()).enumerate() {
136 if a != b {
137 break;
138 }
139 if a == b'/' {
140 last_sep = i + 1;
141 }
142 }
143 if last_sep > 0 && last_sep <= target.len() {
144 &target[last_sep..]
145 } else {
146 target
147 }
148}
149
150fn relative_uri(path: &Path, root: &Path) -> String {
152 normalize_uri(&relative_path(path, root).display().to_string())
153}
154
155#[must_use]
160pub fn normalize_uri(path_str: &str) -> String {
161 path_str
162 .replace('\\', "/")
163 .replace('[', "%5B")
164 .replace(']', "%5D")
165}
166
167#[derive(Clone, Copy, Debug)]
169pub enum Level {
170 Warn,
171 Info,
172 Error,
173}
174
175#[must_use]
176pub const fn severity_to_level(s: Severity) -> Level {
177 match s {
178 Severity::Error => Level::Error,
179 Severity::Warn => Level::Warn,
180 Severity::Off => Level::Info,
181 }
182}
183
184#[must_use]
190pub fn print_results(
191 results: &AnalysisResults,
192 ctx: &ReportContext<'_>,
193 output: OutputFormat,
194 regression: Option<&crate::regression::RegressionOutcome>,
195) -> ExitCode {
196 if let Some(ref resolver) = ctx.group_by {
197 let groups = grouping::group_analysis_results(results, ctx.root, resolver);
198 return print_grouped_results(&groups, results, ctx, output, resolver);
199 }
200
201 match output {
202 OutputFormat::Human => {
203 if ctx.summary {
204 human::check::print_check_summary(
205 results,
206 ctx.rules,
207 ctx.elapsed,
208 ctx.quiet,
209 ctx.summary_heading,
210 );
211 } else {
212 human::print_human(&human::PrintHumanInput {
213 results,
214 root: ctx.root,
215 rules: ctx.rules,
216 elapsed: ctx.elapsed,
217 quiet: ctx.quiet,
218 top: ctx.top,
219 show_explain_tip: ctx.show_explain_tip,
220 explain: ctx.explain,
221 });
222 }
223 ExitCode::SUCCESS
224 }
225 OutputFormat::Json => json::print_json(&json::PrintJsonInput {
226 results,
227 root: ctx.root,
228 elapsed: ctx.elapsed,
229 explain: ctx.explain,
230 regression,
231 baseline_matched: ctx.baseline_matched,
232 config_fixable: ctx.config_fixable,
233 }),
234 OutputFormat::Compact => {
235 compact::print_compact(results, ctx.root);
236 ExitCode::SUCCESS
237 }
238 OutputFormat::Sarif => sarif::print_sarif(results, ctx.root, ctx.rules),
239 OutputFormat::Markdown => {
240 markdown::print_markdown(results, ctx.root);
241 ExitCode::SUCCESS
242 }
243 OutputFormat::CodeClimate => codeclimate::print_codeclimate(results, ctx.root, ctx.rules),
244 ci_format => print_results_ci_comment(results, ctx, ci_format),
245 }
246}
247
248fn print_results_ci_comment(
250 results: &AnalysisResults,
251 ctx: &ReportContext<'_>,
252 output: OutputFormat,
253) -> ExitCode {
254 let issues = codeclimate::build_codeclimate(results, ctx.root, ctx.rules);
255 let value = codeclimate::issues_to_value(&issues);
256 print_ci_comment_format("dead-code", &value, output).unwrap_or_else(|| {
257 eprintln!("Error: badge format is only supported for the health command");
258 ExitCode::from(2)
259 })
260}
261
262#[must_use]
264fn print_grouped_results(
265 groups: &[grouping::ResultGroup],
266 original: &AnalysisResults,
267 ctx: &ReportContext<'_>,
268 output: OutputFormat,
269 resolver: &OwnershipResolver,
270) -> ExitCode {
271 match output {
272 OutputFormat::Human => {
273 human::print_grouped_human(&human::PrintGroupedHumanInput {
274 groups,
275 root: ctx.root,
276 rules: ctx.rules,
277 elapsed: ctx.elapsed,
278 quiet: ctx.quiet,
279 resolver: Some(resolver),
280 explain: ctx.explain,
281 });
282 ExitCode::SUCCESS
283 }
284 OutputFormat::Json => json::print_grouped_json(&json::PrintGroupedJsonInput {
285 groups,
286 original,
287 root: ctx.root,
288 elapsed: ctx.elapsed,
289 explain: ctx.explain,
290 resolver,
291 config_fixable: ctx.config_fixable,
292 }),
293 OutputFormat::Compact => {
294 compact::print_grouped_compact(groups, ctx.root);
295 ExitCode::SUCCESS
296 }
297 OutputFormat::Markdown => {
298 markdown::print_grouped_markdown(groups, ctx.root);
299 ExitCode::SUCCESS
300 }
301 OutputFormat::Sarif => sarif::print_grouped_sarif(original, ctx.root, ctx.rules, resolver),
302 OutputFormat::CodeClimate => {
303 codeclimate::print_grouped_codeclimate(original, ctx.root, ctx.rules, resolver)
304 }
305 ci_format => print_results_ci_comment(original, ctx, ci_format),
306 }
307}
308
309#[must_use]
311pub fn print_duplication_report(
312 report: &DuplicationReport,
313 ctx: &ReportContext<'_>,
314 output: OutputFormat,
315) -> ExitCode {
316 if let Some(ref resolver) = ctx.group_by {
317 let grouping = dupes_grouping::build_duplication_grouping(report, ctx.root, resolver);
318 return print_grouped_duplication_report(report, &grouping, ctx, output, resolver);
319 }
320
321 match output {
322 OutputFormat::Human => {
323 if ctx.summary {
324 human::dupes::print_duplication_summary(
325 report,
326 ctx.elapsed,
327 ctx.quiet,
328 ctx.summary_heading,
329 );
330 } else {
331 human::print_duplication_human(
332 report,
333 ctx.root,
334 ctx.elapsed,
335 ctx.quiet,
336 ctx.show_explain_tip,
337 ctx.explain,
338 );
339 }
340 ExitCode::SUCCESS
341 }
342 OutputFormat::Json => {
343 json::print_duplication_json(report, ctx.root, ctx.elapsed, ctx.explain)
344 }
345 OutputFormat::Compact => {
346 compact::print_duplication_compact(report, ctx.root);
347 ExitCode::SUCCESS
348 }
349 OutputFormat::Sarif => sarif::print_duplication_sarif(report, ctx.root),
350 OutputFormat::Markdown => {
351 markdown::print_duplication_markdown(report, ctx.root);
352 ExitCode::SUCCESS
353 }
354 OutputFormat::CodeClimate => codeclimate::print_duplication_codeclimate(report, ctx.root),
355 ci_format => print_duplication_ci_comment(report, ctx.root, ci_format),
356 }
357}
358
359fn print_duplication_ci_comment(
361 report: &DuplicationReport,
362 root: &Path,
363 output: OutputFormat,
364) -> ExitCode {
365 let issues = codeclimate::build_duplication_codeclimate(report, root);
366 let value = codeclimate::issues_to_value(&issues);
367 print_ci_comment_format("dupes", &value, output).unwrap_or_else(|| {
368 eprintln!("Error: badge format is only supported for the health command");
369 ExitCode::from(2)
370 })
371}
372
373#[must_use]
375fn print_grouped_duplication_report(
376 report: &DuplicationReport,
377 grouping: &dupes_grouping::DuplicationGrouping,
378 ctx: &ReportContext<'_>,
379 output: OutputFormat,
380 resolver: &OwnershipResolver,
381) -> ExitCode {
382 match output {
383 OutputFormat::Human => {
384 human::print_grouped_duplication_human(
385 report,
386 grouping,
387 ctx.root,
388 ctx.elapsed,
389 ctx.quiet,
390 );
391 ExitCode::SUCCESS
392 }
393 OutputFormat::Json => json::print_grouped_duplication_json(
394 report,
395 grouping,
396 ctx.root,
397 ctx.elapsed,
398 ctx.explain,
399 ),
400 OutputFormat::Sarif => sarif::print_grouped_duplication_sarif(report, ctx.root, resolver),
401 OutputFormat::CodeClimate => {
402 codeclimate::print_grouped_duplication_codeclimate(report, ctx.root, resolver)
403 }
404 OutputFormat::PrCommentGithub
405 | OutputFormat::PrCommentGitlab
406 | OutputFormat::ReviewGithub
407 | OutputFormat::ReviewGitlab => print_duplication_ci_comment(report, ctx.root, output),
408 OutputFormat::Compact => {
409 compact::print_duplication_compact(report, ctx.root);
410 warn_dupes_grouping_unsupported(grouping, "compact");
411 ExitCode::SUCCESS
412 }
413 OutputFormat::Markdown => {
414 markdown::print_duplication_markdown(report, ctx.root);
415 warn_dupes_grouping_unsupported(grouping, "markdown");
416 ExitCode::SUCCESS
417 }
418 OutputFormat::Badge => {
419 eprintln!("Error: badge format is only supported for the health command");
420 ExitCode::from(2)
421 }
422 }
423}
424
425fn print_ci_comment_format(
430 analysis: &str,
431 value: &serde_json::Value,
432 output: OutputFormat,
433) -> Option<ExitCode> {
434 let exit = match output {
435 OutputFormat::PrCommentGithub => {
436 ci::pr_comment::print_pr_comment(analysis, ci::pr_comment::Provider::Github, value)
437 }
438 OutputFormat::PrCommentGitlab => {
439 ci::pr_comment::print_pr_comment(analysis, ci::pr_comment::Provider::Gitlab, value)
440 }
441 OutputFormat::ReviewGithub => {
442 ci::review::print_review_envelope(analysis, ci::pr_comment::Provider::Github, value)
443 }
444 OutputFormat::ReviewGitlab => {
445 ci::review::print_review_envelope(analysis, ci::pr_comment::Provider::Gitlab, value)
446 }
447 _ => return None,
448 };
449 Some(exit)
450}
451
452fn warn_dupes_grouping_unsupported(grouping: &dupes_grouping::DuplicationGrouping, format: &str) {
453 eprintln!(
454 "note: --group-by {} is not supported for {format} duplication output, falling back to \
455 ungrouped output (use --format json for the full grouped envelope)",
456 grouping.mode
457 );
458}
459
460#[must_use]
477pub fn print_health_report(
478 report: &crate::health_types::HealthReport,
479 grouping: Option<&crate::health_types::HealthGrouping>,
480 group_resolver: Option<&grouping::OwnershipResolver>,
481 ctx: &ReportContext<'_>,
482 output: OutputFormat,
483) -> ExitCode {
484 match output {
485 OutputFormat::Human => {
486 print_health_human_report(report, grouping, ctx);
487 ExitCode::SUCCESS
488 }
489 OutputFormat::Compact => {
490 compact::print_health_compact(report, ctx.root);
491 warn_grouping_unsupported(grouping, "compact");
492 ExitCode::SUCCESS
493 }
494 OutputFormat::Markdown => {
495 markdown::print_health_markdown(report, ctx.root);
496 warn_grouping_unsupported(grouping, "markdown");
497 ExitCode::SUCCESS
498 }
499 OutputFormat::Sarif => match group_resolver {
500 Some(resolver) => sarif::print_grouped_health_sarif(report, ctx.root, resolver),
501 None => sarif::print_health_sarif(report, ctx.root),
502 },
503 OutputFormat::Json => match grouping {
504 Some(grouping) => json::print_grouped_health_json(
505 report,
506 grouping,
507 ctx.root,
508 ctx.elapsed,
509 ctx.explain,
510 ),
511 None => json::print_health_json(report, ctx.root, ctx.elapsed, ctx.explain),
512 },
513 OutputFormat::CodeClimate => match group_resolver {
514 Some(resolver) => {
515 codeclimate::print_grouped_health_codeclimate(report, ctx.root, resolver)
516 }
517 None => codeclimate::print_health_codeclimate(report, ctx.root),
518 },
519 OutputFormat::PrCommentGithub
520 | OutputFormat::PrCommentGitlab
521 | OutputFormat::ReviewGithub
522 | OutputFormat::ReviewGitlab => print_health_ci_comment(report, ctx.root, output),
523 OutputFormat::Badge => {
524 warn_grouping_unsupported(grouping, "badge");
525 badge::print_health_badge(report)
526 }
527 }
528}
529
530fn print_health_human_report(
532 report: &crate::health_types::HealthReport,
533 grouping: Option<&crate::health_types::HealthGrouping>,
534 ctx: &ReportContext<'_>,
535) {
536 if ctx.summary {
537 human::health::print_health_summary(report, ctx.elapsed, ctx.quiet, ctx.summary_heading);
538 return;
539 }
540 human::print_health_human(&human::PrintHealthHumanInput {
541 report,
542 root: ctx.root,
543 elapsed: ctx.elapsed,
544 quiet: ctx.quiet,
545 show_explain_tip: ctx.show_explain_tip,
546 explain: ctx.explain,
547 skip_score_and_trend: ctx.skip_score_and_trend,
548 });
549 if let Some(grouping) = grouping {
550 human::print_health_grouping(grouping, ctx.root, ctx.quiet);
551 }
552}
553
554fn print_health_ci_comment(
556 report: &crate::health_types::HealthReport,
557 root: &Path,
558 output: OutputFormat,
559) -> ExitCode {
560 let issues = codeclimate::build_health_codeclimate(report, root);
561 let value = codeclimate::issues_to_value(&issues);
562 print_ci_comment_format("health", &value, output).unwrap_or_else(|| {
563 eprintln!("Error: badge format is only supported for the health command");
564 ExitCode::from(2)
565 })
566}
567
568fn warn_grouping_unsupported(grouping: Option<&crate::health_types::HealthGrouping>, format: &str) {
569 if let Some(g) = grouping {
570 eprintln!(
571 "note: --group-by {} is not supported for {format} output, falling back to \
572 ungrouped output (use --format json for the full grouped envelope)",
573 g.mode
574 );
575 }
576}
577
578pub fn print_cross_reference_findings(
582 cross_ref: &fallow_core::cross_reference::CrossReferenceResult,
583 root: &Path,
584 quiet: bool,
585 output: OutputFormat,
586) {
587 human::print_cross_reference_findings(cross_ref, root, quiet, output);
588}
589
590pub fn print_export_trace(trace: &ExportTrace, format: OutputFormat) {
592 match format {
593 OutputFormat::Json => json::print_trace_json(trace),
594 _ => human::print_export_trace_human(trace),
595 }
596}
597
598pub fn print_file_trace(trace: &FileTrace, format: OutputFormat) {
600 match format {
601 OutputFormat::Json => json::print_trace_json(trace),
602 _ => human::print_file_trace_human(trace),
603 }
604}
605
606pub fn print_dependency_trace(trace: &DependencyTrace, format: OutputFormat) {
608 match format {
609 OutputFormat::Json => json::print_trace_json(trace),
610 _ => human::print_dependency_trace_human(trace),
611 }
612}
613
614pub fn print_clone_trace(trace: &CloneTrace, root: &Path, format: OutputFormat) {
616 match format {
617 OutputFormat::Json => json::print_trace_json(trace),
618 _ => human::print_clone_trace_human(trace, root),
619 }
620}
621
622pub fn print_performance(timings: &PipelineTimings, format: OutputFormat) {
625 match format {
626 OutputFormat::Json => match serde_json::to_string_pretty(timings) {
627 Ok(json) => eprintln!("{json}"),
628 Err(e) => eprintln!("Error: failed to serialize timings: {e}"),
629 },
630 _ => human::print_performance_human(timings),
631 }
632}
633
634pub fn print_health_performance(
637 timings: &crate::health_types::HealthTimings,
638 format: OutputFormat,
639) {
640 match format {
641 OutputFormat::Json => match serde_json::to_string_pretty(timings) {
642 Ok(json) => eprintln!("{json}"),
643 Err(e) => eprintln!("Error: failed to serialize timings: {e}"),
644 },
645 _ => human::print_health_performance_human(timings),
646 }
647}
648
649#[allow(
650 unused_imports,
651 reason = "target-dependent: used in lib, unused in bin"
652)]
653pub use codeclimate::build_codeclimate;
654#[allow(
655 unused_imports,
656 reason = "target-dependent: used in lib, unused in bin"
657)]
658pub use codeclimate::build_duplication_codeclimate;
659#[allow(
660 unused_imports,
661 reason = "target-dependent: used in lib, unused in bin"
662)]
663pub use codeclimate::build_health_codeclimate;
664#[allow(
665 unused_imports,
666 reason = "target-dependent: used in lib, unused in bin"
667)]
668pub use codeclimate::issues_to_value as codeclimate_issues_to_value;
669#[allow(
670 unused_imports,
671 reason = "target-dependent: used in lib, unused in bin"
672)]
673pub use compact::build_compact_lines;
674#[allow(
675 clippy::redundant_pub_crate,
676 reason = "pub(crate) deliberately limits visibility, report is pub but these are internal"
677)]
678pub(crate) use json::SCHEMA_VERSION;
679pub use json::build_baseline_deltas_json;
680pub use json::build_check_json_payload_with_config_fixable;
681#[allow(
682 unused_imports,
683 reason = "target-dependent: used in lib, unused in bin"
684)]
685pub use json::build_duplication_json;
686#[allow(
687 unused_imports,
688 reason = "target-dependent: used in lib, unused in bin"
689)]
690pub use json::build_grouped_duplication_json;
691#[allow(
692 unused_imports,
693 reason = "target-dependent: used in lib, unused in bin"
694)]
695pub use json::build_health_json;
696#[allow(
697 unused_imports,
698 reason = "target-dependent: used in bin audit.rs, unused in lib"
699)]
700#[allow(
701 clippy::redundant_pub_crate,
702 reason = "pub(crate) deliberately limits visibility, report is pub but these are internal"
703)]
704pub(crate) use json::harmonize_multi_kind_suppress_line_actions;
705#[allow(
706 unused_imports,
707 reason = "target-dependent: used in lib, unused in bin"
708)]
709pub use json::{build_json, build_json_with_config_fixable};
710#[allow(
711 unused_imports,
712 reason = "target-dependent: used in lib, unused in bin"
713)]
714pub use markdown::build_duplication_markdown;
715#[allow(
716 unused_imports,
717 reason = "target-dependent: used in lib, unused in bin"
718)]
719pub use markdown::build_health_markdown;
720#[allow(
721 unused_imports,
722 reason = "target-dependent: used in lib, unused in bin"
723)]
724pub use markdown::build_markdown;
725#[allow(
726 unused_imports,
727 reason = "target-dependent: used in lib, unused in bin"
728)]
729pub use sarif::build_health_sarif;
730#[allow(
731 unused_imports,
732 reason = "target-dependent: used in lib, unused in bin"
733)]
734pub use sarif::build_sarif;
735
736#[cfg(test)]
737mod tests {
738 use super::*;
739 use std::path::{Path, PathBuf};
740
741 fn test_context<'a>(root: &'a Path, rules: &'a RulesConfig) -> ReportContext<'a> {
742 ReportContext {
743 root,
744 rules,
745 elapsed: Duration::default(),
746 quiet: true,
747 explain: false,
748 group_by: None,
749 top: None,
750 summary: false,
751 summary_heading: false,
752 show_explain_tip: false,
753 baseline_matched: None,
754 config_fixable: false,
755 skip_score_and_trend: false,
756 }
757 }
758
759 #[test]
760 fn normalize_uri_forward_slashes_unchanged() {
761 assert_eq!(normalize_uri("src/utils.ts"), "src/utils.ts");
762 }
763
764 #[test]
765 fn normalize_uri_backslashes_replaced() {
766 assert_eq!(normalize_uri("src\\utils\\index.ts"), "src/utils/index.ts");
767 }
768
769 #[test]
770 fn normalize_uri_mixed_slashes() {
771 assert_eq!(normalize_uri("src\\utils/index.ts"), "src/utils/index.ts");
772 }
773
774 #[test]
775 fn normalize_uri_path_with_spaces() {
776 assert_eq!(
777 normalize_uri("src\\my folder\\file.ts"),
778 "src/my folder/file.ts"
779 );
780 }
781
782 #[test]
783 fn normalize_uri_empty_string() {
784 assert_eq!(normalize_uri(""), "");
785 }
786
787 #[test]
788 fn relative_path_strips_root_prefix() {
789 let root = Path::new("/project");
790 let path = Path::new("/project/src/utils.ts");
791 assert_eq!(relative_path(path, root), Path::new("src/utils.ts"));
792 }
793
794 #[test]
795 fn relative_path_returns_full_path_when_no_prefix() {
796 let root = Path::new("/other");
797 let path = Path::new("/project/src/utils.ts");
798 assert_eq!(relative_path(path, root), path);
799 }
800
801 #[test]
802 fn relative_path_at_root_returns_empty_or_file() {
803 let root = Path::new("/project");
804 let path = Path::new("/project/file.ts");
805 assert_eq!(relative_path(path, root), Path::new("file.ts"));
806 }
807
808 #[test]
809 fn relative_path_deeply_nested() {
810 let root = Path::new("/project");
811 let path = Path::new("/project/packages/ui/src/components/Button.tsx");
812 assert_eq!(
813 relative_path(path, root),
814 Path::new("packages/ui/src/components/Button.tsx")
815 );
816 }
817
818 #[test]
819 fn format_display_path_returns_workspace_relative() {
820 let root = Path::new("/project");
821 let path = Path::new("/project/apps/server/src/index.ts");
822 assert_eq!(format_display_path(path, root), "apps/server/src/index.ts");
823 }
824
825 #[test]
826 fn format_display_path_collides_in_nx_layout_renders_full_relative() {
827 let root = Path::new("/project");
828 let server = Path::new("/project/apps/server/src/index.ts");
829 let client = Path::new("/project/apps/client/src/index.ts");
830 assert_eq!(
831 format_display_path(server, root),
832 "apps/server/src/index.ts"
833 );
834 assert_eq!(
835 format_display_path(client, root),
836 "apps/client/src/index.ts"
837 );
838 }
839
840 #[test]
841 fn format_display_path_angular_component_renders_parent_directory() {
842 let root = Path::new("/project");
843 let path = Path::new(
844 "/project/apps/admin/src/app/payments/payment-list/payment-list.component.html",
845 );
846 assert_eq!(
847 format_display_path(path, root),
848 "apps/admin/src/app/payments/payment-list/payment-list.component.html"
849 );
850 }
851
852 #[test]
853 fn format_display_path_falls_back_to_full_path_when_root_does_not_prefix() {
854 let root = Path::new("/other");
855 let path = Path::new("/project/src/utils.ts");
856 let rendered = format_display_path(path, root);
857 assert!(rendered.contains("project"));
858 assert!(rendered.ends_with("utils.ts"));
859 assert!(!rendered.contains('\\'));
860 }
861
862 #[test]
863 fn format_display_path_normalizes_backslashes_to_forward_slashes() {
864 let root = Path::new("/project");
865 let path = Path::new("/project/src/sub\\file.ts");
866 let rendered = format_display_path(path, root);
867 assert!(
868 !rendered.contains('\\'),
869 "backslashes must be normalized: {rendered}"
870 );
871 }
872
873 #[test]
874 fn format_display_path_handles_brackets_verbatim() {
875 let root = Path::new("/project");
876 let path = Path::new("/project/app/[slug]/page.tsx");
877 assert_eq!(format_display_path(path, root), "app/[slug]/page.tsx");
878 }
879
880 #[test]
881 fn format_display_path_path_equals_root_returns_empty() {
882 let root = Path::new("/project");
883 let path = Path::new("/project");
884 assert_eq!(format_display_path(path, root), "");
885 }
886
887 #[test]
888 fn format_display_path_basename_only_when_path_is_at_root() {
889 let root = Path::new("/project");
890 let path = Path::new("/project/Cargo.toml");
891 assert_eq!(format_display_path(path, root), "Cargo.toml");
892 }
893
894 #[test]
895 fn relative_uri_produces_forward_slash_path() {
896 let root = PathBuf::from("/project");
897 let path = root.join("src").join("utils.ts");
898 let uri = relative_uri(&path, &root);
899 assert_eq!(uri, "src/utils.ts");
900 }
901
902 #[test]
903 fn relative_uri_encodes_brackets() {
904 let root = PathBuf::from("/project");
905 let path = root.join("src/app/[...slug]/page.tsx");
906 let uri = relative_uri(&path, &root);
907 assert_eq!(uri, "src/app/%5B...slug%5D/page.tsx");
908 }
909
910 #[test]
911 fn relative_uri_encodes_nested_dynamic_routes() {
912 let root = PathBuf::from("/project");
913 let path = root.join("src/app/[slug]/[id]/page.tsx");
914 let uri = relative_uri(&path, &root);
915 assert_eq!(uri, "src/app/%5Bslug%5D/%5Bid%5D/page.tsx");
916 }
917
918 #[test]
919 fn relative_uri_no_common_prefix_returns_full() {
920 let root = PathBuf::from("/other");
921 let path = PathBuf::from("/project/src/utils.ts");
922 let uri = relative_uri(&path, &root);
923 assert!(uri.contains("project"));
924 assert!(uri.contains("utils.ts"));
925 }
926
927 #[test]
928 fn severity_error_maps_to_level_error() {
929 assert!(matches!(severity_to_level(Severity::Error), Level::Error));
930 }
931
932 #[test]
933 fn severity_warn_maps_to_level_warn() {
934 assert!(matches!(severity_to_level(Severity::Warn), Level::Warn));
935 }
936
937 #[test]
938 fn severity_off_maps_to_level_info() {
939 assert!(matches!(severity_to_level(Severity::Off), Level::Info));
940 }
941
942 #[test]
943 fn normalize_uri_single_bracket_pair() {
944 assert_eq!(normalize_uri("app/[id]/page.tsx"), "app/%5Bid%5D/page.tsx");
945 }
946
947 #[test]
948 fn normalize_uri_catch_all_route() {
949 assert_eq!(
950 normalize_uri("app/[...slug]/page.tsx"),
951 "app/%5B...slug%5D/page.tsx"
952 );
953 }
954
955 #[test]
956 fn normalize_uri_optional_catch_all_route() {
957 assert_eq!(
958 normalize_uri("app/[[...slug]]/page.tsx"),
959 "app/%5B%5B...slug%5D%5D/page.tsx"
960 );
961 }
962
963 #[test]
964 fn normalize_uri_multiple_dynamic_segments() {
965 assert_eq!(
966 normalize_uri("app/[lang]/posts/[id]"),
967 "app/%5Blang%5D/posts/%5Bid%5D"
968 );
969 }
970
971 #[test]
972 fn normalize_uri_no_special_chars() {
973 let plain = "src/components/Button.tsx";
974 assert_eq!(normalize_uri(plain), plain);
975 }
976
977 #[test]
978 fn normalize_uri_only_backslashes() {
979 assert_eq!(normalize_uri("a\\b\\c"), "a/b/c");
980 }
981
982 #[test]
983 fn relative_path_identical_paths_returns_empty() {
984 let root = Path::new("/project");
985 assert_eq!(relative_path(root, root), Path::new(""));
986 }
987
988 #[test]
989 fn relative_path_partial_name_match_not_stripped() {
990 let root = Path::new("/project");
991 let path = Path::new("/project-two/src/a.ts");
992 assert_eq!(relative_path(path, root), path);
993 }
994
995 #[test]
996 fn relative_uri_combines_stripping_and_encoding() {
997 let root = PathBuf::from("/project");
998 let path = root.join("src/app/[slug]/page.tsx");
999 let uri = relative_uri(&path, &root);
1000 assert_eq!(uri, "src/app/%5Bslug%5D/page.tsx");
1001 assert!(!uri.starts_with('/'));
1002 }
1003
1004 #[test]
1005 fn relative_uri_at_root_file() {
1006 let root = PathBuf::from("/project");
1007 let path = root.join("index.ts");
1008 assert_eq!(relative_uri(&path, &root), "index.ts");
1009 }
1010
1011 #[test]
1012 fn severity_to_level_is_const_evaluable() {
1013 const LEVEL_FROM_ERROR: Level = severity_to_level(Severity::Error);
1014 const LEVEL_FROM_WARN: Level = severity_to_level(Severity::Warn);
1015 const LEVEL_FROM_OFF: Level = severity_to_level(Severity::Off);
1016 assert!(matches!(LEVEL_FROM_ERROR, Level::Error));
1017 assert!(matches!(LEVEL_FROM_WARN, Level::Warn));
1018 assert!(matches!(LEVEL_FROM_OFF, Level::Info));
1019 }
1020
1021 #[test]
1022 fn level_is_copy() {
1023 let level = severity_to_level(Severity::Error);
1024 let copy = level;
1025 assert!(matches!(level, Level::Error));
1026 assert!(matches!(copy, Level::Error));
1027 }
1028
1029 #[test]
1030 fn print_results_rejects_badge_for_dead_code_reports() {
1031 let root = Path::new("/project");
1032 let rules = RulesConfig::default();
1033 let ctx = test_context(root, &rules);
1034
1035 let code = print_results(&AnalysisResults::default(), &ctx, OutputFormat::Badge, None);
1036
1037 assert_eq!(code, ExitCode::from(2));
1038 }
1039
1040 #[test]
1041 fn print_duplication_report_rejects_badge_format() {
1042 let root = Path::new("/project");
1043 let rules = RulesConfig::default();
1044 let ctx = test_context(root, &rules);
1045
1046 let code =
1047 print_duplication_report(&DuplicationReport::default(), &ctx, OutputFormat::Badge);
1048
1049 assert_eq!(code, ExitCode::from(2));
1050 }
1051
1052 #[test]
1053 fn elide_common_prefix_shared_dir() {
1054 assert_eq!(
1055 elide_common_prefix("src/components/A.tsx", "src/components/B.tsx"),
1056 "B.tsx"
1057 );
1058 }
1059
1060 #[test]
1061 fn elide_common_prefix_partial_shared() {
1062 assert_eq!(
1063 elide_common_prefix("src/components/A.tsx", "src/utils/B.tsx"),
1064 "utils/B.tsx"
1065 );
1066 }
1067
1068 #[test]
1069 fn elide_common_prefix_no_shared() {
1070 assert_eq!(
1071 elide_common_prefix("pkg-a/src/A.tsx", "pkg-b/src/B.tsx"),
1072 "pkg-b/src/B.tsx"
1073 );
1074 }
1075
1076 #[test]
1077 fn elide_common_prefix_identical_files() {
1078 assert_eq!(elide_common_prefix("a/b/x.ts", "a/b/y.ts"), "y.ts");
1079 }
1080
1081 #[test]
1082 fn elide_common_prefix_no_dirs() {
1083 assert_eq!(elide_common_prefix("foo.ts", "bar.ts"), "bar.ts");
1084 }
1085
1086 #[test]
1087 fn elide_common_prefix_deep_monorepo() {
1088 assert_eq!(
1089 elide_common_prefix(
1090 "packages/rap/src/rap/components/SearchSelect/SearchSelect.tsx",
1091 "packages/rap/src/rap/components/SearchSelect/SearchSelectItem.tsx"
1092 ),
1093 "SearchSelectItem.tsx"
1094 );
1095 }
1096
1097 #[test]
1098 fn split_dir_filename_with_dir() {
1099 let (dir, file) = split_dir_filename("src/utils/index.ts");
1100 assert_eq!(dir, "src/utils/");
1101 assert_eq!(file, "index.ts");
1102 }
1103
1104 #[test]
1105 fn split_dir_filename_no_dir() {
1106 let (dir, file) = split_dir_filename("file.ts");
1107 assert_eq!(dir, "");
1108 assert_eq!(file, "file.ts");
1109 }
1110
1111 #[test]
1112 fn split_dir_filename_deeply_nested() {
1113 let (dir, file) = split_dir_filename("a/b/c/d/e.ts");
1114 assert_eq!(dir, "a/b/c/d/");
1115 assert_eq!(file, "e.ts");
1116 }
1117
1118 #[test]
1119 fn split_dir_filename_trailing_slash() {
1120 let (dir, file) = split_dir_filename("src/");
1121 assert_eq!(dir, "src/");
1122 assert_eq!(file, "");
1123 }
1124
1125 #[test]
1126 fn split_dir_filename_empty() {
1127 let (dir, file) = split_dir_filename("");
1128 assert_eq!(dir, "");
1129 assert_eq!(file, "");
1130 }
1131
1132 #[test]
1133 fn plural_zero_is_plural() {
1134 assert_eq!(plural(0), "s");
1135 }
1136
1137 #[test]
1138 fn plural_one_is_singular() {
1139 assert_eq!(plural(1), "");
1140 }
1141
1142 #[test]
1143 fn plural_two_is_plural() {
1144 assert_eq!(plural(2), "s");
1145 }
1146
1147 #[test]
1148 fn plural_large_number() {
1149 assert_eq!(plural(999), "s");
1150 }
1151
1152 #[test]
1153 fn elide_common_prefix_empty_base() {
1154 assert_eq!(elide_common_prefix("", "src/foo.ts"), "src/foo.ts");
1155 }
1156
1157 #[test]
1158 fn elide_common_prefix_empty_target() {
1159 assert_eq!(elide_common_prefix("src/foo.ts", ""), "");
1160 }
1161
1162 #[test]
1163 fn elide_common_prefix_both_empty() {
1164 assert_eq!(elide_common_prefix("", ""), "");
1165 }
1166
1167 #[test]
1168 fn elide_common_prefix_same_file_different_extension() {
1169 assert_eq!(
1170 elide_common_prefix("src/utils.ts", "src/utils.js"),
1171 "utils.js"
1172 );
1173 }
1174
1175 #[test]
1176 fn elide_common_prefix_partial_filename_match_not_stripped() {
1177 assert_eq!(
1178 elide_common_prefix("src/App.tsx", "src/AppUtils.tsx"),
1179 "AppUtils.tsx"
1180 );
1181 }
1182
1183 #[test]
1184 fn elide_common_prefix_identical_paths() {
1185 assert_eq!(elide_common_prefix("src/foo.ts", "src/foo.ts"), "foo.ts");
1186 }
1187
1188 #[test]
1189 fn split_dir_filename_single_slash() {
1190 let (dir, file) = split_dir_filename("/file.ts");
1191 assert_eq!(dir, "/");
1192 assert_eq!(file, "file.ts");
1193 }
1194
1195 #[test]
1196 fn emit_json_returns_success_for_valid_value() {
1197 let value = serde_json::json!({"key": "value"});
1198 let code = emit_json(&value, "test");
1199 assert_eq!(code, ExitCode::SUCCESS);
1200 }
1201
1202 mod proptests {
1203 use super::*;
1204 use proptest::prelude::*;
1205
1206 proptest! {
1207 #[test]
1209 fn split_dir_filename_reconstructs_path(path in "[a-zA-Z0-9_./\\-]{0,100}") {
1210 let (dir, file) = split_dir_filename(&path);
1211 let reconstructed = format!("{dir}{file}");
1212 prop_assert_eq!(
1213 reconstructed, path,
1214 "dir+file should reconstruct the original path"
1215 );
1216 }
1217
1218 #[test]
1220 fn plural_returns_empty_or_s(n: usize) {
1221 let result = plural(n);
1222 prop_assert!(
1223 result.is_empty() || result == "s",
1224 "plural should return \"\" or \"s\", got {:?}",
1225 result
1226 );
1227 }
1228
1229 #[test]
1231 fn plural_singular_only_for_one(n: usize) {
1232 let result = plural(n);
1233 if n == 1 {
1234 prop_assert_eq!(result, "", "plural(1) should be empty");
1235 } else {
1236 prop_assert_eq!(result, "s", "plural({}) should be \"s\"", n);
1237 }
1238 }
1239
1240 #[test]
1242 fn normalize_uri_no_backslashes(path in "[a-zA-Z0-9_.\\\\/ \\[\\]%-]{0,100}") {
1243 let result = normalize_uri(&path);
1244 prop_assert!(
1245 !result.contains('\\'),
1246 "Result should not contain backslashes: {result}"
1247 );
1248 }
1249
1250 #[test]
1252 fn normalize_uri_encodes_all_brackets(path in "[a-zA-Z0-9_./\\[\\]%-]{0,80}") {
1253 let result = normalize_uri(&path);
1254 prop_assert!(
1255 !result.contains('[') && !result.contains(']'),
1256 "Result should not contain raw brackets: {result}"
1257 );
1258 }
1259
1260 #[test]
1262 fn elide_common_prefix_returns_suffix_of_target(
1263 base in "[a-zA-Z0-9_./]{0,50}",
1264 target in "[a-zA-Z0-9_./]{0,50}",
1265 ) {
1266 let result = elide_common_prefix(&base, &target);
1267 prop_assert!(
1268 target.ends_with(result),
1269 "Result {:?} should be a suffix of target {:?}",
1270 result, target
1271 );
1272 }
1273
1274 #[test]
1276 fn relative_path_never_panics(
1277 root in "/[a-zA-Z0-9_/]{0,30}",
1278 suffix in "[a-zA-Z0-9_./]{0,30}",
1279 ) {
1280 let root_path = Path::new(&root);
1281 let full = PathBuf::from(format!("{root}/{suffix}"));
1282 let _ = relative_path(&full, root_path);
1283 }
1284 }
1285 }
1286}