Skip to main content

fallow_graph/resolve/
types.rs

1//! Type definitions and constants for import resolution.
2
3use std::path::{Path, PathBuf};
4use std::sync::Mutex;
5
6use oxc_resolver::Resolver;
7use rustc_hash::{FxHashMap, FxHashSet};
8use serde_json::Value;
9
10use fallow_types::discover::FileId;
11
12/// Result of resolving an import specifier.
13#[derive(Debug, Clone)]
14pub enum ResolveResult {
15    /// Resolved to a file within the project.
16    InternalModule(FileId),
17    /// Resolved to a project file through a framework convention auto-import.
18    SyntheticAutoImport(FileId),
19    /// Resolved to a workspace or self package source file while preserving
20    /// dependency usage for package accounting.
21    InternalPackageModule {
22        /// Internal source file reached by the package map.
23        file_id: FileId,
24        /// Package name that was used in the import specifier.
25        package_name: String,
26    },
27    /// Resolved to a file outside the project (`node_modules`, `.json`, etc.).
28    ExternalFile(PathBuf),
29    /// Bare specifier — an npm package.
30    NpmPackage(String),
31    /// Could not resolve.
32    Unresolvable(String),
33}
34
35impl ResolveResult {
36    /// Return the target file for any project-internal result.
37    #[must_use]
38    pub const fn internal_file_id(&self) -> Option<FileId> {
39        match self {
40            Self::InternalModule(file_id)
41            | Self::SyntheticAutoImport(file_id)
42            | Self::InternalPackageModule { file_id, .. } => Some(*file_id),
43            Self::ExternalFile(_) | Self::NpmPackage(_) | Self::Unresolvable(_) => None,
44        }
45    }
46
47    /// Return whether this edge was synthesized from framework auto-import conventions.
48    #[must_use]
49    pub const fn is_synthetic_auto_import(&self) -> bool {
50        matches!(self, Self::SyntheticAutoImport(_))
51    }
52
53    /// Return the package name that should receive dependency usage credit.
54    #[must_use]
55    pub fn package_usage_name(&self) -> Option<&str> {
56        match self {
57            Self::InternalPackageModule { package_name, .. } | Self::NpmPackage(package_name) => {
58                Some(package_name)
59            }
60            Self::InternalModule(_)
61            | Self::SyntheticAutoImport(_)
62            | Self::ExternalFile(_)
63            | Self::Unresolvable(_) => None,
64        }
65    }
66}
67
68/// A resolved import with its target.
69#[derive(Debug, Clone)]
70pub struct ResolvedImport {
71    /// The original import information.
72    pub info: fallow_types::extract::ImportInfo,
73    /// Where the import resolved to.
74    pub target: ResolveResult,
75}
76
77/// A resolved re-export with its target.
78#[derive(Debug, Clone)]
79pub struct ResolvedReExport {
80    /// The original re-export information.
81    pub info: fallow_types::extract::ReExportInfo,
82    /// Where the re-export source resolved to.
83    pub target: ResolveResult,
84}
85
86/// Any source-bearing module edge that resolves one literal specifier.
87pub enum ResolvedSourceEdge<'a> {
88    /// Static or literal dynamic import edge.
89    Import(&'a ResolvedImport),
90    /// Re-export source edge.
91    ReExport(&'a ResolvedReExport),
92}
93
94impl<'a> ResolvedSourceEdge<'a> {
95    /// Return the original source specifier.
96    #[must_use]
97    pub fn source_specifier(&self) -> &'a str {
98        match self {
99            Self::Import(import) => &import.info.source,
100            Self::ReExport(re_export) => &re_export.info.source,
101        }
102    }
103
104    /// Return the resolved target.
105    #[must_use]
106    pub const fn target(&self) -> &'a ResolveResult {
107        match self {
108            Self::Import(import) => &import.target,
109            Self::ReExport(re_export) => &re_export.target,
110        }
111    }
112
113    /// Return whether this edge is type-only.
114    #[must_use]
115    pub const fn is_type_only(&self) -> bool {
116        match self {
117            Self::Import(import) => import.info.is_type_only,
118            Self::ReExport(re_export) => re_export.info.is_type_only,
119        }
120    }
121
122    /// Return the span of the full import or re-export declaration.
123    #[must_use]
124    pub const fn span(&self) -> oxc_span::Span {
125        match self {
126            Self::Import(import) => import.info.span,
127            Self::ReExport(re_export) => re_export.info.span,
128        }
129    }
130
131    /// Return the source literal span when the extractor has one.
132    #[must_use]
133    pub const fn source_span(&self) -> oxc_span::Span {
134        match self {
135            Self::Import(import) => import.info.source_span,
136            Self::ReExport(_) => oxc_span::Span::new(0, 0),
137        }
138    }
139}
140
141/// Fully resolved module with all imports mapped to targets.
142#[derive(Debug)]
143pub struct ResolvedModule {
144    /// Unique file identifier.
145    pub file_id: FileId,
146    /// Absolute path to the module file.
147    pub path: PathBuf,
148    /// All export declarations in this module.
149    pub exports: Vec<fallow_types::extract::ExportInfo>,
150    /// All re-exports with resolved targets.
151    pub re_exports: Vec<ResolvedReExport>,
152    /// All static imports with resolved targets.
153    pub resolved_imports: Vec<ResolvedImport>,
154    /// All dynamic imports with resolved targets.
155    pub resolved_dynamic_imports: Vec<ResolvedImport>,
156    /// Dynamic import patterns matched against discovered files.
157    pub resolved_dynamic_patterns: Vec<(fallow_types::extract::DynamicImportPattern, Vec<FileId>)>,
158    /// Static member accesses (e.g., `Status.Active`).
159    pub member_accesses: Vec<fallow_types::extract::MemberAccess>,
160    /// Typed semantic facts produced by extraction for cross-layer analysis.
161    pub semantic_facts: Box<[fallow_types::extract::SemanticFact]>,
162    /// Identifiers used as whole objects (Object.values, for..in, spread, etc.).
163    pub whole_object_uses: Box<[String]>,
164    /// Whether this module uses `CommonJS` exports.
165    pub has_cjs_exports: bool,
166    /// Whether this module declares at least one Angular `@Component({
167    /// templateUrl: ... })` decorator. Mirrors `ModuleInfo.has_angular_component_template_url`;
168    /// see that field for the contract this gate enforces.
169    pub has_angular_component_template_url: bool,
170    /// Local names of import bindings that are never referenced in this file.
171    pub unused_import_bindings: FxHashSet<String>,
172    /// Local import bindings referenced from type positions.
173    pub type_referenced_import_bindings: Vec<String>,
174    /// Local import bindings referenced from runtime/value positions.
175    pub value_referenced_import_bindings: Vec<String>,
176    /// Namespace-import aliases re-exported through an object literal.
177    /// See `fallow_types::extract::NamespaceObjectAlias` for the shape.
178    pub namespace_object_aliases: Vec<fallow_types::extract::NamespaceObjectAlias>,
179    /// Exported free-function factories that provably return one class instance.
180    /// See `fallow_types::extract::FactoryReturnExport` and issue #1441 (Part A).
181    pub exported_factory_returns: Box<[fallow_types::extract::FactoryReturnExport]>,
182}
183
184impl Default for ResolvedModule {
185    fn default() -> Self {
186        Self {
187            file_id: FileId(0),
188            path: PathBuf::new(),
189            exports: vec![],
190            re_exports: vec![],
191            resolved_imports: vec![],
192            resolved_dynamic_imports: vec![],
193            resolved_dynamic_patterns: vec![],
194            member_accesses: vec![],
195            semantic_facts: Box::default(),
196            whole_object_uses: Box::default(),
197            has_cjs_exports: false,
198            has_angular_component_template_url: false,
199            unused_import_bindings: FxHashSet::default(),
200            type_referenced_import_bindings: vec![],
201            value_referenced_import_bindings: vec![],
202            namespace_object_aliases: vec![],
203            exported_factory_returns: Box::default(),
204        }
205    }
206}
207
208impl ResolvedModule {
209    /// Iterate over all concrete resolved imports in source order buckets.
210    ///
211    /// Includes static `import`/`require` edges and literal dynamic `import()`
212    /// edges. Dynamic import patterns are intentionally excluded because they
213    /// resolve to sets of files rather than single import specifiers.
214    pub fn all_resolved_imports(&self) -> impl Iterator<Item = &ResolvedImport> {
215        self.resolved_imports
216            .iter()
217            .chain(self.resolved_dynamic_imports.iter())
218    }
219
220    /// Iterate over every literal source edge that has one resolved target.
221    ///
222    /// Includes static imports, literal dynamic imports, and re-export sources.
223    /// Dynamic import patterns are excluded because they resolve to sets of
224    /// files rather than single import specifiers.
225    pub fn all_resolved_source_edges(&self) -> impl Iterator<Item = ResolvedSourceEdge<'_>> {
226        self.resolved_imports
227            .iter()
228            .map(ResolvedSourceEdge::Import)
229            .chain(
230                self.resolved_dynamic_imports
231                    .iter()
232                    .map(ResolvedSourceEdge::Import),
233            )
234            .chain(self.re_exports.iter().map(ResolvedSourceEdge::ReExport))
235    }
236}
237
238/// Shared context for resolving import specifiers.
239///
240/// Groups the immutable lookup tables and caches that are shared across all
241/// `resolve_specifier` calls within a single `resolve_all_imports` invocation.
242pub(super) struct ResolveContext<'a> {
243    /// The oxc_resolver instance (configured once, shared across threads).
244    pub resolver: &'a Resolver,
245    /// CSS-only resolver with package.json `sass` and `style` conditions enabled.
246    /// Used only for stylesheet package subpaths so JS/TS imports do not
247    /// accidentally prefer CSS export branches.
248    pub style_resolver: &'a Resolver,
249    /// Ordered extension list used by the resolver.
250    pub extensions: &'a [String],
251    /// Canonical path → FileId lookup (raw paths when root is canonical).
252    pub path_to_id: &'a FxHashMap<&'a Path, FileId>,
253    /// Raw (non-canonical) path → FileId lookup.
254    pub raw_path_to_id: &'a FxHashMap<&'a Path, FileId>,
255    /// Workspace name → canonical root path.
256    pub workspace_roots: &'a FxHashMap<&'a str, &'a Path>,
257    /// Package manifests for the root package and workspace packages.
258    pub package_manifests: &'a [PackageManifestInfo],
259    /// Ordered package condition names matching the resolver configuration.
260    pub condition_names: &'a [String],
261    /// Plugin-provided path aliases (prefix, replacement).
262    pub path_aliases: &'a [(String, String)],
263    /// Absolute directories to search when resolving bare SCSS/Sass
264    /// `@import` / `@use` specifiers. Populated from Angular's
265    /// `stylePreprocessorOptions.includePaths` and equivalent settings.
266    pub scss_include_paths: &'a [PathBuf],
267    /// Static directory URL mappings from framework config.
268    /// Each tuple is `(absolute_source_dir, normalized_url_mount)`.
269    pub static_dir_mappings: &'a [(PathBuf, String)],
270    /// Project root directory.
271    pub root: &'a Path,
272    /// Lazy canonical path → FileId fallback for intra-project symlinks.
273    /// Only initialized on first miss when root is canonical. `None` when
274    /// path_to_id already uses canonical paths (root is not canonical).
275    pub canonical_fallback: Option<&'a CanonicalFallback<'a>>,
276    /// Dedup set for broken-tsconfig warnings. Emits one `tracing::warn!`
277    /// per unique error message instead of spamming the log with one
278    /// warning per affected file. Shared across all parallel resolver
279    /// threads via `Mutex`. Empty and unused when no tsconfig errors occur.
280    pub tsconfig_warned: &'a Mutex<FxHashSet<String>>,
281    /// Per-analysis cache for local tsconfig discovery and JSON parsing.
282    /// Import resolution calls these fallbacks for every unresolved or
283    /// tsconfig-poisoned specifier, so keeping it session-local avoids
284    /// repeated filesystem work without risking stale data across runs.
285    pub tsconfig_cache: &'a TsconfigCache,
286    /// Per-analysis cache of `dunce::canonicalize` results keyed by resolved
287    /// path. Every import resolving to a `node_modules` / output-dir / symlinked
288    /// target is realpath'd during classification, and the same package path is
289    /// re-canonicalized for every file that imports the package. The result is a
290    /// pure function of the path's on-disk state (constant within a run), so the
291    /// cache is session-local for watch-mode safety.
292    pub canonicalize_cache: &'a CanonicalizeCache,
293}
294
295/// Session-local cache of `dunce::canonicalize` results keyed by input path.
296#[derive(Default)]
297pub(super) struct CanonicalizeCache {
298    map: Mutex<FxHashMap<PathBuf, Option<PathBuf>>>,
299}
300
301impl CanonicalizeCache {
302    /// Return the cached `dunce::canonicalize(path)` outcome, computing it on
303    /// first miss. `None` (a path that fails to canonicalize) is cached too so a
304    /// repeated probe of the same missing path does not re-issue the syscall.
305    pub fn get(&self, path: &Path) -> Option<PathBuf> {
306        if let Ok(cache) = self.map.lock()
307            && let Some(value) = cache.get(path)
308        {
309            return value.clone();
310        }
311        let value = dunce::canonicalize(path).ok();
312        if let Ok(mut cache) = self.map.lock() {
313            cache.insert(path.to_path_buf(), value.clone());
314        }
315        value
316    }
317}
318
319/// Session-local cache for tsconfig helper lookups used during import resolution.
320#[derive(Default)]
321pub(super) struct TsconfigCache {
322    json: Mutex<FxHashMap<PathBuf, Option<Value>>>,
323    chains: Mutex<FxHashMap<PathBuf, Vec<PathBuf>>>,
324}
325
326impl TsconfigCache {
327    /// Return a cached parsed tsconfig JSON value, loading it on first miss.
328    pub fn json(&self, path: &Path, load: impl FnOnce(&Path) -> Option<Value>) -> Option<Value> {
329        if let Ok(cache) = self.json.lock()
330            && let Some(value) = cache.get(path)
331        {
332            return value.clone();
333        }
334
335        let value = load(path);
336        if let Ok(mut cache) = self.json.lock() {
337            cache.insert(path.to_path_buf(), value.clone());
338        }
339        value
340    }
341
342    /// Return the cached tsconfig chain for a source file, if one exists.
343    pub fn chain(&self, from_file: &Path) -> Option<Vec<PathBuf>> {
344        self.chains
345            .lock()
346            .ok()
347            .and_then(|cache| cache.get(from_file).cloned())
348    }
349
350    /// Store the computed tsconfig chain for a source file.
351    pub fn store_chain(&self, from_file: &Path, chain: Vec<PathBuf>) {
352        if let Ok(mut cache) = self.chains.lock() {
353            cache.insert(from_file.to_path_buf(), chain);
354        }
355    }
356}
357
358/// Package manifest data used by source fallbacks.
359#[derive(Debug, Clone)]
360pub(super) struct PackageManifestInfo {
361    /// Package root path as discovered from the workspace tree.
362    pub root: PathBuf,
363    /// Canonical package root path for node_modules symlink comparisons.
364    pub canonical_root: PathBuf,
365    /// Parsed package name.
366    pub name: Option<String>,
367    /// Parsed package.json fields.
368    pub package_json: fallow_config::PackageJson,
369}
370
371/// Thread-safe lazy canonical path index, built on first access.
372pub(super) struct CanonicalFallback<'a> {
373    files: &'a [fallow_types::discover::DiscoveredFile],
374    map: std::sync::OnceLock<FxHashMap<std::path::PathBuf, FileId>>,
375}
376
377impl<'a> CanonicalFallback<'a> {
378    pub const fn new(files: &'a [fallow_types::discover::DiscoveredFile]) -> Self {
379        Self {
380            files,
381            map: std::sync::OnceLock::new(),
382        }
383    }
384
385    /// Look up a canonical path, lazily building the index on first call.
386    pub fn get(&self, canonical: &Path) -> Option<FileId> {
387        let map = self.map.get_or_init(|| {
388            tracing::debug!(
389                "intra-project symlinks detected, building canonical path index ({} files)",
390                self.files.len()
391            );
392            self.files
393                .iter()
394                .filter_map(|f| {
395                    dunce::canonicalize(&f.path)
396                        .ok()
397                        .map(|canonical| (canonical, f.id))
398                })
399                .collect()
400        });
401        map.get(canonical).copied()
402    }
403}
404
405#[cfg(all(test, not(miri)))]
406mod tests {
407    use super::*;
408    use fallow_types::discover::DiscoveredFile;
409
410    #[test]
411    fn canonical_fallback_returns_none_for_empty_files() {
412        let files: Vec<DiscoveredFile> = vec![];
413        let fallback = CanonicalFallback::new(&files);
414        assert!(fallback.get(Path::new("/nonexistent")).is_none());
415    }
416
417    #[test]
418    fn canonical_fallback_finds_existing_file() {
419        let temp = std::env::temp_dir().join("fallow-test-canonical-fallback");
420        let _ = std::fs::create_dir_all(&temp);
421        let test_file = temp.join("test.ts");
422        std::fs::write(&test_file, "").unwrap();
423
424        let files = vec![DiscoveredFile {
425            id: FileId(42),
426            path: test_file.clone(),
427            size_bytes: 0,
428        }];
429        let fallback = CanonicalFallback::new(&files);
430
431        let canonical = dunce::canonicalize(&test_file).unwrap();
432        assert_eq!(fallback.get(&canonical), Some(FileId(42)));
433
434        assert_eq!(fallback.get(&canonical), Some(FileId(42)));
435
436        let _ = std::fs::remove_dir_all(&temp);
437    }
438
439    #[test]
440    fn canonical_fallback_returns_none_for_missing_path() {
441        let temp = std::env::temp_dir().join("fallow-test-canonical-miss");
442        let _ = std::fs::create_dir_all(&temp);
443        let test_file = temp.join("exists.ts");
444        std::fs::write(&test_file, "").unwrap();
445
446        let files = vec![DiscoveredFile {
447            id: FileId(1),
448            path: test_file,
449            size_bytes: 0,
450        }];
451        let fallback = CanonicalFallback::new(&files);
452        assert!(fallback.get(Path::new("/nonexistent/file.ts")).is_none());
453
454        let _ = std::fs::remove_dir_all(&temp);
455    }
456}
457
458/// Known output directory names that may appear in exports map targets.
459/// When an exports map points to `./dist/utils.js`, we try replacing these
460/// prefixes with `src/` (the conventional source directory) to find the tracked
461/// source file.
462pub const OUTPUT_DIRS: &[&str] = &["dist", "build", "out", "esm", "cjs"];
463
464/// Source extensions to try when mapping a built output file back to source.
465pub const SOURCE_EXTS: &[&str] = &["ts", "tsx", "mts", "cts", "js", "jsx", "mjs", "cjs"];
466
467/// React Native platform extension prefixes.
468/// Metro resolves platform-specific files (e.g., `./foo` -> `./foo.web.tsx` on web).
469pub const RN_PLATFORM_PREFIXES: &[&str] = &[".web", ".ios", ".android", ".native"];