use std::time::Duration;
use tokio::process::Command;
use tracing::debug;
use crate::Codex;
use crate::error::{Error, Result};
#[derive(Debug, Clone)]
pub struct CommandOutput {
pub stdout: String,
pub stderr: String,
pub exit_code: i32,
pub success: bool,
}
pub async fn run_codex(codex: &Codex, args: Vec<String>) -> Result<CommandOutput> {
run_codex_with_retry(codex, args, None).await
}
pub async fn run_codex_with_retry(
codex: &Codex,
args: Vec<String>,
retry_override: Option<&crate::retry::RetryPolicy>,
) -> Result<CommandOutput> {
let policy = retry_override.or(codex.retry_policy.as_ref());
match policy {
Some(policy) => {
crate::retry::with_retry(policy, || run_codex_once(codex, args.clone())).await
}
None => run_codex_once(codex, args).await,
}
}
async fn run_codex_once(codex: &Codex, args: Vec<String>) -> Result<CommandOutput> {
let mut command_args = Vec::new();
command_args.extend(codex.global_args.clone());
command_args.extend(args);
debug!(binary = %codex.binary.display(), args = ?command_args, "executing codex command");
let output = if let Some(timeout) = codex.timeout {
run_with_timeout(
&codex.binary,
&command_args,
&codex.env,
codex.working_dir.as_deref(),
timeout,
)
.await?
} else {
run_internal(
&codex.binary,
&command_args,
&codex.env,
codex.working_dir.as_deref(),
)
.await?
};
Ok(output)
}
pub async fn run_codex_allow_exit_codes(
codex: &Codex,
args: Vec<String>,
allowed_codes: &[i32],
) -> Result<CommandOutput> {
let output = run_codex(codex, args).await;
match output {
Err(Error::CommandFailed {
exit_code,
stdout,
stderr,
..
}) if allowed_codes.contains(&exit_code) => Ok(CommandOutput {
stdout,
stderr,
exit_code,
success: false,
}),
other => other,
}
}
async fn run_internal(
binary: &std::path::Path,
args: &[String],
env: &std::collections::HashMap<String, String>,
working_dir: Option<&std::path::Path>,
) -> Result<CommandOutput> {
let mut cmd = Command::new(binary);
cmd.args(args);
cmd.stdin(std::process::Stdio::null());
if let Some(dir) = working_dir {
cmd.current_dir(dir);
}
for (key, value) in env {
cmd.env(key, value);
}
let output = cmd.output().await.map_err(|e| Error::Io {
message: format!("failed to spawn codex: {e}"),
source: e,
working_dir: working_dir.map(|p| p.to_path_buf()),
})?;
let stdout = String::from_utf8_lossy(&output.stdout).to_string();
let stderr = String::from_utf8_lossy(&output.stderr).to_string();
let exit_code = output.status.code().unwrap_or(-1);
if !output.status.success() {
return Err(Error::CommandFailed {
command: format!("{} {}", binary.display(), args.join(" ")),
exit_code,
stdout,
stderr,
working_dir: working_dir.map(|p| p.to_path_buf()),
});
}
Ok(CommandOutput {
stdout,
stderr,
exit_code,
success: true,
})
}
async fn run_with_timeout(
binary: &std::path::Path,
args: &[String],
env: &std::collections::HashMap<String, String>,
working_dir: Option<&std::path::Path>,
timeout: Duration,
) -> Result<CommandOutput> {
tokio::time::timeout(timeout, run_internal(binary, args, env, working_dir))
.await
.map_err(|_| Error::Timeout {
timeout_seconds: timeout.as_secs(),
})?
}