use std::env;
use std::ops::{Deref, DerefMut};
use std::path::PathBuf;
use std::process::{self, Command};
use anyhow::{Context, Result};
use clap::Parser;
use crate::options::XWinOptions;
#[derive(Clone, Debug, Default, Parser)]
#[command(
display_order = 1,
after_help = "Run `cargo help test` for more detailed information.\nRun `cargo test -- --help` for test binary options."
)]
pub struct Test {
#[command(flatten)]
pub xwin: XWinOptions,
#[command(flatten)]
pub cargo: cargo_options::Test,
}
impl Test {
#[allow(clippy::field_reassign_with_default)]
pub fn new(manifest_path: Option<PathBuf>) -> Self {
let mut build = Self::default();
build.manifest_path = manifest_path;
build
}
pub fn execute(&self) -> Result<()> {
let mut test = self.build_command()?;
for target in &self.cargo.target {
if target.contains("msvc") {
if env::var_os("WINEDEBUG").is_none() {
test.env("WINEDEBUG", "-all");
}
let env_target = target.to_uppercase().replace('-', "_");
let runner_env = format!("CARGO_TARGET_{}_RUNNER", env_target);
if env::var_os(&runner_env).is_none() {
test.env(runner_env, "wine");
}
}
}
let mut child = test.spawn().context("Failed to run cargo test")?;
let status = child.wait().expect("Failed to wait on cargo test process");
if !status.success() {
process::exit(status.code().unwrap_or(1));
}
Ok(())
}
pub fn build_command(&self) -> Result<Command> {
let mut build = self.cargo.command();
self.xwin.apply_command_env(
self.manifest_path.as_deref(),
&self.cargo.common,
&mut build,
)?;
Ok(build)
}
}
impl Deref for Test {
type Target = cargo_options::Test;
fn deref(&self) -> &Self::Target {
&self.cargo
}
}
impl DerefMut for Test {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.cargo
}
}
impl From<cargo_options::Test> for Test {
fn from(cargo: cargo_options::Test) -> Self {
Self {
cargo,
..Default::default()
}
}
}