2021-03-16 00:55:45 +00:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
2021-10-29 16:14:22 +00:00
|
|
|
use clippy_utils::is_test_module_or_function;
|
2021-03-14 23:17:44 +00:00
|
|
|
use clippy_utils::source::{snippet, snippet_with_applicability};
|
2020-01-07 16:53:56 +00:00
|
|
|
use if_chain::if_chain;
|
|
|
|
use rustc_errors::Applicability;
|
2020-01-14 12:52:08 +00:00
|
|
|
use rustc_hir::{
|
|
|
|
def::{DefKind, Res},
|
2020-05-11 18:23:47 +00:00
|
|
|
Item, ItemKind, PathSegment, UseKind,
|
2020-01-14 12:52:08 +00:00
|
|
|
};
|
2020-01-07 16:53:56 +00:00
|
|
|
use rustc_lint::{LateContext, LateLintPass};
|
2020-05-11 18:23:47 +00:00
|
|
|
use rustc_session::{declare_tool_lint, impl_lint_pass};
|
2021-01-15 09:56:44 +00:00
|
|
|
use rustc_span::symbol::kw;
|
|
|
|
use rustc_span::{sym, BytePos};
|
2020-01-07 16:53:56 +00:00
|
|
|
|
2020-01-14 12:52:08 +00:00
|
|
|
declare_clippy_lint! {
|
2021-07-02 18:37:11 +00:00
|
|
|
/// ### What it does
|
|
|
|
/// Checks for `use Enum::*`.
|
2020-01-14 12:52:08 +00:00
|
|
|
///
|
2021-07-02 18:37:11 +00:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// It is usually better style to use the prefixed name of
|
2020-01-14 12:52:08 +00:00
|
|
|
/// an enumeration variant, rather than importing variants.
|
|
|
|
///
|
2021-07-02 18:37:11 +00:00
|
|
|
/// ### Known problems
|
|
|
|
/// Old-style enumerations that prefix the variants are
|
2020-01-14 12:52:08 +00:00
|
|
|
/// still around.
|
|
|
|
///
|
2021-07-02 18:37:11 +00:00
|
|
|
/// ### Example
|
2020-06-09 14:36:01 +00:00
|
|
|
/// ```rust,ignore
|
|
|
|
/// // Bad
|
2020-01-14 12:52:08 +00:00
|
|
|
/// use std::cmp::Ordering::*;
|
2020-06-09 14:36:01 +00:00
|
|
|
/// foo(Less);
|
|
|
|
///
|
|
|
|
/// // Good
|
|
|
|
/// use std::cmp::Ordering;
|
|
|
|
/// foo(Ordering::Less)
|
2020-01-14 12:52:08 +00:00
|
|
|
/// ```
|
Added `clippy::version` attribute to all normal lints
So, some context for this, well, more a story. I'm not used to scripting, I've never really scripted anything, even if it's a valuable skill. I just never really needed it. Now, `@flip1995` correctly suggested using a script for this in `rust-clippy#7813`...
And I decided to write a script using nushell because why not? This was a mistake... I spend way more time on this than I would like to admit. It has definitely been more than 4 hours. It shouldn't take that long, but me being new to scripting and nushell just wasn't a good mixture... Anyway, here is the script that creates another script which adds the versions. Fun...
Just execute this on the `gh-pages` branch and the resulting `replacer.sh` in `clippy_lints` and it should all work.
```nu
mv v0.0.212 rust-1.00.0;
mv beta rust-1.57.0;
mv master rust-1.58.0;
let paths = (open ./rust-1.58.0/lints.json | select id id_span | flatten | select id path);
let versions = (
ls | where name =~ "rust-" | select name | format {name}/lints.json |
each { open $it | select id | insert version $it | str substring "5,11" version} |
group-by id | rotate counter-clockwise id version |
update version {get version | first 1} | flatten | select id version);
$paths | each { |row|
let version = ($versions | where id == ($row.id) | format {version})
let idu = ($row.id | str upcase)
$"sed -i '0,/($idu),/{s/pub ($idu),/#[clippy::version = "($version)"]\n pub ($idu),/}' ($row.path)"
} | str collect ";" | str find-replace --all '1.00.0' 'pre 1.29.0' | save "replacer.sh";
```
And this still has some problems, but at this point I just want to be done -.-
2021-10-21 19:06:26 +00:00
|
|
|
#[clippy::version = "pre 1.29.0"]
|
2020-01-14 12:52:08 +00:00
|
|
|
pub ENUM_GLOB_USE,
|
|
|
|
pedantic,
|
|
|
|
"use items that import all variants of an enum"
|
|
|
|
}
|
|
|
|
|
2020-01-07 16:53:56 +00:00
|
|
|
declare_clippy_lint! {
|
2021-07-02 18:37:11 +00:00
|
|
|
/// ### What it does
|
|
|
|
/// Checks for wildcard imports `use _::*`.
|
2020-01-07 16:53:56 +00:00
|
|
|
///
|
2021-07-02 18:37:11 +00:00
|
|
|
/// ### Why is this bad?
|
|
|
|
/// wildcard imports can pollute the namespace. This is especially bad if
|
2020-01-07 16:53:56 +00: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-02 18:37:11 +00:00
|
|
|
/// ### Exceptions
|
2020-05-11 18:23:47 +00:00
|
|
|
/// Wildcard imports are allowed from modules named `prelude`. Many crates (including the standard library)
|
|
|
|
/// provide modules named "prelude" specifically designed for wildcard import.
|
|
|
|
///
|
|
|
|
/// `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-02 18:37:11 +00:00
|
|
|
/// ### Known problems
|
|
|
|
/// If macros are imported through the wildcard, this macro is not included
|
2020-01-07 16:53:56 +00:00
|
|
|
/// by the suggestion and has to be added by hand.
|
|
|
|
///
|
2020-02-21 09:15:38 +00:00
|
|
|
/// Applying the suggestion when explicit imports of the things imported with a glob import
|
|
|
|
/// exist, may result in `unused_imports` warnings.
|
|
|
|
///
|
2021-07-02 18:37:11 +00:00
|
|
|
/// ### Example
|
2020-01-07 16:53:56 +00:00
|
|
|
/// ```rust,ignore
|
2020-06-09 14:36:01 +00:00
|
|
|
/// // Bad
|
2020-01-07 16:53:56 +00:00
|
|
|
/// use crate1::*;
|
|
|
|
///
|
|
|
|
/// foo();
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// ```rust,ignore
|
2020-06-09 14:36:01 +00:00
|
|
|
/// // Good
|
2020-01-07 16:53:56 +00:00
|
|
|
/// use crate1::foo;
|
|
|
|
///
|
|
|
|
/// foo();
|
|
|
|
/// ```
|
Added `clippy::version` attribute to all normal lints
So, some context for this, well, more a story. I'm not used to scripting, I've never really scripted anything, even if it's a valuable skill. I just never really needed it. Now, `@flip1995` correctly suggested using a script for this in `rust-clippy#7813`...
And I decided to write a script using nushell because why not? This was a mistake... I spend way more time on this than I would like to admit. It has definitely been more than 4 hours. It shouldn't take that long, but me being new to scripting and nushell just wasn't a good mixture... Anyway, here is the script that creates another script which adds the versions. Fun...
Just execute this on the `gh-pages` branch and the resulting `replacer.sh` in `clippy_lints` and it should all work.
```nu
mv v0.0.212 rust-1.00.0;
mv beta rust-1.57.0;
mv master rust-1.58.0;
let paths = (open ./rust-1.58.0/lints.json | select id id_span | flatten | select id path);
let versions = (
ls | where name =~ "rust-" | select name | format {name}/lints.json |
each { open $it | select id | insert version $it | str substring "5,11" version} |
group-by id | rotate counter-clockwise id version |
update version {get version | first 1} | flatten | select id version);
$paths | each { |row|
let version = ($versions | where id == ($row.id) | format {version})
let idu = ($row.id | str upcase)
$"sed -i '0,/($idu),/{s/pub ($idu),/#[clippy::version = "($version)"]\n pub ($idu),/}' ($row.path)"
} | str collect ";" | str find-replace --all '1.00.0' 'pre 1.29.0' | save "replacer.sh";
```
And this still has some problems, but at this point I just want to be done -.-
2021-10-21 19:06:26 +00:00
|
|
|
#[clippy::version = "1.43.0"]
|
2020-01-07 16:53:56 +00:00
|
|
|
pub WILDCARD_IMPORTS,
|
|
|
|
pedantic,
|
|
|
|
"lint `use _::*` statements"
|
|
|
|
}
|
|
|
|
|
2020-05-11 18:23:47 +00: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 16:53:56 +00:00
|
|
|
|
2020-06-25 20:41:36 +00:00
|
|
|
impl LateLintPass<'_> for WildcardImports {
|
|
|
|
fn check_item(&mut self, cx: &LateContext<'_>, item: &Item<'_>) {
|
2021-06-19 18:14:05 +00:00
|
|
|
if is_test_module_or_function(cx.tcx, item) {
|
2020-05-11 18:23:47 +00:00
|
|
|
self.test_modules_deep = self.test_modules_deep.saturating_add(1);
|
|
|
|
}
|
2020-01-07 16:53:56 +00:00
|
|
|
if item.vis.node.is_pub() || item.vis.node.is_pub_restricted() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if_chain! {
|
|
|
|
if let ItemKind::Use(use_path, UseKind::Glob) = &item.kind;
|
2020-05-11 18:23:47 +00:00
|
|
|
if self.warn_on_all || !self.check_exceptions(item, use_path.segments);
|
2021-01-30 16:47:51 +00:00
|
|
|
let used_imports = cx.tcx.names_imported_by_glob_use(item.def_id);
|
2020-01-09 17:12:54 +00:00
|
|
|
if !used_imports.is_empty(); // Already handled by `unused_imports`
|
2020-01-07 16:53:56 +00:00
|
|
|
then {
|
2020-01-09 17:12:54 +00:00
|
|
|
let mut applicability = Applicability::MachineApplicable;
|
2020-02-21 09:15:38 +00:00
|
|
|
let import_source_snippet = snippet_with_applicability(cx, use_path.span, "..", &mut applicability);
|
|
|
|
let (span, braced_glob) = if import_source_snippet.is_empty() {
|
2020-01-09 17:12:54 +00:00
|
|
|
// This is a `_::{_, *}` import
|
2020-02-21 09:15:38 +00:00
|
|
|
// In this case `use_path.span` is empty and ends directly in front of the `*`,
|
|
|
|
// so we need to extend it by one byte.
|
2020-01-09 17:12:54 +00:00
|
|
|
(
|
|
|
|
use_path.span.with_hi(use_path.span.hi() + BytePos(1)),
|
|
|
|
true,
|
|
|
|
)
|
2020-01-07 16:53:56 +00:00
|
|
|
} else {
|
2020-02-21 09:15:38 +00:00
|
|
|
// 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));
|
|
|
|
}
|
2020-01-09 17:12:54 +00:00
|
|
|
(
|
2020-05-11 18:23:47 +00:00
|
|
|
span, false,
|
2020-01-09 17:12:54 +00:00
|
|
|
)
|
2020-01-07 16:53:56 +00:00
|
|
|
};
|
|
|
|
|
2020-01-09 17:12:54 +00:00
|
|
|
let imports_string = if used_imports.len() == 1 {
|
|
|
|
used_imports.iter().next().unwrap().to_string()
|
|
|
|
} else {
|
|
|
|
let mut imports = used_imports
|
2020-01-07 16:53:56 +00:00
|
|
|
.iter()
|
2020-01-09 17:12:54 +00:00
|
|
|
.map(ToString::to_string)
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
imports.sort();
|
|
|
|
if braced_glob {
|
|
|
|
imports.join(", ")
|
2020-01-07 16:53:56 +00:00
|
|
|
} else {
|
2020-01-09 17:12:54 +00:00
|
|
|
format!("{{{}}}", imports.join(", "))
|
2020-01-07 16:53:56 +00:00
|
|
|
}
|
2020-01-09 17:12:54 +00:00
|
|
|
};
|
2020-01-07 16:53:56 +00:00
|
|
|
|
2020-02-21 09:15:38 +00:00
|
|
|
let sugg = if braced_glob {
|
2020-01-09 17:12:54 +00:00
|
|
|
imports_string
|
2020-01-07 16:53:56 +00:00
|
|
|
} else {
|
2020-02-21 09:15:38 +00:00
|
|
|
format!("{}::{}", import_source_snippet, imports_string)
|
2020-01-09 17:12:54 +00:00
|
|
|
};
|
2020-01-07 16:53:56 +00:00
|
|
|
|
2020-01-14 12:52:08 +00:00
|
|
|
let (lint, message) = if let Res::Def(DefKind::Enum, _) = use_path.res {
|
|
|
|
(ENUM_GLOB_USE, "usage of wildcard import for enum variants")
|
|
|
|
} else {
|
|
|
|
(WILDCARD_IMPORTS, "usage of wildcard import")
|
|
|
|
};
|
|
|
|
|
2020-01-09 17:12:54 +00:00
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
2020-01-14 12:52:08 +00:00
|
|
|
lint,
|
2020-01-09 17:12:54 +00:00
|
|
|
span,
|
2020-01-14 12:52:08 +00:00
|
|
|
message,
|
2020-01-09 17:12:54 +00:00
|
|
|
"try",
|
|
|
|
sugg,
|
|
|
|
applicability,
|
|
|
|
);
|
2020-01-07 16:53:56 +00:00
|
|
|
}
|
2020-01-09 17:12:54 +00:00
|
|
|
}
|
2020-01-07 16:53:56 +00:00
|
|
|
}
|
2020-05-11 18:23:47 +00:00
|
|
|
|
2021-06-19 18:14:05 +00:00
|
|
|
fn check_item_post(&mut self, cx: &LateContext<'_>, item: &Item<'_>) {
|
|
|
|
if is_test_module_or_function(cx.tcx, item) {
|
2020-05-11 18:23:47 +00:00
|
|
|
self.test_modules_deep = self.test_modules_deep.saturating_sub(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl WildcardImports {
|
|
|
|
fn check_exceptions(&self, item: &Item<'_>, segments: &[PathSegment<'_>]) -> bool {
|
2021-10-29 16:14:22 +00:00
|
|
|
item.span.from_expansion()
|
2020-05-11 18:23:47 +00:00
|
|
|
|| is_prelude_import(segments)
|
|
|
|
|| (is_super_only_import(segments) && self.test_modules_deep > 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-28 14:10:16 +00:00
|
|
|
// Allow "...prelude::..::*" imports.
|
2020-05-11 18:23:47 +00:00
|
|
|
// Many crates have a prelude, and it is imported as a glob by design.
|
|
|
|
fn is_prelude_import(segments: &[PathSegment<'_>]) -> bool {
|
2021-01-15 09:56:44 +00:00
|
|
|
segments.iter().any(|ps| ps.ident.name == sym::prelude)
|
2020-05-11 18:23:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Allow "super::*" imports in tests.
|
|
|
|
fn is_super_only_import(segments: &[PathSegment<'_>]) -> bool {
|
2021-01-15 09:56:44 +00:00
|
|
|
segments.len() == 1 && segments[0].ident.name == kw::Super
|
2020-05-11 18:23:47 +00:00
|
|
|
}
|