2021-03-25 19:29:11 +01:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
2021-12-06 12:33:31 +01:00
|
|
|
use clippy_utils::is_test_module_or_function;
|
2021-03-25 19:29:11 +01:00
|
|
|
use clippy_utils::source::{snippet, snippet_with_applicability};
|
2020-01-07 17:53:56 +01:00
|
|
|
use rustc_errors::Applicability;
|
2023-07-03 17:42:48 +00:00
|
|
|
use rustc_hir::def::{DefKind, Res};
|
|
|
|
use rustc_hir::{Item, ItemKind, PathSegment, UseKind};
|
2023-05-20 15:39:26 +02:00
|
|
|
use rustc_lint::{LateContext, LateLintPass, LintContext};
|
2022-02-13 11:30:48 +01:00
|
|
|
use rustc_middle::ty;
|
2020-05-11 20:23:47 +02:00
|
|
|
use rustc_session::{declare_tool_lint, impl_lint_pass};
|
2021-01-15 10:56:44 +01:00
|
|
|
use rustc_span::symbol::kw;
|
|
|
|
use rustc_span::{sym, BytePos};
|
2020-01-07 17:53:56 +01:00
|
|
|
|
2020-01-14 13:52:08 +01:00
|
|
|
declare_clippy_lint! {
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### What it does
|
|
|
|
/// Checks for `use Enum::*`.
|
2020-01-14 13:52:08 +01:00
|
|
|
///
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// It is usually better style to use the prefixed name of
|
2020-01-14 13:52:08 +01:00
|
|
|
/// an enumeration variant, rather than importing variants.
|
|
|
|
///
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### Known problems
|
|
|
|
/// Old-style enumerations that prefix the variants are
|
2020-01-14 13:52:08 +01:00
|
|
|
/// still around.
|
|
|
|
///
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### Example
|
2023-10-23 13:49:18 +00:00
|
|
|
/// ```no_run
|
2020-01-14 13:52:08 +01:00
|
|
|
/// use std::cmp::Ordering::*;
|
2022-06-16 17:39:06 +02:00
|
|
|
///
|
|
|
|
/// # fn foo(_: std::cmp::Ordering) {}
|
2020-06-09 14:36:01 +00:00
|
|
|
/// foo(Less);
|
2022-06-16 17:39:06 +02:00
|
|
|
/// ```
|
2020-06-09 14:36:01 +00:00
|
|
|
///
|
2022-06-16 17:39:06 +02:00
|
|
|
/// Use instead:
|
2023-10-23 13:49:18 +00:00
|
|
|
/// ```no_run
|
2020-06-09 14:36:01 +00:00
|
|
|
/// use std::cmp::Ordering;
|
2022-06-16 17:39:06 +02:00
|
|
|
///
|
|
|
|
/// # fn foo(_: Ordering) {}
|
2020-06-09 14:36:01 +00:00
|
|
|
/// foo(Ordering::Less)
|
2020-01-14 13:52:08 +01:00
|
|
|
/// ```
|
2021-12-06 12:33:31 +01:00
|
|
|
#[clippy::version = "pre 1.29.0"]
|
2020-01-14 13:52:08 +01:00
|
|
|
pub ENUM_GLOB_USE,
|
|
|
|
pedantic,
|
|
|
|
"use items that import all variants of an enum"
|
|
|
|
}
|
|
|
|
|
2020-01-07 17:53:56 +01:00
|
|
|
declare_clippy_lint! {
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### What it does
|
|
|
|
/// Checks for wildcard imports `use _::*`.
|
2020-01-07 17:53:56 +01:00
|
|
|
///
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// wildcard imports can pollute the namespace. This is especially bad if
|
2020-01-07 17:53:56 +01:00
|
|
|
/// you try to import something through a wildcard, that already has been imported by name from
|
|
|
|
/// a different source:
|
|
|
|
///
|
|
|
|
/// ```rust,ignore
|
|
|
|
/// use crate1::foo; // Imports a function named foo
|
|
|
|
/// use crate2::*; // Has a function named foo
|
|
|
|
///
|
|
|
|
/// foo(); // Calls crate1::foo
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// This can lead to confusing error messages at best and to unexpected behavior at worst.
|
|
|
|
///
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### Exceptions
|
2023-06-02 11:41:57 +02:00
|
|
|
/// Wildcard imports are allowed from modules that their name contains `prelude`. Many crates
|
|
|
|
/// (including the standard library) provide modules named "prelude" specifically designed
|
|
|
|
/// for wildcard import.
|
2020-05-11 20:23:47 +02:00
|
|
|
///
|
|
|
|
/// `use super::*` is allowed in test modules. This is defined as any module with "test" in the name.
|
|
|
|
///
|
|
|
|
/// These exceptions can be disabled using the `warn-on-all-wildcard-imports` configuration flag.
|
2020-03-13 17:07:11 +00:00
|
|
|
///
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### Known problems
|
|
|
|
/// If macros are imported through the wildcard, this macro is not included
|
2020-01-07 17:53:56 +01:00
|
|
|
/// by the suggestion and has to be added by hand.
|
|
|
|
///
|
2020-02-21 10:15:38 +01:00
|
|
|
/// Applying the suggestion when explicit imports of the things imported with a glob import
|
|
|
|
/// exist, may result in `unused_imports` warnings.
|
|
|
|
///
|
2021-07-29 12:16:06 +02:00
|
|
|
/// ### Example
|
2020-01-07 17:53:56 +01:00
|
|
|
/// ```rust,ignore
|
|
|
|
/// use crate1::*;
|
|
|
|
///
|
|
|
|
/// foo();
|
|
|
|
/// ```
|
|
|
|
///
|
2022-06-16 17:39:06 +02:00
|
|
|
/// Use instead:
|
2020-01-07 17:53:56 +01:00
|
|
|
/// ```rust,ignore
|
|
|
|
/// use crate1::foo;
|
|
|
|
///
|
|
|
|
/// foo();
|
|
|
|
/// ```
|
2021-12-06 12:33:31 +01:00
|
|
|
#[clippy::version = "1.43.0"]
|
2020-01-07 17:53:56 +01:00
|
|
|
pub WILDCARD_IMPORTS,
|
|
|
|
pedantic,
|
|
|
|
"lint `use _::*` statements"
|
|
|
|
}
|
|
|
|
|
2020-05-11 20:23:47 +02:00
|
|
|
#[derive(Default)]
|
|
|
|
pub struct WildcardImports {
|
|
|
|
warn_on_all: bool,
|
|
|
|
test_modules_deep: u32,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl WildcardImports {
|
|
|
|
pub fn new(warn_on_all: bool) -> Self {
|
|
|
|
Self {
|
|
|
|
warn_on_all,
|
|
|
|
test_modules_deep: 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl_lint_pass!(WildcardImports => [ENUM_GLOB_USE, WILDCARD_IMPORTS]);
|
2020-01-07 17:53:56 +01:00
|
|
|
|
2020-06-25 23:41:36 +03:00
|
|
|
impl LateLintPass<'_> for WildcardImports {
|
|
|
|
fn check_item(&mut self, cx: &LateContext<'_>, item: &Item<'_>) {
|
2023-05-20 15:39:26 +02:00
|
|
|
if cx.sess().is_test_crate() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-07-01 18:17:38 +02:00
|
|
|
if is_test_module_or_function(cx.tcx, item) {
|
2020-05-11 20:23:47 +02:00
|
|
|
self.test_modules_deep = self.test_modules_deep.saturating_add(1);
|
|
|
|
}
|
2022-10-27 14:02:18 +11:00
|
|
|
let module = cx.tcx.parent_module_from_def_id(item.owner_id.def_id);
|
|
|
|
if cx.tcx.visibility(item.owner_id.def_id) != ty::Visibility::Restricted(module.to_def_id()) {
|
2020-01-07 17:53:56 +01:00
|
|
|
return;
|
|
|
|
}
|
2023-11-10 17:29:28 +00:00
|
|
|
if let ItemKind::Use(use_path, UseKind::Glob) = &item.kind
|
|
|
|
&& (self.warn_on_all || !self.check_exceptions(item, use_path.segments))
|
|
|
|
&& let used_imports = cx.tcx.names_imported_by_glob_use(item.owner_id.def_id)
|
|
|
|
&& !used_imports.is_empty() // Already handled by `unused_imports`
|
|
|
|
&& !used_imports.contains(&kw::Underscore)
|
|
|
|
{
|
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
|
|
|
let import_source_snippet = snippet_with_applicability(cx, use_path.span, "..", &mut applicability);
|
|
|
|
let (span, braced_glob) = if import_source_snippet.is_empty() {
|
|
|
|
// This is a `_::{_, *}` import
|
|
|
|
// In this case `use_path.span` is empty and ends directly in front of the `*`,
|
|
|
|
// so we need to extend it by one byte.
|
2023-11-02 19:23:36 +00:00
|
|
|
(use_path.span.with_hi(use_path.span.hi() + BytePos(1)), true)
|
2023-11-10 17:29:28 +00:00
|
|
|
} else {
|
|
|
|
// In this case, the `use_path.span` ends right before the `::*`, so we need to
|
|
|
|
// extend it up to the `*`. Since it is hard to find the `*` in weird
|
|
|
|
// formattings like `use _ :: *;`, we extend it up to, but not including the
|
|
|
|
// `;`. In nested imports, like `use _::{inner::*, _}` there is no `;` and we
|
|
|
|
// can just use the end of the item span
|
|
|
|
let mut span = use_path.span.with_hi(item.span.hi());
|
|
|
|
if snippet(cx, span, "").ends_with(';') {
|
|
|
|
span = use_path.span.with_hi(item.span.hi() - BytePos(1));
|
|
|
|
}
|
2023-11-02 19:23:36 +00:00
|
|
|
(span, false)
|
2023-11-10 17:29:28 +00:00
|
|
|
};
|
2020-01-07 17:53:56 +01:00
|
|
|
|
2023-11-10 17:29:28 +00:00
|
|
|
let mut imports = used_imports.items().map(ToString::to_string).into_sorted_stable_ord();
|
|
|
|
let imports_string = if imports.len() == 1 {
|
|
|
|
imports.pop().unwrap()
|
|
|
|
} else if braced_glob {
|
|
|
|
imports.join(", ")
|
|
|
|
} else {
|
|
|
|
format!("{{{}}}", imports.join(", "))
|
|
|
|
};
|
2020-01-07 17:53:56 +01:00
|
|
|
|
2023-11-10 17:29:28 +00:00
|
|
|
let sugg = if braced_glob {
|
|
|
|
imports_string
|
|
|
|
} else {
|
|
|
|
format!("{import_source_snippet}::{imports_string}")
|
|
|
|
};
|
2020-01-07 17:53:56 +01:00
|
|
|
|
2023-11-10 17:29:28 +00:00
|
|
|
// Glob imports always have a single resolution.
|
|
|
|
let (lint, message) = if let Res::Def(DefKind::Enum, _) = use_path.res[0] {
|
|
|
|
(ENUM_GLOB_USE, "usage of wildcard import for enum variants")
|
|
|
|
} else {
|
|
|
|
(WILDCARD_IMPORTS, "usage of wildcard import")
|
|
|
|
};
|
2020-01-14 13:52:08 +01:00
|
|
|
|
2023-11-02 19:23:36 +00:00
|
|
|
span_lint_and_sugg(cx, lint, span, message, "try", sugg, applicability);
|
2020-01-09 18:12:54 +01:00
|
|
|
}
|
2020-01-07 17:53:56 +01:00
|
|
|
}
|
2020-05-11 20:23:47 +02:00
|
|
|
|
2021-07-01 18:17:38 +02:00
|
|
|
fn check_item_post(&mut self, cx: &LateContext<'_>, item: &Item<'_>) {
|
|
|
|
if is_test_module_or_function(cx.tcx, item) {
|
2020-05-11 20:23:47 +02:00
|
|
|
self.test_modules_deep = self.test_modules_deep.saturating_sub(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl WildcardImports {
|
|
|
|
fn check_exceptions(&self, item: &Item<'_>, segments: &[PathSegment<'_>]) -> bool {
|
2021-12-06 12:33:31 +01:00
|
|
|
item.span.from_expansion()
|
2020-05-11 20:23:47 +02:00
|
|
|
|| is_prelude_import(segments)
|
|
|
|
|| (is_super_only_import(segments) && self.test_modules_deep > 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-28 16:10:16 +02:00
|
|
|
// Allow "...prelude::..::*" imports.
|
2020-05-11 20:23:47 +02:00
|
|
|
// Many crates have a prelude, and it is imported as a glob by design.
|
|
|
|
fn is_prelude_import(segments: &[PathSegment<'_>]) -> bool {
|
2023-06-02 11:41:57 +02:00
|
|
|
segments
|
|
|
|
.iter()
|
|
|
|
.any(|ps| ps.ident.name.as_str().contains(sym::prelude.as_str()))
|
2020-05-11 20:23:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Allow "super::*" imports in tests.
|
|
|
|
fn is_super_only_import(segments: &[PathSegment<'_>]) -> bool {
|
2021-01-15 10:56:44 +01:00
|
|
|
segments.len() == 1 && segments[0].ident.name == kw::Super
|
2020-05-11 20:23:47 +02:00
|
|
|
}
|