@lenylvt/pi-ai 0.64.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 (167) hide show
  1. package/README.md +203 -0
  2. package/dist/api-registry.d.ts +20 -0
  3. package/dist/api-registry.d.ts.map +1 -0
  4. package/dist/api-registry.js +44 -0
  5. package/dist/api-registry.js.map +1 -0
  6. package/dist/cli.d.ts +3 -0
  7. package/dist/cli.d.ts.map +1 -0
  8. package/dist/cli.js +119 -0
  9. package/dist/cli.js.map +1 -0
  10. package/dist/env-api-keys.d.ts +7 -0
  11. package/dist/env-api-keys.d.ts.map +1 -0
  12. package/dist/env-api-keys.js +13 -0
  13. package/dist/env-api-keys.js.map +1 -0
  14. package/dist/index.d.ts +20 -0
  15. package/dist/index.d.ts.map +1 -0
  16. package/dist/index.js +14 -0
  17. package/dist/index.js.map +1 -0
  18. package/dist/models.d.ts +24 -0
  19. package/dist/models.d.ts.map +1 -0
  20. package/dist/models.generated.d.ts +2332 -0
  21. package/dist/models.generated.d.ts.map +1 -0
  22. package/dist/models.generated.js +2186 -0
  23. package/dist/models.generated.js.map +1 -0
  24. package/dist/models.js +60 -0
  25. package/dist/models.js.map +1 -0
  26. package/dist/oauth.d.ts +2 -0
  27. package/dist/oauth.d.ts.map +1 -0
  28. package/dist/oauth.js +2 -0
  29. package/dist/oauth.js.map +1 -0
  30. package/dist/providers/anthropic.d.ts +40 -0
  31. package/dist/providers/anthropic.d.ts.map +1 -0
  32. package/dist/providers/anthropic.js +749 -0
  33. package/dist/providers/anthropic.js.map +1 -0
  34. package/dist/providers/faux.d.ts +56 -0
  35. package/dist/providers/faux.d.ts.map +1 -0
  36. package/dist/providers/faux.js +367 -0
  37. package/dist/providers/faux.js.map +1 -0
  38. package/dist/providers/github-copilot-headers.d.ts +8 -0
  39. package/dist/providers/github-copilot-headers.d.ts.map +1 -0
  40. package/dist/providers/github-copilot-headers.js +29 -0
  41. package/dist/providers/github-copilot-headers.js.map +1 -0
  42. package/dist/providers/openai-codex-responses.d.ts +9 -0
  43. package/dist/providers/openai-codex-responses.d.ts.map +1 -0
  44. package/dist/providers/openai-codex-responses.js +741 -0
  45. package/dist/providers/openai-codex-responses.js.map +1 -0
  46. package/dist/providers/openai-completions.d.ts +15 -0
  47. package/dist/providers/openai-completions.d.ts.map +1 -0
  48. package/dist/providers/openai-completions.js +687 -0
  49. package/dist/providers/openai-completions.js.map +1 -0
  50. package/dist/providers/openai-responses-shared.d.ts +17 -0
  51. package/dist/providers/openai-responses-shared.d.ts.map +1 -0
  52. package/dist/providers/openai-responses-shared.js +458 -0
  53. package/dist/providers/openai-responses-shared.js.map +1 -0
  54. package/dist/providers/openai-responses.d.ts +13 -0
  55. package/dist/providers/openai-responses.d.ts.map +1 -0
  56. package/dist/providers/openai-responses.js +190 -0
  57. package/dist/providers/openai-responses.js.map +1 -0
  58. package/dist/providers/register-builtins.d.ts +16 -0
  59. package/dist/providers/register-builtins.d.ts.map +1 -0
  60. package/dist/providers/register-builtins.js +140 -0
  61. package/dist/providers/register-builtins.js.map +1 -0
  62. package/dist/providers/simple-options.d.ts +8 -0
  63. package/dist/providers/simple-options.d.ts.map +1 -0
  64. package/dist/providers/simple-options.js +35 -0
  65. package/dist/providers/simple-options.js.map +1 -0
  66. package/dist/providers/transform-messages.d.ts +8 -0
  67. package/dist/providers/transform-messages.d.ts.map +1 -0
  68. package/dist/providers/transform-messages.js +155 -0
  69. package/dist/providers/transform-messages.js.map +1 -0
  70. package/dist/stream.d.ts +8 -0
  71. package/dist/stream.d.ts.map +1 -0
  72. package/dist/stream.js +27 -0
  73. package/dist/stream.js.map +1 -0
  74. package/dist/types.d.ts +283 -0
  75. package/dist/types.d.ts.map +1 -0
  76. package/dist/types.js +2 -0
  77. package/dist/types.js.map +1 -0
  78. package/dist/utils/event-stream.d.ts +21 -0
  79. package/dist/utils/event-stream.d.ts.map +1 -0
  80. package/dist/utils/event-stream.js +81 -0
  81. package/dist/utils/event-stream.js.map +1 -0
  82. package/dist/utils/hash.d.ts +3 -0
  83. package/dist/utils/hash.d.ts.map +1 -0
  84. package/dist/utils/hash.js +14 -0
  85. package/dist/utils/hash.js.map +1 -0
  86. package/dist/utils/json-parse.d.ts +9 -0
  87. package/dist/utils/json-parse.d.ts.map +1 -0
  88. package/dist/utils/json-parse.js +29 -0
  89. package/dist/utils/json-parse.js.map +1 -0
  90. package/dist/utils/oauth/anthropic.d.ts +25 -0
  91. package/dist/utils/oauth/anthropic.d.ts.map +1 -0
  92. package/dist/utils/oauth/anthropic.js +335 -0
  93. package/dist/utils/oauth/anthropic.js.map +1 -0
  94. package/dist/utils/oauth/github-copilot.d.ts +30 -0
  95. package/dist/utils/oauth/github-copilot.d.ts.map +1 -0
  96. package/dist/utils/oauth/github-copilot.js +292 -0
  97. package/dist/utils/oauth/github-copilot.js.map +1 -0
  98. package/dist/utils/oauth/index.d.ts +36 -0
  99. package/dist/utils/oauth/index.d.ts.map +1 -0
  100. package/dist/utils/oauth/index.js +92 -0
  101. package/dist/utils/oauth/index.js.map +1 -0
  102. package/dist/utils/oauth/oauth-page.d.ts +3 -0
  103. package/dist/utils/oauth/oauth-page.d.ts.map +1 -0
  104. package/dist/utils/oauth/oauth-page.js +105 -0
  105. package/dist/utils/oauth/oauth-page.js.map +1 -0
  106. package/dist/utils/oauth/openai-codex.d.ts +34 -0
  107. package/dist/utils/oauth/openai-codex.d.ts.map +1 -0
  108. package/dist/utils/oauth/openai-codex.js +373 -0
  109. package/dist/utils/oauth/openai-codex.js.map +1 -0
  110. package/dist/utils/oauth/pkce.d.ts +13 -0
  111. package/dist/utils/oauth/pkce.d.ts.map +1 -0
  112. package/dist/utils/oauth/pkce.js +31 -0
  113. package/dist/utils/oauth/pkce.js.map +1 -0
  114. package/dist/utils/oauth/types.d.ts +47 -0
  115. package/dist/utils/oauth/types.d.ts.map +1 -0
  116. package/dist/utils/oauth/types.js +2 -0
  117. package/dist/utils/oauth/types.js.map +1 -0
  118. package/dist/utils/overflow.d.ts +53 -0
  119. package/dist/utils/overflow.d.ts.map +1 -0
  120. package/dist/utils/overflow.js +119 -0
  121. package/dist/utils/overflow.js.map +1 -0
  122. package/dist/utils/sanitize-unicode.d.ts +22 -0
  123. package/dist/utils/sanitize-unicode.d.ts.map +1 -0
  124. package/dist/utils/sanitize-unicode.js +26 -0
  125. package/dist/utils/sanitize-unicode.js.map +1 -0
  126. package/dist/utils/typebox-helpers.d.ts +17 -0
  127. package/dist/utils/typebox-helpers.d.ts.map +1 -0
  128. package/dist/utils/typebox-helpers.js +21 -0
  129. package/dist/utils/typebox-helpers.js.map +1 -0
  130. package/dist/utils/validation.d.ts +18 -0
  131. package/dist/utils/validation.d.ts.map +1 -0
  132. package/dist/utils/validation.js +80 -0
  133. package/dist/utils/validation.js.map +1 -0
  134. package/package.json +89 -0
  135. package/src/api-registry.ts +98 -0
  136. package/src/cli.ts +136 -0
  137. package/src/env-api-keys.ts +22 -0
  138. package/src/index.ts +29 -0
  139. package/src/models.generated.ts +2188 -0
  140. package/src/models.ts +82 -0
  141. package/src/oauth.ts +1 -0
  142. package/src/providers/anthropic.ts +905 -0
  143. package/src/providers/faux.ts +498 -0
  144. package/src/providers/github-copilot-headers.ts +37 -0
  145. package/src/providers/openai-codex-responses.ts +929 -0
  146. package/src/providers/openai-completions.ts +811 -0
  147. package/src/providers/openai-responses-shared.ts +513 -0
  148. package/src/providers/openai-responses.ts +251 -0
  149. package/src/providers/register-builtins.ts +232 -0
  150. package/src/providers/simple-options.ts +46 -0
  151. package/src/providers/transform-messages.ts +172 -0
  152. package/src/stream.ts +59 -0
  153. package/src/types.ts +294 -0
  154. package/src/utils/event-stream.ts +87 -0
  155. package/src/utils/hash.ts +13 -0
  156. package/src/utils/json-parse.ts +28 -0
  157. package/src/utils/oauth/anthropic.ts +402 -0
  158. package/src/utils/oauth/github-copilot.ts +396 -0
  159. package/src/utils/oauth/index.ts +123 -0
  160. package/src/utils/oauth/oauth-page.ts +109 -0
  161. package/src/utils/oauth/openai-codex.ts +450 -0
  162. package/src/utils/oauth/pkce.ts +34 -0
  163. package/src/utils/oauth/types.ts +59 -0
  164. package/src/utils/overflow.ts +125 -0
  165. package/src/utils/sanitize-unicode.ts +25 -0
  166. package/src/utils/typebox-helpers.ts +24 -0
  167. package/src/utils/validation.ts +93 -0
