use rustc_hash::{FxHashMap, FxHashSet};
use crate::discover::FileId;
use crate::extract::{ANGULAR_TPL_SENTINEL, MemberKind};
use crate::graph::ModuleGraph;
use crate::resolve::{ResolveResult, ResolvedModule};
use crate::results::UnusedMember;
use crate::suppress::{IssueKind, SuppressionContext};
use super::predicates::{is_angular_lifecycle_method, is_react_lifecycle_method};
use super::{LineOffsetsMap, byte_offset_to_line_col};
#[expect(
clippy::too_many_lines,
reason = "member tracking requires many graph traversal steps; split candidate for sig-audit-loop"
)]
pub fn find_unused_members(
graph: &ModuleGraph,
resolved_modules: &[ResolvedModule],
suppressions: &SuppressionContext<'_>,
line_offsets_by_file: &LineOffsetsMap<'_>,
user_class_member_allowlist: &FxHashSet<&str>,
) -> (Vec<UnusedMember>, Vec<UnusedMember>) {
let mut unused_enum_members = Vec::new();
let mut unused_class_members = Vec::new();
let mut accessed_members: FxHashMap<String, FxHashSet<String>> = FxHashMap::default();
let mut self_accessed_members: FxHashMap<crate::discover::FileId, FxHashSet<String>> =
FxHashMap::default();
let mut whole_object_used_exports: FxHashSet<String> = FxHashSet::default();
for resolved in resolved_modules {
let local_to_imported: FxHashMap<&str, &str> = resolved
.resolved_imports
.iter()
.filter_map(|imp| match &imp.info.imported_name {
crate::extract::ImportedName::Named(name) => {
Some((imp.info.local_name.as_str(), name.as_str()))
}
crate::extract::ImportedName::Default => {
Some((imp.info.local_name.as_str(), "default"))
}
_ => None,
})
.collect();
for access in &resolved.member_accesses {
if access.object == "this" {
self_accessed_members
.entry(resolved.file_id)
.or_default()
.insert(access.member.clone());
continue;
}
let export_name = local_to_imported
.get(access.object.as_str())
.copied()
.unwrap_or(access.object.as_str());
accessed_members
.entry(export_name.to_string())
.or_default()
.insert(access.member.clone());
}
for local_name in &resolved.whole_object_uses {
let export_name = local_to_imported
.get(local_name.as_str())
.copied()
.unwrap_or(local_name.as_str());
whole_object_used_exports.insert(export_name.to_string());
}
}
let mut parent_to_children: FxHashMap<(String, FileId), Vec<String>> = FxHashMap::default();
for resolved in resolved_modules {
let local_to_imported: FxHashMap<&str, (&str, FileId)> = resolved
.resolved_imports
.iter()
.filter_map(|imp| {
let imported_name = match &imp.info.imported_name {
crate::extract::ImportedName::Named(name) => name.as_str(),
crate::extract::ImportedName::Default => "default",
_ => return None,
};
if let ResolveResult::InternalModule(file_id) = &imp.target {
Some((imp.info.local_name.as_str(), (imported_name, *file_id)))
} else {
None
}
})
.collect();
for export in &resolved.exports {
if let Some(super_local) = &export.super_class {
let child_name = export.name.to_string();
if let Some(&(parent_name, parent_file_id)) =
local_to_imported.get(super_local.as_str())
{
parent_to_children
.entry((parent_name.to_string(), parent_file_id))
.or_default()
.push(child_name);
} else {
parent_to_children
.entry((super_local.clone(), resolved.file_id))
.or_default()
.push(child_name);
}
}
}
}
if !parent_to_children.is_empty() {
let export_name_to_file: FxHashMap<String, FileId> = graph
.modules
.iter()
.flat_map(|m| {
m.exports
.iter()
.map(move |e| (e.name.to_string(), m.file_id))
})
.collect();
let mut propagations: Vec<(FileId, Vec<String>)> = Vec::new();
for ((parent_name, parent_fid), children) in &parent_to_children {
if let Some(parent_self_accesses) = self_accessed_members.get(parent_fid) {
let accesses: Vec<String> = parent_self_accesses.iter().cloned().collect();
for child_name in children {
if let Some(&child_fid) = export_name_to_file.get(child_name.as_str()) {
propagations.push((child_fid, accesses.clone()));
}
}
}
let parent_accesses: Option<FxHashSet<String>> =
accessed_members.get(parent_name.as_str()).cloned();
let mut child_accesses_to_propagate: FxHashSet<String> = FxHashSet::default();
for child_name in children {
if let Some(child_accesses) = accessed_members.get(child_name.as_str()) {
child_accesses_to_propagate.extend(child_accesses.iter().cloned());
}
}
if let Some(ref parent_acc) = parent_accesses {
for child_name in children {
accessed_members
.entry(child_name.clone())
.or_default()
.extend(parent_acc.iter().cloned());
}
}
if !child_accesses_to_propagate.is_empty() {
accessed_members
.entry(parent_name.clone())
.or_default()
.extend(child_accesses_to_propagate);
}
}
for (file_id, members) in propagations {
let entry = self_accessed_members.entry(file_id).or_default();
for member in members {
entry.insert(member);
}
}
}
let angular_tpl_refs: FxHashMap<FileId, Vec<&str>> = resolved_modules
.iter()
.filter_map(|m| {
let refs: Vec<&str> = m
.member_accesses
.iter()
.filter(|a| a.object == ANGULAR_TPL_SENTINEL)
.map(|a| a.member.as_str())
.collect();
if refs.is_empty() {
None
} else {
Some((m.file_id, refs))
}
})
.collect();
if !angular_tpl_refs.is_empty() {
for resolved in resolved_modules {
if let Some(refs) = angular_tpl_refs.get(&resolved.file_id) {
let entry = self_accessed_members.entry(resolved.file_id).or_default();
for &ref_name in refs {
entry.insert(ref_name.to_string());
}
}
for import in &resolved.resolved_imports {
if let ResolveResult::InternalModule(target_id) = &import.target
&& let Some(refs) = angular_tpl_refs.get(target_id)
{
let entry = self_accessed_members.entry(resolved.file_id).or_default();
for &ref_name in refs {
entry.insert(ref_name.to_string());
}
}
}
}
}
for module in &graph.modules {
if !module.is_reachable() || module.is_entry_point() {
continue;
}
for export in &module.exports {
if export.members.is_empty() {
continue;
}
if export.references.is_empty() && !graph.has_namespace_import(module.file_id) {
continue;
}
let export_name = export.name.to_string();
if whole_object_used_exports.contains(&export_name) {
continue;
}
let file_self_accesses = self_accessed_members.get(&module.file_id);
for member in &export.members {
if matches!(member.kind, MemberKind::NamespaceMember) {
continue;
}
if accessed_members
.get(&export_name)
.is_some_and(|s| s.contains(&member.name))
{
continue;
}
if matches!(
member.kind,
MemberKind::ClassMethod | MemberKind::ClassProperty
) && file_self_accesses.is_some_and(|accesses| accesses.contains(&member.name))
{
continue;
}
if member.has_decorator {
continue;
}
if matches!(
member.kind,
MemberKind::ClassMethod | MemberKind::ClassProperty
) && (is_react_lifecycle_method(&member.name)
|| is_angular_lifecycle_method(&member.name)
|| user_class_member_allowlist.contains(member.name.as_str()))
{
continue;
}
let (line, col) = byte_offset_to_line_col(
line_offsets_by_file,
module.file_id,
member.span.start,
);
let issue_kind = match member.kind {
MemberKind::EnumMember => IssueKind::UnusedEnumMember,
MemberKind::ClassMethod | MemberKind::ClassProperty => {
IssueKind::UnusedClassMember
}
MemberKind::NamespaceMember => unreachable!(),
};
if suppressions.is_suppressed(module.file_id, line, issue_kind) {
continue;
}
let unused = UnusedMember {
path: module.path.clone(),
parent_name: export_name.clone(),
member_name: member.name.clone(),
kind: member.kind,
line,
col,
};
match member.kind {
MemberKind::EnumMember => unused_enum_members.push(unused),
MemberKind::ClassMethod | MemberKind::ClassProperty => {
unused_class_members.push(unused);
}
MemberKind::NamespaceMember => unreachable!(),
}
}
}
}
(unused_enum_members, unused_class_members)
}
#[cfg(test)]
mod tests {
use super::*;
use crate::discover::{DiscoveredFile, EntryPoint, EntryPointSource, FileId};
use crate::extract::{
ExportName, ImportInfo, ImportedName, MemberAccess, MemberInfo, MemberKind, VisibilityTag,
};
use crate::graph::{ExportSymbol, ModuleGraph, SymbolReference};
use crate::resolve::{ResolveResult, ResolvedImport, ResolvedModule};
use oxc_span::Span;
use std::path::PathBuf;
#[expect(
clippy::cast_possible_truncation,
reason = "test file counts are trivially small"
)]
fn build_graph(file_specs: &[(&str, bool)]) -> ModuleGraph {
let files: Vec<DiscoveredFile> = file_specs
.iter()
.enumerate()
.map(|(i, (path, _))| DiscoveredFile {
id: FileId(i as u32),
path: PathBuf::from(path),
size_bytes: 0,
})
.collect();
let entry_points: Vec<EntryPoint> = file_specs
.iter()
.filter(|(_, is_entry)| *is_entry)
.map(|(path, _)| EntryPoint {
path: PathBuf::from(path),
source: EntryPointSource::ManualEntry,
})
.collect();
let resolved_modules: Vec<ResolvedModule> = files
.iter()
.map(|f| ResolvedModule {
file_id: f.id,
path: f.path.clone(),
..Default::default()
})
.collect();
ModuleGraph::build(&resolved_modules, &entry_points, &files)
}
fn make_member(name: &str, kind: MemberKind) -> MemberInfo {
MemberInfo {
name: name.to_string(),
kind,
span: Span::new(10, 20),
has_decorator: false,
}
}
fn make_export_with_members(
name: &str,
members: Vec<MemberInfo>,
ref_from: Option<u32>,
) -> ExportSymbol {
let references = ref_from
.map(|from| {
vec![SymbolReference {
from_file: FileId(from),
kind: crate::graph::ReferenceKind::NamedImport,
import_span: Span::new(0, 10),
}]
})
.unwrap_or_default();
ExportSymbol {
name: ExportName::Named(name.to_string()),
is_type_only: false,
visibility: VisibilityTag::None,
span: Span::new(0, 10),
references,
members,
}
}
#[test]
fn unused_members_empty_graph() {
let graph = build_graph(&[]);
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(enum_members.is_empty());
assert!(class_members.is_empty());
}
#[test]
fn unused_enum_member_detected() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![
make_member("Active", MemberKind::EnumMember),
make_member("Inactive", MemberKind::EnumMember),
],
Some(0), )];
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(enum_members.len(), 2);
assert!(class_members.is_empty());
let names: FxHashSet<&str> = enum_members
.iter()
.map(|m| m.member_name.as_str())
.collect();
assert!(names.contains("Active"));
assert!(names.contains("Inactive"));
}
#[test]
fn accessed_enum_member_not_flagged() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![
make_member("Active", MemberKind::EnumMember),
make_member("Inactive", MemberKind::EnumMember),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(0),
path: PathBuf::from("/src/entry.ts"),
resolved_imports: vec![ResolvedImport {
info: ImportInfo {
source: "./enums".to_string(),
imported_name: ImportedName::Named("Status".to_string()),
local_name: "Status".to_string(),
is_type_only: false,
span: Span::new(0, 30),
source_span: Span::default(),
},
target: ResolveResult::InternalModule(FileId(1)),
}],
member_accesses: vec![MemberAccess {
object: "Status".to_string(),
member: "Active".to_string(),
}],
..Default::default()
}];
let (enum_members, _) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(enum_members.len(), 1);
assert_eq!(enum_members[0].member_name, "Inactive");
}
#[test]
fn whole_object_use_skips_all_members() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![
make_member("Active", MemberKind::EnumMember),
make_member("Inactive", MemberKind::EnumMember),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(0),
path: PathBuf::from("/src/entry.ts"),
resolved_imports: vec![ResolvedImport {
info: ImportInfo {
source: "./enums".to_string(),
imported_name: ImportedName::Named("Status".to_string()),
local_name: "Status".to_string(),
is_type_only: false,
span: Span::new(0, 30),
source_span: Span::default(),
},
target: ResolveResult::InternalModule(FileId(1)),
}],
whole_object_uses: vec!["Status".to_string()],
..Default::default()
}];
let (enum_members, class_members) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(enum_members.is_empty());
assert!(class_members.is_empty());
}
#[test]
fn decorated_class_member_not_flagged() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/entity.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"User",
vec![MemberInfo {
name: "name".to_string(),
kind: MemberKind::ClassProperty,
span: Span::new(10, 20),
has_decorator: true, }],
Some(0),
)];
let (_, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(class_members.is_empty());
}
#[test]
fn react_lifecycle_method_not_flagged() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/component.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"MyComponent",
vec![
make_member("render", MemberKind::ClassMethod),
make_member("componentDidMount", MemberKind::ClassMethod),
make_member("customMethod", MemberKind::ClassMethod),
],
Some(0),
)];
let (_, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(class_members.len(), 1);
assert_eq!(class_members[0].member_name, "customMethod");
}
#[test]
fn angular_lifecycle_method_not_flagged() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/component.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"AppComponent",
vec![
make_member("ngOnInit", MemberKind::ClassMethod),
make_member("ngOnDestroy", MemberKind::ClassMethod),
make_member("myHelper", MemberKind::ClassMethod),
],
Some(0),
)];
let (_, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(class_members.len(), 1);
assert_eq!(class_members[0].member_name, "myHelper");
}
#[test]
fn user_class_member_allowlist_not_flagged() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/renderer.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"MyRendererComponent",
vec![
make_member("agInit", MemberKind::ClassMethod),
make_member("refresh", MemberKind::ClassMethod),
make_member("customHelper", MemberKind::ClassMethod),
],
Some(0),
)];
let mut allowlist: FxHashSet<&str> = FxHashSet::default();
allowlist.insert("agInit");
allowlist.insert("refresh");
let (_, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&allowlist,
);
assert_eq!(
class_members.len(),
1,
"only customHelper should remain unused"
);
assert_eq!(class_members[0].member_name, "customHelper");
}
#[test]
fn user_class_member_allowlist_does_not_affect_enums() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/status.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![make_member("refresh", MemberKind::EnumMember)],
Some(0),
)];
let mut allowlist: FxHashSet<&str> = FxHashSet::default();
allowlist.insert("refresh");
let (enum_members, _) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&allowlist,
);
assert_eq!(enum_members.len(), 1);
assert_eq!(enum_members[0].member_name, "refresh");
}
#[test]
fn this_member_access_not_flagged() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/service.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Service",
vec![
make_member("label", MemberKind::ClassProperty),
make_member("unused_prop", MemberKind::ClassProperty),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(1), path: PathBuf::from("/src/service.ts"),
member_accesses: vec![MemberAccess {
object: "this".to_string(),
member: "label".to_string(),
}],
..Default::default()
}];
let (_, class_members) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(class_members.len(), 1);
assert_eq!(class_members[0].member_name, "unused_prop");
}
#[test]
fn unreferenced_export_skips_member_analysis() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![make_member("Active", MemberKind::EnumMember)],
None, )];
let (enum_members, _) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(enum_members.is_empty());
}
#[test]
fn unreachable_module_skips_member_analysis() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/dead.ts", false)]);
graph.modules[1].exports = vec![make_export_with_members(
"DeadEnum",
vec![make_member("X", MemberKind::EnumMember)],
Some(0),
)];
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(enum_members.is_empty());
assert!(class_members.is_empty());
}
#[test]
fn entry_point_module_skips_member_analysis() {
let mut graph = build_graph(&[("/src/entry.ts", true)]);
graph.modules[0].exports = vec![make_export_with_members(
"EntryEnum",
vec![make_member("X", MemberKind::EnumMember)],
None,
)];
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(enum_members.is_empty());
assert!(class_members.is_empty());
}
#[test]
fn enum_member_kind_routed_to_enum_results() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![make_member("Active", MemberKind::EnumMember)],
Some(0),
)];
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(enum_members.len(), 1);
assert_eq!(enum_members[0].kind, MemberKind::EnumMember);
assert!(class_members.is_empty());
}
#[test]
fn class_member_kind_routed_to_class_results() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/class.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"MyClass",
vec![
make_member("myMethod", MemberKind::ClassMethod),
make_member("myProp", MemberKind::ClassProperty),
],
Some(0),
)];
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(enum_members.is_empty());
assert_eq!(class_members.len(), 2);
assert!(
class_members
.iter()
.any(|m| m.kind == MemberKind::ClassMethod)
);
assert!(
class_members
.iter()
.any(|m| m.kind == MemberKind::ClassProperty)
);
}
#[test]
fn instance_member_access_not_flagged() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/service.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"MyService",
vec![
make_member("greet", MemberKind::ClassMethod),
make_member("unusedMethod", MemberKind::ClassMethod),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(0),
path: PathBuf::from("/src/entry.ts"),
resolved_imports: vec![ResolvedImport {
info: ImportInfo {
source: "./service".to_string(),
imported_name: ImportedName::Named("MyService".to_string()),
local_name: "MyService".to_string(),
is_type_only: false,
span: Span::new(0, 30),
source_span: Span::default(),
},
target: ResolveResult::InternalModule(FileId(1)),
}],
member_accesses: vec![MemberAccess {
object: "MyService".to_string(),
member: "greet".to_string(),
}],
..Default::default()
}];
let (_, class_members) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(class_members.len(), 1);
assert_eq!(class_members[0].member_name, "unusedMethod");
}
#[test]
fn this_access_does_not_skip_enum_members() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Direction",
vec![
make_member("Up", MemberKind::EnumMember),
make_member("Down", MemberKind::EnumMember),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(1),
path: PathBuf::from("/src/enums.ts"),
member_accesses: vec![MemberAccess {
object: "this".to_string(),
member: "Up".to_string(),
}],
..Default::default()
}];
let (enum_members, _) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(enum_members.len(), 2);
}
#[test]
fn mixed_enum_and_class_in_same_module() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/mixed.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![
make_export_with_members(
"Status",
vec![make_member("Active", MemberKind::EnumMember)],
Some(0),
),
make_export_with_members(
"Service",
vec![make_member("doWork", MemberKind::ClassMethod)],
Some(0),
),
];
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(enum_members.len(), 1);
assert_eq!(enum_members[0].parent_name, "Status");
assert_eq!(class_members.len(), 1);
assert_eq!(class_members[0].parent_name, "Service");
}
#[test]
fn local_name_mapped_to_imported_name() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![
make_member("Active", MemberKind::EnumMember),
make_member("Inactive", MemberKind::EnumMember),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(0),
path: PathBuf::from("/src/entry.ts"),
resolved_imports: vec![ResolvedImport {
info: ImportInfo {
source: "./enums".to_string(),
imported_name: ImportedName::Named("Status".to_string()),
local_name: "S".to_string(), is_type_only: false,
span: Span::new(0, 30),
source_span: Span::default(),
},
target: ResolveResult::InternalModule(FileId(1)),
}],
member_accesses: vec![MemberAccess {
object: "S".to_string(), member: "Active".to_string(),
}],
..Default::default()
}];
let (enum_members, _) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(enum_members.len(), 1);
assert_eq!(enum_members[0].member_name, "Inactive");
}
#[test]
fn default_import_maps_to_default_export() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"default",
vec![
make_member("X", MemberKind::EnumMember),
make_member("Y", MemberKind::EnumMember),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(0),
path: PathBuf::from("/src/entry.ts"),
resolved_imports: vec![ResolvedImport {
info: ImportInfo {
source: "./enums".to_string(),
imported_name: ImportedName::Default,
local_name: "MyEnum".to_string(),
is_type_only: false,
span: Span::new(0, 30),
source_span: Span::default(),
},
target: ResolveResult::InternalModule(FileId(1)),
}],
member_accesses: vec![MemberAccess {
object: "MyEnum".to_string(),
member: "X".to_string(),
}],
..Default::default()
}];
let (enum_members, _) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(enum_members.len(), 1);
assert_eq!(enum_members[0].member_name, "Y");
}
#[test]
fn suppressed_enum_member_not_flagged() {
use crate::suppress::{IssueKind, Suppression};
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![make_member("Active", MemberKind::EnumMember)],
Some(0),
)];
let supps = vec![Suppression {
line: 1,
comment_line: 0,
kind: Some(IssueKind::UnusedEnumMember),
}];
let mut supp_map: FxHashMap<FileId, &[Suppression]> = FxHashMap::default();
supp_map.insert(FileId(1), &supps);
let suppressions = SuppressionContext::from_map(supp_map);
let (enum_members, _) = find_unused_members(
&graph,
&[],
&suppressions,
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(
enum_members.is_empty(),
"suppressed enum member should not be flagged"
);
}
#[test]
fn suppressed_class_member_not_flagged() {
use crate::suppress::{IssueKind, Suppression};
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/service.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Service",
vec![make_member("doWork", MemberKind::ClassMethod)],
Some(0),
)];
let supps = vec![Suppression {
line: 1,
comment_line: 0,
kind: Some(IssueKind::UnusedClassMember),
}];
let mut supp_map: FxHashMap<FileId, &[Suppression]> = FxHashMap::default();
supp_map.insert(FileId(1), &supps);
let suppressions = SuppressionContext::from_map(supp_map);
let (_, class_members) = find_unused_members(
&graph,
&[],
&suppressions,
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(
class_members.is_empty(),
"suppressed class member should not be flagged"
);
}
#[test]
fn whole_object_use_via_aliased_import() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/enums.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"Status",
vec![
make_member("A", MemberKind::EnumMember),
make_member("B", MemberKind::EnumMember),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(0),
path: PathBuf::from("/src/entry.ts"),
resolved_imports: vec![ResolvedImport {
info: ImportInfo {
source: "./enums".to_string(),
imported_name: ImportedName::Named("Status".to_string()),
local_name: "S".to_string(),
is_type_only: false,
span: Span::new(0, 30),
source_span: Span::default(),
},
target: ResolveResult::InternalModule(FileId(1)),
}],
whole_object_uses: vec!["S".to_string()], ..Default::default()
}];
let (enum_members, _) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(
enum_members.is_empty(),
"whole object use via alias should suppress all members"
);
}
#[test]
fn this_field_chained_access_not_flagged() {
let mut graph = build_graph(&[("/src/main.ts", true), ("/src/service.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"MyService",
vec![
make_member("doWork", MemberKind::ClassMethod),
make_member("unusedMethod", MemberKind::ClassMethod),
],
Some(0),
)];
let resolved_modules = vec![ResolvedModule {
file_id: FileId(0),
path: PathBuf::from("/src/main.ts"),
resolved_imports: vec![ResolvedImport {
info: ImportInfo {
source: "./service".to_string(),
imported_name: ImportedName::Named("MyService".to_string()),
local_name: "MyService".to_string(),
is_type_only: false,
span: Span::new(0, 30),
source_span: Span::default(),
},
target: ResolveResult::InternalModule(FileId(1)),
}],
member_accesses: vec![MemberAccess {
object: "MyService".to_string(),
member: "doWork".to_string(),
}],
..Default::default()
}];
let (_, class_members) = find_unused_members(
&graph,
&resolved_modules,
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert_eq!(class_members.len(), 1);
assert_eq!(class_members[0].member_name, "unusedMethod");
}
#[test]
fn export_with_no_members_skipped() {
let mut graph = build_graph(&[("/src/entry.ts", true), ("/src/utils.ts", false)]);
graph.modules[1].set_reachable(true);
graph.modules[1].exports = vec![make_export_with_members(
"helper",
vec![], Some(0),
)];
let (enum_members, class_members) = find_unused_members(
&graph,
&[],
&SuppressionContext::empty(),
&FxHashMap::default(),
&FxHashSet::default(),
);
assert!(enum_members.is_empty());
assert!(class_members.is_empty());
}
}