Skip to main content

fallow_api/
list_runtime.rs

1//! Programmatic list-command runtime helpers.
2
3use 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/// Options for MCP/project metadata listing through the programmatic API.
22#[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/// Options for `fallow list --boundaries` through the programmatic API.
32#[derive(Debug, Clone, Default)]
33pub struct ListBoundariesOptions {
34    pub analysis: AnalysisOptions,
35}
36
37/// Typed output for project metadata listing before JSON serialization.
38#[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
49/// Serialize typed project-info output to the stable JSON contract.
50///
51/// # Errors
52///
53/// Returns a structured programmatic error when JSON serialization fails.
54pub 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/// Typed output for `fallow list --boundaries` before JSON serialization.
76#[derive(Debug, Clone)]
77pub struct ListBoundariesProgrammaticOutput {
78    pub boundaries: BoundariesListing,
79    pub envelope_mode: RootEnvelopeMode,
80}
81
82/// Serialize typed boundary-list output to the stable JSON contract.
83///
84/// # Errors
85///
86/// Returns a structured programmatic error when JSON serialization fails.
87pub 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/// Owned boundary listing data shared by CLI and programmatic renderers.
112#[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/// View-model mirror of [`LogicalGroup`] with derived file-count totals.
134#[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
151/// Run `list_boundaries` through the API-owned runtime path.
152///
153/// # Errors
154///
155/// Returns a structured programmatic error for invalid options or config-load
156/// failures.
157pub 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
175/// Run project metadata listing through the API-owned runtime path.
176///
177/// # Errors
178///
179/// Returns a structured programmatic error for invalid options, config-load
180/// failures, or plugin regex errors.
181pub 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/// Compute boundary listing data from resolved config and optional discovery.
414#[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/// Convert boundary listing data to the stable output contract.
538#[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}