-
-
Notifications
You must be signed in to change notification settings - Fork 14.6k
Add new rustdoc broken_footnote lint
#137803
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Open
GuillaumeGomez
wants to merge
8
commits into
rust-lang:main
Choose a base branch
from
GuillaumeGomez:broken-footnote
base: main
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
+204
−0
Open
Changes from all commits
Commits
Show all changes
8 commits
Select commit
Hold shift + click to select a range
963746f
Add new rustdoc `broken_footnote` lint
GuillaumeGomez e7b4bc8
Add ui test for rustdoc `broken_footnote` lint
GuillaumeGomez 6b500aa
Add new `unused_footnote_definition` rustdoc lint
GuillaumeGomez 621d984
Add ui test for new `unused_footnote_definition` rustdoc lint
GuillaumeGomez 4ddd48c
Improve description of new rustdoc lints
GuillaumeGomez 5deff4b
Remove outdated comment
GuillaumeGomez 7b8b405
Correctly handle "escaped footnotes"
GuillaumeGomez c470ca8
Improve suggestion for `broken_footnote` lint
GuillaumeGomez File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,96 @@ | ||
| use std::ops::Range; | ||
|
|
||
| use rustc_data_structures::fx::{FxHashMap, FxHashSet}; | ||
| use rustc_errors::DiagDecorator; | ||
| use rustc_hir::HirId; | ||
| use rustc_lint_defs::Applicability; | ||
| use rustc_resolve::rustdoc::pulldown_cmark::{Event, Options, Parser, Tag}; | ||
| use rustc_resolve::rustdoc::source_span_for_markdown_range; | ||
|
|
||
| use crate::clean::Item; | ||
| use crate::core::DocContext; | ||
|
|
||
| pub(crate) fn visit_item(cx: &DocContext<'_>, item: &Item, hir_id: HirId, dox: &str) { | ||
| let tcx = cx.tcx; | ||
|
|
||
| let mut missing_footnote_references = FxHashSet::default(); | ||
| let mut footnote_references = FxHashSet::default(); | ||
| let mut footnote_definitions = FxHashMap::default(); | ||
|
|
||
| let options = Options::ENABLE_FOOTNOTES; | ||
| let mut parser = Parser::new_ext(dox, options).into_offset_iter().peekable(); | ||
| while let Some((event, span)) = parser.next() { | ||
| match event { | ||
| Event::Text(text) | ||
| if &*text == "[" | ||
| && let Some((Event::Text(_), range)) = parser.next() | ||
| && dox[span.end..range.end].starts_with('^') => | ||
| { | ||
| loop { | ||
| let Some((Event::Text(text), new_span)) = parser.peek() else { break }; | ||
| if &**text != "]" { | ||
| parser.next(); | ||
| continue; | ||
| } | ||
| let text = &dox[span.end..new_span.end]; | ||
| if !text.ends_with("\\]") { | ||
| missing_footnote_references | ||
| .insert(Range { start: span.start, end: new_span.end }); | ||
| } | ||
| break; | ||
| } | ||
| } | ||
| Event::FootnoteReference(label) => { | ||
| footnote_references.insert(label); | ||
| } | ||
| Event::Start(Tag::FootnoteDefinition(label)) => { | ||
| footnote_definitions.insert(label, span.start + 1); | ||
| } | ||
| _ => {} | ||
| } | ||
| } | ||
|
|
||
| #[allow(rustc::potential_query_instability)] | ||
| for (footnote, span) in footnote_definitions { | ||
| if !footnote_references.contains(&footnote) { | ||
| let (span, _) = source_span_for_markdown_range( | ||
| tcx, | ||
| dox, | ||
| &(span..span + 1), | ||
| &item.attrs.doc_strings, | ||
| ) | ||
| .unwrap_or_else(|| (item.attr_span(tcx), false)); | ||
|
|
||
| tcx.emit_node_span_lint( | ||
| crate::lint::UNUSED_FOOTNOTE_DEFINITION, | ||
| hir_id, | ||
| span, | ||
| DiagDecorator(|lint| { | ||
| lint.primary_message("unused footnote definition"); | ||
| }), | ||
| ); | ||
| } | ||
| } | ||
|
|
||
| #[allow(rustc::potential_query_instability)] | ||
| for span in missing_footnote_references { | ||
| let ref_span = source_span_for_markdown_range(tcx, dox, &span, &item.attrs.doc_strings) | ||
| .map(|(span, _)| span) | ||
| .unwrap_or_else(|| item.attr_span(tcx)); | ||
|
|
||
| tcx.emit_node_span_lint( | ||
| crate::lint::BROKEN_FOOTNOTE, | ||
| hir_id, | ||
| ref_span, | ||
| DiagDecorator(|lint| { | ||
| lint.primary_message("no footnote definition matching this footnote"); | ||
| lint.span_suggestion( | ||
| ref_span.shrink_to_lo(), | ||
| "if it should not be a footnote, escape it", | ||
| format!("\\{}", &dox[span]), | ||
| Applicability::MaybeIncorrect, | ||
| ); | ||
| }), | ||
| ); | ||
| } | ||
| } |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,27 @@ | ||
| #![deny(rustdoc::broken_footnote)] | ||
|
|
||
| //! Footnote referenced [^1]. And [^2]. And [^bla]. | ||
| //! | ||
| //! [^1]: footnote defined | ||
| //~^^^ ERROR: no footnote definition matching this footnote | ||
| //~| ERROR: no footnote definition matching this footnote | ||
|
|
||
| //! [^*] special characters can appear within footnote references | ||
| //~^ ERROR: no footnote definition matching this footnote | ||
| //! | ||
| //! [^**] | ||
| //! | ||
| //! [^**]: not an error | ||
| //! | ||
| //! [^\_] so can escaped characters | ||
| //~^ ERROR: no footnote definition matching this footnote | ||
|
|
||
| // Backslash escaped footnotes should not be recognized: | ||
| //! [\^4] | ||
| //! | ||
| //! [^5\] | ||
| //! | ||
| //! \[^yup] | ||
| //! | ||
| //! [^foo\ | ||
| //! bar] |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,40 @@ | ||
| error: no footnote definition matching this footnote | ||
| --> $DIR/broken-footnote.rs:9:5 | ||
| | | ||
| LL | //! [^*] special characters can appear within footnote references | ||
| | -^^^ | ||
| | | | ||
| | help: if it should not be a footnote, escape it: `\[^*]` | ||
| | | ||
| note: the lint level is defined here | ||
| --> $DIR/broken-footnote.rs:1:9 | ||
| | | ||
| LL | #![deny(rustdoc::broken_footnote)] | ||
| | ^^^^^^^^^^^^^^^^^^^^^^^^ | ||
|
|
||
| error: no footnote definition matching this footnote | ||
| --> $DIR/broken-footnote.rs:3:35 | ||
| | | ||
| LL | //! Footnote referenced [^1]. And [^2]. And [^bla]. | ||
| | -^^^ | ||
| | | | ||
| | help: if it should not be a footnote, escape it: `\[^2]` | ||
|
|
||
| error: no footnote definition matching this footnote | ||
| --> $DIR/broken-footnote.rs:3:45 | ||
| | | ||
| LL | //! Footnote referenced [^1]. And [^2]. And [^bla]. | ||
| | -^^^^^ | ||
| | | | ||
| | help: if it should not be a footnote, escape it: `\[^bla]` | ||
|
|
||
| error: no footnote definition matching this footnote | ||
| --> $DIR/broken-footnote.rs:16:5 | ||
| | | ||
| LL | //! [^\_] so can escaped characters | ||
| | -^^^^ | ||
| | | | ||
| | help: if it should not be a footnote, escape it: `\[^\_]` | ||
|
|
||
| error: aborting due to 4 previous errors | ||
|
|
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,9 @@ | ||
| // This test ensures that the `rustdoc::unused_footnote` lint is working as expected. | ||
|
|
||
| #![deny(rustdoc::unused_footnote_definition)] | ||
|
|
||
| //! Footnote referenced. [^2] | ||
| //! | ||
| //! [^1]: footnote defined | ||
| //! [^2]: footnote defined | ||
| //~^^ ERROR: unused_footnote_definition |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,14 @@ | ||
| error: unused footnote definition | ||
| --> $DIR/unused-footnote.rs:7:6 | ||
| | | ||
| LL | //! [^1]: footnote defined | ||
| | ^ | ||
| | | ||
| note: the lint level is defined here | ||
| --> $DIR/unused-footnote.rs:3:9 | ||
| | | ||
| LL | #![deny(rustdoc::unused_footnote_definition)] | ||
| | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||
|
|
||
| error: aborting due to 1 previous error | ||
|
|
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Still missing a few:
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah good point, adding them as well.