mirror of
https://github.com/openai/codex.git
synced 2026-04-29 02:41:12 +03:00
Previously, clients would call `thread/start` with dynamic_tools set, and when a model invokes a dynamic tool, it would just make the server->client `item/tool/call` request and wait for the client's response to complete the tool call. This works, but it doesn't have an `item/started` or `item/completed` event. Now we are doing this: - [new] emit `item/started` with `DynamicToolCall` populated with the call arguments - send an `item/tool/call` server request - [new] once the client responds, emit `item/completed` with `DynamicToolCall` populated with the response. Also, with `persistExtendedHistory: true`, dynamic tool calls are now reconstructable in `thread/read` and `thread/resume` as `ThreadItem::DynamicToolCall`.
74 lines
2.3 KiB
Rust
74 lines
2.3 KiB
Rust
use codex_app_server_protocol::DynamicToolCallOutputContentItem;
|
|
use codex_app_server_protocol::DynamicToolCallResponse;
|
|
use codex_core::CodexThread;
|
|
use codex_protocol::dynamic_tools::DynamicToolCallOutputContentItem as CoreDynamicToolCallOutputContentItem;
|
|
use codex_protocol::dynamic_tools::DynamicToolResponse as CoreDynamicToolResponse;
|
|
use codex_protocol::protocol::Op;
|
|
use std::sync::Arc;
|
|
use tokio::sync::oneshot;
|
|
use tracing::error;
|
|
|
|
use crate::outgoing_message::ClientRequestResult;
|
|
|
|
pub(crate) async fn on_call_response(
|
|
call_id: String,
|
|
receiver: oneshot::Receiver<ClientRequestResult>,
|
|
conversation: Arc<CodexThread>,
|
|
) {
|
|
let response = receiver.await;
|
|
let (response, _error) = match response {
|
|
Ok(Ok(value)) => decode_response(value),
|
|
Ok(Err(err)) => {
|
|
error!("request failed with client error: {err:?}");
|
|
fallback_response("dynamic tool request failed")
|
|
}
|
|
Err(err) => {
|
|
error!("request failed: {err:?}");
|
|
fallback_response("dynamic tool request failed")
|
|
}
|
|
};
|
|
|
|
let DynamicToolCallResponse {
|
|
content_items,
|
|
success,
|
|
} = response.clone();
|
|
let core_response = CoreDynamicToolResponse {
|
|
content_items: content_items
|
|
.into_iter()
|
|
.map(CoreDynamicToolCallOutputContentItem::from)
|
|
.collect(),
|
|
success,
|
|
};
|
|
if let Err(err) = conversation
|
|
.submit(Op::DynamicToolResponse {
|
|
id: call_id.clone(),
|
|
response: core_response,
|
|
})
|
|
.await
|
|
{
|
|
error!("failed to submit DynamicToolResponse: {err}");
|
|
}
|
|
}
|
|
|
|
fn decode_response(value: serde_json::Value) -> (DynamicToolCallResponse, Option<String>) {
|
|
match serde_json::from_value::<DynamicToolCallResponse>(value) {
|
|
Ok(response) => (response, None),
|
|
Err(err) => {
|
|
error!("failed to deserialize DynamicToolCallResponse: {err}");
|
|
fallback_response("dynamic tool response was invalid")
|
|
}
|
|
}
|
|
}
|
|
|
|
fn fallback_response(message: &str) -> (DynamicToolCallResponse, Option<String>) {
|
|
(
|
|
DynamicToolCallResponse {
|
|
content_items: vec![DynamicToolCallOutputContentItem::InputText {
|
|
text: message.to_string(),
|
|
}],
|
|
success: false,
|
|
},
|
|
Some(message.to_string()),
|
|
)
|
|
}
|