mirror of
https://github.com/decolua/9router.git
synced 2026-05-08 12:01:28 +00:00
- Added detailed request logging and latency tracking in handleChatCore. - Improved error handling for SSE to JSON conversion and JSON parsing in streamToJsonConverter. - Introduced a safe JSON parsing utility to handle potential parsing errors gracefully in requestDetailsDb. Co-authored-by: zx <me@char.moe>
104 lines
3.2 KiB
JavaScript
104 lines
3.2 KiB
JavaScript
/**
|
|
* Stream-to-JSON Converter
|
|
* Converts Responses API SSE stream to single JSON response
|
|
* Used when client requests non-streaming but provider forces streaming (e.g., Codex)
|
|
*/
|
|
|
|
/**
|
|
* Process a single SSE message and update state accordingly.
|
|
*/
|
|
function processSSEMessage(msg, state) {
|
|
if (!msg.trim()) return;
|
|
|
|
const eventMatch = msg.match(/^event:\s*(.+)$/m);
|
|
const dataMatch = msg.match(/^data:\s*(.+)$/m);
|
|
if (!eventMatch || !dataMatch) return;
|
|
|
|
const eventType = eventMatch[1].trim();
|
|
const dataStr = dataMatch[1].trim();
|
|
if (dataStr === "[DONE]") return;
|
|
|
|
let parsed;
|
|
try { parsed = JSON.parse(dataStr); }
|
|
catch { return; }
|
|
|
|
if (eventType === "response.created") {
|
|
state.responseId = parsed.response?.id || state.responseId;
|
|
state.created = parsed.response?.created_at || state.created;
|
|
} else if (eventType === "response.output_item.done") {
|
|
state.items.set(parsed.output_index ?? 0, parsed.item);
|
|
} else if (eventType === "response.completed") {
|
|
state.status = "completed";
|
|
if (parsed.response?.usage) {
|
|
state.usage.input_tokens = parsed.response.usage.input_tokens || 0;
|
|
state.usage.output_tokens = parsed.response.usage.output_tokens || 0;
|
|
state.usage.total_tokens = parsed.response.usage.total_tokens || 0;
|
|
}
|
|
} else if (eventType === "response.failed") {
|
|
state.status = "failed";
|
|
}
|
|
}
|
|
|
|
const EMPTY_RESPONSE = { input_tokens: 0, output_tokens: 0, total_tokens: 0 };
|
|
|
|
/**
|
|
* Convert Responses API SSE stream to single JSON response
|
|
* @param {ReadableStream} stream - SSE stream from provider
|
|
* @returns {Promise<Object>} Final JSON response in Responses API format
|
|
*/
|
|
export async function convertResponsesStreamToJson(stream) {
|
|
if (!stream || typeof stream.getReader !== "function") {
|
|
return { id: `resp_${Date.now()}`, object: "response", created_at: Math.floor(Date.now() / 1000), status: "failed", output: [], usage: { ...EMPTY_RESPONSE } };
|
|
}
|
|
|
|
const reader = stream.getReader();
|
|
const decoder = new TextDecoder();
|
|
let buffer = "";
|
|
|
|
const state = {
|
|
responseId: "",
|
|
created: Math.floor(Date.now() / 1000),
|
|
status: "in_progress",
|
|
usage: { ...EMPTY_RESPONSE },
|
|
items: new Map()
|
|
};
|
|
|
|
try {
|
|
while (true) {
|
|
const { done, value } = await reader.read();
|
|
if (done) break;
|
|
|
|
buffer += decoder.decode(value, { stream: true });
|
|
const messages = buffer.split("\n\n");
|
|
buffer = messages.pop() || "";
|
|
|
|
for (const msg of messages) {
|
|
processSSEMessage(msg, state);
|
|
}
|
|
}
|
|
|
|
// Flush remaining buffer (last event may not end with \n\n)
|
|
if (buffer.trim()) {
|
|
processSSEMessage(buffer, state);
|
|
}
|
|
} finally {
|
|
reader.releaseLock();
|
|
}
|
|
|
|
// Build output array from accumulated items (ordered by index)
|
|
const output = [];
|
|
const maxIndex = state.items.size > 0 ? Math.max(...state.items.keys()) : -1;
|
|
for (let i = 0; i <= maxIndex; i++) {
|
|
output.push(state.items.get(i) || { type: "message", content: [], role: "assistant" });
|
|
}
|
|
|
|
return {
|
|
id: state.responseId || `resp_${Date.now()}_${Math.random().toString(36).slice(2, 8)}`,
|
|
object: "response",
|
|
created_at: state.created,
|
|
status: state.status || "completed",
|
|
output,
|
|
usage: state.usage
|
|
};
|
|
}
|