1
Fork 0

New rustdoc lint to respect -Dwarnings correctly

This adds a new lint to `rustc` that is used in rustdoc when a code
block is empty or cannot be parsed as valid Rust code.

Previously this was unconditionally a warning. As such some
documentation comments were (unknowingly) abusing this to pass despite
the `-Dwarnings` used when compiling `rustc`, this should not be the
case anymore.
This commit is contained in:
Alexis Bourget 2020-12-28 23:07:20 +01:00 committed by Joshua Nelson
parent 3e99439f4d
commit b574c67b93
8 changed files with 116 additions and 41 deletions

View file

@ -1241,7 +1241,7 @@ impl<'tcx> RegionInferenceContext<'tcx> {
/// it. However, it works pretty well in practice. In particular, /// it. However, it works pretty well in practice. In particular,
/// this is needed to deal with projection outlives bounds like /// this is needed to deal with projection outlives bounds like
/// ///
/// ```ignore (internal compiler representation so lifetime syntax is invalid) /// ```text (internal compiler representation so lifetime syntax is invalid)
/// <T as Foo<'0>>::Item: '1 /// <T as Foo<'0>>::Item: '1
/// ``` /// ```
/// ///

View file

@ -46,6 +46,7 @@ pub struct OpaqueTypeDecl<'tcx> {
/// type Foo = impl Baz; /// type Foo = impl Baz;
/// fn bar() -> Foo { /// fn bar() -> Foo {
/// // ^^^ This is the span we are looking for! /// // ^^^ This is the span we are looking for!
/// }
/// ``` /// ```
/// ///
/// In cases where the fn returns `(impl Trait, impl Trait)` or /// In cases where the fn returns `(impl Trait, impl Trait)` or

View file

@ -323,7 +323,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
/// ///
/// InferBorrowKind results in a structure like this: /// InferBorrowKind results in a structure like this:
/// ///
/// ``` /// ```text
/// { /// {
/// Place(base: hir_id_s, projections: [], ....) -> { /// Place(base: hir_id_s, projections: [], ....) -> {
/// capture_kind_expr: hir_id_L5, /// capture_kind_expr: hir_id_L5,
@ -348,7 +348,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
/// ``` /// ```
/// ///
/// After the min capture analysis, we get: /// After the min capture analysis, we get:
/// ``` /// ```text
/// { /// {
/// hir_id_s -> [ /// hir_id_s -> [
/// Place(base: hir_id_s, projections: [], ....) -> { /// Place(base: hir_id_s, projections: [], ....) -> {

View file

@ -294,6 +294,43 @@ warning: unclosed HTML tag `h1`
warning: 2 warnings emitted warning: 2 warnings emitted
``` ```
## invalid_rust_codeblock
This lint **warns by default**. It detects Rust code blocks in documentation
examples that are invalid (e.g. empty, not parsable as Rust). For example:
```rust
/// Empty code blocks (with and without the `rust` marker):
///
/// ```rust
/// ```
///
/// Unclosed code blocks (with and without the `rust` marker):
///
/// ```rust
fn main() {}
```
Which will give:
```text
warning: Rust code block is empty
--> src/lib.rs:3:5
|
3 | /// ```rust
| _____^
4 | | /// ```
| |_______^
|
= note: `#[warn(rustdoc::invalid_rust_codeblock)]` on by default
warning: Rust code block is empty
--> src/lib.rs:8:5
|
8 | /// ```rust
| ^^^^^^^
```
## bare_urls ## bare_urls
This lint is **warn-by-default**. It detects URLs which are not links. This lint is **warn-by-default**. It detects URLs which are not links.

View file

@ -157,6 +157,18 @@ declare_rustdoc_lint! {
"detects URLs that are not hyperlinks" "detects URLs that are not hyperlinks"
} }
declare_rustdoc_lint! {
/// The `invalid_rust_codeblock` lint detects Rust code blocks in
/// documentation examples that are invalid (e.g. empty, not parsable as
/// Rust code). This is a `rustdoc` only lint, see the documentation in the
/// [rustdoc book].
///
/// [rustdoc book]: ../../../rustdoc/lints.html#invalid_rust_codeblock
INVALID_RUST_CODEBLOCK,
Warn,
"codeblock could not be parsed as valid Rust or is empty"
}
crate static RUSTDOC_LINTS: Lazy<Vec<&'static Lint>> = Lazy::new(|| { crate static RUSTDOC_LINTS: Lazy<Vec<&'static Lint>> = Lazy::new(|| {
vec![ vec![
BROKEN_INTRA_DOC_LINKS, BROKEN_INTRA_DOC_LINKS,
@ -164,6 +176,7 @@ crate static RUSTDOC_LINTS: Lazy<Vec<&'static Lint>> = Lazy::new(|| {
MISSING_DOC_CODE_EXAMPLES, MISSING_DOC_CODE_EXAMPLES,
PRIVATE_DOC_TESTS, PRIVATE_DOC_TESTS,
INVALID_CODEBLOCK_ATTRIBUTES, INVALID_CODEBLOCK_ATTRIBUTES,
INVALID_RUST_CODEBLOCK,
INVALID_HTML_TAGS, INVALID_HTML_TAGS,
BARE_URLS, BARE_URLS,
MISSING_CRATE_LEVEL_DOCS, MISSING_CRATE_LEVEL_DOCS,

View file

@ -1,5 +1,6 @@
use rustc_data_structures::sync::{Lock, Lrc}; use rustc_data_structures::sync::{Lock, Lrc};
use rustc_errors::{emitter::Emitter, Applicability, Diagnostic, Handler}; use rustc_errors::{emitter::Emitter, Applicability, Diagnostic, Handler};
use rustc_middle::lint::LintDiagnosticBuilder;
use rustc_parse::parse_stream_from_source_str; use rustc_parse::parse_stream_from_source_str;
use rustc_session::parse::ParseSess; use rustc_session::parse::ParseSess;
use rustc_span::source_map::{FilePathMapping, SourceMap}; use rustc_span::source_map::{FilePathMapping, SourceMap};
@ -47,50 +48,65 @@ impl<'a, 'tcx> SyntaxChecker<'a, 'tcx> {
.unwrap_or(false); .unwrap_or(false);
let buffer = buffer.borrow(); let buffer = buffer.borrow();
if buffer.has_errors || is_empty { if !(buffer.has_errors || is_empty) {
let mut diag = if let Some(sp) = super::source_span_for_markdown_range( // No errors in a non-empty program.
self.cx.tcx, return;
&dox, }
&code_block.range,
&item.attrs, let local_id = match item.def_id.as_local() {
) { Some(id) => id,
let (warning_message, suggest_using_text) = if buffer.has_errors { // We don't need to check the syntax for other crates so returning
("could not parse code block as Rust code", true) // without doing anything should not be a problem.
None => return,
};
let hir_id = self.cx.tcx.hir().local_def_id_to_hir_id(local_id);
let suggest_using_text = code_block.syntax.is_none() && code_block.is_fenced;
let is_ignore = code_block.is_ignore;
// The span and whether it is precise or not.
let (sp, precise_span) = match super::source_span_for_markdown_range(
self.cx.tcx,
&dox,
&code_block.range,
&item.attrs,
) {
Some(sp) => (sp, true),
None => (item.attr_span(self.cx.tcx), false),
};
// lambda that will use the lint to start a new diagnostic and add
// a suggestion to it when needed.
let diag_builder = |lint: LintDiagnosticBuilder<'_>| {
let mut diag = if precise_span {
let msg = if buffer.has_errors {
"could not parse code block as Rust code"
} else { } else {
("Rust code block is empty", false) "Rust code block is empty"
}; };
let mut diag = self.cx.sess().struct_span_warn(sp, warning_message); let mut diag = lint.build(msg);
if code_block.syntax.is_none() && code_block.is_fenced { if suggest_using_text {
let sp = sp.from_inner(InnerSpan::new(0, 3)); let extended_msg = if is_ignore {
diag.span_suggestion(
sp,
"mark blocks that do not contain Rust code as text",
String::from("```text"),
Applicability::MachineApplicable,
);
} else if suggest_using_text && code_block.is_ignore {
let sp = sp.from_inner(InnerSpan::new(0, 3));
diag.span_suggestion(
sp,
"`ignore` code blocks require valid Rust code for syntax highlighting. \ "`ignore` code blocks require valid Rust code for syntax highlighting. \
Mark blocks that do not contain Rust code as text", Mark blocks that do not contain Rust code as text"
String::from("```text,"), } else {
"mark blocks that do not contain Rust code as text"
};
diag.span_suggestion(
sp.from_inner(InnerSpan::new(0, 3)),
extended_msg,
String::from("```text"),
Applicability::MachineApplicable, Applicability::MachineApplicable,
); );
} }
diag diag
} else { } else {
// We couldn't calculate the span of the markdown block that had the error, so our let mut diag = lint.build("doc comment contains an invalid Rust code block");
// diagnostics are going to be a bit lacking. if suggest_using_text {
let mut diag = self.cx.sess().struct_span_warn(
item.attr_span(self.cx.tcx),
"doc comment contains an invalid Rust code block",
);
if code_block.syntax.is_none() && code_block.is_fenced {
diag.help("mark blocks that do not contain Rust code as text: ```text"); diag.help("mark blocks that do not contain Rust code as text: ```text");
} }
@ -103,7 +119,17 @@ impl<'a, 'tcx> SyntaxChecker<'a, 'tcx> {
} }
diag.emit(); diag.emit();
} };
// Finally build and emit the completed diagnostic.
// All points of divergence have been handled earlier so this can be
// done the same way whether the span is precise or not.
self.cx.tcx.struct_span_lint_hir(
crate::lint::INVALID_RUST_CODEBLOCK,
hir_id,
sp,
diag_builder,
);
} }
} }

View file

@ -7,11 +7,8 @@ LL | | /// let heart = '❤️';
LL | | /// ``` LL | | /// ```
| |_______^ | |_______^
| |
= note: `#[warn(invalid_rust_codeblock)]` on by default
= note: error from rustc: character literal may only contain one codepoint = note: error from rustc: character literal may only contain one codepoint
help: `ignore` code blocks require valid Rust code for syntax highlighting. Mark blocks that do not contain Rust code as text
|
LL | /// ```text,ignore (to-prevent-tidy-error)
| ^^^^^^^^
warning: 1 warning emitted warning: 1 warning emitted

View file

@ -7,6 +7,7 @@ LL | | /// \__________pkt->size___________/ \_result->size_/ \__pkt->si
LL | | /// ``` LL | | /// ```
| |_______^ | |_______^
| |
= note: `#[warn(invalid_rust_codeblock)]` on by default
= note: error from rustc: unknown start of token: \ = note: error from rustc: unknown start of token: \
= note: error from rustc: unknown start of token: \ = note: error from rustc: unknown start of token: \
= note: error from rustc: unknown start of token: \ = note: error from rustc: unknown start of token: \