use crate::{error::PsError, output::Output, Result, POWERSHELL_NAME};
use std::os::windows::process::CommandExt;
use std::{
env,
io::Write,
path::Path,
process::{self, Command, Stdio},
};
const PATH_SPLITTER: char = ';';
const CREATE_NO_WINDOW: u32 = 0x08000000;
pub struct PsScript {
pub(crate) args: Vec<&'static str>,
pub(crate) hidden: bool,
pub(crate) print_commands: bool,
}
impl PsScript {
pub fn run(&self, script: &str) -> Result<Output> {
let proc_output = self.run_raw(script)?;
let output = Output::from(proc_output);
if output.success {
Ok(output)
} else {
Err(PsError::Powershell(output))
}
}
fn run_raw(&self, script: &str) -> Result<process::Output> {
let pws_path = get_powershell_path()?;
let mut cmd = Command::new(pws_path);
cmd.stdin(Stdio::piped());
cmd.stdout(Stdio::piped());
cmd.stderr(Stdio::piped());
cmd.args(&self.args);
if self.hidden {
cmd.creation_flags(CREATE_NO_WINDOW);
}
let mut process = cmd.spawn()?;
let stdin = process.stdin.as_mut().ok_or(PsError::ChildStdinNotFound)?;
for line in script.lines() {
if self.print_commands {
println!("{}", line)
};
writeln!(stdin, "{}", line)?;
}
let output = process.wait_with_output()?;
Ok(output)
}
}
fn is_program_on_path(program_name: &str) -> Option<bool> {
let system_path = match env::var("PATH") {
Ok(x) => x,
Err(_e) => return None,
};
for path_dir in system_path.split(PATH_SPLITTER) {
let path = std::path::Path::new(path_dir).join(&program_name);
if path.exists() {
return Some(true);
}
}
return Some(false);
}
fn get_powershell_path() -> Result<String> {
if is_program_on_path(POWERSHELL_NAME).unwrap() {
return Ok(POWERSHELL_NAME.to_string());
}
let system_root = match env::var("SYSTEMROOT") {
Ok(x) => x,
Err(_e) => return Err(PsError::PowershellNotFound),
};
let path_candidate =
Path::new(&system_root).join(r#"System32\WindowsPowerShell\v1.0\powershell.exe"#);
if path_candidate.exists() {
Ok(path_candidate.to_string_lossy().to_string())
} else {
Err(PsError::PowershellNotFound)
}
}