rust-analyzer/crates/ra_syntax/src/syntax_node.rs

279 lines
7.2 KiB
Rust
Raw Normal View History

2019-02-21 12:24:42 +00:00
//! This module defines Concrete Syntax Tree (CST), used by rust-analyzer.
//!
//! The CST includes comments and whitespace, provides a single node type,
//! `SyntaxNode`, and a basic traversal API (parent, children, siblings).
//!
//! The *real* implementation is in the (language-agnostic) `rowan` crate, this
//! modules just wraps its API.
2019-02-21 13:12:15 +00:00
use std::{
fmt::{self, Write},
borrow::Borrow,
};
2019-01-07 13:15:47 +00:00
use rowan::{Types, TransparentNewType};
2018-08-10 14:49:45 +00:00
2019-02-20 13:16:14 +00:00
use crate::{
2019-02-21 12:51:22 +00:00
SmolStr, SyntaxKind, TextRange, SyntaxText, SourceFile, AstNode,
2019-02-20 13:16:14 +00:00
syntax_error::SyntaxError,
};
2019-01-07 13:15:47 +00:00
pub use rowan::WalkEvent;
2018-10-02 14:07:12 +00:00
#[derive(Debug, Clone, Copy)]
pub enum RaTypes {}
impl Types for RaTypes {
type Kind = SyntaxKind;
type RootData = Vec<SyntaxError>;
2018-08-10 14:49:45 +00:00
}
2019-02-21 13:12:15 +00:00
pub(crate) type GreenNode = rowan::GreenNode<RaTypes>;
/// Marker trait for CST and AST nodes
pub trait SyntaxNodeWrapper: TransparentNewType<Repr = rowan::SyntaxNode<RaTypes>> {}
impl<T: TransparentNewType<Repr = rowan::SyntaxNode<RaTypes>>> SyntaxNodeWrapper for T {}
2019-01-07 13:42:10 +00:00
2019-02-21 13:12:15 +00:00
/// An owning smart pointer for CST or AST node.
2019-01-07 14:00:39 +00:00
#[derive(PartialEq, Eq, Hash)]
2019-02-21 13:12:15 +00:00
pub struct TreeArc<T: SyntaxNodeWrapper>(pub(crate) rowan::TreeArc<RaTypes, T>);
2019-01-07 13:42:10 +00:00
2019-02-21 13:12:15 +00:00
impl<T: SyntaxNodeWrapper> Borrow<T> for TreeArc<T> {
fn borrow(&self) -> &T {
&*self
}
}
impl<T> TreeArc<T>
2019-01-07 13:42:10 +00:00
where
2019-02-21 13:12:15 +00:00
T: SyntaxNodeWrapper,
2019-01-07 13:42:10 +00:00
{
pub(crate) fn cast<U>(this: TreeArc<T>) -> TreeArc<U>
2019-01-07 13:42:10 +00:00
where
2019-02-21 13:12:15 +00:00
U: SyntaxNodeWrapper,
2019-01-07 13:42:10 +00:00
{
TreeArc(rowan::TreeArc::cast(this.0))
2019-01-07 13:42:10 +00:00
}
}
impl<T> std::ops::Deref for TreeArc<T>
2019-01-07 13:42:10 +00:00
where
2019-02-21 13:12:15 +00:00
T: SyntaxNodeWrapper,
2019-01-07 13:42:10 +00:00
{
type Target = T;
fn deref(&self) -> &T {
self.0.deref()
}
}
impl<T> PartialEq<T> for TreeArc<T>
2019-01-08 08:28:42 +00:00
where
2019-02-21 13:12:15 +00:00
T: SyntaxNodeWrapper,
2019-01-08 08:28:42 +00:00
T: PartialEq<T>,
{
fn eq(&self, other: &T) -> bool {
let t: &T = self;
t == other
}
}
impl<T> Clone for TreeArc<T>
2019-01-07 14:00:39 +00:00
where
2019-02-21 13:12:15 +00:00
T: SyntaxNodeWrapper,
2019-01-07 14:00:39 +00:00
{
fn clone(&self) -> TreeArc<T> {
TreeArc(self.0.clone())
2019-01-07 14:00:39 +00:00
}
}
impl<T> fmt::Debug for TreeArc<T>
2019-01-07 13:42:10 +00:00
where
2019-02-21 13:12:15 +00:00
T: SyntaxNodeWrapper,
2019-01-07 13:42:10 +00:00
T: fmt::Debug,
{
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
fmt::Debug::fmt(&self.0, fmt)
}
}
2018-08-17 18:10:55 +00:00
2019-01-07 13:15:47 +00:00
#[derive(PartialEq, Eq, Hash)]
#[repr(transparent)]
2019-01-07 13:42:10 +00:00
pub struct SyntaxNode(pub(crate) rowan::SyntaxNode<RaTypes>);
2019-01-07 13:15:47 +00:00
unsafe impl TransparentNewType for SyntaxNode {
2019-01-07 13:42:10 +00:00
type Repr = rowan::SyntaxNode<RaTypes>;
2018-10-02 14:07:12 +00:00
}
2018-08-17 18:10:55 +00:00
2019-02-21 13:04:03 +00:00
impl ToOwned for SyntaxNode {
type Owned = TreeArc<SyntaxNode>;
fn to_owned(&self) -> TreeArc<SyntaxNode> {
let ptr = TreeArc(self.0.to_owned());
TreeArc::cast(ptr)
2018-10-02 14:07:12 +00:00
}
}
2018-10-02 15:14:33 +00:00
2019-02-21 13:04:03 +00:00
impl fmt::Debug for SyntaxNode {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
write!(fmt, "{:?}@{:?}", self.kind(), self.range())?;
if has_short_text(self.kind()) {
write!(fmt, " \"{}\"", self.text())?;
}
Ok(())
}
}
2018-10-02 15:14:33 +00:00
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum Direction {
Next,
Prev,
}
2019-01-07 13:15:47 +00:00
impl SyntaxNode {
2019-02-21 13:04:03 +00:00
pub(crate) fn new(green: GreenNode, errors: Vec<SyntaxError>) -> TreeArc<SyntaxNode> {
let ptr = TreeArc(rowan::SyntaxNode::new(green, errors));
TreeArc::cast(ptr)
}
pub fn kind(&self) -> SyntaxKind {
self.0.kind()
}
pub fn range(&self) -> TextRange {
self.0.range()
}
pub fn text(&self) -> SyntaxText {
SyntaxText::new(self)
}
pub fn is_leaf(&self) -> bool {
self.0.is_leaf()
}
2019-01-07 13:15:47 +00:00
pub fn leaf_text(&self) -> Option<&SmolStr> {
2018-10-02 14:07:12 +00:00
self.0.leaf_text()
2018-08-17 18:10:55 +00:00
}
2019-02-21 13:04:03 +00:00
pub fn parent(&self) -> Option<&SyntaxNode> {
self.0.parent().map(SyntaxNode::from_repr)
}
pub fn first_child(&self) -> Option<&SyntaxNode> {
self.0.first_child().map(SyntaxNode::from_repr)
}
pub fn last_child(&self) -> Option<&SyntaxNode> {
self.0.last_child().map(SyntaxNode::from_repr)
}
pub fn next_sibling(&self) -> Option<&SyntaxNode> {
self.0.next_sibling().map(SyntaxNode::from_repr)
}
pub fn prev_sibling(&self) -> Option<&SyntaxNode> {
self.0.prev_sibling().map(SyntaxNode::from_repr)
}
pub fn children(&self) -> SyntaxNodeChildren {
SyntaxNodeChildren(self.0.children())
}
2019-01-07 13:15:47 +00:00
pub fn ancestors(&self) -> impl Iterator<Item = &SyntaxNode> {
2018-10-15 16:55:32 +00:00
crate::algo::generate(Some(self), |&node| node.parent())
}
2019-02-21 13:04:03 +00:00
2019-01-07 13:15:47 +00:00
pub fn descendants(&self) -> impl Iterator<Item = &SyntaxNode> {
2018-10-17 16:52:25 +00:00
self.preorder().filter_map(|event| match event {
WalkEvent::Enter(node) => Some(node),
WalkEvent::Leave(_) => None,
})
}
2019-02-21 13:04:03 +00:00
2019-01-07 13:15:47 +00:00
pub fn siblings(&self, direction: Direction) -> impl Iterator<Item = &SyntaxNode> {
2018-10-15 16:55:32 +00:00
crate::algo::generate(Some(self), move |&node| match direction {
2018-10-02 15:14:33 +00:00
Direction::Next => node.next_sibling(),
Direction::Prev => node.prev_sibling(),
})
}
2019-02-21 13:04:03 +00:00
2019-01-07 13:15:47 +00:00
pub fn preorder(&self) -> impl Iterator<Item = WalkEvent<&SyntaxNode>> {
2018-10-17 16:52:25 +00:00
self.0.preorder().map(|event| match event {
2019-01-07 13:15:47 +00:00
WalkEvent::Enter(n) => WalkEvent::Enter(SyntaxNode::from_repr(n)),
WalkEvent::Leave(n) => WalkEvent::Leave(SyntaxNode::from_repr(n)),
2018-10-17 16:52:25 +00:00
})
}
2019-02-21 12:51:22 +00:00
2019-02-21 13:04:03 +00:00
pub fn memory_size_of_subtree(&self) -> usize {
self.0.memory_size_of_subtree()
}
2019-02-21 12:51:22 +00:00
pub fn debug_dump(&self) -> String {
let mut errors: Vec<_> = match self.ancestors().find_map(SourceFile::cast) {
Some(file) => file.errors(),
None => self.root_data().to_vec(),
};
errors.sort_by_key(|e| e.offset());
let mut err_pos = 0;
let mut level = 0;
let mut buf = String::new();
macro_rules! indent {
() => {
for _ in 0..level {
buf.push_str(" ");
}
};
}
for event in self.preorder() {
match event {
WalkEvent::Enter(node) => {
indent!();
writeln!(buf, "{:?}", node).unwrap();
if node.first_child().is_none() {
let off = node.range().end();
while err_pos < errors.len() && errors[err_pos].offset() <= off {
indent!();
writeln!(buf, "err: `{}`", errors[err_pos]).unwrap();
err_pos += 1;
}
}
level += 1;
}
WalkEvent::Leave(_) => level -= 1,
}
}
assert_eq!(level, 0);
for err in errors[err_pos..].iter() {
writeln!(buf, "err: `{}`", err).unwrap();
}
buf
}
2018-08-17 18:10:55 +00:00
2018-10-02 14:07:12 +00:00
pub(crate) fn root_data(&self) -> &Vec<SyntaxError> {
self.0.root_data()
}
2018-10-02 14:07:12 +00:00
pub(crate) fn replace_with(&self, replacement: GreenNode) -> GreenNode {
2019-01-07 13:15:47 +00:00
self.0.replace_self(replacement)
2018-10-02 14:07:12 +00:00
}
2018-08-10 14:49:45 +00:00
}
2018-10-02 14:07:12 +00:00
#[derive(Debug)]
2019-01-07 13:42:10 +00:00
pub struct SyntaxNodeChildren<'a>(rowan::SyntaxNodeChildren<'a, RaTypes>);
2018-08-10 14:49:45 +00:00
2019-01-07 13:15:47 +00:00
impl<'a> Iterator for SyntaxNodeChildren<'a> {
type Item = &'a SyntaxNode;
2018-08-10 14:49:45 +00:00
2019-01-07 13:15:47 +00:00
fn next(&mut self) -> Option<&'a SyntaxNode> {
self.0.next().map(SyntaxNode::from_repr)
2018-08-10 14:49:45 +00:00
}
}
2018-10-02 14:07:12 +00:00
fn has_short_text(kind: SyntaxKind) -> bool {
2018-10-15 16:55:32 +00:00
use crate::SyntaxKind::*;
2018-10-02 14:07:12 +00:00
match kind {
IDENT | LIFETIME | INT_NUMBER | FLOAT_NUMBER => true,
_ => false,
}
2018-08-10 14:49:45 +00:00
}