mirror of
https://github.com/openai/codex.git
synced 2026-04-24 06:35:50 +00:00
181 lines
6.7 KiB
Rust
181 lines
6.7 KiB
Rust
#![cfg(not(target_os = "windows"))]
|
|
#![allow(clippy::expect_used, clippy::unwrap_used)]
|
|
|
|
use std::net::TcpListener;
|
|
use std::path::PathBuf;
|
|
use std::process::Stdio;
|
|
use std::time::Duration;
|
|
|
|
use core_test_support::responses;
|
|
use core_test_support::test_codex_exec::test_codex_exec;
|
|
use pretty_assertions::assert_eq;
|
|
use serde_json::json;
|
|
use tokio::process::Command;
|
|
|
|
fn extract_output_text(item: &serde_json::Value) -> String {
|
|
item.get("output")
|
|
.and_then(|value| match value {
|
|
serde_json::Value::String(text) => Some(text.clone()),
|
|
serde_json::Value::Object(obj) => obj
|
|
.get("content")
|
|
.and_then(serde_json::Value::as_str)
|
|
.map(str::to_string),
|
|
_ => None,
|
|
})
|
|
.expect("function call output should include text content")
|
|
}
|
|
|
|
#[tokio::test(flavor = "multi_thread", worker_threads = 4)]
|
|
async fn exec_cli_can_route_remote_exec_and_read_file_through_exec_server() -> anyhow::Result<()> {
|
|
let test = test_codex_exec();
|
|
let external_websocket_url = std::env::var("CODEX_EXEC_SERVER_TEST_WS_URL")
|
|
.ok()
|
|
.filter(|value| !value.trim().is_empty());
|
|
let external_remote_root = std::env::var("CODEX_EXEC_SERVER_TEST_REMOTE_ROOT")
|
|
.ok()
|
|
.filter(|value| !value.trim().is_empty())
|
|
.map(PathBuf::from);
|
|
let websocket_url = if let Some(websocket_url) = external_websocket_url {
|
|
websocket_url
|
|
} else {
|
|
let websocket_listener = TcpListener::bind("127.0.0.1:0")?;
|
|
let websocket_port = websocket_listener.local_addr()?.port();
|
|
drop(websocket_listener);
|
|
format!("ws://127.0.0.1:{websocket_port}")
|
|
};
|
|
|
|
let mut exec_server = if std::env::var("CODEX_EXEC_SERVER_TEST_WS_URL").is_ok() {
|
|
None
|
|
} else {
|
|
let child = Command::new(codex_utils_cargo_bin::cargo_bin("codex-exec-server")?)
|
|
.arg("--listen")
|
|
.arg(&websocket_url)
|
|
.stdin(Stdio::null())
|
|
.stdout(Stdio::null())
|
|
.stderr(Stdio::inherit())
|
|
.spawn()?;
|
|
tokio::time::sleep(Duration::from_millis(250)).await;
|
|
Some(child)
|
|
};
|
|
|
|
let local_workspace_root = test.cwd_path().to_path_buf();
|
|
let remote_workspace_root = external_remote_root
|
|
.clone()
|
|
.unwrap_or_else(|| local_workspace_root.clone());
|
|
let seed_path = local_workspace_root.join("remote_exec_seed.txt");
|
|
if external_remote_root.is_none() {
|
|
std::fs::write(&seed_path, "remote-fs-seed\n")?;
|
|
}
|
|
|
|
let server = responses::start_mock_server().await;
|
|
let response_mock = responses::mount_sse_sequence(
|
|
&server,
|
|
vec![
|
|
responses::sse(vec![
|
|
responses::ev_response_created("resp-1"),
|
|
responses::ev_function_call(
|
|
"call-exec",
|
|
"exec_command",
|
|
&serde_json::to_string(&json!({
|
|
"cmd": if external_remote_root.is_some() {
|
|
"printf remote-fs-seed > remote_exec_seed.txt && printf from-remote > remote_exec_generated.txt"
|
|
} else {
|
|
"printf from-remote > remote_exec_generated.txt"
|
|
},
|
|
"yield_time_ms": 500,
|
|
}))?,
|
|
),
|
|
responses::ev_completed("resp-1"),
|
|
]),
|
|
responses::sse(vec![
|
|
responses::ev_response_created("resp-2"),
|
|
responses::ev_function_call(
|
|
"call-read",
|
|
"read_file",
|
|
&serde_json::to_string(&json!({
|
|
"file_path": seed_path,
|
|
}))?,
|
|
),
|
|
responses::ev_completed("resp-2"),
|
|
]),
|
|
responses::sse(vec![
|
|
responses::ev_response_created("resp-3"),
|
|
responses::ev_function_call(
|
|
"call-list",
|
|
"list_dir",
|
|
&serde_json::to_string(&json!({
|
|
"dir_path": local_workspace_root,
|
|
"offset": 1,
|
|
"limit": 20,
|
|
"depth": 1,
|
|
}))?,
|
|
),
|
|
responses::ev_completed("resp-3"),
|
|
]),
|
|
responses::sse(vec![
|
|
responses::ev_response_created("resp-4"),
|
|
responses::ev_assistant_message("msg-1", "done"),
|
|
responses::ev_completed("resp-4"),
|
|
]),
|
|
],
|
|
)
|
|
.await;
|
|
|
|
test.cmd_with_server(&server)
|
|
.arg("--skip-git-repo-check")
|
|
.arg("-s")
|
|
.arg("danger-full-access")
|
|
.arg("-c")
|
|
.arg("experimental_use_unified_exec_tool=true")
|
|
.arg("-c")
|
|
.arg("zsh_path=\"/usr/bin/zsh\"")
|
|
.arg("-c")
|
|
.arg("experimental_unified_exec_use_exec_server=true")
|
|
.arg("-c")
|
|
.arg(format!(
|
|
"experimental_unified_exec_exec_server_websocket_url={}",
|
|
serde_json::to_string(&websocket_url)?
|
|
))
|
|
.arg("-c")
|
|
.arg(format!(
|
|
"experimental_unified_exec_exec_server_workspace_root={}",
|
|
serde_json::to_string(&remote_workspace_root)?
|
|
))
|
|
.arg("-c")
|
|
.arg("experimental_supported_tools=[\"read_file\",\"list_dir\"]")
|
|
.arg("run remote exec-server tools")
|
|
.assert()
|
|
.success();
|
|
|
|
if external_remote_root.is_none() {
|
|
let generated_path = test.cwd_path().join("remote_exec_generated.txt");
|
|
let deadline = tokio::time::Instant::now() + Duration::from_secs(5);
|
|
while tokio::time::Instant::now() < deadline && !generated_path.exists() {
|
|
tokio::time::sleep(Duration::from_millis(50)).await;
|
|
}
|
|
assert_eq!(std::fs::read_to_string(&generated_path)?, "from-remote");
|
|
}
|
|
|
|
let requests = response_mock.requests();
|
|
let read_output = extract_output_text(&requests[2].function_call_output("call-read"));
|
|
assert!(
|
|
read_output.contains("remote-fs-seed"),
|
|
"expected read_file tool output to include remote file contents, got {read_output:?}"
|
|
);
|
|
let list_output = extract_output_text(&requests[3].function_call_output("call-list"));
|
|
assert!(
|
|
list_output.contains("remote_exec_seed.txt"),
|
|
"expected list_dir output to include remote_exec_seed.txt, got {list_output:?}"
|
|
);
|
|
assert!(
|
|
list_output.contains("remote_exec_generated.txt"),
|
|
"expected list_dir output to include remote_exec_generated.txt, got {list_output:?}"
|
|
);
|
|
|
|
if let Some(exec_server) = exec_server.as_mut() {
|
|
exec_server.start_kill()?;
|
|
let _ = exec_server.wait().await;
|
|
}
|
|
Ok(())
|
|
}
|