1use std::path::{Path, PathBuf};
4use std::sync::Mutex;
5
6use oxc_resolver::Resolver;
7use rustc_hash::{FxHashMap, FxHashSet};
8
9use fallow_types::discover::FileId;
10
11#[derive(Debug, Clone)]
13pub enum ResolveResult {
14 InternalModule(FileId),
16 ExternalFile(PathBuf),
18 NpmPackage(String),
20 Unresolvable(String),
22}
23
24#[derive(Debug, Clone)]
26pub struct ResolvedImport {
27 pub info: fallow_types::extract::ImportInfo,
29 pub target: ResolveResult,
31}
32
33#[derive(Debug, Clone)]
35pub struct ResolvedReExport {
36 pub info: fallow_types::extract::ReExportInfo,
38 pub target: ResolveResult,
40}
41
42#[derive(Debug)]
44pub struct ResolvedModule {
45 pub file_id: FileId,
47 pub path: PathBuf,
49 pub exports: Vec<fallow_types::extract::ExportInfo>,
51 pub re_exports: Vec<ResolvedReExport>,
53 pub resolved_imports: Vec<ResolvedImport>,
55 pub resolved_dynamic_imports: Vec<ResolvedImport>,
57 pub resolved_dynamic_patterns: Vec<(fallow_types::extract::DynamicImportPattern, Vec<FileId>)>,
59 pub member_accesses: Vec<fallow_types::extract::MemberAccess>,
61 pub whole_object_uses: Vec<String>,
63 pub has_cjs_exports: bool,
65 pub unused_import_bindings: FxHashSet<String>,
67 pub type_referenced_import_bindings: Vec<String>,
69 pub value_referenced_import_bindings: Vec<String>,
71}
72
73impl Default for ResolvedModule {
74 fn default() -> Self {
75 Self {
76 file_id: FileId(0),
77 path: PathBuf::new(),
78 exports: vec![],
79 re_exports: vec![],
80 resolved_imports: vec![],
81 resolved_dynamic_imports: vec![],
82 resolved_dynamic_patterns: vec![],
83 member_accesses: vec![],
84 whole_object_uses: vec![],
85 has_cjs_exports: false,
86 unused_import_bindings: FxHashSet::default(),
87 type_referenced_import_bindings: vec![],
88 value_referenced_import_bindings: vec![],
89 }
90 }
91}
92
93pub(super) struct ResolveContext<'a> {
98 pub resolver: &'a Resolver,
100 pub path_to_id: &'a FxHashMap<&'a Path, FileId>,
102 pub raw_path_to_id: &'a FxHashMap<&'a Path, FileId>,
104 pub workspace_roots: &'a FxHashMap<&'a str, &'a Path>,
106 pub path_aliases: &'a [(String, String)],
108 pub scss_include_paths: &'a [PathBuf],
112 pub root: &'a Path,
114 pub canonical_fallback: Option<&'a CanonicalFallback<'a>>,
118 pub tsconfig_warned: &'a Mutex<FxHashSet<String>>,
123}
124
125pub(super) struct CanonicalFallback<'a> {
127 files: &'a [fallow_types::discover::DiscoveredFile],
128 map: std::sync::OnceLock<FxHashMap<std::path::PathBuf, FileId>>,
129}
130
131impl<'a> CanonicalFallback<'a> {
132 pub const fn new(files: &'a [fallow_types::discover::DiscoveredFile]) -> Self {
133 Self {
134 files,
135 map: std::sync::OnceLock::new(),
136 }
137 }
138
139 pub fn get(&self, canonical: &Path) -> Option<FileId> {
141 let map = self.map.get_or_init(|| {
142 tracing::debug!(
143 "intra-project symlinks detected — building canonical path index ({} files)",
144 self.files.len()
145 );
146 self.files
147 .iter()
148 .filter_map(|f| {
149 dunce::canonicalize(&f.path)
150 .ok()
151 .map(|canonical| (canonical, f.id))
152 })
153 .collect()
154 });
155 map.get(canonical).copied()
156 }
157}
158
159#[cfg(all(test, not(miri)))]
160mod tests {
161 use super::*;
162 use fallow_types::discover::DiscoveredFile;
163
164 #[test]
165 fn canonical_fallback_returns_none_for_empty_files() {
166 let files: Vec<DiscoveredFile> = vec![];
167 let fallback = CanonicalFallback::new(&files);
168 assert!(fallback.get(Path::new("/nonexistent")).is_none());
169 }
170
171 #[test]
172 fn canonical_fallback_finds_existing_file() {
173 let temp = std::env::temp_dir().join("fallow-test-canonical-fallback");
174 let _ = std::fs::create_dir_all(&temp);
175 let test_file = temp.join("test.ts");
176 std::fs::write(&test_file, "").unwrap();
177
178 let files = vec![DiscoveredFile {
179 id: FileId(42),
180 path: test_file.clone(),
181 size_bytes: 0,
182 }];
183 let fallback = CanonicalFallback::new(&files);
184
185 let canonical = dunce::canonicalize(&test_file).unwrap();
186 assert_eq!(fallback.get(&canonical), Some(FileId(42)));
187
188 assert_eq!(fallback.get(&canonical), Some(FileId(42)));
190
191 let _ = std::fs::remove_dir_all(&temp);
192 }
193
194 #[test]
195 fn canonical_fallback_returns_none_for_missing_path() {
196 let temp = std::env::temp_dir().join("fallow-test-canonical-miss");
197 let _ = std::fs::create_dir_all(&temp);
198 let test_file = temp.join("exists.ts");
199 std::fs::write(&test_file, "").unwrap();
200
201 let files = vec![DiscoveredFile {
202 id: FileId(1),
203 path: test_file,
204 size_bytes: 0,
205 }];
206 let fallback = CanonicalFallback::new(&files);
207 assert!(fallback.get(Path::new("/nonexistent/file.ts")).is_none());
208
209 let _ = std::fs::remove_dir_all(&temp);
210 }
211}
212
213pub const OUTPUT_DIRS: &[&str] = &["dist", "build", "out", "esm", "cjs"];
218
219pub const SOURCE_EXTS: &[&str] = &["ts", "tsx", "mts", "cts", "js", "jsx", "mjs", "cjs"];
221
222pub const RN_PLATFORM_PREFIXES: &[&str] = &[".web", ".ios", ".android", ".native"];