2022-02-01 11:21:42 +00:00
|
|
|
use clippy_utils::diagnostics::span_lint;
|
|
|
|
use itertools::Itertools;
|
|
|
|
use rustc_ast::{AttrKind, Attribute};
|
|
|
|
use rustc_lint::{EarlyContext, EarlyLintPass};
|
|
|
|
use rustc_session::{declare_lint_pass, declare_tool_lint};
|
|
|
|
|
|
|
|
declare_clippy_lint! {
|
|
|
|
/// ### What it does
|
|
|
|
/// Detects the syntax `['foo']` in documentation comments (notice quotes instead of backticks)
|
|
|
|
/// outside of code blocks
|
|
|
|
/// ### Why is this bad?
|
|
|
|
/// It is likely a typo when defining an intra-doc link
|
|
|
|
///
|
|
|
|
/// ### Example
|
|
|
|
/// ```rust
|
|
|
|
/// /// See also: ['foo']
|
|
|
|
/// fn bar() {}
|
|
|
|
/// ```
|
|
|
|
/// Use instead:
|
|
|
|
/// ```rust
|
|
|
|
/// /// See also: [`foo`]
|
|
|
|
/// fn bar() {}
|
|
|
|
/// ```
|
2022-08-12 21:19:53 +00:00
|
|
|
#[clippy::version = "1.63.0"]
|
2022-02-01 11:21:42 +00:00
|
|
|
pub DOC_LINK_WITH_QUOTES,
|
|
|
|
pedantic,
|
|
|
|
"possible typo for an intra-doc link"
|
|
|
|
}
|
|
|
|
declare_lint_pass!(DocLinkWithQuotes => [DOC_LINK_WITH_QUOTES]);
|
|
|
|
|
|
|
|
impl EarlyLintPass for DocLinkWithQuotes {
|
|
|
|
fn check_attribute(&mut self, ctx: &EarlyContext<'_>, attr: &Attribute) {
|
|
|
|
if let AttrKind::DocComment(_, symbol) = attr.kind {
|
|
|
|
if contains_quote_link(symbol.as_str()) {
|
|
|
|
span_lint(
|
|
|
|
ctx,
|
|
|
|
DOC_LINK_WITH_QUOTES,
|
|
|
|
attr.span,
|
|
|
|
"possible intra-doc link using quotes instead of backticks",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn contains_quote_link(s: &str) -> bool {
|
|
|
|
let mut in_backticks = false;
|
|
|
|
let mut found_opening = false;
|
|
|
|
|
|
|
|
for c in s.chars().tuple_windows::<(char, char)>() {
|
|
|
|
match c {
|
|
|
|
('`', _) => in_backticks = !in_backticks,
|
|
|
|
('[', '\'') if !in_backticks => found_opening = true,
|
|
|
|
('\'', ']') if !in_backticks && found_opening => return true,
|
|
|
|
_ => {},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|