fallow_graph/resolve/
types.rs1use 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}
68
69impl Default for ResolvedModule {
70 fn default() -> Self {
71 Self {
72 file_id: FileId(0),
73 path: PathBuf::new(),
74 exports: vec![],
75 re_exports: vec![],
76 resolved_imports: vec![],
77 resolved_dynamic_imports: vec![],
78 resolved_dynamic_patterns: vec![],
79 member_accesses: vec![],
80 whole_object_uses: vec![],
81 has_cjs_exports: false,
82 unused_import_bindings: FxHashSet::default(),
83 }
84 }
85}
86
87pub(super) struct ResolveContext<'a> {
92 pub resolver: &'a Resolver,
94 pub path_to_id: &'a FxHashMap<&'a Path, FileId>,
96 pub raw_path_to_id: &'a FxHashMap<&'a Path, FileId>,
98 pub workspace_roots: &'a FxHashMap<&'a str, &'a Path>,
100 pub path_aliases: &'a [(String, String)],
102 pub scss_include_paths: &'a [PathBuf],
106 pub root: &'a Path,
108 pub canonical_fallback: Option<&'a CanonicalFallback<'a>>,
112 pub tsconfig_warned: &'a Mutex<FxHashSet<String>>,
117}
118
119pub(super) struct CanonicalFallback<'a> {
121 files: &'a [fallow_types::discover::DiscoveredFile],
122 map: std::sync::OnceLock<FxHashMap<std::path::PathBuf, FileId>>,
123}
124
125impl<'a> CanonicalFallback<'a> {
126 pub const fn new(files: &'a [fallow_types::discover::DiscoveredFile]) -> Self {
127 Self {
128 files,
129 map: std::sync::OnceLock::new(),
130 }
131 }
132
133 pub fn get(&self, canonical: &Path) -> Option<FileId> {
135 let map = self.map.get_or_init(|| {
136 tracing::debug!(
137 "intra-project symlinks detected — building canonical path index ({} files)",
138 self.files.len()
139 );
140 self.files
141 .iter()
142 .filter_map(|f| {
143 dunce::canonicalize(&f.path)
144 .ok()
145 .map(|canonical| (canonical, f.id))
146 })
147 .collect()
148 });
149 map.get(canonical).copied()
150 }
151}
152
153#[cfg(all(test, not(miri)))]
154mod tests {
155 use super::*;
156 use fallow_types::discover::DiscoveredFile;
157
158 #[test]
159 fn canonical_fallback_returns_none_for_empty_files() {
160 let files: Vec<DiscoveredFile> = vec![];
161 let fallback = CanonicalFallback::new(&files);
162 assert!(fallback.get(Path::new("/nonexistent")).is_none());
163 }
164
165 #[test]
166 fn canonical_fallback_finds_existing_file() {
167 let temp = std::env::temp_dir().join("fallow-test-canonical-fallback");
168 let _ = std::fs::create_dir_all(&temp);
169 let test_file = temp.join("test.ts");
170 std::fs::write(&test_file, "").unwrap();
171
172 let files = vec![DiscoveredFile {
173 id: FileId(42),
174 path: test_file.clone(),
175 size_bytes: 0,
176 }];
177 let fallback = CanonicalFallback::new(&files);
178
179 let canonical = dunce::canonicalize(&test_file).unwrap();
180 assert_eq!(fallback.get(&canonical), Some(FileId(42)));
181
182 assert_eq!(fallback.get(&canonical), Some(FileId(42)));
184
185 let _ = std::fs::remove_dir_all(&temp);
186 }
187
188 #[test]
189 fn canonical_fallback_returns_none_for_missing_path() {
190 let temp = std::env::temp_dir().join("fallow-test-canonical-miss");
191 let _ = std::fs::create_dir_all(&temp);
192 let test_file = temp.join("exists.ts");
193 std::fs::write(&test_file, "").unwrap();
194
195 let files = vec![DiscoveredFile {
196 id: FileId(1),
197 path: test_file,
198 size_bytes: 0,
199 }];
200 let fallback = CanonicalFallback::new(&files);
201 assert!(fallback.get(Path::new("/nonexistent/file.ts")).is_none());
202
203 let _ = std::fs::remove_dir_all(&temp);
204 }
205}
206
207pub const OUTPUT_DIRS: &[&str] = &["dist", "build", "out", "esm", "cjs"];
212
213pub const SOURCE_EXTS: &[&str] = &["ts", "tsx", "mts", "cts", "js", "jsx", "mjs", "cjs"];
215
216pub const RN_PLATFORM_PREFIXES: &[&str] = &[".web", ".ios", ".android", ".native"];