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