blob: b95f3a5cfe5a8de17c9445ecbb9cabf69e2a6022 [file] [log] [blame]
use std::ffi::{OsStr, OsString};
use std::path::PathBuf;
use std::{env, panic};
use crate::command::{Command, CompletedProcess};
use crate::util::handle_failed_output;
use crate::{cwd, env_var};
#[track_caller]
fn run_common(name: &str, args: Option<&[&str]>) -> Command {
let mut bin_path = PathBuf::new();
bin_path.push(cwd());
bin_path.push(name);
let ld_lib_path_envvar = env_var("LD_LIB_PATH_ENVVAR");
let mut cmd = if let Some(rtc) = env::var_os("REMOTE_TEST_CLIENT") {
let mut cmd = Command::new(rtc);
cmd.arg("run");
// FIXME: the "0" indicates how many support files should be uploaded along with the binary
// to execute. If a test requires additional files to be pushed to the remote machine, this
// will have to be changed (and the support files will have to be uploaded).
cmd.arg("0");
cmd.arg(bin_path);
cmd
} else if let Ok(runner) = std::env::var("RUNNER") {
let mut args = split_maybe_args(&runner);
let prog = args.remove(0);
let mut cmd = Command::new(prog);
for arg in args {
cmd.arg(arg);
}
cmd.arg("--");
cmd.arg(bin_path);
cmd
} else {
Command::new(bin_path)
};
if let Some(args) = args {
for arg in args {
cmd.arg(arg);
}
}
cmd.env(&ld_lib_path_envvar, {
let mut paths = vec![];
paths.push(cwd());
for p in env::split_paths(&env_var("TARGET_EXE_DYLIB_PATH")) {
paths.push(p.to_path_buf());
}
for p in env::split_paths(&env_var(&ld_lib_path_envvar)) {
paths.push(p.to_path_buf());
}
env::join_paths(paths.iter()).unwrap()
});
cmd.env("LC_ALL", "C"); // force english locale
cmd
}
/// Run a built binary and make sure it succeeds.
#[track_caller]
pub fn run(name: &str) -> CompletedProcess {
let caller = panic::Location::caller();
let mut cmd = run_common(name, None);
let output = cmd.run();
if !output.status().success() {
handle_failed_output(&cmd, output, caller.line());
}
output
}
/// Run a built binary with one or more argument(s) and make sure it succeeds.
#[track_caller]
pub fn run_with_args(name: &str, args: &[&str]) -> CompletedProcess {
let caller = panic::Location::caller();
let mut cmd = run_common(name, Some(args));
let output = cmd.run();
if !output.status().success() {
handle_failed_output(&cmd, output, caller.line());
}
output
}
/// Run a built binary and make sure it fails.
#[track_caller]
pub fn run_fail(name: &str) -> CompletedProcess {
let caller = panic::Location::caller();
let mut cmd = run_common(name, None);
let output = cmd.run_fail();
if output.status().success() {
handle_failed_output(&cmd, output, caller.line());
}
output
}
/// Create a new custom [`Command`]. This should be preferred to creating [`std::process::Command`]
/// directly.
#[track_caller]
pub fn cmd<S: AsRef<OsStr>>(program: S) -> Command {
let mut command = Command::new(program);
command.env("LC_ALL", "C"); // force english locale
command
}
fn split_maybe_args(s: &str) -> Vec<OsString> {
// FIXME(132599): implement proper env var/shell argument splitting.
s.split(' ')
.filter_map(|s| {
if s.chars().all(|c| c.is_whitespace()) { None } else { Some(OsString::from(s)) }
})
.collect()
}