//! Syntax Tree library used throughout the rust-analyzer. //! //! Properties: //! - easy and fast incremental re-parsing //! - graceful handling of errors //! - full-fidelity representation (*any* text can be precisely represented as //! a syntax tree) //! //! For more information, see the [RFC]. Current implementation is inspired by //! the [Swift] one. //! //! The most interesting modules here are `syntax_node` (which defines concrete //! syntax tree) and `ast` (which defines abstract syntax tree on top of the //! CST). The actual parser live in a separate `parser` crate, though the //! lexer lives in this crate. //! //! See `api_walkthrough` test in this file for a quick API tour! //! //! [RFC]: //! [Swift]: #![cfg_attr(feature = "in-rust-tree", feature(rustc_private))] #![warn(rust_2018_idioms, unused_lifetimes, semicolon_in_expressions_from_macros)] #[allow(unused)] macro_rules! eprintln { ($($tt:tt)*) => { stdx::eprintln!($($tt)*) }; } mod syntax_node; mod syntax_error; mod parsing; mod validation; mod ptr; mod token_text; #[cfg(test)] mod tests; pub mod algo; pub mod ast; #[doc(hidden)] pub mod fuzz; pub mod utils; pub mod ted; pub mod hacks; use std::marker::PhantomData; use stdx::format_to; use text_edit::Indel; use triomphe::Arc; pub use crate::{ ast::{AstNode, AstToken}, ptr::{AstPtr, SyntaxNodePtr}, syntax_error::SyntaxError, syntax_node::{ PreorderWithTokens, RustLanguage, SyntaxElement, SyntaxElementChildren, SyntaxNode, SyntaxNodeChildren, SyntaxToken, SyntaxTreeBuilder, }, token_text::TokenText, }; pub use parser::{SyntaxKind, T}; pub use rowan::{ api::Preorder, Direction, GreenNode, NodeOrToken, SyntaxText, TextRange, TextSize, TokenAtOffset, WalkEvent, }; pub use smol_str::SmolStr; /// `Parse` is the result of the parsing: a syntax tree and a collection of /// errors. /// /// Note that we always produce a syntax tree, even for completely invalid /// files. #[derive(Debug, PartialEq, Eq)] pub struct Parse { green: GreenNode, errors: Arc<[SyntaxError]>, _ty: PhantomData T>, } impl Clone for Parse { fn clone(&self) -> Parse { Parse { green: self.green.clone(), errors: self.errors.clone(), _ty: PhantomData } } } impl Parse { fn new(green: GreenNode, errors: Vec) -> Parse { Parse { green, errors: errors.into(), _ty: PhantomData } } pub fn syntax_node(&self) -> SyntaxNode { SyntaxNode::new_root(self.green.clone()) } pub fn errors(&self) -> &[SyntaxError] { &self.errors } } impl Parse { pub fn to_syntax(self) -> Parse { Parse { green: self.green, errors: self.errors, _ty: PhantomData } } pub fn tree(&self) -> T { T::cast(self.syntax_node()).unwrap() } pub fn ok(self) -> Result> { if self.errors.is_empty() { Ok(self.tree()) } else { Err(self.errors) } } } impl Parse { pub fn cast(self) -> Option> { if N::cast(self.syntax_node()).is_some() { Some(Parse { green: self.green, errors: self.errors, _ty: PhantomData }) } else { None } } } impl Parse { pub fn debug_dump(&self) -> String { let mut buf = format!("{:#?}", self.tree().syntax()); for err in self.errors.iter() { format_to!(buf, "error {:?}: {}\n", err.range(), err); } buf } pub fn reparse(&self, indel: &Indel) -> Parse { self.incremental_reparse(indel).unwrap_or_else(|| self.full_reparse(indel)) } fn incremental_reparse(&self, indel: &Indel) -> Option> { // FIXME: validation errors are not handled here parsing::incremental_reparse(self.tree().syntax(), indel, self.errors.to_vec()).map( |(green_node, errors, _reparsed_range)| Parse { green: green_node, errors: errors.into(), _ty: PhantomData, }, ) } fn full_reparse(&self, indel: &Indel) -> Parse { let mut text = self.tree().syntax().text().to_string(); indel.apply(&mut text); SourceFile::parse(&text) } } /// `SourceFile` represents a parse tree for a single Rust file. pub use crate::ast::SourceFile; impl SourceFile { pub fn parse(text: &str) -> Parse { let (green, mut errors) = parsing::parse_text(text); let root = SyntaxNode::new_root(green.clone()); errors.extend(validation::validate(&root)); assert_eq!(root.kind(), SyntaxKind::SOURCE_FILE); Parse { green, errors: errors.into(), _ty: PhantomData } } } impl ast::TokenTree { pub fn reparse_as_comma_separated_expr(self) -> Parse { let tokens = self.syntax().descendants_with_tokens().filter_map(NodeOrToken::into_token); let mut parser_input = parser::Input::default(); let mut was_joint = false; for t in tokens { let kind = t.kind(); if kind.is_trivia() { was_joint = false } else if kind == SyntaxKind::IDENT { let token_text = t.text(); let contextual_kw = SyntaxKind::from_contextual_keyword(token_text).unwrap_or(SyntaxKind::IDENT); parser_input.push_ident(contextual_kw); } else { if was_joint { parser_input.was_joint(); } parser_input.push(kind); // Tag the token as joint if it is float with a fractional part // we use this jointness to inform the parser about what token split // event to emit when we encounter a float literal in a field access if kind == SyntaxKind::FLOAT_NUMBER { if !t.text().ends_with('.') { parser_input.was_joint(); } else { was_joint = false; } } else { was_joint = true; } } } let parser_output = parser::TopEntryPoint::MacroEagerInput.parse(&parser_input); let mut tokens = self.syntax().descendants_with_tokens().filter_map(NodeOrToken::into_token); let mut text = String::new(); let mut pos = TextSize::from(0); let mut builder = SyntaxTreeBuilder::default(); for event in parser_output.iter() { match event { parser::Step::Token { kind, n_input_tokens } => { let mut token = tokens.next().unwrap(); while token.kind().is_trivia() { let text = token.text(); pos += TextSize::from(text.len() as u32); builder.token(token.kind(), text); token = tokens.next().unwrap(); } text.push_str(token.text()); for _ in 1..n_input_tokens { let token = tokens.next().unwrap(); text.push_str(token.text()); } pos += TextSize::from(text.len() as u32); builder.token(kind, &text); text.clear(); } parser::Step::FloatSplit { ends_in_dot: has_pseudo_dot } => { let token = tokens.next().unwrap(); let text = token.text(); match text.split_once('.') { Some((left, right)) => { assert!(!left.is_empty()); builder.start_node(SyntaxKind::NAME_REF); builder.token(SyntaxKind::INT_NUMBER, left); builder.finish_node(); // here we move the exit up, the original exit has been deleted in process builder.finish_node(); builder.token(SyntaxKind::DOT, "."); if has_pseudo_dot { assert!(right.is_empty(), "{left}.{right}"); } else { assert!(!right.is_empty(), "{left}.{right}"); builder.start_node(SyntaxKind::NAME_REF); builder.token(SyntaxKind::INT_NUMBER, right); builder.finish_node(); // the parser creates an unbalanced start node, we are required to close it here builder.finish_node(); } } None => unreachable!(), } pos += TextSize::from(text.len() as u32); } parser::Step::Enter { kind } => builder.start_node(kind), parser::Step::Exit => builder.finish_node(), parser::Step::Error { msg } => builder.error(msg.to_owned(), pos), } } let (green, errors) = builder.finish_raw(); Parse { green, errors: errors.into(), _ty: PhantomData } } } /// Matches a `SyntaxNode` against an `ast` type. /// /// # Example: /// /// ```ignore /// match_ast! { /// match node { /// ast::CallExpr(it) => { ... }, /// ast::MethodCallExpr(it) => { ... }, /// ast::MacroCall(it) => { ... }, /// _ => None, /// } /// } /// ``` #[macro_export] macro_rules! match_ast { (match $node:ident { $($tt:tt)* }) => { $crate::match_ast!(match ($node) { $($tt)* }) }; (match ($node:expr) { $( $( $path:ident )::+ ($it:pat) => $res:expr, )* _ => $catch_all:expr $(,)? }) => {{ $( if let Some($it) = $($path::)+cast($node.clone()) { $res } else )* { $catch_all } }}; } /// This test does not assert anything and instead just shows off the crate's /// API. #[test] fn api_walkthrough() { use ast::{HasModuleItem, HasName}; let source_code = " fn foo() { 1 + 1 } "; // `SourceFile` is the main entry point. // // The `parse` method returns a `Parse` -- a pair of syntax tree and a list // of errors. That is, syntax tree is constructed even in presence of errors. let parse = SourceFile::parse(source_code); assert!(parse.errors().is_empty()); // The `tree` method returns an owned syntax node of type `SourceFile`. // Owned nodes are cheap: inside, they are `Rc` handles to the underling data. let file: SourceFile = parse.tree(); // `SourceFile` is the root of the syntax tree. We can iterate file's items. // Let's fetch the `foo` function. let mut func = None; for item in file.items() { match item { ast::Item::Fn(f) => func = Some(f), _ => unreachable!(), } } let func: ast::Fn = func.unwrap(); // Each AST node has a bunch of getters for children. All getters return // `Option`s though, to account for incomplete code. Some getters are common // for several kinds of node. In this case, a trait like `ast::NameOwner` // usually exists. By convention, all ast types should be used with `ast::` // qualifier. let name: Option = func.name(); let name = name.unwrap(); assert_eq!(name.text(), "foo"); // Let's get the `1 + 1` expression! let body: ast::BlockExpr = func.body().unwrap(); let stmt_list: ast::StmtList = body.stmt_list().unwrap(); let expr: ast::Expr = stmt_list.tail_expr().unwrap(); // Enums are used to group related ast nodes together, and can be used for // matching. However, because there are no public fields, it's possible to // match only the top level enum: that is the price we pay for increased API // flexibility let bin_expr: &ast::BinExpr = match &expr { ast::Expr::BinExpr(e) => e, _ => unreachable!(), }; // Besides the "typed" AST API, there's an untyped CST one as well. // To switch from AST to CST, call `.syntax()` method: let expr_syntax: &SyntaxNode = expr.syntax(); // Note how `expr` and `bin_expr` are in fact the same node underneath: assert!(expr_syntax == bin_expr.syntax()); // To go from CST to AST, `AstNode::cast` function is used: let _expr: ast::Expr = match ast::Expr::cast(expr_syntax.clone()) { Some(e) => e, None => unreachable!(), }; // The two properties each syntax node has is a `SyntaxKind`: assert_eq!(expr_syntax.kind(), SyntaxKind::BIN_EXPR); // And text range: assert_eq!(expr_syntax.text_range(), TextRange::new(32.into(), 37.into())); // You can get node's text as a `SyntaxText` object, which will traverse the // tree collecting token's text: let text: SyntaxText = expr_syntax.text(); assert_eq!(text.to_string(), "1 + 1"); // There's a bunch of traversal methods on `SyntaxNode`: assert_eq!(expr_syntax.parent().as_ref(), Some(stmt_list.syntax())); assert_eq!(stmt_list.syntax().first_child_or_token().map(|it| it.kind()), Some(T!['{'])); assert_eq!( expr_syntax.next_sibling_or_token().map(|it| it.kind()), Some(SyntaxKind::WHITESPACE) ); // As well as some iterator helpers: let f = expr_syntax.ancestors().find_map(ast::Fn::cast); assert_eq!(f, Some(func)); assert!(expr_syntax.siblings_with_tokens(Direction::Next).any(|it| it.kind() == T!['}'])); assert_eq!( expr_syntax.descendants_with_tokens().count(), 8, // 5 tokens `1`, ` `, `+`, ` `, `!` // 2 child literal expressions: `1`, `1` // 1 the node itself: `1 + 1` ); // There's also a `preorder` method with a more fine-grained iteration control: let mut buf = String::new(); let mut indent = 0; for event in expr_syntax.preorder_with_tokens() { match event { WalkEvent::Enter(node) => { let text = match &node { NodeOrToken::Node(it) => it.text().to_string(), NodeOrToken::Token(it) => it.text().to_string(), }; format_to!(buf, "{:indent$}{:?} {:?}\n", " ", text, node.kind(), indent = indent); indent += 2; } WalkEvent::Leave(_) => indent -= 2, } } assert_eq!(indent, 0); assert_eq!( buf.trim(), r#" "1 + 1" BIN_EXPR "1" LITERAL "1" INT_NUMBER " " WHITESPACE "+" PLUS " " WHITESPACE "1" LITERAL "1" INT_NUMBER "# .trim() ); // To recursively process the tree, there are three approaches: // 1. explicitly call getter methods on AST nodes. // 2. use descendants and `AstNode::cast`. // 3. use descendants and `match_ast!`. // // Here's how the first one looks like: let exprs_cast: Vec = file .syntax() .descendants() .filter_map(ast::Expr::cast) .map(|expr| expr.syntax().text().to_string()) .collect(); // An alternative is to use a macro. let mut exprs_visit = Vec::new(); for node in file.syntax().descendants() { match_ast! { match node { ast::Expr(it) => { let res = it.syntax().text().to_string(); exprs_visit.push(res); }, _ => (), } } } assert_eq!(exprs_cast, exprs_visit); }