use std::time::Duration; use anyhow::Result; use async_trait::async_trait; use bytes::Bytes; use codex_api::AuthProvider; use codex_api::Compression; use codex_api::Provider; use codex_api::ResponseEvent; use codex_api::ResponsesClient; use codex_client::HttpTransport; use codex_client::Request; use codex_client::Response; use codex_client::StreamResponse; use codex_client::TransportError; use codex_protocol::models::ResponseItem; use futures::StreamExt; use http::HeaderMap; use http::StatusCode; use pretty_assertions::assert_eq; use serde_json::Value; #[derive(Clone)] struct FixtureSseTransport { body: String, } impl FixtureSseTransport { fn new(body: String) -> Self { Self { body } } } #[async_trait] impl HttpTransport for FixtureSseTransport { async fn execute(&self, _req: Request) -> Result { Err(TransportError::Build("execute should not run".to_string())) } async fn stream(&self, _req: Request) -> Result { let stream = futures::stream::iter(vec![Ok::(Bytes::from( self.body.clone(), ))]); Ok(StreamResponse { status: StatusCode::OK, headers: HeaderMap::new(), bytes: Box::pin(stream), }) } } #[derive(Clone, Default)] struct NoAuth; impl AuthProvider for NoAuth { fn add_auth_headers(&self, _headers: &mut HeaderMap) {} } fn provider(name: &str) -> Provider { Provider { name: name.to_string(), base_url: "https://example.com/v1".to_string(), query_params: None, headers: HeaderMap::new(), retry: codex_api::RetryConfig { max_attempts: 1, base_delay: Duration::from_millis(1), retry_429: false, retry_5xx: false, retry_transport: true, }, stream_idle_timeout: Duration::from_millis(50), } } fn build_responses_body(events: Vec) -> String { let mut body = String::new(); for e in events { let kind = e .get("type") .and_then(|v| v.as_str()) .unwrap_or_else(|| panic!("fixture event missing type in SSE fixture: {e}")); if e.as_object().map(|o| o.len() == 1).unwrap_or(false) { body.push_str(&format!("event: {kind}\n\n")); } else { body.push_str(&format!("event: {kind}\ndata: {e}\n\n")); } } body } #[tokio::test] async fn responses_stream_parses_items_and_completed_end_to_end() -> Result<()> { let item1 = serde_json::json!({ "type": "response.output_item.done", "item": { "type": "message", "role": "assistant", "content": [{"type": "output_text", "text": "Hello"}] } }); let item2 = serde_json::json!({ "type": "response.output_item.done", "item": { "type": "message", "role": "assistant", "content": [{"type": "output_text", "text": "World"}] } }); let completed = serde_json::json!({ "type": "response.completed", "response": { "id": "resp1" } }); let body = build_responses_body(vec![item1, item2, completed]); let transport = FixtureSseTransport::new(body); let client = ResponsesClient::new(transport, provider("openai"), NoAuth); let mut stream = client .stream( serde_json::json!({"echo": true}), HeaderMap::new(), Compression::None, /*turn_state*/ None, ) .await?; let mut events = Vec::new(); while let Some(ev) = stream.next().await { events.push(ev?); } let events: Vec = events .into_iter() .filter(|ev| !matches!(ev, ResponseEvent::RateLimits(_))) .collect(); assert_eq!(events.len(), 3); match &events[0] { ResponseEvent::OutputItemDone(ResponseItem::Message { role, .. }) => { assert_eq!(role, "assistant"); } other => panic!("unexpected first event: {other:?}"), } match &events[1] { ResponseEvent::OutputItemDone(ResponseItem::Message { role, .. }) => { assert_eq!(role, "assistant"); } other => panic!("unexpected second event: {other:?}"), } match &events[2] { ResponseEvent::Completed { response_id, token_usage, } => { assert_eq!(response_id, "resp1"); assert!(token_usage.is_none()); } other => panic!("unexpected third event: {other:?}"), } Ok(()) }