mirror of
https://github.com/openai/codex.git
synced 2026-04-29 10:53:24 +03:00
codex-api: share websocket pump and simplify realtime events
This commit is contained in:
@@ -19,15 +19,8 @@ pub struct RealtimeAudioFrame {
|
||||
pub samples_per_channel: Option<u32>,
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||
pub enum RealtimeConnectionState {
|
||||
Connected,
|
||||
Disconnected,
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||
pub enum RealtimeEvent {
|
||||
State(RealtimeConnectionState),
|
||||
SessionCreated { session_id: String },
|
||||
SessionUpdated { backend_prompt: Option<String> },
|
||||
AudioOut(RealtimeAudioFrame),
|
||||
@@ -86,84 +79,68 @@ pub(super) struct ConversationItemContent {
|
||||
pub(super) text: String,
|
||||
}
|
||||
|
||||
#[derive(Debug, Deserialize)]
|
||||
#[serde(tag = "type")]
|
||||
enum RealtimeInboundMessage {
|
||||
#[serde(rename = "session.created")]
|
||||
SessionCreated {
|
||||
session_id: Option<String>,
|
||||
session: Option<RealtimeInboundSession>,
|
||||
},
|
||||
#[serde(rename = "session.updated")]
|
||||
SessionUpdated {
|
||||
session: Option<RealtimeInboundSession>,
|
||||
},
|
||||
#[serde(rename = "response.output_audio.delta")]
|
||||
OutputAudioDelta {
|
||||
delta: Option<String>,
|
||||
data: Option<String>,
|
||||
sample_rate: Option<u32>,
|
||||
num_channels: Option<u16>,
|
||||
samples_per_channel: Option<u32>,
|
||||
},
|
||||
#[serde(rename = "conversation.item.added")]
|
||||
ConversationItemAdded { item: Option<Value> },
|
||||
#[serde(rename = "error")]
|
||||
Error {
|
||||
error: Option<Value>,
|
||||
message: Option<String>,
|
||||
},
|
||||
}
|
||||
|
||||
#[derive(Debug, Deserialize)]
|
||||
struct RealtimeInboundSession {
|
||||
id: Option<String>,
|
||||
backend_prompt: Option<String>,
|
||||
}
|
||||
|
||||
pub(super) fn parse_realtime_event(payload: &str) -> Option<RealtimeEvent> {
|
||||
let parsed: RealtimeInboundMessage = match serde_json::from_str(payload) {
|
||||
Ok(msg) => msg,
|
||||
let parsed: Value = match serde_json::from_str(payload) {
|
||||
Ok(value) => value,
|
||||
Err(err) => {
|
||||
debug!("failed to parse realtime event: {err}, data: {payload}");
|
||||
return None;
|
||||
}
|
||||
};
|
||||
|
||||
match parsed {
|
||||
RealtimeInboundMessage::SessionCreated {
|
||||
session_id,
|
||||
session,
|
||||
} => {
|
||||
let session_id = session.and_then(|s| s.id).or(session_id);
|
||||
session_id.map(|id| RealtimeEvent::SessionCreated { session_id: id })
|
||||
let event_type = parsed.get("type")?.as_str()?;
|
||||
match event_type {
|
||||
"session.created" => {
|
||||
let session_id = parsed
|
||||
.pointer("/session/id")
|
||||
.and_then(Value::as_str)
|
||||
.or_else(|| parsed.get("session_id").and_then(Value::as_str))?;
|
||||
Some(RealtimeEvent::SessionCreated {
|
||||
session_id: session_id.to_string(),
|
||||
})
|
||||
}
|
||||
RealtimeInboundMessage::SessionUpdated { session } => Some(RealtimeEvent::SessionUpdated {
|
||||
backend_prompt: session.and_then(|s| s.backend_prompt),
|
||||
"session.updated" => Some(RealtimeEvent::SessionUpdated {
|
||||
backend_prompt: parsed
|
||||
.pointer("/session/backend_prompt")
|
||||
.and_then(Value::as_str)
|
||||
.map(ToString::to_string),
|
||||
}),
|
||||
RealtimeInboundMessage::OutputAudioDelta {
|
||||
delta,
|
||||
data,
|
||||
sample_rate,
|
||||
num_channels,
|
||||
samples_per_channel,
|
||||
} => {
|
||||
let data = delta.or(data)?;
|
||||
let sample_rate = sample_rate?;
|
||||
let num_channels = num_channels?;
|
||||
"response.output_audio.delta" => {
|
||||
let data = parsed
|
||||
.get("delta")
|
||||
.and_then(Value::as_str)
|
||||
.or_else(|| parsed.get("data").and_then(Value::as_str))?;
|
||||
let sample_rate = parsed
|
||||
.get("sample_rate")
|
||||
.and_then(Value::as_u64)
|
||||
.and_then(|value| u32::try_from(value).ok())?;
|
||||
let num_channels = parsed
|
||||
.get("num_channels")
|
||||
.and_then(Value::as_u64)
|
||||
.and_then(|value| u16::try_from(value).ok())?;
|
||||
let samples_per_channel = parsed
|
||||
.get("samples_per_channel")
|
||||
.and_then(Value::as_u64)
|
||||
.and_then(|value| u32::try_from(value).ok());
|
||||
Some(RealtimeEvent::AudioOut(RealtimeAudioFrame {
|
||||
data,
|
||||
data: data.to_string(),
|
||||
sample_rate,
|
||||
num_channels,
|
||||
samples_per_channel,
|
||||
}))
|
||||
}
|
||||
RealtimeInboundMessage::ConversationItemAdded { item } => {
|
||||
item.map(RealtimeEvent::ConversationItemAdded)
|
||||
}
|
||||
RealtimeInboundMessage::Error { error, message } => {
|
||||
let message = message.or_else(|| error.map(|e| e.to_string()))?;
|
||||
"conversation.item.added" => parsed
|
||||
.get("item")
|
||||
.cloned()
|
||||
.map(RealtimeEvent::ConversationItemAdded),
|
||||
"error" => {
|
||||
let message = parsed
|
||||
.get("message")
|
||||
.and_then(Value::as_str)
|
||||
.map(ToString::to_string)
|
||||
.or_else(|| parsed.get("error").map(ToString::to_string))?;
|
||||
Some(RealtimeEvent::Error(message))
|
||||
}
|
||||
_ => None,
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user