Skip to main content

fallow_core/discover/
entry_points.rs

1use std::path::{Component, Path, PathBuf};
2
3use super::parse_scripts::extract_script_file_refs;
4use super::walk::SOURCE_EXTENSIONS;
5use fallow_config::{EntryPointRole, PackageJson, ResolvedConfig};
6use fallow_types::discover::{DiscoveredFile, EntryPoint, EntryPointSource};
7use fallow_types::path_util::is_absolute_path_any_platform;
8use rustc_hash::{FxHashMap, FxHashSet};
9
10/// Known output directory names from exports maps.
11/// When an entry point path is inside one of these directories, we also try
12/// the `src/` equivalent to find the tracked source file.
13const OUTPUT_DIRS: &[&str] = &["dist", "build", "out", "esm", "cjs"];
14const SKIPPED_ENTRY_WARNING_PREVIEW: usize = 5;
15
16fn format_skipped_entry_warning(skipped_entries: &FxHashMap<String, usize>) -> Option<String> {
17    if skipped_entries.is_empty() {
18        return None;
19    }
20
21    let mut entries = skipped_entries
22        .iter()
23        .map(|(path, count)| (path.as_str(), *count))
24        .collect::<Vec<_>>();
25    entries.sort_by(|a, b| b.1.cmp(&a.1).then_with(|| a.0.cmp(b.0)));
26
27    let preview = entries
28        .iter()
29        .take(SKIPPED_ENTRY_WARNING_PREVIEW)
30        .map(|(path, count)| {
31            if *count > 1 {
32                format!("{path} ({count}x)")
33            } else {
34                (*path).to_owned()
35            }
36        })
37        .collect::<Vec<_>>();
38
39    let omitted = entries.len().saturating_sub(SKIPPED_ENTRY_WARNING_PREVIEW);
40    let tail = if omitted > 0 {
41        format!(" (and {omitted} more)")
42    } else {
43        String::new()
44    };
45    let total = entries.iter().map(|(_, count)| *count).sum::<usize>();
46    let noun = if total == 1 {
47        "package.json entry point"
48    } else {
49        "package.json entry points"
50    };
51
52    Some(format!(
53        "Skipped {total} {noun} outside project root or containing parent directory traversal: {}{tail}",
54        preview.join(", ")
55    ))
56}
57
58pub fn warn_skipped_entry_summary(skipped_entries: &FxHashMap<String, usize>) {
59    let Some(message) = format_skipped_entry_warning(skipped_entries) else {
60        return;
61    };
62    if should_warn_skipped_entry(&message) {
63        tracing::warn!("{message}");
64    }
65}
66
67/// Process-wide dedupe for [`warn_skipped_entry_summary`]. Returns `true` when
68/// `message` was newly inserted (caller should emit). On a poisoned mutex
69/// returns `true` so over-warning beats swallowing.
70fn should_warn_skipped_entry(message: &str) -> bool {
71    static WARNED: std::sync::OnceLock<std::sync::Mutex<FxHashSet<String>>> =
72        std::sync::OnceLock::new();
73    let warned = WARNED.get_or_init(|| std::sync::Mutex::new(FxHashSet::default()));
74    warned
75        .lock()
76        .map_or(true, |mut set| set.insert(message.to_owned()))
77}
78
79/// Entry points grouped by reachability role.
80#[derive(Debug, Clone, Default)]
81pub struct CategorizedEntryPoints {
82    pub all: Vec<EntryPoint>,
83    pub runtime: Vec<EntryPoint>,
84    pub test: Vec<EntryPoint>,
85}
86
87impl CategorizedEntryPoints {
88    pub fn push_runtime(&mut self, entry: EntryPoint) {
89        self.runtime.push(entry.clone());
90        self.all.push(entry);
91    }
92
93    pub fn push_test(&mut self, entry: EntryPoint) {
94        self.test.push(entry.clone());
95        self.all.push(entry);
96    }
97
98    pub fn push_support(&mut self, entry: EntryPoint) {
99        self.all.push(entry);
100    }
101
102    pub fn extend_runtime<I>(&mut self, entries: I)
103    where
104        I: IntoIterator<Item = EntryPoint>,
105    {
106        for entry in entries {
107            self.push_runtime(entry);
108        }
109    }
110
111    pub fn extend_test<I>(&mut self, entries: I)
112    where
113        I: IntoIterator<Item = EntryPoint>,
114    {
115        for entry in entries {
116            self.push_test(entry);
117        }
118    }
119
120    pub fn extend_support<I>(&mut self, entries: I)
121    where
122        I: IntoIterator<Item = EntryPoint>,
123    {
124        for entry in entries {
125            self.push_support(entry);
126        }
127    }
128
129    pub fn extend(&mut self, other: Self) {
130        self.all.extend(other.all);
131        self.runtime.extend(other.runtime);
132        self.test.extend(other.test);
133    }
134
135    #[must_use]
136    pub fn dedup(mut self) -> Self {
137        dedup_entry_paths(&mut self.all);
138        dedup_entry_paths(&mut self.runtime);
139        dedup_entry_paths(&mut self.test);
140        self
141    }
142}
143
144fn dedup_entry_paths(entries: &mut Vec<EntryPoint>) {
145    entries.sort_by(|a, b| a.path.cmp(&b.path));
146    entries.dedup_by(|a, b| a.path == b.path);
147}
148
149#[derive(Debug, Default)]
150pub struct EntryPointDiscovery {
151    pub entries: Vec<EntryPoint>,
152    pub skipped_entries: FxHashMap<String, usize>,
153}
154
155/// Resolve a path relative to a base directory, with security check and extension fallback.
156///
157/// Returns `Some(EntryPoint)` if the path resolves to an existing file within `canonical_root`,
158/// trying source extensions as fallback when the exact path doesn't exist.
159/// Also handles exports map targets in output directories (e.g., `./dist/utils.js`)
160/// by trying to map back to the source file (e.g., `./src/utils.ts`).
161fn resolve_entry_path_with_tracking(
162    base: &Path,
163    entry: &str,
164    canonical_root: &Path,
165    source: EntryPointSource,
166    mut skipped_entries: Option<&mut FxHashMap<String, usize>>,
167) -> Option<EntryPoint> {
168    if entry.contains('*') {
169        return None;
170    }
171
172    if entry_has_parent_dir(entry) {
173        record_or_warn_skipped_entry(
174            skipped_entries.as_deref_mut(),
175            entry,
176            "Skipping entry point containing parent directory traversal",
177        );
178        return None;
179    }
180
181    if let OutputDirEntry::ShortCircuit(result) = resolve_entry_via_output_dir(
182        base,
183        entry,
184        canonical_root,
185        source.clone(),
186        skipped_entries.as_deref_mut(),
187    ) {
188        return result;
189    }
190
191    resolve_entry_via_filesystem_probe(base, entry, canonical_root, source, skipped_entries)
192}
193
194/// Record a skipped entry in the dedup map, or warn when no map is tracking skips.
195fn record_or_warn_skipped_entry(
196    skipped_entries: Option<&mut FxHashMap<String, usize>>,
197    entry: &str,
198    warning: &str,
199) {
200    if let Some(skipped_entries) = skipped_entries {
201        *skipped_entries.entry(entry.to_owned()).or_default() += 1;
202    } else {
203        tracing::warn!(path = %entry, "{warning}");
204    }
205}
206
207/// Outcome of the output-directory mapping step.
208///
209/// `ShortCircuit` means an output-dir branch applied and carries the resolved
210/// entry (which may itself be `None` when validation rejected the candidate);
211/// `Continue` signals that filesystem probing should proceed.
212enum OutputDirEntry {
213    ShortCircuit(Option<EntryPoint>),
214    Continue,
215}
216
217/// Map an output-directory entry back to a source file, short-circuiting resolution.
218fn resolve_entry_via_output_dir(
219    base: &Path,
220    entry: &str,
221    canonical_root: &Path,
222    source: EntryPointSource,
223    mut skipped_entries: Option<&mut FxHashMap<String, usize>>,
224) -> OutputDirEntry {
225    if let Some(source_path) = try_output_to_source_path(base, entry) {
226        return OutputDirEntry::ShortCircuit(validated_entry_point(
227            &source_path,
228            canonical_root,
229            entry,
230            source,
231            skipped_entries.as_deref_mut(),
232        ));
233    }
234
235    if is_entry_in_output_dir(entry)
236        && let Some(source_path) = try_source_index_fallback(base)
237    {
238        tracing::info!(
239            entry = %entry,
240            fallback = %source_path.display(),
241            "package.json entry resolves to an ignored output directory; falling back to source index"
242        );
243        return OutputDirEntry::ShortCircuit(validated_entry_point(
244            &source_path,
245            canonical_root,
246            entry,
247            source,
248            skipped_entries,
249        ));
250    }
251
252    OutputDirEntry::Continue
253}
254
255/// Probe the filesystem for the entry: exact file, extension fallback, directory
256/// index, then a package-root source-index fallback.
257fn resolve_entry_via_filesystem_probe(
258    base: &Path,
259    entry: &str,
260    canonical_root: &Path,
261    source: EntryPointSource,
262    mut skipped_entries: Option<&mut FxHashMap<String, usize>>,
263) -> Option<EntryPoint> {
264    let resolved = base.join(entry);
265
266    if resolved.is_file() {
267        return validated_entry_point(
268            &resolved,
269            canonical_root,
270            entry,
271            source,
272            skipped_entries.as_deref_mut(),
273        );
274    }
275
276    for ext in SOURCE_EXTENSIONS {
277        let with_ext = resolved.with_extension(ext);
278        if with_ext.is_file() {
279            return validated_entry_point(
280                &with_ext,
281                canonical_root,
282                entry,
283                source,
284                skipped_entries.as_deref_mut(),
285            );
286        }
287    }
288
289    if let Some(index_entry) = try_directory_index_entry(&resolved) {
290        return validated_entry_point(
291            &index_entry,
292            canonical_root,
293            entry,
294            source,
295            skipped_entries.as_deref_mut(),
296        );
297    }
298
299    if is_package_root_index_entry(entry)
300        && let Some(source_path) = try_source_index_fallback(base)
301    {
302        tracing::info!(
303            entry = %entry,
304            fallback = %source_path.display(),
305            "package.json root index entry is missing; falling back to source index"
306        );
307        return validated_entry_point(&source_path, canonical_root, entry, source, skipped_entries);
308    }
309    None
310}
311
312fn try_directory_index_entry(resolved: &Path) -> Option<PathBuf> {
313    for ext in SOURCE_EXTENSIONS {
314        let candidate = resolved.join(format!("index.{ext}"));
315        if candidate.is_file() {
316            return Some(candidate);
317        }
318    }
319    None
320}
321
322fn entry_has_parent_dir(entry: &str) -> bool {
323    Path::new(entry)
324        .components()
325        .any(|component| matches!(component, Component::ParentDir))
326}
327
328fn is_package_root_index_entry(entry: &str) -> bool {
329    let mut components = Path::new(entry)
330        .components()
331        .filter(|component| !matches!(component, Component::CurDir));
332
333    let Some(Component::Normal(file_name)) = components.next() else {
334        return false;
335    };
336    if components.next().is_some() {
337        return false;
338    }
339
340    file_name
341        .to_str()
342        .is_some_and(|name| name == "index" || name.starts_with("index."))
343}
344
345fn validated_entry_point(
346    candidate: &Path,
347    canonical_root: &Path,
348    entry: &str,
349    source: EntryPointSource,
350    mut skipped_entries: Option<&mut FxHashMap<String, usize>>,
351) -> Option<EntryPoint> {
352    let canonical_candidate = match dunce::canonicalize(candidate) {
353        Ok(path) => path,
354        Err(err) => {
355            tracing::warn!(
356                path = %candidate.display(),
357                %entry,
358                error = %err,
359                "Skipping entry point that could not be canonicalized"
360            );
361            return None;
362        }
363    };
364
365    if !canonical_candidate.starts_with(canonical_root) {
366        if let Some(skipped_entries) = skipped_entries.as_mut() {
367            *skipped_entries.entry(entry.to_owned()).or_default() += 1;
368        } else {
369            tracing::warn!(
370                path = %candidate.display(),
371                %entry,
372                "Skipping entry point outside project root"
373            );
374        }
375        return None;
376    }
377
378    Some(EntryPoint {
379        path: candidate.to_path_buf(),
380        source,
381    })
382}
383
384pub fn resolve_entry_path(
385    base: &Path,
386    entry: &str,
387    canonical_root: &Path,
388    source: EntryPointSource,
389) -> Option<EntryPoint> {
390    resolve_entry_path_with_tracking(base, entry, canonical_root, source, None)
391}
392/// Try to map an entry path from an output directory to its source equivalent.
393///
394/// Given `base=/project/packages/ui` and `entry=./dist/utils.js`, this tries:
395/// - `/project/packages/ui/src/utils.ts`
396/// - `/project/packages/ui/src/utils.tsx`
397/// - etc. for all source extensions
398///
399/// Preserves any path prefix between the package root and the output dir,
400/// e.g. `./modules/dist/utils.js` → `base/modules/src/utils.ts`.
401///
402/// Returns `Some(path)` if a source file is found.
403fn try_output_to_source_path(base: &Path, entry: &str) -> Option<PathBuf> {
404    let entry_path = Path::new(entry);
405    let components: Vec<_> = entry_path.components().collect();
406
407    let output_pos = components.iter().rposition(|c| {
408        if let std::path::Component::Normal(s) = c
409            && let Some(name) = s.to_str()
410        {
411            return OUTPUT_DIRS.contains(&name);
412        }
413        false
414    })?;
415
416    let prefix: PathBuf = components[..output_pos]
417        .iter()
418        .filter(|c| !matches!(c, std::path::Component::CurDir))
419        .collect();
420
421    let suffix: PathBuf = components[output_pos + 1..].iter().collect();
422
423    for ext in SOURCE_EXTENSIONS {
424        let source_candidate = base
425            .join(&prefix)
426            .join("src")
427            .join(suffix.with_extension(ext));
428        if source_candidate.exists() {
429            return Some(source_candidate);
430        }
431    }
432
433    None
434}
435
436/// Conventional source index file stems probed when a package.json entry lives
437/// in an ignored output directory. Ordered by preference.
438const SOURCE_INDEX_FALLBACK_STEMS: &[&str] = &["src/index", "src/main", "index", "main"];
439
440/// Return `true` when `entry` contains a known output directory component.
441///
442/// Matches any segment in `OUTPUT_DIRS`, e.g. `./dist/esm2022/index.js` →
443/// `true`, `src/main.ts` → `false`.
444fn is_entry_in_output_dir(entry: &str) -> bool {
445    Path::new(entry).components().any(|c| {
446        matches!(
447            c,
448            std::path::Component::Normal(s)
449                if s.to_str().is_some_and(|name| OUTPUT_DIRS.contains(&name))
450        )
451    })
452}
453
454/// Probe a package root for a conventional source index file.
455///
456/// Used when `package.json` points at compiled output but the canonical source
457/// entry is a standard TypeScript/JavaScript index file. Tries `src/index`,
458/// `src/main`, `index`, and `main` with each supported source extension, in
459/// that order.
460fn try_source_index_fallback(base: &Path) -> Option<PathBuf> {
461    for stem in SOURCE_INDEX_FALLBACK_STEMS {
462        for ext in SOURCE_EXTENSIONS {
463            let candidate = base.join(format!("{stem}.{ext}"));
464            if candidate.is_file() {
465                return Some(candidate);
466            }
467        }
468    }
469    None
470}
471
472/// Default index patterns used when no other entry points are found.
473const DEFAULT_INDEX_PATTERNS: &[&str] = &[
474    "src/index.{ts,tsx,js,jsx}",
475    "src/main.{ts,tsx,js,jsx}",
476    "index.{ts,tsx,js,jsx}",
477    "main.{ts,tsx,js,jsx}",
478];
479
480/// Fall back to default index patterns if no entries were found.
481///
482/// When `ws_filter` is `Some`, only files whose path starts with the given
483/// workspace root are considered (used for workspace-scoped discovery).
484fn apply_default_fallback(
485    files: &[DiscoveredFile],
486    root: &Path,
487    ws_filter: Option<&Path>,
488) -> Vec<EntryPoint> {
489    let default_matchers: Vec<globset::GlobMatcher> = DEFAULT_INDEX_PATTERNS
490        .iter()
491        .filter_map(|p| globset::Glob::new(p).ok().map(|g| g.compile_matcher()))
492        .collect();
493
494    let mut entries = Vec::new();
495    for file in files {
496        if let Some(ws_root) = ws_filter
497            && file.path.strip_prefix(ws_root).is_err()
498        {
499            continue;
500        }
501        let relative = file.path.strip_prefix(root).unwrap_or(&file.path);
502        let relative_str = relative.to_string_lossy();
503        if default_matchers
504            .iter()
505            .any(|m| m.is_match(relative_str.as_ref()))
506        {
507            entries.push(EntryPoint {
508                path: file.path.clone(),
509                source: EntryPointSource::DefaultIndex,
510            });
511        }
512    }
513    entries
514}
515
516/// Compute each file's path relative to `root` as a forward-slash-lossy string.
517fn relative_paths_for(files: &[DiscoveredFile], root: &Path) -> Vec<String> {
518    files
519        .iter()
520        .map(|f| {
521            f.path
522                .strip_prefix(root)
523                .unwrap_or(&f.path)
524                .to_string_lossy()
525                .into_owned()
526        })
527        .collect()
528}
529
530/// Push entries for files matching the user-configured manual entry glob patterns.
531#[expect(
532    clippy::expect_used,
533    reason = "entry glob patterns are validated before entry point discovery"
534)]
535fn push_manual_entry_matches(
536    entries: &mut Vec<EntryPoint>,
537    config: &ResolvedConfig,
538    relative_paths: &[String],
539    files: &[DiscoveredFile],
540) {
541    let mut builder = globset::GlobSetBuilder::new();
542    for pattern in &config.entry_patterns {
543        builder.add(
544            globset::Glob::new(pattern).expect("entry pattern was validated at config load time"),
545        );
546    }
547    let Ok(glob_set) = builder.build() else {
548        return;
549    };
550    if glob_set.is_empty() {
551        return;
552    }
553    for (idx, rel) in relative_paths.iter().enumerate() {
554        if glob_set.is_match(rel) {
555            entries.push(EntryPoint {
556                path: files[idx].path.clone(),
557                source: EntryPointSource::ManualEntry,
558            });
559        }
560    }
561}
562
563/// Push entries derived from a package.json's declared entry points and scripts.
564fn push_package_json_entries(
565    discovery: &mut EntryPointDiscovery,
566    root: &Path,
567    pkg: &PackageJson,
568    canonical_root: &Path,
569) {
570    for entry_path in pkg.entry_points() {
571        if let Some(ep) = resolve_entry_path_with_tracking(
572            root,
573            &entry_path,
574            canonical_root,
575            EntryPointSource::PackageJsonMain,
576            Some(&mut discovery.skipped_entries),
577        ) {
578            discovery.entries.push(ep);
579        }
580    }
581
582    let Some(scripts) = &pkg.scripts else {
583        return;
584    };
585    for script_value in scripts.values() {
586        for file_ref in extract_script_file_refs(script_value) {
587            if let Some(ep) = resolve_entry_path_with_tracking(
588                root,
589                &file_ref,
590                canonical_root,
591                EntryPointSource::PackageJsonScript,
592                Some(&mut discovery.skipped_entries),
593            ) {
594                discovery.entries.push(ep);
595            }
596        }
597    }
598}
599
600/// Discover entry points from package.json, framework rules, and defaults.
601fn discover_entry_points_with_warnings_impl(
602    config: &ResolvedConfig,
603    files: &[DiscoveredFile],
604    root_pkg: Option<&PackageJson>,
605    include_nested_package_entries: bool,
606) -> EntryPointDiscovery {
607    let _span = tracing::info_span!("discover_entry_points").entered();
608    let mut discovery = EntryPointDiscovery::default();
609
610    let relative_paths = relative_paths_for(files, &config.root);
611    push_manual_entry_matches(&mut discovery.entries, config, &relative_paths, files);
612
613    let canonical_root = dunce::canonicalize(&config.root).unwrap_or_else(|_| config.root.clone());
614    if let Some(pkg) = root_pkg {
615        push_package_json_entries(&mut discovery, &config.root, pkg, &canonical_root);
616    }
617
618    if include_nested_package_entries {
619        let exports_dirs = root_pkg
620            .map(PackageJson::exports_subdirectories)
621            .unwrap_or_default();
622        discover_nested_package_entries(
623            &config.root,
624            files,
625            &mut discovery.entries,
626            &canonical_root,
627            &exports_dirs,
628            &mut discovery.skipped_entries,
629        );
630    }
631
632    if discovery.entries.is_empty() {
633        discovery.entries = apply_default_fallback(files, &config.root, None);
634    }
635
636    discovery.entries.sort_by(|a, b| a.path.cmp(&b.path));
637    discovery.entries.dedup_by(|a, b| a.path == b.path);
638
639    discovery
640}
641
642pub fn discover_entry_points_with_warnings_from_pkg(
643    config: &ResolvedConfig,
644    files: &[DiscoveredFile],
645    root_pkg: Option<&PackageJson>,
646    include_nested_package_entries: bool,
647) -> EntryPointDiscovery {
648    discover_entry_points_with_warnings_impl(
649        config,
650        files,
651        root_pkg,
652        include_nested_package_entries,
653    )
654}
655
656pub fn discover_entry_points_with_warnings(
657    config: &ResolvedConfig,
658    files: &[DiscoveredFile],
659) -> EntryPointDiscovery {
660    let pkg_path = config.root.join("package.json");
661    let root_pkg = PackageJson::load(&pkg_path).ok();
662    discover_entry_points_with_warnings_impl(config, files, root_pkg.as_ref(), true)
663}
664
665pub fn discover_entry_points(config: &ResolvedConfig, files: &[DiscoveredFile]) -> Vec<EntryPoint> {
666    let discovery = discover_entry_points_with_warnings(config, files);
667    warn_skipped_entry_summary(&discovery.skipped_entries);
668    discovery.entries
669}
670
671/// Discover entry points from nested package.json files in subdirectories.
672///
673/// Scans two sources for sub-packages:
674/// 1. Common monorepo directory patterns (`packages/`, `apps/`, `libs/`, etc.)
675/// 2. Directories derived from the root package.json `exports` map keys
676///    (e.g., `"./compat": {...}` implies `compat/` may be a sub-package)
677///
678/// For each discovered sub-package with a `package.json`, the `main`, `module`,
679/// `source`, `exports`, and `bin` fields are treated as entry points.
680fn discover_nested_package_entries(
681    root: &Path,
682    _files: &[DiscoveredFile],
683    entries: &mut Vec<EntryPoint>,
684    canonical_root: &Path,
685    exports_subdirectories: &[String],
686    skipped_entries: &mut FxHashMap<String, usize>,
687) {
688    let mut visited = rustc_hash::FxHashSet::default();
689
690    let search_dirs = [
691        "packages", "apps", "libs", "modules", "plugins", "services", "tools", "utils",
692    ];
693    for dir_name in &search_dirs {
694        let search_dir = root.join(dir_name);
695        if !search_dir.is_dir() {
696            continue;
697        }
698        let Ok(read_dir) = std::fs::read_dir(&search_dir) else {
699            continue;
700        };
701        for entry in read_dir.flatten() {
702            let pkg_dir = entry.path();
703            if visited.insert(pkg_dir.clone()) {
704                collect_nested_package_entries(&pkg_dir, entries, canonical_root, skipped_entries);
705            }
706        }
707    }
708
709    for dir_name in exports_subdirectories {
710        let pkg_dir = root.join(dir_name);
711        if pkg_dir.is_dir() && visited.insert(pkg_dir.clone()) {
712            collect_nested_package_entries(&pkg_dir, entries, canonical_root, skipped_entries);
713        }
714    }
715}
716
717/// Collect entry points from a single sub-package directory.
718fn collect_nested_package_entries(
719    pkg_dir: &Path,
720    entries: &mut Vec<EntryPoint>,
721    canonical_root: &Path,
722    skipped_entries: &mut FxHashMap<String, usize>,
723) {
724    let pkg_path = pkg_dir.join("package.json");
725    if !pkg_path.exists() {
726        return;
727    }
728    let Ok(pkg) = PackageJson::load(&pkg_path) else {
729        return;
730    };
731    for entry_path in pkg.entry_points() {
732        if entry_path.contains('*') {
733            expand_wildcard_entries(pkg_dir, &entry_path, canonical_root, entries);
734        } else if let Some(ep) = resolve_entry_path_with_tracking(
735            pkg_dir,
736            &entry_path,
737            canonical_root,
738            EntryPointSource::PackageJsonExports,
739            Some(&mut *skipped_entries),
740        ) {
741            entries.push(ep);
742        }
743    }
744    if let Some(scripts) = &pkg.scripts {
745        for script_value in scripts.values() {
746            for file_ref in extract_script_file_refs(script_value) {
747                if let Some(ep) = resolve_entry_path_with_tracking(
748                    pkg_dir,
749                    &file_ref,
750                    canonical_root,
751                    EntryPointSource::PackageJsonScript,
752                    Some(&mut *skipped_entries),
753                ) {
754                    entries.push(ep);
755                }
756            }
757        }
758    }
759}
760
761/// Expand wildcard subpath exports to matching files on disk.
762///
763/// Handles patterns like `./src/themes/*.css` from package.json exports maps
764/// (`"./themes/*": { "import": "./src/themes/*.css" }`). Expands the `*` to
765/// match actual files in the target directory.
766fn expand_wildcard_entries(
767    base: &Path,
768    pattern: &str,
769    canonical_root: &Path,
770    entries: &mut Vec<EntryPoint>,
771) {
772    let full_pattern = base.join(pattern).to_string_lossy().to_string();
773    let Ok(matches) = glob::glob(&full_pattern) else {
774        return;
775    };
776    for path_result in matches {
777        let Ok(path) = path_result else {
778            continue;
779        };
780        if let Ok(canonical) = dunce::canonicalize(&path)
781            && canonical.starts_with(canonical_root)
782        {
783            entries.push(EntryPoint {
784                path,
785                source: EntryPointSource::PackageJsonExports,
786            });
787        }
788    }
789}
790
791/// Discover entry points for a workspace package.
792#[must_use]
793fn discover_workspace_entry_points_with_warnings_impl(
794    ws_root: &Path,
795    all_files: &[DiscoveredFile],
796    pkg: Option<&PackageJson>,
797) -> EntryPointDiscovery {
798    let mut discovery = EntryPointDiscovery::default();
799
800    if let Some(pkg) = pkg {
801        let canonical_ws_root =
802            dunce::canonicalize(ws_root).unwrap_or_else(|_| ws_root.to_path_buf());
803        for entry_path in pkg.entry_points() {
804            if entry_path.contains('*') {
805                expand_wildcard_entries(
806                    ws_root,
807                    &entry_path,
808                    &canonical_ws_root,
809                    &mut discovery.entries,
810                );
811            } else if let Some(ep) = resolve_entry_path_with_tracking(
812                ws_root,
813                &entry_path,
814                &canonical_ws_root,
815                EntryPointSource::PackageJsonMain,
816                Some(&mut discovery.skipped_entries),
817            ) {
818                discovery.entries.push(ep);
819            }
820        }
821
822        if let Some(scripts) = &pkg.scripts {
823            for script_value in scripts.values() {
824                for file_ref in extract_script_file_refs(script_value) {
825                    if let Some(ep) = resolve_entry_path_with_tracking(
826                        ws_root,
827                        &file_ref,
828                        &canonical_ws_root,
829                        EntryPointSource::PackageJsonScript,
830                        Some(&mut discovery.skipped_entries),
831                    ) {
832                        discovery.entries.push(ep);
833                    }
834                }
835            }
836        }
837    }
838
839    if discovery.entries.is_empty() {
840        discovery.entries = apply_default_fallback(all_files, ws_root, Some(ws_root));
841    }
842
843    discovery.entries.sort_by(|a, b| a.path.cmp(&b.path));
844    discovery.entries.dedup_by(|a, b| a.path == b.path);
845    discovery
846}
847
848pub fn discover_workspace_entry_points_with_warnings_from_pkg(
849    ws_root: &Path,
850    all_files: &[DiscoveredFile],
851    pkg: Option<&PackageJson>,
852) -> EntryPointDiscovery {
853    discover_workspace_entry_points_with_warnings_impl(ws_root, all_files, pkg)
854}
855
856#[must_use]
857pub fn discover_workspace_entry_points_with_warnings(
858    ws_root: &Path,
859    _config: &ResolvedConfig,
860    all_files: &[DiscoveredFile],
861) -> EntryPointDiscovery {
862    let pkg_path = ws_root.join("package.json");
863    let pkg = PackageJson::load(&pkg_path).ok();
864    discover_workspace_entry_points_with_warnings_impl(ws_root, all_files, pkg.as_ref())
865}
866
867#[must_use]
868pub fn discover_workspace_entry_points(
869    ws_root: &Path,
870    config: &ResolvedConfig,
871    all_files: &[DiscoveredFile],
872) -> Vec<EntryPoint> {
873    let discovery = discover_workspace_entry_points_with_warnings(ws_root, config, all_files);
874    warn_skipped_entry_summary(&discovery.skipped_entries);
875    discovery.entries
876}
877
878/// Discover entry points from plugin results (dynamic config parsing).
879///
880/// Converts plugin-discovered patterns and setup files into concrete entry points
881/// by matching them against the discovered file list.
882#[must_use]
883pub fn discover_plugin_entry_points(
884    plugin_result: &crate::plugins::AggregatedPluginResult,
885    config: &ResolvedConfig,
886    files: &[DiscoveredFile],
887) -> Vec<EntryPoint> {
888    discover_plugin_entry_point_sets(plugin_result, config, files).all
889}
890
891/// Discover plugin-derived entry points with runtime/test/support roles preserved.
892#[must_use]
893pub fn discover_plugin_entry_point_sets(
894    plugin_result: &crate::plugins::AggregatedPluginResult,
895    config: &ResolvedConfig,
896    files: &[DiscoveredFile],
897) -> CategorizedEntryPoints {
898    let mut entries = CategorizedEntryPoints::default();
899
900    let relative_paths = relative_paths_for(files, &config.root);
901    let (glob_set, glob_meta) = build_plugin_glob_meta(plugin_result);
902    if let Some(glob_set) = glob_set.filter(|set| !set.is_empty()) {
903        match_plugin_entry_files(&mut entries, &glob_set, &glob_meta, &relative_paths, files);
904    }
905
906    push_plugin_setup_files(&mut entries, &config.root, plugin_result);
907
908    entries.dedup()
909}
910
911/// Compile plugin entry-pattern and support globs into a glob set plus per-glob metadata.
912fn build_plugin_glob_meta(
913    plugin_result: &crate::plugins::AggregatedPluginResult,
914) -> (Option<globset::GlobSet>, Vec<CompiledEntryRule<'_>>) {
915    let mut builder = globset::GlobSetBuilder::new();
916    let mut glob_meta: Vec<CompiledEntryRule<'_>> = Vec::new();
917    for (rule, pname) in &plugin_result.entry_patterns {
918        if let Some((include, compiled)) = compile_entry_rule(rule, pname, plugin_result) {
919            builder.add(include);
920            glob_meta.push(compiled);
921        }
922    }
923    for (pattern, pname) in plugin_result
924        .discovered_always_used
925        .iter()
926        .chain(plugin_result.always_used.iter())
927        .chain(plugin_result.fixture_patterns.iter())
928    {
929        if let Ok(glob) = globset::GlobBuilder::new(pattern)
930            .literal_separator(true)
931            .build()
932        {
933            builder.add(glob);
934            if let Some(path) = crate::plugins::CompiledPathRule::for_entry_rule(
935                &crate::plugins::PathRule::new(pattern.clone()),
936                "support entry pattern",
937            ) {
938                glob_meta.push(CompiledEntryRule {
939                    path,
940                    plugin_name: pname,
941                    role: EntryPointRole::Support,
942                });
943            }
944        }
945    }
946    (builder.build().ok(), glob_meta)
947}
948
949/// Match each file against the compiled plugin glob set, categorizing hits by role.
950fn match_plugin_entry_files(
951    entries: &mut CategorizedEntryPoints,
952    glob_set: &globset::GlobSet,
953    glob_meta: &[CompiledEntryRule<'_>],
954    relative_paths: &[String],
955    files: &[DiscoveredFile],
956) {
957    for (idx, rel) in relative_paths.iter().enumerate() {
958        let matches: Vec<usize> = glob_set
959            .matches(rel)
960            .into_iter()
961            .filter(|match_idx| glob_meta[*match_idx].matches(rel))
962            .collect();
963        if matches.is_empty() {
964            continue;
965        }
966        let name = glob_meta[matches[0]].plugin_name;
967        let entry = EntryPoint {
968            path: files[idx].path.clone(),
969            source: EntryPointSource::Plugin {
970                name: name.to_string(),
971            },
972        };
973        categorize_plugin_match(entries, entry, glob_meta, &matches);
974    }
975}
976
977/// Push a matched entry into the runtime/test/support buckets implied by its roles.
978fn categorize_plugin_match(
979    entries: &mut CategorizedEntryPoints,
980    entry: EntryPoint,
981    glob_meta: &[CompiledEntryRule<'_>],
982    matches: &[usize],
983) {
984    let mut has_runtime = false;
985    let mut has_test = false;
986    let mut has_support = false;
987    for &match_idx in matches {
988        match glob_meta[match_idx].role {
989            EntryPointRole::Runtime => has_runtime = true,
990            EntryPointRole::Test => has_test = true,
991            EntryPointRole::Support => has_support = true,
992        }
993    }
994
995    if has_runtime {
996        entries.push_runtime(entry.clone());
997    }
998    if has_test {
999        entries.push_test(entry.clone());
1000    }
1001    if has_support || (!has_runtime && !has_test) {
1002        entries.push_support(entry);
1003    }
1004}
1005
1006/// Resolve plugin-declared setup files (with source-extension fallback) into support entries.
1007fn push_plugin_setup_files(
1008    entries: &mut CategorizedEntryPoints,
1009    root: &Path,
1010    plugin_result: &crate::plugins::AggregatedPluginResult,
1011) {
1012    for (setup_file, pname) in &plugin_result.setup_files {
1013        let resolved = resolve_plugin_setup_file(root, setup_file);
1014        if resolved.exists() {
1015            entries.push_support(EntryPoint {
1016                path: resolved,
1017                source: EntryPointSource::Plugin {
1018                    name: pname.clone(),
1019                },
1020            });
1021            continue;
1022        }
1023        for ext in SOURCE_EXTENSIONS {
1024            let with_ext = resolved.with_extension(ext);
1025            if with_ext.exists() {
1026                entries.push_support(EntryPoint {
1027                    path: with_ext,
1028                    source: EntryPointSource::Plugin {
1029                        name: pname.clone(),
1030                    },
1031                });
1032                break;
1033            }
1034        }
1035    }
1036}
1037
1038fn resolve_plugin_setup_file(root: &Path, setup_file: &Path) -> PathBuf {
1039    if is_absolute_path_any_platform(setup_file) {
1040        setup_file.to_path_buf()
1041    } else {
1042        root.join(setup_file)
1043    }
1044}
1045
1046/// Discover entry points from `dynamicallyLoaded` config patterns.
1047///
1048/// Matches the configured glob patterns against the discovered file list and
1049/// marks matching files as entry points so they are never flagged as unused.
1050#[expect(
1051    clippy::expect_used,
1052    reason = "dynamicallyLoaded glob patterns are validated before entry point discovery"
1053)]
1054#[must_use]
1055pub fn discover_dynamically_loaded_entry_points(
1056    config: &ResolvedConfig,
1057    files: &[DiscoveredFile],
1058) -> Vec<EntryPoint> {
1059    if config.dynamically_loaded.is_empty() {
1060        return Vec::new();
1061    }
1062
1063    let mut builder = globset::GlobSetBuilder::new();
1064    for pattern in &config.dynamically_loaded {
1065        builder.add(
1066            globset::Glob::new(pattern)
1067                .expect("dynamicallyLoaded pattern was validated at config load time"),
1068        );
1069    }
1070    let Ok(glob_set) = builder.build() else {
1071        return Vec::new();
1072    };
1073    if glob_set.is_empty() {
1074        return Vec::new();
1075    }
1076
1077    let mut entries = Vec::new();
1078    for file in files {
1079        let rel = file
1080            .path
1081            .strip_prefix(&config.root)
1082            .unwrap_or(&file.path)
1083            .to_string_lossy();
1084        if glob_set.is_match(rel.as_ref()) {
1085            entries.push(EntryPoint {
1086                path: file.path.clone(),
1087                source: EntryPointSource::DynamicallyLoaded,
1088            });
1089        }
1090    }
1091    entries
1092}
1093
1094struct CompiledEntryRule<'a> {
1095    path: crate::plugins::CompiledPathRule,
1096    plugin_name: &'a str,
1097    role: EntryPointRole,
1098}
1099
1100impl CompiledEntryRule<'_> {
1101    fn matches(&self, path: &str) -> bool {
1102        self.path.matches(path)
1103    }
1104}
1105
1106fn compile_entry_rule<'a>(
1107    rule: &'a crate::plugins::PathRule,
1108    plugin_name: &'a str,
1109    plugin_result: &'a crate::plugins::AggregatedPluginResult,
1110) -> Option<(globset::Glob, CompiledEntryRule<'a>)> {
1111    let include = match globset::GlobBuilder::new(&rule.pattern)
1112        .literal_separator(true)
1113        .build()
1114    {
1115        Ok(glob) => glob,
1116        Err(err) => {
1117            tracing::warn!("invalid entry pattern '{}': {err}", rule.pattern);
1118            return None;
1119        }
1120    };
1121    let role = plugin_result
1122        .entry_point_roles
1123        .get(plugin_name)
1124        .copied()
1125        .unwrap_or(EntryPointRole::Support);
1126    Some((
1127        include,
1128        CompiledEntryRule {
1129            path: crate::plugins::CompiledPathRule::for_entry_rule(rule, "entry pattern")?,
1130            plugin_name,
1131            role,
1132        },
1133    ))
1134}
1135
1136/// Pre-compile a set of glob patterns for efficient matching against many paths.
1137#[must_use]
1138pub fn compile_glob_set(patterns: &[String]) -> Option<globset::GlobSet> {
1139    if patterns.is_empty() {
1140        return None;
1141    }
1142    let mut builder = globset::GlobSetBuilder::new();
1143    for pattern in patterns {
1144        if let Ok(glob) = globset::GlobBuilder::new(pattern)
1145            .literal_separator(true)
1146            .build()
1147        {
1148            builder.add(glob);
1149        }
1150    }
1151    builder.build().ok()
1152}
1153
1154#[cfg(test)]
1155mod tests {
1156    use super::*;
1157    use fallow_config::{FallowConfig, OutputFormat, RulesConfig};
1158    use fallow_types::discover::FileId;
1159    use proptest::prelude::*;
1160
1161    proptest! {
1162        /// Valid glob patterns should never panic when compiled via globset.
1163        #[test]
1164        fn glob_patterns_never_panic_on_compile(
1165            prefix in "[a-zA-Z0-9_]{1,20}",
1166            ext in prop::sample::select(vec!["ts", "tsx", "js", "jsx", "vue", "svelte", "astro", "mdx"]),
1167        ) {
1168            let pattern = format!("**/{prefix}*.{ext}");
1169            let result = globset::Glob::new(&pattern);
1170            prop_assert!(result.is_ok(), "Glob::new should not fail for well-formed patterns");
1171        }
1172
1173        /// Non-source extensions should NOT be in the SOURCE_EXTENSIONS list.
1174        #[test]
1175        fn non_source_extensions_not_in_list(
1176            ext in prop::sample::select(vec!["py", "rb", "rs", "go", "java", "xml", "yaml", "toml", "md", "txt", "png", "jpg", "wasm", "lock"]),
1177        ) {
1178            prop_assert!(
1179                !SOURCE_EXTENSIONS.contains(&ext),
1180                "Extension '{ext}' should NOT be in SOURCE_EXTENSIONS"
1181            );
1182        }
1183
1184        /// compile_glob_set should never panic on arbitrary well-formed glob patterns.
1185        #[test]
1186        fn compile_glob_set_no_panic(
1187            patterns in prop::collection::vec("[a-zA-Z0-9_*/.]{1,30}", 0..10),
1188        ) {
1189            let _ = compile_glob_set(&patterns);
1190        }
1191    }
1192
1193    #[test]
1194    fn compile_glob_set_empty_input() {
1195        assert!(
1196            compile_glob_set(&[]).is_none(),
1197            "empty patterns should return None"
1198        );
1199    }
1200
1201    #[test]
1202    fn compile_glob_set_valid_patterns() {
1203        let patterns = vec!["**/*.ts".to_string(), "src/**/*.js".to_string()];
1204        let set = compile_glob_set(&patterns);
1205        assert!(set.is_some(), "valid patterns should compile");
1206        let set = set.unwrap();
1207        assert!(set.is_match("src/foo.ts"));
1208        assert!(set.is_match("src/bar.js"));
1209        assert!(!set.is_match("src/bar.py"));
1210    }
1211
1212    #[test]
1213    fn compile_glob_set_keeps_star_within_a_single_path_segment() {
1214        let patterns = vec!["composables/*.{ts,js}".to_string()];
1215        let set = compile_glob_set(&patterns).expect("pattern should compile");
1216
1217        assert!(set.is_match("composables/useFoo.ts"));
1218        assert!(!set.is_match("composables/nested/useFoo.ts"));
1219    }
1220
1221    #[test]
1222    #[expect(
1223        clippy::too_many_lines,
1224        reason = "test fixture; linear setup/assert, length is not a maintainability concern"
1225    )]
1226    fn plugin_entry_point_sets_preserve_runtime_test_and_support_roles() {
1227        let dir = tempfile::tempdir().expect("create temp dir");
1228        let root = dir.path();
1229        std::fs::create_dir_all(root.join("src")).unwrap();
1230        std::fs::create_dir_all(root.join("tests")).unwrap();
1231        std::fs::write(root.join("src/runtime.ts"), "export const runtime = 1;").unwrap();
1232        std::fs::write(root.join("src/setup.ts"), "export const setup = 1;").unwrap();
1233        std::fs::write(root.join("tests/app.test.ts"), "export const test = 1;").unwrap();
1234
1235        let config = FallowConfig {
1236            schema: None,
1237            extends: vec![],
1238            entry: vec![],
1239            ignore_patterns: vec![],
1240            framework: vec![],
1241            workspaces: None,
1242            ignore_dependencies: vec![],
1243            ignore_unresolved_imports: vec![],
1244            ignore_exports: vec![],
1245            ignore_catalog_references: vec![],
1246            ignore_dependency_overrides: vec![],
1247            ignore_exports_used_in_file: fallow_config::IgnoreExportsUsedInFileConfig::default(),
1248            used_class_members: vec![],
1249            ignore_decorators: vec![],
1250            unused_component_props: fallow_config::UnusedComponentPropsConfig::default(),
1251            duplicates: fallow_config::DuplicatesConfig::default(),
1252            health: fallow_config::HealthConfig::default(),
1253            rules: RulesConfig::default(),
1254            boundaries: fallow_config::BoundaryConfig::default(),
1255            production: false.into(),
1256            plugins: vec![],
1257            rule_packs: vec![],
1258            dynamically_loaded: vec![],
1259            overrides: vec![],
1260            regression: None,
1261            audit: fallow_config::AuditConfig::default(),
1262            codeowners: None,
1263            public_packages: vec![],
1264            flags: fallow_config::FlagsConfig::default(),
1265            security: fallow_config::SecurityConfig::default(),
1266            fix: fallow_config::FixConfig::default(),
1267            resolve: fallow_config::ResolveConfig::default(),
1268            sealed: false,
1269            include_entry_exports: false,
1270            auto_imports: false,
1271            cache: fallow_config::CacheConfig::default(),
1272        }
1273        .resolve(root.to_path_buf(), OutputFormat::Human, 4, true, true, None);
1274
1275        let files = vec![
1276            DiscoveredFile {
1277                id: FileId(0),
1278                path: root.join("src/runtime.ts"),
1279                size_bytes: 1,
1280            },
1281            DiscoveredFile {
1282                id: FileId(1),
1283                path: root.join("src/setup.ts"),
1284                size_bytes: 1,
1285            },
1286            DiscoveredFile {
1287                id: FileId(2),
1288                path: root.join("tests/app.test.ts"),
1289                size_bytes: 1,
1290            },
1291        ];
1292
1293        let mut plugin_result = crate::plugins::AggregatedPluginResult::default();
1294        plugin_result.entry_patterns.push((
1295            crate::plugins::PathRule::new("src/runtime.ts"),
1296            "runtime-plugin".to_string(),
1297        ));
1298        plugin_result.entry_patterns.push((
1299            crate::plugins::PathRule::new("tests/app.test.ts"),
1300            "test-plugin".to_string(),
1301        ));
1302        plugin_result
1303            .always_used
1304            .push(("src/setup.ts".to_string(), "support-plugin".to_string()));
1305        plugin_result
1306            .entry_point_roles
1307            .insert("runtime-plugin".to_string(), EntryPointRole::Runtime);
1308        plugin_result
1309            .entry_point_roles
1310            .insert("test-plugin".to_string(), EntryPointRole::Test);
1311        plugin_result
1312            .entry_point_roles
1313            .insert("support-plugin".to_string(), EntryPointRole::Support);
1314
1315        let entries = discover_plugin_entry_point_sets(&plugin_result, &config, &files);
1316
1317        assert_eq!(entries.runtime.len(), 1, "expected one runtime entry");
1318        assert!(
1319            entries.runtime[0].path.ends_with("src/runtime.ts"),
1320            "runtime entry should stay runtime-only"
1321        );
1322        assert_eq!(entries.test.len(), 1, "expected one test entry");
1323        assert!(
1324            entries.test[0].path.ends_with("tests/app.test.ts"),
1325            "test entry should stay test-only"
1326        );
1327        assert_eq!(
1328            entries.all.len(),
1329            3,
1330            "support entries should stay in all entries"
1331        );
1332        assert!(
1333            entries
1334                .all
1335                .iter()
1336                .any(|entry| entry.path.ends_with("src/setup.ts")),
1337            "support entries should remain in the overall entry-point set"
1338        );
1339        assert!(
1340            !entries
1341                .runtime
1342                .iter()
1343                .any(|entry| entry.path.ends_with("src/setup.ts")),
1344            "support entries should not bleed into runtime reachability"
1345        );
1346        assert!(
1347            !entries
1348                .test
1349                .iter()
1350                .any(|entry| entry.path.ends_with("src/setup.ts")),
1351            "support entries should not bleed into test reachability"
1352        );
1353    }
1354
1355    #[test]
1356    fn resolve_plugin_setup_file_preserves_windows_absolute_path_on_any_host() {
1357        let root = Path::new("/workspace/project");
1358        let setup_file = Path::new(r"C:\workspace\project\setup.ts");
1359
1360        assert_eq!(
1361            resolve_plugin_setup_file(root, setup_file),
1362            setup_file.to_path_buf()
1363        );
1364    }
1365
1366    #[cfg(windows)]
1367    #[test]
1368    fn resolve_plugin_setup_file_preserves_posix_rooted_path_on_windows() {
1369        let root = Path::new(r"C:\workspace\project");
1370        let setup_file = Path::new(r"/workspace/project/setup.ts");
1371
1372        assert_eq!(
1373            resolve_plugin_setup_file(root, setup_file),
1374            setup_file.to_path_buf()
1375        );
1376    }
1377
1378    #[test]
1379    fn plugin_entry_point_rules_respect_exclusions() {
1380        let dir = tempfile::tempdir().expect("create temp dir");
1381        let root = dir.path();
1382        std::fs::create_dir_all(root.join("app/pages")).unwrap();
1383        std::fs::write(
1384            root.join("app/pages/index.tsx"),
1385            "export default function Page() { return null; }",
1386        )
1387        .unwrap();
1388        std::fs::write(
1389            root.join("app/pages/-helper.ts"),
1390            "export const helper = 1;",
1391        )
1392        .unwrap();
1393
1394        let config = FallowConfig {
1395            schema: None,
1396            extends: vec![],
1397            entry: vec![],
1398            ignore_patterns: vec![],
1399            framework: vec![],
1400            workspaces: None,
1401            ignore_dependencies: vec![],
1402            ignore_unresolved_imports: vec![],
1403            ignore_exports: vec![],
1404            ignore_catalog_references: vec![],
1405            ignore_dependency_overrides: vec![],
1406            ignore_exports_used_in_file: fallow_config::IgnoreExportsUsedInFileConfig::default(),
1407            used_class_members: vec![],
1408            ignore_decorators: vec![],
1409            unused_component_props: fallow_config::UnusedComponentPropsConfig::default(),
1410            duplicates: fallow_config::DuplicatesConfig::default(),
1411            health: fallow_config::HealthConfig::default(),
1412            rules: RulesConfig::default(),
1413            boundaries: fallow_config::BoundaryConfig::default(),
1414            production: false.into(),
1415            plugins: vec![],
1416            rule_packs: vec![],
1417            dynamically_loaded: vec![],
1418            overrides: vec![],
1419            regression: None,
1420            audit: fallow_config::AuditConfig::default(),
1421            codeowners: None,
1422            public_packages: vec![],
1423            flags: fallow_config::FlagsConfig::default(),
1424            security: fallow_config::SecurityConfig::default(),
1425            fix: fallow_config::FixConfig::default(),
1426            resolve: fallow_config::ResolveConfig::default(),
1427            sealed: false,
1428            include_entry_exports: false,
1429            auto_imports: false,
1430            cache: fallow_config::CacheConfig::default(),
1431        }
1432        .resolve(root.to_path_buf(), OutputFormat::Human, 4, true, true, None);
1433
1434        let files = vec![
1435            DiscoveredFile {
1436                id: FileId(0),
1437                path: root.join("app/pages/index.tsx"),
1438                size_bytes: 1,
1439            },
1440            DiscoveredFile {
1441                id: FileId(1),
1442                path: root.join("app/pages/-helper.ts"),
1443                size_bytes: 1,
1444            },
1445        ];
1446
1447        let mut plugin_result = crate::plugins::AggregatedPluginResult::default();
1448        plugin_result.entry_patterns.push((
1449            crate::plugins::PathRule::new("app/pages/**/*.{ts,tsx,js,jsx}")
1450                .with_excluded_globs(["app/pages/**/-*", "app/pages/**/-*/**/*"]),
1451            "tanstack-router".to_string(),
1452        ));
1453        plugin_result
1454            .entry_point_roles
1455            .insert("tanstack-router".to_string(), EntryPointRole::Runtime);
1456
1457        let entries = discover_plugin_entry_point_sets(&plugin_result, &config, &files);
1458        let entry_paths: Vec<_> = entries
1459            .all
1460            .iter()
1461            .map(|entry| {
1462                entry
1463                    .path
1464                    .strip_prefix(root)
1465                    .unwrap()
1466                    .to_string_lossy()
1467                    .into_owned()
1468            })
1469            .collect();
1470
1471        assert!(entry_paths.contains(&"app/pages/index.tsx".to_string()));
1472        assert!(!entry_paths.contains(&"app/pages/-helper.ts".to_string()));
1473    }
1474
1475    mod resolve_entry_path_tests {
1476        use super::*;
1477
1478        #[test]
1479        fn resolves_existing_file() {
1480            let dir = tempfile::tempdir().expect("create temp dir");
1481            let src = dir.path().join("src");
1482            std::fs::create_dir_all(&src).unwrap();
1483            std::fs::write(src.join("index.ts"), "export const a = 1;").unwrap();
1484
1485            let canonical = dunce::canonicalize(dir.path()).unwrap();
1486            let result = resolve_entry_path(
1487                dir.path(),
1488                "src/index.ts",
1489                &canonical,
1490                EntryPointSource::PackageJsonMain,
1491            );
1492            assert!(result.is_some(), "should resolve an existing file");
1493            assert!(result.unwrap().path.ends_with("src/index.ts"));
1494        }
1495
1496        #[test]
1497        fn resolves_with_extension_fallback() {
1498            let dir = tempfile::tempdir().expect("create temp dir");
1499            let canonical = dunce::canonicalize(dir.path()).unwrap();
1500            let src = canonical.join("src");
1501            std::fs::create_dir_all(&src).unwrap();
1502            std::fs::write(src.join("index.ts"), "export const a = 1;").unwrap();
1503
1504            let result = resolve_entry_path(
1505                &canonical,
1506                "src/index",
1507                &canonical,
1508                EntryPointSource::PackageJsonMain,
1509            );
1510            assert!(
1511                result.is_some(),
1512                "should resolve via extension fallback when exact path doesn't exist"
1513            );
1514            let ep = result.unwrap();
1515            assert!(
1516                ep.path.to_string_lossy().contains("index.ts"),
1517                "should find index.ts via extension fallback"
1518            );
1519        }
1520
1521        #[test]
1522        fn exact_file_wins_before_directory_index_fallback() {
1523            let dir = tempfile::tempdir().expect("create temp dir");
1524            let canonical = dunce::canonicalize(dir.path()).unwrap();
1525            let scripts = canonical.join("scripts");
1526            std::fs::create_dir_all(scripts.join("process-messages")).unwrap();
1527            std::fs::write(
1528                scripts.join("process-messages.js"),
1529                "export const direct = true;",
1530            )
1531            .unwrap();
1532            std::fs::write(
1533                scripts.join("process-messages").join("index.js"),
1534                "export const index = true;",
1535            )
1536            .unwrap();
1537
1538            let result = resolve_entry_path(
1539                &canonical,
1540                "scripts/process-messages.js",
1541                &canonical,
1542                EntryPointSource::PackageJsonScript,
1543            )
1544            .expect("exact file should resolve");
1545
1546            assert!(result.path.ends_with("scripts/process-messages.js"));
1547        }
1548
1549        #[test]
1550        fn extension_fallback_wins_before_directory_index_fallback() {
1551            let dir = tempfile::tempdir().expect("create temp dir");
1552            let canonical = dunce::canonicalize(dir.path()).unwrap();
1553            let scripts = canonical.join("scripts");
1554            std::fs::create_dir_all(scripts.join("process-messages")).unwrap();
1555            std::fs::write(
1556                scripts.join("process-messages.ts"),
1557                "export const withExt = true;",
1558            )
1559            .unwrap();
1560            std::fs::write(
1561                scripts.join("process-messages").join("index.js"),
1562                "export const index = true;",
1563            )
1564            .unwrap();
1565
1566            let result = resolve_entry_path(
1567                &canonical,
1568                "scripts/process-messages",
1569                &canonical,
1570                EntryPointSource::PackageJsonScript,
1571            )
1572            .expect("extension fallback should resolve");
1573
1574            assert!(result.path.ends_with("scripts/process-messages.ts"));
1575        }
1576
1577        #[test]
1578        fn resolves_directory_index_after_exact_and_extension_fallbacks() {
1579            let dir = tempfile::tempdir().expect("create temp dir");
1580            let canonical = dunce::canonicalize(dir.path()).unwrap();
1581            let scripts = canonical.join("scripts/process-messages");
1582            std::fs::create_dir_all(&scripts).unwrap();
1583            std::fs::write(scripts.join("index.js"), "export const index = true;").unwrap();
1584
1585            let result = resolve_entry_path(
1586                &canonical,
1587                "scripts/process-messages",
1588                &canonical,
1589                EntryPointSource::PackageJsonScript,
1590            )
1591            .expect("directory index should resolve");
1592
1593            assert!(result.path.ends_with("scripts/process-messages/index.js"));
1594        }
1595
1596        #[test]
1597        fn directory_index_fallback_ignores_wildcards_and_url_like_entries() {
1598            let dir = tempfile::tempdir().expect("create temp dir");
1599            let canonical = dunce::canonicalize(dir.path()).unwrap();
1600            std::fs::create_dir_all(canonical.join("scripts/process-messages")).unwrap();
1601            std::fs::write(
1602                canonical.join("scripts/process-messages/index.js"),
1603                "export const index = true;",
1604            )
1605            .unwrap();
1606
1607            for entry in [
1608                "scripts/*",
1609                "https://example.com/scripts/process-messages",
1610                "@scope/package/scripts/process-messages",
1611            ] {
1612                let result = resolve_entry_path(
1613                    &canonical,
1614                    entry,
1615                    &canonical,
1616                    EntryPointSource::PackageJsonScript,
1617                );
1618                assert!(result.is_none(), "{entry} should not resolve");
1619            }
1620        }
1621
1622        #[test]
1623        fn returns_none_for_nonexistent_file() {
1624            let dir = tempfile::tempdir().expect("create temp dir");
1625            let canonical = dunce::canonicalize(dir.path()).unwrap();
1626            let result = resolve_entry_path(
1627                dir.path(),
1628                "does/not/exist.ts",
1629                &canonical,
1630                EntryPointSource::PackageJsonMain,
1631            );
1632            assert!(result.is_none(), "should return None for nonexistent files");
1633        }
1634
1635        #[test]
1636        fn maps_dist_output_to_src() {
1637            let dir = tempfile::tempdir().expect("create temp dir");
1638            let src = dir.path().join("src");
1639            std::fs::create_dir_all(&src).unwrap();
1640            std::fs::write(src.join("utils.ts"), "export const u = 1;").unwrap();
1641
1642            let dist = dir.path().join("dist");
1643            std::fs::create_dir_all(&dist).unwrap();
1644            std::fs::write(dist.join("utils.js"), "// compiled").unwrap();
1645
1646            let canonical = dunce::canonicalize(dir.path()).unwrap();
1647            let result = resolve_entry_path(
1648                dir.path(),
1649                "./dist/utils.js",
1650                &canonical,
1651                EntryPointSource::PackageJsonExports,
1652            );
1653            assert!(result.is_some(), "should resolve dist/ path to src/");
1654            let ep = result.unwrap();
1655            assert!(
1656                ep.path
1657                    .to_string_lossy()
1658                    .replace('\\', "/")
1659                    .contains("src/utils.ts"),
1660                "should map ./dist/utils.js to src/utils.ts"
1661            );
1662        }
1663
1664        #[test]
1665        fn maps_build_output_to_src() {
1666            let dir = tempfile::tempdir().expect("create temp dir");
1667            let canonical = dunce::canonicalize(dir.path()).unwrap();
1668            let src = canonical.join("src");
1669            std::fs::create_dir_all(&src).unwrap();
1670            std::fs::write(src.join("index.tsx"), "export default () => {};").unwrap();
1671
1672            let result = resolve_entry_path(
1673                &canonical,
1674                "./build/index.js",
1675                &canonical,
1676                EntryPointSource::PackageJsonExports,
1677            );
1678            assert!(result.is_some(), "should map build/ output to src/");
1679            let ep = result.unwrap();
1680            assert!(
1681                ep.path
1682                    .to_string_lossy()
1683                    .replace('\\', "/")
1684                    .contains("src/index.tsx"),
1685                "should map ./build/index.js to src/index.tsx"
1686            );
1687        }
1688
1689        #[test]
1690        fn preserves_entry_point_source() {
1691            let dir = tempfile::tempdir().expect("create temp dir");
1692            std::fs::write(dir.path().join("index.ts"), "export const a = 1;").unwrap();
1693
1694            let canonical = dunce::canonicalize(dir.path()).unwrap();
1695            let result = resolve_entry_path(
1696                dir.path(),
1697                "index.ts",
1698                &canonical,
1699                EntryPointSource::PackageJsonScript,
1700            );
1701            assert!(result.is_some());
1702            assert!(
1703                matches!(result.unwrap().source, EntryPointSource::PackageJsonScript),
1704                "should preserve the source kind"
1705            );
1706        }
1707        #[test]
1708        fn tracks_skipped_entries_without_logging_each_repeat() {
1709            let dir = tempfile::tempdir().expect("create temp dir");
1710            let canonical = dunce::canonicalize(dir.path()).unwrap();
1711            let mut skipped_entries = FxHashMap::default();
1712
1713            let result = resolve_entry_path_with_tracking(
1714                dir.path(),
1715                "../scripts/build.js",
1716                &canonical,
1717                EntryPointSource::PackageJsonScript,
1718                Some(&mut skipped_entries),
1719            );
1720
1721            assert!(result.is_none(), "unsafe entry should be skipped");
1722            assert_eq!(
1723                skipped_entries.get("../scripts/build.js"),
1724                Some(&1),
1725                "warning tracker should count the skipped path"
1726            );
1727        }
1728
1729        #[test]
1730        fn formats_skipped_entry_warning_with_counts() {
1731            let mut skipped_entries = FxHashMap::default();
1732            skipped_entries.insert("../../scripts/rm.mjs".to_owned(), 8);
1733            skipped_entries.insert("../utils/bar.js".to_owned(), 2);
1734
1735            let warning =
1736                format_skipped_entry_warning(&skipped_entries).expect("warning should be rendered");
1737
1738            assert_eq!(
1739                warning,
1740                "Skipped 10 package.json entry points outside project root or containing parent directory traversal: ../../scripts/rm.mjs (8x), ../utils/bar.js (2x)"
1741            );
1742        }
1743
1744        #[test]
1745        fn skipped_entry_summary_dedupes_identical_messages() {
1746            let message = format!(
1747                "Skipped 1 package.json entry point outside project root: ../../pkg-{}/bin/x",
1748                std::process::id()
1749            );
1750            assert!(
1751                should_warn_skipped_entry(&message),
1752                "first occurrence of a message emits"
1753            );
1754            assert!(
1755                !should_warn_skipped_entry(&message),
1756                "identical repeat is suppressed"
1757            );
1758        }
1759
1760        #[test]
1761        fn rejects_parent_dir_escape_for_exact_file() {
1762            let sandbox = tempfile::tempdir().expect("create sandbox");
1763            let root = sandbox.path().join("project");
1764            std::fs::create_dir_all(&root).unwrap();
1765            std::fs::write(
1766                sandbox.path().join("escape.ts"),
1767                "export const escape = true;",
1768            )
1769            .unwrap();
1770
1771            let canonical = dunce::canonicalize(&root).unwrap();
1772            let result = resolve_entry_path(
1773                &root,
1774                "../escape.ts",
1775                &canonical,
1776                EntryPointSource::PackageJsonMain,
1777            );
1778
1779            assert!(
1780                result.is_none(),
1781                "should reject exact paths that escape the root"
1782            );
1783        }
1784
1785        #[test]
1786        fn rejects_parent_dir_escape_via_extension_fallback() {
1787            let sandbox = tempfile::tempdir().expect("create sandbox");
1788            let root = sandbox.path().join("project");
1789            std::fs::create_dir_all(&root).unwrap();
1790            std::fs::write(
1791                sandbox.path().join("escape.ts"),
1792                "export const escape = true;",
1793            )
1794            .unwrap();
1795
1796            let canonical = dunce::canonicalize(&root).unwrap();
1797            let result = resolve_entry_path(
1798                &root,
1799                "../escape",
1800                &canonical,
1801                EntryPointSource::PackageJsonMain,
1802            );
1803
1804            assert!(
1805                result.is_none(),
1806                "should reject extension fallback paths that escape the root"
1807            );
1808        }
1809    }
1810
1811    mod output_to_source_tests {
1812        use super::*;
1813
1814        #[test]
1815        fn maps_dist_to_src_with_ts_extension() {
1816            let dir = tempfile::tempdir().expect("create temp dir");
1817            let src = dir.path().join("src");
1818            std::fs::create_dir_all(&src).unwrap();
1819            std::fs::write(src.join("utils.ts"), "export const u = 1;").unwrap();
1820
1821            let result = try_output_to_source_path(dir.path(), "./dist/utils.js");
1822            assert!(result.is_some());
1823            assert!(
1824                result
1825                    .unwrap()
1826                    .to_string_lossy()
1827                    .replace('\\', "/")
1828                    .contains("src/utils.ts")
1829            );
1830        }
1831
1832        #[test]
1833        fn returns_none_when_no_source_file_exists() {
1834            let dir = tempfile::tempdir().expect("create temp dir");
1835            let result = try_output_to_source_path(dir.path(), "./dist/missing.js");
1836            assert!(result.is_none());
1837        }
1838
1839        #[test]
1840        fn ignores_non_output_directories() {
1841            let dir = tempfile::tempdir().expect("create temp dir");
1842            let src = dir.path().join("src");
1843            std::fs::create_dir_all(&src).unwrap();
1844            std::fs::write(src.join("foo.ts"), "export const f = 1;").unwrap();
1845
1846            let result = try_output_to_source_path(dir.path(), "./lib/foo.js");
1847            assert!(result.is_none());
1848        }
1849
1850        #[test]
1851        fn maps_nested_output_path_preserving_prefix() {
1852            let dir = tempfile::tempdir().expect("create temp dir");
1853            let modules_src = dir.path().join("modules").join("src");
1854            std::fs::create_dir_all(&modules_src).unwrap();
1855            std::fs::write(modules_src.join("helper.ts"), "export const h = 1;").unwrap();
1856
1857            let result = try_output_to_source_path(dir.path(), "./modules/dist/helper.js");
1858            assert!(result.is_some());
1859            assert!(
1860                result
1861                    .unwrap()
1862                    .to_string_lossy()
1863                    .replace('\\', "/")
1864                    .contains("modules/src/helper.ts")
1865            );
1866        }
1867    }
1868
1869    mod source_index_fallback_tests {
1870        use super::*;
1871
1872        #[test]
1873        fn detects_dist_entry_in_output_dir() {
1874            assert!(is_entry_in_output_dir("./dist/esm2022/index.js"));
1875            assert!(is_entry_in_output_dir("dist/index.js"));
1876            assert!(is_entry_in_output_dir("./build/index.js"));
1877            assert!(is_entry_in_output_dir("./out/main.js"));
1878            assert!(is_entry_in_output_dir("./esm/index.js"));
1879            assert!(is_entry_in_output_dir("./cjs/index.js"));
1880        }
1881
1882        #[test]
1883        fn rejects_non_output_entry_paths() {
1884            assert!(!is_entry_in_output_dir("./src/index.ts"));
1885            assert!(!is_entry_in_output_dir("src/main.ts"));
1886            assert!(!is_entry_in_output_dir("./index.js"));
1887            assert!(!is_entry_in_output_dir(""));
1888        }
1889
1890        #[test]
1891        fn root_index_entries_are_recognized_for_source_fallback() {
1892            assert!(is_package_root_index_entry("./index.js"));
1893            assert!(is_package_root_index_entry("index.cjs"));
1894            assert!(is_package_root_index_entry("./index.d.ts"));
1895            assert!(!is_package_root_index_entry("./src/index.js"));
1896            assert!(!is_package_root_index_entry("./main.js"));
1897            assert!(!is_package_root_index_entry(""));
1898        }
1899
1900        #[test]
1901        fn rejects_substring_match_for_output_dir() {
1902            assert!(!is_entry_in_output_dir("./distro/index.js"));
1903            assert!(!is_entry_in_output_dir("./build-scripts/run.js"));
1904        }
1905
1906        #[test]
1907        fn finds_src_index_ts() {
1908            let dir = tempfile::tempdir().expect("create temp dir");
1909            let src = dir.path().join("src");
1910            std::fs::create_dir_all(&src).unwrap();
1911            let index_path = src.join("index.ts");
1912            std::fs::write(&index_path, "export const a = 1;").unwrap();
1913
1914            let result = try_source_index_fallback(dir.path());
1915            assert_eq!(result.as_deref(), Some(index_path.as_path()));
1916        }
1917
1918        #[test]
1919        fn finds_src_index_tsx_when_ts_missing() {
1920            let dir = tempfile::tempdir().expect("create temp dir");
1921            let src = dir.path().join("src");
1922            std::fs::create_dir_all(&src).unwrap();
1923            let index_path = src.join("index.tsx");
1924            std::fs::write(&index_path, "export default 1;").unwrap();
1925
1926            let result = try_source_index_fallback(dir.path());
1927            assert_eq!(result.as_deref(), Some(index_path.as_path()));
1928        }
1929
1930        #[test]
1931        fn prefers_src_index_over_root_index() {
1932            let dir = tempfile::tempdir().expect("create temp dir");
1933            let src = dir.path().join("src");
1934            std::fs::create_dir_all(&src).unwrap();
1935            let src_index = src.join("index.ts");
1936            std::fs::write(&src_index, "export const a = 1;").unwrap();
1937            let root_index = dir.path().join("index.ts");
1938            std::fs::write(&root_index, "export const b = 2;").unwrap();
1939
1940            let result = try_source_index_fallback(dir.path());
1941            assert_eq!(result.as_deref(), Some(src_index.as_path()));
1942        }
1943
1944        #[test]
1945        fn falls_back_to_src_main() {
1946            let dir = tempfile::tempdir().expect("create temp dir");
1947            let src = dir.path().join("src");
1948            std::fs::create_dir_all(&src).unwrap();
1949            let main_path = src.join("main.ts");
1950            std::fs::write(&main_path, "export const a = 1;").unwrap();
1951
1952            let result = try_source_index_fallback(dir.path());
1953            assert_eq!(result.as_deref(), Some(main_path.as_path()));
1954        }
1955
1956        #[test]
1957        fn falls_back_to_root_index_when_no_src() {
1958            let dir = tempfile::tempdir().expect("create temp dir");
1959            let index_path = dir.path().join("index.js");
1960            std::fs::write(&index_path, "module.exports = {};").unwrap();
1961
1962            let result = try_source_index_fallback(dir.path());
1963            assert_eq!(result.as_deref(), Some(index_path.as_path()));
1964        }
1965
1966        #[test]
1967        fn returns_none_when_nothing_matches() {
1968            let dir = tempfile::tempdir().expect("create temp dir");
1969            let result = try_source_index_fallback(dir.path());
1970            assert!(result.is_none());
1971        }
1972
1973        #[test]
1974        fn resolve_entry_path_falls_back_to_src_index_for_dist_entry() {
1975            let dir = tempfile::tempdir().expect("create temp dir");
1976            let canonical = dunce::canonicalize(dir.path()).unwrap();
1977
1978            let dist_dir = canonical.join("dist").join("esm2022");
1979            std::fs::create_dir_all(&dist_dir).unwrap();
1980            std::fs::write(dist_dir.join("index.js"), "export const x = 1;").unwrap();
1981
1982            let src = canonical.join("src");
1983            std::fs::create_dir_all(&src).unwrap();
1984            let src_index = src.join("index.ts");
1985            std::fs::write(&src_index, "export const x = 1;").unwrap();
1986
1987            let result = resolve_entry_path(
1988                &canonical,
1989                "./dist/esm2022/index.js",
1990                &canonical,
1991                EntryPointSource::PackageJsonMain,
1992            );
1993            assert!(result.is_some());
1994            let entry = result.unwrap();
1995            assert_eq!(entry.path, src_index);
1996        }
1997
1998        #[test]
1999        fn resolve_entry_path_uses_direct_src_mirror_when_available() {
2000            let dir = tempfile::tempdir().expect("create temp dir");
2001            let canonical = dunce::canonicalize(dir.path()).unwrap();
2002
2003            let src_mirror = canonical.join("src").join("esm2022");
2004            std::fs::create_dir_all(&src_mirror).unwrap();
2005            let mirror_index = src_mirror.join("index.ts");
2006            std::fs::write(&mirror_index, "export const x = 1;").unwrap();
2007
2008            let src_index = canonical.join("src").join("index.ts");
2009            std::fs::write(&src_index, "export const y = 2;").unwrap();
2010
2011            let result = resolve_entry_path(
2012                &canonical,
2013                "./dist/esm2022/index.js",
2014                &canonical,
2015                EntryPointSource::PackageJsonMain,
2016            );
2017            assert_eq!(result.map(|e| e.path), Some(mirror_index));
2018        }
2019
2020        #[test]
2021        fn resolve_entry_path_falls_back_to_src_index_for_missing_root_index() {
2022            let dir = tempfile::tempdir().expect("create temp dir");
2023            let canonical = dunce::canonicalize(dir.path()).unwrap();
2024
2025            let src = canonical.join("src");
2026            std::fs::create_dir_all(&src).unwrap();
2027            let src_index = src.join("index.ts");
2028            std::fs::write(&src_index, "export const x = 1;").unwrap();
2029
2030            let result = resolve_entry_path(
2031                &canonical,
2032                "./index.js",
2033                &canonical,
2034                EntryPointSource::PackageJsonMain,
2035            );
2036            assert_eq!(result.map(|entry| entry.path), Some(src_index));
2037        }
2038    }
2039
2040    mod default_fallback_tests {
2041        use super::*;
2042
2043        #[test]
2044        fn finds_src_index_ts_as_fallback() {
2045            let dir = tempfile::tempdir().expect("create temp dir");
2046            let src = dir.path().join("src");
2047            std::fs::create_dir_all(&src).unwrap();
2048            let index_path = src.join("index.ts");
2049            std::fs::write(&index_path, "export const a = 1;").unwrap();
2050
2051            let files = vec![DiscoveredFile {
2052                id: FileId(0),
2053                path: index_path.clone(),
2054                size_bytes: 20,
2055            }];
2056
2057            let entries = apply_default_fallback(&files, dir.path(), None);
2058            assert_eq!(entries.len(), 1);
2059            assert_eq!(entries[0].path, index_path);
2060            assert!(matches!(entries[0].source, EntryPointSource::DefaultIndex));
2061        }
2062
2063        #[test]
2064        fn finds_root_index_js_as_fallback() {
2065            let dir = tempfile::tempdir().expect("create temp dir");
2066            let index_path = dir.path().join("index.js");
2067            std::fs::write(&index_path, "module.exports = {};").unwrap();
2068
2069            let files = vec![DiscoveredFile {
2070                id: FileId(0),
2071                path: index_path.clone(),
2072                size_bytes: 21,
2073            }];
2074
2075            let entries = apply_default_fallback(&files, dir.path(), None);
2076            assert_eq!(entries.len(), 1);
2077            assert_eq!(entries[0].path, index_path);
2078        }
2079
2080        #[test]
2081        fn returns_empty_when_no_index_file() {
2082            let dir = tempfile::tempdir().expect("create temp dir");
2083            let other_path = dir.path().join("src").join("utils.ts");
2084
2085            let files = vec![DiscoveredFile {
2086                id: FileId(0),
2087                path: other_path,
2088                size_bytes: 10,
2089            }];
2090
2091            let entries = apply_default_fallback(&files, dir.path(), None);
2092            assert!(
2093                entries.is_empty(),
2094                "non-index files should not match default fallback"
2095            );
2096        }
2097
2098        #[test]
2099        fn workspace_filter_restricts_scope() {
2100            let dir = tempfile::tempdir().expect("create temp dir");
2101            let ws_a = dir.path().join("packages").join("a").join("src");
2102            std::fs::create_dir_all(&ws_a).unwrap();
2103            let ws_b = dir.path().join("packages").join("b").join("src");
2104            std::fs::create_dir_all(&ws_b).unwrap();
2105
2106            let index_a = ws_a.join("index.ts");
2107            let index_b = ws_b.join("index.ts");
2108
2109            let files = vec![
2110                DiscoveredFile {
2111                    id: FileId(0),
2112                    path: index_a.clone(),
2113                    size_bytes: 10,
2114                },
2115                DiscoveredFile {
2116                    id: FileId(1),
2117                    path: index_b,
2118                    size_bytes: 10,
2119                },
2120            ];
2121
2122            let ws_root = dir.path().join("packages").join("a");
2123            let entries = apply_default_fallback(&files, &ws_root, Some(&ws_root));
2124            assert_eq!(entries.len(), 1);
2125            assert_eq!(entries[0].path, index_a);
2126        }
2127    }
2128
2129    mod wildcard_entry_tests {
2130        use super::*;
2131
2132        #[test]
2133        fn expands_wildcard_css_entries() {
2134            let dir = tempfile::tempdir().expect("create temp dir");
2135            let themes = dir.path().join("src").join("themes");
2136            std::fs::create_dir_all(&themes).unwrap();
2137            std::fs::write(themes.join("dark.css"), ":root { --bg: #000; }").unwrap();
2138            std::fs::write(themes.join("light.css"), ":root { --bg: #fff; }").unwrap();
2139
2140            let canonical = dunce::canonicalize(dir.path()).unwrap();
2141            let mut entries = Vec::new();
2142            expand_wildcard_entries(dir.path(), "./src/themes/*.css", &canonical, &mut entries);
2143
2144            assert_eq!(entries.len(), 2, "should expand wildcard to 2 CSS files");
2145            let paths: Vec<String> = entries
2146                .iter()
2147                .map(|ep| ep.path.file_name().unwrap().to_string_lossy().to_string())
2148                .collect();
2149            assert!(paths.contains(&"dark.css".to_string()));
2150            assert!(paths.contains(&"light.css".to_string()));
2151            assert!(
2152                entries
2153                    .iter()
2154                    .all(|ep| matches!(ep.source, EntryPointSource::PackageJsonExports))
2155            );
2156        }
2157
2158        #[test]
2159        fn wildcard_does_not_match_nonexistent_files() {
2160            let dir = tempfile::tempdir().expect("create temp dir");
2161            std::fs::create_dir_all(dir.path().join("src/themes")).unwrap();
2162
2163            let canonical = dunce::canonicalize(dir.path()).unwrap();
2164            let mut entries = Vec::new();
2165            expand_wildcard_entries(dir.path(), "./src/themes/*.css", &canonical, &mut entries);
2166
2167            assert!(
2168                entries.is_empty(),
2169                "should return empty when no files match the wildcard"
2170            );
2171        }
2172
2173        #[test]
2174        fn wildcard_only_matches_specified_extension() {
2175            let dir = tempfile::tempdir().expect("create temp dir");
2176            let themes = dir.path().join("src").join("themes");
2177            std::fs::create_dir_all(&themes).unwrap();
2178            std::fs::write(themes.join("dark.css"), ":root {}").unwrap();
2179            std::fs::write(themes.join("index.ts"), "export {};").unwrap();
2180
2181            let canonical = dunce::canonicalize(dir.path()).unwrap();
2182            let mut entries = Vec::new();
2183            expand_wildcard_entries(dir.path(), "./src/themes/*.css", &canonical, &mut entries);
2184
2185            assert_eq!(entries.len(), 1, "should only match CSS files");
2186            assert!(
2187                entries[0]
2188                    .path
2189                    .file_name()
2190                    .unwrap()
2191                    .to_string_lossy()
2192                    .ends_with(".css")
2193            );
2194        }
2195    }
2196}