2018-07-31 19:14:56 +00:00
|
|
|
use std::{fmt::Write};
|
|
|
|
use {
|
|
|
|
SyntaxNode,
|
|
|
|
algo::walk::{WalkEvent, walk},
|
|
|
|
};
|
2018-07-30 12:25:52 +00:00
|
|
|
|
|
|
|
/// Parse a file and create a string representation of the resulting parse tree.
|
|
|
|
pub fn dump_tree(syntax: &SyntaxNode) -> String {
|
2018-07-30 20:45:10 +00:00
|
|
|
let syntax = syntax.as_ref();
|
2018-07-31 19:14:56 +00:00
|
|
|
let mut errors: Vec<_> = syntax.root.errors.iter().cloned().collect();
|
|
|
|
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(" ");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2018-07-30 12:25:52 +00:00
|
|
|
|
2018-07-31 19:14:56 +00:00
|
|
|
for event in walk(syntax) {
|
|
|
|
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].message).unwrap();
|
|
|
|
err_pos += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
level += 1;
|
|
|
|
},
|
|
|
|
WalkEvent::Exit(_) => level -= 1,
|
2018-07-30 12:25:52 +00:00
|
|
|
}
|
2018-07-31 19:14:56 +00:00
|
|
|
}
|
2018-07-30 12:25:52 +00:00
|
|
|
|
2018-07-31 19:14:56 +00:00
|
|
|
assert_eq!(level, 0);
|
|
|
|
for err in errors[err_pos..].iter() {
|
|
|
|
writeln!(buf, "err: `{}`", err.message).unwrap();
|
2018-07-30 12:25:52 +00:00
|
|
|
}
|
2018-07-31 19:14:56 +00:00
|
|
|
|
|
|
|
return buf;
|
2018-07-30 12:25:52 +00:00
|
|
|
}
|