@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,513 @@
|
|
|
1
|
+
// src/transformer/transformers/OpenAIResponseTransformer.ts
|
|
2
|
+
var OpenAIResponseTransformer = class {
|
|
3
|
+
static TransformerName = "openai-response";
|
|
4
|
+
name = "openai-response";
|
|
5
|
+
endPoint = "/v1/responses";
|
|
6
|
+
logger;
|
|
7
|
+
/**
|
|
8
|
+
* Handle authentication - Bearer token
|
|
9
|
+
*/
|
|
10
|
+
async auth(request, provider, _context) {
|
|
11
|
+
return {
|
|
12
|
+
body: request,
|
|
13
|
+
config: {
|
|
14
|
+
headers: {
|
|
15
|
+
Authorization: `Bearer ${provider.apiKey}`,
|
|
16
|
+
"Content-Type": "application/json"
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Transform unified request → Response API format
|
|
23
|
+
*/
|
|
24
|
+
async transformRequestIn(request, provider, _context) {
|
|
25
|
+
const input = [];
|
|
26
|
+
for (const msg of request.messages) {
|
|
27
|
+
if (msg.role === "system") {
|
|
28
|
+
input.push({
|
|
29
|
+
role: "developer",
|
|
30
|
+
content: typeof msg.content === "string" ? msg.content : flattenContent(msg.content)
|
|
31
|
+
});
|
|
32
|
+
} else if (msg.role === "tool") {
|
|
33
|
+
input.push({
|
|
34
|
+
type: "function_call_output",
|
|
35
|
+
call_id: msg.tool_call_id,
|
|
36
|
+
output: typeof msg.content === "string" ? msg.content : ""
|
|
37
|
+
});
|
|
38
|
+
} else {
|
|
39
|
+
const entry = {
|
|
40
|
+
role: msg.role,
|
|
41
|
+
content: typeof msg.content === "string" ? msg.content : flattenContent(msg.content)
|
|
42
|
+
};
|
|
43
|
+
if (msg.role === "assistant" && msg.tool_calls?.length) {
|
|
44
|
+
input.push(entry);
|
|
45
|
+
for (const tc of msg.tool_calls) {
|
|
46
|
+
input.push({
|
|
47
|
+
type: "function_call",
|
|
48
|
+
id: tc.id,
|
|
49
|
+
call_id: tc.id,
|
|
50
|
+
name: tc.function.name,
|
|
51
|
+
arguments: tc.function.arguments
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
continue;
|
|
55
|
+
}
|
|
56
|
+
input.push(entry);
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
const body = {
|
|
60
|
+
model: request.model,
|
|
61
|
+
input,
|
|
62
|
+
stream: request.stream ?? false,
|
|
63
|
+
...request.max_tokens ? { max_output_tokens: request.max_tokens } : {},
|
|
64
|
+
...request.temperature !== void 0 ? { temperature: request.temperature } : {}
|
|
65
|
+
};
|
|
66
|
+
if (request.reasoning?.effort && request.reasoning.effort !== "none") {
|
|
67
|
+
body.reasoning = { effort: request.reasoning.effort, summary: "auto" };
|
|
68
|
+
}
|
|
69
|
+
if (request.tools?.length) {
|
|
70
|
+
body.tools = request.tools.map((tool) => ({
|
|
71
|
+
type: "function",
|
|
72
|
+
name: tool.function.name,
|
|
73
|
+
description: tool.function.description,
|
|
74
|
+
parameters: tool.function.parameters
|
|
75
|
+
}));
|
|
76
|
+
}
|
|
77
|
+
if (request.tool_choice) {
|
|
78
|
+
if (typeof request.tool_choice === "string") {
|
|
79
|
+
body.tool_choice = request.tool_choice;
|
|
80
|
+
} else if (typeof request.tool_choice === "object" && "function" in request.tool_choice) {
|
|
81
|
+
body.tool_choice = { type: "function", name: request.tool_choice.function.name };
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
const url = new URL("/v1/responses", provider.baseUrl);
|
|
85
|
+
return { body, config: { url } };
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Transform Response API request → unified format
|
|
89
|
+
*/
|
|
90
|
+
async transformRequestOut(request, _context) {
|
|
91
|
+
const req = request;
|
|
92
|
+
const messages = [];
|
|
93
|
+
if (req.input) {
|
|
94
|
+
for (const item of req.input) {
|
|
95
|
+
const entry = item;
|
|
96
|
+
if (entry.type === "function_call_output") {
|
|
97
|
+
messages.push({
|
|
98
|
+
role: "tool",
|
|
99
|
+
content: entry.output || "",
|
|
100
|
+
tool_call_id: entry.call_id || void 0
|
|
101
|
+
});
|
|
102
|
+
continue;
|
|
103
|
+
}
|
|
104
|
+
if (entry.type === "function_call") {
|
|
105
|
+
const toolCall = {
|
|
106
|
+
id: (entry.call_id ?? entry.id) || "",
|
|
107
|
+
type: "function",
|
|
108
|
+
function: {
|
|
109
|
+
name: entry.name || "",
|
|
110
|
+
arguments: typeof entry.arguments === "string" ? entry.arguments : ""
|
|
111
|
+
}
|
|
112
|
+
};
|
|
113
|
+
const last = messages[messages.length - 1];
|
|
114
|
+
if (last && last.role === "assistant") {
|
|
115
|
+
(last.tool_calls ??= []).push(toolCall);
|
|
116
|
+
} else {
|
|
117
|
+
messages.push({ role: "assistant", content: null, tool_calls: [toolCall] });
|
|
118
|
+
}
|
|
119
|
+
continue;
|
|
120
|
+
}
|
|
121
|
+
const role = entry.role;
|
|
122
|
+
if (role === "developer") {
|
|
123
|
+
messages.push({
|
|
124
|
+
role: "system",
|
|
125
|
+
content: typeof entry.content === "string" ? entry.content : JSON.stringify(entry.content)
|
|
126
|
+
});
|
|
127
|
+
} else if (role === "user" || role === "assistant") {
|
|
128
|
+
messages.push({
|
|
129
|
+
role,
|
|
130
|
+
content: typeof entry.content === "string" ? entry.content : JSON.stringify(entry.content)
|
|
131
|
+
});
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
const result = {
|
|
136
|
+
messages,
|
|
137
|
+
model: req.model,
|
|
138
|
+
max_tokens: req.max_output_tokens,
|
|
139
|
+
temperature: req.temperature,
|
|
140
|
+
stream: req.stream
|
|
141
|
+
};
|
|
142
|
+
if (req.reasoning?.effort) {
|
|
143
|
+
result.reasoning = {
|
|
144
|
+
effort: req.reasoning.effort,
|
|
145
|
+
enabled: true
|
|
146
|
+
};
|
|
147
|
+
}
|
|
148
|
+
if (req.tools?.length) {
|
|
149
|
+
result.tools = req.tools.filter((t) => t.type === "function").map((t) => ({
|
|
150
|
+
type: "function",
|
|
151
|
+
function: {
|
|
152
|
+
name: t.name || "",
|
|
153
|
+
description: t.description || "",
|
|
154
|
+
parameters: t.parameters || {}
|
|
155
|
+
}
|
|
156
|
+
}));
|
|
157
|
+
}
|
|
158
|
+
return result;
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Transform Response API response → unified (OpenAI CC) format
|
|
162
|
+
*/
|
|
163
|
+
async transformResponseOut(response, _context) {
|
|
164
|
+
const contentType = response.headers.get("Content-Type") ?? "";
|
|
165
|
+
if (contentType.includes("text/event-stream")) {
|
|
166
|
+
if (!response.body) {
|
|
167
|
+
throw new Error("Stream response body is null");
|
|
168
|
+
}
|
|
169
|
+
return new Response(convertResponseApiStreamToOpenAI(response.body), {
|
|
170
|
+
headers: {
|
|
171
|
+
"Content-Type": "text/event-stream",
|
|
172
|
+
"Cache-Control": "no-cache",
|
|
173
|
+
Connection: "keep-alive"
|
|
174
|
+
}
|
|
175
|
+
});
|
|
176
|
+
}
|
|
177
|
+
const data = await response.json();
|
|
178
|
+
return new Response(JSON.stringify(convertResponseApiJsonToOpenAI(data)), {
|
|
179
|
+
headers: { "Content-Type": "application/json" }
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Transform OpenAI CC response → Response API format
|
|
184
|
+
*/
|
|
185
|
+
async transformResponseIn(response, _context) {
|
|
186
|
+
const contentType = response.headers.get("Content-Type") ?? "";
|
|
187
|
+
if (contentType.includes("text/event-stream")) {
|
|
188
|
+
if (!response.body) {
|
|
189
|
+
throw new Error("Stream response body is null");
|
|
190
|
+
}
|
|
191
|
+
return new Response(convertOpenAIStreamToResponseApi(response.body), {
|
|
192
|
+
headers: {
|
|
193
|
+
"Content-Type": "text/event-stream",
|
|
194
|
+
"Cache-Control": "no-cache",
|
|
195
|
+
Connection: "keep-alive"
|
|
196
|
+
}
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
const data = await response.json();
|
|
200
|
+
return new Response(JSON.stringify(convertOpenAIJsonToResponseApi(data)), {
|
|
201
|
+
headers: { "Content-Type": "application/json" }
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
};
|
|
205
|
+
function flattenContent(content) {
|
|
206
|
+
if (typeof content === "string") return content;
|
|
207
|
+
if (!content) return "";
|
|
208
|
+
if (Array.isArray(content)) {
|
|
209
|
+
return content.filter((c) => c.type === "text").map((c) => c.text || "").join("\n");
|
|
210
|
+
}
|
|
211
|
+
return "";
|
|
212
|
+
}
|
|
213
|
+
function convertResponseApiJsonToOpenAI(data) {
|
|
214
|
+
let textContent = "";
|
|
215
|
+
const toolCalls = [];
|
|
216
|
+
const output = data.output;
|
|
217
|
+
if (output) {
|
|
218
|
+
for (const item of output) {
|
|
219
|
+
if (item.type === "message") {
|
|
220
|
+
const content = item.content;
|
|
221
|
+
if (content) {
|
|
222
|
+
for (const part of content) {
|
|
223
|
+
if (part.type === "output_text" && typeof part.text === "string") {
|
|
224
|
+
textContent += part.text;
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
} else if (item.type === "function_call") {
|
|
229
|
+
toolCalls.push({
|
|
230
|
+
id: item.call_id || item.id || `call_${Date.now()}`,
|
|
231
|
+
type: "function",
|
|
232
|
+
function: {
|
|
233
|
+
name: item.name,
|
|
234
|
+
arguments: typeof item.arguments === "string" ? item.arguments : JSON.stringify(item.arguments || {})
|
|
235
|
+
}
|
|
236
|
+
});
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
const usage = data.usage;
|
|
241
|
+
const message = {
|
|
242
|
+
role: "assistant",
|
|
243
|
+
content: textContent || null
|
|
244
|
+
};
|
|
245
|
+
if (toolCalls.length > 0) {
|
|
246
|
+
message.tool_calls = toolCalls;
|
|
247
|
+
}
|
|
248
|
+
return {
|
|
249
|
+
id: data.id || `chatcmpl-${Date.now()}`,
|
|
250
|
+
object: "chat.completion",
|
|
251
|
+
created: Math.floor(Date.now() / 1e3),
|
|
252
|
+
model: data.model || "unknown",
|
|
253
|
+
choices: [
|
|
254
|
+
{
|
|
255
|
+
index: 0,
|
|
256
|
+
message,
|
|
257
|
+
finish_reason: toolCalls.length > 0 ? "tool_calls" : "stop"
|
|
258
|
+
}
|
|
259
|
+
],
|
|
260
|
+
usage: usage ? {
|
|
261
|
+
prompt_tokens: usage.input_tokens || 0,
|
|
262
|
+
completion_tokens: usage.output_tokens || 0,
|
|
263
|
+
total_tokens: (usage.input_tokens || 0) + (usage.output_tokens || 0)
|
|
264
|
+
} : void 0
|
|
265
|
+
};
|
|
266
|
+
}
|
|
267
|
+
function convertOpenAIJsonToResponseApi(data) {
|
|
268
|
+
const choices = data.choices;
|
|
269
|
+
const message = choices?.[0]?.message;
|
|
270
|
+
const output = [];
|
|
271
|
+
if (message) {
|
|
272
|
+
const contentParts = [];
|
|
273
|
+
if (message.content) {
|
|
274
|
+
contentParts.push({ type: "output_text", text: message.content });
|
|
275
|
+
}
|
|
276
|
+
if (contentParts.length > 0) {
|
|
277
|
+
output.push({ type: "message", role: "assistant", content: contentParts });
|
|
278
|
+
}
|
|
279
|
+
const toolCalls = message.tool_calls;
|
|
280
|
+
if (toolCalls?.length) {
|
|
281
|
+
for (const tc of toolCalls) {
|
|
282
|
+
const func = tc.function;
|
|
283
|
+
output.push({
|
|
284
|
+
type: "function_call",
|
|
285
|
+
id: tc.id,
|
|
286
|
+
call_id: tc.id,
|
|
287
|
+
name: func?.name,
|
|
288
|
+
arguments: func?.arguments
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
const usage = data.usage;
|
|
294
|
+
return {
|
|
295
|
+
id: data.id || `resp_${Date.now()}`,
|
|
296
|
+
object: "response",
|
|
297
|
+
status: "completed",
|
|
298
|
+
model: data.model || "unknown",
|
|
299
|
+
output,
|
|
300
|
+
usage: usage ? {
|
|
301
|
+
input_tokens: usage.prompt_tokens || 0,
|
|
302
|
+
output_tokens: usage.completion_tokens || 0,
|
|
303
|
+
total_tokens: usage.total_tokens || (usage.prompt_tokens || 0) + (usage.completion_tokens || 0)
|
|
304
|
+
} : void 0
|
|
305
|
+
};
|
|
306
|
+
}
|
|
307
|
+
function convertResponseApiStreamToOpenAI(responseApiStream) {
|
|
308
|
+
const decoder = new TextDecoder();
|
|
309
|
+
const encoder = new TextEncoder();
|
|
310
|
+
return new ReadableStream({
|
|
311
|
+
start: async (controller) => {
|
|
312
|
+
const reader = responseApiStream.getReader();
|
|
313
|
+
let buffer = "";
|
|
314
|
+
let isClosed = false;
|
|
315
|
+
const messageId = `chatcmpl-${Date.now()}`;
|
|
316
|
+
let model = "unknown";
|
|
317
|
+
let hasEmittedRole = false;
|
|
318
|
+
const safeEnqueue = (str) => {
|
|
319
|
+
if (!isClosed) {
|
|
320
|
+
try {
|
|
321
|
+
controller.enqueue(encoder.encode(str));
|
|
322
|
+
} catch {
|
|
323
|
+
isClosed = true;
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
};
|
|
327
|
+
const emitChunk = (choices, usage) => {
|
|
328
|
+
const chunk = {
|
|
329
|
+
id: messageId,
|
|
330
|
+
object: "chat.completion.chunk",
|
|
331
|
+
created: Math.floor(Date.now() / 1e3),
|
|
332
|
+
model,
|
|
333
|
+
choices
|
|
334
|
+
};
|
|
335
|
+
if (usage) chunk.usage = usage;
|
|
336
|
+
safeEnqueue(`data: ${JSON.stringify(chunk)}
|
|
337
|
+
|
|
338
|
+
`);
|
|
339
|
+
};
|
|
340
|
+
try {
|
|
341
|
+
while (true) {
|
|
342
|
+
const { done, value } = await reader.read();
|
|
343
|
+
if (done) break;
|
|
344
|
+
buffer += decoder.decode(value, { stream: true });
|
|
345
|
+
const lines = buffer.split("\n");
|
|
346
|
+
buffer = lines.pop() || "";
|
|
347
|
+
for (const line of lines) {
|
|
348
|
+
if (isClosed) break;
|
|
349
|
+
if (!line.startsWith("data:")) continue;
|
|
350
|
+
const data = line.slice(5).trim();
|
|
351
|
+
if (data === "[DONE]") continue;
|
|
352
|
+
try {
|
|
353
|
+
const event = JSON.parse(data);
|
|
354
|
+
model = event.model || event.response?.model || model;
|
|
355
|
+
switch (event.type) {
|
|
356
|
+
case "response.output_text.delta":
|
|
357
|
+
if (event.delta) {
|
|
358
|
+
if (!hasEmittedRole) {
|
|
359
|
+
emitChunk([{ index: 0, delta: { role: "assistant", content: "" }, finish_reason: null }]);
|
|
360
|
+
hasEmittedRole = true;
|
|
361
|
+
}
|
|
362
|
+
emitChunk([{ index: 0, delta: { content: event.delta }, finish_reason: null }]);
|
|
363
|
+
}
|
|
364
|
+
break;
|
|
365
|
+
case "response.reasoning_summary_text.delta":
|
|
366
|
+
if (event.delta) {
|
|
367
|
+
emitChunk([{
|
|
368
|
+
index: 0,
|
|
369
|
+
delta: { thinking: { content: event.delta } },
|
|
370
|
+
finish_reason: null
|
|
371
|
+
}]);
|
|
372
|
+
}
|
|
373
|
+
break;
|
|
374
|
+
case "response.completed": {
|
|
375
|
+
const resp = event.response;
|
|
376
|
+
const respUsage = resp?.usage;
|
|
377
|
+
const usage = respUsage ? {
|
|
378
|
+
prompt_tokens: respUsage.input_tokens || 0,
|
|
379
|
+
completion_tokens: respUsage.output_tokens || 0,
|
|
380
|
+
total_tokens: (respUsage.input_tokens || 0) + (respUsage.output_tokens || 0)
|
|
381
|
+
} : void 0;
|
|
382
|
+
emitChunk([{ index: 0, delta: {}, finish_reason: "stop" }], usage);
|
|
383
|
+
safeEnqueue("data: [DONE]\n\n");
|
|
384
|
+
break;
|
|
385
|
+
}
|
|
386
|
+
case "error":
|
|
387
|
+
emitChunk([{
|
|
388
|
+
index: 0,
|
|
389
|
+
delta: { content: `[Error: ${event.error?.message || "Unknown error"}]` },
|
|
390
|
+
finish_reason: "stop"
|
|
391
|
+
}]);
|
|
392
|
+
break;
|
|
393
|
+
default:
|
|
394
|
+
break;
|
|
395
|
+
}
|
|
396
|
+
} catch {
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
} catch (e) {
|
|
401
|
+
if (!isClosed) controller.error(e);
|
|
402
|
+
} finally {
|
|
403
|
+
if (!isClosed) {
|
|
404
|
+
try {
|
|
405
|
+
controller.close();
|
|
406
|
+
} catch {
|
|
407
|
+
}
|
|
408
|
+
}
|
|
409
|
+
reader.releaseLock();
|
|
410
|
+
}
|
|
411
|
+
}
|
|
412
|
+
});
|
|
413
|
+
}
|
|
414
|
+
function convertOpenAIStreamToResponseApi(openaiStream) {
|
|
415
|
+
const decoder = new TextDecoder();
|
|
416
|
+
const encoder = new TextEncoder();
|
|
417
|
+
return new ReadableStream({
|
|
418
|
+
start: async (controller) => {
|
|
419
|
+
const reader = openaiStream.getReader();
|
|
420
|
+
let buffer = "";
|
|
421
|
+
let isClosed = false;
|
|
422
|
+
let accumulatedContent = "";
|
|
423
|
+
let model = "unknown";
|
|
424
|
+
const responseId = `resp_${Date.now()}`;
|
|
425
|
+
const safeEnqueue = (str) => {
|
|
426
|
+
if (!isClosed) {
|
|
427
|
+
try {
|
|
428
|
+
controller.enqueue(encoder.encode(str));
|
|
429
|
+
} catch {
|
|
430
|
+
isClosed = true;
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
};
|
|
434
|
+
const emitEvent = (event) => {
|
|
435
|
+
safeEnqueue(`data: ${JSON.stringify(event)}
|
|
436
|
+
|
|
437
|
+
`);
|
|
438
|
+
};
|
|
439
|
+
emitEvent({
|
|
440
|
+
type: "response.created",
|
|
441
|
+
response: { id: responseId, status: "in_progress" }
|
|
442
|
+
});
|
|
443
|
+
try {
|
|
444
|
+
while (true) {
|
|
445
|
+
const { done, value } = await reader.read();
|
|
446
|
+
if (done) break;
|
|
447
|
+
buffer += decoder.decode(value, { stream: true });
|
|
448
|
+
const lines = buffer.split("\n");
|
|
449
|
+
buffer = lines.pop() || "";
|
|
450
|
+
for (const line of lines) {
|
|
451
|
+
if (isClosed) break;
|
|
452
|
+
if (!line.startsWith("data:")) continue;
|
|
453
|
+
const data = line.slice(5).trim();
|
|
454
|
+
if (data === "[DONE]") continue;
|
|
455
|
+
try {
|
|
456
|
+
const chunk = JSON.parse(data);
|
|
457
|
+
const choice = chunk.choices?.[0];
|
|
458
|
+
model = chunk.model || model;
|
|
459
|
+
if (!choice) continue;
|
|
460
|
+
if (choice.delta?.content) {
|
|
461
|
+
accumulatedContent += choice.delta.content;
|
|
462
|
+
emitEvent({ type: "response.output_text.delta", delta: choice.delta.content });
|
|
463
|
+
}
|
|
464
|
+
if (choice.delta?.thinking?.content) {
|
|
465
|
+
emitEvent({
|
|
466
|
+
type: "response.reasoning_summary_text.delta",
|
|
467
|
+
delta: choice.delta.thinking.content
|
|
468
|
+
});
|
|
469
|
+
}
|
|
470
|
+
if (choice.finish_reason) {
|
|
471
|
+
emitEvent({ type: "response.output_text.done", text: accumulatedContent });
|
|
472
|
+
emitEvent({
|
|
473
|
+
type: "response.completed",
|
|
474
|
+
response: {
|
|
475
|
+
id: responseId,
|
|
476
|
+
status: "completed",
|
|
477
|
+
model,
|
|
478
|
+
output: [
|
|
479
|
+
{
|
|
480
|
+
type: "message",
|
|
481
|
+
role: "assistant",
|
|
482
|
+
content: [{ type: "output_text", text: accumulatedContent }]
|
|
483
|
+
}
|
|
484
|
+
],
|
|
485
|
+
usage: chunk.usage ? {
|
|
486
|
+
input_tokens: chunk.usage.prompt_tokens || 0,
|
|
487
|
+
output_tokens: chunk.usage.completion_tokens || 0,
|
|
488
|
+
total_tokens: chunk.usage.total_tokens || 0
|
|
489
|
+
} : void 0
|
|
490
|
+
}
|
|
491
|
+
});
|
|
492
|
+
}
|
|
493
|
+
} catch {
|
|
494
|
+
}
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
} catch (e) {
|
|
498
|
+
if (!isClosed) controller.error(e);
|
|
499
|
+
} finally {
|
|
500
|
+
if (!isClosed) {
|
|
501
|
+
try {
|
|
502
|
+
controller.close();
|
|
503
|
+
} catch {
|
|
504
|
+
}
|
|
505
|
+
}
|
|
506
|
+
reader.releaseLock();
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
});
|
|
510
|
+
}
|
|
511
|
+
export {
|
|
512
|
+
OpenAIResponseTransformer
|
|
513
|
+
};
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __export = (target, all) => {
|
|
7
|
+
for (var name in all)
|
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
+
};
|
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
+
for (let key of __getOwnPropNames(from))
|
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
+
|
|
20
|
+
// src/transformer/transformers/OpenCodeGoTransformer.ts
|
|
21
|
+
var OpenCodeGoTransformer_exports = {};
|
|
22
|
+
__export(OpenCodeGoTransformer_exports, {
|
|
23
|
+
OpenCodeGoTransformer: () => OpenCodeGoTransformer
|
|
24
|
+
});
|
|
25
|
+
module.exports = __toCommonJS(OpenCodeGoTransformer_exports);
|
|
26
|
+
var OpenCodeGoTransformer = class {
|
|
27
|
+
static TransformerName = "opencodego";
|
|
28
|
+
name = "opencodego";
|
|
29
|
+
endPoint = "/v1/chat/completions";
|
|
30
|
+
logger;
|
|
31
|
+
async auth(request, _provider, _context) {
|
|
32
|
+
return { body: request, config: { headers: {} } };
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Unified → OpenAI Chat Completions.
|
|
36
|
+
* Unified IS chat completions shape; this is mostly stripping `meta` and
|
|
37
|
+
* normalizing string-content single-block messages.
|
|
38
|
+
*/
|
|
39
|
+
async transformRequestIn(request, _provider, _context) {
|
|
40
|
+
const messages = request.messages.map((m) => {
|
|
41
|
+
const content = m.content;
|
|
42
|
+
if (Array.isArray(content) && content.length === 1 && content[0]?.type === "text") {
|
|
43
|
+
return { ...m, content: content[0].text };
|
|
44
|
+
}
|
|
45
|
+
return m;
|
|
46
|
+
});
|
|
47
|
+
const out = {
|
|
48
|
+
model: request.model,
|
|
49
|
+
messages,
|
|
50
|
+
stream: request.stream
|
|
51
|
+
};
|
|
52
|
+
if (request.temperature !== void 0) out.temperature = request.temperature;
|
|
53
|
+
if (request.max_tokens !== void 0) out.max_tokens = request.max_tokens;
|
|
54
|
+
if (request.tools && request.tools.length > 0) out.tools = request.tools;
|
|
55
|
+
if (request.tool_choice !== void 0) out.tool_choice = request.tool_choice;
|
|
56
|
+
if (request.reasoning?.effort && request.reasoning.effort !== "none") {
|
|
57
|
+
out.reasoning_effort = request.reasoning.effort;
|
|
58
|
+
}
|
|
59
|
+
return out;
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* OpenAI Chat Completions response → Unified.
|
|
63
|
+
* The upstream's response shape already matches Unified — pass through.
|
|
64
|
+
* The endpoint AnthropicTransformer re-encodes to Anthropic for the SDK.
|
|
65
|
+
*/
|
|
66
|
+
async transformResponseOut(response, _context) {
|
|
67
|
+
return response;
|
|
68
|
+
}
|
|
69
|
+
};
|
|
70
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
71
|
+
0 && (module.exports = {
|
|
72
|
+
OpenCodeGoTransformer
|
|
73
|
+
});
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { Transformer, TransformerLogger, LLMProvider, TransformerContext, UnifiedChatRequest } from '../types.cjs';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* OpenCodeGoTransformer — minimal request/response normalizer for OpenCodeGo's
|
|
5
|
+
* OpenAI-shape `/v1/chat/completions` upstream.
|
|
6
|
+
*
|
|
7
|
+
* The internal `UnifiedChatRequest` is already OpenAI-shape, so most of the
|
|
8
|
+
* conversion work happens upstream of this transformer (AnthropicTransformer
|
|
9
|
+
* decodes the SDK's Anthropic body to Unified at the chain's endpoint). This
|
|
10
|
+
* transformer's job is to:
|
|
11
|
+
*
|
|
12
|
+
* - Drop `meta` (internal routing metadata) before serializing.
|
|
13
|
+
* - Normalize `messages[].content` from `Array<MessageContent>` to a plain
|
|
14
|
+
* string when it's a single text block — some OpenCodeGo models reject
|
|
15
|
+
* the array form.
|
|
16
|
+
* - Drop fields the upstream rejects (e.g. `reasoning` when the model
|
|
17
|
+
* isn't a thinking model).
|
|
18
|
+
*
|
|
19
|
+
* **MiniMax models route to the Anthropic-shape `/v1/messages` upstream
|
|
20
|
+
* BYPASSING this transformer entirely** — see the dispatch proxy's
|
|
21
|
+
* subscription branch and `model-shape.ts`.
|
|
22
|
+
*
|
|
23
|
+
* @module transformer/transformers/OpenCodeGoTransformer
|
|
24
|
+
*/
|
|
25
|
+
|
|
26
|
+
declare class OpenCodeGoTransformer implements Transformer {
|
|
27
|
+
static TransformerName: string;
|
|
28
|
+
name: string;
|
|
29
|
+
endPoint: string;
|
|
30
|
+
logger?: TransformerLogger;
|
|
31
|
+
auth(request: unknown, _provider: LLMProvider, _context: TransformerContext): Promise<{
|
|
32
|
+
body: unknown;
|
|
33
|
+
config: {
|
|
34
|
+
headers: Record<string, string | undefined>;
|
|
35
|
+
};
|
|
36
|
+
}>;
|
|
37
|
+
/**
|
|
38
|
+
* Unified → OpenAI Chat Completions.
|
|
39
|
+
* Unified IS chat completions shape; this is mostly stripping `meta` and
|
|
40
|
+
* normalizing string-content single-block messages.
|
|
41
|
+
*/
|
|
42
|
+
transformRequestIn(request: UnifiedChatRequest, _provider: LLMProvider, _context: TransformerContext): Promise<Record<string, unknown>>;
|
|
43
|
+
/**
|
|
44
|
+
* OpenAI Chat Completions response → Unified.
|
|
45
|
+
* The upstream's response shape already matches Unified — pass through.
|
|
46
|
+
* The endpoint AnthropicTransformer re-encodes to Anthropic for the SDK.
|
|
47
|
+
*/
|
|
48
|
+
transformResponseOut(response: Response, _context: TransformerContext): Promise<Response>;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
export { OpenCodeGoTransformer };
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { Transformer, TransformerLogger, LLMProvider, TransformerContext, UnifiedChatRequest } from '../types.js';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* OpenCodeGoTransformer — minimal request/response normalizer for OpenCodeGo's
|
|
5
|
+
* OpenAI-shape `/v1/chat/completions` upstream.
|
|
6
|
+
*
|
|
7
|
+
* The internal `UnifiedChatRequest` is already OpenAI-shape, so most of the
|
|
8
|
+
* conversion work happens upstream of this transformer (AnthropicTransformer
|
|
9
|
+
* decodes the SDK's Anthropic body to Unified at the chain's endpoint). This
|
|
10
|
+
* transformer's job is to:
|
|
11
|
+
*
|
|
12
|
+
* - Drop `meta` (internal routing metadata) before serializing.
|
|
13
|
+
* - Normalize `messages[].content` from `Array<MessageContent>` to a plain
|
|
14
|
+
* string when it's a single text block — some OpenCodeGo models reject
|
|
15
|
+
* the array form.
|
|
16
|
+
* - Drop fields the upstream rejects (e.g. `reasoning` when the model
|
|
17
|
+
* isn't a thinking model).
|
|
18
|
+
*
|
|
19
|
+
* **MiniMax models route to the Anthropic-shape `/v1/messages` upstream
|
|
20
|
+
* BYPASSING this transformer entirely** — see the dispatch proxy's
|
|
21
|
+
* subscription branch and `model-shape.ts`.
|
|
22
|
+
*
|
|
23
|
+
* @module transformer/transformers/OpenCodeGoTransformer
|
|
24
|
+
*/
|
|
25
|
+
|
|
26
|
+
declare class OpenCodeGoTransformer implements Transformer {
|
|
27
|
+
static TransformerName: string;
|
|
28
|
+
name: string;
|
|
29
|
+
endPoint: string;
|
|
30
|
+
logger?: TransformerLogger;
|
|
31
|
+
auth(request: unknown, _provider: LLMProvider, _context: TransformerContext): Promise<{
|
|
32
|
+
body: unknown;
|
|
33
|
+
config: {
|
|
34
|
+
headers: Record<string, string | undefined>;
|
|
35
|
+
};
|
|
36
|
+
}>;
|
|
37
|
+
/**
|
|
38
|
+
* Unified → OpenAI Chat Completions.
|
|
39
|
+
* Unified IS chat completions shape; this is mostly stripping `meta` and
|
|
40
|
+
* normalizing string-content single-block messages.
|
|
41
|
+
*/
|
|
42
|
+
transformRequestIn(request: UnifiedChatRequest, _provider: LLMProvider, _context: TransformerContext): Promise<Record<string, unknown>>;
|
|
43
|
+
/**
|
|
44
|
+
* OpenAI Chat Completions response → Unified.
|
|
45
|
+
* The upstream's response shape already matches Unified — pass through.
|
|
46
|
+
* The endpoint AnthropicTransformer re-encodes to Anthropic for the SDK.
|
|
47
|
+
*/
|
|
48
|
+
transformResponseOut(response: Response, _context: TransformerContext): Promise<Response>;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
export { OpenCodeGoTransformer };
|