-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Auto merge of #13322 - RuairidhWilliamson:anon-trait-import, r=y21
Unused trait imports (formerly anonymous trait import) For #11969 I'm looking for help and feedback on implementing a new lint for suggesting `use ... as _` for traits where possible. I have had a go at implementing this but I don't know if this is the best way to do it as I am new to clippy. There are some edge cases I can think of where this doesn't work but have aired on the side of false negatives instead of false positives. An example of a false negative. I couldn't figure out the best way to resolve an import from within clippy. The sub module imports MyAny so that cannot be anonymized but `use std::any::Any` could be. In this case it is not caught because `Any` and `MyAny` have the same DefId. ```rust mod nested_mod_used_bad1 { use std::any::Any; use std::any::Any as MyAny; mod foo { use crate::nested_mod_used_bad1::MyAny; fn foo() { println!("{:?}", MyAny::type_id("foo")); } } } ``` Any feedback is much appreciated. ------- changelog: new lint: `unused_trait_names`
- Loading branch information
Showing
11 changed files
with
747 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,94 @@ | ||
use clippy_config::msrvs::{self, Msrv}; | ||
use clippy_config::Conf; | ||
use clippy_utils::diagnostics::span_lint_and_sugg; | ||
use clippy_utils::is_from_proc_macro; | ||
use clippy_utils::source::snippet_opt; | ||
use rustc_errors::Applicability; | ||
use rustc_hir::def::{DefKind, Res}; | ||
use rustc_hir::{Item, ItemKind, UseKind}; | ||
use rustc_lint::{LateContext, LateLintPass, LintContext as _}; | ||
use rustc_middle::lint::in_external_macro; | ||
use rustc_middle::ty::Visibility; | ||
use rustc_session::impl_lint_pass; | ||
use rustc_span::symbol::kw; | ||
|
||
declare_clippy_lint! { | ||
/// ### What it does | ||
/// Checks for `use Trait` where the Trait is only used for its methods and not referenced by a path directly. | ||
/// | ||
/// ### Why is this bad? | ||
/// Traits imported that aren't used directly can be imported anonymously with `use Trait as _`. | ||
/// It is more explicit, avoids polluting the current scope with unused names and can be useful to show which imports are required for traits. | ||
/// | ||
/// ### Example | ||
/// ```no_run | ||
/// use std::fmt::Write; | ||
/// | ||
/// fn main() { | ||
/// let mut s = String::new(); | ||
/// let _ = write!(s, "hello, world!"); | ||
/// println!("{s}"); | ||
/// } | ||
/// ``` | ||
/// Use instead: | ||
/// ```no_run | ||
/// use std::fmt::Write as _; | ||
/// | ||
/// fn main() { | ||
/// let mut s = String::new(); | ||
/// let _ = write!(s, "hello, world!"); | ||
/// println!("{s}"); | ||
/// } | ||
/// ``` | ||
#[clippy::version = "1.83.0"] | ||
pub UNUSED_TRAIT_NAMES, | ||
restriction, | ||
"use items that import a trait but only use it anonymously" | ||
} | ||
|
||
pub struct UnusedTraitNames { | ||
msrv: Msrv, | ||
} | ||
|
||
impl UnusedTraitNames { | ||
pub fn new(conf: &'static Conf) -> Self { | ||
Self { | ||
msrv: conf.msrv.clone(), | ||
} | ||
} | ||
} | ||
|
||
impl_lint_pass!(UnusedTraitNames => [UNUSED_TRAIT_NAMES]); | ||
|
||
impl<'tcx> LateLintPass<'tcx> for UnusedTraitNames { | ||
fn check_item(&mut self, cx: &LateContext<'tcx>, item: &'tcx Item<'tcx>) { | ||
if self.msrv.meets(msrvs::UNDERSCORE_IMPORTS) | ||
&& !in_external_macro(cx.sess(), item.span) | ||
&& let ItemKind::Use(path, UseKind::Single) = item.kind | ||
// Ignore imports that already use Underscore | ||
&& item.ident.name != kw::Underscore | ||
// Only check traits | ||
&& let Some(Res::Def(DefKind::Trait, _)) = path.res.first() | ||
&& cx.tcx.maybe_unused_trait_imports(()).contains(&item.owner_id.def_id) | ||
// Only check this import if it is visible to its module only (no pub, pub(crate), ...) | ||
&& let module = cx.tcx.parent_module_from_def_id(item.owner_id.def_id) | ||
&& cx.tcx.visibility(item.owner_id.def_id) == Visibility::Restricted(module.to_def_id()) | ||
&& let Some(last_segment) = path.segments.last() | ||
&& let Some(snip) = snippet_opt(cx, last_segment.ident.span) | ||
&& !is_from_proc_macro(cx, &last_segment.ident) | ||
{ | ||
let complete_span = last_segment.ident.span.to(item.ident.span); | ||
span_lint_and_sugg( | ||
cx, | ||
UNUSED_TRAIT_NAMES, | ||
complete_span, | ||
"importing trait that is only used anonymously", | ||
"use", | ||
format!("{snip} as _"), | ||
Applicability::MachineApplicable, | ||
); | ||
} | ||
} | ||
|
||
extract_msrv_attr!(LateContext); | ||
} |
Oops, something went wrong.