@omnicross/core 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/NOTICE +57 -0
- package/README.md +15 -0
- package/dist/ApiKeyPoolService-BmMkau07.d.cts +170 -0
- package/dist/ApiKeyPoolService-BmMkau07.d.ts +170 -0
- package/dist/ProviderProxy-f_8ziIhW.d.cts +120 -0
- package/dist/ProviderProxy-vjt8sQQk.d.ts +120 -0
- package/dist/SubscriptionAuthSource-Cr4fVEYY.d.cts +264 -0
- package/dist/SubscriptionAuthSource-D89zmiSS.d.ts +264 -0
- package/dist/auth/GeminiCodeAssistProjectResolver.cjs +218 -0
- package/dist/auth/GeminiCodeAssistProjectResolver.d.cts +68 -0
- package/dist/auth/GeminiCodeAssistProjectResolver.d.ts +68 -0
- package/dist/auth/GeminiCodeAssistProjectResolver.js +189 -0
- package/dist/completion/ApiKeyPoolService.cjs +331 -0
- package/dist/completion/ApiKeyPoolService.d.cts +2 -0
- package/dist/completion/ApiKeyPoolService.d.ts +2 -0
- package/dist/completion/ApiKeyPoolService.js +306 -0
- package/dist/completion.cjs +4027 -0
- package/dist/completion.d.cts +17 -0
- package/dist/completion.d.ts +17 -0
- package/dist/completion.js +3983 -0
- package/dist/index-BTSmc9Sm.d.ts +645 -0
- package/dist/index-DXazdTzZ.d.cts +645 -0
- package/dist/index.cjs +10428 -0
- package/dist/index.d.cts +128 -0
- package/dist/index.d.ts +128 -0
- package/dist/index.js +10339 -0
- package/dist/outbound-api/subscriptionRegistryPort.cjs +38 -0
- package/dist/outbound-api/subscriptionRegistryPort.d.cts +73 -0
- package/dist/outbound-api/subscriptionRegistryPort.d.ts +73 -0
- package/dist/outbound-api/subscriptionRegistryPort.js +12 -0
- package/dist/outbound-api.cjs +5264 -0
- package/dist/outbound-api.d.cts +320 -0
- package/dist/outbound-api.d.ts +320 -0
- package/dist/outbound-api.js +5218 -0
- package/dist/pipeline/SubscriptionAuthSource.cjs +131 -0
- package/dist/pipeline/SubscriptionAuthSource.d.cts +3 -0
- package/dist/pipeline/SubscriptionAuthSource.d.ts +3 -0
- package/dist/pipeline/SubscriptionAuthSource.js +103 -0
- package/dist/pipeline/SubscriptionAuthStrategy.cjs +18 -0
- package/dist/pipeline/SubscriptionAuthStrategy.d.cts +61 -0
- package/dist/pipeline/SubscriptionAuthStrategy.d.ts +61 -0
- package/dist/pipeline/SubscriptionAuthStrategy.js +0 -0
- package/dist/ports/gemini-code-assist-resolver.cjs +38 -0
- package/dist/ports/gemini-code-assist-resolver.d.cts +26 -0
- package/dist/ports/gemini-code-assist-resolver.d.ts +26 -0
- package/dist/ports/gemini-code-assist-resolver.js +12 -0
- package/dist/ports.cjs +18 -0
- package/dist/ports.d.cts +15 -0
- package/dist/ports.d.ts +15 -0
- package/dist/ports.js +0 -0
- package/dist/provider-proxy/ingress/providerProxyShared.cjs +2958 -0
- package/dist/provider-proxy/ingress/providerProxyShared.d.cts +77 -0
- package/dist/provider-proxy/ingress/providerProxyShared.d.ts +77 -0
- package/dist/provider-proxy/ingress/providerProxyShared.js +2925 -0
- package/dist/provider-proxy/matchText.cjs +73 -0
- package/dist/provider-proxy/matchText.d.cts +47 -0
- package/dist/provider-proxy/matchText.d.ts +47 -0
- package/dist/provider-proxy/matchText.js +45 -0
- package/dist/provider-proxy/types.cjs +18 -0
- package/dist/provider-proxy/types.d.cts +12 -0
- package/dist/provider-proxy/types.d.ts +12 -0
- package/dist/provider-proxy/types.js +0 -0
- package/dist/provider-proxy.cjs +4667 -0
- package/dist/provider-proxy.d.cts +69 -0
- package/dist/provider-proxy.d.ts +69 -0
- package/dist/provider-proxy.js +4636 -0
- package/dist/serializeError.cjs +82 -0
- package/dist/serializeError.d.cts +24 -0
- package/dist/serializeError.d.ts +24 -0
- package/dist/serializeError.js +57 -0
- package/dist/sse-parser.cjs +456 -0
- package/dist/sse-parser.d.cts +143 -0
- package/dist/sse-parser.d.ts +143 -0
- package/dist/sse-parser.js +430 -0
- package/dist/transformer/TransformerChainExecutor.cjs +321 -0
- package/dist/transformer/TransformerChainExecutor.d.cts +104 -0
- package/dist/transformer/TransformerChainExecutor.d.ts +104 -0
- package/dist/transformer/TransformerChainExecutor.js +294 -0
- package/dist/transformer/TransformerService.cjs +290 -0
- package/dist/transformer/TransformerService.d.cts +138 -0
- package/dist/transformer/TransformerService.d.ts +138 -0
- package/dist/transformer/TransformerService.js +265 -0
- package/dist/transformer/transformers/GeminiCodeAssistTransformer.cjs +1115 -0
- package/dist/transformer/transformers/GeminiCodeAssistTransformer.d.cts +102 -0
- package/dist/transformer/transformers/GeminiCodeAssistTransformer.d.ts +102 -0
- package/dist/transformer/transformers/GeminiCodeAssistTransformer.js +1085 -0
- package/dist/transformer/transformers/GeminiTransformer.cjs +1013 -0
- package/dist/transformer/transformers/GeminiTransformer.d.cts +70 -0
- package/dist/transformer/transformers/GeminiTransformer.d.ts +70 -0
- package/dist/transformer/transformers/GeminiTransformer.js +986 -0
- package/dist/transformer/transformers/OpenAIResponseTransformer.cjs +538 -0
- package/dist/transformer/transformers/OpenAIResponseTransformer.d.cts +53 -0
- package/dist/transformer/transformers/OpenAIResponseTransformer.d.ts +53 -0
- package/dist/transformer/transformers/OpenAIResponseTransformer.js +513 -0
- package/dist/transformer/transformers/OpenCodeGoTransformer.cjs +73 -0
- package/dist/transformer/transformers/OpenCodeGoTransformer.d.cts +51 -0
- package/dist/transformer/transformers/OpenCodeGoTransformer.d.ts +51 -0
- package/dist/transformer/transformers/OpenCodeGoTransformer.js +48 -0
- package/dist/transformer/types.cjs +18 -0
- package/dist/transformer/types.d.cts +405 -0
- package/dist/transformer/types.d.ts +405 -0
- package/dist/transformer/types.js +0 -0
- package/dist/transformer.cjs +3736 -0
- package/dist/transformer.d.cts +33 -0
- package/dist/transformer.d.ts +33 -0
- package/dist/transformer.js +3712 -0
- package/dist/types-CGGrKqC_.d.cts +142 -0
- package/dist/types-CbCN2NQP.d.ts +142 -0
- package/dist/types-DCzHkhJt.d.ts +467 -0
- package/dist/types-DZIQbgp0.d.cts +467 -0
- package/dist/usage-event-sink-BX7FE1NL.d.cts +59 -0
- package/dist/usage-event-sink-BX7FE1NL.d.ts +59 -0
- package/package.json +62 -0
|
@@ -0,0 +1,430 @@
|
|
|
1
|
+
// src/sse-parser.ts
|
|
2
|
+
var DEBUG_SSE = true;
|
|
3
|
+
function debugSSE(prefix, ...args) {
|
|
4
|
+
if (DEBUG_SSE) {
|
|
5
|
+
console.log(`[SSE-Parser] ${prefix}`, ...args);
|
|
6
|
+
}
|
|
7
|
+
}
|
|
8
|
+
function createSSEParser(format, callbacks) {
|
|
9
|
+
const state = {
|
|
10
|
+
buffer: "",
|
|
11
|
+
content: "",
|
|
12
|
+
reasoning: "",
|
|
13
|
+
usage: void 0,
|
|
14
|
+
isDone: false,
|
|
15
|
+
streamStartTime: void 0,
|
|
16
|
+
firstTokenTime: void 0,
|
|
17
|
+
streamEndTime: void 0,
|
|
18
|
+
blocks: [],
|
|
19
|
+
currentBlockIndex: -1,
|
|
20
|
+
currentBlockType: "",
|
|
21
|
+
inputJsonBuffer: "",
|
|
22
|
+
audios: [],
|
|
23
|
+
videos: []
|
|
24
|
+
};
|
|
25
|
+
function parseOpenAIEvent(data) {
|
|
26
|
+
if (data === "[DONE]") {
|
|
27
|
+
debugSSE("OpenAI [DONE] received, content length:", state.content.length);
|
|
28
|
+
state.isDone = true;
|
|
29
|
+
callbacks.onDone?.();
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
try {
|
|
33
|
+
const json = JSON.parse(data);
|
|
34
|
+
callbacks.onRawEvent?.({ data: json, raw: data });
|
|
35
|
+
const delta = json.choices?.[0]?.delta;
|
|
36
|
+
const finishReason = json.choices?.[0]?.finish_reason;
|
|
37
|
+
if (finishReason) {
|
|
38
|
+
debugSSE("OpenAI finish_reason:", finishReason, "content length so far:", state.content.length);
|
|
39
|
+
}
|
|
40
|
+
if (delta?.content) {
|
|
41
|
+
if (!state.firstTokenTime) {
|
|
42
|
+
state.firstTokenTime = Date.now();
|
|
43
|
+
}
|
|
44
|
+
state.content += delta.content;
|
|
45
|
+
callbacks.onDelta?.(delta.content);
|
|
46
|
+
}
|
|
47
|
+
if (delta?.reasoning_content) {
|
|
48
|
+
state.reasoning += delta.reasoning_content;
|
|
49
|
+
callbacks.onReasoning?.(delta.reasoning_content);
|
|
50
|
+
}
|
|
51
|
+
if (delta?.audio) {
|
|
52
|
+
const audioData = delta.audio;
|
|
53
|
+
if (audioData.data) {
|
|
54
|
+
const mimeType = audioData.format ? `audio/${audioData.format}` : "audio/wav";
|
|
55
|
+
const audio = {
|
|
56
|
+
url: audioData.data.startsWith("data:") ? audioData.data : `data:${mimeType};base64,${audioData.data}`,
|
|
57
|
+
mimeType
|
|
58
|
+
};
|
|
59
|
+
state.audios.push(audio);
|
|
60
|
+
callbacks.onAudio?.(audio);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
if (json.usage) {
|
|
64
|
+
debugSSE("OpenAI usage received:", JSON.stringify(json.usage));
|
|
65
|
+
debugSSE("OpenAI final chunk full data:", JSON.stringify(json));
|
|
66
|
+
state.usage = {
|
|
67
|
+
promptTokens: json.usage.prompt_tokens || 0,
|
|
68
|
+
completionTokens: json.usage.completion_tokens || 0,
|
|
69
|
+
totalTokens: json.usage.total_tokens || 0
|
|
70
|
+
};
|
|
71
|
+
callbacks.onUsage?.(state.usage);
|
|
72
|
+
}
|
|
73
|
+
if (finishReason === "stop" || finishReason === "end_turn" || finishReason === "length") {
|
|
74
|
+
debugSSE("OpenAI stream marked done due to finish_reason:", finishReason);
|
|
75
|
+
}
|
|
76
|
+
} catch {
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
function parseAnthropicEvent(data) {
|
|
80
|
+
try {
|
|
81
|
+
const json = JSON.parse(data);
|
|
82
|
+
callbacks.onRawEvent?.({ type: json.type, data: json, raw: data });
|
|
83
|
+
if (json.type === "content_block_start") {
|
|
84
|
+
const contentBlock = json.content_block;
|
|
85
|
+
const blockIndex = json.index ?? -1;
|
|
86
|
+
if (contentBlock?.type === "thinking") {
|
|
87
|
+
debugSSE("THINKING BLOCK STARTED!");
|
|
88
|
+
}
|
|
89
|
+
if (contentBlock?.type === "server_tool_use") {
|
|
90
|
+
state.currentBlockIndex = blockIndex;
|
|
91
|
+
state.currentBlockType = "server_tool_use";
|
|
92
|
+
state.inputJsonBuffer = "";
|
|
93
|
+
const toolUseBlock = {
|
|
94
|
+
id: contentBlock.id || `block_${Date.now()}`,
|
|
95
|
+
type: "tool_use",
|
|
96
|
+
toolId: contentBlock.id || "",
|
|
97
|
+
toolName: contentBlock.name || "web_search",
|
|
98
|
+
input: contentBlock.input || {},
|
|
99
|
+
status: "running"
|
|
100
|
+
};
|
|
101
|
+
state.blocks.push(toolUseBlock);
|
|
102
|
+
callbacks.onBlock?.(toolUseBlock);
|
|
103
|
+
}
|
|
104
|
+
if (contentBlock?.type === "web_search_tool_result") {
|
|
105
|
+
state.currentBlockIndex = blockIndex;
|
|
106
|
+
state.currentBlockType = "web_search_tool_result";
|
|
107
|
+
const searchResults = contentBlock.content;
|
|
108
|
+
let output = "";
|
|
109
|
+
if (Array.isArray(searchResults)) {
|
|
110
|
+
output = searchResults.filter((r) => r.type === "web_search_result").map(
|
|
111
|
+
(r) => `**${r.title || "Untitled"}**
|
|
112
|
+
${r.url || ""}
|
|
113
|
+
${r.encrypted_content ? "(encrypted)" : r.page_content || r.snippet || ""}`
|
|
114
|
+
).join("\n\n");
|
|
115
|
+
}
|
|
116
|
+
const lastToolUse = [...state.blocks].reverse().find(
|
|
117
|
+
(b) => b.type === "tool_use"
|
|
118
|
+
);
|
|
119
|
+
const toolResultBlock = {
|
|
120
|
+
id: `result_${Date.now()}`,
|
|
121
|
+
type: "tool_result",
|
|
122
|
+
toolId: lastToolUse?.toolId || "",
|
|
123
|
+
toolName: lastToolUse?.toolName || "web_search",
|
|
124
|
+
output: output || "(no results)"
|
|
125
|
+
};
|
|
126
|
+
state.blocks.push(toolResultBlock);
|
|
127
|
+
callbacks.onBlock?.(toolResultBlock);
|
|
128
|
+
if (lastToolUse) {
|
|
129
|
+
lastToolUse.status = "completed";
|
|
130
|
+
callbacks.onBlock?.(lastToolUse);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
if (json.type === "content_block_delta") {
|
|
135
|
+
const delta = json.delta;
|
|
136
|
+
if (delta?.type === "text_delta" && delta?.text) {
|
|
137
|
+
if (!state.firstTokenTime) {
|
|
138
|
+
state.firstTokenTime = Date.now();
|
|
139
|
+
}
|
|
140
|
+
state.content += delta.text;
|
|
141
|
+
callbacks.onDelta?.(delta.text);
|
|
142
|
+
}
|
|
143
|
+
if (delta?.type === "thinking_delta" && delta?.thinking) {
|
|
144
|
+
state.reasoning += delta.thinking;
|
|
145
|
+
callbacks.onReasoning?.(delta.thinking);
|
|
146
|
+
}
|
|
147
|
+
if (delta?.type === "input_json_delta" && delta?.partial_json) {
|
|
148
|
+
state.inputJsonBuffer += delta.partial_json;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
if (json.type === "content_block_stop") {
|
|
152
|
+
if (state.currentBlockType === "server_tool_use" && state.inputJsonBuffer) {
|
|
153
|
+
const lastToolUse = [...state.blocks].reverse().find(
|
|
154
|
+
(b) => b.type === "tool_use"
|
|
155
|
+
);
|
|
156
|
+
if (lastToolUse) {
|
|
157
|
+
try {
|
|
158
|
+
lastToolUse.input = JSON.parse(state.inputJsonBuffer);
|
|
159
|
+
} catch {
|
|
160
|
+
lastToolUse.input = { raw: state.inputJsonBuffer };
|
|
161
|
+
}
|
|
162
|
+
callbacks.onBlock?.(lastToolUse);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
state.currentBlockIndex = -1;
|
|
166
|
+
state.currentBlockType = "";
|
|
167
|
+
state.inputJsonBuffer = "";
|
|
168
|
+
}
|
|
169
|
+
if (json.type === "message_delta" && json.usage) {
|
|
170
|
+
state.usage = {
|
|
171
|
+
promptTokens: json.usage.input_tokens || 0,
|
|
172
|
+
completionTokens: json.usage.output_tokens || 0,
|
|
173
|
+
totalTokens: (json.usage.input_tokens || 0) + (json.usage.output_tokens || 0)
|
|
174
|
+
};
|
|
175
|
+
callbacks.onUsage?.(state.usage);
|
|
176
|
+
}
|
|
177
|
+
if (json.type === "message_stop") {
|
|
178
|
+
state.isDone = true;
|
|
179
|
+
callbacks.onDone?.();
|
|
180
|
+
}
|
|
181
|
+
if (json.type === "error") {
|
|
182
|
+
callbacks.onError?.(json.error?.message || "Unknown error");
|
|
183
|
+
}
|
|
184
|
+
} catch {
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
function parseGeminiEvent(data) {
|
|
188
|
+
try {
|
|
189
|
+
const json = JSON.parse(data);
|
|
190
|
+
callbacks.onRawEvent?.({ data: json, raw: data });
|
|
191
|
+
const candidates = json.candidates;
|
|
192
|
+
if (candidates && candidates.length > 0) {
|
|
193
|
+
const candidate = candidates[0];
|
|
194
|
+
const content = candidate.content;
|
|
195
|
+
if (content?.parts) {
|
|
196
|
+
for (const part of content.parts) {
|
|
197
|
+
if (part.thought === true && part.text) {
|
|
198
|
+
state.reasoning += part.text;
|
|
199
|
+
callbacks.onReasoning?.(part.text);
|
|
200
|
+
} else if (part.text) {
|
|
201
|
+
if (!state.firstTokenTime) {
|
|
202
|
+
state.firstTokenTime = Date.now();
|
|
203
|
+
}
|
|
204
|
+
state.content += part.text;
|
|
205
|
+
callbacks.onDelta?.(part.text);
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
const finishReason = candidate.finishReason;
|
|
210
|
+
if (finishReason === "STOP" || finishReason === "MAX_TOKENS" || finishReason === "SAFETY") {
|
|
211
|
+
debugSSE("Gemini finish reason:", finishReason);
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
const usage = json.usageMetadata;
|
|
215
|
+
if (usage) {
|
|
216
|
+
state.usage = {
|
|
217
|
+
promptTokens: usage.promptTokenCount || 0,
|
|
218
|
+
completionTokens: usage.candidatesTokenCount || 0,
|
|
219
|
+
totalTokens: usage.totalTokenCount || 0
|
|
220
|
+
};
|
|
221
|
+
callbacks.onUsage?.(state.usage);
|
|
222
|
+
debugSSE("Gemini usage:", state.usage);
|
|
223
|
+
}
|
|
224
|
+
if (json.error) {
|
|
225
|
+
callbacks.onError?.(json.error.message || "Gemini API error");
|
|
226
|
+
}
|
|
227
|
+
} catch {
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
function parseOpenAIResponseEvent(data) {
|
|
231
|
+
try {
|
|
232
|
+
const json = JSON.parse(data);
|
|
233
|
+
callbacks.onRawEvent?.({ type: json.type, data: json, raw: data });
|
|
234
|
+
switch (json.type) {
|
|
235
|
+
case "response.output_text.delta":
|
|
236
|
+
if (json.delta) {
|
|
237
|
+
if (!state.firstTokenTime) {
|
|
238
|
+
state.firstTokenTime = Date.now();
|
|
239
|
+
}
|
|
240
|
+
state.content += json.delta;
|
|
241
|
+
callbacks.onDelta?.(json.delta);
|
|
242
|
+
}
|
|
243
|
+
break;
|
|
244
|
+
case "response.reasoning_summary_text.delta":
|
|
245
|
+
if (json.delta) {
|
|
246
|
+
state.reasoning += json.delta;
|
|
247
|
+
callbacks.onReasoning?.(json.delta);
|
|
248
|
+
}
|
|
249
|
+
break;
|
|
250
|
+
case "response.completed": {
|
|
251
|
+
const response = json.response;
|
|
252
|
+
if (response?.usage) {
|
|
253
|
+
state.usage = {
|
|
254
|
+
promptTokens: response.usage.input_tokens || 0,
|
|
255
|
+
completionTokens: response.usage.output_tokens || 0,
|
|
256
|
+
totalTokens: (response.usage.input_tokens || 0) + (response.usage.output_tokens || 0)
|
|
257
|
+
};
|
|
258
|
+
callbacks.onUsage?.(state.usage);
|
|
259
|
+
}
|
|
260
|
+
state.isDone = true;
|
|
261
|
+
callbacks.onDone?.();
|
|
262
|
+
break;
|
|
263
|
+
}
|
|
264
|
+
case "error":
|
|
265
|
+
callbacks.onError?.(json.error?.message || json.message || "Unknown error");
|
|
266
|
+
break;
|
|
267
|
+
// Informational events — no-op
|
|
268
|
+
// response.created, response.in_progress, response.output_item.added,
|
|
269
|
+
// response.content_part.added, response.output_text.done,
|
|
270
|
+
// response.content_part.done, response.output_item.done, etc.
|
|
271
|
+
default:
|
|
272
|
+
break;
|
|
273
|
+
}
|
|
274
|
+
} catch {
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
function processEventBlock(eventBlock) {
|
|
278
|
+
const lines = eventBlock.split("\n");
|
|
279
|
+
const dataLines = [];
|
|
280
|
+
for (const line of lines) {
|
|
281
|
+
if (line.startsWith("data: ")) {
|
|
282
|
+
dataLines.push(line.slice(6));
|
|
283
|
+
} else if (line.trim() !== "" && !line.startsWith(":")) {
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
if (dataLines.length > 0) {
|
|
287
|
+
const data = dataLines.join("\n");
|
|
288
|
+
if (format === "openai") {
|
|
289
|
+
parseOpenAIEvent(data);
|
|
290
|
+
} else if (format === "gemini") {
|
|
291
|
+
parseGeminiEvent(data);
|
|
292
|
+
} else if (format === "openai-response") {
|
|
293
|
+
parseOpenAIResponseEvent(data);
|
|
294
|
+
} else {
|
|
295
|
+
parseAnthropicEvent(data);
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
return {
|
|
300
|
+
/**
|
|
301
|
+
* Push a chunk of data to the parser
|
|
302
|
+
* @param chunk - Raw SSE chunk from response stream
|
|
303
|
+
*/
|
|
304
|
+
push(chunk) {
|
|
305
|
+
if (state.isDone) return;
|
|
306
|
+
if (!state.streamStartTime) {
|
|
307
|
+
state.streamStartTime = Date.now();
|
|
308
|
+
}
|
|
309
|
+
state.buffer += chunk;
|
|
310
|
+
const events = state.buffer.split("\n\n");
|
|
311
|
+
state.buffer = events.pop() || "";
|
|
312
|
+
for (const eventBlock of events) {
|
|
313
|
+
processEventBlock(eventBlock);
|
|
314
|
+
if (state.isDone) break;
|
|
315
|
+
}
|
|
316
|
+
},
|
|
317
|
+
/**
|
|
318
|
+
* Flush any remaining data in the buffer
|
|
319
|
+
*/
|
|
320
|
+
flush() {
|
|
321
|
+
debugSSE("flush() called, buffer length:", state.buffer.length, "isDone:", state.isDone);
|
|
322
|
+
if (state.buffer.trim()) {
|
|
323
|
+
debugSSE("flush() processing remaining buffer:", state.buffer.substring(0, 200));
|
|
324
|
+
processEventBlock(state.buffer);
|
|
325
|
+
state.buffer = "";
|
|
326
|
+
}
|
|
327
|
+
state.streamEndTime = Date.now();
|
|
328
|
+
if (!state.isDone) {
|
|
329
|
+
debugSSE("flush() marking stream as done, final content length:", state.content.length);
|
|
330
|
+
state.isDone = true;
|
|
331
|
+
callbacks.onDone?.();
|
|
332
|
+
}
|
|
333
|
+
},
|
|
334
|
+
/**
|
|
335
|
+
* Get accumulated results
|
|
336
|
+
*/
|
|
337
|
+
getResult() {
|
|
338
|
+
let metrics;
|
|
339
|
+
if (state.usage && state.firstTokenTime && state.streamEndTime) {
|
|
340
|
+
const timeCompletionMs = state.streamEndTime - state.firstTokenTime;
|
|
341
|
+
const timeFirstTokenMs = state.streamStartTime ? state.firstTokenTime - state.streamStartTime : void 0;
|
|
342
|
+
metrics = {
|
|
343
|
+
completionTokens: state.usage.completionTokens,
|
|
344
|
+
timeCompletionMs: timeCompletionMs > 0 ? timeCompletionMs : 1,
|
|
345
|
+
// Ensure at least 1ms
|
|
346
|
+
timeFirstTokenMs
|
|
347
|
+
};
|
|
348
|
+
}
|
|
349
|
+
if (state.reasoning) {
|
|
350
|
+
const thinkingBlock = {
|
|
351
|
+
id: `thinking_${Date.now()}`,
|
|
352
|
+
type: "thinking",
|
|
353
|
+
content: state.reasoning
|
|
354
|
+
};
|
|
355
|
+
state.blocks.unshift(thinkingBlock);
|
|
356
|
+
}
|
|
357
|
+
return {
|
|
358
|
+
content: state.content,
|
|
359
|
+
reasoning: state.reasoning,
|
|
360
|
+
usage: state.usage,
|
|
361
|
+
metrics,
|
|
362
|
+
blocks: state.blocks,
|
|
363
|
+
audios: state.audios,
|
|
364
|
+
videos: state.videos
|
|
365
|
+
};
|
|
366
|
+
},
|
|
367
|
+
/**
|
|
368
|
+
* Check if stream is complete
|
|
369
|
+
*/
|
|
370
|
+
isDone() {
|
|
371
|
+
return state.isDone;
|
|
372
|
+
},
|
|
373
|
+
/**
|
|
374
|
+
* Reset parser state for reuse
|
|
375
|
+
*/
|
|
376
|
+
reset() {
|
|
377
|
+
state.buffer = "";
|
|
378
|
+
state.content = "";
|
|
379
|
+
state.reasoning = "";
|
|
380
|
+
state.usage = void 0;
|
|
381
|
+
state.isDone = false;
|
|
382
|
+
state.streamStartTime = void 0;
|
|
383
|
+
state.firstTokenTime = void 0;
|
|
384
|
+
state.streamEndTime = void 0;
|
|
385
|
+
state.blocks = [];
|
|
386
|
+
state.currentBlockIndex = -1;
|
|
387
|
+
state.currentBlockType = "";
|
|
388
|
+
state.inputJsonBuffer = "";
|
|
389
|
+
state.audios = [];
|
|
390
|
+
state.videos = [];
|
|
391
|
+
}
|
|
392
|
+
};
|
|
393
|
+
}
|
|
394
|
+
async function streamSSEResponse(response, format, callbacks) {
|
|
395
|
+
debugSSE("streamSSEResponse started, format:", format);
|
|
396
|
+
const reader = response.body?.getReader();
|
|
397
|
+
if (!reader) {
|
|
398
|
+
callbacks.onError?.("No response body");
|
|
399
|
+
return { content: "", reasoning: "", blocks: [], audios: [], videos: [] };
|
|
400
|
+
}
|
|
401
|
+
const decoder = new TextDecoder();
|
|
402
|
+
const parser = createSSEParser(format, callbacks);
|
|
403
|
+
let chunkCount = 0;
|
|
404
|
+
try {
|
|
405
|
+
while (true) {
|
|
406
|
+
const { done, value } = await reader.read();
|
|
407
|
+
if (done) {
|
|
408
|
+
debugSSE("Reader done signal received after", chunkCount, "chunks");
|
|
409
|
+
break;
|
|
410
|
+
}
|
|
411
|
+
const chunk = decoder.decode(value, { stream: true });
|
|
412
|
+
chunkCount++;
|
|
413
|
+
parser.push(chunk);
|
|
414
|
+
if (parser.isDone()) {
|
|
415
|
+
debugSSE("Parser isDone after", chunkCount, "chunks");
|
|
416
|
+
break;
|
|
417
|
+
}
|
|
418
|
+
}
|
|
419
|
+
parser.flush();
|
|
420
|
+
const result = parser.getResult();
|
|
421
|
+
debugSSE("streamSSEResponse complete, content length:", result.content.length, "completionTokens:", result.usage?.completionTokens, "usage:", result.usage);
|
|
422
|
+
return result;
|
|
423
|
+
} finally {
|
|
424
|
+
reader.releaseLock();
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
export {
|
|
428
|
+
createSSEParser,
|
|
429
|
+
streamSSEResponse
|
|
430
|
+
};
|