cargo/diagnostics/rules/
unused_workspace_dependencies.rs1use std::path::Path;
2
3use cargo_util_schemas::manifest::InheritableDependency;
4use cargo_util_terminal::report::AnnotationKind;
5use cargo_util_terminal::report::Group;
6use cargo_util_terminal::report::Level;
7use cargo_util_terminal::report::Origin;
8use cargo_util_terminal::report::Patch;
9use cargo_util_terminal::report::Snippet;
10use indexmap::IndexSet;
11use tracing::instrument;
12
13use super::SUSPICIOUS;
14use crate::CargoResult;
15use crate::GlobalContext;
16use crate::core::MaybePackage;
17use crate::core::Workspace;
18use crate::diagnostics::DiagnosticStats;
19use crate::diagnostics::Lint;
20use crate::diagnostics::LintLevelProduct;
21use crate::diagnostics::get_key_value_span;
22use crate::diagnostics::rel_cwd_manifest_path;
23
24pub static LINT: &Lint = &Lint {
25 name: "unused_workspace_dependencies",
26 desc: "unused workspace dependency",
27 primary_group: &SUSPICIOUS,
28 msrv: Some(super::CARGO_LINTS_MSRV),
29 feature_gate: None,
30 docs: Some(
31 r#"
32### What it does
33Checks for any entry in `[workspace.dependencies]` that has not been inherited
34
35### Why it is bad
36They can give the false impression that these dependencies are used
37
38### Example
39```toml
40[workspace.dependencies]
41regex = "1"
42
43[dependencies]
44```
45"#,
46 ),
47};
48
49#[instrument(skip_all)]
50pub(crate) fn lint_workspace(
51 ws: &Workspace<'_>,
52 maybe_pkg: &MaybePackage,
53 manifest_path: &Path,
54 level: LintLevelProduct,
55 stats: &mut DiagnosticStats,
56 gctx: &GlobalContext,
57) -> CargoResult<()> {
58 let LintLevelProduct {
59 level: lint_level,
60 source,
61 } = level;
62
63 let workspace_deps: IndexSet<_> = maybe_pkg
64 .original_toml()
65 .and_then(|t| t.workspace.as_ref())
66 .and_then(|w| w.dependencies.as_ref())
67 .iter()
68 .flat_map(|d| d.keys())
69 .collect();
70
71 let mut inherited_deps = IndexSet::new();
72 for member in ws.members() {
73 let Some(original_toml) = member.manifest().original_toml() else {
74 return Ok(());
75 };
76 inherited_deps.extend(
77 original_toml
78 .build_dependencies()
79 .into_iter()
80 .flatten()
81 .filter(|(_, d)| is_inherited(d))
82 .map(|(name, _)| name),
83 );
84 inherited_deps.extend(
85 original_toml
86 .dependencies
87 .iter()
88 .flatten()
89 .filter(|(_, d)| is_inherited(d))
90 .map(|(name, _)| name),
91 );
92 inherited_deps.extend(
93 original_toml
94 .dev_dependencies()
95 .into_iter()
96 .flatten()
97 .filter(|(_, d)| is_inherited(d))
98 .map(|(name, _)| name),
99 );
100 for target in original_toml.target.iter().flat_map(|t| t.values()) {
101 inherited_deps.extend(
102 target
103 .build_dependencies()
104 .into_iter()
105 .flatten()
106 .filter(|(_, d)| is_inherited(d))
107 .map(|(name, _)| name),
108 );
109 inherited_deps.extend(
110 target
111 .dependencies
112 .iter()
113 .flatten()
114 .filter(|(_, d)| is_inherited(d))
115 .map(|(name, _)| name),
116 );
117 inherited_deps.extend(
118 target
119 .dev_dependencies()
120 .into_iter()
121 .flatten()
122 .filter(|(_, d)| is_inherited(d))
123 .map(|(name, _)| name),
124 );
125 }
126 }
127
128 for (i, unused) in workspace_deps.difference(&inherited_deps).enumerate() {
129 let document = maybe_pkg.document();
130 let contents = maybe_pkg.contents();
131 let level = lint_level.to_diagnostic_level();
132 let manifest_path = rel_cwd_manifest_path(manifest_path, gctx);
133 let emitted_source = LINT.emitted_source(lint_level, source);
134
135 let mut primary = Group::with_title(level.primary_title(LINT.desc));
136 if let Some(document) = document
137 && let Some(contents) = contents
138 {
139 let mut snippet = Snippet::source(contents).path(&manifest_path);
140 if let Some(span) =
141 get_key_value_span(document, &["workspace", "dependencies", unused.as_str()])
142 {
143 snippet = snippet.annotation(AnnotationKind::Primary.span(span.key));
144 }
145 primary = primary.element(snippet);
146 } else {
147 primary = primary.element(Origin::path(&manifest_path));
148 }
149 if i == 0 {
150 primary = primary.element(Level::NOTE.message(emitted_source));
151 }
152 let mut report = vec![primary];
153 if let Some(document) = document
154 && let Some(contents) = contents
155 {
156 let mut help = Group::with_title(
157 Level::HELP.secondary_title("consider removing the unused dependency"),
158 );
159 let mut snippet = Snippet::source(contents).path(&manifest_path);
160 if let Some(span) =
161 get_key_value_span(document, &["workspace", "dependencies", unused.as_str()])
162 {
163 snippet = snippet.patch(Patch::new(span.key.start..span.value.end, ""));
164 }
165 help = help.element(snippet);
166 report.push(help);
167 }
168
169 stats.record_lint(lint_level);
170 gctx.shell().print_report(&report, lint_level.force())?;
171 }
172
173 Ok(())
174}
175
176fn is_inherited(dep: &InheritableDependency) -> bool {
177 matches!(dep, InheritableDependency::Inherit(_))
178}