2019-02-21 12:24:42 +00:00
|
|
|
//! Abstract Syntax Tree, layered on top of untyped `SyntaxNode`s
|
2018-08-09 14:43:39 +00:00
|
|
|
mod generated;
|
2019-04-02 07:03:19 +00:00
|
|
|
mod traits;
|
2018-08-09 14:43:39 +00:00
|
|
|
|
2018-09-07 22:16:07 +00:00
|
|
|
use std::marker::PhantomData;
|
|
|
|
|
2018-08-16 09:51:40 +00:00
|
|
|
use itertools::Itertools;
|
2018-08-13 11:24:22 +00:00
|
|
|
|
2018-10-15 16:55:32 +00:00
|
|
|
use crate::{
|
2019-04-02 07:03:19 +00:00
|
|
|
syntax_node::{SyntaxNode, SyntaxNodeChildren, TreeArc, RaTypes, SyntaxToken, SyntaxElement},
|
2018-10-15 21:44:23 +00:00
|
|
|
SmolStr,
|
|
|
|
SyntaxKind::*,
|
2018-08-09 13:03:21 +00:00
|
|
|
};
|
2018-07-30 18:58:49 +00:00
|
|
|
|
2019-04-02 07:03:19 +00:00
|
|
|
pub use self::{
|
|
|
|
generated::*,
|
|
|
|
traits::*,
|
|
|
|
};
|
|
|
|
|
2018-11-06 19:06:58 +00:00
|
|
|
/// The main trait to go from untyped `SyntaxNode` to a typed ast. The
|
|
|
|
/// conversion itself has zero runtime cost: ast and syntax nodes have exactly
|
|
|
|
/// the same representation: a pointer to the tree root and a pointer to the
|
|
|
|
/// node itself.
|
2019-01-24 23:09:31 +00:00
|
|
|
pub trait AstNode:
|
|
|
|
rowan::TransparentNewType<Repr = rowan::SyntaxNode<RaTypes>> + ToOwned<Owned = TreeArc<Self>>
|
|
|
|
{
|
2019-01-07 13:15:47 +00:00
|
|
|
fn cast(syntax: &SyntaxNode) -> Option<&Self>
|
2018-10-15 21:44:23 +00:00
|
|
|
where
|
|
|
|
Self: Sized;
|
2019-01-07 13:15:47 +00:00
|
|
|
fn syntax(&self) -> &SyntaxNode;
|
2018-08-09 13:03:21 +00:00
|
|
|
}
|
|
|
|
|
2019-04-02 07:09:52 +00:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct AstChildren<'a, N> {
|
|
|
|
inner: SyntaxNodeChildren<'a>,
|
|
|
|
ph: PhantomData<N>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, N> AstChildren<'a, N> {
|
|
|
|
fn new(parent: &'a SyntaxNode) -> Self {
|
|
|
|
AstChildren { inner: parent.children(), ph: PhantomData }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, N: AstNode + 'a> Iterator for AstChildren<'a, N> {
|
|
|
|
type Item = &'a N;
|
|
|
|
fn next(&mut self) -> Option<&'a N> {
|
|
|
|
self.inner.by_ref().find_map(N::cast)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl Attr {
|
2019-02-17 17:08:34 +00:00
|
|
|
pub fn is_inner(&self) -> bool {
|
|
|
|
let tt = match self.value() {
|
|
|
|
None => return false,
|
|
|
|
Some(tt) => tt,
|
|
|
|
};
|
|
|
|
|
|
|
|
let prev = match tt.syntax().prev_sibling() {
|
|
|
|
None => return false,
|
|
|
|
Some(prev) => prev,
|
|
|
|
};
|
|
|
|
|
|
|
|
prev.kind() == EXCL
|
|
|
|
}
|
|
|
|
|
2018-08-16 10:11:20 +00:00
|
|
|
pub fn as_atom(&self) -> Option<SmolStr> {
|
|
|
|
let tt = self.value()?;
|
2019-03-30 10:25:53 +00:00
|
|
|
let (_bra, attr, _ket) = tt.syntax().children_with_tokens().collect_tuple()?;
|
2018-08-16 10:11:20 +00:00
|
|
|
if attr.kind() == IDENT {
|
2019-03-30 10:25:53 +00:00
|
|
|
Some(attr.as_token()?.text().clone())
|
2018-08-16 10:11:20 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
pub fn as_call(&self) -> Option<(SmolStr, &TokenTree)> {
|
2018-08-16 10:11:20 +00:00
|
|
|
let tt = self.value()?;
|
2019-03-30 10:25:53 +00:00
|
|
|
let (_bra, attr, args, _ket) = tt.syntax().children_with_tokens().collect_tuple()?;
|
|
|
|
let args = TokenTree::cast(args.as_node()?)?;
|
2018-08-16 10:11:20 +00:00
|
|
|
if attr.kind() == IDENT {
|
2019-03-30 10:25:53 +00:00
|
|
|
Some((attr.as_token()?.text().clone(), args))
|
2018-08-16 10:11:20 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2018-08-09 13:03:21 +00:00
|
|
|
}
|
2019-02-05 22:05:46 +00:00
|
|
|
|
|
|
|
pub fn as_named(&self) -> Option<SmolStr> {
|
|
|
|
let tt = self.value()?;
|
2019-03-30 10:25:53 +00:00
|
|
|
let attr = tt.syntax().children_with_tokens().nth(1)?;
|
2019-02-05 22:05:46 +00:00
|
|
|
if attr.kind() == IDENT {
|
2019-03-30 10:25:53 +00:00
|
|
|
Some(attr.as_token()?.text().clone())
|
2019-02-05 22:05:46 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
2018-08-09 13:03:21 +00:00
|
|
|
}
|
|
|
|
|
2019-03-30 10:25:53 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct Comment<'a>(SyntaxToken<'a>);
|
|
|
|
|
|
|
|
impl<'a> Comment<'a> {
|
|
|
|
pub fn cast(token: SyntaxToken<'a>) -> Option<Self> {
|
|
|
|
if token.kind() == COMMENT {
|
|
|
|
Some(Comment(token))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn syntax(&self) -> SyntaxToken<'a> {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn text(&self) -> &'a SmolStr {
|
|
|
|
self.0.text()
|
|
|
|
}
|
|
|
|
|
2018-10-11 14:25:35 +00:00
|
|
|
pub fn flavor(&self) -> CommentFlavor {
|
|
|
|
let text = self.text();
|
|
|
|
if text.starts_with("///") {
|
|
|
|
CommentFlavor::Doc
|
|
|
|
} else if text.starts_with("//!") {
|
|
|
|
CommentFlavor::ModuleDoc
|
|
|
|
} else if text.starts_with("//") {
|
|
|
|
CommentFlavor::Line
|
|
|
|
} else {
|
|
|
|
CommentFlavor::Multiline
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-04 13:29:00 +00:00
|
|
|
pub fn is_doc_comment(&self) -> bool {
|
|
|
|
self.flavor().is_doc_comment()
|
|
|
|
}
|
|
|
|
|
2018-10-11 14:25:35 +00:00
|
|
|
pub fn prefix(&self) -> &'static str {
|
|
|
|
self.flavor().prefix()
|
|
|
|
}
|
2019-03-30 10:25:53 +00:00
|
|
|
}
|
2018-10-12 17:20:58 +00:00
|
|
|
|
2018-10-12 17:49:08 +00:00
|
|
|
#[derive(Debug, PartialEq, Eq)]
|
2018-10-11 14:25:35 +00:00
|
|
|
pub enum CommentFlavor {
|
|
|
|
Line,
|
|
|
|
Doc,
|
|
|
|
ModuleDoc,
|
2018-10-15 21:44:23 +00:00
|
|
|
Multiline,
|
2018-10-11 14:25:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl CommentFlavor {
|
|
|
|
pub fn prefix(&self) -> &'static str {
|
|
|
|
use self::CommentFlavor::*;
|
|
|
|
match *self {
|
|
|
|
Line => "//",
|
|
|
|
Doc => "///",
|
|
|
|
ModuleDoc => "//!",
|
2018-10-15 21:44:23 +00:00
|
|
|
Multiline => "/*",
|
2018-10-11 14:25:35 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-04 13:29:00 +00:00
|
|
|
|
|
|
|
pub fn is_doc_comment(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
CommentFlavor::Doc | CommentFlavor::ModuleDoc => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
2018-10-11 14:25:35 +00:00
|
|
|
}
|
|
|
|
|
2019-03-30 10:25:53 +00:00
|
|
|
pub struct Whitespace<'a>(SyntaxToken<'a>);
|
|
|
|
|
|
|
|
impl<'a> Whitespace<'a> {
|
|
|
|
pub fn cast(token: SyntaxToken<'a>) -> Option<Self> {
|
|
|
|
if token.kind() == WHITESPACE {
|
|
|
|
Some(Whitespace(token))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn syntax(&self) -> SyntaxToken<'a> {
|
|
|
|
self.0
|
2018-10-12 17:20:58 +00:00
|
|
|
}
|
|
|
|
|
2019-03-30 10:25:53 +00:00
|
|
|
pub fn text(&self) -> &'a SmolStr {
|
|
|
|
self.0.text()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn spans_multiple_lines(&self) -> bool {
|
|
|
|
let text = self.text();
|
|
|
|
text.find('\n').map_or(false, |idx| text[idx + 1..].contains('\n'))
|
2018-10-12 17:20:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl Name {
|
2019-01-08 09:23:10 +00:00
|
|
|
pub fn text(&self) -> &SmolStr {
|
2019-03-30 10:25:53 +00:00
|
|
|
let ident = self.syntax().first_child_or_token().unwrap().as_token().unwrap();
|
|
|
|
ident.text()
|
2018-07-30 18:58:49 +00:00
|
|
|
}
|
|
|
|
}
|
2018-08-13 13:35:17 +00:00
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl NameRef {
|
2019-01-08 09:23:10 +00:00
|
|
|
pub fn text(&self) -> &SmolStr {
|
2019-03-30 10:25:53 +00:00
|
|
|
let ident = self.syntax().first_child_or_token().unwrap().as_token().unwrap();
|
|
|
|
ident.text()
|
2018-08-13 13:35:17 +00:00
|
|
|
}
|
|
|
|
}
|
2018-08-14 09:38:20 +00:00
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl ImplBlock {
|
|
|
|
pub fn target_type(&self) -> Option<&TypeRef> {
|
2018-08-14 09:38:20 +00:00
|
|
|
match self.target() {
|
|
|
|
(Some(t), None) | (_, Some(t)) => Some(t),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
pub fn target_trait(&self) -> Option<&TypeRef> {
|
2018-08-14 09:38:20 +00:00
|
|
|
match self.target() {
|
|
|
|
(Some(t), Some(_)) => Some(t),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
fn target(&self) -> (Option<&TypeRef>, Option<&TypeRef>) {
|
2018-08-22 14:01:51 +00:00
|
|
|
let mut types = children(self);
|
2018-08-14 09:38:20 +00:00
|
|
|
let first = types.next();
|
|
|
|
let second = types.next();
|
|
|
|
(first, second)
|
|
|
|
}
|
|
|
|
}
|
2018-08-17 12:37:17 +00:00
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl Module {
|
|
|
|
pub fn has_semi(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
match self.syntax().last_child_or_token() {
|
2018-08-17 12:37:17 +00:00
|
|
|
None => false,
|
|
|
|
Some(node) => node.kind() == SEMI,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-08-22 14:01:51 +00:00
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl LetStmt {
|
|
|
|
pub fn has_semi(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
match self.syntax().last_child_or_token() {
|
2018-08-28 08:12:42 +00:00
|
|
|
None => false,
|
|
|
|
Some(node) => node.kind() == SEMI,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-26 21:23:07 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
|
|
pub enum ElseBranchFlavor<'a> {
|
|
|
|
Block(&'a Block),
|
|
|
|
IfExpr(&'a IfExpr),
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl IfExpr {
|
|
|
|
pub fn then_branch(&self) -> Option<&Block> {
|
2018-08-27 09:22:09 +00:00
|
|
|
self.blocks().nth(0)
|
|
|
|
}
|
2019-01-26 21:23:07 +00:00
|
|
|
pub fn else_branch(&self) -> Option<ElseBranchFlavor> {
|
|
|
|
let res = match self.blocks().nth(1) {
|
|
|
|
Some(block) => ElseBranchFlavor::Block(block),
|
|
|
|
None => {
|
|
|
|
let elif: &IfExpr = child_opt(self)?;
|
|
|
|
ElseBranchFlavor::IfExpr(elif)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Some(res)
|
2018-08-27 09:22:09 +00:00
|
|
|
}
|
2019-01-26 21:23:07 +00:00
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
fn blocks(&self) -> AstChildren<Block> {
|
2018-08-27 09:22:09 +00:00
|
|
|
children(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-16 04:26:46 +00:00
|
|
|
impl ExprStmt {
|
|
|
|
pub fn has_semi(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
match self.syntax().last_child_or_token() {
|
2019-01-16 04:26:46 +00:00
|
|
|
None => false,
|
|
|
|
Some(node) => node.kind() == SEMI,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-21 22:29:59 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
2018-10-24 15:37:25 +00:00
|
|
|
pub enum PathSegmentKind<'a> {
|
2019-01-07 13:15:47 +00:00
|
|
|
Name(&'a NameRef),
|
2018-10-24 15:37:25 +00:00
|
|
|
SelfKw,
|
|
|
|
SuperKw,
|
|
|
|
CrateKw,
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl PathSegment {
|
|
|
|
pub fn parent_path(&self) -> &Path {
|
2019-02-08 11:49:43 +00:00
|
|
|
self.syntax().parent().and_then(Path::cast).expect("segments are always nested in paths")
|
2018-10-24 15:37:25 +00:00
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
pub fn kind(&self) -> Option<PathSegmentKind> {
|
2018-10-24 15:37:25 +00:00
|
|
|
let res = if let Some(name_ref) = self.name_ref() {
|
|
|
|
PathSegmentKind::Name(name_ref)
|
|
|
|
} else {
|
2019-03-30 10:25:53 +00:00
|
|
|
match self.syntax().first_child_or_token()?.kind() {
|
2018-10-24 15:37:25 +00:00
|
|
|
SELF_KW => PathSegmentKind::SelfKw,
|
|
|
|
SUPER_KW => PathSegmentKind::SuperKw,
|
|
|
|
CRATE_KW => PathSegmentKind::CrateKw,
|
|
|
|
_ => return None,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
Some(res)
|
|
|
|
}
|
2019-01-23 05:21:29 +00:00
|
|
|
|
|
|
|
pub fn has_colon_colon(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
match self.syntax.first_child_or_token().map(|s| s.kind()) {
|
2019-01-23 05:21:29 +00:00
|
|
|
Some(COLONCOLON) => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
2018-10-24 15:37:25 +00:00
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl Path {
|
|
|
|
pub fn parent_path(&self) -> Option<&Path> {
|
2019-01-05 10:45:18 +00:00
|
|
|
self.syntax().parent().and_then(Path::cast)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl UseTree {
|
|
|
|
pub fn has_star(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
self.syntax().children_with_tokens().any(|it| it.kind() == STAR)
|
2018-11-20 16:24:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl UseTreeList {
|
|
|
|
pub fn parent_use_tree(&self) -> &UseTree {
|
2018-11-07 18:38:41 +00:00
|
|
|
self.syntax()
|
|
|
|
.parent()
|
|
|
|
.and_then(UseTree::cast)
|
|
|
|
.expect("UseTreeLists are always nested in UseTrees")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-13 10:34:57 +00:00
|
|
|
impl RefPat {
|
|
|
|
pub fn is_mut(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
self.syntax().children_with_tokens().any(|n| n.kind() == MUT_KW)
|
2019-01-13 10:34:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
fn child_opt<P: AstNode, C: AstNode>(parent: &P) -> Option<&C> {
|
2018-08-22 14:01:51 +00:00
|
|
|
children(parent).next()
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
fn children<P: AstNode, C: AstNode>(parent: &P) -> AstChildren<C> {
|
2018-09-07 22:35:20 +00:00
|
|
|
AstChildren::new(parent.syntax())
|
2018-09-07 22:16:07 +00:00
|
|
|
}
|
|
|
|
|
2018-12-25 12:01:47 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
|
|
pub enum StructFlavor<'a> {
|
2019-01-25 21:24:12 +00:00
|
|
|
Tuple(&'a PosFieldDefList),
|
2019-01-07 13:15:47 +00:00
|
|
|
Named(&'a NamedFieldDefList),
|
2018-12-25 12:01:47 +00:00
|
|
|
Unit,
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl StructFlavor<'_> {
|
|
|
|
fn from_node<N: AstNode>(node: &N) -> StructFlavor {
|
2018-12-25 12:01:47 +00:00
|
|
|
if let Some(nfdl) = child_opt::<_, NamedFieldDefList>(node) {
|
|
|
|
StructFlavor::Named(nfdl)
|
2019-01-25 21:24:12 +00:00
|
|
|
} else if let Some(pfl) = child_opt::<_, PosFieldDefList>(node) {
|
2018-12-25 12:01:47 +00:00
|
|
|
StructFlavor::Tuple(pfl)
|
|
|
|
} else {
|
|
|
|
StructFlavor::Unit
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl StructDef {
|
|
|
|
pub fn flavor(&self) -> StructFlavor {
|
2018-12-25 12:01:47 +00:00
|
|
|
StructFlavor::from_node(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl EnumVariant {
|
2019-01-24 15:56:38 +00:00
|
|
|
pub fn parent_enum(&self) -> &EnumDef {
|
|
|
|
self.syntax()
|
|
|
|
.parent()
|
|
|
|
.and_then(|it| it.parent())
|
|
|
|
.and_then(EnumDef::cast)
|
|
|
|
.expect("EnumVariants are always nested in Enums")
|
|
|
|
}
|
2019-01-07 13:15:47 +00:00
|
|
|
pub fn flavor(&self) -> StructFlavor {
|
2018-12-25 12:01:47 +00:00
|
|
|
StructFlavor::from_node(self)
|
|
|
|
}
|
|
|
|
}
|
2018-12-25 16:17:39 +00:00
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl PointerType {
|
2018-12-25 16:17:39 +00:00
|
|
|
pub fn is_mut(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
self.syntax().children_with_tokens().any(|n| n.kind() == MUT_KW)
|
2018-12-25 16:17:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl ReferenceType {
|
2018-12-25 16:17:39 +00:00
|
|
|
pub fn is_mut(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
self.syntax().children_with_tokens().any(|n| n.kind() == MUT_KW)
|
2018-12-25 16:17:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl RefExpr {
|
2018-12-25 16:17:39 +00:00
|
|
|
pub fn is_mut(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
self.syntax().children_with_tokens().any(|n| n.kind() == MUT_KW)
|
2018-12-25 16:17:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum PrefixOp {
|
|
|
|
/// The `*` operator for dereferencing
|
|
|
|
Deref,
|
|
|
|
/// The `!` operator for logical inversion
|
|
|
|
Not,
|
|
|
|
/// The `-` operator for negation
|
|
|
|
Neg,
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl PrefixExpr {
|
2019-03-24 21:21:22 +00:00
|
|
|
pub fn op_kind(&self) -> Option<PrefixOp> {
|
2019-03-30 10:25:53 +00:00
|
|
|
match self.op_token()?.kind() {
|
2018-12-25 16:17:39 +00:00
|
|
|
STAR => Some(PrefixOp::Deref),
|
|
|
|
EXCL => Some(PrefixOp::Not),
|
|
|
|
MINUS => Some(PrefixOp::Neg),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
2019-03-24 21:21:22 +00:00
|
|
|
|
2019-03-30 10:25:53 +00:00
|
|
|
pub fn op_token(&self) -> Option<SyntaxToken> {
|
|
|
|
self.syntax().first_child_or_token()?.as_token()
|
2019-03-24 21:21:22 +00:00
|
|
|
}
|
2018-12-25 16:17:39 +00:00
|
|
|
}
|
2019-01-04 13:51:45 +00:00
|
|
|
|
2018-12-29 20:32:07 +00:00
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
|
2019-01-05 20:28:30 +00:00
|
|
|
pub enum BinOp {
|
|
|
|
/// The `||` operator for boolean OR
|
|
|
|
BooleanOr,
|
|
|
|
/// The `&&` operator for boolean AND
|
|
|
|
BooleanAnd,
|
|
|
|
/// The `==` operator for equality testing
|
|
|
|
EqualityTest,
|
2019-02-18 07:09:44 +00:00
|
|
|
/// The `!=` operator for equality testing
|
|
|
|
NegatedEqualityTest,
|
2019-01-05 20:28:30 +00:00
|
|
|
/// The `<=` operator for lesser-equal testing
|
|
|
|
LesserEqualTest,
|
|
|
|
/// The `>=` operator for greater-equal testing
|
|
|
|
GreaterEqualTest,
|
|
|
|
/// The `<` operator for comparison
|
|
|
|
LesserTest,
|
|
|
|
/// The `>` operator for comparison
|
|
|
|
GreaterTest,
|
2019-01-07 18:03:25 +00:00
|
|
|
/// The `+` operator for addition
|
|
|
|
Addition,
|
|
|
|
/// The `*` operator for multiplication
|
|
|
|
Multiplication,
|
|
|
|
/// The `-` operator for subtraction
|
|
|
|
Subtraction,
|
|
|
|
/// The `/` operator for division
|
|
|
|
Division,
|
|
|
|
/// The `%` operator for remainder after division
|
|
|
|
Remainder,
|
|
|
|
/// The `<<` operator for left shift
|
|
|
|
LeftShift,
|
|
|
|
/// The `>>` operator for right shift
|
|
|
|
RightShift,
|
|
|
|
/// The `^` operator for bitwise XOR
|
|
|
|
BitwiseXor,
|
|
|
|
/// The `|` operator for bitwise OR
|
|
|
|
BitwiseOr,
|
|
|
|
/// The `&` operator for bitwise AND
|
|
|
|
BitwiseAnd,
|
|
|
|
/// The `..` operator for right-open ranges
|
|
|
|
RangeRightOpen,
|
|
|
|
/// The `..=` operator for right-closed ranges
|
|
|
|
RangeRightClosed,
|
|
|
|
/// The `=` operator for assignment
|
|
|
|
Assignment,
|
2019-02-18 07:09:44 +00:00
|
|
|
/// The `+=` operator for assignment after addition
|
2019-01-07 18:03:25 +00:00
|
|
|
AddAssign,
|
|
|
|
/// The `/=` operator for assignment after division
|
|
|
|
DivAssign,
|
|
|
|
/// The `*=` operator for assignment after multiplication
|
|
|
|
MulAssign,
|
|
|
|
/// The `%=` operator for assignment after remainders
|
|
|
|
RemAssign,
|
|
|
|
/// The `>>=` operator for assignment after shifting right
|
|
|
|
ShrAssign,
|
|
|
|
/// The `<<=` operator for assignment after shifting left
|
|
|
|
ShlAssign,
|
|
|
|
/// The `-=` operator for assignment after subtraction
|
|
|
|
SubAssign,
|
|
|
|
/// The `|=` operator for assignment after bitwise OR
|
|
|
|
BitOrAssign,
|
|
|
|
/// The `&=` operator for assignment after bitwise AND
|
|
|
|
BitAndAssign,
|
|
|
|
/// The `^=` operator for assignment after bitwise XOR
|
2019-01-07 19:11:31 +00:00
|
|
|
BitXorAssign,
|
2019-01-05 20:28:30 +00:00
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl BinExpr {
|
2019-03-30 10:25:53 +00:00
|
|
|
fn op_details(&self) -> Option<(SyntaxToken, BinOp)> {
|
|
|
|
self.syntax().children_with_tokens().filter_map(|it| it.as_token()).find_map(|c| {
|
|
|
|
match c.kind() {
|
|
|
|
PIPEPIPE => Some((c, BinOp::BooleanOr)),
|
|
|
|
AMPAMP => Some((c, BinOp::BooleanAnd)),
|
|
|
|
EQEQ => Some((c, BinOp::EqualityTest)),
|
|
|
|
NEQ => Some((c, BinOp::NegatedEqualityTest)),
|
|
|
|
LTEQ => Some((c, BinOp::LesserEqualTest)),
|
|
|
|
GTEQ => Some((c, BinOp::GreaterEqualTest)),
|
|
|
|
L_ANGLE => Some((c, BinOp::LesserTest)),
|
|
|
|
R_ANGLE => Some((c, BinOp::GreaterTest)),
|
|
|
|
PLUS => Some((c, BinOp::Addition)),
|
|
|
|
STAR => Some((c, BinOp::Multiplication)),
|
|
|
|
MINUS => Some((c, BinOp::Subtraction)),
|
|
|
|
SLASH => Some((c, BinOp::Division)),
|
|
|
|
PERCENT => Some((c, BinOp::Remainder)),
|
|
|
|
SHL => Some((c, BinOp::LeftShift)),
|
|
|
|
SHR => Some((c, BinOp::RightShift)),
|
|
|
|
CARET => Some((c, BinOp::BitwiseXor)),
|
|
|
|
PIPE => Some((c, BinOp::BitwiseOr)),
|
|
|
|
AMP => Some((c, BinOp::BitwiseAnd)),
|
|
|
|
DOTDOT => Some((c, BinOp::RangeRightOpen)),
|
|
|
|
DOTDOTEQ => Some((c, BinOp::RangeRightClosed)),
|
|
|
|
EQ => Some((c, BinOp::Assignment)),
|
|
|
|
PLUSEQ => Some((c, BinOp::AddAssign)),
|
|
|
|
SLASHEQ => Some((c, BinOp::DivAssign)),
|
|
|
|
STAREQ => Some((c, BinOp::MulAssign)),
|
|
|
|
PERCENTEQ => Some((c, BinOp::RemAssign)),
|
|
|
|
SHREQ => Some((c, BinOp::ShrAssign)),
|
|
|
|
SHLEQ => Some((c, BinOp::ShlAssign)),
|
|
|
|
MINUSEQ => Some((c, BinOp::SubAssign)),
|
|
|
|
PIPEEQ => Some((c, BinOp::BitOrAssign)),
|
|
|
|
AMPEQ => Some((c, BinOp::BitAndAssign)),
|
|
|
|
CARETEQ => Some((c, BinOp::BitXorAssign)),
|
|
|
|
_ => None,
|
|
|
|
}
|
2019-03-25 07:59:42 +00:00
|
|
|
})
|
2019-01-05 20:28:30 +00:00
|
|
|
}
|
2019-01-06 20:39:36 +00:00
|
|
|
|
2019-03-24 21:21:22 +00:00
|
|
|
pub fn op_kind(&self) -> Option<BinOp> {
|
|
|
|
self.op_details().map(|t| t.1)
|
|
|
|
}
|
|
|
|
|
2019-03-30 10:25:53 +00:00
|
|
|
pub fn op_token(&self) -> Option<SyntaxToken> {
|
2019-03-24 21:21:22 +00:00
|
|
|
self.op_details().map(|t| t.0)
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
pub fn lhs(&self) -> Option<&Expr> {
|
2019-01-06 20:39:36 +00:00
|
|
|
children(self).nth(0)
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
pub fn rhs(&self) -> Option<&Expr> {
|
2019-01-06 20:39:36 +00:00
|
|
|
children(self).nth(1)
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
pub fn sub_exprs(&self) -> (Option<&Expr>, Option<&Expr>) {
|
2019-01-06 20:39:36 +00:00
|
|
|
let mut children = children(self);
|
|
|
|
let first = children.next();
|
|
|
|
let second = children.next();
|
|
|
|
(first, second)
|
|
|
|
}
|
2019-01-05 20:28:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
|
2018-12-29 20:32:07 +00:00
|
|
|
pub enum SelfParamFlavor {
|
|
|
|
/// self
|
|
|
|
Owned,
|
|
|
|
/// &self
|
|
|
|
Ref,
|
|
|
|
/// &mut self
|
|
|
|
MutRef,
|
|
|
|
}
|
|
|
|
|
2019-01-07 13:15:47 +00:00
|
|
|
impl SelfParam {
|
2019-03-30 10:25:53 +00:00
|
|
|
pub fn self_kw_token(&self) -> SyntaxToken {
|
|
|
|
self.syntax()
|
|
|
|
.children_with_tokens()
|
|
|
|
.filter_map(|it| it.as_token())
|
|
|
|
.find(|it| it.kind() == SELF_KW)
|
|
|
|
.expect("invalid tree: self param must have self")
|
|
|
|
}
|
|
|
|
|
2018-12-29 20:32:07 +00:00
|
|
|
pub fn flavor(&self) -> SelfParamFlavor {
|
2019-03-30 10:25:53 +00:00
|
|
|
let borrowed = self.syntax().children_with_tokens().any(|n| n.kind() == AMP);
|
2018-12-29 20:32:07 +00:00
|
|
|
if borrowed {
|
|
|
|
// check for a `mut` coming after the & -- `mut &self` != `&mut self`
|
2019-03-30 10:25:53 +00:00
|
|
|
if self
|
|
|
|
.syntax()
|
|
|
|
.children_with_tokens()
|
|
|
|
.skip_while(|n| n.kind() != AMP)
|
|
|
|
.any(|n| n.kind() == MUT_KW)
|
2018-12-29 20:32:07 +00:00
|
|
|
{
|
|
|
|
SelfParamFlavor::MutRef
|
|
|
|
} else {
|
|
|
|
SelfParamFlavor::Ref
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SelfParamFlavor::Owned
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-14 18:30:21 +00:00
|
|
|
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum LiteralFlavor {
|
|
|
|
String,
|
|
|
|
ByteString,
|
|
|
|
Char,
|
|
|
|
Byte,
|
|
|
|
IntNumber { suffix: Option<SmolStr> },
|
|
|
|
FloatNumber { suffix: Option<SmolStr> },
|
|
|
|
Bool,
|
|
|
|
}
|
|
|
|
|
2019-03-30 10:25:53 +00:00
|
|
|
impl Literal {
|
|
|
|
pub fn token(&self) -> SyntaxToken {
|
|
|
|
match self.syntax().first_child_or_token().unwrap() {
|
|
|
|
SyntaxElement::Token(token) => token,
|
|
|
|
_ => unreachable!(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-14 18:30:21 +00:00
|
|
|
pub fn flavor(&self) -> LiteralFlavor {
|
2019-03-30 10:25:53 +00:00
|
|
|
match self.token().kind() {
|
2019-01-14 18:30:21 +00:00
|
|
|
INT_NUMBER => {
|
|
|
|
let allowed_suffix_list = [
|
|
|
|
"isize", "i128", "i64", "i32", "i16", "i8", "usize", "u128", "u64", "u32",
|
|
|
|
"u16", "u8",
|
|
|
|
];
|
2019-03-30 10:25:53 +00:00
|
|
|
let text = self.token().text().to_string();
|
2019-01-14 18:30:21 +00:00
|
|
|
let suffix = allowed_suffix_list
|
|
|
|
.iter()
|
|
|
|
.find(|&s| text.ends_with(s))
|
|
|
|
.map(|&suf| SmolStr::new(suf));
|
2019-03-30 10:25:53 +00:00
|
|
|
LiteralFlavor::IntNumber { suffix }
|
2019-01-14 18:30:21 +00:00
|
|
|
}
|
|
|
|
FLOAT_NUMBER => {
|
|
|
|
let allowed_suffix_list = ["f64", "f32"];
|
2019-03-30 10:25:53 +00:00
|
|
|
let text = self.token().text().to_string();
|
2019-01-14 18:30:21 +00:00
|
|
|
let suffix = allowed_suffix_list
|
|
|
|
.iter()
|
|
|
|
.find(|&s| text.ends_with(s))
|
|
|
|
.map(|&suf| SmolStr::new(suf));
|
|
|
|
LiteralFlavor::FloatNumber { suffix: suffix }
|
|
|
|
}
|
|
|
|
STRING | RAW_STRING => LiteralFlavor::String,
|
|
|
|
TRUE_KW | FALSE_KW => LiteralFlavor::Bool,
|
|
|
|
BYTE_STRING | RAW_BYTE_STRING => LiteralFlavor::ByteString,
|
|
|
|
CHAR => LiteralFlavor::Char,
|
|
|
|
BYTE => LiteralFlavor::Byte,
|
|
|
|
_ => unreachable!(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-24 14:01:56 +00:00
|
|
|
impl NamedField {
|
|
|
|
pub fn parent_struct_lit(&self) -> &StructLit {
|
|
|
|
self.syntax().ancestors().find_map(StructLit::cast).unwrap()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-17 12:08:18 +00:00
|
|
|
impl BindPat {
|
|
|
|
pub fn is_mutable(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
self.syntax().children_with_tokens().any(|n| n.kind() == MUT_KW)
|
2019-01-17 12:08:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_ref(&self) -> bool {
|
2019-03-30 10:25:53 +00:00
|
|
|
self.syntax().children_with_tokens().any(|n| n.kind() == REF_KW)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl LifetimeParam {
|
|
|
|
pub fn lifetime_token(&self) -> Option<SyntaxToken> {
|
|
|
|
self.syntax()
|
|
|
|
.children_with_tokens()
|
|
|
|
.filter_map(|it| it.as_token())
|
|
|
|
.find(|it| it.kind() == LIFETIME)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl WherePred {
|
|
|
|
pub fn lifetime_token(&self) -> Option<SyntaxToken> {
|
|
|
|
self.syntax()
|
|
|
|
.children_with_tokens()
|
|
|
|
.filter_map(|it| it.as_token())
|
|
|
|
.find(|it| it.kind() == LIFETIME)
|
2019-01-17 12:08:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-26 15:35:23 +00:00
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_none() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
// non-doc
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
|
|
|
assert!(module.doc_comment_text().is_none());
|
|
|
|
}
|
|
|
|
|
2019-01-04 13:51:45 +00:00
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_of_items() {
|
2019-01-07 13:15:47 +00:00
|
|
|
let file = SourceFile::parse(
|
2019-01-04 13:51:45 +00:00
|
|
|
r#"
|
|
|
|
//! doc
|
|
|
|
// non-doc
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2019-01-26 15:35:23 +00:00
|
|
|
assert_eq!("doc", module.doc_comment_text().unwrap());
|
2019-01-04 13:51:45 +00:00
|
|
|
}
|
2019-01-26 02:31:31 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_doc_comment_preserves_indents() {
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
/// doc1
|
|
|
|
/// ```
|
|
|
|
/// fn foo() {
|
|
|
|
/// // ...
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
mod foo {}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
let module = file.syntax().descendants().find_map(Module::cast).unwrap();
|
2019-02-08 11:49:43 +00:00
|
|
|
assert_eq!("doc1\n```\nfn foo() {\n // ...\n}\n```", module.doc_comment_text().unwrap());
|
2019-01-04 13:51:45 +00:00
|
|
|
}
|
2019-03-24 17:45:11 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_where_predicates() {
|
|
|
|
fn assert_bound(text: &str, bound: Option<&TypeBound>) {
|
|
|
|
assert_eq!(text, bound.unwrap().syntax().text().to_string());
|
|
|
|
}
|
|
|
|
|
|
|
|
let file = SourceFile::parse(
|
|
|
|
r#"
|
|
|
|
fn foo()
|
|
|
|
where
|
|
|
|
T: Clone + Copy + Debug + 'static,
|
|
|
|
'a: 'b + 'c,
|
|
|
|
Iterator::Item: 'a + Debug,
|
|
|
|
Iterator::Item: Debug + 'a,
|
|
|
|
<T as Iterator>::Item: Debug + 'a,
|
|
|
|
for<'a> F: Fn(&'a str)
|
|
|
|
{}
|
|
|
|
"#,
|
|
|
|
);
|
|
|
|
let where_clause = file.syntax().descendants().find_map(WhereClause::cast).unwrap();
|
|
|
|
|
|
|
|
let mut predicates = where_clause.predicates();
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
|
|
|
assert_eq!("T", pred.type_ref().unwrap().syntax().text().to_string());
|
|
|
|
assert_bound("Clone", bounds.next());
|
|
|
|
assert_bound("Copy", bounds.next());
|
|
|
|
assert_bound("Debug", bounds.next());
|
|
|
|
assert_bound("'static", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
2019-03-30 10:25:53 +00:00
|
|
|
assert_eq!("'a", pred.lifetime_token().unwrap().text());
|
2019-03-24 17:45:11 +00:00
|
|
|
|
|
|
|
assert_bound("'b", bounds.next());
|
|
|
|
assert_bound("'c", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
|
|
|
assert_eq!("Iterator::Item", pred.type_ref().unwrap().syntax().text().to_string());
|
|
|
|
assert_bound("'a", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
|
|
|
assert_eq!("Iterator::Item", pred.type_ref().unwrap().syntax().text().to_string());
|
|
|
|
assert_bound("Debug", bounds.next());
|
|
|
|
assert_bound("'a", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
|
|
|
assert_eq!("<T as Iterator>::Item", pred.type_ref().unwrap().syntax().text().to_string());
|
|
|
|
assert_bound("Debug", bounds.next());
|
|
|
|
assert_bound("'a", bounds.next());
|
|
|
|
|
|
|
|
let pred = predicates.next().unwrap();
|
|
|
|
let mut bounds = pred.type_bound_list().unwrap().bounds();
|
|
|
|
|
|
|
|
assert_eq!("for<'a> F", pred.type_ref().unwrap().syntax().text().to_string());
|
|
|
|
assert_bound("Fn(&'a str)", bounds.next());
|
|
|
|
}
|