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 duplicates: fallow_config::DuplicatesConfig::default(),
1251 health: fallow_config::HealthConfig::default(),
1252 rules: RulesConfig::default(),
1253 boundaries: fallow_config::BoundaryConfig::default(),
1254 production: false.into(),
1255 plugins: vec![],
1256 rule_packs: vec![],
1257 dynamically_loaded: vec![],
1258 overrides: vec![],
1259 regression: None,
1260 audit: fallow_config::AuditConfig::default(),
1261 codeowners: None,
1262 public_packages: vec![],
1263 flags: fallow_config::FlagsConfig::default(),
1264 security: fallow_config::SecurityConfig::default(),
1265 fix: fallow_config::FixConfig::default(),
1266 resolve: fallow_config::ResolveConfig::default(),
1267 sealed: false,
1268 include_entry_exports: false,
1269 auto_imports: false,
1270 cache: fallow_config::CacheConfig::default(),
1271 }
1272 .resolve(root.to_path_buf(), OutputFormat::Human, 4, true, true, None);
1273
1274 let files = vec![
1275 DiscoveredFile {
1276 id: FileId(0),
1277 path: root.join("src/runtime.ts"),
1278 size_bytes: 1,
1279 },
1280 DiscoveredFile {
1281 id: FileId(1),
1282 path: root.join("src/setup.ts"),
1283 size_bytes: 1,
1284 },
1285 DiscoveredFile {
1286 id: FileId(2),
1287 path: root.join("tests/app.test.ts"),
1288 size_bytes: 1,
1289 },
1290 ];
1291
1292 let mut plugin_result = crate::plugins::AggregatedPluginResult::default();
1293 plugin_result.entry_patterns.push((
1294 crate::plugins::PathRule::new("src/runtime.ts"),
1295 "runtime-plugin".to_string(),
1296 ));
1297 plugin_result.entry_patterns.push((
1298 crate::plugins::PathRule::new("tests/app.test.ts"),
1299 "test-plugin".to_string(),
1300 ));
1301 plugin_result
1302 .always_used
1303 .push(("src/setup.ts".to_string(), "support-plugin".to_string()));
1304 plugin_result
1305 .entry_point_roles
1306 .insert("runtime-plugin".to_string(), EntryPointRole::Runtime);
1307 plugin_result
1308 .entry_point_roles
1309 .insert("test-plugin".to_string(), EntryPointRole::Test);
1310 plugin_result
1311 .entry_point_roles
1312 .insert("support-plugin".to_string(), EntryPointRole::Support);
1313
1314 let entries = discover_plugin_entry_point_sets(&plugin_result, &config, &files);
1315
1316 assert_eq!(entries.runtime.len(), 1, "expected one runtime entry");
1317 assert!(
1318 entries.runtime[0].path.ends_with("src/runtime.ts"),
1319 "runtime entry should stay runtime-only"
1320 );
1321 assert_eq!(entries.test.len(), 1, "expected one test entry");
1322 assert!(
1323 entries.test[0].path.ends_with("tests/app.test.ts"),
1324 "test entry should stay test-only"
1325 );
1326 assert_eq!(
1327 entries.all.len(),
1328 3,
1329 "support entries should stay in all entries"
1330 );
1331 assert!(
1332 entries
1333 .all
1334 .iter()
1335 .any(|entry| entry.path.ends_with("src/setup.ts")),
1336 "support entries should remain in the overall entry-point set"
1337 );
1338 assert!(
1339 !entries
1340 .runtime
1341 .iter()
1342 .any(|entry| entry.path.ends_with("src/setup.ts")),
1343 "support entries should not bleed into runtime reachability"
1344 );
1345 assert!(
1346 !entries
1347 .test
1348 .iter()
1349 .any(|entry| entry.path.ends_with("src/setup.ts")),
1350 "support entries should not bleed into test reachability"
1351 );
1352 }
1353
1354 #[test]
1355 fn resolve_plugin_setup_file_preserves_windows_absolute_path_on_any_host() {
1356 let root = Path::new("/workspace/project");
1357 let setup_file = Path::new(r"C:\workspace\project\setup.ts");
1358
1359 assert_eq!(
1360 resolve_plugin_setup_file(root, setup_file),
1361 setup_file.to_path_buf()
1362 );
1363 }
1364
1365 #[cfg(windows)]
1366 #[test]
1367 fn resolve_plugin_setup_file_preserves_posix_rooted_path_on_windows() {
1368 let root = Path::new(r"C:\workspace\project");
1369 let setup_file = Path::new(r"/workspace/project/setup.ts");
1370
1371 assert_eq!(
1372 resolve_plugin_setup_file(root, setup_file),
1373 setup_file.to_path_buf()
1374 );
1375 }
1376
1377 #[test]
1378 fn plugin_entry_point_rules_respect_exclusions() {
1379 let dir = tempfile::tempdir().expect("create temp dir");
1380 let root = dir.path();
1381 std::fs::create_dir_all(root.join("app/pages")).unwrap();
1382 std::fs::write(
1383 root.join("app/pages/index.tsx"),
1384 "export default function Page() { return null; }",
1385 )
1386 .unwrap();
1387 std::fs::write(
1388 root.join("app/pages/-helper.ts"),
1389 "export const helper = 1;",
1390 )
1391 .unwrap();
1392
1393 let config = FallowConfig {
1394 schema: None,
1395 extends: vec![],
1396 entry: vec![],
1397 ignore_patterns: vec![],
1398 framework: vec![],
1399 workspaces: None,
1400 ignore_dependencies: vec![],
1401 ignore_unresolved_imports: vec![],
1402 ignore_exports: vec![],
1403 ignore_catalog_references: vec![],
1404 ignore_dependency_overrides: vec![],
1405 ignore_exports_used_in_file: fallow_config::IgnoreExportsUsedInFileConfig::default(),
1406 used_class_members: vec![],
1407 ignore_decorators: vec![],
1408 duplicates: fallow_config::DuplicatesConfig::default(),
1409 health: fallow_config::HealthConfig::default(),
1410 rules: RulesConfig::default(),
1411 boundaries: fallow_config::BoundaryConfig::default(),
1412 production: false.into(),
1413 plugins: vec![],
1414 rule_packs: vec![],
1415 dynamically_loaded: vec![],
1416 overrides: vec![],
1417 regression: None,
1418 audit: fallow_config::AuditConfig::default(),
1419 codeowners: None,
1420 public_packages: vec![],
1421 flags: fallow_config::FlagsConfig::default(),
1422 security: fallow_config::SecurityConfig::default(),
1423 fix: fallow_config::FixConfig::default(),
1424 resolve: fallow_config::ResolveConfig::default(),
1425 sealed: false,
1426 include_entry_exports: false,
1427 auto_imports: false,
1428 cache: fallow_config::CacheConfig::default(),
1429 }
1430 .resolve(root.to_path_buf(), OutputFormat::Human, 4, true, true, None);
1431
1432 let files = vec![
1433 DiscoveredFile {
1434 id: FileId(0),
1435 path: root.join("app/pages/index.tsx"),
1436 size_bytes: 1,
1437 },
1438 DiscoveredFile {
1439 id: FileId(1),
1440 path: root.join("app/pages/-helper.ts"),
1441 size_bytes: 1,
1442 },
1443 ];
1444
1445 let mut plugin_result = crate::plugins::AggregatedPluginResult::default();
1446 plugin_result.entry_patterns.push((
1447 crate::plugins::PathRule::new("app/pages/**/*.{ts,tsx,js,jsx}")
1448 .with_excluded_globs(["app/pages/**/-*", "app/pages/**/-*/**/*"]),
1449 "tanstack-router".to_string(),
1450 ));
1451 plugin_result
1452 .entry_point_roles
1453 .insert("tanstack-router".to_string(), EntryPointRole::Runtime);
1454
1455 let entries = discover_plugin_entry_point_sets(&plugin_result, &config, &files);
1456 let entry_paths: Vec<_> = entries
1457 .all
1458 .iter()
1459 .map(|entry| {
1460 entry
1461 .path
1462 .strip_prefix(root)
1463 .unwrap()
1464 .to_string_lossy()
1465 .into_owned()
1466 })
1467 .collect();
1468
1469 assert!(entry_paths.contains(&"app/pages/index.tsx".to_string()));
1470 assert!(!entry_paths.contains(&"app/pages/-helper.ts".to_string()));
1471 }
1472
1473 mod resolve_entry_path_tests {
1474 use super::*;
1475
1476 #[test]
1477 fn resolves_existing_file() {
1478 let dir = tempfile::tempdir().expect("create temp dir");
1479 let src = dir.path().join("src");
1480 std::fs::create_dir_all(&src).unwrap();
1481 std::fs::write(src.join("index.ts"), "export const a = 1;").unwrap();
1482
1483 let canonical = dunce::canonicalize(dir.path()).unwrap();
1484 let result = resolve_entry_path(
1485 dir.path(),
1486 "src/index.ts",
1487 &canonical,
1488 EntryPointSource::PackageJsonMain,
1489 );
1490 assert!(result.is_some(), "should resolve an existing file");
1491 assert!(result.unwrap().path.ends_with("src/index.ts"));
1492 }
1493
1494 #[test]
1495 fn resolves_with_extension_fallback() {
1496 let dir = tempfile::tempdir().expect("create temp dir");
1497 let canonical = dunce::canonicalize(dir.path()).unwrap();
1498 let src = canonical.join("src");
1499 std::fs::create_dir_all(&src).unwrap();
1500 std::fs::write(src.join("index.ts"), "export const a = 1;").unwrap();
1501
1502 let result = resolve_entry_path(
1503 &canonical,
1504 "src/index",
1505 &canonical,
1506 EntryPointSource::PackageJsonMain,
1507 );
1508 assert!(
1509 result.is_some(),
1510 "should resolve via extension fallback when exact path doesn't exist"
1511 );
1512 let ep = result.unwrap();
1513 assert!(
1514 ep.path.to_string_lossy().contains("index.ts"),
1515 "should find index.ts via extension fallback"
1516 );
1517 }
1518
1519 #[test]
1520 fn exact_file_wins_before_directory_index_fallback() {
1521 let dir = tempfile::tempdir().expect("create temp dir");
1522 let canonical = dunce::canonicalize(dir.path()).unwrap();
1523 let scripts = canonical.join("scripts");
1524 std::fs::create_dir_all(scripts.join("process-messages")).unwrap();
1525 std::fs::write(
1526 scripts.join("process-messages.js"),
1527 "export const direct = true;",
1528 )
1529 .unwrap();
1530 std::fs::write(
1531 scripts.join("process-messages").join("index.js"),
1532 "export const index = true;",
1533 )
1534 .unwrap();
1535
1536 let result = resolve_entry_path(
1537 &canonical,
1538 "scripts/process-messages.js",
1539 &canonical,
1540 EntryPointSource::PackageJsonScript,
1541 )
1542 .expect("exact file should resolve");
1543
1544 assert!(result.path.ends_with("scripts/process-messages.js"));
1545 }
1546
1547 #[test]
1548 fn extension_fallback_wins_before_directory_index_fallback() {
1549 let dir = tempfile::tempdir().expect("create temp dir");
1550 let canonical = dunce::canonicalize(dir.path()).unwrap();
1551 let scripts = canonical.join("scripts");
1552 std::fs::create_dir_all(scripts.join("process-messages")).unwrap();
1553 std::fs::write(
1554 scripts.join("process-messages.ts"),
1555 "export const withExt = true;",
1556 )
1557 .unwrap();
1558 std::fs::write(
1559 scripts.join("process-messages").join("index.js"),
1560 "export const index = true;",
1561 )
1562 .unwrap();
1563
1564 let result = resolve_entry_path(
1565 &canonical,
1566 "scripts/process-messages",
1567 &canonical,
1568 EntryPointSource::PackageJsonScript,
1569 )
1570 .expect("extension fallback should resolve");
1571
1572 assert!(result.path.ends_with("scripts/process-messages.ts"));
1573 }
1574
1575 #[test]
1576 fn resolves_directory_index_after_exact_and_extension_fallbacks() {
1577 let dir = tempfile::tempdir().expect("create temp dir");
1578 let canonical = dunce::canonicalize(dir.path()).unwrap();
1579 let scripts = canonical.join("scripts/process-messages");
1580 std::fs::create_dir_all(&scripts).unwrap();
1581 std::fs::write(scripts.join("index.js"), "export const index = true;").unwrap();
1582
1583 let result = resolve_entry_path(
1584 &canonical,
1585 "scripts/process-messages",
1586 &canonical,
1587 EntryPointSource::PackageJsonScript,
1588 )
1589 .expect("directory index should resolve");
1590
1591 assert!(result.path.ends_with("scripts/process-messages/index.js"));
1592 }
1593
1594 #[test]
1595 fn directory_index_fallback_ignores_wildcards_and_url_like_entries() {
1596 let dir = tempfile::tempdir().expect("create temp dir");
1597 let canonical = dunce::canonicalize(dir.path()).unwrap();
1598 std::fs::create_dir_all(canonical.join("scripts/process-messages")).unwrap();
1599 std::fs::write(
1600 canonical.join("scripts/process-messages/index.js"),
1601 "export const index = true;",
1602 )
1603 .unwrap();
1604
1605 for entry in [
1606 "scripts/*",
1607 "https://example.com/scripts/process-messages",
1608 "@scope/package/scripts/process-messages",
1609 ] {
1610 let result = resolve_entry_path(
1611 &canonical,
1612 entry,
1613 &canonical,
1614 EntryPointSource::PackageJsonScript,
1615 );
1616 assert!(result.is_none(), "{entry} should not resolve");
1617 }
1618 }
1619
1620 #[test]
1621 fn returns_none_for_nonexistent_file() {
1622 let dir = tempfile::tempdir().expect("create temp dir");
1623 let canonical = dunce::canonicalize(dir.path()).unwrap();
1624 let result = resolve_entry_path(
1625 dir.path(),
1626 "does/not/exist.ts",
1627 &canonical,
1628 EntryPointSource::PackageJsonMain,
1629 );
1630 assert!(result.is_none(), "should return None for nonexistent files");
1631 }
1632
1633 #[test]
1634 fn maps_dist_output_to_src() {
1635 let dir = tempfile::tempdir().expect("create temp dir");
1636 let src = dir.path().join("src");
1637 std::fs::create_dir_all(&src).unwrap();
1638 std::fs::write(src.join("utils.ts"), "export const u = 1;").unwrap();
1639
1640 let dist = dir.path().join("dist");
1641 std::fs::create_dir_all(&dist).unwrap();
1642 std::fs::write(dist.join("utils.js"), "// compiled").unwrap();
1643
1644 let canonical = dunce::canonicalize(dir.path()).unwrap();
1645 let result = resolve_entry_path(
1646 dir.path(),
1647 "./dist/utils.js",
1648 &canonical,
1649 EntryPointSource::PackageJsonExports,
1650 );
1651 assert!(result.is_some(), "should resolve dist/ path to src/");
1652 let ep = result.unwrap();
1653 assert!(
1654 ep.path
1655 .to_string_lossy()
1656 .replace('\\', "/")
1657 .contains("src/utils.ts"),
1658 "should map ./dist/utils.js to src/utils.ts"
1659 );
1660 }
1661
1662 #[test]
1663 fn maps_build_output_to_src() {
1664 let dir = tempfile::tempdir().expect("create temp dir");
1665 let canonical = dunce::canonicalize(dir.path()).unwrap();
1666 let src = canonical.join("src");
1667 std::fs::create_dir_all(&src).unwrap();
1668 std::fs::write(src.join("index.tsx"), "export default () => {};").unwrap();
1669
1670 let result = resolve_entry_path(
1671 &canonical,
1672 "./build/index.js",
1673 &canonical,
1674 EntryPointSource::PackageJsonExports,
1675 );
1676 assert!(result.is_some(), "should map build/ output to src/");
1677 let ep = result.unwrap();
1678 assert!(
1679 ep.path
1680 .to_string_lossy()
1681 .replace('\\', "/")
1682 .contains("src/index.tsx"),
1683 "should map ./build/index.js to src/index.tsx"
1684 );
1685 }
1686
1687 #[test]
1688 fn preserves_entry_point_source() {
1689 let dir = tempfile::tempdir().expect("create temp dir");
1690 std::fs::write(dir.path().join("index.ts"), "export const a = 1;").unwrap();
1691
1692 let canonical = dunce::canonicalize(dir.path()).unwrap();
1693 let result = resolve_entry_path(
1694 dir.path(),
1695 "index.ts",
1696 &canonical,
1697 EntryPointSource::PackageJsonScript,
1698 );
1699 assert!(result.is_some());
1700 assert!(
1701 matches!(result.unwrap().source, EntryPointSource::PackageJsonScript),
1702 "should preserve the source kind"
1703 );
1704 }
1705 #[test]
1706 fn tracks_skipped_entries_without_logging_each_repeat() {
1707 let dir = tempfile::tempdir().expect("create temp dir");
1708 let canonical = dunce::canonicalize(dir.path()).unwrap();
1709 let mut skipped_entries = FxHashMap::default();
1710
1711 let result = resolve_entry_path_with_tracking(
1712 dir.path(),
1713 "../scripts/build.js",
1714 &canonical,
1715 EntryPointSource::PackageJsonScript,
1716 Some(&mut skipped_entries),
1717 );
1718
1719 assert!(result.is_none(), "unsafe entry should be skipped");
1720 assert_eq!(
1721 skipped_entries.get("../scripts/build.js"),
1722 Some(&1),
1723 "warning tracker should count the skipped path"
1724 );
1725 }
1726
1727 #[test]
1728 fn formats_skipped_entry_warning_with_counts() {
1729 let mut skipped_entries = FxHashMap::default();
1730 skipped_entries.insert("../../scripts/rm.mjs".to_owned(), 8);
1731 skipped_entries.insert("../utils/bar.js".to_owned(), 2);
1732
1733 let warning =
1734 format_skipped_entry_warning(&skipped_entries).expect("warning should be rendered");
1735
1736 assert_eq!(
1737 warning,
1738 "Skipped 10 package.json entry points outside project root or containing parent directory traversal: ../../scripts/rm.mjs (8x), ../utils/bar.js (2x)"
1739 );
1740 }
1741
1742 #[test]
1743 fn skipped_entry_summary_dedupes_identical_messages() {
1744 let message = format!(
1745 "Skipped 1 package.json entry point outside project root: ../../pkg-{}/bin/x",
1746 std::process::id()
1747 );
1748 assert!(
1749 should_warn_skipped_entry(&message),
1750 "first occurrence of a message emits"
1751 );
1752 assert!(
1753 !should_warn_skipped_entry(&message),
1754 "identical repeat is suppressed"
1755 );
1756 }
1757
1758 #[test]
1759 fn rejects_parent_dir_escape_for_exact_file() {
1760 let sandbox = tempfile::tempdir().expect("create sandbox");
1761 let root = sandbox.path().join("project");
1762 std::fs::create_dir_all(&root).unwrap();
1763 std::fs::write(
1764 sandbox.path().join("escape.ts"),
1765 "export const escape = true;",
1766 )
1767 .unwrap();
1768
1769 let canonical = dunce::canonicalize(&root).unwrap();
1770 let result = resolve_entry_path(
1771 &root,
1772 "../escape.ts",
1773 &canonical,
1774 EntryPointSource::PackageJsonMain,
1775 );
1776
1777 assert!(
1778 result.is_none(),
1779 "should reject exact paths that escape the root"
1780 );
1781 }
1782
1783 #[test]
1784 fn rejects_parent_dir_escape_via_extension_fallback() {
1785 let sandbox = tempfile::tempdir().expect("create sandbox");
1786 let root = sandbox.path().join("project");
1787 std::fs::create_dir_all(&root).unwrap();
1788 std::fs::write(
1789 sandbox.path().join("escape.ts"),
1790 "export const escape = true;",
1791 )
1792 .unwrap();
1793
1794 let canonical = dunce::canonicalize(&root).unwrap();
1795 let result = resolve_entry_path(
1796 &root,
1797 "../escape",
1798 &canonical,
1799 EntryPointSource::PackageJsonMain,
1800 );
1801
1802 assert!(
1803 result.is_none(),
1804 "should reject extension fallback paths that escape the root"
1805 );
1806 }
1807 }
1808
1809 mod output_to_source_tests {
1810 use super::*;
1811
1812 #[test]
1813 fn maps_dist_to_src_with_ts_extension() {
1814 let dir = tempfile::tempdir().expect("create temp dir");
1815 let src = dir.path().join("src");
1816 std::fs::create_dir_all(&src).unwrap();
1817 std::fs::write(src.join("utils.ts"), "export const u = 1;").unwrap();
1818
1819 let result = try_output_to_source_path(dir.path(), "./dist/utils.js");
1820 assert!(result.is_some());
1821 assert!(
1822 result
1823 .unwrap()
1824 .to_string_lossy()
1825 .replace('\\', "/")
1826 .contains("src/utils.ts")
1827 );
1828 }
1829
1830 #[test]
1831 fn returns_none_when_no_source_file_exists() {
1832 let dir = tempfile::tempdir().expect("create temp dir");
1833 let result = try_output_to_source_path(dir.path(), "./dist/missing.js");
1834 assert!(result.is_none());
1835 }
1836
1837 #[test]
1838 fn ignores_non_output_directories() {
1839 let dir = tempfile::tempdir().expect("create temp dir");
1840 let src = dir.path().join("src");
1841 std::fs::create_dir_all(&src).unwrap();
1842 std::fs::write(src.join("foo.ts"), "export const f = 1;").unwrap();
1843
1844 let result = try_output_to_source_path(dir.path(), "./lib/foo.js");
1845 assert!(result.is_none());
1846 }
1847
1848 #[test]
1849 fn maps_nested_output_path_preserving_prefix() {
1850 let dir = tempfile::tempdir().expect("create temp dir");
1851 let modules_src = dir.path().join("modules").join("src");
1852 std::fs::create_dir_all(&modules_src).unwrap();
1853 std::fs::write(modules_src.join("helper.ts"), "export const h = 1;").unwrap();
1854
1855 let result = try_output_to_source_path(dir.path(), "./modules/dist/helper.js");
1856 assert!(result.is_some());
1857 assert!(
1858 result
1859 .unwrap()
1860 .to_string_lossy()
1861 .replace('\\', "/")
1862 .contains("modules/src/helper.ts")
1863 );
1864 }
1865 }
1866
1867 mod source_index_fallback_tests {
1868 use super::*;
1869
1870 #[test]
1871 fn detects_dist_entry_in_output_dir() {
1872 assert!(is_entry_in_output_dir("./dist/esm2022/index.js"));
1873 assert!(is_entry_in_output_dir("dist/index.js"));
1874 assert!(is_entry_in_output_dir("./build/index.js"));
1875 assert!(is_entry_in_output_dir("./out/main.js"));
1876 assert!(is_entry_in_output_dir("./esm/index.js"));
1877 assert!(is_entry_in_output_dir("./cjs/index.js"));
1878 }
1879
1880 #[test]
1881 fn rejects_non_output_entry_paths() {
1882 assert!(!is_entry_in_output_dir("./src/index.ts"));
1883 assert!(!is_entry_in_output_dir("src/main.ts"));
1884 assert!(!is_entry_in_output_dir("./index.js"));
1885 assert!(!is_entry_in_output_dir(""));
1886 }
1887
1888 #[test]
1889 fn root_index_entries_are_recognized_for_source_fallback() {
1890 assert!(is_package_root_index_entry("./index.js"));
1891 assert!(is_package_root_index_entry("index.cjs"));
1892 assert!(is_package_root_index_entry("./index.d.ts"));
1893 assert!(!is_package_root_index_entry("./src/index.js"));
1894 assert!(!is_package_root_index_entry("./main.js"));
1895 assert!(!is_package_root_index_entry(""));
1896 }
1897
1898 #[test]
1899 fn rejects_substring_match_for_output_dir() {
1900 assert!(!is_entry_in_output_dir("./distro/index.js"));
1901 assert!(!is_entry_in_output_dir("./build-scripts/run.js"));
1902 }
1903
1904 #[test]
1905 fn finds_src_index_ts() {
1906 let dir = tempfile::tempdir().expect("create temp dir");
1907 let src = dir.path().join("src");
1908 std::fs::create_dir_all(&src).unwrap();
1909 let index_path = src.join("index.ts");
1910 std::fs::write(&index_path, "export const a = 1;").unwrap();
1911
1912 let result = try_source_index_fallback(dir.path());
1913 assert_eq!(result.as_deref(), Some(index_path.as_path()));
1914 }
1915
1916 #[test]
1917 fn finds_src_index_tsx_when_ts_missing() {
1918 let dir = tempfile::tempdir().expect("create temp dir");
1919 let src = dir.path().join("src");
1920 std::fs::create_dir_all(&src).unwrap();
1921 let index_path = src.join("index.tsx");
1922 std::fs::write(&index_path, "export default 1;").unwrap();
1923
1924 let result = try_source_index_fallback(dir.path());
1925 assert_eq!(result.as_deref(), Some(index_path.as_path()));
1926 }
1927
1928 #[test]
1929 fn prefers_src_index_over_root_index() {
1930 let dir = tempfile::tempdir().expect("create temp dir");
1931 let src = dir.path().join("src");
1932 std::fs::create_dir_all(&src).unwrap();
1933 let src_index = src.join("index.ts");
1934 std::fs::write(&src_index, "export const a = 1;").unwrap();
1935 let root_index = dir.path().join("index.ts");
1936 std::fs::write(&root_index, "export const b = 2;").unwrap();
1937
1938 let result = try_source_index_fallback(dir.path());
1939 assert_eq!(result.as_deref(), Some(src_index.as_path()));
1940 }
1941
1942 #[test]
1943 fn falls_back_to_src_main() {
1944 let dir = tempfile::tempdir().expect("create temp dir");
1945 let src = dir.path().join("src");
1946 std::fs::create_dir_all(&src).unwrap();
1947 let main_path = src.join("main.ts");
1948 std::fs::write(&main_path, "export const a = 1;").unwrap();
1949
1950 let result = try_source_index_fallback(dir.path());
1951 assert_eq!(result.as_deref(), Some(main_path.as_path()));
1952 }
1953
1954 #[test]
1955 fn falls_back_to_root_index_when_no_src() {
1956 let dir = tempfile::tempdir().expect("create temp dir");
1957 let index_path = dir.path().join("index.js");
1958 std::fs::write(&index_path, "module.exports = {};").unwrap();
1959
1960 let result = try_source_index_fallback(dir.path());
1961 assert_eq!(result.as_deref(), Some(index_path.as_path()));
1962 }
1963
1964 #[test]
1965 fn returns_none_when_nothing_matches() {
1966 let dir = tempfile::tempdir().expect("create temp dir");
1967 let result = try_source_index_fallback(dir.path());
1968 assert!(result.is_none());
1969 }
1970
1971 #[test]
1972 fn resolve_entry_path_falls_back_to_src_index_for_dist_entry() {
1973 let dir = tempfile::tempdir().expect("create temp dir");
1974 let canonical = dunce::canonicalize(dir.path()).unwrap();
1975
1976 let dist_dir = canonical.join("dist").join("esm2022");
1977 std::fs::create_dir_all(&dist_dir).unwrap();
1978 std::fs::write(dist_dir.join("index.js"), "export const x = 1;").unwrap();
1979
1980 let src = canonical.join("src");
1981 std::fs::create_dir_all(&src).unwrap();
1982 let src_index = src.join("index.ts");
1983 std::fs::write(&src_index, "export const x = 1;").unwrap();
1984
1985 let result = resolve_entry_path(
1986 &canonical,
1987 "./dist/esm2022/index.js",
1988 &canonical,
1989 EntryPointSource::PackageJsonMain,
1990 );
1991 assert!(result.is_some());
1992 let entry = result.unwrap();
1993 assert_eq!(entry.path, src_index);
1994 }
1995
1996 #[test]
1997 fn resolve_entry_path_uses_direct_src_mirror_when_available() {
1998 let dir = tempfile::tempdir().expect("create temp dir");
1999 let canonical = dunce::canonicalize(dir.path()).unwrap();
2000
2001 let src_mirror = canonical.join("src").join("esm2022");
2002 std::fs::create_dir_all(&src_mirror).unwrap();
2003 let mirror_index = src_mirror.join("index.ts");
2004 std::fs::write(&mirror_index, "export const x = 1;").unwrap();
2005
2006 let src_index = canonical.join("src").join("index.ts");
2007 std::fs::write(&src_index, "export const y = 2;").unwrap();
2008
2009 let result = resolve_entry_path(
2010 &canonical,
2011 "./dist/esm2022/index.js",
2012 &canonical,
2013 EntryPointSource::PackageJsonMain,
2014 );
2015 assert_eq!(result.map(|e| e.path), Some(mirror_index));
2016 }
2017
2018 #[test]
2019 fn resolve_entry_path_falls_back_to_src_index_for_missing_root_index() {
2020 let dir = tempfile::tempdir().expect("create temp dir");
2021 let canonical = dunce::canonicalize(dir.path()).unwrap();
2022
2023 let src = canonical.join("src");
2024 std::fs::create_dir_all(&src).unwrap();
2025 let src_index = src.join("index.ts");
2026 std::fs::write(&src_index, "export const x = 1;").unwrap();
2027
2028 let result = resolve_entry_path(
2029 &canonical,
2030 "./index.js",
2031 &canonical,
2032 EntryPointSource::PackageJsonMain,
2033 );
2034 assert_eq!(result.map(|entry| entry.path), Some(src_index));
2035 }
2036 }
2037
2038 mod default_fallback_tests {
2039 use super::*;
2040
2041 #[test]
2042 fn finds_src_index_ts_as_fallback() {
2043 let dir = tempfile::tempdir().expect("create temp dir");
2044 let src = dir.path().join("src");
2045 std::fs::create_dir_all(&src).unwrap();
2046 let index_path = src.join("index.ts");
2047 std::fs::write(&index_path, "export const a = 1;").unwrap();
2048
2049 let files = vec![DiscoveredFile {
2050 id: FileId(0),
2051 path: index_path.clone(),
2052 size_bytes: 20,
2053 }];
2054
2055 let entries = apply_default_fallback(&files, dir.path(), None);
2056 assert_eq!(entries.len(), 1);
2057 assert_eq!(entries[0].path, index_path);
2058 assert!(matches!(entries[0].source, EntryPointSource::DefaultIndex));
2059 }
2060
2061 #[test]
2062 fn finds_root_index_js_as_fallback() {
2063 let dir = tempfile::tempdir().expect("create temp dir");
2064 let index_path = dir.path().join("index.js");
2065 std::fs::write(&index_path, "module.exports = {};").unwrap();
2066
2067 let files = vec![DiscoveredFile {
2068 id: FileId(0),
2069 path: index_path.clone(),
2070 size_bytes: 21,
2071 }];
2072
2073 let entries = apply_default_fallback(&files, dir.path(), None);
2074 assert_eq!(entries.len(), 1);
2075 assert_eq!(entries[0].path, index_path);
2076 }
2077
2078 #[test]
2079 fn returns_empty_when_no_index_file() {
2080 let dir = tempfile::tempdir().expect("create temp dir");
2081 let other_path = dir.path().join("src").join("utils.ts");
2082
2083 let files = vec![DiscoveredFile {
2084 id: FileId(0),
2085 path: other_path,
2086 size_bytes: 10,
2087 }];
2088
2089 let entries = apply_default_fallback(&files, dir.path(), None);
2090 assert!(
2091 entries.is_empty(),
2092 "non-index files should not match default fallback"
2093 );
2094 }
2095
2096 #[test]
2097 fn workspace_filter_restricts_scope() {
2098 let dir = tempfile::tempdir().expect("create temp dir");
2099 let ws_a = dir.path().join("packages").join("a").join("src");
2100 std::fs::create_dir_all(&ws_a).unwrap();
2101 let ws_b = dir.path().join("packages").join("b").join("src");
2102 std::fs::create_dir_all(&ws_b).unwrap();
2103
2104 let index_a = ws_a.join("index.ts");
2105 let index_b = ws_b.join("index.ts");
2106
2107 let files = vec![
2108 DiscoveredFile {
2109 id: FileId(0),
2110 path: index_a.clone(),
2111 size_bytes: 10,
2112 },
2113 DiscoveredFile {
2114 id: FileId(1),
2115 path: index_b,
2116 size_bytes: 10,
2117 },
2118 ];
2119
2120 let ws_root = dir.path().join("packages").join("a");
2121 let entries = apply_default_fallback(&files, &ws_root, Some(&ws_root));
2122 assert_eq!(entries.len(), 1);
2123 assert_eq!(entries[0].path, index_a);
2124 }
2125 }
2126
2127 mod wildcard_entry_tests {
2128 use super::*;
2129
2130 #[test]
2131 fn expands_wildcard_css_entries() {
2132 let dir = tempfile::tempdir().expect("create temp dir");
2133 let themes = dir.path().join("src").join("themes");
2134 std::fs::create_dir_all(&themes).unwrap();
2135 std::fs::write(themes.join("dark.css"), ":root { --bg: #000; }").unwrap();
2136 std::fs::write(themes.join("light.css"), ":root { --bg: #fff; }").unwrap();
2137
2138 let canonical = dunce::canonicalize(dir.path()).unwrap();
2139 let mut entries = Vec::new();
2140 expand_wildcard_entries(dir.path(), "./src/themes/*.css", &canonical, &mut entries);
2141
2142 assert_eq!(entries.len(), 2, "should expand wildcard to 2 CSS files");
2143 let paths: Vec<String> = entries
2144 .iter()
2145 .map(|ep| ep.path.file_name().unwrap().to_string_lossy().to_string())
2146 .collect();
2147 assert!(paths.contains(&"dark.css".to_string()));
2148 assert!(paths.contains(&"light.css".to_string()));
2149 assert!(
2150 entries
2151 .iter()
2152 .all(|ep| matches!(ep.source, EntryPointSource::PackageJsonExports))
2153 );
2154 }
2155
2156 #[test]
2157 fn wildcard_does_not_match_nonexistent_files() {
2158 let dir = tempfile::tempdir().expect("create temp dir");
2159 std::fs::create_dir_all(dir.path().join("src/themes")).unwrap();
2160
2161 let canonical = dunce::canonicalize(dir.path()).unwrap();
2162 let mut entries = Vec::new();
2163 expand_wildcard_entries(dir.path(), "./src/themes/*.css", &canonical, &mut entries);
2164
2165 assert!(
2166 entries.is_empty(),
2167 "should return empty when no files match the wildcard"
2168 );
2169 }
2170
2171 #[test]
2172 fn wildcard_only_matches_specified_extension() {
2173 let dir = tempfile::tempdir().expect("create temp dir");
2174 let themes = dir.path().join("src").join("themes");
2175 std::fs::create_dir_all(&themes).unwrap();
2176 std::fs::write(themes.join("dark.css"), ":root {}").unwrap();
2177 std::fs::write(themes.join("index.ts"), "export {};").unwrap();
2178
2179 let canonical = dunce::canonicalize(dir.path()).unwrap();
2180 let mut entries = Vec::new();
2181 expand_wildcard_entries(dir.path(), "./src/themes/*.css", &canonical, &mut entries);
2182
2183 assert_eq!(entries.len(), 1, "should only match CSS files");
2184 assert!(
2185 entries[0]
2186 .path
2187 .file_name()
2188 .unwrap()
2189 .to_string_lossy()
2190 .ends_with(".css")
2191 );
2192 }
2193 }
2194}