Skip to main content

fallow_graph/resolve/
fallbacks.rs

1//! Resolution fallback strategies for import specifiers.
2//!
3//! Handles path alias fallbacks, output-to-source directory mapping, pnpm virtual
4//! store detection, node_modules package extraction, and dynamic import glob patterns.
5
6use 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
16/// Return the post-prefix remainder when `specifier` matches the alias `prefix`
17/// at a path boundary, else `None`.
18///
19/// The match is segment-aware: a bare exact-key alias (e.g. `@scope/sdk` or
20/// `vscode`) matches only on an exact hit or a `/`-delimited continuation, so it
21/// never captures a longer package that merely shares the prefix
22/// (`@scope/sdk-extra`). Prefixes that already end in `/` (`~/`, `@/`, `$lib/`)
23/// match any continuation by construction, preserving their existing behavior.
24fn 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
30/// Try resolving a specifier using plugin-provided path aliases.
31///
32/// Substitutes a matching alias prefix (e.g., `~/`) with a directory relative to the
33/// project root (e.g., `app/`) and resolves the resulting path. This handles framework
34/// aliases like Nuxt's `~/`, `~~/`, `#shared/` that aren't defined in tsconfig.json
35/// but map to real filesystem paths.
36pub(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
78/// Try SCSS partial resolution: `_filename` and `_index` conventions.
79///
80/// SCSS resolves imports in this order:
81/// 1. `@use 'variables'` → `_variables.scss` (partial convention)
82/// 2. `@use 'components'` → `components/_index.scss` or `components/index.scss` (directory index)
83///
84/// Handles both relative (`../styles/variables`) and bare (`variables`) specifiers
85/// that were normalized to `./variables` during extraction.
86pub(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
124/// Try non-partial CSS-extension resolution: `<spec>.scss`, `<spec>.sass`,
125/// `<spec>.css` from the importing file's parent.
126///
127/// This is needed when the standard resolver's extension list contains both
128/// `.vue` / `.svelte` / `.astro` AND CSS extensions. For an SFC `<style>` block
129/// importing `./Foo`, the standard resolver picks `Foo.vue` (the SFC itself!)
130/// before `Foo.scss` because `.vue` comes earlier in the extension list. SCSS
131/// imports must restrict resolution to CSS-family extensions to avoid this
132/// self-import collision. Only invoked when `from_style = true`. See issue #195.
133pub(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
162/// Attempt to resolve a single SCSS specifier and map to an internal module.
163fn 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
182/// Try SCSS `includePaths` fallback: resolve the specifier against each
183/// framework-contributed include directory.
184///
185/// Angular's `stylePreprocessorOptions.includePaths` (and Nx's equivalent via
186/// project.json) adds extra search paths that SCSS resolves against before
187/// falling back to node_modules. Bare `@use 'variables'` statements that were
188/// normalized to `./variables` at extraction time fail the usual file-local
189/// resolution, so when the importing file is `.scss`/`.sass` and the spec
190/// originated from such a bare specifier, we retry against each include path,
191/// applying the SCSS partial (`_variables`) and directory-index conventions.
192/// SFC `<style lang="scss">` imports pass `from_style = true` because their
193/// filesystem importer is `.vue` / `.svelte`, not `.scss` / `.sass`.
194///
195/// The specifier arrives with a `./` prefix because `normalize_css_import_path`
196/// rewrites bare extensionless SCSS specifiers to relative ones. We strip that
197/// prefix here to re-enter the include-path search from the root of each
198/// directory. Relative specifiers that already escape the importing file
199/// (e.g. `../shared/variables`) are left untouched — include paths only
200/// disambiguate bare specifiers, not explicit relative paths.
201pub(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
232/// Probe an SCSS include directory for a bare specifier, applying the standard
233/// SCSS resolution order: exact file, `_`-prefixed partial, `_index` / `index`
234/// directory conventions. Supports `.scss` and `.sass` extensions.
235fn 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
295/// Look up an absolute candidate path in the file index, falling back to
296/// canonical path lookup for intra-project symlinks.
297fn 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
314/// Try SCSS `node_modules` fallback: resolve a bare specifier by walking up
315/// from the importing file and probing each ancestor's `node_modules/` dir.
316///
317/// Sass's `@import` / `@use` resolution algorithm searches `node_modules/` for
318/// bare specifiers after the file-local and `includePaths` searches fail.
319/// `@import 'bootstrap/scss/functions'` resolves to
320/// `node_modules/bootstrap/scss/_functions.scss` via the standard partial
321/// convention; `@import 'animate.css/animate.min'` resolves to
322/// `node_modules/animate.css/animate.min.css` via the CSS-extension fallback.
323///
324/// Files inside `node_modules/` are not in fallow's file index (the default
325/// ignore patterns exclude them), so this function returns
326/// `ResolveResult::NpmPackage` when a candidate exists on disk. That ensures
327/// (1) the `@import` is not reported as unresolved and (2) the npm package is
328/// marked as a used dependency so `unused-dependencies` / `unlisted-dependencies`
329/// stay accurate.
330///
331/// The specifier arrives with a `./` prefix because `normalize_css_import_path`
332/// rewrites bare extensionless SCSS specifiers to relative ones. Parent-relative
333/// specifiers are skipped — they explicitly escape the importing file and must
334/// not be silently redirected to `node_modules`. See issue #125.
335pub(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
375/// Probe candidate filesystem paths for a bare SCSS specifier inside a single
376/// `node_modules/` directory, applying Sass resolution conventions.
377///
378/// Candidate order:
379/// 1. `<bare>.scss` / `<bare>.sass` / `<bare>.css` (extension append)
380/// 2. `<parent>/_<stem>.scss` / `<parent>/_<stem>.sass` (partial convention)
381/// 3. `<bare>/_index.scss` / `<bare>/index.scss` (and `.sass` variants)
382/// 4. `<bare>` (exact, for specifiers that already carry an extension)
383fn 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
420/// Try to map a resolved output path (e.g., `packages/ui/dist/utils.js`) back to
421/// the corresponding source file (e.g., `packages/ui/src/utils.ts`).
422///
423/// This handles cross-workspace imports that go through `exports` maps pointing to
424/// built output directories. Since fallow ignores `dist/`, `build/`, etc. by default,
425/// the resolved path won't be in the file set, but the source file will be.
426///
427/// Nested output subdirectories (e.g., `dist/esm/utils.mjs`, `build/cjs/index.cjs`)
428/// are handled by finding the last output directory component (closest to the file,
429/// avoiding false matches on parent directories) and then walking backwards to collect
430/// all consecutive output directory components before it.
431pub(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()?; // Ensure the suffix has a filename
457
458    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
468/// Try to resolve a package `imports` entry from the nearest owning package.
469///
470/// `#...` specifiers are package-local by definition, so this fallback is only
471/// allowed when the importing file's nearest package manifest has a matching
472/// `imports` key. That keeps unrelated hash-prefixed path aliases unresolved.
473pub(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
505/// Resolve a relative import that lands on a known package root whose built
506/// entry points are absent but whose package metadata points at source files.
507pub(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
849/// Extract npm package name from a resolved path inside `node_modules`.
850///
851/// Given a path like `/project/node_modules/react/index.js`, returns `Some("react")`.
852/// Given a path like `/project/node_modules/@scope/pkg/dist/index.js`, returns `Some("@scope/pkg")`.
853/// Returns `None` if the path doesn't contain a `node_modules` segment.
854pub 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
881/// Try to map a pnpm virtual store path back to a workspace source file.
882///
883/// When pnpm uses injected dependencies or certain linking strategies, canonical
884/// paths go through `.pnpm`:
885///   `/project/node_modules/.pnpm/@myorg+ui@1.0.0/node_modules/@myorg/ui/dist/index.js`
886///
887/// This function detects such paths, extracts the package name, checks if it
888/// matches a workspace package, and tries to find the source file in that workspace.
889pub(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
940/// Try to resolve a bare specifier as a workspace package reference.
941///
942/// When the specifier's package name matches a workspace package, resolve the
943/// subpath against that package's root directory directly instead of going
944/// through `node_modules`. Covers two cases:
945///
946/// 1. **Self-referencing package imports**: Node.js v12+ lets a package import
947///    itself via its own name (`import { X } from '@org/pkg/subentry'` from
948///    inside `@org/pkg`). Angular libraries built with `ng-packagr` rely on
949///    this to declare secondary entry points.
950/// 2. **Cross-workspace imports without `node_modules` symlinks**: monorepos
951///    that have not been installed yet, or bundlers that bypass `node_modules`
952///    entirely, still need to resolve `@org/other-pkg/sub` to the sibling
953///    workspace's source file.
954///
955/// Strategy: prefer a matching package `exports` target when the manifest has
956/// one, then try the package source layout directly when no `exports` map exists,
957/// and finally resolve the stripped subpath as a relative path from inside the
958/// package root. The manifest branches cover source-only workspaces whose
959/// package metadata points at missing `dist` output.
960///
961/// See issues #106, #641, and #725.
962pub(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
993/// Outcome of attempting workspace resolution through a matching package
994/// manifest's `exports` map or source layout.
995enum ManifestWorkspaceResolution {
996    /// A target was resolved.
997    Resolved(ResolveResult),
998    /// An `exports` map exists but the subpath is unmatched or null-blocked.
999    Blocked,
1000    /// No manifest matched, or it had no usable resolution; keep trying.
1001    Continue,
1002}
1003
1004/// Try resolving via the package manifest: `exports` map first, then the source
1005/// layout for manifests without an `exports` map.
1006fn 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
1050/// Resolve the stripped subpath as a relative import from inside the package
1051/// root, mapping the resolved path back to an internal module via the id maps
1052/// and source fallback.
1053fn 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
1100/// Convert a `DynamicImportPattern` to a glob string for file matching.
1101pub(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(&regular_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    // --- package_map_target: non-object map branches (lines 583-586) ---
2243
2244    #[test]
2245    fn package_map_target_string_value_dot_key() {
2246        // A non-object top-level map with specifier_key "." delegates to
2247        // package_map_match_value immediately.
2248        let map = serde_json::Value::String("./src/index.ts".to_string());
2249        let conds = conditions();
2250        // A string value resolves to Targets.
2251        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        // A non-object top-level map with a non-"." specifier returns NoMatch.
2261        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        // A null top-level map with "." returns Blocked (null means blocked).
2273        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        // A bool top-level map with non-dot key is not an object, returns NoMatch.
2285        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    // --- package_map_target: condition-only object map (lines 592-596) ---
2295
2296    #[test]
2297    fn package_map_target_condition_only_object_dot_key() {
2298        // An object whose keys are all conditions (not "." or "./...") is treated
2299        // as a condition map when specifier_key is ".".
2300        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        // Same object, but specifier_key != "." returns NoMatch because no
2315        // subpath key like "./foo" exists.
2316        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    // --- resolve_package_map_value: unmatched conditions, bool, null (lines 641-654) ---
2329
2330    #[test]
2331    fn resolve_package_map_value_unmatched_conditions_returns_none() {
2332        // Object whose only key is not in the active condition set returns None.
2333        let value = serde_json::json!({ "browser": "./src/browser.js" });
2334        let conds = conditions(); // does not include "browser"
2335        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        // Array where every element resolves to None yields None.
2378        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        // Array where one element is a valid string target.
2390        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    // --- package_map_pattern_capture: two-star and no-star branches (lines 659-665) ---
2401
2402    #[test]
2403    fn package_map_pattern_capture_no_star_returns_none() {
2404        // A pattern without '*' returns None (no star found).
2405        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        // A pattern with more than one '*' returns None.
2415        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        // Sanity: the happy path still works after the guard checks.
2425        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        // Specifier does not start with the pattern prefix.
2434        assert_eq!(
2435            package_map_pattern_capture("./lib/*.js", "./src/foo.js"),
2436            None,
2437        );
2438    }
2439
2440    // --- resolve_package_map_target: parent-dir and root-absolute guard (lines 719-721) ---
2441
2442    #[test]
2443    fn resolve_package_map_target_no_dot_slash_prefix_returns_none() {
2444        // A target that does not start with "./" is rejected by strip_prefix.
2445        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        // A target starting with "../" is rejected as a path escape.
2456        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        // A target starting with "/" after stripping "./" prefix is rejected.
2467        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            // "./" + "/" -> "/" after strip_prefix("./") which is no-op, but
2471            // an absolute path disguised as ".//abs" yields "/" start after strip.
2472            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        // A valid "./" target resolves when the path is in raw_path_to_id.
2480        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    // --- package_import_source_subpath: variants (lines 673-689) ---
2500
2501    #[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        // When the specifier after '#' does not start with the package name,
2520        // the full stripped specifier is returned.
2521        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        // "#" with nothing after returns None because stripped is empty and
2538        // the empty string is rejected by the is_empty guard.
2539        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        // "#" strips to "", which is_empty is true, so returns None.
2546        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        // A specifier not starting with '#' returns None.
2556        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        // When the manifest has no name the full stripped specifier is returned.
2569        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    // --- nearest_package_manifest: deepest selection (lines 691-701) ---
2584
2585    #[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        // File is outside the manifest root.
2622        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    // --- lookup_internal_file_id: path_to_id fallback (line 832) ---
2631
2632    #[test]
2633    fn lookup_internal_file_id_uses_path_to_id_when_raw_misses() {
2634        // When raw_path_to_id does not contain the path but path_to_id does,
2635        // lookup_internal_file_id should fall back to path_to_id.
2636        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    // --- try_scss_partial_fallback: colon guard (line 91) ---
2672
2673    #[test]
2674    fn try_scss_partial_fallback_rejects_colon_specifier() {
2675        // A specifier containing ':' (e.g. a Sass built-in like "sass:math")
2676        // must return None immediately.
2677        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        // A specifier whose filename already starts with '_' (i.e. it's already a
2692        // partial path) must return None immediately.
2693        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    // --- try_workspace_package_fallback: bare-specifier guard (lines 967-968) ---
2706
2707    #[test]
2708    fn try_workspace_package_fallback_rejects_relative_specifier() {
2709        // Relative specifiers (starting with "./" or "../") are not bare specifiers
2710        // and must return None without touching any manifest.
2711        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        // Absolute paths are not bare specifiers either.
2725        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}