1use 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#[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
14pub enum ResolveResult {
15 InternalModule(FileId),
17 SyntheticAutoImport(FileId),
19 InternalPackageModule {
22 file_id: FileId,
24 package_name: String,
26 },
27 ExternalFile(PathBuf),
29 NpmPackage(String),
31 Unresolvable(String),
33}
34
35impl ResolveResult {
36 #[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 #[must_use]
49 pub const fn is_synthetic_auto_import(&self) -> bool {
50 matches!(self, Self::SyntheticAutoImport(_))
51 }
52
53 #[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#[derive(Debug, Clone)]
70pub struct ResolvedImport {
71 pub info: fallow_types::extract::ImportInfo,
73 pub target: ResolveResult,
75}
76
77#[derive(Debug, Clone)]
79pub struct ResolvedReExport {
80 pub info: fallow_types::extract::ReExportInfo,
82 pub target: ResolveResult,
84}
85
86pub enum ResolvedSourceEdge<'a> {
88 Import(&'a ResolvedImport),
90 ReExport(&'a ResolvedReExport),
92}
93
94impl<'a> ResolvedSourceEdge<'a> {
95 #[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 #[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 #[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 #[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 #[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#[derive(Debug)]
143pub struct ResolvedModule {
144 pub file_id: FileId,
146 pub path: PathBuf,
148 pub exports: Vec<fallow_types::extract::ExportInfo>,
150 pub re_exports: Vec<ResolvedReExport>,
152 pub resolved_imports: Vec<ResolvedImport>,
154 pub resolved_dynamic_imports: Vec<ResolvedImport>,
156 pub resolved_dynamic_patterns: Vec<(fallow_types::extract::DynamicImportPattern, Vec<FileId>)>,
158 pub member_accesses: Vec<fallow_types::extract::MemberAccess>,
160 pub semantic_facts: Box<[fallow_types::extract::SemanticFact]>,
162 pub whole_object_uses: Box<[String]>,
164 pub has_cjs_exports: bool,
166 pub has_angular_component_template_url: bool,
170 pub unused_import_bindings: FxHashSet<String>,
172 pub type_referenced_import_bindings: Vec<String>,
174 pub value_referenced_import_bindings: Vec<String>,
176 pub namespace_object_aliases: Vec<fallow_types::extract::NamespaceObjectAlias>,
179 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 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 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
238pub(super) struct ResolveContext<'a> {
243 pub resolver: &'a Resolver,
245 pub style_resolver: &'a Resolver,
249 pub extensions: &'a [String],
251 pub path_to_id: &'a FxHashMap<&'a Path, FileId>,
253 pub raw_path_to_id: &'a FxHashMap<&'a Path, FileId>,
255 pub workspace_roots: &'a FxHashMap<&'a str, &'a Path>,
257 pub package_manifests: &'a [PackageManifestInfo],
259 pub condition_names: &'a [String],
261 pub path_aliases: &'a [(String, String)],
263 pub scss_include_paths: &'a [PathBuf],
267 pub static_dir_mappings: &'a [(PathBuf, String)],
270 pub root: &'a Path,
272 pub canonical_fallback: Option<&'a CanonicalFallback<'a>>,
276 pub tsconfig_warned: &'a Mutex<FxHashSet<String>>,
281 pub tsconfig_cache: &'a TsconfigCache,
286 pub canonicalize_cache: &'a CanonicalizeCache,
293}
294
295#[derive(Default)]
297pub(super) struct CanonicalizeCache {
298 map: Mutex<FxHashMap<PathBuf, Option<PathBuf>>>,
299}
300
301impl CanonicalizeCache {
302 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#[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 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 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 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#[derive(Debug, Clone)]
360pub(super) struct PackageManifestInfo {
361 pub root: PathBuf,
363 pub canonical_root: PathBuf,
365 pub name: Option<String>,
367 pub package_json: fallow_config::PackageJson,
369}
370
371pub(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 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
458pub const OUTPUT_DIRS: &[&str] = &["dist", "build", "out", "esm", "cjs"];
463
464pub const SOURCE_EXTS: &[&str] = &["ts", "tsx", "mts", "cts", "js", "jsx", "mjs", "cjs"];
466
467pub const RN_PLATFORM_PREFIXES: &[&str] = &[".web", ".ios", ".android", ".native"];