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