rust-clippy/clippy_lints/src/empty_structs_with_brackets.rs

100 lines
2.9 KiB
Rust
Raw Normal View History

2022-03-27 18:10:10 +00:00
use clippy_utils::{diagnostics::span_lint_and_then, source::snippet_opt};
2022-03-27 13:33:31 +00:00
use rustc_ast::ast::{Item, ItemKind, VariantData};
2022-03-27 12:16:08 +00:00
use rustc_errors::Applicability;
use rustc_lexer::TokenKind;
2022-03-27 12:16:08 +00:00
use rustc_lint::{EarlyContext, EarlyLintPass};
use rustc_session::{declare_lint_pass, declare_tool_lint};
use rustc_span::Span;
2022-03-27 12:16:08 +00:00
declare_clippy_lint! {
/// ### What it does
/// Finds structs without fields (a so-called "empty struct") that are declared with brackets.
2022-03-27 12:16:08 +00:00
///
/// ### Why is this bad?
/// Empty brackets after a struct declaration can be omitted.
///
/// ### Example
/// ```rust
/// struct Cookie {}
/// ```
/// Use instead:
/// ```rust
/// struct Cookie;
/// ```
2022-03-28 09:18:20 +00:00
#[clippy::version = "1.62.0"]
pub EMPTY_STRUCTS_WITH_BRACKETS,
2022-03-30 11:33:10 +00:00
restriction,
2022-03-27 12:16:08 +00:00
"finds struct declarations with empty brackets"
}
declare_lint_pass!(EmptyStructsWithBrackets => [EMPTY_STRUCTS_WITH_BRACKETS]);
2022-03-27 12:16:08 +00:00
impl EarlyLintPass for EmptyStructsWithBrackets {
2022-03-27 12:16:08 +00:00
fn check_item(&mut self, cx: &EarlyContext<'_>, item: &Item) {
let span_after_ident = item.span.with_lo(item.ident.span.hi());
2022-03-27 12:16:08 +00:00
2022-03-28 09:18:20 +00:00
if let ItemKind::Struct(var_data, _) = &item.kind
2022-04-04 06:48:49 +00:00
&& has_brackets(var_data)
2022-03-28 09:18:20 +00:00
&& has_no_fields(cx, var_data, span_after_ident) {
2022-03-27 18:10:10 +00:00
span_lint_and_then(
2022-03-27 12:16:08 +00:00
cx,
EMPTY_STRUCTS_WITH_BRACKETS,
2022-03-27 12:16:08 +00:00
span_after_ident,
"found empty brackets on struct declaration",
2022-03-27 18:10:10 +00:00
|diagnostic| {
diagnostic.span_suggestion_hidden(
span_after_ident,
"remove the brackets",
";".to_string(),
Applicability::MachineApplicable);
},
2022-03-27 12:16:08 +00:00
);
}
}
}
fn has_no_ident_token(braces_span_str: &str) -> bool {
!rustc_lexer::tokenize(braces_span_str).any(|t| t.kind == TokenKind::Ident)
}
2022-04-04 06:48:49 +00:00
fn has_brackets(var_data: &VariantData) -> bool {
!matches!(var_data, VariantData::Unit(_))
2022-03-28 09:18:20 +00:00
}
fn has_no_fields(cx: &EarlyContext<'_>, var_data: &VariantData, braces_span: Span) -> bool {
2022-03-28 09:18:20 +00:00
if !var_data.fields().is_empty() {
return false;
}
2022-03-28 09:18:20 +00:00
// there might still be field declarations hidden from the AST
2022-04-15 18:25:55 +00:00
// (conditionally compiled code using #[cfg(..)])
let Some(braces_span_str) = snippet_opt(cx, braces_span) else {
return false;
};
has_no_ident_token(braces_span_str.as_ref())
}
#[cfg(test)]
mod unit_test {
use super::*;
#[test]
fn test_has_no_ident_token() {
let input = "{ field: u8 }";
assert!(!has_no_ident_token(input));
let input = "(u8, String);";
assert!(!has_no_ident_token(input));
let input = " {
// test = 5
}
";
assert!(has_no_ident_token(input));
let input = " ();";
assert!(has_no_ident_token(input));
2022-03-27 12:16:08 +00:00
}
}