rust/clippy_lints/src/allow_attributes.rs

75 lines
2.6 KiB
Rust
Raw Normal View History

use ast::AttrStyle;
use clippy_utils::{diagnostics::span_lint_and_sugg, is_from_proc_macro};
2023-03-11 00:23:58 +01:00
use rustc_ast as ast;
use rustc_errors::Applicability;
2023-04-21 05:12:14 -05:00
use rustc_lint::{LateContext, LateLintPass, LintContext};
use rustc_middle::lint::in_external_macro;
use rustc_session::{declare_lint_pass, declare_tool_lint};
2023-03-11 00:23:58 +01:00
declare_clippy_lint! {
2023-04-12 13:48:03 +02:00
/// Checks for usage of the `#[allow]` attribute and suggests replacing it with
/// the `#[expect]` (See [RFC 2383](https://rust-lang.github.io/rfcs/2383-lint-reasons.html))
///
/// The expect attribute is still unstable and requires the `lint_reasons`
/// on nightly. It can be enabled by adding `#![feature(lint_reasons)]` to
/// the crate root.
///
/// This lint only warns outer attributes (`#[allow]`), as inner attributes
/// (`#![allow]`) are usually used to enable or disable lints on a global scale.
///
2023-03-11 00:23:58 +01:00
/// ### Why is this bad?
///
/// `#[expect]` attributes suppress the lint emission, but emit a warning, if
/// the expectation is unfulfilled. This can be useful to be notified when the
/// lint is no longer triggered.
///
2023-03-11 00:23:58 +01:00
/// ### Example
2023-03-11 01:08:03 +01:00
/// ```rust,ignore
2023-03-11 00:23:58 +01:00
/// #[allow(unused_mut)]
/// fn foo() -> usize {
/// let mut a = Vec::new();
/// a.len()
/// }
2023-03-11 00:23:58 +01:00
/// ```
/// Use instead:
2023-03-11 01:08:03 +01:00
/// ```rust,ignore
/// #![feature(lint_reasons)]
2023-03-11 00:23:58 +01:00
/// #[expect(unused_mut)]
/// fn foo() -> usize {
/// let mut a = Vec::new();
/// a.len()
/// }
/// ```
#[clippy::version = "1.70.0"]
2023-03-15 23:18:25 +01:00
pub ALLOW_ATTRIBUTES,
2023-03-11 00:23:58 +01:00
restriction,
"`#[allow]` will not trigger if a warning isn't found. `#[expect]` triggers if there are no warnings."
}
2023-03-15 23:18:25 +01:00
declare_lint_pass!(AllowAttribute => [ALLOW_ATTRIBUTES]);
2023-03-11 00:23:58 +01:00
impl LateLintPass<'_> for AllowAttribute {
// Separate each crate's features.
fn check_attribute(&mut self, cx: &LateContext<'_>, attr: &ast::Attribute) {
if_chain! {
2023-04-21 05:12:14 -05:00
if !in_external_macro(cx.sess(), attr.span);
if cx.tcx.features().lint_reasons;
2023-03-11 00:23:58 +01:00
if let AttrStyle::Outer = attr.style;
if let Some(ident) = attr.ident();
if ident.name == rustc_span::symbol::sym::allow;
if !is_from_proc_macro(cx, &(attr, cx));
2023-03-11 00:23:58 +01:00
then {
span_lint_and_sugg(
cx,
2023-03-15 23:18:25 +01:00
ALLOW_ATTRIBUTES,
ident.span,
2023-03-11 00:23:58 +01:00
"#[allow] attribute found",
"replace it with",
"expect".into(),
Applicability::MachineApplicable,
);
2023-03-11 00:23:58 +01:00
}
}
}
}