1use std::path::{Path, PathBuf};
4
5use fallow_config::{AuthoredRule, LogicalGroup, LogicalGroupStatus, ResolvedBoundaryConfig};
6use fallow_output::{
7 ListEntryPointOutput, RootEnvelopeMode, WorkspaceInfo as WorkspaceOutputInfo, WorkspacesOutput,
8};
9use fallow_types::discover::{DiscoveredFile, EntryPoint};
10use rustc_hash::FxHashMap;
11
12use crate::{
13 AnalysisOptions, BoundariesListLogicalGroup, BoundariesListRule, BoundariesListZone,
14 BoundariesListing, ListJsonEnvelope, ListJsonOutputInput, ProgrammaticError,
15 analysis_context::changed_files_for_run, resolve_programmatic_analysis_context,
16 serialize_list_json_output,
17};
18
19type ProgrammaticResult<T> = Result<T, ProgrammaticError>;
20
21#[derive(Debug, Clone, Default)]
23pub struct ProjectInfoOptions {
24 pub analysis: AnalysisOptions,
25 pub entry_points: bool,
26 pub files: bool,
27 pub plugins: bool,
28 pub boundaries: bool,
29}
30
31#[derive(Debug, Clone, Default)]
33pub struct ListBoundariesOptions {
34 pub analysis: AnalysisOptions,
35}
36
37#[derive(Debug, Clone)]
39pub struct ProjectInfoProgrammaticOutput {
40 pub plugins: Option<Vec<String>>,
41 pub files: Option<Vec<String>>,
42 pub entry_points: Option<Vec<ListEntryPointOutput>>,
43 pub boundaries: Option<BoundariesListing>,
44 pub workspaces: Option<WorkspacesOutput<fallow_config::WorkspaceDiagnostic>>,
45 pub envelope: ListJsonEnvelope,
46 pub envelope_mode: RootEnvelopeMode,
47}
48
49pub fn serialize_project_info_programmatic_json(
55 output: ProjectInfoProgrammaticOutput,
56) -> ProgrammaticResult<serde_json::Value> {
57 serialize_list_json_output(
58 ListJsonOutputInput {
59 plugins: output.plugins,
60 files: output.files,
61 entry_points: output.entry_points,
62 boundaries: output.boundaries,
63 workspaces: output.workspaces,
64 },
65 output.envelope_mode,
66 output.envelope,
67 )
68 .map_err(|err| {
69 ProgrammaticError::new(format!("failed to serialize project info output: {err}"), 2)
70 .with_code("FALLOW_PROJECT_INFO_SERIALIZE_FAILED")
71 .with_context("project_info")
72 })
73}
74
75#[derive(Debug, Clone)]
77pub struct ListBoundariesProgrammaticOutput {
78 pub boundaries: BoundariesListing,
79 pub envelope_mode: RootEnvelopeMode,
80}
81
82pub fn serialize_list_boundaries_programmatic_json(
88 output: ListBoundariesProgrammaticOutput,
89) -> ProgrammaticResult<serde_json::Value> {
90 serialize_list_json_output(
91 ListJsonOutputInput::<BoundariesListing, serde_json::Value> {
92 plugins: None,
93 files: None,
94 entry_points: None,
95 boundaries: Some(output.boundaries),
96 workspaces: None,
97 },
98 output.envelope_mode,
99 ListJsonEnvelope::Boundaries,
100 )
101 .map_err(|err| {
102 ProgrammaticError::new(
103 format!("failed to serialize list boundaries output: {err}"),
104 2,
105 )
106 .with_code("FALLOW_LIST_BOUNDARIES_SERIALIZE_FAILED")
107 .with_context("list_boundaries")
108 })
109}
110
111#[derive(Debug, Clone)]
113pub struct BoundaryData {
114 pub zones: Vec<ZoneInfo>,
115 pub rules: Vec<RuleInfo>,
116 pub logical_groups: Vec<LogicalGroupInfo>,
117 pub is_empty: bool,
118}
119
120#[derive(Debug, Clone)]
121pub struct ZoneInfo {
122 pub name: String,
123 pub patterns: Vec<String>,
124 pub file_count: usize,
125}
126
127#[derive(Debug, Clone)]
128pub struct RuleInfo {
129 pub from: String,
130 pub allow: Vec<String>,
131}
132
133#[derive(Debug, Clone)]
135pub struct LogicalGroupInfo {
136 pub name: String,
137 pub children: Vec<String>,
138 pub auto_discover: Vec<String>,
139 pub authored_rule: Option<AuthoredRule>,
140 pub fallback_zone: Option<String>,
141 pub source_zone_index: usize,
142 pub status: LogicalGroupStatus,
143 pub file_count: usize,
144 pub child_file_count: usize,
145 pub fallback_file_count: usize,
146 pub merged_from: Option<Vec<usize>>,
147 pub original_zone_root: Option<String>,
148 pub child_source_indices: Vec<usize>,
149}
150
151pub fn run_list_boundaries(
158 options: &ListBoundariesOptions,
159) -> ProgrammaticResult<ListBoundariesProgrammaticOutput> {
160 let resolved = resolve_programmatic_analysis_context(&options.analysis)?;
161 resolved.install(|| {
162 let project_config = load_list_project_config(&resolved)?;
163 let session = fallow_engine::session::AnalysisSession::from_config(project_config);
164 let changed_files = changed_files_for_run(&resolved)?;
165 let discovered = scoped_discovered_files(session.files(), changed_files.as_ref());
166 let data = compute_boundary_data(session.config(), Some(&discovered));
167
168 Ok(ListBoundariesProgrammaticOutput {
169 boundaries: boundary_data_to_output(&data),
170 envelope_mode: RootEnvelopeMode::Tagged,
171 })
172 })
173}
174
175pub fn run_project_info(
182 options: &ProjectInfoOptions,
183) -> ProgrammaticResult<ProjectInfoProgrammaticOutput> {
184 let resolved = resolve_programmatic_analysis_context(&options.analysis)?;
185 resolved.install(|| {
186 let project_config = load_list_project_config(&resolved)?;
187 let session = fallow_engine::session::AnalysisSession::from_config(project_config);
188 let config = session.config();
189 let workspaces = session.workspaces();
190 let show_all = project_info_should_show_all(options);
191 let changed_files = changed_files_for_run(&resolved)?;
192 let discovered =
193 project_info_discovered_files(options, show_all, &session, changed_files.as_ref());
194 let discovered_ref = discovered.as_deref();
195
196 let plugin_result = collect_plugin_result(
197 resolved.root(),
198 config,
199 options,
200 show_all,
201 discovered_ref,
202 workspaces,
203 )?;
204 let entry_points = collect_entry_points(
205 config,
206 options,
207 show_all,
208 discovered_ref,
209 workspaces,
210 plugin_result.as_ref(),
211 );
212 let boundaries = options
213 .boundaries
214 .then(|| boundary_data_to_output(&compute_boundary_data(config, discovered_ref)));
215 let workspaces = if show_all {
216 Some(collect_workspace_output(
217 resolved.root(),
218 workspaces,
219 session.workspace_diagnostics(),
220 ))
221 } else {
222 None
223 };
224 let envelope = if boundaries.is_some() {
225 ListJsonEnvelope::Boundaries
226 } else {
227 ListJsonEnvelope::Plain
228 };
229
230 Ok(ProjectInfoProgrammaticOutput {
231 plugins: collect_plugins(options, show_all, plugin_result.as_ref()),
232 files: collect_files(options, show_all, discovered_ref, resolved.root()),
233 entry_points: entry_points
234 .map(|entries| entry_points_to_output(&entries, resolved.root())),
235 boundaries,
236 workspaces,
237 envelope,
238 envelope_mode: RootEnvelopeMode::Tagged,
239 })
240 })
241}
242
243fn project_info_discovered_files(
244 options: &ProjectInfoOptions,
245 show_all: bool,
246 session: &fallow_engine::session::AnalysisSession,
247 changed_files: Option<&rustc_hash::FxHashSet<PathBuf>>,
248) -> Option<Vec<DiscoveredFile>> {
249 let needs_discovery =
250 options.files || options.entry_points || options.boundaries || options.plugins || show_all;
251 needs_discovery.then(|| scoped_discovered_files(session.files(), changed_files))
252}
253
254fn scoped_discovered_files(
255 files: &[DiscoveredFile],
256 changed_files: Option<&rustc_hash::FxHashSet<PathBuf>>,
257) -> Vec<DiscoveredFile> {
258 let Some(changed_files) = changed_files else {
259 return files.to_vec();
260 };
261 files
262 .iter()
263 .filter(|file| changed_files.contains(&file.path))
264 .cloned()
265 .collect()
266}
267
268fn load_list_project_config(
269 resolved: &crate::ProgrammaticAnalysisContext,
270) -> ProgrammaticResult<fallow_engine::project_config::ProjectConfig> {
271 fallow_engine::project_config::config_for_project_analysis(
272 resolved.root(),
273 resolved.config_path().as_deref(),
274 fallow_engine::project_config::ProjectConfigOptions {
275 output: fallow_types::output_format::OutputFormat::Json,
276 no_cache: resolved.no_cache(),
277 threads: resolved.threads(),
278 production_override: resolved.production_override(),
279 quiet: true,
280 analysis: fallow_config::ProductionAnalysis::DeadCode,
281 },
282 )
283 .map_err(|err| {
284 ProgrammaticError::new(format!("failed to load config: {err}"), 2)
285 .with_code("FALLOW_CONFIG_LOAD_FAILED")
286 .with_context("analysis.configPath")
287 })
288}
289
290const fn project_info_should_show_all(options: &ProjectInfoOptions) -> bool {
291 !options.entry_points && !options.files && !options.plugins && !options.boundaries
292}
293
294fn collect_plugins(
295 options: &ProjectInfoOptions,
296 show_all: bool,
297 plugin_result: Option<&fallow_engine::plugins::AggregatedPluginResult>,
298) -> Option<Vec<String>> {
299 if options.plugins || show_all {
300 plugin_result.map(|plugin_result| plugin_result.active_plugins().to_vec())
301 } else {
302 None
303 }
304}
305
306fn collect_files(
307 options: &ProjectInfoOptions,
308 show_all: bool,
309 discovered: Option<&[DiscoveredFile]>,
310 root: &Path,
311) -> Option<Vec<String>> {
312 if options.files || show_all {
313 discovered.map(|files| {
314 files
315 .iter()
316 .map(|file| format_display_path(&file.path, root))
317 .collect()
318 })
319 } else {
320 None
321 }
322}
323
324fn collect_plugin_result(
325 root: &Path,
326 config: &fallow_config::ResolvedConfig,
327 options: &ProjectInfoOptions,
328 show_all: bool,
329 discovered: Option<&[DiscoveredFile]>,
330 workspaces: &[fallow_config::WorkspaceInfo],
331) -> ProgrammaticResult<Option<fallow_engine::plugins::AggregatedPluginResult>> {
332 if !(options.plugins || options.entry_points || show_all) {
333 return Ok(None);
334 }
335 let Some(files) = discovered else {
336 return Ok(None);
337 };
338 fallow_engine::list_inventory::collect_active_plugins(root, config, files, workspaces)
339 .map(Some)
340 .map_err(|err| match err {
341 fallow_engine::list_inventory::ListInventoryError::PluginRegex(errors) => {
342 ProgrammaticError::new(
343 fallow_engine::plugins::registry::format_plugin_regex_errors(&errors),
344 2,
345 )
346 .with_code("FALLOW_PLUGIN_REGEX_FAILED")
347 .with_context("project_info.plugins")
348 }
349 })
350}
351
352fn collect_entry_points(
353 config: &fallow_config::ResolvedConfig,
354 options: &ProjectInfoOptions,
355 show_all: bool,
356 discovered: Option<&[DiscoveredFile]>,
357 workspaces: &[fallow_config::WorkspaceInfo],
358 plugin_result: Option<&fallow_engine::plugins::AggregatedPluginResult>,
359) -> Option<Vec<EntryPoint>> {
360 if !(options.entry_points || show_all) {
361 return None;
362 }
363 let discovered = discovered?;
364 Some(fallow_engine::list_inventory::collect_entry_points(
365 config,
366 discovered,
367 workspaces,
368 plugin_result,
369 ))
370}
371
372fn entry_points_to_output(entries: &[EntryPoint], root: &Path) -> Vec<ListEntryPointOutput> {
373 entries
374 .iter()
375 .map(|entry| ListEntryPointOutput {
376 path: format_display_path(&entry.path, root),
377 source: entry.source.to_string(),
378 })
379 .collect()
380}
381
382fn collect_workspace_output(
383 root: &Path,
384 workspaces: &[fallow_config::WorkspaceInfo],
385 diagnostics: &[fallow_config::WorkspaceDiagnostic],
386) -> WorkspacesOutput<fallow_config::WorkspaceDiagnostic> {
387 let workspaces = workspaces
388 .iter()
389 .map(|workspace| {
390 let relative = workspace.root.strip_prefix(root).unwrap_or(&workspace.root);
391 WorkspaceOutputInfo {
392 name: workspace.name.clone(),
393 path: relative.display().to_string().replace('\\', "/"),
394 is_internal_dependency: workspace.is_internal_dependency,
395 }
396 })
397 .collect::<Vec<_>>();
398 WorkspacesOutput {
399 workspace_count: workspaces.len(),
400 workspaces,
401 workspace_diagnostics: diagnostics.to_vec(),
402 }
403}
404
405fn format_display_path(path: &Path, root: &Path) -> String {
406 path.strip_prefix(root)
407 .unwrap_or(path)
408 .display()
409 .to_string()
410 .replace('\\', "/")
411}
412
413#[must_use]
415pub fn compute_boundary_data(
416 config: &fallow_config::ResolvedConfig,
417 discovered: Option<&[DiscoveredFile]>,
418) -> BoundaryData {
419 let boundaries = &config.boundaries;
420
421 if boundaries.is_empty() {
422 return BoundaryData {
423 zones: vec![],
424 rules: vec![],
425 logical_groups: vec![],
426 is_empty: true,
427 };
428 }
429
430 let zones = build_boundary_zones(config, discovered);
431 let rules = build_boundary_rules(boundaries);
432 let logical_groups = build_logical_groups(boundaries, &zones);
433
434 BoundaryData {
435 zones,
436 rules,
437 logical_groups,
438 is_empty: false,
439 }
440}
441
442fn build_boundary_zones(
443 config: &fallow_config::ResolvedConfig,
444 discovered: Option<&[DiscoveredFile]>,
445) -> Vec<ZoneInfo> {
446 config
447 .boundaries
448 .zones
449 .iter()
450 .map(|zone| ZoneInfo {
451 name: zone.name.clone(),
452 patterns: zone.matchers.iter().map(|m| m.glob().to_string()).collect(),
453 file_count: count_boundary_zone_files(config, discovered, &zone.name),
454 })
455 .collect()
456}
457
458fn count_boundary_zone_files(
459 config: &fallow_config::ResolvedConfig,
460 discovered: Option<&[DiscoveredFile]>,
461 zone_name: &str,
462) -> usize {
463 discovered.map_or(0, |files| {
464 files
465 .iter()
466 .filter(|file| {
467 let rel = file
468 .path
469 .strip_prefix(&config.root)
470 .ok()
471 .map(|path| path.to_string_lossy().replace('\\', "/"));
472 rel.is_some_and(|path| config.boundaries.classify_zone(&path) == Some(zone_name))
473 })
474 .count()
475 })
476}
477
478fn build_boundary_rules(boundaries: &ResolvedBoundaryConfig) -> Vec<RuleInfo> {
479 boundaries
480 .rules
481 .iter()
482 .map(|rule| RuleInfo {
483 from: rule.from_zone.clone(),
484 allow: rule.allowed_zones.clone(),
485 })
486 .collect()
487}
488
489fn build_logical_groups(
490 boundaries: &ResolvedBoundaryConfig,
491 zones: &[ZoneInfo],
492) -> Vec<LogicalGroupInfo> {
493 let zone_count_by_name: FxHashMap<&str, usize> = zones
494 .iter()
495 .map(|zone| (zone.name.as_str(), zone.file_count))
496 .collect();
497
498 boundaries
499 .logical_groups
500 .iter()
501 .map(|group| logical_group_info(group, &zone_count_by_name))
502 .collect()
503}
504
505fn logical_group_info(
506 group: &LogicalGroup,
507 zone_count_by_name: &FxHashMap<&str, usize>,
508) -> LogicalGroupInfo {
509 let child_file_count: usize = group
510 .children
511 .iter()
512 .filter_map(|child| zone_count_by_name.get(child.as_str()).copied())
513 .sum();
514 let fallback_file_count = group
515 .fallback_zone
516 .as_deref()
517 .and_then(|fallback| zone_count_by_name.get(fallback).copied())
518 .unwrap_or(0);
519
520 LogicalGroupInfo {
521 name: group.name.clone(),
522 children: group.children.clone(),
523 auto_discover: group.auto_discover.clone(),
524 authored_rule: group.authored_rule.clone(),
525 fallback_zone: group.fallback_zone.clone(),
526 source_zone_index: group.source_zone_index,
527 status: group.status,
528 file_count: child_file_count + fallback_file_count,
529 child_file_count,
530 fallback_file_count,
531 merged_from: group.merged_from.clone(),
532 original_zone_root: group.original_zone_root.clone(),
533 child_source_indices: group.child_source_indices.clone(),
534 }
535}
536
537#[must_use]
539pub fn boundary_data_to_output(data: &BoundaryData) -> BoundariesListing {
540 if data.is_empty {
541 return BoundariesListing {
542 configured: false,
543 zone_count: 0,
544 zones: Vec::new(),
545 rule_count: 0,
546 rules: Vec::new(),
547 logical_group_count: 0,
548 logical_groups: Vec::new(),
549 };
550 }
551
552 BoundariesListing {
553 configured: true,
554 zone_count: data.zones.len(),
555 zones: data
556 .zones
557 .iter()
558 .map(|zone| BoundariesListZone {
559 name: zone.name.clone(),
560 patterns: zone.patterns.clone(),
561 file_count: zone.file_count,
562 })
563 .collect(),
564 rule_count: data.rules.len(),
565 rules: data
566 .rules
567 .iter()
568 .map(|rule| BoundariesListRule {
569 from: rule.from.clone(),
570 allow: rule.allow.clone(),
571 })
572 .collect(),
573 logical_group_count: data.logical_groups.len(),
574 logical_groups: data
575 .logical_groups
576 .iter()
577 .map(logical_group_info_to_output)
578 .collect(),
579 }
580}
581
582fn logical_group_info_to_output(group: &LogicalGroupInfo) -> BoundariesListLogicalGroup {
583 BoundariesListLogicalGroup {
584 name: group.name.clone(),
585 children: group.children.clone(),
586 auto_discover: group.auto_discover.clone(),
587 status: group.status,
588 source_zone_index: group.source_zone_index,
589 file_count: group.file_count,
590 authored_rule: group.authored_rule.clone(),
591 fallback_zone: group.fallback_zone.clone(),
592 merged_from: group.merged_from.clone(),
593 original_zone_root: group.original_zone_root.clone(),
594 child_source_indices: group.child_source_indices.clone(),
595 }
596}
597
598#[cfg(test)]
599mod tests {
600 use std::process::Command;
601
602 use serde_json::json;
603
604 use super::*;
605
606 fn empty_boundary_data() -> BoundaryData {
607 BoundaryData {
608 zones: vec![],
609 rules: vec![],
610 logical_groups: vec![],
611 is_empty: true,
612 }
613 }
614
615 fn boundary_data_to_json(data: &BoundaryData) -> serde_json::Value {
616 serde_json::to_value(boundary_data_to_output(data))
617 .expect("boundary list output should serialize")
618 }
619
620 fn git(project: &Path, args: &[&str]) {
621 let status = Command::new("git")
622 .args(args)
623 .current_dir(project)
624 .status()
625 .expect("git command should run");
626 assert!(status.success(), "git {args:?} failed");
627 }
628
629 fn setup_changed_boundary_project() -> tempfile::TempDir {
630 let project = tempfile::tempdir().expect("project");
631 std::fs::write(
632 project.path().join("package.json"),
633 r#"{"name":"changed-list-api","main":"src/app/index.ts"}"#,
634 )
635 .expect("write package");
636 std::fs::write(
637 project.path().join(".fallowrc.json"),
638 r#"{
639 "boundaries": {
640 "zones": [
641 { "name": "app", "patterns": ["src/app/**"] },
642 { "name": "shared", "patterns": ["src/shared/**"] }
643 ]
644 }
645 }"#,
646 )
647 .expect("write config");
648 std::fs::create_dir_all(project.path().join("src/app")).expect("create app");
649 std::fs::create_dir_all(project.path().join("src/shared")).expect("create shared");
650 std::fs::write(
651 project.path().join("src/app/index.ts"),
652 "export const app = 1;\n",
653 )
654 .expect("write app");
655 std::fs::write(
656 project.path().join("src/shared/index.ts"),
657 "export const shared = 1;\n",
658 )
659 .expect("write shared");
660
661 git(project.path(), &["init", "-q"]);
662 git(
663 project.path(),
664 &["config", "user.email", "test@example.com"],
665 );
666 git(project.path(), &["config", "user.name", "Test User"]);
667 git(project.path(), &["config", "commit.gpgsign", "false"]);
668 git(project.path(), &["add", "."]);
669 git(project.path(), &["commit", "-qm", "initial"]);
670 std::fs::write(
671 project.path().join("src/app/index.ts"),
672 "export const app = 2;\n",
673 )
674 .expect("modify app");
675 project
676 }
677
678 #[test]
679 fn project_info_default_sections_match_plain_list_contract() {
680 let project = tempfile::tempdir().expect("project");
681 std::fs::write(
682 project.path().join("package.json"),
683 r#"{"name":"project-info-api","main":"src/index.ts"}"#,
684 )
685 .expect("write package");
686 std::fs::create_dir_all(project.path().join("src")).expect("create src");
687 std::fs::write(
688 project.path().join("src/index.ts"),
689 "export const value = 1;\n",
690 )
691 .expect("write source");
692
693 let output = serialize_project_info_programmatic_json(
694 run_project_info(&ProjectInfoOptions {
695 analysis: AnalysisOptions {
696 root: Some(project.path().to_path_buf()),
697 no_cache: true,
698 ..AnalysisOptions::default()
699 },
700 ..ProjectInfoOptions::default()
701 })
702 .expect("project info should run"),
703 )
704 .expect("project info should serialize");
705
706 assert_eq!(output["file_count"], 1);
707 assert_eq!(output["files"][0], "src/index.ts");
708 assert_eq!(output["entry_point_count"], 1);
709 assert_eq!(output["workspace_count"], 0);
710 assert!(output.get("kind").is_none());
711 }
712
713 #[test]
714 fn project_info_surfaces_malformed_root_package_json() {
715 let project = tempfile::tempdir().expect("project");
716 std::fs::write(project.path().join("package.json"), "{").expect("write package");
717
718 let err = run_project_info(&ProjectInfoOptions {
719 analysis: AnalysisOptions {
720 root: Some(project.path().to_path_buf()),
721 no_cache: true,
722 ..AnalysisOptions::default()
723 },
724 ..ProjectInfoOptions::default()
725 })
726 .expect_err("malformed root package.json must fail project info");
727
728 assert_eq!(err.exit_code, 2);
729 assert_eq!(err.code.as_deref(), Some("FALLOW_CONFIG_LOAD_FAILED"));
730 assert!(
731 err.message.contains("package.json"),
732 "error should name the malformed root package.json"
733 );
734 }
735
736 #[test]
737 fn project_info_default_sections_include_undeclared_workspace_diagnostic() {
738 let project = tempfile::tempdir().expect("project");
739 std::fs::write(
740 project.path().join("package.json"),
741 r#"{"name":"project-info-api","workspaces":["packages/*"]}"#,
742 )
743 .expect("write package");
744 std::fs::create_dir_all(project.path().join("packages/app")).expect("workspace dir");
745 std::fs::write(
746 project.path().join("packages/app/package.json"),
747 r#"{"name":"app","main":"src/index.ts"}"#,
748 )
749 .expect("write workspace package");
750 std::fs::create_dir_all(project.path().join("tools/extra")).expect("extra package dir");
751 std::fs::write(
752 project.path().join("tools/extra/package.json"),
753 r#"{"name":"extra"}"#,
754 )
755 .expect("write extra package");
756
757 let output = serialize_project_info_programmatic_json(
758 run_project_info(&ProjectInfoOptions {
759 analysis: AnalysisOptions {
760 root: Some(project.path().to_path_buf()),
761 no_cache: true,
762 ..AnalysisOptions::default()
763 },
764 ..ProjectInfoOptions::default()
765 })
766 .expect("project info should run"),
767 )
768 .expect("project info should serialize");
769
770 let diagnostics = output["workspace_diagnostics"]
771 .as_array()
772 .expect("project info should include workspace_diagnostics");
773 assert!(
774 diagnostics.iter().any(|diagnostic| {
775 diagnostic["kind"].as_str() == Some("undeclared-workspace")
776 && diagnostic["path"]
777 .as_str()
778 .is_some_and(|path| path.replace('\\', "/").ends_with("/tools/extra"))
779 }),
780 "project info must include undeclared workspace diagnostics from the reused session, got {diagnostics:#?}"
781 );
782 }
783
784 #[test]
785 fn list_runtimes_scope_files_and_boundary_counts_to_changed_since() {
786 let project = setup_changed_boundary_project();
787 let analysis = AnalysisOptions {
788 root: Some(project.path().to_path_buf()),
789 changed_since: Some("HEAD".to_string()),
790 no_cache: true,
791 ..AnalysisOptions::default()
792 };
793
794 let project_info = serialize_project_info_programmatic_json(
795 run_project_info(&ProjectInfoOptions {
796 analysis: analysis.clone(),
797 files: true,
798 boundaries: true,
799 ..ProjectInfoOptions::default()
800 })
801 .expect("project info should run"),
802 )
803 .expect("project info should serialize");
804 let files = project_info["files"].as_array().expect("files array");
805 assert_eq!(files, &[json!("src/app/index.ts")]);
806 assert_eq!(project_info["boundaries"]["zones"][0]["file_count"], 1);
807 assert_eq!(project_info["boundaries"]["zones"][1]["file_count"], 0);
808
809 let boundaries = serialize_list_boundaries_programmatic_json(
810 run_list_boundaries(&ListBoundariesOptions { analysis })
811 .expect("list boundaries should run"),
812 )
813 .expect("list boundaries should serialize");
814 assert_eq!(boundaries["boundaries"]["zones"][0]["file_count"], 1);
815 assert_eq!(boundaries["boundaries"]["zones"][1]["file_count"], 0);
816 }
817
818 #[test]
819 fn boundary_json_empty_includes_logical_groups_key() {
820 let value = boundary_data_to_json(&empty_boundary_data());
821
822 assert_eq!(value["configured"], false);
823 assert_eq!(value["zone_count"], 0);
824 assert_eq!(value["rule_count"], 0);
825 assert_eq!(value["logical_group_count"], 0);
826 assert_eq!(value["logical_groups"], json!([]));
827 }
828
829 #[test]
830 fn boundary_json_logical_group_carries_all_fields() {
831 let data = BoundaryData {
832 zones: vec![ZoneInfo {
833 name: "features/auth".to_string(),
834 patterns: vec!["src/features/auth/**".to_string()],
835 file_count: 3,
836 }],
837 rules: vec![],
838 logical_groups: vec![LogicalGroupInfo {
839 name: "features".to_string(),
840 children: vec!["features/auth".to_string()],
841 auto_discover: vec!["./src/features/".to_string()],
842 authored_rule: Some(AuthoredRule {
843 allow: vec!["shared".to_string()],
844 allow_type_only: vec!["types".to_string()],
845 }),
846 fallback_zone: None,
847 source_zone_index: 1,
848 status: LogicalGroupStatus::Ok,
849 file_count: 3,
850 child_file_count: 3,
851 fallback_file_count: 0,
852 merged_from: None,
853 original_zone_root: None,
854 child_source_indices: vec![],
855 }],
856 is_empty: false,
857 };
858
859 let value = boundary_data_to_json(&data);
860 let group = &value["logical_groups"][0];
861
862 assert_eq!(value["logical_group_count"], 1);
863 assert_eq!(group["name"], "features");
864 assert_eq!(group["children"][0], "features/auth");
865 assert_eq!(group["auto_discover"][0], "./src/features/");
866 assert_eq!(group["status"], "ok");
867 assert_eq!(group["source_zone_index"], 1);
868 assert_eq!(group["file_count"], 3);
869 assert_eq!(group["authored_rule"]["allow"][0], "shared");
870 assert_eq!(group["authored_rule"]["allow_type_only"][0], "types");
871 assert!(group.get("fallback_zone").is_none());
872 assert!(group.get("merged_from").is_none());
873 assert!(group.get("original_zone_root").is_none());
874 assert!(group.get("child_source_indices").is_none());
875 }
876
877 #[test]
878 fn boundary_json_logical_group_optional_fields_round_trip() {
879 let data = BoundaryData {
880 zones: vec![],
881 rules: vec![],
882 logical_groups: vec![LogicalGroupInfo {
883 name: "features".to_string(),
884 children: vec!["features/auth".to_string(), "features/billing".to_string()],
885 auto_discover: vec!["src/features".to_string(), "src/modules".to_string()],
886 authored_rule: None,
887 fallback_zone: Some("features".to_string()),
888 source_zone_index: 0,
889 status: LogicalGroupStatus::Empty,
890 file_count: 2,
891 child_file_count: 0,
892 fallback_file_count: 2,
893 merged_from: Some(vec![0, 3]),
894 original_zone_root: Some("packages/app/".to_string()),
895 child_source_indices: vec![0, 1],
896 }],
897 is_empty: false,
898 };
899
900 let group = &boundary_data_to_json(&data)["logical_groups"][0];
901
902 assert_eq!(group["status"], "empty");
903 assert_eq!(group["fallback_zone"], "features");
904 assert_eq!(group["merged_from"][1], 3);
905 assert_eq!(group["original_zone_root"], "packages/app/");
906 assert_eq!(group["child_source_indices"][1], 1);
907 }
908}