mirror of
https://github.com/nushell/nushell
synced 2025-01-15 14:44:14 +00:00
Update external spawn (#406)
* Simplify external spawn, improve arg cleaning * Fix tests * Fix windows test
This commit is contained in:
parent
ff673ba0ba
commit
ccd5f59314
2 changed files with 65 additions and 16 deletions
|
@ -10,7 +10,10 @@ use nu_protocol::engine::{EngineState, Stack};
|
|||
use nu_protocol::{ast::Call, engine::Command, ShellError, Signature, SyntaxShape, Value};
|
||||
use nu_protocol::{Category, Config, IntoInterruptiblePipelineData, PipelineData, Span, Spanned};
|
||||
|
||||
use itertools::Itertools;
|
||||
|
||||
use nu_engine::CallExt;
|
||||
use regex::Regex;
|
||||
|
||||
const OUTPUT_BUFFER_SIZE: usize = 8192;
|
||||
|
||||
|
@ -206,7 +209,32 @@ impl ExternalCommand {
|
|||
//TODO. This should be modifiable from the config file.
|
||||
// We could give the option to call from powershell
|
||||
// for minimal builds cwd is unused
|
||||
let mut process = CommandSys::new("cmd");
|
||||
if self.name.item.ends_with(".cmd") || self.name.item.ends_with(".bat") {
|
||||
self.spawn_cmd_command()
|
||||
} else {
|
||||
self.spawn_simple_command()
|
||||
}
|
||||
} else if self.name.item.ends_with(".sh") {
|
||||
self.spawn_sh_command()
|
||||
} else {
|
||||
self.spawn_simple_command()
|
||||
}
|
||||
}
|
||||
|
||||
/// Spawn a command without shelling out to an external shell
|
||||
fn spawn_simple_command(&self) -> std::process::Command {
|
||||
let mut process = std::process::Command::new(&self.name.item);
|
||||
|
||||
for arg in &self.args {
|
||||
process.arg(&arg);
|
||||
}
|
||||
|
||||
process
|
||||
}
|
||||
|
||||
/// Spawn a cmd command with `cmd /c args...`
|
||||
fn spawn_cmd_command(&self) -> std::process::Command {
|
||||
let mut process = std::process::Command::new("cmd");
|
||||
process.arg("/c");
|
||||
process.arg(&self.name.item);
|
||||
for arg in &self.args {
|
||||
|
@ -217,13 +245,34 @@ impl ExternalCommand {
|
|||
process.arg(&arg);
|
||||
}
|
||||
process
|
||||
} else {
|
||||
let cmd_with_args = vec![self.name.item.clone(), self.args.join(" ")].join(" ");
|
||||
let mut process = CommandSys::new("sh");
|
||||
}
|
||||
|
||||
/// Spawn a sh command with `sh -c args...`
|
||||
fn spawn_sh_command(&self) -> std::process::Command {
|
||||
let joined_and_escaped_arguments =
|
||||
self.args.iter().map(|arg| shell_arg_escape(arg)).join(" ");
|
||||
let cmd_with_args = vec![self.name.item.clone(), joined_and_escaped_arguments].join(" ");
|
||||
let mut process = std::process::Command::new("sh");
|
||||
process.arg("-c").arg(cmd_with_args);
|
||||
process
|
||||
}
|
||||
}
|
||||
|
||||
fn has_unsafe_shell_characters(arg: &str) -> bool {
|
||||
let re: Regex = Regex::new(r"[^\w@%+=:,./-]").expect("regex to be valid");
|
||||
|
||||
re.is_match(arg)
|
||||
}
|
||||
|
||||
fn shell_arg_escape(arg: &str) -> String {
|
||||
match arg {
|
||||
"" => String::from("''"),
|
||||
s if !has_unsafe_shell_characters(s) => String::from(s),
|
||||
_ => {
|
||||
let single_quotes_escaped = arg.split('\'').join("'\"'\"'");
|
||||
format!("'{}'", single_quotes_escaped)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// The piped data from stdout from the external command can be either String
|
||||
|
|
|
@ -56,9 +56,9 @@ fn fail_test(input: &str, expected: &str) -> TestResult {
|
|||
|
||||
fn not_found_msg() -> &'static str {
|
||||
if cfg!(windows) {
|
||||
"not recognized"
|
||||
"cannot find"
|
||||
} else {
|
||||
"not found"
|
||||
"No such"
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in a new issue