2023-06-03 14:38:16 -05:00
|
|
|
use ast::{AttrStyle, Attribute};
|
2023-07-03 12:42:48 -05:00
|
|
|
use clippy_utils::diagnostics::span_lint_and_sugg;
|
|
|
|
use clippy_utils::is_from_proc_macro;
|
2023-03-10 17:23:58 -06: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;
|
2023-03-11 14:30:34 -06:00
|
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
2023-03-10 17:23:58 -06:00
|
|
|
|
|
|
|
declare_clippy_lint! {
|
2023-09-28 20:08:34 -05:00
|
|
|
/// ### What it does
|
2023-04-12 06:48:03 -05:00
|
|
|
/// Checks for usage of the `#[allow]` attribute and suggests replacing it with
|
2023-03-11 14:30:34 -06:00
|
|
|
/// 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-10 17:23:58 -06:00
|
|
|
/// ### Why is this bad?
|
2023-03-11 14:30:34 -06:00
|
|
|
/// `#[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-10 17:23:58 -06:00
|
|
|
/// ### Example
|
2023-03-10 18:08:03 -06:00
|
|
|
/// ```rust,ignore
|
2023-03-10 17:23:58 -06:00
|
|
|
/// #[allow(unused_mut)]
|
|
|
|
/// fn foo() -> usize {
|
|
|
|
/// let mut a = Vec::new();
|
|
|
|
/// a.len()
|
2023-03-15 17:20:52 -05:00
|
|
|
/// }
|
2023-03-10 17:23:58 -06:00
|
|
|
/// ```
|
|
|
|
/// Use instead:
|
2023-03-10 18:08:03 -06:00
|
|
|
/// ```rust,ignore
|
2023-03-15 17:20:52 -05:00
|
|
|
/// #![feature(lint_reasons)]
|
2023-03-10 17:23:58 -06:00
|
|
|
/// #[expect(unused_mut)]
|
|
|
|
/// fn foo() -> usize {
|
|
|
|
/// let mut a = Vec::new();
|
|
|
|
/// a.len()
|
|
|
|
/// }
|
|
|
|
/// ```
|
2023-05-28 17:46:15 -05:00
|
|
|
#[clippy::version = "1.70.0"]
|
2023-03-15 17:18:25 -05:00
|
|
|
pub ALLOW_ATTRIBUTES,
|
2023-03-10 17:23:58 -06:00
|
|
|
restriction,
|
|
|
|
"`#[allow]` will not trigger if a warning isn't found. `#[expect]` triggers if there are no warnings."
|
|
|
|
}
|
|
|
|
|
2023-03-15 17:18:25 -05:00
|
|
|
declare_lint_pass!(AllowAttribute => [ALLOW_ATTRIBUTES]);
|
2023-03-10 17:23:58 -06:00
|
|
|
|
|
|
|
impl LateLintPass<'_> for AllowAttribute {
|
|
|
|
// Separate each crate's features.
|
2023-06-03 14:38:16 -05:00
|
|
|
fn check_attribute<'cx>(&mut self, cx: &LateContext<'cx>, attr: &'cx Attribute) {
|
2023-11-10 11:29:28 -06:00
|
|
|
if !in_external_macro(cx.sess(), attr.span)
|
|
|
|
&& cx.tcx.features().lint_reasons
|
|
|
|
&& let AttrStyle::Outer = attr.style
|
|
|
|
&& let Some(ident) = attr.ident()
|
|
|
|
&& ident.name == rustc_span::symbol::sym::allow
|
|
|
|
&& !is_from_proc_macro(cx, &attr)
|
|
|
|
{
|
|
|
|
span_lint_and_sugg(
|
|
|
|
cx,
|
|
|
|
ALLOW_ATTRIBUTES,
|
|
|
|
ident.span,
|
|
|
|
"#[allow] attribute found",
|
|
|
|
"replace it with",
|
|
|
|
"expect".into(),
|
|
|
|
Applicability::MachineApplicable,
|
|
|
|
);
|
2023-03-10 17:23:58 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|