2019-01-10 17:11:57 -08:00
|
|
|
use super::languages;
|
2019-01-11 13:30:45 -08:00
|
|
|
use crate::generate;
|
|
|
|
|
use crate::loader::Loader;
|
2019-01-10 17:11:57 -08:00
|
|
|
use crate::test::{parse_tests, TestEntry};
|
2019-01-11 13:30:45 -08:00
|
|
|
use std::fs;
|
2019-01-11 17:26:45 -08:00
|
|
|
use std::path::PathBuf;
|
2019-01-10 17:11:57 -08:00
|
|
|
use tree_sitter::{Language, Parser};
|
|
|
|
|
|
|
|
|
|
lazy_static! {
|
|
|
|
|
static ref LANGUAGES: [(&'static str, Language); 6] = [
|
|
|
|
|
("c", languages::c()),
|
|
|
|
|
("cpp", languages::cpp()),
|
|
|
|
|
("embedded-template", languages::embedded_template()),
|
|
|
|
|
("go", languages::go()),
|
|
|
|
|
("html", languages::html()),
|
|
|
|
|
("javascript", languages::javascript()),
|
|
|
|
|
];
|
2019-01-11 13:30:45 -08:00
|
|
|
static ref ROOT_DIR: PathBuf = [env!("CARGO_MANIFEST_DIR"), ".."].iter().collect();
|
|
|
|
|
static ref HEADER_DIR: PathBuf = ROOT_DIR.join("lib").join("include");
|
|
|
|
|
static ref SCRATCH_DIR: PathBuf = ROOT_DIR.join("target").join("scratch");
|
|
|
|
|
static ref FIXTURES_DIR: PathBuf = ROOT_DIR.join("test").join("fixtures");
|
2019-01-11 17:26:45 -08:00
|
|
|
static ref EXEC_PATH: PathBuf = std::env::current_exe().unwrap();
|
2019-01-10 17:11:57 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
2019-01-11 13:30:45 -08:00
|
|
|
fn test_real_language_corpus_files() {
|
2019-01-10 17:11:57 -08:00
|
|
|
let mut parser = Parser::new();
|
2019-01-11 13:30:45 -08:00
|
|
|
let grammars_dir = FIXTURES_DIR.join("grammars");
|
2019-01-10 17:11:57 -08:00
|
|
|
|
|
|
|
|
for (name, language) in LANGUAGES.iter().cloned() {
|
|
|
|
|
let corpus_dir = grammars_dir.join(name).join("corpus");
|
|
|
|
|
let test = parse_tests(&corpus_dir).unwrap();
|
|
|
|
|
parser.set_language(language).unwrap();
|
|
|
|
|
run_mutation_tests(&mut parser, test);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-11 13:30:45 -08:00
|
|
|
#[test]
|
|
|
|
|
fn test_feature_corpus_files() {
|
|
|
|
|
fs::create_dir_all(SCRATCH_DIR.as_path()).unwrap();
|
|
|
|
|
|
2019-01-11 17:26:45 -08:00
|
|
|
let filter = std::env::var("TREE_SITTER_TEST_FILTER").ok();
|
2019-01-11 13:30:45 -08:00
|
|
|
let mut loader = Loader::new(SCRATCH_DIR.clone());
|
|
|
|
|
let mut parser = Parser::new();
|
|
|
|
|
let test_grammars_dir = FIXTURES_DIR.join("test_grammars");
|
|
|
|
|
|
|
|
|
|
for entry in fs::read_dir(&test_grammars_dir).unwrap() {
|
|
|
|
|
let entry = entry.unwrap();
|
2019-01-11 17:26:45 -08:00
|
|
|
if !entry.metadata().unwrap().is_dir() {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2019-01-11 13:30:45 -08:00
|
|
|
let test_name = entry.file_name();
|
|
|
|
|
let test_name = test_name.to_str().unwrap();
|
|
|
|
|
|
2019-01-11 17:26:45 -08:00
|
|
|
if let Some(filter) = filter.as_ref() {
|
|
|
|
|
if !test_name.contains(filter.as_str()) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
eprintln!("test: {:?}", test_name);
|
|
|
|
|
|
2019-01-11 13:30:45 -08:00
|
|
|
let test_path = entry.path();
|
|
|
|
|
let grammar_path = test_path.join("grammar.json");
|
|
|
|
|
let error_message_path = test_path.join("expected_error.txt");
|
|
|
|
|
let grammar_json = fs::read_to_string(grammar_path).unwrap();
|
|
|
|
|
let generate_result = generate::generate_parser_for_grammar(&grammar_json);
|
2019-01-11 17:26:45 -08:00
|
|
|
|
2019-01-11 13:30:45 -08:00
|
|
|
if error_message_path.exists() {
|
2019-01-11 17:26:45 -08:00
|
|
|
let expected_message = fs::read_to_string(&error_message_path).unwrap();
|
2019-01-11 13:30:45 -08:00
|
|
|
if let Err(e) = generate_result {
|
2019-01-11 17:26:45 -08:00
|
|
|
if e.0 != expected_message {
|
|
|
|
|
panic!(
|
|
|
|
|
"Unexpected error message.\n\nExpected:\n\n{}\nActual:\n\n{}\n",
|
|
|
|
|
expected_message, e.0
|
|
|
|
|
);
|
|
|
|
|
}
|
2019-01-11 13:30:45 -08:00
|
|
|
} else {
|
|
|
|
|
panic!(
|
|
|
|
|
"Expected error message but got none for test grammar '{}'",
|
|
|
|
|
test_name
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2019-01-11 17:26:45 -08:00
|
|
|
let corpus_path = test_path.join("corpus.txt");
|
2019-01-11 13:30:45 -08:00
|
|
|
let c_code = generate_result.unwrap();
|
|
|
|
|
let parser_c_path = SCRATCH_DIR.join(&format!("{}-parser.c", test_name));
|
2019-01-11 17:26:45 -08:00
|
|
|
if !fs::read_to_string(&parser_c_path)
|
|
|
|
|
.map(|content| content == c_code)
|
|
|
|
|
.unwrap_or(false)
|
|
|
|
|
{
|
|
|
|
|
fs::write(&parser_c_path, c_code).unwrap();
|
|
|
|
|
}
|
2019-01-11 13:30:45 -08:00
|
|
|
let scanner_path = test_path.join("scanner.c");
|
|
|
|
|
let scanner_path = if scanner_path.exists() {
|
|
|
|
|
Some(scanner_path)
|
|
|
|
|
} else {
|
|
|
|
|
None
|
|
|
|
|
};
|
|
|
|
|
let language = loader
|
|
|
|
|
.load_language_from_sources(test_name, &HEADER_DIR, &parser_c_path, &scanner_path)
|
|
|
|
|
.unwrap();
|
2019-01-11 17:26:45 -08:00
|
|
|
let test = parse_tests(&corpus_path).unwrap();
|
2019-01-11 13:30:45 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// for (name, language) in LANGUAGES.iter().cloned() {
|
|
|
|
|
// let corpus_dir = grammars_dir.join(name).join("corpus");
|
|
|
|
|
// let test = parse_tests(&corpus_dir).unwrap();
|
|
|
|
|
// parser.set_language(language).unwrap();
|
|
|
|
|
// run_mutation_tests(&mut parser, test);
|
|
|
|
|
// }
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-10 17:11:57 -08:00
|
|
|
fn run_mutation_tests(parser: &mut Parser, test: TestEntry) {
|
|
|
|
|
match test {
|
|
|
|
|
TestEntry::Example {
|
|
|
|
|
name,
|
|
|
|
|
input,
|
|
|
|
|
output,
|
|
|
|
|
} => {
|
|
|
|
|
let tree = parser
|
|
|
|
|
.parse_utf8(&mut |byte_offset, _| &input[byte_offset..], None)
|
|
|
|
|
.unwrap();
|
|
|
|
|
let actual = tree.root_node().to_sexp();
|
|
|
|
|
assert_eq!(actual, output);
|
|
|
|
|
}
|
|
|
|
|
TestEntry::Group { name, children } => {
|
|
|
|
|
for child in children {
|
|
|
|
|
run_mutation_tests(parser, child);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|