fallow_graph/resolve/
mod.rs1mod dynamic_imports;
19pub(crate) mod fallbacks;
20mod path_info;
21mod re_exports;
22mod react_native;
23mod require_imports;
24mod specifier;
25mod static_imports;
26#[cfg(test)]
27mod tests;
28mod types;
29mod upgrades;
30
31pub use path_info::{extract_package_name, is_bare_specifier, is_path_alias};
32pub use types::{ResolveResult, ResolvedImport, ResolvedModule, ResolvedReExport};
33
34use std::path::{Path, PathBuf};
35use std::sync::Mutex;
36
37use rayon::prelude::*;
38use rustc_hash::{FxHashMap, FxHashSet};
39
40use fallow_types::discover::{DiscoveredFile, FileId};
41use fallow_types::extract::ModuleInfo;
42
43use dynamic_imports::{resolve_dynamic_imports, resolve_dynamic_patterns};
44use re_exports::resolve_re_exports;
45use require_imports::resolve_require_imports;
46use specifier::create_resolver;
47use static_imports::resolve_static_imports;
48use types::ResolveContext;
49use upgrades::apply_specifier_upgrades;
50
51#[must_use]
53#[expect(
54 clippy::too_many_arguments,
55 reason = "resolver inputs come from disjoint sources (config, plugins, workspace, filesystem); \
56 bundling them into a struct would be a cross-cutting refactor outside this task"
57)]
58pub fn resolve_all_imports(
59 modules: &[ModuleInfo],
60 files: &[DiscoveredFile],
61 workspaces: &[fallow_config::WorkspaceInfo],
62 active_plugins: &[String],
63 path_aliases: &[(String, String)],
64 scss_include_paths: &[PathBuf],
65 root: &Path,
66 extra_conditions: &[String],
67) -> Vec<ResolvedModule> {
68 let canonical_ws_roots: Vec<PathBuf> = workspaces
73 .par_iter()
74 .map(|ws| dunce::canonicalize(&ws.root).unwrap_or_else(|_| ws.root.clone()))
75 .collect();
76 let workspace_roots: FxHashMap<&str, &Path> = workspaces
77 .iter()
78 .zip(canonical_ws_roots.iter())
79 .map(|(ws, canonical)| (ws.name.as_str(), canonical.as_path()))
80 .collect();
81
82 let root_is_canonical = dunce::canonicalize(root).is_ok_and(|c| c == root);
87
88 let canonical_paths: Vec<PathBuf> = if root_is_canonical {
91 Vec::new()
92 } else {
93 files
94 .par_iter()
95 .map(|f| dunce::canonicalize(&f.path).unwrap_or_else(|_| f.path.clone()))
96 .collect()
97 };
98
99 let path_to_id: FxHashMap<&Path, FileId> = if root_is_canonical {
102 files.iter().map(|f| (f.path.as_path(), f.id)).collect()
103 } else {
104 canonical_paths
105 .iter()
106 .enumerate()
107 .map(|(idx, canonical)| (canonical.as_path(), files[idx].id))
108 .collect()
109 };
110
111 let raw_path_to_id: FxHashMap<&Path, FileId> =
113 files.iter().map(|f| (f.path.as_path(), f.id)).collect();
114
115 let file_paths: Vec<&Path> = files.iter().map(|f| f.path.as_path()).collect();
117
118 let resolver = create_resolver(active_plugins, extra_conditions);
120
121 let canonical_fallback = if root_is_canonical {
124 Some(types::CanonicalFallback::new(files))
125 } else {
126 None
127 };
128
129 let tsconfig_warned: Mutex<FxHashSet<String>> = Mutex::new(FxHashSet::default());
131
132 let ctx = ResolveContext {
134 resolver: &resolver,
135 path_to_id: &path_to_id,
136 raw_path_to_id: &raw_path_to_id,
137 workspace_roots: &workspace_roots,
138 path_aliases,
139 scss_include_paths,
140 root,
141 canonical_fallback: canonical_fallback.as_ref(),
142 tsconfig_warned: &tsconfig_warned,
143 };
144
145 let mut resolved: Vec<ResolvedModule> = modules
150 .par_iter()
151 .filter_map(|module| {
152 let Some(file_path) = file_paths.get(module.file_id.0 as usize) else {
153 tracing::warn!(
154 file_id = module.file_id.0,
155 "Skipping module with unknown file_id during resolution"
156 );
157 return None;
158 };
159
160 let mut all_imports = resolve_static_imports(&ctx, file_path, &module.imports);
161 all_imports.extend(resolve_require_imports(
162 &ctx,
163 file_path,
164 &module.require_calls,
165 ));
166
167 let from_dir = if canonical_paths.is_empty() {
168 file_path.parent().unwrap_or(file_path)
170 } else {
171 canonical_paths
172 .get(module.file_id.0 as usize)
173 .and_then(|p| p.parent())
174 .unwrap_or(file_path)
175 };
176
177 Some(ResolvedModule {
178 file_id: module.file_id,
179 path: file_path.to_path_buf(),
180 exports: module.exports.clone(),
181 re_exports: resolve_re_exports(&ctx, file_path, &module.re_exports),
182 resolved_imports: all_imports,
183 resolved_dynamic_imports: resolve_dynamic_imports(
184 &ctx,
185 file_path,
186 &module.dynamic_imports,
187 ),
188 resolved_dynamic_patterns: resolve_dynamic_patterns(
189 from_dir,
190 &module.dynamic_import_patterns,
191 &canonical_paths,
192 files,
193 ),
194 member_accesses: module.member_accesses.clone(),
195 whole_object_uses: module.whole_object_uses.clone(),
196 has_cjs_exports: module.has_cjs_exports,
197 unused_import_bindings: module.unused_import_bindings.iter().cloned().collect(),
198 })
199 })
200 .collect();
201
202 apply_specifier_upgrades(&mut resolved);
203
204 resolved
205}