use rustc_hash::FxHashMap;
use fallow_config::ResolvedConfig;
use fallow_types::extract::{ImportedName, ModuleInfo};
use fallow_types::results::BoundaryCallViolation;
use crate::discover::FileId;
use crate::graph::ModuleGraph;
use crate::suppress::{IssueKind, SuppressionContext};
use super::security::CalleePattern;
use super::{LineOffsetsMap, byte_offset_to_line_col};
pub fn find_boundary_call_violations(
graph: &ModuleGraph,
modules: &[ModuleInfo],
config: &ResolvedConfig,
suppressions: &SuppressionContext<'_>,
line_offsets_by_file: &LineOffsetsMap<'_>,
) -> Vec<BoundaryCallViolation> {
let forbidden = &config.boundaries.calls_forbidden_by_zone;
if forbidden.is_empty() {
return Vec::new();
}
let patterns_by_zone: FxHashMap<&str, Vec<CalleePattern>> = forbidden
.iter()
.map(|(zone, patterns)| {
(
zone.as_str(),
patterns
.iter()
.filter_map(|raw| CalleePattern::parse(raw))
.collect(),
)
})
.collect();
let modules_by_id: FxHashMap<FileId, &ModuleInfo> =
modules.iter().map(|m| (m.file_id, m)).collect();
let mut zone_file_counts: FxHashMap<&str, usize> =
patterns_by_zone.keys().map(|zone| (*zone, 0)).collect();
let mut violations = Vec::new();
for node in &graph.modules {
if !node.is_reachable() && !node.is_entry_point() {
continue;
}
let Ok(relative) = node.path.strip_prefix(&config.root) else {
continue;
};
let relative = relative.to_string_lossy().replace('\\', "/");
let Some(zone) = config.boundaries.classify_zone(&relative) else {
continue;
};
if let Some(count) = zone_file_counts.get_mut(zone) {
*count += 1;
}
let Some(patterns) = patterns_by_zone.get(zone) else {
continue;
};
let Some(module) = modules_by_id.get(&node.file_id) else {
continue;
};
if suppressions.is_file_suppressed(node.file_id, IssueKind::BoundaryViolation) {
continue;
}
for callee_use in &module.callee_uses {
let Some(pattern) = first_matching_pattern(patterns, &callee_use.callee_path, module)
else {
continue;
};
let (line, col) =
byte_offset_to_line_col(line_offsets_by_file, node.file_id, callee_use.span_start);
if suppressions.is_suppressed(node.file_id, line, IssueKind::BoundaryViolation) {
continue;
}
violations.push(BoundaryCallViolation {
path: node.path.clone(),
line,
col,
zone: zone.to_owned(),
callee: callee_use.callee_path.clone(),
pattern: pattern.raw().to_owned(),
});
}
}
for (zone, count) in &zone_file_counts {
if *count == 0 {
tracing::warn!(
"boundaries.calls.forbidden references zone '{zone}', but no analyzed file \
classified into that zone; forbidden-call rules only apply to files matched \
by a zone's patterns"
);
}
}
violations
}
fn first_matching_pattern<'p>(
patterns: &'p [CalleePattern],
callee_path: &str,
module: &ModuleInfo,
) -> Option<&'p CalleePattern> {
if let Some(pattern) = patterns.iter().find(|p| p.matches(callee_path)) {
return Some(pattern);
}
let canonical = canonical_callee_path(module, callee_path)?;
patterns.iter().find(|p| p.matches(&canonical))
}
pub(super) fn canonical_callee_path(module: &ModuleInfo, callee_path: &str) -> Option<String> {
let (head, rest) = match callee_path.split_once('.') {
Some((head, rest)) => (head, Some(rest)),
None => (callee_path, None),
};
let import = module
.imports
.iter()
.find(|import| import.local_name == head)?;
if import.is_type_only {
return None;
}
let source = import.source.as_str();
if source.starts_with('.') || source.starts_with('/') || source.starts_with('#') {
return None;
}
let source = source.strip_prefix("node:").unwrap_or(source);
match &import.imported_name {
ImportedName::Named(imported) => Some(match rest {
Some(rest) => format!("{source}.{imported}.{rest}"),
None => format!("{source}.{imported}"),
}),
ImportedName::Namespace | ImportedName::Default => Some(match rest {
Some(rest) => format!("{source}.{rest}"),
None => source.to_owned(),
}),
ImportedName::SideEffect => None,
}
}
#[cfg(test)]
mod tests;