Improve the ShellExtension to allow the user to specify different shells

This commit is contained in:
Federico Terzi 2020-05-06 20:01:40 +02:00
parent 406372638e
commit 53e72bb8a3

View File

@ -18,7 +18,7 @@
*/ */
use serde_yaml::{Mapping, Value}; use serde_yaml::{Mapping, Value};
use std::process::Command; use std::process::{Command, Output};
use log::{warn, error}; use log::{warn, error};
use regex::{Regex, Captures}; use regex::{Regex, Captures};
@ -30,6 +30,71 @@ lazy_static! {
}; };
} }
pub enum Shell {
Cmd,
Powershell,
WSL,
Bash,
Sh,
}
impl Shell {
fn execute_cmd(&self, cmd: &str) -> std::io::Result<Output>{
match self {
Shell::Cmd => {
Command::new("cmd")
.args(&["/C", &cmd])
.output()
},
Shell::Powershell => {
Command::new("powershell")
.args(&["-Command", &cmd])
.output()
},
Shell::WSL => {
Command::new("wsl")
.args(&["bash", "-c", &cmd])
.output()
},
Shell::Bash => {
Command::new("bash")
.args(&["-c", &cmd])
.output()
},
Shell::Sh => {
Command::new("sh")
.args(&["-c", &cmd])
.output()
},
}
}
fn from_string(shell: &str) -> Option<Shell> {
match shell {
"cmd" => Some(Shell::Cmd),
"powershell" => Some(Shell::Powershell),
"wsl" => Some(Shell::WSL),
"bash" => Some(Shell::Bash),
"sh" => Some(Shell::Sh),
_ => None
}
}
}
impl Default for Shell {
fn default() -> Shell {
if cfg!(target_os = "windows") {
Shell::Powershell
}else if cfg!(target_os = "macos") {
Shell::Sh
}else if cfg!(target_os = "linux") {
Shell::Bash
}else{
panic!("invalid target os for shell")
}
}
}
pub struct ShellExtension {} pub struct ShellExtension {}
impl ShellExtension { impl ShellExtension {
@ -62,17 +127,23 @@ impl super::Extension for ShellExtension {
} }
}).to_string(); }).to_string();
let output = if cfg!(target_os = "windows") { let shell_param = params.get(&Value::from("shell"));
Command::new("powershell") let shell = if let Some(shell_param) = shell_param {
.args(&["-Command", &cmd]) let shell_param = shell_param.as_str().expect("invalid shell parameter");
.output() let shell = Shell::from_string(shell_param);
} else {
Command::new("sh") if shell.is_none() {
.arg("-c") error!("Invalid shell parameter, please select a valid one.");
.arg(&cmd) return None;
.output() }
shell.unwrap()
}else{
Shell::default()
}; };
let output = shell.execute_cmd(&cmd);
match output { match output {
Ok(output) => { Ok(output) => {
let output_str = String::from_utf8_lossy(output.stdout.as_slice()); let output_str = String::from_utf8_lossy(output.stdout.as_slice());
@ -115,7 +186,7 @@ mod tests {
#[test] #[test]
fn test_shell_basic() { fn test_shell_basic() {
let mut params = Mapping::new(); let mut params = Mapping::new();
params.insert(Value::from("cmd"), Value::from("echo hello world")); params.insert(Value::from("cmd"), Value::from("echo \"hello world\""));
let extension = ShellExtension::new(); let extension = ShellExtension::new();
let output = extension.calculate(&params, &vec![]); let output = extension.calculate(&params, &vec![]);
@ -132,7 +203,7 @@ mod tests {
#[test] #[test]
fn test_shell_trimmed() { fn test_shell_trimmed() {
let mut params = Mapping::new(); let mut params = Mapping::new();
params.insert(Value::from("cmd"), Value::from("echo hello world")); params.insert(Value::from("cmd"), Value::from("echo \"hello world\""));
params.insert(Value::from("trim"), Value::from(true)); params.insert(Value::from("trim"), Value::from(true));
let extension = ShellExtension::new(); let extension = ShellExtension::new();
@ -145,11 +216,7 @@ mod tests {
#[test] #[test]
fn test_shell_trimmed_2() { fn test_shell_trimmed_2() {
let mut params = Mapping::new(); let mut params = Mapping::new();
if cfg!(target_os = "windows") {
params.insert(Value::from("cmd"), Value::from("echo hello world "));
}else{
params.insert(Value::from("cmd"), Value::from("echo \" hello world \"")); params.insert(Value::from("cmd"), Value::from("echo \" hello world \""));
}
params.insert(Value::from("trim"), Value::from(true)); params.insert(Value::from("trim"), Value::from(true));
@ -163,7 +230,7 @@ mod tests {
#[test] #[test]
fn test_shell_trimmed_malformed() { fn test_shell_trimmed_malformed() {
let mut params = Mapping::new(); let mut params = Mapping::new();
params.insert(Value::from("cmd"), Value::from("echo hello world")); params.insert(Value::from("cmd"), Value::from("echo \"hello world\""));
params.insert(Value::from("trim"), Value::from("error")); params.insert(Value::from("trim"), Value::from("error"));
let extension = ShellExtension::new(); let extension = ShellExtension::new();