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
10const 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
67fn 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#[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
155fn 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
194fn 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
207enum OutputDirEntry {
213 ShortCircuit(Option<EntryPoint>),
214 Continue,
215}
216
217fn 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
255fn 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}
392fn 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
436const SOURCE_INDEX_FALLBACK_STEMS: &[&str] = &["src/index", "src/main", "index", "main"];
439
440fn 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
454fn 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
472const 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
480fn 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
516fn 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#[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
563fn 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
600fn 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
671fn 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
717fn 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
761fn 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#[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#[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#[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
911fn 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
949fn 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
977fn 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
1006fn 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#[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#[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 #[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 #[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 #[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}