use super::EnvironmentVariable; use crate::fs::{binaries as test_bins_path, executable_path}; use std::{ ffi::{OsStr, OsString}, fmt, process::{Command, ExitStatus}, }; pub trait Executable { fn execute(&mut self) -> Result; } #[derive(Clone, Debug)] pub struct Outcome { pub out: Vec, pub err: Vec, } impl Outcome { pub fn new(out: &[u8], err: &[u8]) -> Outcome { Outcome { out: out.to_vec(), err: err.to_vec(), } } } #[derive(Debug)] pub struct NuError { pub desc: String, pub exit: Option, pub output: Option, } #[derive(Clone, Debug, Default)] pub struct NuProcess { pub arguments: Vec, pub environment_vars: Vec, pub cwd: Option, } impl fmt::Display for NuProcess { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!(f, "`nu")?; for arg in &self.arguments { write!(f, " {}", arg.to_string_lossy())?; } write!(f, "`") } } impl NuProcess { pub fn arg>(&mut self, arg: T) -> &mut Self { self.arguments.push(arg.as_ref().to_os_string()); self } pub fn args>(&mut self, arguments: &[T]) -> &mut NuProcess { self.arguments .extend(arguments.iter().map(|t| t.as_ref().to_os_string())); self } pub fn cwd>(&mut self, path: T) -> &mut NuProcess { self.cwd = Some(path.as_ref().to_os_string()); self } pub fn construct(&self) -> Command { let mut command = Command::new(executable_path()); if let Some(cwd) = &self.cwd { command.current_dir(cwd); } command.env_clear(); let paths = [test_bins_path()]; let paths_joined = match std::env::join_paths(paths) { Ok(all) => all, Err(_) => panic!("Couldn't join paths for PATH var."), }; command.env(crate::NATIVE_PATH_ENV_VAR, paths_joined); for env_var in &self.environment_vars { command.env(&env_var.name, &env_var.value); } for arg in &self.arguments { command.arg(arg); } command } }