@@ -0,0 +1,929 @@
1
+ import type * as NodeOs from "node:os";
2
+ import type { Tool as OpenAITool, ResponseInput, ResponseStreamEvent } from "openai/resources/responses/responses.js";
3
+
4
+ // NEVER convert to top-level runtime imports - breaks browser/Vite builds (web-ui)
5
+ let _os: typeof NodeOs | null = null;
6
+
7
+ type DynamicImport = (specifier: string) => Promise<unknown>;
8
+
9
+ const dynamicImport: DynamicImport = (specifier) => import(specifier);
10
+ const NODE_OS_SPECIFIER = "node:" + "os";
11
+
12
+ if (typeof process !== "undefined" && (process.versions?.node || process.versions?.bun)) {
13
+ dynamicImport(NODE_OS_SPECIFIER).then((m) => {
14
+ _os = m as typeof NodeOs;
15
+ });
16
+ }
17
+
18
+ import { getEnvApiKey } from "../env-api-keys.js";
19
+ import { supportsXhigh } from "../models.js";
20
+ import type {
21
+ Api,
22
+ AssistantMessage,
23
+ Context,
24
+ Model,
25
+ SimpleStreamOptions,
26
+ StreamFunction,
27
+ StreamOptions,
28
+ } from "../types.js";
29
+ import { AssistantMessageEventStream } from "../utils/event-stream.js";
30
+ import { convertResponsesMessages, convertResponsesTools, processResponsesStream } from "./openai-responses-shared.js";
31
+ import { buildBaseOptions, clampReasoning } from "./simple-options.js";
32
+
33
+ // ============================================================================
34
+ // Configuration
35
+ // ============================================================================
36
+
37
+ const DEFAULT_CODEX_BASE_URL = "https://chatgpt.com/backend-api";
38
+ const JWT_CLAIM_PATH = "https://api.openai.com/auth" as const;
39
+ const MAX_RETRIES = 3;
40
+ const BASE_DELAY_MS = 1000;
41
+ const CODEX_TOOL_CALL_PROVIDERS = new Set(["openai-codex"]);
42
+
43
+ const CODEX_RESPONSE_STATUSES = new Set<CodexResponseStatus>([
44
+ "completed",
45
+ "incomplete",
46
+ "failed",
47
+ "cancelled",
48
+ "queued",
49
+ "in_progress",
50
+ ]);
51
+
52
+ // ============================================================================
53
+ // Types
54
+ // ============================================================================
55
+
56
+ export interface OpenAICodexResponsesOptions extends StreamOptions {
57
+ reasoningEffort?: "none" | "minimal" | "low" | "medium" | "high" | "xhigh";
58
+ reasoningSummary?: "auto" | "concise" | "detailed" | "off" | "on" | null;
59
+ textVerbosity?: "low" | "medium" | "high";
60
+ }
61
+
62
+ type CodexResponseStatus = "completed" | "incomplete" | "failed" | "cancelled" | "queued" | "in_progress";
63
+
64
+ interface RequestBody {
65
+ model: string;
66
+ store?: boolean;
67
+ stream?: boolean;
68
+ instructions?: string;
69
+ input?: ResponseInput;
70
+ tools?: OpenAITool[];
71
+ tool_choice?: "auto";
72
+ parallel_tool_calls?: boolean;
73
+ temperature?: number;
74
+ reasoning?: { effort?: string; summary?: string };
75
+ text?: { verbosity?: string };
76
+ include?: string[];
77
+ prompt_cache_key?: string;
78
+ [key: string]: unknown;
79
+ }
80
+
81
+ // ============================================================================
82
+ // Retry Helpers
83
+ // ============================================================================
84
+
85
+ function isRetryableError(status: number, errorText: string): boolean {
86
+ if (status === 429 || status === 500 || status === 502 || status === 503 || status === 504) {
87
+ return true;
88
+ }
89
+ return /rate.?limit|overloaded|service.?unavailable|upstream.?connect|connection.?refused/i.test(errorText);
90
+ }
91
+
92
+ function sleep(ms: number, signal?: AbortSignal): Promise<void> {
93
+ return new Promise((resolve, reject) => {
94
+ if (signal?.aborted) {
95
+ reject(new Error("Request was aborted"));
96
+ return;
97
+ }
98
+ const timeout = setTimeout(resolve, ms);
99
+ signal?.addEventListener("abort", () => {
100
+ clearTimeout(timeout);
101
+ reject(new Error("Request was aborted"));
102
+ });
103
+ });
104
+ }
105
+
106
+ // ============================================================================
107
+ // Main Stream Function
108
+ // ============================================================================
109
+
110
+ export const streamOpenAICodexResponses: StreamFunction<"openai-codex-responses", OpenAICodexResponsesOptions> = (
111
+ model: Model<"openai-codex-responses">,
112
+ context: Context,
113
+ options?: OpenAICodexResponsesOptions,
114
+ ): AssistantMessageEventStream => {
115
+ const stream = new AssistantMessageEventStream();
116
+
117
+ (async () => {
118
+ const output: AssistantMessage = {
119
+ role: "assistant",
120
+ content: [],
121
+ api: "openai-codex-responses" as Api,
122
+ provider: model.provider,
123
+ model: model.id,
124
+ usage: {
125
+ input: 0,
126
+ output: 0,
127
+ cacheRead: 0,
128
+ cacheWrite: 0,
129
+ totalTokens: 0,
130
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
131
+ },
132
+ stopReason: "stop",
133
+ timestamp: Date.now(),
134
+ };
135
+
136
+ try {
137
+ const apiKey = options?.apiKey || getEnvApiKey(model.provider) || "";
138
+ if (!apiKey) {
139
+ throw new Error(`No API key for provider: ${model.provider}`);
140
+ }
141
+
142
+ const accountId = extractAccountId(apiKey);
143
+ let body = buildRequestBody(model, context, options);
144
+ const nextBody = await options?.onPayload?.(body, model);
145
+ if (nextBody !== undefined) {
146
+ body = nextBody as RequestBody;
147
+ }
148
+ const websocketRequestId = options?.sessionId || createCodexRequestId();
149
+ const sseHeaders = buildSSEHeaders(model.headers, options?.headers, accountId, apiKey, options?.sessionId);
150
+ const websocketHeaders = buildWebSocketHeaders(
151
+ model.headers,
152
+ options?.headers,
153
+ accountId,
154
+ apiKey,
155
+ websocketRequestId,
156
+ );
157
+ const bodyJson = JSON.stringify(body);
158
+ const transport = options?.transport || "sse";
159
+
160
+ if (transport !== "sse") {
161
+ let websocketStarted = false;
162
+ try {
163
+ await processWebSocketStream(
164
+ resolveCodexWebSocketUrl(model.baseUrl),
165
+ body,
166
+ websocketHeaders,
167
+ output,
168
+ stream,
169
+ model,
170
+ () => {
171
+ websocketStarted = true;
172
+ },
173
+ options,
174
+ );
175
+
176
+ if (options?.signal?.aborted) {
177
+ throw new Error("Request was aborted");
178
+ }
179
+ stream.push({
180
+ type: "done",
181
+ reason: output.stopReason as "stop" | "length" | "toolUse",
182
+ message: output,
183
+ });
184
+ stream.end();
185
+ return;
186
+ } catch (error) {
187
+ if (transport === "websocket" || websocketStarted) {
188
+ throw error;
189
+ }
190
+ }
191
+ }
192
+
193
+ // Fetch with retry logic for rate limits and transient errors
194
+ let response: Response | undefined;
195
+ let lastError: Error | undefined;
196
+
197
+ for (let attempt = 0; attempt <= MAX_RETRIES; attempt++) {
198
+ if (options?.signal?.aborted) {
199
+ throw new Error("Request was aborted");
200
+ }
201
+
202
+ try {
203
+ response = await fetch(resolveCodexUrl(model.baseUrl), {
204
+ method: "POST",
205
+ headers: sseHeaders,
206
+ body: bodyJson,
207
+ signal: options?.signal,
208
+ });
209
+
210
+ if (response.ok) {
211
+ break;
212
+ }
213
+
214
+ const errorText = await response.text();
215
+ if (attempt < MAX_RETRIES && isRetryableError(response.status, errorText)) {
216
+ const delayMs = BASE_DELAY_MS * 2 ** attempt;
217
+ await sleep(delayMs, options?.signal);
218
+ continue;
219
+ }
220
+
221
+ // Parse error for friendly message on final attempt or non-retryable error
222
+ const fakeResponse = new Response(errorText, {
223
+ status: response.status,
224
+ statusText: response.statusText,
225
+ });
226
+ const info = await parseErrorResponse(fakeResponse);
227
+ throw new Error(info.friendlyMessage || info.message);
228
+ } catch (error) {
229
+ if (error instanceof Error) {
230
+ if (error.name === "AbortError" || error.message === "Request was aborted") {
231
+ throw new Error("Request was aborted");
232
+ }
233
+ }
234
+ lastError = error instanceof Error ? error : new Error(String(error));
235
+ // Network errors are retryable
236
+ if (attempt < MAX_RETRIES && !lastError.message.includes("usage limit")) {
237
+ const delayMs = BASE_DELAY_MS * 2 ** attempt;
238
+ await sleep(delayMs, options?.signal);
239
+ continue;
240
+ }
241
+ throw lastError;
242
+ }
243
+ }
244
+
245
+ if (!response?.ok) {
246
+ throw lastError ?? new Error("Failed after retries");
247
+ }
248
+
249
+ if (!response.body) {
250
+ throw new Error("No response body");
251
+ }
252
+
253
+ stream.push({ type: "start", partial: output });
254
+ await processStream(response, output, stream, model);
255
+
256
+ if (options?.signal?.aborted) {
257
+ throw new Error("Request was aborted");
258
+ }
259
+
260
+ stream.push({ type: "done", reason: output.stopReason as "stop" | "length" | "toolUse", message: output });
261
+ stream.end();
262
+ } catch (error) {
263
+ output.stopReason = options?.signal?.aborted ? "aborted" : "error";
264
+ output.errorMessage = error instanceof Error ? error.message : String(error);
265
+ stream.push({ type: "error", reason: output.stopReason, error: output });
266
+ stream.end();
267
+ }
268
+ })();
269
+
270
+ return stream;
271
+ };
272
+
273
+ export const streamSimpleOpenAICodexResponses: StreamFunction<"openai-codex-responses", SimpleStreamOptions> = (
274
+ model: Model<"openai-codex-responses">,
275
+ context: Context,
276
+ options?: SimpleStreamOptions,
277
+ ): AssistantMessageEventStream => {
278
+ const apiKey = options?.apiKey || getEnvApiKey(model.provider);
279
+ if (!apiKey) {
280
+ throw new Error(`No API key for provider: ${model.provider}`);
281
+ }
282
+
283
+ const base = buildBaseOptions(model, options, apiKey);
284
+ const reasoningEffort = supportsXhigh(model) ? options?.reasoning : clampReasoning(options?.reasoning);
285
+
286
+ return streamOpenAICodexResponses(model, context, {
287
+ ...base,
288
+ reasoningEffort,
289
+ } satisfies OpenAICodexResponsesOptions);
290
+ };
291
+
292
+ // ============================================================================
293
+ // Request Building
294
+ // ============================================================================
295
+
296
+ function buildRequestBody(
297
+ model: Model<"openai-codex-responses">,
298
+ context: Context,
299
+ options?: OpenAICodexResponsesOptions,
300
+ ): RequestBody {
301
+ const messages = convertResponsesMessages(model, context, CODEX_TOOL_CALL_PROVIDERS, {
302
+ includeSystemPrompt: false,
303
+ });
304
+
305
+ const body: RequestBody = {
306
+ model: model.id,
307
+ store: false,
308
+ stream: true,
309
+ instructions: context.systemPrompt,
310
+ input: messages,
311
+ text: { verbosity: options?.textVerbosity || "medium" },
312
+ include: ["reasoning.encrypted_content"],
313
+ prompt_cache_key: options?.sessionId,
314
+ tool_choice: "auto",
315
+ parallel_tool_calls: true,
316
+ };
317
+
318
+ if (options?.temperature !== undefined) {
319
+ body.temperature = options.temperature;
320
+ }
321
+
322
+ if (context.tools) {
323
+ body.tools = convertResponsesTools(context.tools, { strict: null });
324
+ }
325
+
326
+ if (options?.reasoningEffort !== undefined) {
327
+ body.reasoning = {
328
+ effort: clampReasoningEffort(model.id, options.reasoningEffort),
329
+ summary: options.reasoningSummary ?? "auto",
330
+ };
331
+ }
332
+
333
+ return body;
334
+ }
335
+
336
+ function clampReasoningEffort(modelId: string, effort: string): string {
337
+ const id = modelId.includes("/") ? modelId.split("/").pop()! : modelId;
338
+ if ((id.startsWith("gpt-5.2") || id.startsWith("gpt-5.3") || id.startsWith("gpt-5.4")) && effort === "minimal")
339
+ return "low";
340
+ if (id === "gpt-5.1" && effort === "xhigh") return "high";
341
+ if (id === "gpt-5.1-codex-mini") return effort === "high" || effort === "xhigh" ? "high" : "medium";
342
+ return effort;
343
+ }
344
+
345
+ function resolveCodexUrl(baseUrl?: string): string {
346
+ const raw = baseUrl && baseUrl.trim().length > 0 ? baseUrl : DEFAULT_CODEX_BASE_URL;
347
+ const normalized = raw.replace(/\/+$/, "");
348
+ if (normalized.endsWith("/codex/responses")) return normalized;
349
+ if (normalized.endsWith("/codex")) return `${normalized}/responses`;
350
+ return `${normalized}/codex/responses`;
351
+ }
352
+
353
+ function resolveCodexWebSocketUrl(baseUrl?: string): string {
354
+ const url = new URL(resolveCodexUrl(baseUrl));
355
+ if (url.protocol === "https:") url.protocol = "wss:";
356
+ if (url.protocol === "http:") url.protocol = "ws:";
357
+ return url.toString();
358
+ }
359
+
360
+ // ============================================================================
361
+ // Response Processing
362
+ // ============================================================================
363
+
364
+ async function processStream(
365
+ response: Response,
366
+ output: AssistantMessage,
367
+ stream: AssistantMessageEventStream,
368
+ model: Model<"openai-codex-responses">,
369
+ ): Promise<void> {
370
+ await processResponsesStream(mapCodexEvents(parseSSE(response)), output, stream, model);
371
+ }
372
+
373
+ async function* mapCodexEvents(events: AsyncIterable<Record<string, unknown>>): AsyncGenerator<ResponseStreamEvent> {
374
+ for await (const event of events) {
375
+ const type = typeof event.type === "string" ? event.type : undefined;
376
+ if (!type) continue;
377
+
378
+ if (type === "error") {
379
+ const code = (event as { code?: string }).code || "";
380
+ const message = (event as { message?: string }).message || "";
381
+ throw new Error(`Codex error: ${message || code || JSON.stringify(event)}`);
382
+ }
383
+
384
+ if (type === "response.failed") {
385
+ const msg = (event as { response?: { error?: { message?: string } } }).response?.error?.message;
386
+ throw new Error(msg || "Codex response failed");
387
+ }
388
+
389
+ if (type === "response.done" || type === "response.completed" || type === "response.incomplete") {
390
+ const response = (event as { response?: { status?: unknown } }).response;
391
+ const normalizedResponse = response
392
+ ? { ...response, status: normalizeCodexStatus(response.status) }
393
+ : response;
394
+ yield { ...event, type: "response.completed", response: normalizedResponse } as ResponseStreamEvent;
395
+ return;
396
+ }
397
+
398
+ yield event as unknown as ResponseStreamEvent;
399
+ }
400
+ }
401
+
402
+ function normalizeCodexStatus(status: unknown): CodexResponseStatus | undefined {
403
+ if (typeof status !== "string") return undefined;
404
+ return CODEX_RESPONSE_STATUSES.has(status as CodexResponseStatus) ? (status as CodexResponseStatus) : undefined;
405
+ }
406
+
407
+ // ============================================================================
408
+ // SSE Parsing
409
+ // ============================================================================
410
+
411
+ async function* parseSSE(response: Response): AsyncGenerator<Record<string, unknown>> {
412
+ if (!response.body) return;
413
+
414
+ const reader = response.body.getReader();
415
+ const decoder = new TextDecoder();
416
+ let buffer = "";
417
+
418
+ try {
419
+ while (true) {
420
+ const { done, value } = await reader.read();
421
+ if (done) break;
422
+ buffer += decoder.decode(value, { stream: true });
423
+
424
+ let idx = buffer.indexOf("\n\n");
425
+ while (idx !== -1) {
426
+ const chunk = buffer.slice(0, idx);
427
+ buffer = buffer.slice(idx + 2);
428
+
429
+ const dataLines = chunk
430
+ .split("\n")
431
+ .filter((l) => l.startsWith("data:"))
432
+ .map((l) => l.slice(5).trim());
433
+ if (dataLines.length > 0) {
434
+ const data = dataLines.join("\n").trim();
435
+ if (data && data !== "[DONE]") {
436
+ try {
437
+ yield JSON.parse(data);
438
+ } catch {}
439
+ }
440
+ }
441
+ idx = buffer.indexOf("\n\n");
442
+ }
443
+ }
444
+ } finally {
445
+ try {
446
+ await reader.cancel();
447
+ } catch {}
448
+ try {
449
+ reader.releaseLock();
450
+ } catch {}
451
+ }
452
+ }
453
+
454
+ // ============================================================================
455
+ // WebSocket Parsing
456
+ // ============================================================================
457
+
458
+ const OPENAI_BETA_RESPONSES_WEBSOCKETS = "responses_websockets=2026-02-06";
459
+ const SESSION_WEBSOCKET_CACHE_TTL_MS = 5 * 60 * 1000;
460
+
461
+ type WebSocketEventType = "open" | "message" | "error" | "close";
462
+ type WebSocketListener = (event: unknown) => void;
463
+
464
+ interface WebSocketLike {
465
+ close(code?: number, reason?: string): void;
466
+ send(data: string): void;
467
+ addEventListener(type: WebSocketEventType, listener: WebSocketListener): void;
468
+ removeEventListener(type: WebSocketEventType, listener: WebSocketListener): void;
469
+ }
470
+
471
+ interface CachedWebSocketConnection {
472
+ socket: WebSocketLike;
473
+ busy: boolean;
474
+ idleTimer?: ReturnType<typeof setTimeout>;
475
+ }
476
+
477
+ const websocketSessionCache = new Map<string, CachedWebSocketConnection>();
478
+
479
+ type WebSocketConstructor = new (
480
+ url: string,
481
+ protocols?: string | string[] | { headers?: Record<string, string> },
482
+ ) => WebSocketLike;
483
+
484
+ function getWebSocketConstructor(): WebSocketConstructor | null {
485
+ const ctor = (globalThis as { WebSocket?: unknown }).WebSocket;
486
+ if (typeof ctor !== "function") return null;
487
+ return ctor as unknown as WebSocketConstructor;
488
+ }
489
+
490
+ function headersToRecord(headers: Headers): Record<string, string> {
491
+ const out: Record<string, string> = {};
492
+ for (const [key, value] of headers.entries()) {
493
+ out[key] = value;
494
+ }
495
+ return out;
496
+ }
497
+
498
+ function getWebSocketReadyState(socket: WebSocketLike): number | undefined {
499
+ const readyState = (socket as { readyState?: unknown }).readyState;
500
+ return typeof readyState === "number" ? readyState : undefined;
501
+ }
502
+
503
+ function isWebSocketReusable(socket: WebSocketLike): boolean {
504
+ const readyState = getWebSocketReadyState(socket);
505
+ // If readyState is unavailable, assume the runtime keeps it open/reusable.
506
+ return readyState === undefined || readyState === 1;
507
+ }
508
+
509
+ function closeWebSocketSilently(socket: WebSocketLike, code = 1000, reason = "done"): void {
510
+ try {
511
+ socket.close(code, reason);
512
+ } catch {}
513
+ }
514
+
515
+ function scheduleSessionWebSocketExpiry(sessionId: string, entry: CachedWebSocketConnection): void {
516
+ if (entry.idleTimer) {
517
+ clearTimeout(entry.idleTimer);
518
+ }
519
+ entry.idleTimer = setTimeout(() => {
520
+ if (entry.busy) return;
521
+ closeWebSocketSilently(entry.socket, 1000, "idle_timeout");
522
+ websocketSessionCache.delete(sessionId);
523
+ }, SESSION_WEBSOCKET_CACHE_TTL_MS);
524
+ }
525
+
526
+ async function connectWebSocket(url: string, headers: Headers, signal?: AbortSignal): Promise<WebSocketLike> {
527
+ const WebSocketCtor = getWebSocketConstructor();
528
+ if (!WebSocketCtor) {
529
+ throw new Error("WebSocket transport is not available in this runtime");
530
+ }
531
+
532
+ const wsHeaders = headersToRecord(headers);
533
+ delete wsHeaders["OpenAI-Beta"];
534
+
535
+ return new Promise<WebSocketLike>((resolve, reject) => {
536
+ let settled = false;
537
+ let socket: WebSocketLike;
538
+
539
+ try {
540
+ socket = new WebSocketCtor(url, { headers: wsHeaders });
541
+ } catch (error) {
542
+ reject(error instanceof Error ? error : new Error(String(error)));
543
+ return;
544
+ }
545
+
546
+ const onOpen: WebSocketListener = () => {
547
+ if (settled) return;
548
+ settled = true;
549
+ cleanup();
550
+ resolve(socket);
551
+ };
552
+ const onError: WebSocketListener = (event) => {
553
+ const error = extractWebSocketError(event);
554
+ if (settled) return;
555
+ settled = true;
556
+ cleanup();
557
+ reject(error);
558
+ };
559
+ const onClose: WebSocketListener = (event) => {
560
+ const error = extractWebSocketCloseError(event);
561
+ if (settled) return;
562
+ settled = true;
563
+ cleanup();
564
+ reject(error);
565
+ };
566
+ const onAbort = () => {
567
+ if (settled) return;
568
+ settled = true;
569
+ cleanup();
570
+ socket.close(1000, "aborted");
571
+ reject(new Error("Request was aborted"));
572
+ };
573
+
574
+ const cleanup = () => {
575
+ socket.removeEventListener("open", onOpen);
576
+ socket.removeEventListener("error", onError);
577
+ socket.removeEventListener("close", onClose);
578
+ signal?.removeEventListener("abort", onAbort);
579
+ };
580
+
581
+ socket.addEventListener("open", onOpen);
582
+ socket.addEventListener("error", onError);
583
+ socket.addEventListener("close", onClose);
584
+ signal?.addEventListener("abort", onAbort);
585
+ });
586
+ }
587
+
588
+ async function acquireWebSocket(
589
+ url: string,
590
+ headers: Headers,
591
+ sessionId: string | undefined,
592
+ signal?: AbortSignal,
593
+ ): Promise<{ socket: WebSocketLike; release: (options?: { keep?: boolean }) => void }> {
594
+ if (!sessionId) {
595
+ const socket = await connectWebSocket(url, headers, signal);
596
+ return {
597
+ socket,
598
+ release: ({ keep } = {}) => {
599
+ if (keep === false) {
600
+ closeWebSocketSilently(socket);
601
+ return;
602
+ }
603
+ closeWebSocketSilently(socket);
604
+ },
605
+ };
606
+ }
607
+
608
+ const cached = websocketSessionCache.get(sessionId);
609
+ if (cached) {
610
+ if (cached.idleTimer) {
611
+ clearTimeout(cached.idleTimer);
612
+ cached.idleTimer = undefined;
613
+ }
614
+ if (!cached.busy && isWebSocketReusable(cached.socket)) {
615
+ cached.busy = true;
616
+ return {
617
+ socket: cached.socket,
618
+ release: ({ keep } = {}) => {
619
+ if (!keep || !isWebSocketReusable(cached.socket)) {
620
+ closeWebSocketSilently(cached.socket);
621
+ websocketSessionCache.delete(sessionId);
622
+ return;
623
+ }
624
+ cached.busy = false;
625
+ scheduleSessionWebSocketExpiry(sessionId, cached);
626
+ },
627
+ };
628
+ }
629
+ if (cached.busy) {
630
+ const socket = await connectWebSocket(url, headers, signal);
631
+ return {
632
+ socket,
633
+ release: () => {
634
+ closeWebSocketSilently(socket);
635
+ },
636
+ };
637
+ }
638
+ if (!isWebSocketReusable(cached.socket)) {
639
+ closeWebSocketSilently(cached.socket);
640
+ websocketSessionCache.delete(sessionId);
641
+ }
642
+ }
643
+
644
+ const socket = await connectWebSocket(url, headers, signal);
645
+ const entry: CachedWebSocketConnection = { socket, busy: true };
646
+ websocketSessionCache.set(sessionId, entry);
647
+ return {
648
+ socket,
649
+ release: ({ keep } = {}) => {
650
+ if (!keep || !isWebSocketReusable(entry.socket)) {
651
+ closeWebSocketSilently(entry.socket);
652
+ if (entry.idleTimer) clearTimeout(entry.idleTimer);
653
+ if (websocketSessionCache.get(sessionId) === entry) {
654
+ websocketSessionCache.delete(sessionId);
655
+ }
656
+ return;
657
+ }
658
+ entry.busy = false;
659
+ scheduleSessionWebSocketExpiry(sessionId, entry);
660
+ },
661
+ };
662
+ }
663
+
664
+ function extractWebSocketError(event: unknown): Error {
665
+ if (event && typeof event === "object" && "message" in event) {
666
+ const message = (event as { message?: unknown }).message;
667
+ if (typeof message === "string" && message.length > 0) {
668
+ return new Error(message);
669
+ }
670
+ }
671
+ return new Error("WebSocket error");
672
+ }
673
+
674
+ function extractWebSocketCloseError(event: unknown): Error {
675
+ if (event && typeof event === "object") {
676
+ const code = "code" in event ? (event as { code?: unknown }).code : undefined;
677
+ const reason = "reason" in event ? (event as { reason?: unknown }).reason : undefined;
678
+ const codeText = typeof code === "number" ? ` ${code}` : "";
679
+ const reasonText = typeof reason === "string" && reason.length > 0 ? ` ${reason}` : "";
680
+ return new Error(`WebSocket closed${codeText}${reasonText}`.trim());
681
+ }
682
+ return new Error("WebSocket closed");
683
+ }
684
+
685
+ async function decodeWebSocketData(data: unknown): Promise<string | null> {
686
+ if (typeof data === "string") return data;
687
+ if (data instanceof ArrayBuffer) {
688
+ return new TextDecoder().decode(new Uint8Array(data));
689
+ }
690
+ if (ArrayBuffer.isView(data)) {
691
+ const view = data as ArrayBufferView;
692
+ return new TextDecoder().decode(new Uint8Array(view.buffer, view.byteOffset, view.byteLength));
693
+ }
694
+ if (data && typeof data === "object" && "arrayBuffer" in data) {
695
+ const blobLike = data as { arrayBuffer: () => Promise<ArrayBuffer> };
696
+ const arrayBuffer = await blobLike.arrayBuffer();
697
+ return new TextDecoder().decode(new Uint8Array(arrayBuffer));
698
+ }
699
+ return null;
700
+ }
701
+
702
+ async function* parseWebSocket(socket: WebSocketLike, signal?: AbortSignal): AsyncGenerator<Record<string, unknown>> {
703
+ const queue: Record<string, unknown>[] = [];
704
+ let pending: (() => void) | null = null;
705
+ let done = false;
706
+ let failed: Error | null = null;
707
+ let sawCompletion = false;
708
+
709
+ const wake = () => {
710
+ if (!pending) return;
711
+ const resolve = pending;
712
+ pending = null;
713
+ resolve();
714
+ };
715
+
716
+ const onMessage: WebSocketListener = (event) => {
717
+ void (async () => {
718
+ if (!event || typeof event !== "object" || !("data" in event)) return;
719
+ const text = await decodeWebSocketData((event as { data?: unknown }).data);
720
+ if (!text) return;
721
+ try {
722
+ const parsed = JSON.parse(text) as Record<string, unknown>;
723
+ const type = typeof parsed.type === "string" ? parsed.type : "";
724
+ if (type === "response.completed" || type === "response.done" || type === "response.incomplete") {
725
+ sawCompletion = true;
726
+ done = true;
727
+ }
728
+ queue.push(parsed);
729
+ wake();
730
+ } catch {}
731
+ })();
732
+ };
733
+
734
+ const onError: WebSocketListener = (event) => {
735
+ failed = extractWebSocketError(event);
736
+ done = true;
737
+ wake();
738
+ };
739
+
740
+ const onClose: WebSocketListener = (event) => {
741
+ if (sawCompletion) {
742
+ done = true;
743
+ wake();
744
+ return;
745
+ }
746
+ if (!failed) {
747
+ failed = extractWebSocketCloseError(event);
748
+ }
749
+ done = true;
750
+ wake();
751
+ };
752
+
753
+ const onAbort = () => {
754
+ failed = new Error("Request was aborted");
755
+ done = true;
756
+ wake();
757
+ };
758
+
759
+ socket.addEventListener("message", onMessage);
760
+ socket.addEventListener("error", onError);
761
+ socket.addEventListener("close", onClose);
762
+ signal?.addEventListener("abort", onAbort);
763
+
764
+ try {
765
+ while (true) {
766
+ if (signal?.aborted) {
767
+ throw new Error("Request was aborted");
768
+ }
769
+ if (queue.length > 0) {
770
+ yield queue.shift()!;
771
+ continue;
772
+ }
773
+ if (done) break;
774
+ await new Promise<void>((resolve) => {
775
+ pending = resolve;
776
+ });
777
+ }
778
+
779
+ if (failed) {
780
+ throw failed;
781
+ }
782
+ if (!sawCompletion) {
783
+ throw new Error("WebSocket stream closed before response.completed");
784
+ }
785
+ } finally {
786
+ socket.removeEventListener("message", onMessage);
787
+ socket.removeEventListener("error", onError);
788
+ socket.removeEventListener("close", onClose);
789
+ signal?.removeEventListener("abort", onAbort);
790
+ }
791
+ }
792
+
793
+ async function processWebSocketStream(
794
+ url: string,
795
+ body: RequestBody,
796
+ headers: Headers,
797
+ output: AssistantMessage,
798
+ stream: AssistantMessageEventStream,
799
+ model: Model<"openai-codex-responses">,
800
+ onStart: () => void,
801
+ options?: OpenAICodexResponsesOptions,
802
+ ): Promise<void> {
803
+ const { socket, release } = await acquireWebSocket(url, headers, options?.sessionId, options?.signal);
804
+ let keepConnection = true;
805
+ try {
806
+ socket.send(JSON.stringify({ type: "response.create", ...body }));
807
+ onStart();
808
+ stream.push({ type: "start", partial: output });
809
+ await processResponsesStream(mapCodexEvents(parseWebSocket(socket, options?.signal)), output, stream, model);
810
+ if (options?.signal?.aborted) {
811
+ keepConnection = false;
812
+ }
813
+ } catch (error) {
814
+ keepConnection = false;
815
+ throw error;
816
+ } finally {
817
+ release({ keep: keepConnection });
818
+ }
819
+ }
820
+
821
+ // ============================================================================
822
+ // Error Handling
823
+ // ============================================================================
824
+
825
+ async function parseErrorResponse(response: Response): Promise<{ message: string; friendlyMessage?: string }> {
826
+ const raw = await response.text();
827
+ let message = raw || response.statusText || "Request failed";
828
+ let friendlyMessage: string | undefined;
829
+
830
+ try {
831
+ const parsed = JSON.parse(raw) as {
832
+ error?: { code?: string; type?: string; message?: string; plan_type?: string; resets_at?: number };
833
+ };
834
+ const err = parsed?.error;
835
+ if (err) {
836
+ const code = err.code || err.type || "";
837
+ if (/usage_limit_reached|usage_not_included|rate_limit_exceeded/i.test(code) || response.status === 429) {
838
+ const plan = err.plan_type ? ` (${err.plan_type.toLowerCase()} plan)` : "";
839
+ const mins = err.resets_at
840
+ ? Math.max(0, Math.round((err.resets_at * 1000 - Date.now()) / 60000))
841
+ : undefined;
842
+ const when = mins !== undefined ? ` Try again in ~${mins} min.` : "";
843
+ friendlyMessage = `You have hit your ChatGPT usage limit${plan}.${when}`.trim();
844
+ }
845
+ message = err.message || friendlyMessage || message;
846
+ }
847
+ } catch {}
848
+
849
+ return { message, friendlyMessage };
850
+ }
851
+
852
+ // ============================================================================
853
+ // Auth & Headers
854
+ // ============================================================================
855
+
856
+ function extractAccountId(token: string): string {
857
+ try {
858
+ const parts = token.split(".");
859
+ if (parts.length !== 3) throw new Error("Invalid token");
860
+ const payload = JSON.parse(atob(parts[1]));
861
+ const accountId = payload?.[JWT_CLAIM_PATH]?.chatgpt_account_id;
862
+ if (!accountId) throw new Error("No account ID in token");
863
+ return accountId;
864
+ } catch {
865
+ throw new Error("Failed to extract accountId from token");
866
+ }
867
+ }
868
+
869
+ function createCodexRequestId(): string {
870
+ if (typeof globalThis.crypto?.randomUUID === "function") {
871
+ return globalThis.crypto.randomUUID();
872
+ }
873
+ return `codex_${Date.now()}_${Math.random().toString(36).slice(2, 10)}`;
874
+ }
875
+
876
+ function buildBaseCodexHeaders(
877
+ initHeaders: Record<string, string> | undefined,
878
+ additionalHeaders: Record<string, string> | undefined,
879
+ accountId: string,
880
+ token: string,
881
+ ): Headers {
882
+ const headers = new Headers(initHeaders);
883
+ for (const [key, value] of Object.entries(additionalHeaders || {})) {
884
+ headers.set(key, value);
885
+ }
886
+ headers.set("Authorization", `Bearer ${token}`);
887
+ headers.set("chatgpt-account-id", accountId);
888
+ headers.set("originator", "pi");
889
+ const userAgent = _os ? `pi (${_os.platform()} ${_os.release()}; ${_os.arch()})` : "pi (browser)";
890
+ headers.set("User-Agent", userAgent);
891
+ return headers;
892
+ }
893
+
894
+ function buildSSEHeaders(
895
+ initHeaders: Record<string, string> | undefined,
896
+ additionalHeaders: Record<string, string> | undefined,
897
+ accountId: string,
898
+ token: string,
899
+ sessionId?: string,
900
+ ): Headers {
901
+ const headers = buildBaseCodexHeaders(initHeaders, additionalHeaders, accountId, token);
902
+ headers.set("OpenAI-Beta", "responses=experimental");
903
+ headers.set("accept", "text/event-stream");
904
+ headers.set("content-type", "application/json");
905
+
906
+ if (sessionId) {
907
+ headers.set("session_id", sessionId);
908
+ }
909
+
910
+ return headers;
911
+ }
912
+
913
+ function buildWebSocketHeaders(
914
+ initHeaders: Record<string, string> | undefined,
915
+ additionalHeaders: Record<string, string> | undefined,
916
+ accountId: string,
917
+ token: string,
918
+ requestId: string,
919
+ ): Headers {
920
+ const headers = buildBaseCodexHeaders(initHeaders, additionalHeaders, accountId, token);
921
+ headers.delete("accept");
922
+ headers.delete("content-type");
923
+ headers.delete("OpenAI-Beta");
924
+ headers.delete("openai-beta");
925
+ headers.set("OpenAI-Beta", OPENAI_BETA_RESPONSES_WEBSOCKETS);
926
+ headers.set("x-client-request-id", requestId);
927
+ headers.set("session_id", requestId);
928
+ return headers;
929
+ }