mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-27 20:35:09 +00:00
Better interface for doc comment and attribute processing
This commit is contained in:
parent
69dbfc7754
commit
08adce61a1
5 changed files with 66 additions and 64 deletions
|
@ -5,7 +5,7 @@ use std::{fmt, hash::Hash, ops, sync::Arc};
|
||||||
use base_db::CrateId;
|
use base_db::CrateId;
|
||||||
use cfg::{CfgExpr, CfgOptions};
|
use cfg::{CfgExpr, CfgOptions};
|
||||||
use either::Either;
|
use either::Either;
|
||||||
use hir_expand::{hygiene::Hygiene, name::AsName, AstId, InFile};
|
use hir_expand::{hygiene::Hygiene, name::AsName, AstId, HirFileId, InFile};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use la_arena::ArenaMap;
|
use la_arena::ArenaMap;
|
||||||
use mbe::{syntax_node_to_token_tree, DelimiterKind, Punct};
|
use mbe::{syntax_node_to_token_tree, DelimiterKind, Punct};
|
||||||
|
@ -84,6 +84,14 @@ impl ops::Deref for Attrs {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ops::Index<AttrId> for Attrs {
|
||||||
|
type Output = Attr;
|
||||||
|
|
||||||
|
fn index(&self, AttrId { ast_index, .. }: AttrId) -> &Self::Output {
|
||||||
|
&(**self)[ast_index as usize]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl ops::Deref for AttrsWithOwner {
|
impl ops::Deref for AttrsWithOwner {
|
||||||
type Target = Attrs;
|
type Target = Attrs;
|
||||||
|
|
||||||
|
@ -509,23 +517,23 @@ fn inner_attributes(
|
||||||
) -> Option<(impl Iterator<Item = ast::Attr>, impl Iterator<Item = ast::Comment>)> {
|
) -> Option<(impl Iterator<Item = ast::Attr>, impl Iterator<Item = ast::Comment>)> {
|
||||||
let (attrs, docs) = match_ast! {
|
let (attrs, docs) = match_ast! {
|
||||||
match syntax {
|
match syntax {
|
||||||
ast::SourceFile(it) => (it.attrs(), ast::CommentIter::from_syntax_node(it.syntax())),
|
ast::SourceFile(it) => (it.attrs(), ast::DocCommentIter::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(), ast::CommentIter::from_syntax_node(extern_item_list.syntax()))
|
(extern_item_list.attrs(), ast::DocCommentIter::from_syntax_node(extern_item_list.syntax()))
|
||||||
},
|
},
|
||||||
ast::Fn(it) => {
|
ast::Fn(it) => {
|
||||||
let body = it.body()?;
|
let body = it.body()?;
|
||||||
let stmt_list = body.stmt_list()?;
|
let stmt_list = body.stmt_list()?;
|
||||||
(stmt_list.attrs(), ast::CommentIter::from_syntax_node(body.syntax()))
|
(stmt_list.attrs(), ast::DocCommentIter::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(), ast::CommentIter::from_syntax_node(assoc_item_list.syntax()))
|
(assoc_item_list.attrs(), ast::DocCommentIter::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(), ast::CommentIter::from_syntax_node(item_list.syntax()))
|
(item_list.attrs(), ast::DocCommentIter::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:
|
||||||
|
@ -542,27 +550,20 @@ fn inner_attributes(
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct AttrSourceMap {
|
pub struct AttrSourceMap {
|
||||||
attrs: Vec<InFile<ast::Attr>>,
|
source: Vec<Either<ast::Attr, ast::Comment>>,
|
||||||
doc_comments: Vec<InFile<ast::Comment>>,
|
file_id: HirFileId,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AttrSourceMap {
|
impl AttrSourceMap {
|
||||||
fn new(owner: InFile<&dyn ast::HasAttrs>) -> Self {
|
fn new(owner: InFile<&dyn ast::HasAttrs>) -> Self {
|
||||||
let mut attrs = Vec::new();
|
Self {
|
||||||
let mut doc_comments = Vec::new();
|
source: collect_attrs(owner.value).map(|(_, it)| it).collect(),
|
||||||
for (_, attr) in collect_attrs(owner.value) {
|
file_id: owner.file_id,
|
||||||
match attr {
|
|
||||||
Either::Left(attr) => attrs.push(owner.with_value(attr)),
|
|
||||||
Either::Right(comment) => doc_comments.push(owner.with_value(comment)),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Self { attrs, doc_comments }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn merge(&mut self, other: Self) {
|
fn merge(&mut self, other: Self) {
|
||||||
self.attrs.extend(other.attrs);
|
self.source.extend(other.source);
|
||||||
self.doc_comments.extend(other.doc_comments);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Maps the lowered `Attr` back to its original syntax node.
|
/// Maps the lowered `Attr` back to its original syntax node.
|
||||||
|
@ -571,24 +572,15 @@ impl AttrSourceMap {
|
||||||
///
|
///
|
||||||
/// Note that the returned syntax node might be a `#[cfg_attr]`, or a doc comment, instead of
|
/// Note that the returned syntax node might be a `#[cfg_attr]`, or a doc comment, instead of
|
||||||
/// the attribute represented by `Attr`.
|
/// the attribute represented by `Attr`.
|
||||||
pub fn source_of(&self, attr: &Attr) -> InFile<Either<ast::Attr, ast::Comment>> {
|
pub fn source_of(&self, attr: &Attr) -> InFile<&Either<ast::Attr, ast::Comment>> {
|
||||||
self.source_of_id(attr.id)
|
self.source_of_id(attr.id)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn source_of_id(&self, id: AttrId) -> InFile<Either<ast::Attr, ast::Comment>> {
|
fn source_of_id(&self, id: AttrId) -> InFile<&Either<ast::Attr, ast::Comment>> {
|
||||||
if id.is_doc_comment {
|
self.source
|
||||||
self.doc_comments
|
.get(id.ast_index as usize)
|
||||||
.get(id.ast_index as usize)
|
.map(|it| InFile::new(self.file_id, it))
|
||||||
.unwrap_or_else(|| panic!("cannot find doc comment at index {:?}", id))
|
.unwrap_or_else(|| panic!("cannot find attr at index {:?}", id))
|
||||||
.clone()
|
|
||||||
.map(Either::Right)
|
|
||||||
} else {
|
|
||||||
self.attrs
|
|
||||||
.get(id.ast_index as usize)
|
|
||||||
.unwrap_or_else(|| panic!("cannot find `Attr` at index {:?}", id))
|
|
||||||
.clone()
|
|
||||||
.map(Either::Left)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -656,8 +648,7 @@ fn get_doc_string_in_attr(it: &ast::Attr) -> Option<ast::String> {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
||||||
pub(crate) struct AttrId {
|
pub struct AttrId {
|
||||||
is_doc_comment: bool,
|
|
||||||
pub(crate) ast_index: u32,
|
pub(crate) ast_index: u32,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -816,27 +807,20 @@ fn collect_attrs(
|
||||||
.map_or((None, None), |(attrs, docs)| (Some(attrs), Some(docs)));
|
.map_or((None, None), |(attrs, docs)| (Some(attrs), Some(docs)));
|
||||||
|
|
||||||
let outer_attrs = owner.attrs().filter(|attr| attr.kind().is_outer());
|
let outer_attrs = owner.attrs().filter(|attr| attr.kind().is_outer());
|
||||||
let attrs =
|
let attrs = outer_attrs
|
||||||
outer_attrs.chain(inner_attrs.into_iter().flatten()).enumerate().map(|(idx, attr)| {
|
.chain(inner_attrs.into_iter().flatten())
|
||||||
(
|
.map(|attr| (attr.syntax().text_range().start(), Either::Left(attr)));
|
||||||
AttrId { ast_index: idx as u32, is_doc_comment: false },
|
|
||||||
attr.syntax().text_range().start(),
|
|
||||||
Either::Left(attr),
|
|
||||||
)
|
|
||||||
});
|
|
||||||
|
|
||||||
let outer_docs =
|
let outer_docs =
|
||||||
ast::CommentIter::from_syntax_node(owner.syntax()).filter(ast::Comment::is_outer);
|
ast::DocCommentIter::from_syntax_node(owner.syntax()).filter(ast::Comment::is_outer);
|
||||||
let docs =
|
let docs = outer_docs
|
||||||
outer_docs.chain(inner_docs.into_iter().flatten()).enumerate().map(|(idx, docs_text)| {
|
.chain(inner_docs.into_iter().flatten())
|
||||||
(
|
.map(|docs_text| (docs_text.syntax().text_range().start(), Either::Right(docs_text)));
|
||||||
AttrId { ast_index: idx as u32, is_doc_comment: true },
|
|
||||||
docs_text.syntax().text_range().start(),
|
|
||||||
Either::Right(docs_text),
|
|
||||||
)
|
|
||||||
});
|
|
||||||
// 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
|
||||||
docs.chain(attrs).sorted_by_key(|&(_, offset, _)| offset).map(|(id, _, attr)| (id, attr))
|
docs.chain(attrs)
|
||||||
|
.sorted_by_key(|&(offset, _)| offset)
|
||||||
|
.enumerate()
|
||||||
|
.map(|(id, (_, attr))| (AttrId { ast_index: id as u32 }, attr))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn variants_attrs_source_map(
|
pub(crate) fn variants_attrs_source_map(
|
||||||
|
|
|
@ -30,7 +30,7 @@ pub use self::{
|
||||||
QuoteOffsets, Radix,
|
QuoteOffsets, Radix,
|
||||||
},
|
},
|
||||||
traits::{
|
traits::{
|
||||||
CommentIter, HasArgList, HasAttrs, HasDocComments, HasGenericParams, HasLoopBody,
|
DocCommentIter, HasArgList, HasAttrs, HasDocComments, HasGenericParams, HasLoopBody,
|
||||||
HasModuleItem, HasName, HasTypeBounds, HasVisibility,
|
HasModuleItem, HasName, HasTypeBounds, HasVisibility,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -772,3 +772,13 @@ impl ast::HasLoopBody for ast::ForExpr {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ast::HasAttrs for ast::AnyHasDocComments {}
|
impl ast::HasAttrs for ast::AnyHasDocComments {}
|
||||||
|
|
||||||
|
impl From<ast::Adt> for ast::Item {
|
||||||
|
fn from(it: ast::Adt) -> Self {
|
||||||
|
match it {
|
||||||
|
ast::Adt::Enum(it) => ast::Item::Enum(it),
|
||||||
|
ast::Adt::Struct(it) => ast::Item::Struct(it),
|
||||||
|
ast::Adt::Union(it) => ast::Item::Union(it),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -14,6 +14,10 @@ impl ast::Comment {
|
||||||
CommentKind::from_text(self.text())
|
CommentKind::from_text(self.text())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn is_doc(&self) -> bool {
|
||||||
|
self.kind().doc.is_some()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn is_inner(&self) -> bool {
|
pub fn is_inner(&self) -> bool {
|
||||||
self.kind().doc == Some(CommentPlacement::Inner)
|
self.kind().doc == Some(CommentPlacement::Inner)
|
||||||
}
|
}
|
||||||
|
|
|
@ -73,17 +73,17 @@ pub trait HasAttrs: AstNode {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait HasDocComments: HasAttrs {
|
pub trait HasDocComments: HasAttrs {
|
||||||
fn doc_comments(&self) -> CommentIter {
|
fn doc_comments(&self) -> DocCommentIter {
|
||||||
CommentIter { iter: self.syntax().children_with_tokens() }
|
DocCommentIter { iter: self.syntax().children_with_tokens() }
|
||||||
}
|
}
|
||||||
fn doc_comments_and_attrs(&self) -> AttrCommentIter {
|
fn doc_comments_and_attrs(&self) -> AttrCommentIter {
|
||||||
AttrCommentIter { iter: self.syntax().children_with_tokens() }
|
AttrCommentIter { iter: self.syntax().children_with_tokens() }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CommentIter {
|
impl DocCommentIter {
|
||||||
pub fn from_syntax_node(syntax_node: &ast::SyntaxNode) -> CommentIter {
|
pub fn from_syntax_node(syntax_node: &ast::SyntaxNode) -> DocCommentIter {
|
||||||
CommentIter { iter: syntax_node.children_with_tokens() }
|
DocCommentIter { iter: syntax_node.children_with_tokens() }
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
@ -100,14 +100,16 @@ impl CommentIter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct CommentIter {
|
pub struct DocCommentIter {
|
||||||
iter: SyntaxElementChildren,
|
iter: SyntaxElementChildren,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Iterator for CommentIter {
|
impl Iterator for DocCommentIter {
|
||||||
type Item = ast::Comment;
|
type Item = ast::Comment;
|
||||||
fn next(&mut self) -> Option<ast::Comment> {
|
fn next(&mut self) -> Option<ast::Comment> {
|
||||||
self.iter.by_ref().find_map(|el| el.into_token().and_then(ast::Comment::cast))
|
self.iter.by_ref().find_map(|el| {
|
||||||
|
el.into_token().and_then(ast::Comment::cast).filter(ast::Comment::is_doc)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -120,7 +122,9 @@ impl Iterator for AttrCommentIter {
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
self.iter.by_ref().find_map(|el| match el {
|
self.iter.by_ref().find_map(|el| match el {
|
||||||
SyntaxElement::Node(node) => ast::Attr::cast(node).map(Either::Right),
|
SyntaxElement::Node(node) => ast::Attr::cast(node).map(Either::Right),
|
||||||
SyntaxElement::Token(tok) => ast::Comment::cast(tok).map(Either::Left),
|
SyntaxElement::Token(tok) => {
|
||||||
|
ast::Comment::cast(tok).filter(ast::Comment::is_doc).map(Either::Left)
|
||||||
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue