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:
parent
3e99439f4d
commit
b574c67b93
8 changed files with 116 additions and 41 deletions
|
@ -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
|
||||||
/// ```
|
/// ```
|
||||||
///
|
///
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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: [], ....) -> {
|
||||||
|
|
|
@ -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.
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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: \
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue