feat(cli): add include/exclude options to the test subcommand
This commit is contained in:
parent
fcfc934a27
commit
a4a7edc5af
2 changed files with 77 additions and 43 deletions
|
|
@ -1,10 +1,12 @@
|
|||
use anyhow::{anyhow, Context, Error, Result};
|
||||
use clap::{App, AppSettings, Arg, SubCommand};
|
||||
use glob::glob;
|
||||
use regex::Regex;
|
||||
use std::collections::HashSet;
|
||||
use std::path::{Path, PathBuf};
|
||||
use std::{env, fs, u64};
|
||||
use tree_sitter::{ffi, Parser, Point};
|
||||
use tree_sitter_cli::test::TestOptions;
|
||||
use tree_sitter_cli::{
|
||||
generate, highlight, logger,
|
||||
parse::{self, ParseFileOptions, ParseOutput},
|
||||
|
|
@ -236,7 +238,23 @@ fn run() -> Result<()> {
|
|||
.long("filter")
|
||||
.short("f")
|
||||
.takes_value(true)
|
||||
.help("Only run corpus test cases whose name includes the given string"),
|
||||
.help("[DEPRECATED in favor of include]\nOnly run corpus test cases whose name includes the given string"),
|
||||
)
|
||||
.arg(
|
||||
Arg::with_name("include")
|
||||
.long("include")
|
||||
.short("i")
|
||||
.takes_value(true)
|
||||
.help("Only run corpus test cases whose name matches the given regex"),
|
||||
)
|
||||
.arg(
|
||||
Arg::with_name("exclude")
|
||||
.long("exclude")
|
||||
.short("e")
|
||||
.takes_value(true)
|
||||
.help(
|
||||
"Only run corpus test cases whose name does not match the given regex",
|
||||
),
|
||||
)
|
||||
.arg(
|
||||
Arg::with_name("update")
|
||||
|
|
@ -389,6 +407,10 @@ fn run() -> Result<()> {
|
|||
let debug_build = matches.is_present("debug-build");
|
||||
let update = matches.is_present("update");
|
||||
let filter = matches.value_of("filter");
|
||||
let include: Option<Regex> =
|
||||
matches.value_of("include").and_then(|s| Regex::new(s).ok());
|
||||
let exclude: Option<Regex> =
|
||||
matches.value_of("exclude").and_then(|s| Regex::new(s).ok());
|
||||
let apply_all_captures = matches.is_present("apply-all-captures");
|
||||
|
||||
if debug {
|
||||
|
|
@ -423,14 +445,17 @@ fn run() -> Result<()> {
|
|||
test_corpus_dir = current_dir.join("corpus");
|
||||
}
|
||||
if test_corpus_dir.is_dir() {
|
||||
test::run_tests_at_path(
|
||||
&mut parser,
|
||||
&test_corpus_dir,
|
||||
let mut opts = TestOptions {
|
||||
path: test_corpus_dir,
|
||||
debug,
|
||||
debug_graph,
|
||||
filter,
|
||||
include,
|
||||
exclude,
|
||||
update,
|
||||
)?;
|
||||
};
|
||||
|
||||
test::run_tests_at_path(&mut parser, &mut opts)?;
|
||||
}
|
||||
|
||||
// Check that all of the queries are valid.
|
||||
|
|
|
|||
|
|
@ -56,20 +56,23 @@ impl Default for TestEntry {
|
|||
}
|
||||
}
|
||||
|
||||
pub fn run_tests_at_path(
|
||||
parser: &mut Parser,
|
||||
path: &Path,
|
||||
debug: bool,
|
||||
debug_graph: bool,
|
||||
filter: Option<&str>,
|
||||
update: bool,
|
||||
) -> Result<()> {
|
||||
let test_entry = parse_tests(path)?;
|
||||
pub struct TestOptions<'a> {
|
||||
pub path: PathBuf,
|
||||
pub debug: bool,
|
||||
pub debug_graph: bool,
|
||||
pub filter: Option<&'a str>,
|
||||
pub include: Option<Regex>,
|
||||
pub exclude: Option<Regex>,
|
||||
pub update: bool,
|
||||
}
|
||||
|
||||
pub fn run_tests_at_path(parser: &mut Parser, opts: &mut TestOptions) -> Result<()> {
|
||||
let test_entry = parse_tests(&opts.path)?;
|
||||
let mut _log_session = None;
|
||||
|
||||
if debug_graph {
|
||||
if opts.debug_graph {
|
||||
_log_session = Some(util::log_graphs(parser, "log.html")?);
|
||||
} else if debug {
|
||||
} else if opts.debug {
|
||||
parser.set_logger(Some(Box::new(|log_type, message| {
|
||||
if log_type == LogType::Lex {
|
||||
io::stderr().write_all(b" ").unwrap();
|
||||
|
|
@ -83,10 +86,9 @@ pub fn run_tests_at_path(
|
|||
run_tests(
|
||||
parser,
|
||||
test_entry,
|
||||
filter,
|
||||
opts,
|
||||
0,
|
||||
&mut failures,
|
||||
update,
|
||||
&mut corrected_entries,
|
||||
)?;
|
||||
|
||||
|
|
@ -97,7 +99,7 @@ pub fn run_tests_at_path(
|
|||
} else {
|
||||
println!();
|
||||
|
||||
if update {
|
||||
if opts.update {
|
||||
if failures.len() == 1 {
|
||||
println!("1 update:\n");
|
||||
} else {
|
||||
|
|
@ -177,10 +179,9 @@ pub fn print_diff(actual: &str, expected: &str) {
|
|||
fn run_tests(
|
||||
parser: &mut Parser,
|
||||
test_entry: TestEntry,
|
||||
filter: Option<&str>,
|
||||
opts: &mut TestOptions,
|
||||
mut indent_level: i32,
|
||||
failures: &mut Vec<(String, String, String)>,
|
||||
update: bool,
|
||||
corrected_entries: &mut Vec<(String, String, String, usize, usize)>,
|
||||
) -> Result<()> {
|
||||
match test_entry {
|
||||
|
|
@ -192,22 +193,6 @@ fn run_tests(
|
|||
divider_delim_len,
|
||||
has_fields,
|
||||
} => {
|
||||
if let Some(filter) = filter {
|
||||
if !name.contains(filter) {
|
||||
if update {
|
||||
let input = String::from_utf8(input).unwrap();
|
||||
let output = format_sexp(&output);
|
||||
corrected_entries.push((
|
||||
name,
|
||||
input,
|
||||
output,
|
||||
header_delim_len,
|
||||
divider_delim_len,
|
||||
));
|
||||
}
|
||||
return Ok(());
|
||||
}
|
||||
}
|
||||
let tree = parser.parse(&input, None).unwrap();
|
||||
let mut actual = tree.root_node().to_sexp();
|
||||
if !has_fields {
|
||||
|
|
@ -216,7 +201,7 @@ fn run_tests(
|
|||
print!("{}", " ".repeat(indent_level as usize));
|
||||
if actual == output {
|
||||
println!("✓ {}", Colour::Green.paint(&name));
|
||||
if update {
|
||||
if opts.update {
|
||||
let input = String::from_utf8(input).unwrap();
|
||||
let output = format_sexp(&output);
|
||||
corrected_entries.push((
|
||||
|
|
@ -228,7 +213,7 @@ fn run_tests(
|
|||
));
|
||||
}
|
||||
} else {
|
||||
if update {
|
||||
if opts.update {
|
||||
let input = String::from_utf8(input).unwrap();
|
||||
let output = format_sexp(&actual);
|
||||
corrected_entries.push((
|
||||
|
|
@ -247,9 +232,34 @@ fn run_tests(
|
|||
}
|
||||
TestEntry::Group {
|
||||
name,
|
||||
children,
|
||||
mut children,
|
||||
file_path,
|
||||
} => {
|
||||
children.retain(|child| {
|
||||
if let TestEntry::Example { name, .. } = child {
|
||||
if let Some(filter) = opts.filter {
|
||||
if !name.contains(filter) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if let Some(include) = &opts.include {
|
||||
if !include.is_match(name) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if let Some(exclude) = &opts.exclude {
|
||||
if exclude.is_match(name) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
true
|
||||
});
|
||||
|
||||
if children.is_empty() {
|
||||
return Ok(());
|
||||
}
|
||||
|
||||
if indent_level > 0 {
|
||||
print!("{}", " ".repeat(indent_level as usize));
|
||||
println!("{name}:");
|
||||
|
|
@ -262,16 +272,15 @@ fn run_tests(
|
|||
run_tests(
|
||||
parser,
|
||||
child,
|
||||
filter,
|
||||
opts,
|
||||
indent_level,
|
||||
failures,
|
||||
update,
|
||||
corrected_entries,
|
||||
)?;
|
||||
}
|
||||
|
||||
if let Some(file_path) = file_path {
|
||||
if update && failures.len() - failure_count > 0 {
|
||||
if opts.update && failures.len() - failure_count > 0 {
|
||||
write_tests(&file_path, corrected_entries)?;
|
||||
}
|
||||
corrected_entries.clear();
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue