From 1d4b57cfd29dcefaffbee20b120309d670a9e384 Mon Sep 17 00:00:00 2001 From: Marvin Hagemeister Date: Fri, 22 Nov 2024 18:57:39 +0100 Subject: [PATCH] feat(task): add support for task wildcards --- cli/tools/task.rs | 253 +++++++++++--------- tests/specs/task/wildcard/__test__.jsonc | 12 + tests/specs/task/wildcard/deno.json | 15 ++ tests/specs/task/wildcard/wildcard.out | 6 + tests/specs/task/wildcard/wildcard_deps.out | 6 + 5 files changed, 180 insertions(+), 112 deletions(-) create mode 100644 tests/specs/task/wildcard/__test__.jsonc create mode 100644 tests/specs/task/wildcard/deno.json create mode 100644 tests/specs/task/wildcard/wildcard.out create mode 100644 tests/specs/task/wildcard/wildcard_deps.out diff --git a/cli/tools/task.rs b/cli/tools/task.rs index 478853f4e64f61..5da79438047181 100644 --- a/cli/tools/task.rs +++ b/cli/tools/task.rs @@ -27,6 +27,7 @@ use deno_path_util::normalize_path; use deno_runtime::deno_node::NodeResolver; use deno_task_shell::ShellCommand; use indexmap::IndexMap; +use indexmap::IndexSet; use regex::Regex; use crate::args::CliOptions; @@ -65,145 +66,167 @@ pub async fn execute_script( }) .unwrap_or(false); - fn arg_to_regex(input: &str) -> Result { + fn arg_to_regex( + input: &str, + exact: bool, + ) -> Result { let mut regex_str = regex::escape(input); regex_str = regex_str.replace("\\*", ".*"); + let prefix = if exact && !regex_str.starts_with('^') { + "^" + } else { + "" + }; + let suffix = if exact && !regex_str.ends_with('$') { + "$" + } else { + "" + }; - Regex::new(®ex_str) + Regex::new(&format!("{}{}{}", prefix, regex_str, suffix)) } - let packages_task_configs: Vec = if let Some(filter) = - &task_flags.filter - { - let task_name = task_flags.task.as_ref().unwrap(); - - // Filter based on package name - let package_regex = arg_to_regex(filter)?; - let task_regex = arg_to_regex(task_name)?; - - let mut packages_task_info: Vec = vec![]; - - fn matches_package( - config: &FolderConfigs, - force_use_pkg_json: bool, - regex: &Regex, - ) -> bool { - if !force_use_pkg_json { - if let Some(deno_json) = &config.deno_json { - if let Some(name) = &deno_json.json.name { - if regex.is_match(name) { - return true; - } - } - } + // Any of the matched tasks could be a child task of another matched + // one. Therefore we need to filter these out to ensure that every + // task is only run once. + fn match_tasks( + tasks_config: &WorkspaceTasksConfig, + task_regex: &Regex, + ) -> Vec { + let mut matched: IndexSet = IndexSet::new(); + let mut visited: HashSet = HashSet::new(); + + fn visit_task( + tasks_config: &WorkspaceTasksConfig, + visited: &mut HashSet, + name: &str, + ) { + if visited.contains(name) { + return; } - if let Some(package_json) = &config.pkg_json { - if let Some(name) = &package_json.name { - if regex.is_match(name) { - return true; - } + visited.insert(name.to_string()); + + if let Some((_, TaskOrScript::Task(_, task))) = &tasks_config.task(name) { + for dep in &task.dependencies { + visit_task(tasks_config, visited, dep); } } - - false } - let workspace = cli_options.workspace(); - for folder in workspace.config_folders() { - if !matches_package(folder.1, force_use_pkg_json, &package_regex) { - continue; + // Match tasks in deno.json + for name in tasks_config.task_names() { + if task_regex.is_match(name) && !visited.contains(name) { + matched.insert(name.to_string()); + visit_task(&tasks_config, &mut visited, name); } + } - let member_dir = workspace.resolve_member_dir(folder.0); - let mut tasks_config = member_dir.to_tasks_config()?; - if force_use_pkg_json { - tasks_config = tasks_config.with_only_pkg_json(); - } + matched.iter().map(|s| s.to_string()).collect::>() + } - // Any of the matched tasks could be a child task of another matched - // one. Therefore we need to filter these out to ensure that every - // task is only run once. - let mut matched: HashSet = HashSet::new(); - let mut visited: HashSet = HashSet::new(); - - fn visit_task( - tasks_config: &WorkspaceTasksConfig, - visited: &mut HashSet, - name: &str, - ) { - if visited.contains(name) { - return; + let (packages_task_configs, task_name): (Vec, &str) = + if let Some(filter) = &task_flags.filter { + let task_name = task_flags.task.as_ref().unwrap(); + + // Filter based on package name + let package_regex = arg_to_regex(filter, false)?; + let task_regex = arg_to_regex(task_name, true)?; + + let mut packages_task_info: Vec = vec![]; + + fn matches_package( + config: &FolderConfigs, + force_use_pkg_json: bool, + regex: &Regex, + ) -> bool { + if !force_use_pkg_json { + if let Some(deno_json) = &config.deno_json { + if let Some(name) = &deno_json.json.name { + if regex.is_match(name) { + return true; + } + } + } } - visited.insert(name.to_string()); - - if let Some((_, TaskOrScript::Task(_, task))) = &tasks_config.task(name) - { - for dep in &task.dependencies { - visit_task(tasks_config, visited, dep); + if let Some(package_json) = &config.pkg_json { + if let Some(name) = &package_json.name { + if regex.is_match(name) { + return true; + } } } + + false } - // Match tasks in deno.json - for name in tasks_config.task_names() { - if task_regex.is_match(name) && !visited.contains(name) { - matched.insert(name.to_string()); - visit_task(&tasks_config, &mut visited, name); + let workspace = cli_options.workspace(); + for folder in workspace.config_folders() { + if !matches_package(folder.1, force_use_pkg_json, &package_regex) { + continue; + } + + let member_dir = workspace.resolve_member_dir(folder.0); + let mut tasks_config = member_dir.to_tasks_config()?; + if force_use_pkg_json { + tasks_config = tasks_config.with_only_pkg_json(); } + + let matched_tasks = match_tasks(&tasks_config, &task_regex); + + packages_task_info.push(PackageTaskInfo { + matched_tasks, + tasks_config, + }); } - packages_task_info.push(PackageTaskInfo { - matched_tasks: matched - .iter() - .map(|s| s.to_string()) - .collect::>(), - tasks_config, - }); - } + // Logging every task definition would be too spammy. Pnpm only + // logs a simple message too. + if packages_task_info + .iter() + .all(|config| config.matched_tasks.is_empty()) + { + log::warn!( + "{}", + colors::red(format!( + "No matching task or script '{}' found in selected packages.", + task_name + )) + ); + return Ok(0); + } - // Logging every task definition would be too spammy. Pnpm only - // logs a simple message too. - if packages_task_info - .iter() - .all(|config| config.matched_tasks.is_empty()) - { - log::warn!( - "{}", - colors::red(format!( - "No matching task or script '{}' found in selected packages.", - task_name - )) - ); - return Ok(0); - } + // TODO: Sort packages topologically - // FIXME: Sort packages topologically - // + (packages_task_info, task_name) + } else { + let mut tasks_config = start_dir.to_tasks_config()?; - packages_task_info - } else { - let mut tasks_config = start_dir.to_tasks_config()?; + if force_use_pkg_json { + tasks_config = tasks_config.with_only_pkg_json() + } - if force_use_pkg_json { - tasks_config = tasks_config.with_only_pkg_json() - } + let Some(task_name) = &task_flags.task else { + print_available_tasks( + &mut std::io::stdout(), + &cli_options.start_dir, + &tasks_config, + )?; + return Ok(0); + }; - let Some(task_name) = &task_flags.task else { - print_available_tasks( - &mut std::io::stdout(), - &cli_options.start_dir, - &tasks_config, - )?; - return Ok(0); - }; + let task_regex = arg_to_regex(task_name, true)?; + let matched_tasks = match_tasks(&tasks_config, &task_regex); - vec![PackageTaskInfo { - tasks_config, - matched_tasks: vec![task_name.to_string()], - }] - }; + ( + vec![PackageTaskInfo { + tasks_config, + matched_tasks, + }], + task_name, + ) + }; let npm_resolver = factory.npm_resolver().await?; let node_resolver = factory.node_resolver().await?; @@ -240,7 +263,7 @@ pub async fn execute_script( } for task_config in &packages_task_configs { - let exit_code = task_runner.run_tasks(task_config).await?; + let exit_code = task_runner.run_tasks(task_config, task_name).await?; if exit_code > 0 { return Ok(exit_code); } @@ -269,8 +292,9 @@ impl<'a> TaskRunner<'a> { pub async fn run_tasks( &self, pkg_tasks_config: &PackageTaskInfo, + task_name: &str, ) -> Result { - match sort_tasks_topo(pkg_tasks_config) { + match sort_tasks_topo(pkg_tasks_config, task_name) { Ok(sorted) => self.run_tasks_in_parallel(sorted).await, Err(err) => match err { TaskError::NotFound(name) => { @@ -526,6 +550,7 @@ struct ResolvedTask<'a> { fn sort_tasks_topo<'a>( pkg_task_config: &'a PackageTaskInfo, + task_name: &str, ) -> Result>, TaskError> { trait TasksConfig { fn task( @@ -620,6 +645,10 @@ fn sort_tasks_topo<'a>( sort_visit(name, &mut sorted, Vec::new(), &pkg_task_config.tasks_config)?; } + if sorted.is_empty() { + return Err(TaskError::NotFound(task_name.to_string())); + } + Ok(sorted) } diff --git a/tests/specs/task/wildcard/__test__.jsonc b/tests/specs/task/wildcard/__test__.jsonc new file mode 100644 index 00000000000000..83d37432346304 --- /dev/null +++ b/tests/specs/task/wildcard/__test__.jsonc @@ -0,0 +1,12 @@ +{ + "tests": { + "wildcard": { + "args": "task foo-*", + "output": "wildcard.out" + }, + "wildcard_deps": { + "args": "task dep-*", + "output": "wildcard_deps.out" + } + } +} diff --git a/tests/specs/task/wildcard/deno.json b/tests/specs/task/wildcard/deno.json new file mode 100644 index 00000000000000..82feca68292b5a --- /dev/null +++ b/tests/specs/task/wildcard/deno.json @@ -0,0 +1,15 @@ +{ + "tasks": { + "foo-1": "echo 'foo-1'", + "foo-2": "echo 'foo-2'", + "foo-3": "echo 'foo-3'", + "dep-1": { + "command": "echo 'dep-1'", + "dependencies": ["dep-2", "foo-1"] + }, + "dep-2": { + "command": "echo 'dep-2'", + "dependencies": ["foo-1"] + } + } +} diff --git a/tests/specs/task/wildcard/wildcard.out b/tests/specs/task/wildcard/wildcard.out new file mode 100644 index 00000000000000..fe04f12c7f9e17 --- /dev/null +++ b/tests/specs/task/wildcard/wildcard.out @@ -0,0 +1,6 @@ +Task foo-1 echo 'foo-1' +foo-1 +Task foo-2 echo 'foo-2' +foo-2 +Task foo-3 echo 'foo-3' +foo-3 diff --git a/tests/specs/task/wildcard/wildcard_deps.out b/tests/specs/task/wildcard/wildcard_deps.out new file mode 100644 index 00000000000000..9fc9916c492ffe --- /dev/null +++ b/tests/specs/task/wildcard/wildcard_deps.out @@ -0,0 +1,6 @@ +Task foo-1 echo 'foo-1' +foo-1 +Task dep-2 echo 'dep-2' +dep-2 +Task dep-1 echo 'dep-1' +dep-1