ide/
doc_links.rs

1//! Extracts, resolves and rewrites links and intra-doc links in markdown documentation.
2
3#[cfg(test)]
4mod tests;
5
6mod intra_doc_links;
7
8use std::ops::Range;
9
10use pulldown_cmark::{BrokenLink, CowStr, Event, InlineStr, LinkType, Options, Parser, Tag};
11use pulldown_cmark_to_cmark::{Options as CMarkOptions, cmark_resume_with_options};
12use stdx::format_to;
13use url::Url;
14
15use hir::{
16    Adt, AsAssocItem, AssocItem, AssocItemContainer, AttrsWithOwner, HasAttrs, db::HirDatabase, sym,
17};
18use ide_db::{
19    RootDatabase,
20    base_db::{CrateOrigin, LangCrateOrigin, ReleaseChannel, RootQueryDb},
21    defs::{Definition, NameClass, NameRefClass},
22    documentation::{DocsRangeMap, Documentation, HasDocs, docs_with_rangemap},
23    helpers::pick_best_token,
24};
25use syntax::{
26    AstNode, AstToken,
27    SyntaxKind::*,
28    SyntaxNode, SyntaxToken, T, TextRange, TextSize,
29    ast::{self, IsString},
30    match_ast,
31};
32
33use crate::{
34    FilePosition, Semantics,
35    doc_links::intra_doc_links::{parse_intra_doc_link, strip_prefixes_suffixes},
36};
37
38/// Web and local links to an item's documentation.
39#[derive(Default, Debug, Clone, PartialEq, Eq)]
40pub struct DocumentationLinks {
41    /// The URL to the documentation on docs.rs.
42    /// May not lead anywhere.
43    pub web_url: Option<String>,
44    /// The URL to the documentation in the local file system.
45    /// May not lead anywhere.
46    pub local_url: Option<String>,
47}
48
49const MARKDOWN_OPTIONS: Options =
50    Options::ENABLE_FOOTNOTES.union(Options::ENABLE_TABLES).union(Options::ENABLE_TASKLISTS);
51
52/// Rewrite documentation links in markdown to point to an online host (e.g. docs.rs)
53pub(crate) fn rewrite_links(
54    db: &RootDatabase,
55    markdown: &str,
56    definition: Definition,
57    range_map: Option<DocsRangeMap>,
58) -> String {
59    let mut cb = broken_link_clone_cb;
60    let doc = Parser::new_with_broken_link_callback(markdown, MARKDOWN_OPTIONS, Some(&mut cb))
61        .into_offset_iter();
62
63    let doc = map_links(doc, |target, title, range, link_type| {
64        // This check is imperfect, there's some overlap between valid intra-doc links
65        // and valid URLs so we choose to be too eager to try to resolve what might be
66        // a URL.
67        if target.contains("://") {
68            (Some(LinkType::Inline), target.to_owned(), title.to_owned())
69        } else {
70            // Two possibilities:
71            // * path-based links: `../../module/struct.MyStruct.html`
72            // * module-based links (AKA intra-doc links): `super::super::module::MyStruct`
73            let text_range =
74                TextRange::new(range.start.try_into().unwrap(), range.end.try_into().unwrap());
75            let is_inner_doc = range_map
76                .as_ref()
77                .and_then(|range_map| range_map.map(text_range))
78                .map(|(_, attr_id)| attr_id.is_inner_attr())
79                .unwrap_or(false);
80            if let Some((target, title)) =
81                rewrite_intra_doc_link(db, definition, target, title, is_inner_doc, link_type)
82            {
83                (None, target, title)
84            } else if let Some(target) = rewrite_url_link(db, definition, target) {
85                (Some(LinkType::Inline), target, title.to_owned())
86            } else {
87                (None, target.to_owned(), title.to_owned())
88            }
89        }
90    });
91    let mut out = String::new();
92    cmark_resume_with_options(
93        doc,
94        &mut out,
95        None,
96        CMarkOptions { code_block_token_count: 3, ..Default::default() },
97    )
98    .ok();
99    out
100}
101
102/// Remove all links in markdown documentation.
103pub(crate) fn remove_links(markdown: &str) -> String {
104    let mut drop_link = false;
105
106    let mut cb = |_: BrokenLink<'_>| {
107        let empty = InlineStr::try_from("").unwrap();
108        Some((CowStr::Inlined(empty), CowStr::Inlined(empty)))
109    };
110    let doc = Parser::new_with_broken_link_callback(markdown, MARKDOWN_OPTIONS, Some(&mut cb));
111    let doc = doc.filter_map(move |evt| match evt {
112        Event::Start(Tag::Link(link_type, target, title)) => {
113            if link_type == LinkType::Inline && target.contains("://") {
114                Some(Event::Start(Tag::Link(link_type, target, title)))
115            } else {
116                drop_link = true;
117                None
118            }
119        }
120        Event::End(_) if drop_link => {
121            drop_link = false;
122            None
123        }
124        _ => Some(evt),
125    });
126
127    let mut out = String::new();
128    cmark_resume_with_options(
129        doc,
130        &mut out,
131        None,
132        CMarkOptions { code_block_token_count: 3, ..Default::default() },
133    )
134    .ok();
135    out
136}
137
138// Feature: Open Docs
139//
140// Retrieve a links to documentation for the given symbol.
141//
142// The simplest way to use this feature is via the context menu. Right-click on
143// the selected item. The context menu opens. Select **Open Docs**.
144//
145// | Editor  | Action Name |
146// |---------|-------------|
147// | VS Code | **rust-analyzer: Open Docs** |
148pub(crate) fn external_docs(
149    db: &RootDatabase,
150    FilePosition { file_id, offset }: FilePosition,
151    target_dir: Option<&str>,
152    sysroot: Option<&str>,
153) -> Option<DocumentationLinks> {
154    let sema = &Semantics::new(db);
155    let file = sema.parse_guess_edition(file_id).syntax().clone();
156    let token = pick_best_token(file.token_at_offset(offset), |kind| match kind {
157        IDENT | INT_NUMBER | T![self] => 3,
158        T!['('] | T![')'] => 2,
159        kind if kind.is_trivia() => 0,
160        _ => 1,
161    })?;
162    let token = sema.descend_into_macros_single_exact(token);
163
164    let node = token.parent()?;
165    let definition = match_ast! {
166        match node {
167            ast::NameRef(name_ref) => match NameRefClass::classify(sema, &name_ref)? {
168                NameRefClass::Definition(def, _) => def,
169                NameRefClass::FieldShorthand { local_ref: _, field_ref, adt_subst: _ } => {
170                    Definition::Field(field_ref)
171                }
172                NameRefClass::ExternCrateShorthand { decl, .. } => {
173                    Definition::ExternCrateDecl(decl)
174                }
175            },
176            ast::Name(name) => match NameClass::classify(sema, &name)? {
177                NameClass::Definition(it) | NameClass::ConstReference(it) => it,
178                NameClass::PatFieldShorthand { local_def: _, field_ref, adt_subst: _ } => Definition::Field(field_ref),
179            },
180            _ => return None
181        }
182    };
183
184    Some(get_doc_links(db, definition, target_dir, sysroot))
185}
186
187/// Extracts all links from a given markdown text returning the definition text range, link-text
188/// and the namespace if known.
189pub(crate) fn extract_definitions_from_docs(
190    docs: &Documentation,
191) -> Vec<(TextRange, String, Option<hir::Namespace>)> {
192    Parser::new_with_broken_link_callback(
193        docs.as_str(),
194        MARKDOWN_OPTIONS,
195        Some(&mut broken_link_clone_cb),
196    )
197    .into_offset_iter()
198    .filter_map(|(event, range)| match event {
199        Event::Start(Tag::Link(_, target, _)) => {
200            let (link, ns) = parse_intra_doc_link(&target);
201            Some((
202                TextRange::new(range.start.try_into().ok()?, range.end.try_into().ok()?),
203                link.to_owned(),
204                ns,
205            ))
206        }
207        _ => None,
208    })
209    .collect()
210}
211
212pub(crate) fn resolve_doc_path_for_def(
213    db: &dyn HirDatabase,
214    def: Definition,
215    link: &str,
216    ns: Option<hir::Namespace>,
217    is_inner_doc: bool,
218) -> Option<Definition> {
219    match def {
220        Definition::Module(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
221        Definition::Crate(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
222        Definition::Function(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
223        Definition::Adt(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
224        Definition::Variant(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
225        Definition::Const(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
226        Definition::Static(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
227        Definition::Trait(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
228        Definition::TypeAlias(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
229        Definition::Macro(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
230        Definition::Field(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
231        Definition::SelfType(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
232        Definition::ExternCrateDecl(it) => it.resolve_doc_path(db, link, ns, is_inner_doc),
233        Definition::BuiltinAttr(_)
234        | Definition::BuiltinType(_)
235        | Definition::BuiltinLifetime(_)
236        | Definition::ToolModule(_)
237        | Definition::TupleField(_)
238        | Definition::Local(_)
239        | Definition::GenericParam(_)
240        | Definition::Label(_)
241        | Definition::DeriveHelper(_)
242        | Definition::InlineAsmRegOrRegClass(_)
243        | Definition::InlineAsmOperand(_) => None,
244    }
245    .map(Definition::from)
246}
247
248pub(crate) fn doc_attributes(
249    sema: &Semantics<'_, RootDatabase>,
250    node: &SyntaxNode,
251) -> Option<(hir::AttrsWithOwner, Definition)> {
252    match_ast! {
253        match node {
254            ast::SourceFile(it)  => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
255            ast::Module(it)      => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
256            ast::Fn(it)          => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
257            ast::Struct(it)      => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(hir::Adt::Struct(def)))),
258            ast::Union(it)       => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(hir::Adt::Union(def)))),
259            ast::Enum(it)        => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(hir::Adt::Enum(def)))),
260            ast::Variant(it)     => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
261            ast::Trait(it)       => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
262            ast::Static(it)      => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
263            ast::Const(it)       => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
264            ast::TypeAlias(it)   => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
265            ast::Impl(it)        => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
266            ast::RecordField(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
267            ast::TupleField(it)  => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
268            ast::Macro(it)       => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
269            ast::ExternCrate(it) => sema.to_def(&it).map(|def| (def.attrs(sema.db), Definition::from(def))),
270            // ast::Use(it) => sema.to_def(&it).map(|def| (Box::new(it) as _, def.attrs(sema.db))),
271            _ => None
272        }
273    }
274}
275
276pub(crate) struct DocCommentToken {
277    doc_token: SyntaxToken,
278    prefix_len: TextSize,
279}
280
281pub(crate) fn token_as_doc_comment(doc_token: &SyntaxToken) -> Option<DocCommentToken> {
282    (match_ast! {
283        match doc_token {
284            ast::Comment(comment) => TextSize::try_from(comment.prefix().len()).ok(),
285            ast::String(string) => {
286                doc_token.parent_ancestors().find_map(ast::Attr::cast).filter(|attr| attr.simple_name().as_deref() == Some("doc"))?;
287                if doc_token.parent_ancestors().find_map(ast::MacroCall::cast).filter(|mac| mac.path().and_then(|p| p.segment()?.name_ref()).as_ref().map(|n| n.text()).as_deref() == Some("include_str")).is_some() {
288                    return None;
289                }
290                string.open_quote_text_range().map(|it| it.len())
291            },
292            _ => None,
293        }
294    }).map(|prefix_len| DocCommentToken { prefix_len, doc_token: doc_token.clone() })
295}
296
297impl DocCommentToken {
298    pub(crate) fn get_definition_with_descend_at<T>(
299        self,
300        sema: &Semantics<'_, RootDatabase>,
301        offset: TextSize,
302        // Definition, CommentOwner, range of intra doc link in original file
303        mut cb: impl FnMut(Definition, SyntaxNode, TextRange) -> Option<T>,
304    ) -> Option<T> {
305        let DocCommentToken { prefix_len, doc_token } = self;
306        // offset relative to the comments contents
307        let original_start = doc_token.text_range().start();
308        let relative_comment_offset = offset - original_start - prefix_len;
309
310        sema.descend_into_macros(doc_token).into_iter().find_map(|t| {
311            let (node, descended_prefix_len, is_inner) = match_ast!{
312                match t {
313                    ast::Comment(comment) => {
314                        (t.parent()?, TextSize::try_from(comment.prefix().len()).ok()?, comment.is_inner())
315                    },
316                    ast::String(string) => {
317                        let attr = t.parent_ancestors().find_map(ast::Attr::cast)?;
318                        let attr_is_inner = attr.excl_token().map(|excl| excl.kind() == BANG).unwrap_or(false);
319                        (attr.syntax().parent()?, string.open_quote_text_range()?.len(), attr_is_inner)
320                    },
321                    _ => return None,
322                }
323            };
324            let token_start = t.text_range().start();
325            let abs_in_expansion_offset = token_start + relative_comment_offset + descended_prefix_len;
326            let (attributes, def) = Self::doc_attributes(sema, &node, is_inner)?;
327            let (docs, doc_mapping) = docs_with_rangemap(sema.db, &attributes)?;
328            let (in_expansion_range, link, ns, is_inner) =
329                extract_definitions_from_docs(&docs).into_iter().find_map(|(range, link, ns)| {
330                    let (mapped, idx) = doc_mapping.map(range)?;
331                    (mapped.value.contains(abs_in_expansion_offset)).then_some((mapped.value, link, ns, idx.is_inner_attr()))
332                })?;
333            // get the relative range to the doc/attribute in the expansion
334            let in_expansion_relative_range = in_expansion_range - descended_prefix_len - token_start;
335            // Apply relative range to the original input comment
336            let absolute_range = in_expansion_relative_range + original_start + prefix_len;
337            let def = resolve_doc_path_for_def(sema.db, def, &link, ns, is_inner)?;
338            cb(def, node, absolute_range)
339        })
340    }
341
342    /// When we hover a inner doc item, this find a attached definition.
343    /// ```
344    /// // node == ITEM_LIST
345    /// // node.parent == EXPR_BLOCK
346    /// // node.parent().parent() == FN
347    /// fn f() {
348    ///    //! [`S$0`]
349    /// }
350    /// ```
351    fn doc_attributes(
352        sema: &Semantics<'_, RootDatabase>,
353        node: &SyntaxNode,
354        is_inner_doc: bool,
355    ) -> Option<(AttrsWithOwner, Definition)> {
356        if is_inner_doc && node.kind() != SOURCE_FILE {
357            let parent = node.parent()?;
358            doc_attributes(sema, &parent).or(doc_attributes(sema, &parent.parent()?))
359        } else {
360            doc_attributes(sema, node)
361        }
362    }
363}
364
365fn broken_link_clone_cb(link: BrokenLink<'_>) -> Option<(CowStr<'_>, CowStr<'_>)> {
366    Some((/*url*/ link.reference.clone(), /*title*/ link.reference))
367}
368
369// FIXME:
370// BUG: For Option::Some
371// Returns https://doc.rust-lang.org/nightly/core/prelude/v1/enum.Option.html#variant.Some
372// Instead of https://doc.rust-lang.org/nightly/core/option/enum.Option.html
373//
374// This should cease to be a problem if RFC2988 (Stable Rustdoc URLs) is implemented
375// https://github.com/rust-lang/rfcs/pull/2988
376fn get_doc_links(
377    db: &RootDatabase,
378    def: Definition,
379    target_dir: Option<&str>,
380    sysroot: Option<&str>,
381) -> DocumentationLinks {
382    let join_url = |base_url: Option<Url>, path: &str| -> Option<Url> {
383        base_url.and_then(|url| url.join(path).ok())
384    };
385
386    let Some((target, file, frag)) = filename_and_frag_for_def(db, def) else {
387        return Default::default();
388    };
389
390    let (mut web_url, mut local_url) = get_doc_base_urls(db, target, target_dir, sysroot);
391
392    let append_mod = !matches!(def, Definition::Macro(m) if m.is_macro_export(db));
393    if append_mod && let Some(path) = mod_path_of_def(db, target) {
394        web_url = join_url(web_url, &path);
395        local_url = join_url(local_url, &path);
396    }
397
398    web_url = join_url(web_url, &file);
399    local_url = join_url(local_url, &file);
400
401    if let Some(url) = web_url.as_mut() {
402        url.set_fragment(frag.as_deref())
403    }
404    if let Some(url) = local_url.as_mut() {
405        url.set_fragment(frag.as_deref())
406    }
407
408    DocumentationLinks {
409        web_url: web_url.map(|it| it.into()),
410        local_url: local_url.map(|it| it.into()),
411    }
412}
413
414fn rewrite_intra_doc_link(
415    db: &RootDatabase,
416    def: Definition,
417    target: &str,
418    title: &str,
419    is_inner_doc: bool,
420    link_type: LinkType,
421) -> Option<(String, String)> {
422    let (link, ns) = parse_intra_doc_link(target);
423
424    let (link, anchor) = match link.split_once('#') {
425        Some((new_link, anchor)) => (new_link, Some(anchor)),
426        None => (link, None),
427    };
428
429    let resolved = resolve_doc_path_for_def(db, def, link, ns, is_inner_doc)?;
430    let mut url = get_doc_base_urls(db, resolved, None, None).0?;
431
432    let (_, file, frag) = filename_and_frag_for_def(db, resolved)?;
433    if let Some(path) = mod_path_of_def(db, resolved) {
434        url = url.join(&path).ok()?;
435    }
436
437    let frag = anchor.or(frag.as_deref());
438
439    url = url.join(&file).ok()?;
440    url.set_fragment(frag);
441
442    // We want to strip the keyword prefix from the title, but only if the target is implicitly the same
443    // as the title.
444    let title = match link_type {
445        LinkType::Email
446        | LinkType::Autolink
447        | LinkType::Shortcut
448        | LinkType::Collapsed
449        | LinkType::Reference
450        | LinkType::Inline => title.to_owned(),
451        LinkType::ShortcutUnknown | LinkType::CollapsedUnknown | LinkType::ReferenceUnknown => {
452            strip_prefixes_suffixes(title).to_owned()
453        }
454    };
455
456    Some((url.into(), title))
457}
458
459/// Try to resolve path to local documentation via path-based links (i.e. `../gateway/struct.Shard.html`).
460fn rewrite_url_link(db: &RootDatabase, def: Definition, target: &str) -> Option<String> {
461    if !(target.contains('#') || target.contains(".html")) {
462        return None;
463    }
464
465    let mut url = get_doc_base_urls(db, def, None, None).0?;
466    let (def, file, frag) = filename_and_frag_for_def(db, def)?;
467
468    if let Some(path) = mod_path_of_def(db, def) {
469        url = url.join(&path).ok()?;
470    }
471
472    url = url.join(&file).ok()?;
473    url.set_fragment(frag.as_deref());
474    url.join(target).ok().map(Into::into)
475}
476
477fn mod_path_of_def(db: &RootDatabase, def: Definition) -> Option<String> {
478    def.canonical_module_path(db).map(|it| {
479        let mut path = String::new();
480        it.flat_map(|it| it.name(db)).for_each(|name| format_to!(path, "{}/", name.as_str()));
481        path
482    })
483}
484
485/// Rewrites a markdown document, applying 'callback' to each link.
486fn map_links<'e>(
487    events: impl Iterator<Item = (Event<'e>, Range<usize>)>,
488    callback: impl Fn(&str, &str, Range<usize>, LinkType) -> (Option<LinkType>, String, String),
489) -> impl Iterator<Item = Event<'e>> {
490    let mut in_link = false;
491    // holds the origin link target on start event and the rewritten one on end event
492    let mut end_link_target: Option<CowStr<'_>> = None;
493    // normally link's type is determined by the type of link tag in the end event,
494    // however in some cases we want to change the link type, for example,
495    // `Shortcut` type parsed from Start/End tags doesn't make sense for url links
496    let mut end_link_type: Option<LinkType> = None;
497
498    events.map(move |(evt, range)| match evt {
499        Event::Start(Tag::Link(link_type, ref target, _)) => {
500            in_link = true;
501            end_link_target = Some(target.clone());
502            end_link_type = Some(link_type);
503            evt
504        }
505        Event::End(Tag::Link(link_type, target, _)) => {
506            in_link = false;
507            Event::End(Tag::Link(
508                end_link_type.take().unwrap_or(link_type),
509                end_link_target.take().unwrap_or(target),
510                CowStr::Borrowed(""),
511            ))
512        }
513        Event::Text(s) if in_link => {
514            let (link_type, link_target_s, link_name) =
515                callback(&end_link_target.take().unwrap(), &s, range, end_link_type.unwrap());
516            end_link_target = Some(CowStr::Boxed(link_target_s.into()));
517            if !matches!(end_link_type, Some(LinkType::Autolink)) && link_type.is_some() {
518                end_link_type = link_type;
519            }
520            Event::Text(CowStr::Boxed(link_name.into()))
521        }
522        Event::Code(s) if in_link => {
523            let (link_type, link_target_s, link_name) =
524                callback(&end_link_target.take().unwrap(), &s, range, end_link_type.unwrap());
525            end_link_target = Some(CowStr::Boxed(link_target_s.into()));
526            if !matches!(end_link_type, Some(LinkType::Autolink)) && link_type.is_some() {
527                end_link_type = link_type;
528            }
529            Event::Code(CowStr::Boxed(link_name.into()))
530        }
531        _ => evt,
532    })
533}
534
535/// Get the root URL for the documentation of a definition.
536///
537/// ```ignore
538/// https://doc.rust-lang.org/std/iter/trait.Iterator.html#tymethod.next
539/// ^^^^^^^^^^^^^^^^^^^^^^^^^^
540/// file:///project/root/target/doc/std/iter/trait.Iterator.html#tymethod.next
541/// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
542/// ```
543fn get_doc_base_urls(
544    db: &RootDatabase,
545    def: Definition,
546    target_dir: Option<&str>,
547    sysroot: Option<&str>,
548) -> (Option<Url>, Option<Url>) {
549    let local_doc = target_dir
550        .and_then(|path| Url::parse(&format!("file:///{path}/")).ok())
551        .and_then(|it| it.join("doc/").ok());
552    let system_doc = sysroot
553        .map(|sysroot| format!("file:///{sysroot}/share/doc/rust/html/"))
554        .and_then(|it| Url::parse(&it).ok());
555    let krate = def.krate(db);
556    let channel = krate
557        .and_then(|krate| db.toolchain_channel(krate.into()))
558        .unwrap_or(ReleaseChannel::Nightly)
559        .as_str();
560
561    // special case base url of `BuiltinType` to core
562    // https://github.com/rust-lang/rust-analyzer/issues/12250
563    if let Definition::BuiltinType(..) = def {
564        let web_link = Url::parse(&format!("https://doc.rust-lang.org/{channel}/core/")).ok();
565        let system_link = system_doc.and_then(|it| it.join("core/").ok());
566        return (web_link, system_link);
567    };
568
569    let Some(krate) = krate else { return Default::default() };
570    let Some(display_name) = krate.display_name(db) else { return Default::default() };
571    let (web_base, local_base) = match krate.origin(db) {
572        // std and co do not specify `html_root_url` any longer so we gotta handwrite this ourself.
573        // FIXME: Use the toolchains channel instead of nightly
574        CrateOrigin::Lang(
575            origin @ (LangCrateOrigin::Alloc
576            | LangCrateOrigin::Core
577            | LangCrateOrigin::ProcMacro
578            | LangCrateOrigin::Std
579            | LangCrateOrigin::Test),
580        ) => {
581            let system_url = system_doc.and_then(|it| it.join(&format!("{origin}")).ok());
582            let web_url = format!("https://doc.rust-lang.org/{channel}/{origin}");
583            (Some(web_url), system_url)
584        }
585        CrateOrigin::Lang(_) => return (None, None),
586        CrateOrigin::Rustc { name: _ } => {
587            (Some(format!("https://doc.rust-lang.org/{channel}/nightly-rustc/")), None)
588        }
589        CrateOrigin::Local { repo: _, name: _ } => {
590            // FIXME: These should not attempt to link to docs.rs!
591            let weblink = krate.get_html_root_url(db).or_else(|| {
592                let version = krate.version(db);
593                // Fallback to docs.rs. This uses `display_name` and can never be
594                // correct, but that's what fallbacks are about.
595                //
596                // FIXME: clicking on the link should just open the file in the editor,
597                // instead of falling back to external urls.
598                Some(format!(
599                    "https://docs.rs/{krate}/{version}/",
600                    krate = display_name,
601                    version = version.as_deref().unwrap_or("*")
602                ))
603            });
604            (weblink, local_doc)
605        }
606        CrateOrigin::Library { repo: _, name } => {
607            let weblink = krate.get_html_root_url(db).or_else(|| {
608                let version = krate.version(db);
609                // Fallback to docs.rs. This uses `display_name` and can never be
610                // correct, but that's what fallbacks are about.
611                //
612                // FIXME: clicking on the link should just open the file in the editor,
613                // instead of falling back to external urls.
614                Some(format!(
615                    "https://docs.rs/{krate}/{version}/",
616                    krate = name,
617                    version = version.as_deref().unwrap_or("*")
618                ))
619            });
620            (weblink, local_doc)
621        }
622    };
623    let web_base = web_base
624        .and_then(|it| Url::parse(&it).ok())
625        .and_then(|it| it.join(&format!("{display_name}/")).ok());
626    let local_base = local_base.and_then(|it| it.join(&format!("{display_name}/")).ok());
627
628    (web_base, local_base)
629}
630
631/// Get the filename and extension generated for a symbol by rustdoc.
632///
633/// ```ignore
634/// https://doc.rust-lang.org/std/iter/trait.Iterator.html#tymethod.next
635///                                    ^^^^^^^^^^^^^^^^^^^
636/// ```
637fn filename_and_frag_for_def(
638    db: &dyn HirDatabase,
639    def: Definition,
640) -> Option<(Definition, String, Option<String>)> {
641    if let Some(assoc_item) = def.as_assoc_item(db) {
642        let def = match assoc_item.container(db) {
643            AssocItemContainer::Trait(t) => t.into(),
644            AssocItemContainer::Impl(i) => i.self_ty(db).as_adt()?.into(),
645        };
646        let (_, file, _) = filename_and_frag_for_def(db, def)?;
647        let frag = get_assoc_item_fragment(db, assoc_item)?;
648        return Some((def, file, Some(frag)));
649    }
650
651    let res = match def {
652        Definition::Adt(adt) => match adt {
653            Adt::Struct(s) => {
654                format!("struct.{}.html", s.name(db).as_str())
655            }
656            Adt::Enum(e) => format!("enum.{}.html", e.name(db).as_str()),
657            Adt::Union(u) => format!("union.{}.html", u.name(db).as_str()),
658        },
659        Definition::Crate(_) => String::from("index.html"),
660        Definition::Module(m) => match m.name(db) {
661            // `#[doc(keyword = "...")]` is internal used only by rust compiler
662            Some(name) => {
663                match m.attrs(db).by_key(sym::doc).find_string_value_in_tt(sym::keyword) {
664                    Some(kw) => {
665                        format!("keyword.{kw}.html")
666                    }
667                    None => format!("{}/index.html", name.as_str()),
668                }
669            }
670            None => String::from("index.html"),
671        },
672        Definition::Trait(t) => {
673            // FIXME(trait-alias): url should be traitalias. for aliases
674            format!("trait.{}.html", t.name(db).as_str())
675        }
676        Definition::TypeAlias(t) => {
677            format!("type.{}.html", t.name(db).as_str())
678        }
679        Definition::BuiltinType(t) => {
680            format!("primitive.{}.html", t.name().as_str())
681        }
682        Definition::Function(f) => {
683            format!("fn.{}.html", f.name(db).as_str())
684        }
685        Definition::Variant(ev) => {
686            let def = Definition::Adt(ev.parent_enum(db).into());
687            let (_, file, _) = filename_and_frag_for_def(db, def)?;
688            return Some((def, file, Some(format!("variant.{}", ev.name(db).as_str()))));
689        }
690        Definition::Const(c) => {
691            format!("constant.{}.html", c.name(db)?.as_str())
692        }
693        Definition::Static(s) => {
694            format!("static.{}.html", s.name(db).as_str())
695        }
696        Definition::Macro(mac) => match mac.kind(db) {
697            hir::MacroKind::Declarative
698            | hir::MacroKind::AttrBuiltIn
699            | hir::MacroKind::DeclarativeBuiltIn
700            | hir::MacroKind::Attr
701            | hir::MacroKind::ProcMacro => {
702                format!("macro.{}.html", mac.name(db).as_str())
703            }
704            hir::MacroKind::Derive | hir::MacroKind::DeriveBuiltIn => {
705                format!("derive.{}.html", mac.name(db).as_str())
706            }
707        },
708        Definition::Field(field) => {
709            let def = match field.parent_def(db) {
710                hir::VariantDef::Struct(it) => Definition::Adt(it.into()),
711                hir::VariantDef::Union(it) => Definition::Adt(it.into()),
712                hir::VariantDef::Variant(it) => Definition::Variant(it),
713            };
714            let (_, file, _) = filename_and_frag_for_def(db, def)?;
715            return Some((def, file, Some(format!("structfield.{}", field.name(db).as_str()))));
716        }
717        Definition::SelfType(impl_) => {
718            let adt = impl_.self_ty(db).as_adt()?.into();
719            let (_, file, _) = filename_and_frag_for_def(db, adt)?;
720            // FIXME fragment numbering
721            return Some((adt, file, Some(String::from("impl"))));
722        }
723        Definition::ExternCrateDecl(it) => {
724            format!("{}/index.html", it.name(db).as_str())
725        }
726        Definition::Local(_)
727        | Definition::GenericParam(_)
728        | Definition::TupleField(_)
729        | Definition::Label(_)
730        | Definition::BuiltinAttr(_)
731        | Definition::BuiltinLifetime(_)
732        | Definition::ToolModule(_)
733        | Definition::DeriveHelper(_)
734        | Definition::InlineAsmRegOrRegClass(_)
735        | Definition::InlineAsmOperand(_) => return None,
736    };
737
738    Some((def, res, None))
739}
740
741/// Get the fragment required to link to a specific field, method, associated type, or associated constant.
742///
743/// ```ignore
744/// https://doc.rust-lang.org/std/iter/trait.Iterator.html#tymethod.next
745///                                                       ^^^^^^^^^^^^^^
746/// ```
747fn get_assoc_item_fragment(db: &dyn HirDatabase, assoc_item: hir::AssocItem) -> Option<String> {
748    Some(match assoc_item {
749        AssocItem::Function(function) => {
750            let is_trait_method =
751                function.as_assoc_item(db).and_then(|assoc| assoc.container_trait(db)).is_some();
752            // This distinction may get more complicated when specialization is available.
753            // Rustdoc makes this decision based on whether a method 'has defaultness'.
754            // Currently this is only the case for provided trait methods.
755            if is_trait_method && !function.has_body(db) {
756                format!("tymethod.{}", function.name(db).as_str())
757            } else {
758                format!("method.{}", function.name(db).as_str())
759            }
760        }
761        AssocItem::Const(constant) => {
762            format!("associatedconstant.{}", constant.name(db)?.as_str())
763        }
764        AssocItem::TypeAlias(ty) => {
765            format!("associatedtype.{}", ty.name(db).as_str())
766        }
767    })
768}