///
/// ### Example
/// ```ignore
- /// // Bad
/// #[deny(dead_code)]
/// extern crate foo;
/// #[forbid(dead_code)]
/// use foo::bar;
+ /// ```
///
- /// // Ok
+ /// Use instead:
+ /// ```rust,ignore
/// #[allow(unused_imports)]
/// use foo::baz;
/// #[allow(unused_imports)]
///
/// ### Example
/// ```rust
+ /// #[allow(dead_code)]
+ ///
+ /// fn not_quite_good_code() { }
+ /// ```
+ ///
+ /// Use instead:
+ /// ```rust
/// // Good (as inner attribute)
/// #![allow(dead_code)]
///
/// fn this_is_fine() { }
///
- /// // Bad
- /// #[allow(dead_code)]
- ///
- /// fn not_quite_good_code() { }
+ /// // or
///
/// // Good (as outer attribute)
/// #[allow(dead_code)]
/// These lints should only be enabled on a lint-by-lint basis and with careful consideration.
///
/// ### Example
- /// Bad:
/// ```rust
/// #![deny(clippy::restriction)]
/// ```
///
- /// Good:
+ /// Use instead:
/// ```rust
/// #![deny(clippy::as_conversions)]
/// ```
/// [#3123](https://github.com/rust-lang/rust-clippy/pull/3123#issuecomment-422321765)
///
/// ### Example
- /// Bad:
/// ```rust
/// #[cfg_attr(rustfmt, rustfmt_skip)]
/// fn main() { }
/// ```
///
- /// Good:
+ /// Use instead:
/// ```rust
/// #[rustfmt::skip]
/// fn main() { }
/// by the conditional compilation engine.
///
/// ### Example
- /// Bad:
/// ```rust
/// #[cfg(linux)]
/// fn conditional() { }
/// ```
///
- /// Good:
+ /// Use instead:
/// ```rust
+ /// # mod hidden {
/// #[cfg(target_os = "linux")]
/// fn conditional() { }
- /// ```
+ /// # }
+ ///
+ /// // or
///
- /// Or:
- /// ```rust
/// #[cfg(unix)]
/// fn conditional() { }
/// ```
/// ensure that others understand the reasoning
///
/// ### Example
- /// Bad:
/// ```rust
/// #![feature(lint_reasons)]
///
/// #![allow(clippy::some_lint)]
/// ```
///
- /// Good:
+ /// Use instead:
/// ```rust
/// #![feature(lint_reasons)]
///
}
fn check_empty_line_after_outer_attr(cx: &EarlyContext<'_>, item: &rustc_ast::Item) {
- for attr in &item.attrs {
+ let mut iter = item.attrs.iter().peekable();
+ while let Some(attr) = iter.next() {
if matches!(attr.kind, AttrKind::Normal(..))
&& attr.style == AttrStyle::Outer
&& is_present_in_source(cx, attr.span)
{
let begin_of_attr_to_item = Span::new(attr.span.lo(), item.span.lo(), item.span.ctxt(), item.span.parent());
- let end_of_attr_to_item = Span::new(attr.span.hi(), item.span.lo(), item.span.ctxt(), item.span.parent());
+ let end_of_attr_to_next_attr_or_item = Span::new(
+ attr.span.hi(),
+ iter.peek().map_or(item.span.lo(), |next_attr| next_attr.span.lo()),
+ item.span.ctxt(),
+ item.span.parent(),
+ );
- if let Some(snippet) = snippet_opt(cx, end_of_attr_to_item) {
+ if let Some(snippet) = snippet_opt(cx, end_of_attr_to_next_attr_or_item) {
let lines = snippet.split('\n').collect::<Vec<_>>();
let lines = without_block_comments(lines);
if feature_item.has_name(sym::rustfmt);
// check for `rustfmt_skip` and `rustfmt::skip`
if let Some(skip_item) = &items[1].meta_item();
- if skip_item.has_name(sym!(rustfmt_skip)) ||
- skip_item.path.segments.last().expect("empty path in attribute").ident.name == sym::skip;
+ if skip_item.has_name(sym!(rustfmt_skip))
+ || skip_item
+ .path
+ .segments
+ .last()
+ .expect("empty path in attribute")
+ .ident
+ .name
+ == sym::skip;
// Only lint outer attributes, because custom inner attributes are unstable
// Tracking issue: https://github.com/rust-lang/rust/issues/54726
if attr.style == AttrStyle::Outer;