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 style_resolver: &'a Resolver,
104 pub path_to_id: &'a FxHashMap<&'a Path, FileId>,
106 pub raw_path_to_id: &'a FxHashMap<&'a Path, FileId>,
108 pub workspace_roots: &'a FxHashMap<&'a str, &'a Path>,
110 pub path_aliases: &'a [(String, String)],
112 pub scss_include_paths: &'a [PathBuf],
116 pub root: &'a Path,
118 pub canonical_fallback: Option<&'a CanonicalFallback<'a>>,
122 pub tsconfig_warned: &'a Mutex<FxHashSet<String>>,
127}
128
129pub(super) struct CanonicalFallback<'a> {
131 files: &'a [fallow_types::discover::DiscoveredFile],
132 map: std::sync::OnceLock<FxHashMap<std::path::PathBuf, FileId>>,
133}
134
135impl<'a> CanonicalFallback<'a> {
136 pub const fn new(files: &'a [fallow_types::discover::DiscoveredFile]) -> Self {
137 Self {
138 files,
139 map: std::sync::OnceLock::new(),
140 }
141 }
142
143 pub fn get(&self, canonical: &Path) -> Option<FileId> {
145 let map = self.map.get_or_init(|| {
146 tracing::debug!(
147 "intra-project symlinks detected — building canonical path index ({} files)",
148 self.files.len()
149 );
150 self.files
151 .iter()
152 .filter_map(|f| {
153 dunce::canonicalize(&f.path)
154 .ok()
155 .map(|canonical| (canonical, f.id))
156 })
157 .collect()
158 });
159 map.get(canonical).copied()
160 }
161}
162
163#[cfg(all(test, not(miri)))]
164mod tests {
165 use super::*;
166 use fallow_types::discover::DiscoveredFile;
167
168 #[test]
169 fn canonical_fallback_returns_none_for_empty_files() {
170 let files: Vec<DiscoveredFile> = vec![];
171 let fallback = CanonicalFallback::new(&files);
172 assert!(fallback.get(Path::new("/nonexistent")).is_none());
173 }
174
175 #[test]
176 fn canonical_fallback_finds_existing_file() {
177 let temp = std::env::temp_dir().join("fallow-test-canonical-fallback");
178 let _ = std::fs::create_dir_all(&temp);
179 let test_file = temp.join("test.ts");
180 std::fs::write(&test_file, "").unwrap();
181
182 let files = vec![DiscoveredFile {
183 id: FileId(42),
184 path: test_file.clone(),
185 size_bytes: 0,
186 }];
187 let fallback = CanonicalFallback::new(&files);
188
189 let canonical = dunce::canonicalize(&test_file).unwrap();
190 assert_eq!(fallback.get(&canonical), Some(FileId(42)));
191
192 assert_eq!(fallback.get(&canonical), Some(FileId(42)));
194
195 let _ = std::fs::remove_dir_all(&temp);
196 }
197
198 #[test]
199 fn canonical_fallback_returns_none_for_missing_path() {
200 let temp = std::env::temp_dir().join("fallow-test-canonical-miss");
201 let _ = std::fs::create_dir_all(&temp);
202 let test_file = temp.join("exists.ts");
203 std::fs::write(&test_file, "").unwrap();
204
205 let files = vec![DiscoveredFile {
206 id: FileId(1),
207 path: test_file,
208 size_bytes: 0,
209 }];
210 let fallback = CanonicalFallback::new(&files);
211 assert!(fallback.get(Path::new("/nonexistent/file.ts")).is_none());
212
213 let _ = std::fs::remove_dir_all(&temp);
214 }
215}
216
217pub const OUTPUT_DIRS: &[&str] = &["dist", "build", "out", "esm", "cjs"];
222
223pub const SOURCE_EXTS: &[&str] = &["ts", "tsx", "mts", "cts", "js", "jsx", "mjs", "cjs"];
225
226pub const RN_PLATFORM_PREFIXES: &[&str] = &[".web", ".ios", ".android", ".native"];