Properly fetch inner and outer docs on hir-level

This commit is contained in:
Lukas Wirth 2020-12-08 23:30:51 +01:00
parent eac77997bf
commit f8823e8cbc
2 changed files with 33 additions and 17 deletions

View file

@ -110,7 +110,17 @@ impl Attrs {
} }
pub(crate) fn new(owner: &dyn AttrsOwner, hygiene: &Hygiene) -> Attrs { pub(crate) fn new(owner: &dyn AttrsOwner, hygiene: &Hygiene) -> Attrs {
let outer_docs = ast::CommentIter::from_syntax_node(owner.syntax()).map(|docs_text| { let (inner_attrs, inner_docs) = inner_attributes(owner.syntax())
.map_or((None, None), |(attrs, docs)| ((Some(attrs), Some(docs))));
let outer_attrs = owner.attrs().filter(|attr| attr.excl_token().is_none());
let attrs = outer_attrs
.chain(inner_attrs.into_iter().flatten())
.map(|attr| (attr.syntax().text_range().start(), Attr::from_src(attr, hygiene)));
let outer_docs =
ast::CommentIter::from_syntax_node(owner.syntax()).filter(ast::Comment::is_outer);
let docs = outer_docs.chain(inner_docs.into_iter().flatten()).map(|docs_text| {
( (
docs_text.syntax().text_range().start(), docs_text.syntax().text_range().start(),
docs_text.doc_comment().map(|doc| Attr { docs_text.doc_comment().map(|doc| Attr {
@ -119,13 +129,8 @@ impl Attrs {
}), }),
) )
}); });
let outer_attrs = owner.attrs().filter(|attr| attr.excl_token().is_none());
let inner_attrs = inner_attributes(owner.syntax()).into_iter().flatten();
let attrs = outer_attrs
.chain(inner_attrs)
.map(|attr| (attr.syntax().text_range().start(), Attr::from_src(attr, hygiene)));
// sort here by syntax node offset because the source can have doc attributes and doc strings be interleaved // sort here by syntax node offset because the source can have doc attributes and doc strings be interleaved
let attrs: Vec<_> = outer_docs.chain(attrs).sorted_by_key(|&(offset, _)| offset).collect(); let attrs: Vec<_> = docs.chain(attrs).sorted_by_key(|&(offset, _)| offset).collect();
let entries = if attrs.is_empty() { let entries = if attrs.is_empty() {
// Avoid heap allocation // Avoid heap allocation
None None
@ -186,36 +191,39 @@ impl Attrs {
} }
} }
fn inner_attributes(syntax: &SyntaxNode) -> Option<impl Iterator<Item = ast::Attr>> { fn inner_attributes(
let (attrs, _docs) = match_ast! { syntax: &SyntaxNode,
) -> Option<(impl Iterator<Item = ast::Attr>, impl Iterator<Item = ast::Comment>)> {
let (attrs, docs) = match_ast! {
match syntax { match syntax {
ast::SourceFile(it) => (it.attrs(), None::<ast::Comment>), ast::SourceFile(it) => (it.attrs(), ast::CommentIter::from_syntax_node(it.syntax())),
ast::ExternBlock(it) => { ast::ExternBlock(it) => {
let extern_item_list = it.extern_item_list()?; let extern_item_list = it.extern_item_list()?;
(extern_item_list.attrs(), None) (extern_item_list.attrs(), ast::CommentIter::from_syntax_node(extern_item_list.syntax()))
}, },
ast::Fn(it) => { ast::Fn(it) => {
let body = it.body()?; let body = it.body()?;
(body.attrs(), None) (body.attrs(), ast::CommentIter::from_syntax_node(body.syntax()))
}, },
ast::Impl(it) => { ast::Impl(it) => {
let assoc_item_list = it.assoc_item_list()?; let assoc_item_list = it.assoc_item_list()?;
(assoc_item_list.attrs(), None) (assoc_item_list.attrs(), ast::CommentIter::from_syntax_node(assoc_item_list.syntax()))
}, },
ast::Module(it) => { ast::Module(it) => {
let item_list = it.item_list()?; let item_list = it.item_list()?;
(item_list.attrs(), None) (item_list.attrs(), ast::CommentIter::from_syntax_node(item_list.syntax()))
}, },
// FIXME: BlockExpr's only accept inner attributes in specific cases // FIXME: BlockExpr's only accept inner attributes in specific cases
// Excerpt from the reference: // Excerpt from the reference:
// Block expressions accept outer and inner attributes, but only when they are the outer // Block expressions accept outer and inner attributes, but only when they are the outer
// expression of an expression statement or the final expression of another block expression. // expression of an expression statement or the final expression of another block expression.
ast::BlockExpr(it) => return None, ast::BlockExpr(it) => return None,
_ => return None, _ => return None,
} }
}; };
let attrs = attrs.filter(|attr| attr.excl_token().is_some()); let attrs = attrs.filter(|attr| attr.excl_token().is_some());
Some(attrs) let docs = docs.filter(|doc| doc.is_inner());
Some((attrs, docs))
} }
#[derive(Debug, Clone, PartialEq, Eq)] #[derive(Debug, Clone, PartialEq, Eq)]

View file

@ -17,6 +17,14 @@ impl ast::Comment {
CommentKind::from_text(self.text()) CommentKind::from_text(self.text())
} }
pub fn is_inner(&self) -> bool {
self.kind().doc == Some(CommentPlacement::Inner)
}
pub fn is_outer(&self) -> bool {
self.kind().doc == Some(CommentPlacement::Outer)
}
pub fn prefix(&self) -> &'static str { pub fn prefix(&self) -> &'static str {
let &(prefix, _kind) = CommentKind::BY_PREFIX let &(prefix, _kind) = CommentKind::BY_PREFIX
.iter() .iter()