@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.
Files changed (114) hide show
  1. package/LICENSE +21 -0
  2. package/NOTICE +57 -0
  3. package/README.md +15 -0
  4. package/dist/ApiKeyPoolService-BmMkau07.d.cts +170 -0
  5. package/dist/ApiKeyPoolService-BmMkau07.d.ts +170 -0
  6. package/dist/ProviderProxy-f_8ziIhW.d.cts +120 -0
  7. package/dist/ProviderProxy-vjt8sQQk.d.ts +120 -0
  8. package/dist/SubscriptionAuthSource-Cr4fVEYY.d.cts +264 -0
  9. package/dist/SubscriptionAuthSource-D89zmiSS.d.ts +264 -0
  10. package/dist/auth/GeminiCodeAssistProjectResolver.cjs +218 -0
  11. package/dist/auth/GeminiCodeAssistProjectResolver.d.cts +68 -0
  12. package/dist/auth/GeminiCodeAssistProjectResolver.d.ts +68 -0
  13. package/dist/auth/GeminiCodeAssistProjectResolver.js +189 -0
  14. package/dist/completion/ApiKeyPoolService.cjs +331 -0
  15. package/dist/completion/ApiKeyPoolService.d.cts +2 -0
  16. package/dist/completion/ApiKeyPoolService.d.ts +2 -0
  17. package/dist/completion/ApiKeyPoolService.js +306 -0
  18. package/dist/completion.cjs +4027 -0
  19. package/dist/completion.d.cts +17 -0
  20. package/dist/completion.d.ts +17 -0
  21. package/dist/completion.js +3983 -0
  22. package/dist/index-BTSmc9Sm.d.ts +645 -0
  23. package/dist/index-DXazdTzZ.d.cts +645 -0
  24. package/dist/index.cjs +10428 -0
  25. package/dist/index.d.cts +128 -0
  26. package/dist/index.d.ts +128 -0
  27. package/dist/index.js +10339 -0
  28. package/dist/outbound-api/subscriptionRegistryPort.cjs +38 -0
  29. package/dist/outbound-api/subscriptionRegistryPort.d.cts +73 -0
  30. package/dist/outbound-api/subscriptionRegistryPort.d.ts +73 -0
  31. package/dist/outbound-api/subscriptionRegistryPort.js +12 -0
  32. package/dist/outbound-api.cjs +5264 -0
  33. package/dist/outbound-api.d.cts +320 -0
  34. package/dist/outbound-api.d.ts +320 -0
  35. package/dist/outbound-api.js +5218 -0
  36. package/dist/pipeline/SubscriptionAuthSource.cjs +131 -0
  37. package/dist/pipeline/SubscriptionAuthSource.d.cts +3 -0
  38. package/dist/pipeline/SubscriptionAuthSource.d.ts +3 -0
  39. package/dist/pipeline/SubscriptionAuthSource.js +103 -0
  40. package/dist/pipeline/SubscriptionAuthStrategy.cjs +18 -0
  41. package/dist/pipeline/SubscriptionAuthStrategy.d.cts +61 -0
  42. package/dist/pipeline/SubscriptionAuthStrategy.d.ts +61 -0
  43. package/dist/pipeline/SubscriptionAuthStrategy.js +0 -0
  44. package/dist/ports/gemini-code-assist-resolver.cjs +38 -0
  45. package/dist/ports/gemini-code-assist-resolver.d.cts +26 -0
  46. package/dist/ports/gemini-code-assist-resolver.d.ts +26 -0
  47. package/dist/ports/gemini-code-assist-resolver.js +12 -0
  48. package/dist/ports.cjs +18 -0
  49. package/dist/ports.d.cts +15 -0
  50. package/dist/ports.d.ts +15 -0
  51. package/dist/ports.js +0 -0
  52. package/dist/provider-proxy/ingress/providerProxyShared.cjs +2958 -0
  53. package/dist/provider-proxy/ingress/providerProxyShared.d.cts +77 -0
  54. package/dist/provider-proxy/ingress/providerProxyShared.d.ts +77 -0
  55. package/dist/provider-proxy/ingress/providerProxyShared.js +2925 -0
  56. package/dist/provider-proxy/matchText.cjs +73 -0
  57. package/dist/provider-proxy/matchText.d.cts +47 -0
  58. package/dist/provider-proxy/matchText.d.ts +47 -0
  59. package/dist/provider-proxy/matchText.js +45 -0
  60. package/dist/provider-proxy/types.cjs +18 -0
  61. package/dist/provider-proxy/types.d.cts +12 -0
  62. package/dist/provider-proxy/types.d.ts +12 -0
  63. package/dist/provider-proxy/types.js +0 -0
  64. package/dist/provider-proxy.cjs +4667 -0
  65. package/dist/provider-proxy.d.cts +69 -0
  66. package/dist/provider-proxy.d.ts +69 -0
  67. package/dist/provider-proxy.js +4636 -0
  68. package/dist/serializeError.cjs +82 -0
  69. package/dist/serializeError.d.cts +24 -0
  70. package/dist/serializeError.d.ts +24 -0
  71. package/dist/serializeError.js +57 -0
  72. package/dist/sse-parser.cjs +456 -0
  73. package/dist/sse-parser.d.cts +143 -0
  74. package/dist/sse-parser.d.ts +143 -0
  75. package/dist/sse-parser.js +430 -0
  76. package/dist/transformer/TransformerChainExecutor.cjs +321 -0
  77. package/dist/transformer/TransformerChainExecutor.d.cts +104 -0
  78. package/dist/transformer/TransformerChainExecutor.d.ts +104 -0
  79. package/dist/transformer/TransformerChainExecutor.js +294 -0
  80. package/dist/transformer/TransformerService.cjs +290 -0
  81. package/dist/transformer/TransformerService.d.cts +138 -0
  82. package/dist/transformer/TransformerService.d.ts +138 -0
  83. package/dist/transformer/TransformerService.js +265 -0
  84. package/dist/transformer/transformers/GeminiCodeAssistTransformer.cjs +1115 -0
  85. package/dist/transformer/transformers/GeminiCodeAssistTransformer.d.cts +102 -0
  86. package/dist/transformer/transformers/GeminiCodeAssistTransformer.d.ts +102 -0
  87. package/dist/transformer/transformers/GeminiCodeAssistTransformer.js +1085 -0
  88. package/dist/transformer/transformers/GeminiTransformer.cjs +1013 -0
  89. package/dist/transformer/transformers/GeminiTransformer.d.cts +70 -0
  90. package/dist/transformer/transformers/GeminiTransformer.d.ts +70 -0
  91. package/dist/transformer/transformers/GeminiTransformer.js +986 -0
  92. package/dist/transformer/transformers/OpenAIResponseTransformer.cjs +538 -0
  93. package/dist/transformer/transformers/OpenAIResponseTransformer.d.cts +53 -0
  94. package/dist/transformer/transformers/OpenAIResponseTransformer.d.ts +53 -0
  95. package/dist/transformer/transformers/OpenAIResponseTransformer.js +513 -0
  96. package/dist/transformer/transformers/OpenCodeGoTransformer.cjs +73 -0
  97. package/dist/transformer/transformers/OpenCodeGoTransformer.d.cts +51 -0
  98. package/dist/transformer/transformers/OpenCodeGoTransformer.d.ts +51 -0
  99. package/dist/transformer/transformers/OpenCodeGoTransformer.js +48 -0
  100. package/dist/transformer/types.cjs +18 -0
  101. package/dist/transformer/types.d.cts +405 -0
  102. package/dist/transformer/types.d.ts +405 -0
  103. package/dist/transformer/types.js +0 -0
  104. package/dist/transformer.cjs +3736 -0
  105. package/dist/transformer.d.cts +33 -0
  106. package/dist/transformer.d.ts +33 -0
  107. package/dist/transformer.js +3712 -0
  108. package/dist/types-CGGrKqC_.d.cts +142 -0
  109. package/dist/types-CbCN2NQP.d.ts +142 -0
  110. package/dist/types-DCzHkhJt.d.ts +467 -0
  111. package/dist/types-DZIQbgp0.d.cts +467 -0
  112. package/dist/usage-event-sink-BX7FE1NL.d.cts +59 -0
  113. package/dist/usage-event-sink-BX7FE1NL.d.ts +59 -0
  114. 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 };