mirror of
https://github.com/openai/codex.git
synced 2026-05-02 02:17:22 +00:00
257 lines
7.9 KiB
Rust
257 lines
7.9 KiB
Rust
use anyhow::Result;
|
|
use app_test_support::McpProcess;
|
|
use app_test_support::to_response;
|
|
use codex_app_server_protocol::JSONRPCResponse;
|
|
use codex_app_server_protocol::RequestId;
|
|
use codex_app_server_protocol::ThreadListParams;
|
|
use codex_app_server_protocol::ThreadListResponse;
|
|
use serde_json::json;
|
|
use tempfile::TempDir;
|
|
use tokio::time::timeout;
|
|
use uuid::Uuid;
|
|
|
|
const DEFAULT_READ_TIMEOUT: std::time::Duration = std::time::Duration::from_secs(10);
|
|
|
|
#[tokio::test(flavor = "multi_thread", worker_threads = 2)]
|
|
async fn thread_list_basic_empty() -> Result<()> {
|
|
let codex_home = TempDir::new()?;
|
|
create_minimal_config(codex_home.path())?;
|
|
|
|
let mut mcp = McpProcess::new(codex_home.path()).await?;
|
|
timeout(DEFAULT_READ_TIMEOUT, mcp.initialize()).await??;
|
|
|
|
// List threads in an empty CODEX_HOME; should return an empty page with nextCursor: null.
|
|
let list_id = mcp
|
|
.send_thread_list_request(ThreadListParams {
|
|
cursor: None,
|
|
limit: Some(10),
|
|
model_providers: None,
|
|
})
|
|
.await?;
|
|
let list_resp: JSONRPCResponse = timeout(
|
|
DEFAULT_READ_TIMEOUT,
|
|
mcp.read_stream_until_response_message(RequestId::Integer(list_id)),
|
|
)
|
|
.await??;
|
|
let ThreadListResponse { data, next_cursor } = to_response::<ThreadListResponse>(list_resp)?;
|
|
assert!(data.is_empty());
|
|
assert!(next_cursor.is_none());
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Minimal config.toml for listing.
|
|
fn create_minimal_config(codex_home: &std::path::Path) -> std::io::Result<()> {
|
|
let config_toml = codex_home.join("config.toml");
|
|
std::fs::write(
|
|
config_toml,
|
|
"model = \"mock-model\"\napproval_policy = \"never\"\n",
|
|
)
|
|
}
|
|
|
|
fn create_fake_rollout(
|
|
codex_home: &std::path::Path,
|
|
filename_ts: &str,
|
|
meta_rfc3339: &str,
|
|
) -> Result<String> {
|
|
let uuid = Uuid::new_v4();
|
|
let year = &filename_ts[0..4];
|
|
let month = &filename_ts[5..7];
|
|
let day = &filename_ts[8..10];
|
|
let dir = codex_home.join("sessions").join(year).join(month).join(day);
|
|
std::fs::create_dir_all(&dir)?;
|
|
|
|
let file_path = dir.join(format!("rollout-{filename_ts}-{uuid}.jsonl"));
|
|
let mut lines = Vec::new();
|
|
lines.push(
|
|
json!({
|
|
"timestamp": meta_rfc3339,
|
|
"type": "session_meta",
|
|
"payload": {
|
|
"id": uuid,
|
|
"timestamp": meta_rfc3339,
|
|
"cwd": "/",
|
|
"originator": "codex",
|
|
"cli_version": "0.0.0",
|
|
"instructions": null,
|
|
"source": "vscode",
|
|
"model_provider": "mock_provider"
|
|
}
|
|
})
|
|
.to_string(),
|
|
);
|
|
lines.push(
|
|
json!({
|
|
"timestamp": meta_rfc3339,
|
|
"type":"response_item",
|
|
"payload": {
|
|
"type":"message",
|
|
"role":"user",
|
|
"content":[{"type":"input_text","text": "Hello"}]
|
|
}
|
|
})
|
|
.to_string(),
|
|
);
|
|
// Add a matching user_message event so the scanner includes this rollout.
|
|
lines.push(
|
|
json!({
|
|
"timestamp": meta_rfc3339,
|
|
"type":"event_msg",
|
|
"payload": {
|
|
"type":"user_message",
|
|
"message":"Hello",
|
|
"kind":"plain"
|
|
}
|
|
})
|
|
.to_string(),
|
|
);
|
|
std::fs::write(file_path, lines.join("\n") + "\n")?;
|
|
Ok(uuid.to_string())
|
|
}
|
|
|
|
#[tokio::test(flavor = "multi_thread", worker_threads = 2)]
|
|
async fn thread_list_pagination_next_cursor_none_on_last_page() -> Result<()> {
|
|
let codex_home = TempDir::new()?;
|
|
create_minimal_config(codex_home.path())?;
|
|
|
|
// Create three rollouts so we can paginate with limit=2.
|
|
let _a = create_fake_rollout(
|
|
codex_home.path(),
|
|
"2025-01-02T12-00-00",
|
|
"2025-01-02T12:00:00Z",
|
|
)?;
|
|
let _b = create_fake_rollout(
|
|
codex_home.path(),
|
|
"2025-01-01T13-00-00",
|
|
"2025-01-01T13:00:00Z",
|
|
)?;
|
|
let _c = create_fake_rollout(
|
|
codex_home.path(),
|
|
"2025-01-01T12-00-00",
|
|
"2025-01-01T12:00:00Z",
|
|
)?;
|
|
|
|
let mut mcp = McpProcess::new(codex_home.path()).await?;
|
|
timeout(DEFAULT_READ_TIMEOUT, mcp.initialize()).await??;
|
|
|
|
// Page 1: limit 2 → expect next_cursor Some.
|
|
let page1_id = mcp
|
|
.send_thread_list_request(ThreadListParams {
|
|
cursor: None,
|
|
limit: Some(2),
|
|
model_providers: None,
|
|
})
|
|
.await?;
|
|
let page1_resp: JSONRPCResponse = timeout(
|
|
DEFAULT_READ_TIMEOUT,
|
|
mcp.read_stream_until_response_message(RequestId::Integer(page1_id)),
|
|
)
|
|
.await??;
|
|
let ThreadListResponse {
|
|
data: data1,
|
|
next_cursor: cursor1,
|
|
} = to_response::<ThreadListResponse>(page1_resp)?;
|
|
assert_eq!(data1.len(), 2);
|
|
let cursor1 = cursor1.expect("expected nextCursor on first page");
|
|
|
|
// Page 2: with cursor → expect next_cursor None when no more results.
|
|
let page2_id = mcp
|
|
.send_thread_list_request(ThreadListParams {
|
|
cursor: Some(cursor1),
|
|
limit: Some(2),
|
|
model_providers: None,
|
|
})
|
|
.await?;
|
|
let page2_resp: JSONRPCResponse = timeout(
|
|
DEFAULT_READ_TIMEOUT,
|
|
mcp.read_stream_until_response_message(RequestId::Integer(page2_id)),
|
|
)
|
|
.await??;
|
|
let ThreadListResponse {
|
|
data: data2,
|
|
next_cursor: cursor2,
|
|
} = to_response::<ThreadListResponse>(page2_resp)?;
|
|
assert!(data2.len() <= 2);
|
|
assert!(
|
|
cursor2.is_none(),
|
|
"expected nextCursor to be null on last page"
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[tokio::test(flavor = "multi_thread", worker_threads = 2)]
|
|
async fn thread_list_respects_provider_filter() -> Result<()> {
|
|
let codex_home = TempDir::new()?;
|
|
create_minimal_config(codex_home.path())?;
|
|
|
|
// Create rollouts under two providers.
|
|
let _a = create_fake_rollout(
|
|
codex_home.path(),
|
|
"2025-01-02T10-00-00",
|
|
"2025-01-02T10:00:00Z",
|
|
)?; // mock_provider
|
|
// one with a different provider
|
|
let uuid = Uuid::new_v4();
|
|
let dir = codex_home
|
|
.path()
|
|
.join("sessions")
|
|
.join("2025")
|
|
.join("01")
|
|
.join("02");
|
|
std::fs::create_dir_all(&dir)?;
|
|
let file_path = dir.join(format!("rollout-2025-01-02T11-00-00-{uuid}.jsonl"));
|
|
let lines = [
|
|
json!({
|
|
"timestamp": "2025-01-02T11:00:00Z",
|
|
"type": "session_meta",
|
|
"payload": {
|
|
"id": uuid,
|
|
"timestamp": "2025-01-02T11:00:00Z",
|
|
"cwd": "/",
|
|
"originator": "codex",
|
|
"cli_version": "0.0.0",
|
|
"instructions": null,
|
|
"source": "vscode",
|
|
"model_provider": "other_provider"
|
|
}
|
|
})
|
|
.to_string(),
|
|
json!({
|
|
"timestamp": "2025-01-02T11:00:00Z",
|
|
"type":"response_item",
|
|
"payload": {"type":"message","role":"user","content":[{"type":"input_text","text":"X"}]}
|
|
})
|
|
.to_string(),
|
|
json!({
|
|
"timestamp": "2025-01-02T11:00:00Z",
|
|
"type":"event_msg",
|
|
"payload": {"type":"user_message","message":"X","kind":"plain"}
|
|
})
|
|
.to_string(),
|
|
];
|
|
std::fs::write(file_path, lines.join("\n") + "\n")?;
|
|
|
|
let mut mcp = McpProcess::new(codex_home.path()).await?;
|
|
timeout(DEFAULT_READ_TIMEOUT, mcp.initialize()).await??;
|
|
|
|
// Filter to only other_provider; expect 1 item, nextCursor None.
|
|
let list_id = mcp
|
|
.send_thread_list_request(ThreadListParams {
|
|
cursor: None,
|
|
limit: Some(10),
|
|
model_providers: Some(vec!["other_provider".to_string()]),
|
|
})
|
|
.await?;
|
|
let resp: JSONRPCResponse = timeout(
|
|
DEFAULT_READ_TIMEOUT,
|
|
mcp.read_stream_until_response_message(RequestId::Integer(list_id)),
|
|
)
|
|
.await??;
|
|
let ThreadListResponse { data, next_cursor } = to_response::<ThreadListResponse>(resp)?;
|
|
assert_eq!(data.len(), 1);
|
|
assert!(next_cursor.is_none());
|
|
|
|
Ok(())
|
|
}
|