rustdoc: Fix LinkReplacer link matching

This commit is contained in:
benediktwerner 2023-02-25 17:52:25 +01:00 committed by Benedikt Werner
parent 26c98689f2
commit 9968f3ce55
No known key found for this signature in database
3 changed files with 48 additions and 6 deletions

View File

@ -382,7 +382,6 @@ fn next(&mut self) -> Option<Self::Item> {
Some(Event::Code(text)) => { Some(Event::Code(text)) => {
trace!("saw code {}", text); trace!("saw code {}", text);
if let Some(link) = self.shortcut_link { if let Some(link) = self.shortcut_link {
trace!("original text was {}", link.original_text);
// NOTE: this only replaces if the code block is the *entire* text. // NOTE: this only replaces if the code block is the *entire* text.
// If only part of the link has code highlighting, the disambiguator will not be removed. // If only part of the link has code highlighting, the disambiguator will not be removed.
// e.g. [fn@`f`] // e.g. [fn@`f`]
@ -391,8 +390,11 @@ fn next(&mut self) -> Option<Self::Item> {
// So we could never be sure we weren't replacing too much: // So we could never be sure we weren't replacing too much:
// [fn@my_`f`unc] is treated the same as [my_func()] in that pass. // [fn@my_`f`unc] is treated the same as [my_func()] in that pass.
// //
// NOTE: &[1..len() - 1] is to strip the backticks // NOTE: .get(1..len() - 1) is to strip the backticks
if **text == link.original_text[1..link.original_text.len() - 1] { if let Some(link) = self.links.iter().find(|l| {
l.href == link.href
&& Some(&**text) == l.original_text.get(1..l.original_text.len() - 1)
}) {
debug!("replacing {} with {}", text, link.new_text); debug!("replacing {} with {}", text, link.new_text);
*text = CowStr::Borrowed(&link.new_text); *text = CowStr::Borrowed(&link.new_text);
} }
@ -403,9 +405,12 @@ fn next(&mut self) -> Option<Self::Item> {
Some(Event::Text(text)) => { Some(Event::Text(text)) => {
trace!("saw text {}", text); trace!("saw text {}", text);
if let Some(link) = self.shortcut_link { if let Some(link) = self.shortcut_link {
trace!("original text was {}", link.original_text);
// NOTE: same limitations as `Event::Code` // NOTE: same limitations as `Event::Code`
if **text == *link.original_text { if let Some(link) = self
.links
.iter()
.find(|l| l.href == link.href && **text == *l.original_text)
{
debug!("replacing {} with {}", text, link.new_text); debug!("replacing {} with {}", text, link.new_text);
*text = CowStr::Borrowed(&link.new_text); *text = CowStr::Borrowed(&link.new_text);
} }

View File

@ -0,0 +1,37 @@
#![deny(rustdoc::broken_intra_doc_links)]
pub struct S;
pub mod char {}
// Ensure this doesn't ICE due to trying to slice off non-existent backticks from "S"
/// See [S] and [`S`]
pub struct MyStruct1;
// Ensure that link texts are replaced correctly even if there are multiple links with
// the same target but different text
/// See also [crate::char] and [mod@char] and [prim@char]
// @has issue_108459/struct.MyStruct2.html '//*[@href="char/index.html"]' 'crate::char'
// @has - '//*[@href="char/index.html"]' 'char'
// @has - '//*[@href="{{channel}}/std/primitive.char.html"]' 'char'
pub struct MyStruct2;
/// See also [mod@char] and [prim@char] and [crate::char]
// @has issue_108459/struct.MyStruct3.html '//*[@href="char/index.html"]' 'crate::char'
// @has - '//*[@href="char/index.html"]' 'char'
// @has - '//*[@href="{{channel}}/std/primitive.char.html"]' 'char'
pub struct MyStruct3;
// Ensure that links are correct even if there are multiple links with the same text but
// different targets
/// See also [char][mod@char] and [char][prim@char]
// @has issue_108459/struct.MyStruct4.html '//*[@href="char/index.html"]' 'char'
// @has - '//*[@href="{{channel}}/std/primitive.char.html"]' 'char'
pub struct MyStruct4;
/// See also [char][prim@char] and [char][crate::char]
// @has issue_108459/struct.MyStruct5.html '//*[@href="char/index.html"]' 'char'
// @has - '//*[@href="{{channel}}/std/primitive.char.html"]' 'char'
pub struct MyStruct5;

View File

@ -12,5 +12,5 @@ pub mod char {
/// See also [crate::char] and [mod@char] /// See also [crate::char] and [mod@char]
// @has prim_precedence/struct.MyString2.html '//*[@href="char/index.html"]' 'crate::char' // @has prim_precedence/struct.MyString2.html '//*[@href="char/index.html"]' 'crate::char'
// @has - '//*[@href="char/index.html"]' 'mod@char' // @has - '//*[@href="char/index.html"]' 'char'
pub struct MyString2; pub struct MyString2;