2018-07-30 12:25:52 +00:00
|
|
|
use std::{
|
2018-10-15 21:44:23 +00:00
|
|
|
fmt::Write,
|
2020-07-01 10:30:17 +00:00
|
|
|
fs,
|
2020-07-11 10:56:44 +00:00
|
|
|
path::{Path, PathBuf},
|
2018-07-30 12:25:52 +00:00
|
|
|
};
|
2018-01-07 11:56:08 +00:00
|
|
|
|
2021-02-09 18:52:34 +00:00
|
|
|
use ast::NameOwner;
|
2020-08-21 11:19:31 +00:00
|
|
|
use expect_test::expect_file;
|
2020-07-11 10:56:44 +00:00
|
|
|
use rayon::prelude::*;
|
2021-03-08 17:22:33 +00:00
|
|
|
use test_utils::{bench, bench_fixture, project_root, skip_slow_tests};
|
2018-08-25 09:10:35 +00:00
|
|
|
|
2021-02-09 18:52:34 +00:00
|
|
|
use crate::{ast, fuzz, tokenize, AstNode, SourceFile, SyntaxError, TextRange, TextSize, Token};
|
2019-07-24 09:38:21 +00:00
|
|
|
|
2018-08-11 07:03:03 +00:00
|
|
|
#[test]
|
|
|
|
fn lexer_tests() {
|
2020-02-01 20:25:01 +00:00
|
|
|
// FIXME:
|
|
|
|
// * Add tests for unicode escapes in byte-character and [raw]-byte-string literals
|
|
|
|
// * Add tests for unescape errors
|
|
|
|
|
2020-04-06 11:04:26 +00:00
|
|
|
dir_tests(&test_data_dir(), &["lexer/ok"], "txt", |text, path| {
|
2020-02-01 20:25:01 +00:00
|
|
|
let (tokens, errors) = tokenize(text);
|
|
|
|
assert_errors_are_absent(&errors, path);
|
|
|
|
dump_tokens_and_errors(&tokens, &errors, text)
|
|
|
|
});
|
2020-04-06 11:04:26 +00:00
|
|
|
dir_tests(&test_data_dir(), &["lexer/err"], "txt", |text, path| {
|
2020-02-01 20:25:01 +00:00
|
|
|
let (tokens, errors) = tokenize(text);
|
|
|
|
assert_errors_are_present(&errors, path);
|
|
|
|
dump_tokens_and_errors(&tokens, &errors, text)
|
|
|
|
});
|
2018-08-11 07:03:03 +00:00
|
|
|
}
|
|
|
|
|
2019-09-09 11:52:31 +00:00
|
|
|
#[test]
|
|
|
|
fn parse_smoke_test() {
|
|
|
|
let code = r##"
|
|
|
|
fn main() {
|
|
|
|
println!("Hello, world!")
|
|
|
|
}
|
|
|
|
"##;
|
|
|
|
|
|
|
|
let parse = SourceFile::parse(code);
|
2020-03-13 11:09:14 +00:00
|
|
|
// eprintln!("{:#?}", parse.syntax_node());
|
2019-09-09 11:52:31 +00:00
|
|
|
assert!(parse.ok().is_ok());
|
|
|
|
}
|
|
|
|
|
2021-02-09 18:52:34 +00:00
|
|
|
#[test]
|
|
|
|
fn benchmark_parser() {
|
|
|
|
if skip_slow_tests() {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let data = bench_fixture::glorious_old_parser();
|
|
|
|
let tree = {
|
|
|
|
let _b = bench("parsing");
|
|
|
|
let p = SourceFile::parse(&data);
|
|
|
|
assert!(p.errors.is_empty());
|
|
|
|
assert_eq!(p.tree().syntax.text_range().len(), 352474.into());
|
|
|
|
p.tree()
|
|
|
|
};
|
|
|
|
|
|
|
|
{
|
|
|
|
let _b = bench("tree traversal");
|
|
|
|
let fn_names =
|
|
|
|
tree.syntax().descendants().filter_map(ast::Fn::cast).filter_map(|f| f.name()).count();
|
|
|
|
assert_eq!(fn_names, 268);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-10 18:14:09 +00:00
|
|
|
#[test]
|
|
|
|
fn parser_tests() {
|
2020-04-06 11:04:26 +00:00
|
|
|
dir_tests(&test_data_dir(), &["parser/inline/ok", "parser/ok"], "rast", |text, path| {
|
2019-05-28 14:34:28 +00:00
|
|
|
let parse = SourceFile::parse(text);
|
2019-07-12 16:41:13 +00:00
|
|
|
let errors = parse.errors();
|
2020-02-01 20:25:01 +00:00
|
|
|
assert_errors_are_absent(&errors, path);
|
2019-05-28 13:59:22 +00:00
|
|
|
parse.debug_dump()
|
2019-02-08 11:49:43 +00:00
|
|
|
});
|
2020-04-06 11:04:26 +00:00
|
|
|
dir_tests(&test_data_dir(), &["parser/err", "parser/inline/err"], "rast", |text, path| {
|
2019-05-28 14:34:28 +00:00
|
|
|
let parse = SourceFile::parse(text);
|
2019-07-12 16:41:13 +00:00
|
|
|
let errors = parse.errors();
|
2020-02-01 20:25:01 +00:00
|
|
|
assert_errors_are_present(&errors, path);
|
2019-05-28 13:59:22 +00:00
|
|
|
parse.debug_dump()
|
2019-02-08 11:49:43 +00:00
|
|
|
});
|
2018-09-10 18:14:09 +00:00
|
|
|
}
|
|
|
|
|
2020-06-18 21:43:19 +00:00
|
|
|
#[test]
|
|
|
|
fn expr_parser_tests() {
|
|
|
|
fragment_parser_dir_test(
|
|
|
|
&["parser/fragments/expr/ok"],
|
|
|
|
&["parser/fragments/expr/err"],
|
|
|
|
crate::ast::Expr::parse,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn path_parser_tests() {
|
|
|
|
fragment_parser_dir_test(
|
|
|
|
&["parser/fragments/path/ok"],
|
|
|
|
&["parser/fragments/path/err"],
|
|
|
|
crate::ast::Path::parse,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn pattern_parser_tests() {
|
|
|
|
fragment_parser_dir_test(
|
|
|
|
&["parser/fragments/pattern/ok"],
|
|
|
|
&["parser/fragments/pattern/err"],
|
|
|
|
crate::ast::Pat::parse,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn item_parser_tests() {
|
|
|
|
fragment_parser_dir_test(
|
|
|
|
&["parser/fragments/item/ok"],
|
|
|
|
&["parser/fragments/item/err"],
|
2020-07-29 22:23:03 +00:00
|
|
|
crate::ast::Item::parse,
|
2020-06-18 21:43:19 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn type_parser_tests() {
|
|
|
|
fragment_parser_dir_test(
|
|
|
|
&["parser/fragments/type/ok"],
|
|
|
|
&["parser/fragments/type/err"],
|
2020-07-31 10:06:38 +00:00
|
|
|
crate::ast::Type::parse,
|
2020-06-18 21:43:19 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-11-17 14:34:41 +00:00
|
|
|
#[test]
|
|
|
|
fn stmt_parser_tests() {
|
|
|
|
fragment_parser_dir_test(
|
|
|
|
&["parser/fragments/stmt/ok"],
|
|
|
|
&["parser/fragments/stmt/err"],
|
|
|
|
crate::ast::Stmt::parse,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-09-10 18:14:09 +00:00
|
|
|
#[test]
|
|
|
|
fn parser_fuzz_tests() {
|
2020-04-06 11:04:26 +00:00
|
|
|
for (_, text) in collect_rust_files(&test_data_dir(), &["parser/fuzz-failures"]) {
|
2019-03-21 17:05:12 +00:00
|
|
|
fuzz::check_parser(&text)
|
2018-09-10 18:14:09 +00:00
|
|
|
}
|
2018-08-25 11:45:17 +00:00
|
|
|
}
|
|
|
|
|
2019-03-21 17:06:48 +00:00
|
|
|
#[test]
|
|
|
|
fn reparse_fuzz_tests() {
|
2020-04-06 11:04:26 +00:00
|
|
|
for (_, text) in collect_rust_files(&test_data_dir(), &["reparse/fuzz-failures"]) {
|
2019-03-21 17:06:48 +00:00
|
|
|
let check = fuzz::CheckReparse::from_data(text.as_bytes()).unwrap();
|
|
|
|
check.run();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-11 14:15:20 +00:00
|
|
|
/// Test that Rust-analyzer can parse and validate the rust-analyzer
|
2018-12-09 19:19:23 +00:00
|
|
|
#[test]
|
|
|
|
fn self_hosting_parsing() {
|
2021-03-08 17:22:33 +00:00
|
|
|
let dir = project_root().join("crates");
|
2020-07-11 10:56:44 +00:00
|
|
|
let files = walkdir::WalkDir::new(dir)
|
2018-12-09 19:19:23 +00:00
|
|
|
.into_iter()
|
|
|
|
.filter_entry(|entry| {
|
2020-08-12 16:26:51 +00:00
|
|
|
// Get all files which are not in the crates/syntax/test_data folder
|
2020-07-11 10:56:44 +00:00
|
|
|
!entry.path().components().any(|component| component.as_os_str() == "test_data")
|
2018-12-09 19:19:23 +00:00
|
|
|
})
|
|
|
|
.map(|e| e.unwrap())
|
|
|
|
.filter(|entry| {
|
|
|
|
// Get all `.rs ` files
|
2020-07-11 10:56:44 +00:00
|
|
|
!entry.path().is_dir() && (entry.path().extension().unwrap_or_default() == "rs")
|
2018-12-09 19:19:23 +00:00
|
|
|
})
|
2020-07-11 10:56:44 +00:00
|
|
|
.map(|entry| entry.into_path())
|
|
|
|
.collect::<Vec<_>>();
|
2018-12-19 21:19:32 +00:00
|
|
|
assert!(
|
2020-07-11 10:56:44 +00:00
|
|
|
files.len() > 100,
|
2018-12-19 21:19:32 +00:00
|
|
|
"self_hosting_parsing found too few files - is it running in the right directory?"
|
2020-07-11 10:56:44 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
let errors = files
|
|
|
|
.into_par_iter()
|
|
|
|
.filter_map(|file| {
|
|
|
|
let text = read_text(&file);
|
|
|
|
match SourceFile::parse(&text).ok() {
|
|
|
|
Ok(_) => None,
|
|
|
|
Err(err) => Some((file, err)),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
|
|
if !errors.is_empty() {
|
|
|
|
let errors = errors
|
|
|
|
.into_iter()
|
|
|
|
.map(|(path, err)| format!("{}: {:?}\n", path.display(), err))
|
|
|
|
.collect::<String>();
|
|
|
|
panic!("Parsing errors:\n{}\n", errors);
|
|
|
|
}
|
2018-12-09 19:19:23 +00:00
|
|
|
}
|
2018-01-07 11:56:08 +00:00
|
|
|
|
|
|
|
fn test_data_dir() -> PathBuf {
|
2021-03-08 17:22:33 +00:00
|
|
|
project_root().join("crates/syntax/test_data")
|
2018-08-11 07:03:03 +00:00
|
|
|
}
|
|
|
|
|
2020-02-01 20:25:01 +00:00
|
|
|
fn assert_errors_are_present(errors: &[SyntaxError], path: &Path) {
|
|
|
|
assert!(!errors.is_empty(), "There should be errors in the file {:?}", path.display());
|
|
|
|
}
|
|
|
|
fn assert_errors_are_absent(errors: &[SyntaxError], path: &Path) {
|
|
|
|
assert_eq!(
|
|
|
|
errors,
|
|
|
|
&[] as &[SyntaxError],
|
|
|
|
"There should be no errors in the file {:?}",
|
|
|
|
path.display(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn dump_tokens_and_errors(tokens: &[Token], errors: &[SyntaxError], text: &str) -> String {
|
2018-08-11 07:03:03 +00:00
|
|
|
let mut acc = String::new();
|
2020-04-24 22:57:47 +00:00
|
|
|
let mut offset: TextSize = 0.into();
|
2018-08-11 07:03:03 +00:00
|
|
|
for token in tokens {
|
2020-03-13 02:29:44 +00:00
|
|
|
let token_len = token.len;
|
2020-04-24 21:40:41 +00:00
|
|
|
let token_text = &text[TextRange::at(offset, token.len)];
|
2020-03-13 02:29:44 +00:00
|
|
|
offset += token.len;
|
2020-04-24 21:40:41 +00:00
|
|
|
writeln!(acc, "{:?} {:?} {:?}", token.kind, token_len, token_text).unwrap();
|
2020-02-01 20:25:01 +00:00
|
|
|
}
|
|
|
|
for err in errors {
|
2020-02-10 00:08:49 +00:00
|
|
|
writeln!(acc, "> error{:?} token({:?}) msg({})", err.range(), &text[err.range()], err)
|
|
|
|
.unwrap();
|
2020-02-01 20:25:01 +00:00
|
|
|
}
|
2020-02-06 00:33:18 +00:00
|
|
|
acc
|
2018-02-03 09:51:06 +00:00
|
|
|
}
|
2020-06-18 21:43:19 +00:00
|
|
|
|
|
|
|
fn fragment_parser_dir_test<T, F>(ok_paths: &[&str], err_paths: &[&str], f: F)
|
|
|
|
where
|
|
|
|
T: crate::AstNode,
|
|
|
|
F: Fn(&str) -> Result<T, ()>,
|
|
|
|
{
|
|
|
|
dir_tests(&test_data_dir(), ok_paths, "rast", |text, path| {
|
|
|
|
if let Ok(node) = f(text) {
|
|
|
|
format!("{:#?}", crate::ast::AstNode::syntax(&node))
|
|
|
|
} else {
|
|
|
|
panic!("Failed to parse '{:?}'", path);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
dir_tests(&test_data_dir(), err_paths, "rast", |text, path| {
|
|
|
|
if let Ok(_) = f(text) {
|
|
|
|
panic!("'{:?}' successfully parsed when it should have errored", path);
|
|
|
|
} else {
|
|
|
|
"ERROR\n".to_owned()
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2020-07-01 10:30:17 +00:00
|
|
|
|
|
|
|
/// Calls callback `f` with input code and file paths for each `.rs` file in `test_data_dir`
|
|
|
|
/// subdirectories defined by `paths`.
|
|
|
|
///
|
|
|
|
/// If the content of the matching output file differs from the output of `f()`
|
|
|
|
/// the test will fail.
|
|
|
|
///
|
|
|
|
/// If there is no matching output file it will be created and filled with the
|
|
|
|
/// output of `f()`, but the test will fail.
|
|
|
|
fn dir_tests<F>(test_data_dir: &Path, paths: &[&str], outfile_extension: &str, f: F)
|
|
|
|
where
|
|
|
|
F: Fn(&str, &Path) -> String,
|
|
|
|
{
|
|
|
|
for (path, input_code) in collect_rust_files(test_data_dir, paths) {
|
|
|
|
let actual = f(&input_code, &path);
|
|
|
|
let path = path.with_extension(outfile_extension);
|
2020-07-09 09:47:27 +00:00
|
|
|
expect_file![path].assert_eq(&actual)
|
2020-07-01 10:30:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Collects all `.rs` files from `dir` subdirectories defined by `paths`.
|
|
|
|
fn collect_rust_files(root_dir: &Path, paths: &[&str]) -> Vec<(PathBuf, String)> {
|
|
|
|
paths
|
|
|
|
.iter()
|
|
|
|
.flat_map(|path| {
|
|
|
|
let path = root_dir.to_owned().join(path);
|
|
|
|
rust_files_in_dir(&path).into_iter()
|
|
|
|
})
|
|
|
|
.map(|path| {
|
|
|
|
let text = read_text(&path);
|
|
|
|
(path, text)
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Collects paths to all `.rs` files from `dir` in a sorted `Vec<PathBuf>`.
|
|
|
|
fn rust_files_in_dir(dir: &Path) -> Vec<PathBuf> {
|
|
|
|
let mut acc = Vec::new();
|
|
|
|
for file in fs::read_dir(&dir).unwrap() {
|
|
|
|
let file = file.unwrap();
|
|
|
|
let path = file.path();
|
|
|
|
if path.extension().unwrap_or_default() == "rs" {
|
|
|
|
acc.push(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
acc.sort();
|
|
|
|
acc
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Read file and normalize newlines.
|
|
|
|
///
|
|
|
|
/// `rustc` seems to always normalize `\r\n` newlines to `\n`:
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// let s = "
|
|
|
|
/// ";
|
|
|
|
/// assert_eq!(s.as_bytes(), &[10]);
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// so this should always be correct.
|
|
|
|
fn read_text(path: &Path) -> String {
|
|
|
|
fs::read_to_string(path)
|
|
|
|
.unwrap_or_else(|_| panic!("File at {:?} should be valid", path))
|
|
|
|
.replace("\r\n", "\n")
|
|
|
|
}
|