1use std::path::{Path, PathBuf};
7
8use rustc_hash::FxHashMap;
9use serde_json::Value;
10
11use fallow_types::discover::FileId;
12
13use super::path_info::{extract_package_name, is_bare_specifier, is_valid_package_name};
14use super::types::{OUTPUT_DIRS, PackageManifestInfo, ResolveContext, ResolveResult, SOURCE_EXTS};
15
16fn alias_match_remainder<'a>(specifier: &'a str, prefix: &str) -> Option<&'a str> {
25 let remainder = specifier.strip_prefix(prefix)?;
26 (remainder.is_empty() || prefix.ends_with('/') || remainder.starts_with('/'))
27 .then_some(remainder)
28}
29
30pub(super) fn try_path_alias_fallback(
37 ctx: &ResolveContext<'_>,
38 specifier: &str,
39) -> Option<ResolveResult> {
40 for (prefix, replacement) in ctx.path_aliases {
41 let Some(remainder) = alias_match_remainder(specifier, prefix) else {
42 continue;
43 };
44
45 let substituted = match (replacement.is_empty(), remainder.is_empty()) {
46 (true, _) => format!("./{remainder}"),
47 (false, true) => format!("./{replacement}"),
48 (false, false) => format!("./{replacement}/{remainder}"),
49 };
50
51 if let Ok(resolved) = ctx.resolver.resolve(ctx.root, &substituted) {
52 let resolved_path = resolved.path();
53 if let Some(&file_id) = ctx.raw_path_to_id.get(resolved_path) {
54 return Some(ResolveResult::InternalModule(file_id));
55 }
56 if let Ok(canonical) = dunce::canonicalize(resolved_path) {
57 if let Some(&file_id) = ctx.path_to_id.get(canonical.as_path()) {
58 return Some(ResolveResult::InternalModule(file_id));
59 }
60 if let Some(file_id) = try_source_fallback(&canonical, ctx.path_to_id) {
61 return Some(ResolveResult::InternalModule(file_id));
62 }
63 if let Some(file_id) =
64 try_pnpm_workspace_fallback(&canonical, ctx.path_to_id, ctx.workspace_roots)
65 {
66 return Some(ResolveResult::InternalModule(file_id));
67 }
68 if let Some(pkg_name) = extract_package_name_from_node_modules_path(&canonical) {
69 return Some(ResolveResult::NpmPackage(pkg_name));
70 }
71 return Some(ResolveResult::ExternalFile(canonical));
72 }
73 }
74 }
75 None
76}
77
78pub(super) fn try_scss_partial_fallback(
87 ctx: &ResolveContext<'_>,
88 from_file: &Path,
89 specifier: &str,
90) -> Option<ResolveResult> {
91 if specifier.contains(':') {
92 return None;
93 }
94
95 let spec_path = Path::new(specifier);
96 let filename = spec_path.file_name()?.to_str()?;
97
98 if filename.starts_with('_') {
99 return None;
100 }
101
102 let partial_filename = format!("_{filename}");
103 let partial_specifier = if let Some(parent) = spec_path.parent()
104 && !parent.as_os_str().is_empty()
105 {
106 format!("{}/{partial_filename}", parent.display())
107 } else {
108 partial_filename
109 };
110
111 if let Some(result) = try_resolve_scss(ctx, from_file, &partial_specifier) {
112 return Some(result);
113 }
114
115 let index_partial = format!("{specifier}/_index");
116 if let Some(result) = try_resolve_scss(ctx, from_file, &index_partial) {
117 return Some(result);
118 }
119
120 let index_plain = format!("{specifier}/index");
121 try_resolve_scss(ctx, from_file, &index_plain)
122}
123
124pub(super) fn try_css_extension_fallback(
134 ctx: &ResolveContext<'_>,
135 from_file: &Path,
136 specifier: &str,
137) -> Option<ResolveResult> {
138 if specifier.contains(':') {
139 return None;
140 }
141 let spec_path = Path::new(specifier);
142 let already_css_ext = spec_path
143 .extension()
144 .and_then(|e| e.to_str())
145 .is_some_and(|e| {
146 e.eq_ignore_ascii_case("css")
147 || e.eq_ignore_ascii_case("scss")
148 || e.eq_ignore_ascii_case("sass")
149 });
150 if already_css_ext {
151 return try_resolve_scss(ctx, from_file, specifier);
152 }
153 for ext in ["scss", "sass", "css"] {
154 let candidate = format!("{specifier}.{ext}");
155 if let Some(result) = try_resolve_scss(ctx, from_file, &candidate) {
156 return Some(result);
157 }
158 }
159 None
160}
161
162fn try_resolve_scss(
164 ctx: &ResolveContext<'_>,
165 from_file: &Path,
166 specifier: &str,
167) -> Option<ResolveResult> {
168 let resolved = ctx.resolver.resolve_file(from_file, specifier).ok()?;
169 let resolved_path = resolved.path();
170
171 if let Some(&file_id) = ctx.raw_path_to_id.get(resolved_path) {
172 return Some(ResolveResult::InternalModule(file_id));
173 }
174 if let Ok(canonical) = dunce::canonicalize(resolved_path)
175 && let Some(&file_id) = ctx.path_to_id.get(canonical.as_path())
176 {
177 return Some(ResolveResult::InternalModule(file_id));
178 }
179 None
180}
181
182pub(super) fn try_scss_include_path_fallback(
202 ctx: &ResolveContext<'_>,
203 from_file: &Path,
204 specifier: &str,
205 from_style: bool,
206) -> Option<ResolveResult> {
207 if ctx.scss_include_paths.is_empty() {
208 return None;
209 }
210 let is_scss_importer = from_file
211 .extension()
212 .is_some_and(|e| e == "scss" || e == "sass");
213 if !is_scss_importer && !from_style {
214 return None;
215 }
216 if specifier.contains(':') {
217 return None;
218 }
219 let bare = specifier.strip_prefix("./")?;
220 if bare.starts_with("..") || bare.starts_with('/') {
221 return None;
222 }
223
224 for include_dir in ctx.scss_include_paths {
225 if let Some(file_id) = find_scss_in_dir(include_dir, bare, ctx) {
226 return Some(ResolveResult::InternalModule(file_id));
227 }
228 }
229 None
230}
231
232fn find_scss_in_dir(include_dir: &Path, bare: &str, ctx: &ResolveContext<'_>) -> Option<FileId> {
236 let bare_path = Path::new(bare);
237 let has_scss_ext = matches!(
238 bare_path.extension().and_then(|e| e.to_str()),
239 Some(ext) if ext.eq_ignore_ascii_case("scss") || ext.eq_ignore_ascii_case("sass")
240 );
241
242 let parent = bare_path.parent();
243 let stem_with_ext = bare_path.file_name()?.to_str()?;
244 let stem_without_ext = bare_path.file_stem().and_then(|s| s.to_str())?;
245
246 let build = |rel: &Path| -> std::path::PathBuf { include_dir.join(rel) };
247 let join_with_parent = |name: &str| -> std::path::PathBuf {
248 parent.map_or_else(|| build(Path::new(name)), |p| build(&p.join(name)))
249 };
250
251 let exts: &[&str] = if has_scss_ext {
252 &[""]
253 } else {
254 &["scss", "sass"]
255 };
256
257 for ext in exts {
258 let suffix = if ext.is_empty() {
259 String::new()
260 } else {
261 format!(".{ext}")
262 };
263 let direct = if ext.is_empty() {
264 build(bare_path)
265 } else {
266 join_with_parent(&format!("{stem_with_ext}{suffix}"))
267 };
268 if let Some(fid) = lookup_scss_path(&direct, ctx) {
269 return Some(fid);
270 }
271 let partial_name = if ext.is_empty() {
272 format!("_{stem_with_ext}")
273 } else {
274 format!("_{stem_without_ext}{suffix}")
275 };
276 let partial = join_with_parent(&partial_name);
277 if let Some(fid) = lookup_scss_path(&partial, ctx) {
278 return Some(fid);
279 }
280 if ext.is_empty() {
281 continue;
282 }
283 let idx_partial = build(bare_path).join(format!("_index{suffix}"));
284 if let Some(fid) = lookup_scss_path(&idx_partial, ctx) {
285 return Some(fid);
286 }
287 let idx_plain = build(bare_path).join(format!("index{suffix}"));
288 if let Some(fid) = lookup_scss_path(&idx_plain, ctx) {
289 return Some(fid);
290 }
291 }
292 None
293}
294
295fn lookup_scss_path(candidate: &Path, ctx: &ResolveContext<'_>) -> Option<FileId> {
298 if let Some(&file_id) = ctx.raw_path_to_id.get(candidate) {
299 return Some(file_id);
300 }
301 if let Ok(canonical) = dunce::canonicalize(candidate) {
302 if let Some(&file_id) = ctx.path_to_id.get(canonical.as_path()) {
303 return Some(file_id);
304 }
305 if let Some(fallback) = ctx.canonical_fallback
306 && let Some(file_id) = fallback.get(&canonical)
307 {
308 return Some(file_id);
309 }
310 }
311 None
312}
313
314pub(super) fn try_scss_node_modules_fallback(
336 _ctx: &ResolveContext<'_>,
337 from_file: &Path,
338 specifier: &str,
339 from_style: bool,
340) -> Option<ResolveResult> {
341 if specifier.contains(':') {
342 return None;
343 }
344 let is_scss_importer = from_file
345 .extension()
346 .is_some_and(|e| e == "scss" || e == "sass");
347 if !is_scss_importer && !from_style {
348 return None;
349 }
350 let bare = specifier.strip_prefix("./")?;
351 if bare.starts_with("..") || bare.starts_with('/') {
352 return None;
353 }
354 if bare.is_empty() {
355 return None;
356 }
357
358 let mut dir = from_file.parent()?;
359 loop {
360 let nm_dir = dir.join("node_modules");
361 if nm_dir.is_dir()
362 && let Some(path) = find_scss_in_node_modules(&nm_dir, bare)
363 && let Some(pkg_name) = extract_package_name_from_node_modules_path(&path)
364 {
365 return Some(ResolveResult::NpmPackage(pkg_name));
366 }
367 let Some(parent) = dir.parent() else {
368 break;
369 };
370 dir = parent;
371 }
372 None
373}
374
375fn find_scss_in_node_modules(nm_dir: &Path, bare: &str) -> Option<PathBuf> {
384 let bare_path = Path::new(bare);
385 let file_name = bare_path.file_name()?.to_str()?;
386 let parent = bare_path.parent();
387 let join_with_parent = |name: &str| -> PathBuf {
388 parent.map_or_else(|| nm_dir.join(name), |p| nm_dir.join(p).join(name))
389 };
390
391 for ext in &["scss", "sass", "css"] {
392 let candidate = join_with_parent(&format!("{file_name}.{ext}"));
393 if candidate.is_file() {
394 return Some(candidate);
395 }
396 }
397 for ext in &["scss", "sass"] {
398 let candidate = join_with_parent(&format!("_{file_name}.{ext}"));
399 if candidate.is_file() {
400 return Some(candidate);
401 }
402 }
403 for ext in &["scss", "sass"] {
404 let idx_partial = nm_dir.join(bare).join(format!("_index.{ext}"));
405 if idx_partial.is_file() {
406 return Some(idx_partial);
407 }
408 let idx_plain = nm_dir.join(bare).join(format!("index.{ext}"));
409 if idx_plain.is_file() {
410 return Some(idx_plain);
411 }
412 }
413 let exact = nm_dir.join(bare);
414 if exact.is_file() {
415 return Some(exact);
416 }
417 None
418}
419
420pub(super) fn try_source_fallback(
432 resolved: &Path,
433 path_to_id: &FxHashMap<&Path, FileId>,
434) -> Option<FileId> {
435 let components: Vec<_> = resolved.components().collect();
436
437 let is_output_dir = |c: &std::path::Component| -> bool {
438 if let std::path::Component::Normal(s) = c
439 && let Some(name) = s.to_str()
440 {
441 return OUTPUT_DIRS.contains(&name);
442 }
443 false
444 };
445
446 let last_output_pos = components.iter().rposition(&is_output_dir)?;
447
448 let mut first_output_pos = last_output_pos;
449 while first_output_pos > 0 && is_output_dir(&components[first_output_pos - 1]) {
450 first_output_pos -= 1;
451 }
452
453 let prefix: PathBuf = components[..first_output_pos].iter().collect();
454
455 let suffix: PathBuf = components[last_output_pos + 1..].iter().collect();
456 suffix.file_stem()?; for ext in SOURCE_EXTS {
459 let source_candidate = prefix.join("src").join(suffix.with_extension(ext));
460 if let Some(&file_id) = path_to_id.get(source_candidate.as_path()) {
461 return Some(file_id);
462 }
463 }
464
465 None
466}
467
468pub(super) fn try_package_imports_fallback(
474 ctx: &ResolveContext<'_>,
475 from_file: &Path,
476 specifier: &str,
477) -> Option<ResolveResult> {
478 if !specifier.starts_with('#') {
479 return None;
480 }
481 let manifest = nearest_package_manifest(ctx.package_manifests, from_file)?;
482 let imports = manifest.package_json.imports.as_ref()?;
483 let PackageMapTarget::Targets(targets) =
484 package_map_target(imports, specifier, ctx.condition_names)
485 else {
486 return None;
487 };
488 let source_subpath = package_import_source_subpath(manifest, specifier);
489 resolve_package_import_targets(ctx, manifest, &targets, source_subpath.as_deref()).map(
490 |target| match target {
491 PackageImportTarget::Internal(file_id) => match &manifest.name {
492 Some(package_name) => ResolveResult::InternalPackageModule {
493 file_id,
494 package_name: package_name.clone(),
495 },
496 None => ResolveResult::InternalModule(file_id),
497 },
498 PackageImportTarget::ExternalPackage(package_name) => {
499 ResolveResult::NpmPackage(package_name)
500 }
501 },
502 )
503}
504
505pub(super) fn try_relative_package_root_source_fallback(
508 ctx: &ResolveContext<'_>,
509 from_file: &Path,
510 specifier: &str,
511) -> Option<ResolveResult> {
512 if !specifier.starts_with("./") && !specifier.starts_with("../") {
513 return None;
514 }
515
516 let from_dir = from_file.parent()?;
517 let candidate = from_dir.join(specifier);
518 let normalized_candidate = normalize_path_lexically(&candidate);
519 #[cfg(not(miri))]
520 let canonical_candidate = dunce::canonicalize(&candidate).ok();
521 #[cfg(miri)]
522 let canonical_candidate: Option<PathBuf> = None;
523
524 ctx.package_manifests.iter().find_map(|manifest| {
525 let matches_manifest = candidate == manifest.root
526 || normalized_candidate == manifest.root
527 || canonical_candidate
528 .as_deref()
529 .is_some_and(|canonical| canonical == manifest.canonical_root);
530 matches_manifest
531 .then(|| try_source_subpath(ctx, manifest, Path::new("")))
532 .flatten()
533 .map(ResolveResult::InternalModule)
534 })
535}
536
537fn normalize_path_lexically(path: &Path) -> PathBuf {
538 let mut normalized = PathBuf::new();
539 for component in path.components() {
540 match component {
541 std::path::Component::CurDir => {}
542 std::path::Component::ParentDir => {
543 if !normalized.pop() {
544 normalized.push(component.as_os_str());
545 }
546 }
547 std::path::Component::Prefix(_)
548 | std::path::Component::RootDir
549 | std::path::Component::Normal(_) => normalized.push(component.as_os_str()),
550 }
551 }
552 normalized
553}
554
555#[derive(Debug, Clone, PartialEq, Eq)]
556enum PackageMapTarget {
557 NoMatch,
558 Blocked,
559 Targets(Vec<String>),
560}
561
562enum PackageImportTarget {
563 Internal(FileId),
564 ExternalPackage(String),
565}
566
567fn package_map_match_value(
568 value: &Value,
569 condition_names: &[String],
570 capture: Option<&str>,
571) -> PackageMapTarget {
572 resolve_package_map_value(value, condition_names, capture)
573 .filter(|targets| !targets.is_empty())
574 .map_or(PackageMapTarget::Blocked, PackageMapTarget::Targets)
575}
576
577fn package_map_target(
578 map: &Value,
579 specifier_key: &str,
580 condition_names: &[String],
581) -> PackageMapTarget {
582 let Some(obj) = map.as_object() else {
583 if specifier_key == "." {
584 return package_map_match_value(map, condition_names, None);
585 }
586 return PackageMapTarget::NoMatch;
587 };
588
589 let has_subpath_keys = obj
590 .keys()
591 .any(|key| key == "." || key.starts_with("./") || key.starts_with('#'));
592 if !has_subpath_keys {
593 if specifier_key == "." {
594 return package_map_match_value(map, condition_names, None);
595 }
596 return PackageMapTarget::NoMatch;
597 }
598
599 if let Some(value) = obj.get(specifier_key) {
600 return package_map_match_value(value, condition_names, None);
601 }
602
603 let mut patterns: Vec<(&str, &Value, String)> = obj
604 .iter()
605 .filter_map(|(pattern, value)| {
606 package_map_pattern_capture(pattern, specifier_key)
607 .map(|capture| (pattern.as_str(), value, capture))
608 })
609 .collect();
610 patterns.sort_by(|(left, _, _), (right, _, _)| {
611 package_map_pattern_specificity(right).cmp(&package_map_pattern_specificity(left))
612 });
613
614 patterns
615 .first()
616 .map_or(PackageMapTarget::NoMatch, |(_, value, capture)| {
617 package_map_match_value(value, condition_names, Some(capture))
618 })
619}
620
621fn resolve_package_map_value(
622 value: &Value,
623 condition_names: &[String],
624 capture: Option<&str>,
625) -> Option<Vec<String>> {
626 match value {
627 Value::String(target) => Some(vec![match capture {
628 Some(capture) => target.replace('*', capture),
629 None => target.clone(),
630 }]),
631 Value::Object(map) => {
632 for (condition, value) in map {
633 if (condition == "default"
634 || condition_names
635 .iter()
636 .any(|active_condition| active_condition == condition))
637 && let Some(targets) =
638 resolve_package_map_value(value, condition_names, capture)
639 {
640 return Some(targets);
641 }
642 }
643 None
644 }
645 Value::Array(values) => {
646 let targets: Vec<String> = values
647 .iter()
648 .filter_map(|value| resolve_package_map_value(value, condition_names, capture))
649 .flatten()
650 .collect();
651 (!targets.is_empty()).then_some(targets)
652 }
653 Value::Bool(_) | Value::Null | Value::Number(_) => None,
654 }
655}
656
657fn package_map_pattern_capture(pattern: &str, specifier: &str) -> Option<String> {
658 let star = pattern.find('*')?;
659 if pattern[star + 1..].contains('*') {
660 return None;
661 }
662 let (prefix, suffix_with_star) = pattern.split_at(star);
663 let suffix = &suffix_with_star[1..];
664 let captured = specifier.strip_prefix(prefix)?.strip_suffix(suffix)?;
665 Some(captured.to_string())
666}
667
668fn package_map_pattern_specificity(pattern: &str) -> (usize, usize) {
669 let star = pattern.find('*').unwrap_or(pattern.len());
670 (star, pattern.len())
671}
672
673fn package_import_source_subpath(
674 manifest: &PackageManifestInfo,
675 specifier: &str,
676) -> Option<PathBuf> {
677 let stripped = specifier.strip_prefix('#')?;
678 let without_package_name = manifest
679 .name
680 .as_deref()
681 .and_then(|name| stripped.strip_prefix(name))
682 .and_then(|rest| rest.strip_prefix('/'))
683 .unwrap_or(stripped);
684 if without_package_name.is_empty() {
685 None
686 } else {
687 Some(PathBuf::from(without_package_name))
688 }
689}
690
691fn nearest_package_manifest<'a>(
692 manifests: &'a [PackageManifestInfo],
693 from_file: &Path,
694) -> Option<&'a PackageManifestInfo> {
695 manifests
696 .iter()
697 .filter(|manifest| {
698 from_file.starts_with(&manifest.root) || from_file.starts_with(&manifest.canonical_root)
699 })
700 .max_by_key(|manifest| manifest.root.components().count())
701}
702
703fn find_package_manifest<'a>(
704 manifests: &'a [PackageManifestInfo],
705 package_name: &str,
706) -> Option<&'a PackageManifestInfo> {
707 manifests
708 .iter()
709 .find(|manifest| manifest.name.as_deref() == Some(package_name))
710}
711
712fn resolve_package_map_target(
713 ctx: &ResolveContext<'_>,
714 manifest: &PackageManifestInfo,
715 target: &str,
716 source_subpath: Option<&Path>,
717) -> Option<FileId> {
718 let target = target.strip_prefix("./")?;
719 if target.starts_with("../") || target.starts_with('/') {
720 return None;
721 }
722 let target_path = manifest.root.join(target);
723
724 lookup_internal_file_id(ctx, &target_path)
725 .or_else(|| try_source_fallback(&target_path, ctx.raw_path_to_id))
726 .or_else(|| try_source_fallback(&target_path, ctx.path_to_id))
727 .or_else(|| source_subpath.and_then(|subpath| try_source_subpath(ctx, manifest, subpath)))
728}
729
730fn resolve_package_map_targets(
731 ctx: &ResolveContext<'_>,
732 manifest: &PackageManifestInfo,
733 targets: &[String],
734 source_subpath: Option<&Path>,
735) -> Option<FileId> {
736 targets
737 .iter()
738 .find_map(|target| resolve_package_map_target(ctx, manifest, target, source_subpath))
739}
740
741fn resolve_package_import_targets(
742 ctx: &ResolveContext<'_>,
743 manifest: &PackageManifestInfo,
744 targets: &[String],
745 source_subpath: Option<&Path>,
746) -> Option<PackageImportTarget> {
747 targets.iter().find_map(|target| {
748 resolve_package_map_target(ctx, manifest, target, source_subpath)
749 .map(PackageImportTarget::Internal)
750 .or_else(|| {
751 package_import_external_target(target).map(PackageImportTarget::ExternalPackage)
752 })
753 })
754}
755
756fn package_import_external_target(target: &str) -> Option<String> {
757 if is_bare_specifier(target) && is_valid_package_name(target) {
758 Some(extract_package_name(target))
759 } else {
760 None
761 }
762}
763
764fn try_source_subpath(
765 ctx: &ResolveContext<'_>,
766 manifest: &PackageManifestInfo,
767 subpath: &Path,
768) -> Option<FileId> {
769 if subpath.as_os_str().is_empty()
770 && let Some(source) = manifest.package_json.source.as_deref()
771 && let Some(source_path) = safe_relative_package_source_path(source)
772 && let Some(file_id) = lookup_internal_file_id(ctx, &manifest.root.join(source_path))
773 {
774 return Some(file_id);
775 }
776
777 for ext in SOURCE_EXTS {
778 let direct = if subpath.as_os_str().is_empty() {
779 manifest.root.join("src").join(format!("index.{ext}"))
780 } else {
781 manifest.root.join("src").join(subpath).with_extension(ext)
782 };
783 if let Some(file_id) = lookup_internal_file_id(ctx, &direct) {
784 return Some(file_id);
785 }
786
787 if !subpath.as_os_str().is_empty() {
788 let index = manifest
789 .root
790 .join("src")
791 .join(subpath)
792 .join(format!("index.{ext}"));
793 if let Some(file_id) = lookup_internal_file_id(ctx, &index) {
794 return Some(file_id);
795 }
796 }
797
798 if subpath.as_os_str().is_empty() {
799 let root_index = manifest.root.join(format!("index.{ext}"));
800 if let Some(file_id) = lookup_internal_file_id(ctx, &root_index) {
801 return Some(file_id);
802 }
803 }
804 }
805
806 None
807}
808
809fn safe_relative_package_source_path(source: &str) -> Option<&Path> {
810 let source = source.strip_prefix("./").unwrap_or(source);
811 let path = Path::new(source);
812 if path.as_os_str().is_empty()
813 || path.components().any(|component| {
814 matches!(
815 component,
816 std::path::Component::ParentDir
817 | std::path::Component::RootDir
818 | std::path::Component::Prefix(_)
819 )
820 })
821 {
822 None
823 } else {
824 Some(path)
825 }
826}
827
828fn lookup_internal_file_id(ctx: &ResolveContext<'_>, candidate: &Path) -> Option<FileId> {
829 if let Some(&file_id) = ctx.raw_path_to_id.get(candidate) {
830 return Some(file_id);
831 }
832 if let Some(&file_id) = ctx.path_to_id.get(candidate) {
833 return Some(file_id);
834 }
835 #[cfg(not(miri))]
836 if let Ok(canonical) = dunce::canonicalize(candidate) {
837 if let Some(&file_id) = ctx.path_to_id.get(canonical.as_path()) {
838 return Some(file_id);
839 }
840 if let Some(fallback) = ctx.canonical_fallback
841 && let Some(file_id) = fallback.get(&canonical)
842 {
843 return Some(file_id);
844 }
845 }
846 None
847}
848
849pub fn extract_package_name_from_node_modules_path(path: &Path) -> Option<String> {
855 let components: Vec<&str> = path
856 .components()
857 .filter_map(|c| match c {
858 std::path::Component::Normal(s) => s.to_str(),
859 _ => None,
860 })
861 .collect();
862
863 let nm_idx = components.iter().rposition(|&c| c == "node_modules")?;
864
865 let after = &components[nm_idx + 1..];
866 if after.is_empty() {
867 return None;
868 }
869
870 if after[0].starts_with('@') {
871 if after.len() >= 2 {
872 Some(format!("{}/{}", after[0], after[1]))
873 } else {
874 Some(after[0].to_string())
875 }
876 } else {
877 Some(after[0].to_string())
878 }
879}
880
881pub(super) fn try_pnpm_workspace_fallback(
890 path: &Path,
891 path_to_id: &FxHashMap<&Path, FileId>,
892 workspace_roots: &FxHashMap<&str, &Path>,
893) -> Option<FileId> {
894 let components: Vec<&str> = path
895 .components()
896 .filter_map(|c| match c {
897 std::path::Component::Normal(s) => s.to_str(),
898 _ => None,
899 })
900 .collect();
901
902 let pnpm_idx = components.iter().position(|&c| c == ".pnpm")?;
903
904 let after_pnpm = &components[pnpm_idx + 1..];
905
906 let inner_nm_idx = after_pnpm.iter().position(|&c| c == "node_modules")?;
907 let after_inner_nm = &after_pnpm[inner_nm_idx + 1..];
908
909 if after_inner_nm.is_empty() {
910 return None;
911 }
912
913 let (pkg_name, pkg_name_components) = if after_inner_nm[0].starts_with('@') {
914 if after_inner_nm.len() >= 2 {
915 (format!("{}/{}", after_inner_nm[0], after_inner_nm[1]), 2)
916 } else {
917 return None;
918 }
919 } else {
920 (after_inner_nm[0].to_string(), 1)
921 };
922
923 let ws_root = workspace_roots.get(pkg_name.as_str())?;
924
925 let relative_parts = &after_inner_nm[pkg_name_components..];
926 if relative_parts.is_empty() {
927 return None;
928 }
929
930 let relative_path: PathBuf = relative_parts.iter().collect();
931
932 let direct = ws_root.join(&relative_path);
933 if let Some(&file_id) = path_to_id.get(direct.as_path()) {
934 return Some(file_id);
935 }
936
937 try_source_fallback(&direct, path_to_id)
938}
939
940pub(super) fn try_workspace_package_fallback(
963 ctx: &ResolveContext<'_>,
964 specifier: &str,
965) -> Option<ResolveResult> {
966 if !super::path_info::is_bare_specifier(specifier) {
967 return None;
968 }
969 let pkg_name = super::path_info::extract_package_name(specifier);
970
971 let subpath = specifier
972 .strip_prefix(pkg_name.as_str())
973 .and_then(|s| s.strip_prefix('/'))
974 .unwrap_or("");
975 let source_subpath = PathBuf::from(subpath);
976
977 match try_manifest_workspace_resolution(ctx, &pkg_name, subpath, &source_subpath) {
978 ManifestWorkspaceResolution::Resolved(result) => return Some(result),
979 ManifestWorkspaceResolution::Blocked => return None,
980 ManifestWorkspaceResolution::Continue => {}
981 }
982
983 let ws_root =
984 if let Some(manifest) = find_package_manifest(ctx.package_manifests, pkg_name.as_str()) {
985 manifest.root.as_path()
986 } else {
987 *ctx.workspace_roots.get(pkg_name.as_str())?
988 };
989
990 resolve_workspace_self_reference(ctx, ws_root, subpath, pkg_name)
991}
992
993enum ManifestWorkspaceResolution {
996 Resolved(ResolveResult),
998 Blocked,
1000 Continue,
1002}
1003
1004fn try_manifest_workspace_resolution(
1007 ctx: &ResolveContext<'_>,
1008 pkg_name: &str,
1009 subpath: &str,
1010 source_subpath: &Path,
1011) -> ManifestWorkspaceResolution {
1012 let Some(manifest) = find_package_manifest(ctx.package_manifests, pkg_name) else {
1013 return ManifestWorkspaceResolution::Continue;
1014 };
1015
1016 if let Some(exports) = manifest.package_json.exports.as_ref() {
1017 let export_key = if subpath.is_empty() {
1018 ".".to_string()
1019 } else {
1020 format!("./{subpath}")
1021 };
1022 return match package_map_target(exports, &export_key, ctx.condition_names) {
1023 PackageMapTarget::Targets(targets) => {
1024 match resolve_package_map_targets(ctx, manifest, &targets, Some(source_subpath)) {
1025 Some(file_id) => ManifestWorkspaceResolution::Resolved(
1026 ResolveResult::InternalPackageModule {
1027 file_id,
1028 package_name: pkg_name.to_string(),
1029 },
1030 ),
1031 None => ManifestWorkspaceResolution::Continue,
1032 }
1033 }
1034 PackageMapTarget::NoMatch | PackageMapTarget::Blocked => {
1035 ManifestWorkspaceResolution::Blocked
1036 }
1037 };
1038 }
1039
1040 if let Some(file_id) = try_source_subpath(ctx, manifest, source_subpath) {
1041 return ManifestWorkspaceResolution::Resolved(ResolveResult::InternalPackageModule {
1042 file_id,
1043 package_name: pkg_name.to_string(),
1044 });
1045 }
1046
1047 ManifestWorkspaceResolution::Continue
1048}
1049
1050fn resolve_workspace_self_reference(
1054 ctx: &ResolveContext<'_>,
1055 ws_root: &Path,
1056 subpath: &str,
1057 package_name: String,
1058) -> Option<ResolveResult> {
1059 let root_file = ws_root.join("__fallow_ws_self_resolve__");
1060 let rel_spec = if subpath.is_empty() {
1061 "./".to_string()
1062 } else {
1063 format!("./{subpath}")
1064 };
1065
1066 let resolved = ctx.resolver.resolve_file(&root_file, &rel_spec).ok()?;
1067 let resolved_path = resolved.path();
1068
1069 if let Some(&file_id) = ctx.raw_path_to_id.get(resolved_path) {
1070 return Some(ResolveResult::InternalPackageModule {
1071 file_id,
1072 package_name,
1073 });
1074 }
1075 if let Ok(canonical) = dunce::canonicalize(resolved_path) {
1076 if let Some(&file_id) = ctx.path_to_id.get(canonical.as_path()) {
1077 return Some(ResolveResult::InternalPackageModule {
1078 file_id,
1079 package_name,
1080 });
1081 }
1082 if let Some(fallback) = ctx.canonical_fallback
1083 && let Some(file_id) = fallback.get(&canonical)
1084 {
1085 return Some(ResolveResult::InternalPackageModule {
1086 file_id,
1087 package_name,
1088 });
1089 }
1090 if let Some(file_id) = try_source_fallback(&canonical, ctx.path_to_id) {
1091 return Some(ResolveResult::InternalPackageModule {
1092 file_id,
1093 package_name,
1094 });
1095 }
1096 }
1097 None
1098}
1099
1100pub(super) fn make_glob_from_pattern(
1102 pattern: &fallow_types::extract::DynamicImportPattern,
1103) -> String {
1104 if pattern.prefix.contains('*') || pattern.prefix.contains('{') {
1105 return pattern.prefix.clone();
1106 }
1107 pattern.suffix.as_ref().map_or_else(
1108 || format!("{}*", pattern.prefix),
1109 |suffix| format!("{}*{}", pattern.prefix, suffix),
1110 )
1111}
1112
1113#[cfg(test)]
1114mod tests {
1115 use super::*;
1116 use crate::resolve::types::TsconfigCache;
1117 use rustc_hash::FxHashSet;
1118
1119 fn with_package_map_ctx(
1120 root: PathBuf,
1121 name: Option<&str>,
1122 package_json: fallow_config::PackageJson,
1123 raw_files: &[(PathBuf, FileId)],
1124 f: impl FnOnce(&ResolveContext<'_>, &PackageManifestInfo, &Path),
1125 ) {
1126 let manifest = PackageManifestInfo {
1127 root: root.clone(),
1128 canonical_root: root,
1129 name: name.map(str::to_string),
1130 package_json,
1131 };
1132 let manifests = [manifest];
1133 let mut raw_path_to_id = FxHashMap::default();
1134 for (path, file_id) in raw_files {
1135 raw_path_to_id.insert(path.as_path(), *file_id);
1136 }
1137 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
1138 let workspace_roots: FxHashMap<&str, &Path> = FxHashMap::default();
1139 let condition_names = conditions();
1140 let resolver = oxc_resolver::Resolver::new(oxc_resolver::ResolveOptions::default());
1141 let tsconfig_warned = std::sync::Mutex::new(FxHashSet::default());
1142 let tsconfig_cache = TsconfigCache::default();
1143 let ctx = ResolveContext {
1144 resolver: &resolver,
1145 style_resolver: &resolver,
1146 extensions: &[],
1147 path_to_id: &path_to_id,
1148 raw_path_to_id: &raw_path_to_id,
1149 workspace_roots: &workspace_roots,
1150 package_manifests: &manifests,
1151 condition_names: &condition_names,
1152 path_aliases: &[],
1153 scss_include_paths: &[],
1154 static_dir_mappings: &[],
1155 root: &manifests[0].root,
1156 canonical_fallback: None,
1157 tsconfig_warned: &tsconfig_warned,
1158 tsconfig_cache: &tsconfig_cache,
1159 };
1160
1161 f(&ctx, &manifests[0], &manifests[0].root);
1162 }
1163
1164 #[test]
1165 fn alias_match_remainder_exact_key() {
1166 assert_eq!(alias_match_remainder("vscode", "vscode"), Some(""));
1167 assert_eq!(alias_match_remainder("@scope/sdk", "@scope/sdk"), Some(""));
1168 }
1169
1170 #[test]
1171 fn alias_match_remainder_slash_continuation() {
1172 assert_eq!(
1173 alias_match_remainder("@scope/sdk/sub", "@scope/sdk"),
1174 Some("/sub")
1175 );
1176 assert_eq!(alias_match_remainder("@/foo", "@/"), Some("foo"));
1177 assert_eq!(
1178 alias_match_remainder("~/components/x", "~/"),
1179 Some("components/x")
1180 );
1181 assert_eq!(alias_match_remainder("$lib/util", "$lib/"), Some("util"));
1182 }
1183
1184 #[test]
1185 fn alias_match_remainder_rejects_prefix_collision() {
1186 assert_eq!(
1187 alias_match_remainder("@scope/sdk-extra", "@scope/sdk"),
1188 None
1189 );
1190 assert_eq!(
1191 alias_match_remainder("vscode-languageserver", "vscode"),
1192 None
1193 );
1194 assert_eq!(alias_match_remainder("#shared-utils", "#shared"), None);
1195 }
1196
1197 #[test]
1198 fn alias_match_remainder_non_match() {
1199 assert_eq!(alias_match_remainder("react", "vscode"), None);
1200 }
1201
1202 #[test]
1203 fn test_extract_package_name_from_node_modules_path_regular() {
1204 let path = PathBuf::from("/project/node_modules/react/index.js");
1205 assert_eq!(
1206 extract_package_name_from_node_modules_path(&path),
1207 Some("react".to_string())
1208 );
1209 }
1210
1211 #[test]
1212 fn test_extract_package_name_from_node_modules_path_scoped() {
1213 let path = PathBuf::from("/project/node_modules/@babel/core/lib/index.js");
1214 assert_eq!(
1215 extract_package_name_from_node_modules_path(&path),
1216 Some("@babel/core".to_string())
1217 );
1218 }
1219
1220 #[test]
1221 fn test_extract_package_name_from_node_modules_path_nested() {
1222 let path = PathBuf::from("/project/node_modules/pkg-a/node_modules/pkg-b/dist/index.js");
1223 assert_eq!(
1224 extract_package_name_from_node_modules_path(&path),
1225 Some("pkg-b".to_string())
1226 );
1227 }
1228
1229 #[test]
1230 fn test_extract_package_name_from_node_modules_path_deep_subpath() {
1231 let path = PathBuf::from("/project/node_modules/react-dom/cjs/react-dom.production.min.js");
1232 assert_eq!(
1233 extract_package_name_from_node_modules_path(&path),
1234 Some("react-dom".to_string())
1235 );
1236 }
1237
1238 #[test]
1239 fn test_extract_package_name_from_node_modules_path_no_node_modules() {
1240 let path = PathBuf::from("/project/src/components/Button.tsx");
1241 assert_eq!(extract_package_name_from_node_modules_path(&path), None);
1242 }
1243
1244 #[test]
1245 fn test_extract_package_name_from_node_modules_path_just_node_modules() {
1246 let path = PathBuf::from("/project/node_modules");
1247 assert_eq!(extract_package_name_from_node_modules_path(&path), None);
1248 }
1249
1250 #[test]
1251 fn test_extract_package_name_from_node_modules_path_scoped_only_scope() {
1252 let path = PathBuf::from("/project/node_modules/@scope");
1253 assert_eq!(
1254 extract_package_name_from_node_modules_path(&path),
1255 Some("@scope".to_string())
1256 );
1257 }
1258
1259 #[test]
1260 fn test_resolve_specifier_node_modules_returns_npm_package() {
1261 let path =
1262 PathBuf::from("/project/node_modules/styled-components/dist/styled-components.esm.js");
1263 assert_eq!(
1264 extract_package_name_from_node_modules_path(&path),
1265 Some("styled-components".to_string())
1266 );
1267
1268 let path = PathBuf::from("/project/node_modules/next/dist/server/next.js");
1269 assert_eq!(
1270 extract_package_name_from_node_modules_path(&path),
1271 Some("next".to_string())
1272 );
1273 }
1274
1275 #[test]
1276 fn test_try_source_fallback_dist_to_src() {
1277 let src_path = PathBuf::from("/project/packages/ui/src/utils.ts");
1278 let mut path_to_id = FxHashMap::default();
1279 path_to_id.insert(src_path.as_path(), FileId(0));
1280
1281 let dist_path = PathBuf::from("/project/packages/ui/dist/utils.js");
1282 assert_eq!(
1283 try_source_fallback(&dist_path, &path_to_id),
1284 Some(FileId(0)),
1285 "dist/utils.js should fall back to src/utils.ts"
1286 );
1287 }
1288
1289 #[test]
1290 fn test_try_source_fallback_build_to_src() {
1291 let src_path = PathBuf::from("/project/packages/core/src/index.tsx");
1292 let mut path_to_id = FxHashMap::default();
1293 path_to_id.insert(src_path.as_path(), FileId(1));
1294
1295 let build_path = PathBuf::from("/project/packages/core/build/index.js");
1296 assert_eq!(
1297 try_source_fallback(&build_path, &path_to_id),
1298 Some(FileId(1)),
1299 "build/index.js should fall back to src/index.tsx"
1300 );
1301 }
1302
1303 #[test]
1304 fn test_try_source_fallback_no_match() {
1305 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
1306
1307 let dist_path = PathBuf::from("/project/packages/ui/dist/utils.js");
1308 assert_eq!(
1309 try_source_fallback(&dist_path, &path_to_id),
1310 None,
1311 "should return None when no source file exists"
1312 );
1313 }
1314
1315 #[test]
1316 fn test_try_source_fallback_non_output_dir() {
1317 let src_path = PathBuf::from("/project/packages/ui/src/utils.ts");
1318 let mut path_to_id = FxHashMap::default();
1319 path_to_id.insert(src_path.as_path(), FileId(0));
1320
1321 let normal_path = PathBuf::from("/project/packages/ui/scripts/utils.js");
1322 assert_eq!(
1323 try_source_fallback(&normal_path, &path_to_id),
1324 None,
1325 "non-output directory path should not trigger fallback"
1326 );
1327 }
1328
1329 #[test]
1330 fn test_try_source_fallback_nested_path() {
1331 let src_path = PathBuf::from("/project/packages/ui/src/components/Button.ts");
1332 let mut path_to_id = FxHashMap::default();
1333 path_to_id.insert(src_path.as_path(), FileId(2));
1334
1335 let dist_path = PathBuf::from("/project/packages/ui/dist/components/Button.js");
1336 assert_eq!(
1337 try_source_fallback(&dist_path, &path_to_id),
1338 Some(FileId(2)),
1339 "nested dist path should fall back to nested src path"
1340 );
1341 }
1342
1343 #[test]
1344 fn test_try_source_fallback_nested_dist_esm() {
1345 let src_path = PathBuf::from("/project/packages/ui/src/utils.ts");
1346 let mut path_to_id = FxHashMap::default();
1347 path_to_id.insert(src_path.as_path(), FileId(0));
1348
1349 let dist_path = PathBuf::from("/project/packages/ui/dist/esm/utils.mjs");
1350 assert_eq!(
1351 try_source_fallback(&dist_path, &path_to_id),
1352 Some(FileId(0)),
1353 "dist/esm/utils.mjs should fall back to src/utils.ts"
1354 );
1355 }
1356
1357 #[test]
1358 fn test_try_source_fallback_nested_build_cjs() {
1359 let src_path = PathBuf::from("/project/packages/core/src/index.ts");
1360 let mut path_to_id = FxHashMap::default();
1361 path_to_id.insert(src_path.as_path(), FileId(1));
1362
1363 let build_path = PathBuf::from("/project/packages/core/build/cjs/index.cjs");
1364 assert_eq!(
1365 try_source_fallback(&build_path, &path_to_id),
1366 Some(FileId(1)),
1367 "build/cjs/index.cjs should fall back to src/index.ts"
1368 );
1369 }
1370
1371 #[test]
1372 fn test_try_source_fallback_nested_dist_esm_deep_path() {
1373 let src_path = PathBuf::from("/project/packages/ui/src/components/Button.ts");
1374 let mut path_to_id = FxHashMap::default();
1375 path_to_id.insert(src_path.as_path(), FileId(2));
1376
1377 let dist_path = PathBuf::from("/project/packages/ui/dist/esm/components/Button.mjs");
1378 assert_eq!(
1379 try_source_fallback(&dist_path, &path_to_id),
1380 Some(FileId(2)),
1381 "dist/esm/components/Button.mjs should fall back to src/components/Button.ts"
1382 );
1383 }
1384
1385 #[test]
1386 fn test_try_source_fallback_triple_nested_output_dirs() {
1387 let src_path = PathBuf::from("/project/packages/ui/src/utils.ts");
1388 let mut path_to_id = FxHashMap::default();
1389 path_to_id.insert(src_path.as_path(), FileId(0));
1390
1391 let dist_path = PathBuf::from("/project/packages/ui/out/dist/esm/utils.mjs");
1392 assert_eq!(
1393 try_source_fallback(&dist_path, &path_to_id),
1394 Some(FileId(0)),
1395 "out/dist/esm/utils.mjs should fall back to src/utils.ts"
1396 );
1397 }
1398
1399 #[test]
1400 fn test_try_source_fallback_parent_dir_named_build() {
1401 let src_path = PathBuf::from("/home/user/build/my-project/src/utils.ts");
1402 let mut path_to_id = FxHashMap::default();
1403 path_to_id.insert(src_path.as_path(), FileId(0));
1404
1405 let dist_path = PathBuf::from("/home/user/build/my-project/dist/utils.js");
1406 assert_eq!(
1407 try_source_fallback(&dist_path, &path_to_id),
1408 Some(FileId(0)),
1409 "should resolve dist/ within project, not match parent 'build' dir"
1410 );
1411 }
1412
1413 #[test]
1414 fn package_map_exact_entry_beats_pattern_entry() {
1415 let map = serde_json::json!({
1416 "#nitro/runtime/task": "./dist/special/task.mjs",
1417 "#nitro/runtime/*": "./dist/runtime/internal/*.mjs"
1418 });
1419 assert_eq!(
1420 package_map_target(&map, "#nitro/runtime/task", &conditions()),
1421 PackageMapTarget::Targets(vec!["./dist/special/task.mjs".to_string()])
1422 );
1423 }
1424
1425 #[test]
1426 fn package_map_wildcard_substitutes_capture() {
1427 let map = serde_json::json!({
1428 "#nitro/runtime/*": "./dist/runtime/internal/*.mjs"
1429 });
1430 assert_eq!(
1431 package_map_target(&map, "#nitro/runtime/task", &conditions()),
1432 PackageMapTarget::Targets(vec!["./dist/runtime/internal/task.mjs".to_string()])
1433 );
1434 }
1435
1436 #[test]
1437 fn package_map_exact_entry_with_no_target_blocks_pattern_entry() {
1438 let map = serde_json::json!({
1439 "#nitro/runtime/task": null,
1440 "#nitro/runtime/*": "./dist/runtime/internal/*.mjs"
1441 });
1442 assert_eq!(
1443 package_map_target(&map, "#nitro/runtime/task", &conditions()),
1444 PackageMapTarget::Blocked
1445 );
1446 }
1447
1448 #[test]
1449 fn package_map_best_pattern_with_no_target_blocks_broader_pattern() {
1450 let map = serde_json::json!({
1451 "#nitro/runtime/internal/*": null,
1452 "#nitro/runtime/*": "./dist/runtime/*.mjs"
1453 });
1454 assert_eq!(
1455 package_map_target(&map, "#nitro/runtime/internal/task", &conditions()),
1456 PackageMapTarget::Blocked
1457 );
1458 }
1459
1460 #[test]
1461 fn package_map_unmatched_subpath_is_not_a_target() {
1462 let map = serde_json::json!({
1463 "./query": "./dist/query/index.js"
1464 });
1465 assert_eq!(
1466 package_map_target(&map, "./private", &conditions()),
1467 PackageMapTarget::NoMatch
1468 );
1469 }
1470
1471 #[test]
1472 fn package_map_nested_conditions_follow_manifest_order() {
1473 let map = serde_json::json!({
1474 "./query/react": {
1475 "types": "./dist/query/react/index.d.ts",
1476 "import": {
1477 "development": "./src/query/react/index.ts",
1478 "default": "./dist/query/react/index.js"
1479 },
1480 "default": "./dist/query/react/index.cjs"
1481 }
1482 });
1483 assert_eq!(
1484 package_map_target(&map, "./query/react", &conditions()),
1485 PackageMapTarget::Targets(vec!["./dist/query/react/index.d.ts".to_string()])
1486 );
1487 }
1488
1489 #[test]
1490 fn package_map_import_before_types_selects_runtime_branch() {
1491 let map = serde_json::json!({
1492 ".": {
1493 "import": "./dist/index.js",
1494 "types": "./dist/index.d.ts"
1495 }
1496 });
1497 assert_eq!(
1498 package_map_target(&map, ".", &conditions()),
1499 PackageMapTarget::Targets(vec!["./dist/index.js".to_string()])
1500 );
1501 }
1502
1503 #[test]
1504 fn package_map_condition_order_follows_manifest_order() {
1505 let map = serde_json::json!({
1506 ".": {
1507 "node": "./dist/node.js",
1508 "import": "./dist/index.js"
1509 }
1510 });
1511 assert_eq!(
1512 package_map_target(&map, ".", &conditions()),
1513 PackageMapTarget::Targets(vec!["./dist/node.js".to_string()])
1514 );
1515 }
1516
1517 #[test]
1518 fn package_map_arrays_preserve_fallback_order() {
1519 let map = serde_json::json!({
1520 "#array": ["./dist/missing.js", "./src/array.ts"],
1521 "#null": null,
1522 "#false": false
1523 });
1524 assert_eq!(
1525 package_map_target(&map, "#array", &conditions()),
1526 PackageMapTarget::Targets(vec![
1527 "./dist/missing.js".to_string(),
1528 "./src/array.ts".to_string()
1529 ])
1530 );
1531 assert_eq!(
1532 package_map_target(&map, "#null", &conditions()),
1533 PackageMapTarget::Blocked
1534 );
1535 assert_eq!(
1536 package_map_target(&map, "#false", &conditions()),
1537 PackageMapTarget::Blocked
1538 );
1539 }
1540
1541 #[test]
1542 fn package_map_non_relative_target_does_not_trigger_source_fallback() {
1543 with_package_map_ctx(
1544 PathBuf::from("/project"),
1545 Some("pkg"),
1546 fallow_config::PackageJson::default(),
1547 &[],
1548 |ctx, manifest, _| {
1549 assert!(resolve_package_map_target(ctx, manifest, "lodash", None).is_none());
1550 assert!(
1551 resolve_package_map_target(ctx, manifest, "../dist/index.js", None).is_none()
1552 );
1553 },
1554 );
1555 }
1556
1557 #[test]
1558 fn package_map_targets_use_first_reachable_target() {
1559 let root = PathBuf::from("/project");
1560 let src_path = root.join("src/feature.ts");
1561 let targets = vec![
1562 "./dist/missing.js".to_string(),
1563 "./src/feature.ts".to_string(),
1564 ];
1565
1566 with_package_map_ctx(
1567 root,
1568 Some("pkg"),
1569 fallow_config::PackageJson::default(),
1570 &[(src_path, FileId(9))],
1571 |ctx, manifest, _| {
1572 assert_eq!(
1573 resolve_package_map_targets(ctx, manifest, &targets, None),
1574 Some(FileId(9))
1575 );
1576 },
1577 );
1578 }
1579
1580 #[test]
1581 fn package_imports_fallback_supports_external_package_targets() {
1582 let root = PathBuf::from("/project");
1583 with_package_map_ctx(
1584 root,
1585 Some("pkg"),
1586 fallow_config::PackageJson {
1587 imports: Some(serde_json::json!({
1588 "#pad": "left-pad",
1589 "#scoped": "@scope/pkg/subpath"
1590 })),
1591 ..Default::default()
1592 },
1593 &[],
1594 |ctx, _, root| {
1595 let pad = try_package_imports_fallback(ctx, &root.join("src/index.ts"), "#pad");
1596 assert!(matches!(pad, Some(ResolveResult::NpmPackage(pkg)) if pkg == "left-pad"));
1597
1598 let scoped =
1599 try_package_imports_fallback(ctx, &root.join("src/index.ts"), "#scoped");
1600 assert!(
1601 matches!(scoped, Some(ResolveResult::NpmPackage(pkg)) if pkg == "@scope/pkg")
1602 );
1603 },
1604 );
1605 }
1606
1607 #[test]
1608 fn package_imports_fallback_supports_unnamed_packages() {
1609 let root = PathBuf::from("/project");
1610 let src_path = root.join("src/runtime/task.ts");
1611 with_package_map_ctx(
1612 root,
1613 None,
1614 fallow_config::PackageJson {
1615 imports: Some(serde_json::json!({
1616 "#runtime/*": "./dist/runtime/*.mjs"
1617 })),
1618 ..Default::default()
1619 },
1620 &[(src_path, FileId(7))],
1621 |ctx, _, root| {
1622 let result =
1623 try_package_imports_fallback(ctx, &root.join("src/index.ts"), "#runtime/task");
1624 assert!(matches!(
1625 result,
1626 Some(ResolveResult::InternalModule(FileId(7)))
1627 ));
1628 },
1629 );
1630 }
1631
1632 #[test]
1633 #[cfg_attr(miri, ignore)]
1634 fn relative_package_root_source_fallback_uses_package_source_entry() {
1635 let root = PathBuf::from("/project");
1636 let source_path = root.join("custom/entry.js");
1637 with_package_map_ctx(
1638 root,
1639 Some("pkg"),
1640 fallow_config::PackageJson {
1641 source: Some("custom/entry.js".to_string()),
1642 ..Default::default()
1643 },
1644 &[(source_path, FileId(11))],
1645 |ctx, _, root| {
1646 let result = try_relative_package_root_source_fallback(
1647 ctx,
1648 &root.join("test/shared/exports.test.js"),
1649 "../../",
1650 );
1651 assert!(matches!(
1652 result,
1653 Some(ResolveResult::InternalModule(FileId(11)))
1654 ));
1655 },
1656 );
1657 }
1658
1659 #[test]
1660 fn package_source_path_accepts_relative_source_entries() {
1661 assert_eq!(
1662 safe_relative_package_source_path("src/index.js"),
1663 Some(Path::new("src/index.js"))
1664 );
1665 assert_eq!(
1666 safe_relative_package_source_path("./custom/entry.ts"),
1667 Some(Path::new("custom/entry.ts"))
1668 );
1669 }
1670
1671 #[test]
1672 fn package_source_path_rejects_unsafe_entries() {
1673 assert_eq!(safe_relative_package_source_path(""), None);
1674 assert_eq!(safe_relative_package_source_path("./"), None);
1675 assert_eq!(safe_relative_package_source_path("../src/index.js"), None);
1676 assert_eq!(safe_relative_package_source_path("src/../index.js"), None);
1677 assert_eq!(safe_relative_package_source_path("/src/index.js"), None);
1678
1679 #[cfg(windows)]
1680 assert_eq!(safe_relative_package_source_path("C:\\src\\index.js"), None);
1681 }
1682
1683 #[test]
1684 fn test_pnpm_store_path_extract_package_name() {
1685 let path =
1686 PathBuf::from("/project/node_modules/.pnpm/react@18.2.0/node_modules/react/index.js");
1687 assert_eq!(
1688 extract_package_name_from_node_modules_path(&path),
1689 Some("react".to_string())
1690 );
1691 }
1692
1693 #[test]
1694 fn test_pnpm_store_path_scoped_package() {
1695 let path = PathBuf::from(
1696 "/project/node_modules/.pnpm/@babel+core@7.24.0/node_modules/@babel/core/lib/index.js",
1697 );
1698 assert_eq!(
1699 extract_package_name_from_node_modules_path(&path),
1700 Some("@babel/core".to_string())
1701 );
1702 }
1703
1704 fn conditions() -> Vec<String> {
1705 vec![
1706 "development".to_string(),
1707 "import".to_string(),
1708 "require".to_string(),
1709 "default".to_string(),
1710 "types".to_string(),
1711 "node".to_string(),
1712 ]
1713 }
1714
1715 #[test]
1716 fn test_pnpm_store_path_with_peer_deps() {
1717 let path = PathBuf::from(
1718 "/project/node_modules/.pnpm/webpack@5.0.0_esbuild@0.19.0/node_modules/webpack/lib/index.js",
1719 );
1720 assert_eq!(
1721 extract_package_name_from_node_modules_path(&path),
1722 Some("webpack".to_string())
1723 );
1724 }
1725
1726 #[test]
1727 fn test_try_pnpm_workspace_fallback_dist_to_src() {
1728 let src_path = PathBuf::from("/project/packages/ui/src/utils.ts");
1729 let mut path_to_id = FxHashMap::default();
1730 path_to_id.insert(src_path.as_path(), FileId(0));
1731
1732 let mut workspace_roots = FxHashMap::default();
1733 let ws_root = PathBuf::from("/project/packages/ui");
1734 workspace_roots.insert("@myorg/ui", ws_root.as_path());
1735
1736 let pnpm_path = PathBuf::from(
1737 "/project/node_modules/.pnpm/@myorg+ui@1.0.0/node_modules/@myorg/ui/dist/utils.js",
1738 );
1739 assert_eq!(
1740 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
1741 Some(FileId(0)),
1742 ".pnpm workspace path should fall back to src/utils.ts"
1743 );
1744 }
1745
1746 #[test]
1747 fn test_try_pnpm_workspace_fallback_direct_source() {
1748 let src_path = PathBuf::from("/project/packages/core/src/index.ts");
1749 let mut path_to_id = FxHashMap::default();
1750 path_to_id.insert(src_path.as_path(), FileId(1));
1751
1752 let mut workspace_roots = FxHashMap::default();
1753 let ws_root = PathBuf::from("/project/packages/core");
1754 workspace_roots.insert("@myorg/core", ws_root.as_path());
1755
1756 let pnpm_path = PathBuf::from(
1757 "/project/node_modules/.pnpm/@myorg+core@workspace/node_modules/@myorg/core/src/index.ts",
1758 );
1759 assert_eq!(
1760 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
1761 Some(FileId(1)),
1762 ".pnpm workspace path with src/ should resolve directly"
1763 );
1764 }
1765
1766 #[test]
1767 fn test_try_pnpm_workspace_fallback_non_workspace_package() {
1768 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
1769
1770 let mut workspace_roots = FxHashMap::default();
1771 let ws_root = PathBuf::from("/project/packages/ui");
1772 workspace_roots.insert("@myorg/ui", ws_root.as_path());
1773
1774 let pnpm_path =
1775 PathBuf::from("/project/node_modules/.pnpm/react@18.2.0/node_modules/react/index.js");
1776 assert_eq!(
1777 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
1778 None,
1779 "non-workspace package in .pnpm should return None"
1780 );
1781 }
1782
1783 #[test]
1784 fn test_try_pnpm_workspace_fallback_unscoped_package() {
1785 let src_path = PathBuf::from("/project/packages/utils/src/index.ts");
1786 let mut path_to_id = FxHashMap::default();
1787 path_to_id.insert(src_path.as_path(), FileId(2));
1788
1789 let mut workspace_roots = FxHashMap::default();
1790 let ws_root = PathBuf::from("/project/packages/utils");
1791 workspace_roots.insert("my-utils", ws_root.as_path());
1792
1793 let pnpm_path = PathBuf::from(
1794 "/project/node_modules/.pnpm/my-utils@1.0.0/node_modules/my-utils/dist/index.js",
1795 );
1796 assert_eq!(
1797 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
1798 Some(FileId(2)),
1799 "unscoped workspace package in .pnpm should resolve"
1800 );
1801 }
1802
1803 #[test]
1804 fn test_try_pnpm_workspace_fallback_nested_path() {
1805 let src_path = PathBuf::from("/project/packages/ui/src/components/Button.ts");
1806 let mut path_to_id = FxHashMap::default();
1807 path_to_id.insert(src_path.as_path(), FileId(3));
1808
1809 let mut workspace_roots = FxHashMap::default();
1810 let ws_root = PathBuf::from("/project/packages/ui");
1811 workspace_roots.insert("@myorg/ui", ws_root.as_path());
1812
1813 let pnpm_path = PathBuf::from(
1814 "/project/node_modules/.pnpm/@myorg+ui@1.0.0/node_modules/@myorg/ui/dist/components/Button.js",
1815 );
1816 assert_eq!(
1817 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
1818 Some(FileId(3)),
1819 "nested .pnpm workspace path should resolve through source fallback"
1820 );
1821 }
1822
1823 #[test]
1824 fn test_try_pnpm_workspace_fallback_no_pnpm() {
1825 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
1826 let workspace_roots: FxHashMap<&str, &Path> = FxHashMap::default();
1827
1828 let regular_path = PathBuf::from("/project/node_modules/react/index.js");
1829 assert_eq!(
1830 try_pnpm_workspace_fallback(®ular_path, &path_to_id, &workspace_roots),
1831 None,
1832 );
1833 }
1834
1835 #[test]
1836 fn test_try_pnpm_workspace_fallback_with_peer_deps() {
1837 let src_path = PathBuf::from("/project/packages/ui/src/index.ts");
1838 let mut path_to_id = FxHashMap::default();
1839 path_to_id.insert(src_path.as_path(), FileId(4));
1840
1841 let mut workspace_roots = FxHashMap::default();
1842 let ws_root = PathBuf::from("/project/packages/ui");
1843 workspace_roots.insert("@myorg/ui", ws_root.as_path());
1844
1845 let pnpm_path = PathBuf::from(
1846 "/project/node_modules/.pnpm/@myorg+ui@1.0.0_react@18.2.0/node_modules/@myorg/ui/dist/index.js",
1847 );
1848 assert_eq!(
1849 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
1850 Some(FileId(4)),
1851 ".pnpm path with peer dep suffix should still resolve"
1852 );
1853 }
1854
1855 #[test]
1856 fn make_glob_prefix_only_no_suffix() {
1857 let pattern = fallow_types::extract::DynamicImportPattern {
1858 prefix: "./locales/".to_string(),
1859 suffix: None,
1860 span: oxc_span::Span::default(),
1861 };
1862 assert_eq!(make_glob_from_pattern(&pattern), "./locales/*");
1863 }
1864
1865 #[test]
1866 fn make_glob_prefix_with_suffix() {
1867 let pattern = fallow_types::extract::DynamicImportPattern {
1868 prefix: "./locales/".to_string(),
1869 suffix: Some(".json".to_string()),
1870 span: oxc_span::Span::default(),
1871 };
1872 assert_eq!(make_glob_from_pattern(&pattern), "./locales/*.json");
1873 }
1874
1875 #[test]
1876 fn make_glob_passthrough_star() {
1877 let pattern = fallow_types::extract::DynamicImportPattern {
1878 prefix: "./pages/**/*.tsx".to_string(),
1879 suffix: None,
1880 span: oxc_span::Span::default(),
1881 };
1882 assert_eq!(make_glob_from_pattern(&pattern), "./pages/**/*.tsx");
1883 }
1884
1885 #[test]
1886 fn make_glob_passthrough_brace() {
1887 let pattern = fallow_types::extract::DynamicImportPattern {
1888 prefix: "./i18n/{en,de,fr}.json".to_string(),
1889 suffix: None,
1890 span: oxc_span::Span::default(),
1891 };
1892 assert_eq!(make_glob_from_pattern(&pattern), "./i18n/{en,de,fr}.json");
1893 }
1894
1895 #[test]
1896 fn make_glob_empty_prefix_no_suffix() {
1897 let pattern = fallow_types::extract::DynamicImportPattern {
1898 prefix: String::new(),
1899 suffix: None,
1900 span: oxc_span::Span::default(),
1901 };
1902 assert_eq!(make_glob_from_pattern(&pattern), "*");
1903 }
1904
1905 #[test]
1906 fn make_glob_empty_prefix_with_suffix() {
1907 let pattern = fallow_types::extract::DynamicImportPattern {
1908 prefix: String::new(),
1909 suffix: Some(".ts".to_string()),
1910 span: oxc_span::Span::default(),
1911 };
1912 assert_eq!(make_glob_from_pattern(&pattern), "*.ts");
1913 }
1914
1915 #[test]
1916 fn make_glob_template_literal_prefix_only() {
1917 let pattern = fallow_types::extract::DynamicImportPattern {
1918 prefix: "./pages/".to_string(),
1919 suffix: None,
1920 span: oxc_span::Span::default(),
1921 };
1922 assert_eq!(make_glob_from_pattern(&pattern), "./pages/*");
1923 }
1924
1925 #[test]
1926 fn make_glob_template_literal_with_extension_suffix() {
1927 let pattern = fallow_types::extract::DynamicImportPattern {
1928 prefix: "./locales/".to_string(),
1929 suffix: Some(".json".to_string()),
1930 span: oxc_span::Span::default(),
1931 };
1932 assert_eq!(make_glob_from_pattern(&pattern), "./locales/*.json");
1933 }
1934
1935 #[test]
1936 fn make_glob_template_literal_deep_prefix() {
1937 let pattern = fallow_types::extract::DynamicImportPattern {
1938 prefix: "./modules/".to_string(),
1939 suffix: None,
1940 span: oxc_span::Span::default(),
1941 };
1942 assert_eq!(make_glob_from_pattern(&pattern), "./modules/*");
1943 }
1944
1945 #[test]
1946 fn make_glob_string_concat_prefix() {
1947 let pattern = fallow_types::extract::DynamicImportPattern {
1948 prefix: "./pages/".to_string(),
1949 suffix: None,
1950 span: oxc_span::Span::default(),
1951 };
1952 assert_eq!(make_glob_from_pattern(&pattern), "./pages/*");
1953 }
1954
1955 #[test]
1956 fn make_glob_string_concat_with_extension() {
1957 let pattern = fallow_types::extract::DynamicImportPattern {
1958 prefix: "./views/".to_string(),
1959 suffix: Some(".vue".to_string()),
1960 span: oxc_span::Span::default(),
1961 };
1962 assert_eq!(make_glob_from_pattern(&pattern), "./views/*.vue");
1963 }
1964
1965 #[test]
1966 fn make_glob_import_meta_glob_recursive() {
1967 let pattern = fallow_types::extract::DynamicImportPattern {
1968 prefix: "./components/**/*.vue".to_string(),
1969 suffix: None,
1970 span: oxc_span::Span::default(),
1971 };
1972 assert_eq!(
1973 make_glob_from_pattern(&pattern),
1974 "./components/**/*.vue",
1975 "import.meta.glob patterns with * should pass through as-is"
1976 );
1977 }
1978
1979 #[test]
1980 fn make_glob_import_meta_glob_brace_expansion() {
1981 let pattern = fallow_types::extract::DynamicImportPattern {
1982 prefix: "./plugins/{auth,analytics}.ts".to_string(),
1983 suffix: None,
1984 span: oxc_span::Span::default(),
1985 };
1986 assert_eq!(
1987 make_glob_from_pattern(&pattern),
1988 "./plugins/{auth,analytics}.ts",
1989 "import.meta.glob patterns with braces should pass through as-is"
1990 );
1991 }
1992
1993 #[test]
1994 fn make_glob_import_meta_glob_star_with_brace() {
1995 let pattern = fallow_types::extract::DynamicImportPattern {
1996 prefix: "./routes/**/*.{ts,tsx}".to_string(),
1997 suffix: None,
1998 span: oxc_span::Span::default(),
1999 };
2000 assert_eq!(
2001 make_glob_from_pattern(&pattern),
2002 "./routes/**/*.{ts,tsx}",
2003 "combined * and brace patterns should pass through"
2004 );
2005 }
2006
2007 #[test]
2008 fn make_glob_import_meta_glob_ignores_suffix_when_star_present() {
2009 let pattern = fallow_types::extract::DynamicImportPattern {
2010 prefix: "./*.ts".to_string(),
2011 suffix: Some(".extra".to_string()),
2012 span: oxc_span::Span::default(),
2013 };
2014 assert_eq!(
2015 make_glob_from_pattern(&pattern),
2016 "./*.ts",
2017 "when prefix has glob chars, suffix is ignored (prefix used as-is)"
2018 );
2019 }
2020
2021 #[test]
2022 fn make_glob_single_dot_prefix() {
2023 let pattern = fallow_types::extract::DynamicImportPattern {
2024 prefix: "./".to_string(),
2025 suffix: None,
2026 span: oxc_span::Span::default(),
2027 };
2028 assert_eq!(make_glob_from_pattern(&pattern), "./*");
2029 }
2030
2031 #[test]
2032 fn make_glob_prefix_without_trailing_slash() {
2033 let pattern = fallow_types::extract::DynamicImportPattern {
2034 prefix: "./config".to_string(),
2035 suffix: None,
2036 span: oxc_span::Span::default(),
2037 };
2038 assert_eq!(make_glob_from_pattern(&pattern), "./config*");
2039 }
2040
2041 #[test]
2042 fn make_glob_prefix_with_dotdot() {
2043 let pattern = fallow_types::extract::DynamicImportPattern {
2044 prefix: "../shared/".to_string(),
2045 suffix: Some(".ts".to_string()),
2046 span: oxc_span::Span::default(),
2047 };
2048 assert_eq!(make_glob_from_pattern(&pattern), "../shared/*.ts");
2049 }
2050
2051 #[test]
2052 fn test_extract_package_name_with_pnpm_plus_encoded_scope() {
2053 let path = PathBuf::from(
2054 "/project/node_modules/.pnpm/@mui+material@5.15.0/node_modules/@mui/material/index.js",
2055 );
2056 assert_eq!(
2057 extract_package_name_from_node_modules_path(&path),
2058 Some("@mui/material".to_string())
2059 );
2060 }
2061
2062 #[test]
2063 fn test_extract_package_name_windows_style_path() {
2064 let path = PathBuf::from("/project/node_modules/typescript/lib/tsc.js");
2065 assert_eq!(
2066 extract_package_name_from_node_modules_path(&path),
2067 Some("typescript".to_string())
2068 );
2069 }
2070
2071 #[test]
2072 fn test_try_source_fallback_out_dir() {
2073 let src_path = PathBuf::from("/project/packages/api/src/handler.ts");
2074 let mut path_to_id = FxHashMap::default();
2075 path_to_id.insert(src_path.as_path(), FileId(5));
2076
2077 let out_path = PathBuf::from("/project/packages/api/out/handler.js");
2078 assert_eq!(
2079 try_source_fallback(&out_path, &path_to_id),
2080 Some(FileId(5)),
2081 "out/handler.js should fall back to src/handler.ts"
2082 );
2083 }
2084
2085 #[test]
2086 fn test_try_source_fallback_mts_extension() {
2087 let src_path = PathBuf::from("/project/packages/lib/src/utils.mts");
2088 let mut path_to_id = FxHashMap::default();
2089 path_to_id.insert(src_path.as_path(), FileId(6));
2090
2091 let dist_path = PathBuf::from("/project/packages/lib/dist/utils.mjs");
2092 assert_eq!(
2093 try_source_fallback(&dist_path, &path_to_id),
2094 Some(FileId(6)),
2095 "dist/utils.mjs should fall back to src/utils.mts"
2096 );
2097 }
2098
2099 #[test]
2100 fn test_try_source_fallback_cts_extension() {
2101 let src_path = PathBuf::from("/project/packages/lib/src/config.cts");
2102 let mut path_to_id = FxHashMap::default();
2103 path_to_id.insert(src_path.as_path(), FileId(7));
2104
2105 let dist_path = PathBuf::from("/project/packages/lib/dist/config.cjs");
2106 assert_eq!(
2107 try_source_fallback(&dist_path, &path_to_id),
2108 Some(FileId(7)),
2109 "dist/config.cjs should fall back to src/config.cts"
2110 );
2111 }
2112
2113 #[test]
2114 fn test_try_source_fallback_jsx_extension() {
2115 let src_path = PathBuf::from("/project/packages/ui/src/App.jsx");
2116 let mut path_to_id = FxHashMap::default();
2117 path_to_id.insert(src_path.as_path(), FileId(8));
2118
2119 let build_path = PathBuf::from("/project/packages/ui/build/App.js");
2120 assert_eq!(
2121 try_source_fallback(&build_path, &path_to_id),
2122 Some(FileId(8)),
2123 "build/App.js should fall back to src/App.jsx"
2124 );
2125 }
2126
2127 #[test]
2128 fn test_try_source_fallback_no_file_stem() {
2129 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
2130 let dist_path = PathBuf::from("/project/packages/ui/dist/");
2131 assert_eq!(
2132 try_source_fallback(&dist_path, &path_to_id),
2133 None,
2134 "directory path with no file should return None"
2135 );
2136 }
2137
2138 #[test]
2139 fn test_try_source_fallback_esm_subdir() {
2140 let src_path = PathBuf::from("/project/lib/src/index.ts");
2141 let mut path_to_id = FxHashMap::default();
2142 path_to_id.insert(src_path.as_path(), FileId(10));
2143
2144 let dist_path = PathBuf::from("/project/lib/esm/index.mjs");
2145 assert_eq!(
2146 try_source_fallback(&dist_path, &path_to_id),
2147 Some(FileId(10)),
2148 "standalone esm/ directory should fall back to src/"
2149 );
2150 }
2151
2152 #[test]
2153 fn test_try_source_fallback_cjs_subdir() {
2154 let src_path = PathBuf::from("/project/lib/src/index.ts");
2155 let mut path_to_id = FxHashMap::default();
2156 path_to_id.insert(src_path.as_path(), FileId(11));
2157
2158 let cjs_path = PathBuf::from("/project/lib/cjs/index.cjs");
2159 assert_eq!(
2160 try_source_fallback(&cjs_path, &path_to_id),
2161 Some(FileId(11)),
2162 "standalone cjs/ directory should fall back to src/"
2163 );
2164 }
2165
2166 #[test]
2167 fn test_try_pnpm_workspace_fallback_empty_after_pnpm() {
2168 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
2169 let workspace_roots: FxHashMap<&str, &Path> = FxHashMap::default();
2170
2171 let pnpm_path = PathBuf::from("/project/node_modules/.pnpm/pkg@1.0.0/node_modules");
2172 assert_eq!(
2173 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
2174 None,
2175 "path ending at node_modules with nothing after should return None"
2176 );
2177 }
2178
2179 #[test]
2180 fn test_try_pnpm_workspace_fallback_scoped_package_only_scope() {
2181 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
2182 let workspace_roots: FxHashMap<&str, &Path> = FxHashMap::default();
2183
2184 let pnpm_path =
2185 PathBuf::from("/project/node_modules/.pnpm/@scope+pkg@1.0.0/node_modules/@scope");
2186 assert_eq!(
2187 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
2188 None,
2189 "scoped package without full name and no matching workspace should return None"
2190 );
2191 }
2192
2193 #[test]
2194 fn test_try_pnpm_workspace_fallback_no_inner_node_modules() {
2195 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
2196 let workspace_roots: FxHashMap<&str, &Path> = FxHashMap::default();
2197
2198 let pnpm_path = PathBuf::from("/project/node_modules/.pnpm/pkg@1.0.0/dist/index.js");
2199 assert_eq!(
2200 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
2201 None,
2202 "path without inner node_modules after .pnpm should return None"
2203 );
2204 }
2205
2206 #[test]
2207 fn test_try_pnpm_workspace_fallback_package_without_relative_path() {
2208 let path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
2209 let mut workspace_roots = FxHashMap::default();
2210 let ws_root = PathBuf::from("/project/packages/ui");
2211 workspace_roots.insert("@myorg/ui", ws_root.as_path());
2212
2213 let pnpm_path =
2214 PathBuf::from("/project/node_modules/.pnpm/@myorg+ui@1.0.0/node_modules/@myorg/ui");
2215 assert_eq!(
2216 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
2217 None,
2218 "path ending at package name with no relative file should return None"
2219 );
2220 }
2221
2222 #[test]
2223 fn test_try_pnpm_workspace_fallback_nested_dist_esm() {
2224 let src_path = PathBuf::from("/project/packages/ui/src/Button.ts");
2225 let mut path_to_id = FxHashMap::default();
2226 path_to_id.insert(src_path.as_path(), FileId(10));
2227
2228 let mut workspace_roots = FxHashMap::default();
2229 let ws_root = PathBuf::from("/project/packages/ui");
2230 workspace_roots.insert("@myorg/ui", ws_root.as_path());
2231
2232 let pnpm_path = PathBuf::from(
2233 "/project/node_modules/.pnpm/@myorg+ui@1.0.0/node_modules/@myorg/ui/dist/esm/Button.mjs",
2234 );
2235 assert_eq!(
2236 try_pnpm_workspace_fallback(&pnpm_path, &path_to_id, &workspace_roots),
2237 Some(FileId(10)),
2238 "pnpm path with nested dist/esm should resolve through source fallback"
2239 );
2240 }
2241
2242 #[test]
2245 fn package_map_target_string_value_dot_key() {
2246 let map = serde_json::Value::String("./src/index.ts".to_string());
2249 let conds = conditions();
2250 let result = package_map_target(&map, ".", &conds);
2252 assert!(
2253 matches!(result, PackageMapTarget::Targets(_)),
2254 "string map with '.' key should return Targets"
2255 );
2256 }
2257
2258 #[test]
2259 fn package_map_target_string_value_non_dot_key_no_match() {
2260 let map = serde_json::Value::String("./src/index.ts".to_string());
2262 let conds = conditions();
2263 let result = package_map_target(&map, "./sub", &conds);
2264 assert!(
2265 matches!(result, PackageMapTarget::NoMatch),
2266 "string map with non-dot key should return NoMatch"
2267 );
2268 }
2269
2270 #[test]
2271 fn package_map_target_null_value_dot_key() {
2272 let map = serde_json::Value::Null;
2274 let conds = conditions();
2275 let result = package_map_target(&map, ".", &conds);
2276 assert!(
2277 matches!(result, PackageMapTarget::Blocked),
2278 "null map with '.' key should return Blocked"
2279 );
2280 }
2281
2282 #[test]
2283 fn package_map_target_bool_value_non_dot_key() {
2284 let map = serde_json::Value::Bool(true);
2286 let conds = conditions();
2287 let result = package_map_target(&map, "./sub", &conds);
2288 assert!(
2289 matches!(result, PackageMapTarget::NoMatch),
2290 "bool map with non-dot key should return NoMatch"
2291 );
2292 }
2293
2294 #[test]
2297 fn package_map_target_condition_only_object_dot_key() {
2298 let map = serde_json::json!({
2301 "import": "./src/index.mjs",
2302 "require": "./src/index.cjs"
2303 });
2304 let conds = conditions();
2305 let result = package_map_target(&map, ".", &conds);
2306 assert!(
2307 matches!(result, PackageMapTarget::Targets(_)),
2308 "condition-only object with '.' key should return Targets"
2309 );
2310 }
2311
2312 #[test]
2313 fn package_map_target_condition_only_object_non_dot_key() {
2314 let map = serde_json::json!({
2317 "import": "./src/index.mjs",
2318 "require": "./src/index.cjs"
2319 });
2320 let conds = conditions();
2321 let result = package_map_target(&map, "./nonexistent", &conds);
2322 assert!(
2323 matches!(result, PackageMapTarget::NoMatch),
2324 "condition-only object with non-dot key should return NoMatch"
2325 );
2326 }
2327
2328 #[test]
2331 fn resolve_package_map_value_unmatched_conditions_returns_none() {
2332 let value = serde_json::json!({ "browser": "./src/browser.js" });
2334 let conds = conditions(); assert_eq!(
2336 resolve_package_map_value(&value, &conds, None),
2337 None,
2338 "unmatched condition should return None"
2339 );
2340 }
2341
2342 #[test]
2343 fn resolve_package_map_value_bool_returns_none() {
2344 let value = serde_json::Value::Bool(false);
2345 let conds = conditions();
2346 assert_eq!(
2347 resolve_package_map_value(&value, &conds, None),
2348 None,
2349 "bool value should return None"
2350 );
2351 }
2352
2353 #[test]
2354 fn resolve_package_map_value_number_returns_none() {
2355 let value = serde_json::Value::Number(42.into());
2356 let conds = conditions();
2357 assert_eq!(
2358 resolve_package_map_value(&value, &conds, None),
2359 None,
2360 "number value should return None"
2361 );
2362 }
2363
2364 #[test]
2365 fn resolve_package_map_value_null_returns_none() {
2366 let value = serde_json::Value::Null;
2367 let conds = conditions();
2368 assert_eq!(
2369 resolve_package_map_value(&value, &conds, None),
2370 None,
2371 "null value should return None"
2372 );
2373 }
2374
2375 #[test]
2376 fn resolve_package_map_value_array_all_null_returns_none() {
2377 let value = serde_json::json!([null, false, 42]);
2379 let conds = conditions();
2380 assert_eq!(
2381 resolve_package_map_value(&value, &conds, None),
2382 None,
2383 "array of unresolvable values should return None"
2384 );
2385 }
2386
2387 #[test]
2388 fn resolve_package_map_value_array_with_valid_entry() {
2389 let value = serde_json::json!([null, "./src/index.ts"]);
2391 let conds = conditions();
2392 let result = resolve_package_map_value(&value, &conds, None);
2393 assert_eq!(
2394 result,
2395 Some(vec!["./src/index.ts".to_string()]),
2396 "array with a valid string entry should return that entry"
2397 );
2398 }
2399
2400 #[test]
2403 fn package_map_pattern_capture_no_star_returns_none() {
2404 assert_eq!(
2406 package_map_pattern_capture("./exact", "./exact"),
2407 None,
2408 "pattern with no star should return None"
2409 );
2410 }
2411
2412 #[test]
2413 fn package_map_pattern_capture_two_stars_returns_none() {
2414 assert_eq!(
2416 package_map_pattern_capture("./*/*.js", "./foo/bar.js"),
2417 None,
2418 "pattern with two stars should return None"
2419 );
2420 }
2421
2422 #[test]
2423 fn package_map_pattern_capture_single_star_captures() {
2424 assert_eq!(
2426 package_map_pattern_capture("./dist/*/index.js", "./dist/utils/index.js"),
2427 Some("utils".to_string()),
2428 );
2429 }
2430
2431 #[test]
2432 fn package_map_pattern_capture_no_prefix_match_returns_none() {
2433 assert_eq!(
2435 package_map_pattern_capture("./lib/*.js", "./src/foo.js"),
2436 None,
2437 );
2438 }
2439
2440 #[test]
2443 fn resolve_package_map_target_no_dot_slash_prefix_returns_none() {
2444 let root = PathBuf::from("/project/packages/ui");
2446 let pj = fallow_config::PackageJson::default();
2447 with_package_map_ctx(root, Some("@myorg/ui"), pj, &[], |ctx, manifest, _root| {
2448 let result = resolve_package_map_target(ctx, manifest, "src/index.ts", None);
2449 assert_eq!(result, None, "target without './' should return None");
2450 });
2451 }
2452
2453 #[test]
2454 fn resolve_package_map_target_parent_dir_returns_none() {
2455 let root = PathBuf::from("/project/packages/ui");
2457 let pj = fallow_config::PackageJson::default();
2458 with_package_map_ctx(root, Some("@myorg/ui"), pj, &[], |ctx, manifest, _root| {
2459 let result = resolve_package_map_target(ctx, manifest, "./../outside/file.ts", None);
2460 assert_eq!(result, None, "parent-dir target should return None");
2461 });
2462 }
2463
2464 #[test]
2465 fn resolve_package_map_target_absolute_path_returns_none() {
2466 let root = PathBuf::from("/project/packages/ui");
2468 let pj = fallow_config::PackageJson::default();
2469 with_package_map_ctx(root, Some("@myorg/ui"), pj, &[], |ctx, manifest, _root| {
2470 let result = resolve_package_map_target(ctx, manifest, ".//abs/path.ts", None);
2473 assert_eq!(result, None, "absolute target should return None");
2474 });
2475 }
2476
2477 #[test]
2478 fn resolve_package_map_target_valid_target_hits_raw_path_map() {
2479 let root = PathBuf::from("/project/packages/ui");
2481 let src = root.join("src/index.ts");
2482 let pj = fallow_config::PackageJson::default();
2483 with_package_map_ctx(
2484 root,
2485 Some("@myorg/ui"),
2486 pj,
2487 &[(src, FileId(5))],
2488 |ctx, manifest, _root| {
2489 let result = resolve_package_map_target(ctx, manifest, "./src/index.ts", None);
2490 assert_eq!(
2491 result,
2492 Some(FileId(5)),
2493 "valid target in raw_path_to_id should resolve"
2494 );
2495 },
2496 );
2497 }
2498
2499 #[test]
2502 fn package_import_source_subpath_strips_hash_and_package_name() {
2503 let manifest = PackageManifestInfo {
2504 root: PathBuf::from("/project"),
2505 canonical_root: PathBuf::from("/project"),
2506 name: Some("my-pkg".to_string()),
2507 package_json: fallow_config::PackageJson::default(),
2508 };
2509 let result = package_import_source_subpath(&manifest, "#my-pkg/utils");
2510 assert_eq!(
2511 result,
2512 Some(PathBuf::from("utils")),
2513 "should strip '#', package name, and '/' separator"
2514 );
2515 }
2516
2517 #[test]
2518 fn package_import_source_subpath_no_package_name_match_keeps_full_subpath() {
2519 let manifest = PackageManifestInfo {
2522 root: PathBuf::from("/project"),
2523 canonical_root: PathBuf::from("/project"),
2524 name: Some("my-pkg".to_string()),
2525 package_json: fallow_config::PackageJson::default(),
2526 };
2527 let result = package_import_source_subpath(&manifest, "#utils");
2528 assert_eq!(
2529 result,
2530 Some(PathBuf::from("utils")),
2531 "without package-name prefix the full subpath should be kept"
2532 );
2533 }
2534
2535 #[test]
2536 fn package_import_source_subpath_empty_hash_returns_none() {
2537 let manifest = PackageManifestInfo {
2540 root: PathBuf::from("/project"),
2541 canonical_root: PathBuf::from("/project"),
2542 name: Some("my-pkg".to_string()),
2543 package_json: fallow_config::PackageJson::default(),
2544 };
2545 let result = package_import_source_subpath(&manifest, "#");
2547 assert_eq!(
2548 result, None,
2549 "specifier '#' with empty body should return None"
2550 );
2551 }
2552
2553 #[test]
2554 fn package_import_source_subpath_no_hash_returns_none() {
2555 let manifest = PackageManifestInfo {
2557 root: PathBuf::from("/project"),
2558 canonical_root: PathBuf::from("/project"),
2559 name: Some("my-pkg".to_string()),
2560 package_json: fallow_config::PackageJson::default(),
2561 };
2562 let result = package_import_source_subpath(&manifest, "no-hash");
2563 assert_eq!(result, None, "specifier without '#' should return None");
2564 }
2565
2566 #[test]
2567 fn package_import_source_subpath_no_manifest_name() {
2568 let manifest = PackageManifestInfo {
2570 root: PathBuf::from("/project"),
2571 canonical_root: PathBuf::from("/project"),
2572 name: None,
2573 package_json: fallow_config::PackageJson::default(),
2574 };
2575 let result = package_import_source_subpath(&manifest, "#internal/helper");
2576 assert_eq!(
2577 result,
2578 Some(PathBuf::from("internal/helper")),
2579 "manifest without name should return the full stripped specifier"
2580 );
2581 }
2582
2583 #[test]
2586 fn nearest_package_manifest_returns_deepest_match() {
2587 let root1 = PathBuf::from("/project");
2588 let root2 = PathBuf::from("/project/packages/ui");
2589 let m1 = PackageManifestInfo {
2590 root: root1.clone(),
2591 canonical_root: root1,
2592 name: Some("root".to_string()),
2593 package_json: fallow_config::PackageJson::default(),
2594 };
2595 let m2 = PackageManifestInfo {
2596 root: root2.clone(),
2597 canonical_root: root2,
2598 name: Some("@myorg/ui".to_string()),
2599 package_json: fallow_config::PackageJson::default(),
2600 };
2601 let manifests = [m1, m2];
2602 let from_file = Path::new("/project/packages/ui/src/index.ts");
2603 let result = nearest_package_manifest(&manifests, from_file);
2604 assert_eq!(
2605 result.and_then(|m| m.name.as_deref()),
2606 Some("@myorg/ui"),
2607 "should pick the deepest (longest path) manifest that contains the file"
2608 );
2609 }
2610
2611 #[test]
2612 fn nearest_package_manifest_no_match_returns_none() {
2613 let root = PathBuf::from("/project/packages/ui");
2614 let m = PackageManifestInfo {
2615 root: root.clone(),
2616 canonical_root: root,
2617 name: Some("@myorg/ui".to_string()),
2618 package_json: fallow_config::PackageJson::default(),
2619 };
2620 let manifests = [m];
2621 let from_file = Path::new("/other/project/src/index.ts");
2623 let result = nearest_package_manifest(&manifests, from_file);
2624 assert!(
2625 result.is_none(),
2626 "file outside all manifest roots should return None"
2627 );
2628 }
2629
2630 #[test]
2633 fn lookup_internal_file_id_uses_path_to_id_when_raw_misses() {
2634 let target = PathBuf::from("/project/src/index.ts");
2637 let mut path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
2638 path_to_id.insert(target.as_path(), FileId(99));
2639 let raw_path_to_id: FxHashMap<&Path, FileId> = FxHashMap::default();
2640 let workspace_roots: FxHashMap<&str, &Path> = FxHashMap::default();
2641 let condition_names = conditions();
2642 let resolver = oxc_resolver::Resolver::new(oxc_resolver::ResolveOptions::default());
2643 let tsconfig_warned = std::sync::Mutex::new(FxHashSet::default());
2644 let tsconfig_cache = TsconfigCache::default();
2645 let root = PathBuf::from("/project");
2646 let ctx = ResolveContext {
2647 resolver: &resolver,
2648 style_resolver: &resolver,
2649 extensions: &[],
2650 path_to_id: &path_to_id,
2651 raw_path_to_id: &raw_path_to_id,
2652 workspace_roots: &workspace_roots,
2653 package_manifests: &[],
2654 condition_names: &condition_names,
2655 path_aliases: &[],
2656 scss_include_paths: &[],
2657 static_dir_mappings: &[],
2658 root: &root,
2659 canonical_fallback: None,
2660 tsconfig_warned: &tsconfig_warned,
2661 tsconfig_cache: &tsconfig_cache,
2662 };
2663 let result = lookup_internal_file_id(&ctx, &target);
2664 assert_eq!(
2665 result,
2666 Some(FileId(99)),
2667 "should fall back from raw_path_to_id to path_to_id"
2668 );
2669 }
2670
2671 #[test]
2674 fn try_scss_partial_fallback_rejects_colon_specifier() {
2675 let root = PathBuf::from("/project");
2678 let pj = fallow_config::PackageJson::default();
2679 with_package_map_ctx(root.clone(), None, pj, &[], |ctx, _manifest, _r| {
2680 let from_file = root.join("src/main.scss");
2681 let result = try_scss_partial_fallback(ctx, &from_file, "sass:math");
2682 assert!(
2683 result.is_none(),
2684 "specifier with ':' should short-circuit to None"
2685 );
2686 });
2687 }
2688
2689 #[test]
2690 fn try_scss_partial_fallback_rejects_already_partial_filename() {
2691 let root = PathBuf::from("/project");
2694 let pj = fallow_config::PackageJson::default();
2695 with_package_map_ctx(root.clone(), None, pj, &[], |ctx, _manifest, _r| {
2696 let from_file = root.join("src/main.scss");
2697 let result = try_scss_partial_fallback(ctx, &from_file, "./_variables");
2698 assert!(
2699 result.is_none(),
2700 "already-partial filename should short-circuit to None"
2701 );
2702 });
2703 }
2704
2705 #[test]
2708 fn try_workspace_package_fallback_rejects_relative_specifier() {
2709 let root = PathBuf::from("/project");
2712 let pj = fallow_config::PackageJson::default();
2713 with_package_map_ctx(root, None, pj, &[], |ctx, _manifest, _r| {
2714 let result = try_workspace_package_fallback(ctx, "./local/module");
2715 assert!(
2716 result.is_none(),
2717 "relative specifier should return None from workspace fallback"
2718 );
2719 });
2720 }
2721
2722 #[test]
2723 fn try_workspace_package_fallback_rejects_absolute_path() {
2724 let root = PathBuf::from("/project");
2726 let pj = fallow_config::PackageJson::default();
2727 with_package_map_ctx(root, None, pj, &[], |ctx, _manifest, _r| {
2728 let result = try_workspace_package_fallback(ctx, "/absolute/path");
2729 assert!(
2730 result.is_none(),
2731 "absolute path should return None from workspace fallback"
2732 );
2733 });
2734 }
2735}