openai 6.34.0 → 6.35.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 (112) hide show
  1. package/CHANGELOG.md +44 -0
  2. package/core/EventEmitter.d.mts +11 -0
  3. package/core/EventEmitter.d.mts.map +1 -1
  4. package/core/EventEmitter.d.ts +11 -0
  5. package/core/EventEmitter.d.ts.map +1 -1
  6. package/core/EventEmitter.js +15 -1
  7. package/core/EventEmitter.js.map +1 -1
  8. package/core/EventEmitter.mjs +13 -0
  9. package/core/EventEmitter.mjs.map +1 -1
  10. package/internal/types.d.mts +6 -6
  11. package/internal/types.d.mts.map +1 -1
  12. package/internal/types.d.ts +6 -6
  13. package/internal/types.d.ts.map +1 -1
  14. package/internal/utils/env.js +2 -2
  15. package/internal/utils/env.js.map +1 -1
  16. package/internal/utils/env.mjs +2 -2
  17. package/internal/utils/env.mjs.map +1 -1
  18. package/internal/ws-adapter-browser.d.mts +34 -0
  19. package/internal/ws-adapter-browser.d.mts.map +1 -0
  20. package/internal/ws-adapter-browser.d.ts +34 -0
  21. package/internal/ws-adapter-browser.d.ts.map +1 -0
  22. package/internal/ws-adapter-browser.js +88 -0
  23. package/internal/ws-adapter-browser.js.map +1 -0
  24. package/internal/ws-adapter-browser.mjs +84 -0
  25. package/internal/ws-adapter-browser.mjs.map +1 -0
  26. package/internal/ws-adapter-node.d.mts +27 -0
  27. package/internal/ws-adapter-node.d.mts.map +1 -0
  28. package/internal/ws-adapter-node.d.ts +27 -0
  29. package/internal/ws-adapter-node.d.ts.map +1 -0
  30. package/internal/ws-adapter-node.js +90 -0
  31. package/internal/ws-adapter-node.js.map +1 -0
  32. package/internal/ws-adapter-node.mjs +86 -0
  33. package/internal/ws-adapter-node.mjs.map +1 -0
  34. package/internal/ws-adapter.d.mts +24 -0
  35. package/internal/ws-adapter.d.mts.map +1 -0
  36. package/internal/ws-adapter.d.ts +24 -0
  37. package/internal/ws-adapter.d.ts.map +1 -0
  38. package/internal/ws-adapter.js +11 -0
  39. package/internal/ws-adapter.js.map +1 -0
  40. package/internal/ws-adapter.mjs +8 -0
  41. package/internal/ws-adapter.mjs.map +1 -0
  42. package/internal/ws.d.mts +80 -0
  43. package/internal/ws.d.mts.map +1 -0
  44. package/internal/ws.d.ts +80 -0
  45. package/internal/ws.d.ts.map +1 -0
  46. package/internal/ws.js +153 -0
  47. package/internal/ws.js.map +1 -0
  48. package/internal/ws.mjs +147 -0
  49. package/internal/ws.mjs.map +1 -0
  50. package/package.json +13 -13
  51. package/resources/audio/speech.d.mts +2 -2
  52. package/resources/audio/speech.d.ts +2 -2
  53. package/resources/audio/speech.js +2 -2
  54. package/resources/audio/speech.mjs +2 -2
  55. package/resources/chat/completions/completions.d.mts +1 -1
  56. package/resources/chat/completions/completions.d.ts +1 -1
  57. package/resources/completions.d.mts +1 -1
  58. package/resources/completions.d.ts +1 -1
  59. package/resources/responses/index.d.mts +2 -0
  60. package/resources/responses/index.d.mts.map +1 -1
  61. package/resources/responses/index.d.ts +2 -0
  62. package/resources/responses/index.d.ts.map +1 -1
  63. package/resources/responses/internal-base.d.mts +24 -2
  64. package/resources/responses/internal-base.d.mts.map +1 -1
  65. package/resources/responses/internal-base.d.ts +24 -2
  66. package/resources/responses/internal-base.d.ts.map +1 -1
  67. package/resources/responses/internal-base.js +5 -9
  68. package/resources/responses/internal-base.js.map +1 -1
  69. package/resources/responses/internal-base.mjs +5 -9
  70. package/resources/responses/internal-base.mjs.map +1 -1
  71. package/resources/responses/responses.d.mts +19 -3
  72. package/resources/responses/responses.d.mts.map +1 -1
  73. package/resources/responses/responses.d.ts +19 -3
  74. package/resources/responses/responses.d.ts.map +1 -1
  75. package/resources/responses/responses.js.map +1 -1
  76. package/resources/responses/responses.mjs.map +1 -1
  77. package/resources/responses/ws-base.d.mts +106 -0
  78. package/resources/responses/ws-base.d.mts.map +1 -0
  79. package/resources/responses/ws-base.d.ts +106 -0
  80. package/resources/responses/ws-base.d.ts.map +1 -0
  81. package/resources/responses/ws-base.js +474 -0
  82. package/resources/responses/ws-base.js.map +1 -0
  83. package/resources/responses/ws-base.mjs +470 -0
  84. package/resources/responses/ws-base.mjs.map +1 -0
  85. package/resources/responses/ws.d.mts +9 -38
  86. package/resources/responses/ws.d.mts.map +1 -1
  87. package/resources/responses/ws.d.ts +9 -38
  88. package/resources/responses/ws.d.ts.map +1 -1
  89. package/resources/responses/ws.js +17 -171
  90. package/resources/responses/ws.js.map +1 -1
  91. package/resources/responses/ws.mjs +17 -171
  92. package/resources/responses/ws.mjs.map +1 -1
  93. package/src/core/EventEmitter.ts +16 -0
  94. package/src/internal/types.ts +6 -8
  95. package/src/internal/utils/env.ts +2 -2
  96. package/src/internal/ws-adapter-browser.ts +123 -0
  97. package/src/internal/ws-adapter-node.ts +105 -0
  98. package/src/internal/ws-adapter.ts +30 -0
  99. package/src/internal/ws.ts +193 -0
  100. package/src/resources/audio/speech.ts +2 -2
  101. package/src/resources/chat/completions/completions.ts +1 -1
  102. package/src/resources/completions.ts +1 -1
  103. package/src/resources/responses/index.ts +2 -0
  104. package/src/resources/responses/internal-base.ts +26 -10
  105. package/src/resources/responses/responses.ts +22 -3
  106. package/src/resources/responses/ws-base.ts +609 -0
  107. package/src/resources/responses/ws.ts +23 -186
  108. package/src/version.ts +1 -1
  109. package/version.d.mts +1 -1
  110. package/version.d.ts +1 -1
  111. package/version.js +1 -1
  112. package/version.mjs +1 -1
@@ -0,0 +1,609 @@
1
+ // File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
2
+
3
+ import { ResponsesEmitter, ResponsesStreamMessage, WebSocketError, buildURL } from './internal-base';
4
+ import { InternalEventEmitter } from '../../core/EventEmitter';
5
+ import { sleep } from '../../internal/utils/sleep';
6
+ import { type WebSocketLike, ReadyState } from '../../internal/ws-adapter';
7
+ import {
8
+ SendQueue,
9
+ flattenRawData,
10
+ isRecoverableClose,
11
+ type RawWebSocketData,
12
+ type ReconnectingEvent,
13
+ type ReconnectingOverrides,
14
+ type UnsentMessage,
15
+ } from '../../internal/ws';
16
+ import * as ResponsesAPI from './responses';
17
+ import { OpenAI } from '../../client';
18
+ import { OpenAIError } from '../../core/error';
19
+
20
+ export interface ResponsesWSReconnectOptions {
21
+ /**
22
+ * Called before each reconnect attempt. Return an object with
23
+ * `parameters` to override query parameters for the next connection.
24
+ */
25
+ onReconnecting(
26
+ event: ReconnectingEvent<Record<string, unknown>>,
27
+ ): ReconnectingOverrides<Record<string, unknown>> | void;
28
+
29
+ /**
30
+ * Maximum number of reconnection attempts. Default: 5.
31
+ * Set to 0 to disable reconnection entirely.
32
+ */
33
+ maxRetries?: number;
34
+
35
+ /**
36
+ * Initial backoff delay in milliseconds. Default: 500.
37
+ */
38
+ initialDelay?: number;
39
+
40
+ /**
41
+ * Maximum backoff delay in milliseconds. Default: 8000.
42
+ */
43
+ maxDelay?: number;
44
+ }
45
+
46
+ export interface ResponsesWSBaseOptions {
47
+ /**
48
+ * Options for automatic reconnection on recoverable close codes.
49
+ * Automatic reconnection is only enabled when this has a non-null value.
50
+ */
51
+ reconnect?: ResponsesWSReconnectOptions | null | undefined;
52
+
53
+ /**
54
+ * Maximum size of the outgoing message queue in bytes.
55
+ * Messages queued while the socket is connecting or reconnecting are held
56
+ * in memory up to this limit. Once the limit is reached, new messages are
57
+ * discarded and an `error` event is emitted.
58
+ * Default: 1 MB
59
+ */
60
+ maxQueueSize?: number | undefined;
61
+ }
62
+
63
+ export abstract class ResponsesWSBase<TSocket extends WebSocketLike> extends ResponsesEmitter {
64
+ url!: URL;
65
+ socket!: TSocket;
66
+
67
+ protected _client: OpenAI;
68
+ protected _parameters: Record<string, unknown> | null | undefined;
69
+ private _reconnectOptions: ResponsesWSReconnectOptions | null;
70
+ private _sendQueue: SendQueue<ResponsesAPI.ResponsesClientEvent>;
71
+ private _isReconnecting: boolean = false;
72
+ private _intentionallyClosed = false;
73
+ private _closeCode: number = 1000;
74
+ private _closeReason: string = 'OK';
75
+ private _lastCloseCode: number = 1006;
76
+ private _lastCloseReason: string = '';
77
+
78
+ // Necessary to keep the public event interface clean while we manage reconnecting
79
+ private _internalEvents = new InternalEventEmitter<{
80
+ socketSwap: (oldSocket: TSocket, newSocket: TSocket) => void;
81
+ reconnecting: (event: ReconnectingEvent<Record<string, unknown>>) => void;
82
+ reconnected: () => void;
83
+ close: (code: number, reason: string, unsent: UnsentMessage<ResponsesAPI.ResponsesClientEvent>[]) => void;
84
+ }>();
85
+
86
+ constructor(client: OpenAI, options?: ResponsesWSBaseOptions | undefined) {
87
+ super();
88
+ this._client = client;
89
+ this._parameters = undefined;
90
+ this._reconnectOptions = options?.reconnect ?? null;
91
+ this._sendQueue = new SendQueue<ResponsesAPI.ResponsesClientEvent>(options?.maxQueueSize);
92
+ }
93
+
94
+ /** Establishes the initial WebSocket connection. */
95
+ protected _connectInitial(): void {
96
+ this.url = buildURL(this._client, {});
97
+ this.socket = this._connect();
98
+ }
99
+
100
+ /** Creates a platform-specific WebSocket for the given URL and auth headers. */
101
+ protected abstract _createSocket(url: URL, authHeaders: Record<string, string>): TSocket;
102
+
103
+ send(event: ResponsesAPI.ResponsesClientEvent) {
104
+ if (!this.socket) {
105
+ throw new OpenAIError('Internal error: failed to initialize socket. Please report this issue.');
106
+ }
107
+
108
+ if (this._isReconnecting || this.socket.readyState === ReadyState.CONNECTING) {
109
+ if (!this._sendQueue.enqueue(event)) {
110
+ this._onError(null, 'send queue is full, message discarded', undefined);
111
+ }
112
+ return;
113
+ }
114
+ if (this.socket.readyState !== ReadyState.OPEN) {
115
+ this._onError(null, 'cannot send on a closed WebSocket', undefined);
116
+ return;
117
+ }
118
+ try {
119
+ this.socket.send(JSON.stringify(event));
120
+ } catch (err) {
121
+ this._onError(null, 'could not send data', err);
122
+ }
123
+ }
124
+
125
+ sendRaw(data: RawWebSocketData) {
126
+ if (!this.socket) {
127
+ throw new OpenAIError('Internal error: failed to initialize socket. Please report this issue.');
128
+ }
129
+
130
+ if (this._isReconnecting || this.socket.readyState === ReadyState.CONNECTING) {
131
+ if (!this._sendQueue.enqueueRaw(data)) {
132
+ this._onError(null, 'send queue is full, message discarded', undefined);
133
+ }
134
+ return;
135
+ }
136
+ if (this.socket.readyState !== ReadyState.OPEN) {
137
+ this._onError(null, 'cannot send on a closed WebSocket', undefined);
138
+ return;
139
+ }
140
+ try {
141
+ this.socket.send(flattenRawData(data));
142
+ } catch (err) {
143
+ this._onError(null, 'could not send data', err);
144
+ }
145
+ }
146
+
147
+ close(props?: { code: number; reason: string }) {
148
+ if (!this.socket) {
149
+ throw new OpenAIError('Internal error: failed to initialize socket. Please report this issue.');
150
+ }
151
+
152
+ this._intentionallyClosed = true;
153
+ this._closeCode = props?.code ?? 1000;
154
+ this._closeReason = props?.reason ?? 'OK';
155
+ try {
156
+ this.socket.close(this._closeCode, this._closeReason);
157
+ } catch (err) {
158
+ this._onError(null, 'could not close the connection', err);
159
+ }
160
+ }
161
+
162
+ /**
163
+ * Returns an async iterator over WebSocket lifecycle and message events,
164
+ * providing an alternative to the event-based `.on()` API.
165
+ * The iterator will exit if the socket closes but exiting the iterator
166
+ * does not close the socket.
167
+ *
168
+ * @example
169
+ * ```ts
170
+ * for await (const event of client.stream()) {
171
+ * switch (event.type) {
172
+ * case 'message':
173
+ * console.log('received:', event.message);
174
+ * break;
175
+ * case 'error':
176
+ * console.error(event.error);
177
+ * break;
178
+ * case 'close':
179
+ * console.log('connection closed');
180
+ * break;
181
+ * }
182
+ * }
183
+ * ```
184
+ */
185
+ stream(): AsyncIterableIterator<ResponsesStreamMessage> {
186
+ return this[Symbol.asyncIterator]();
187
+ }
188
+
189
+ [Symbol.asyncIterator](): AsyncIterableIterator<ResponsesStreamMessage> {
190
+ if (!this.socket) {
191
+ throw new OpenAIError('Internal error: failed to initialize socket. Please report this issue.');
192
+ }
193
+
194
+ // Two-queue async iterator: `queue` buffers incoming messages,
195
+ // `resolvers` buffers waiting next() calls. A push wakes the
196
+ // oldest next(); a next() drains the oldest message.
197
+ const queue: ResponsesStreamMessage[] = [];
198
+ const resolvers: (() => void)[] = [];
199
+ let done = false;
200
+ let currentSocket = this.socket;
201
+
202
+ const push = (msg: ResponsesStreamMessage) => {
203
+ queue.push(msg);
204
+ resolvers.shift()?.();
205
+ };
206
+
207
+ const onEvent = (event: ResponsesAPI.ResponsesServerEvent) => {
208
+ if (event.type === 'error') return; // handled by onEmitterError
209
+ push({ type: 'message', message: event });
210
+ };
211
+
212
+ const onRaw = (data: RawWebSocketData) => {
213
+ push({ type: 'raw', data });
214
+ };
215
+
216
+ // All errors (API + socket) funnel through _onError → 'error' event
217
+ const onEmitterError = (err: WebSocketError) => {
218
+ push({ type: 'error', error: err });
219
+ };
220
+
221
+ const onOpen = () => {
222
+ push({ type: 'open' });
223
+ };
224
+
225
+ const onReconnecting = (evt: ReconnectingEvent<Record<string, unknown>>) => {
226
+ push({ type: 'reconnecting', reconnect: evt });
227
+ };
228
+
229
+ const onReconnected = () => {
230
+ push({ type: 'reconnected' });
231
+ };
232
+
233
+ const flushResolvers = () => {
234
+ for (let resolver = resolvers.shift(); resolver; resolver = resolvers.shift()) {
235
+ resolver();
236
+ }
237
+ };
238
+
239
+ const onClose = (
240
+ code: number,
241
+ reason: string,
242
+ unsent: UnsentMessage<ResponsesAPI.ResponsesClientEvent>[],
243
+ ) => {
244
+ push({ type: 'close', code, reason, unsent });
245
+ done = true;
246
+ flushResolvers();
247
+ cleanup();
248
+ };
249
+
250
+ const onSocketSwap = (oldSocket: TSocket, newSocket: TSocket) => {
251
+ oldSocket.off('open', onOpen);
252
+ newSocket.on('open', onOpen);
253
+ currentSocket = newSocket;
254
+ };
255
+
256
+ const cleanup = () => {
257
+ this.off('event', onEvent);
258
+ this.off('raw', onRaw);
259
+ this.off('error', onEmitterError);
260
+ currentSocket.off('open', onOpen);
261
+ this._internalEvents.off('close', onClose);
262
+ this._internalEvents.off('socketSwap', onSocketSwap);
263
+ this._internalEvents.off('reconnecting', onReconnecting);
264
+ this._internalEvents.off('reconnected', onReconnected);
265
+ };
266
+
267
+ this.on('event', onEvent);
268
+ this.on('raw', onRaw);
269
+ this.on('error', onEmitterError);
270
+ this.socket.on('open', onOpen);
271
+ this._internalEvents.on('close', onClose);
272
+ this._internalEvents.on('socketSwap', onSocketSwap);
273
+ this._internalEvents.on('reconnecting', onReconnecting);
274
+ this._internalEvents.on('reconnected', onReconnected);
275
+
276
+ if (this._isReconnecting) {
277
+ // A reconnect is already in flight. The socket may be CLOSED but the
278
+ // instance is still alive. Emit 'reconnecting' so the iterator stays
279
+ // open and receives the upcoming reconnected/message events.
280
+ push({
281
+ type: 'reconnecting',
282
+ reconnect: { attempt: 0, maxAttempts: 0, delay: 0, closeCode: 0, parameters: undefined },
283
+ });
284
+ } else {
285
+ switch (this.socket.readyState) {
286
+ case ReadyState.CONNECTING:
287
+ push({ type: 'connecting' });
288
+ break;
289
+ case ReadyState.OPEN:
290
+ push({ type: 'open' });
291
+ break;
292
+ case ReadyState.CLOSING:
293
+ push({ type: 'closing' });
294
+ break;
295
+ case ReadyState.CLOSED:
296
+ push({
297
+ type: 'close',
298
+ code: this._lastCloseCode,
299
+ reason: this._lastCloseReason,
300
+ unsent: this._sendQueue.drain(),
301
+ });
302
+ done = true;
303
+ cleanup();
304
+ break;
305
+ }
306
+ }
307
+
308
+ const resolve = (res: (value: IteratorResult<ResponsesStreamMessage>) => void) => {
309
+ if (queue.length > 0) {
310
+ res({ value: queue.shift()!, done: false });
311
+ } else if (done) {
312
+ res({ value: undefined, done: true });
313
+ } else {
314
+ return false;
315
+ }
316
+ return true;
317
+ };
318
+
319
+ const next = (): Promise<IteratorResult<ResponsesStreamMessage>> =>
320
+ new Promise((res) => {
321
+ if (resolve(res)) return;
322
+ resolvers.push(() => {
323
+ resolve(res);
324
+ });
325
+ });
326
+
327
+ return {
328
+ next,
329
+ return: (): Promise<IteratorReturnResult<undefined>> => {
330
+ done = true;
331
+ cleanup();
332
+ flushResolvers();
333
+ return Promise.resolve({ value: undefined, done: true });
334
+ },
335
+ [Symbol.asyncIterator]() {
336
+ return this;
337
+ },
338
+ };
339
+ }
340
+
341
+ private _connect(): TSocket {
342
+ this.url = buildURL(this._client, this._parameters ?? {});
343
+
344
+ const socket = this._createSocket(this.url, this._authHeaders());
345
+
346
+ socket.on('message', (data: string | ArrayBuffer | ArrayBufferView, isBinary: boolean) => {
347
+ if (isBinary) {
348
+ this._emit('raw', data);
349
+ return;
350
+ }
351
+
352
+ // Coerce to string in case the adapter delivers a typed-array for text frames.
353
+ const text = typeof data === 'string' ? data : String(data);
354
+
355
+ let event: ResponsesAPI.ResponsesServerEvent;
356
+ try {
357
+ event = JSON.parse(text) as ResponsesAPI.ResponsesServerEvent;
358
+ } catch {
359
+ this._emit('raw', data);
360
+ return;
361
+ }
362
+
363
+ this._emit('event', event);
364
+
365
+ if (event.type === 'error') {
366
+ this._onError(event);
367
+ } else {
368
+ // @ts-ignore TS isn't smart enough to get the relationship right here
369
+ this._emit(event.type, event);
370
+ }
371
+ });
372
+
373
+ socket.on('error', (err: Error) => {
374
+ // Suppress transient errors during reconnection — the retry loop
375
+ // already handles them and will surface a close if retries exhaust.
376
+ if (this._isReconnecting) return;
377
+ this._onError(null, err.message, err);
378
+ });
379
+
380
+ socket.on('open', () => {
381
+ this._flushSendQueue();
382
+ });
383
+
384
+ socket.on('close', (code: number, reason: string) => {
385
+ // Ignore close events from superseded sockets — a stale socket's
386
+ // late close must not kick off a second reconnect loop.
387
+ if (socket !== this.socket) return;
388
+ if (!this._intentionallyClosed && this._canReconnect(code)) {
389
+ this._reconnect(code);
390
+ } else if (!this._isReconnecting) {
391
+ this._emitPermanentClose(code, reason);
392
+ }
393
+ });
394
+
395
+ return socket;
396
+ }
397
+
398
+ // Reconnect is opt-in via onReconnecting so callers can pass
399
+ // state (e.g. session IDs) into the new connection.
400
+ private _canReconnect(code: number): boolean {
401
+ if (this._intentionallyClosed) return false;
402
+ if (!this._reconnectOptions) return false;
403
+ if (this._reconnectOptions.maxRetries === 0) return false;
404
+ if (!this._reconnectOptions.onReconnecting) return false;
405
+ return isRecoverableClose(code);
406
+ }
407
+
408
+ private async _reconnect(closeCode: number): Promise<void> {
409
+ if (!this.socket) {
410
+ throw new OpenAIError('Internal error: failed to initialize socket. Please report this issue.');
411
+ }
412
+
413
+ if (this._isReconnecting || !this._reconnectOptions) return;
414
+ this._isReconnecting = true;
415
+
416
+ const maxRetries = this._reconnectOptions.maxRetries ?? 5;
417
+ const initialDelay = this._reconnectOptions.initialDelay ?? 500;
418
+ const maxDelay = this._reconnectOptions.maxDelay ?? 8000;
419
+
420
+ for (let attempt = 1; attempt <= maxRetries; attempt++) {
421
+ if (!this._canReconnect(closeCode)) {
422
+ this._isReconnecting = false;
423
+ if (!this._intentionallyClosed) {
424
+ this._onError(
425
+ null,
426
+ `WebSocket reconnect aborted: non-recoverable close code ${closeCode}`,
427
+ undefined,
428
+ );
429
+ }
430
+ this._emitPermanentClose(
431
+ this._intentionallyClosed ? this._closeCode : closeCode,
432
+ this._intentionallyClosed ? this._closeReason : 'reconnect aborted',
433
+ );
434
+ return;
435
+ }
436
+
437
+ const baseDelay = Math.min(initialDelay * Math.pow(2, attempt - 1), maxDelay);
438
+ // Jitter: rand [0.75, 1.0] to spread out connection attempts without over-delaying
439
+ const jitter = 0.75 + Math.random() * 0.25;
440
+ const actualDelay = Math.round(baseDelay * jitter);
441
+
442
+ let reconnectingEvent: ReconnectingEvent<Record<string, unknown>> = {
443
+ attempt,
444
+ maxAttempts: maxRetries,
445
+ delay: actualDelay,
446
+ closeCode,
447
+ parameters: this._parameters ? { ...this._parameters } : undefined,
448
+ };
449
+
450
+ let overrides: ReconnectingOverrides<Record<string, unknown>> | void;
451
+ try {
452
+ overrides = this._reconnectOptions.onReconnecting(reconnectingEvent);
453
+ } catch (err) {
454
+ this._isReconnecting = false;
455
+ this._onError(null, 'onReconnecting callback threw', err);
456
+ this._emitPermanentClose(closeCode, 'onReconnecting callback threw');
457
+ return;
458
+ }
459
+
460
+ if (overrides && 'abort' in overrides && overrides.abort) {
461
+ this._isReconnecting = false;
462
+ this._emitPermanentClose(closeCode, 'reconnect aborted by handler');
463
+ return;
464
+ }
465
+
466
+ if (overrides && 'parameters' in overrides) {
467
+ this._parameters = overrides.parameters;
468
+ reconnectingEvent = { ...reconnectingEvent, parameters: this._parameters };
469
+ }
470
+
471
+ try {
472
+ this._emit('reconnecting', reconnectingEvent);
473
+ } catch (err) {
474
+ this._onError(null, 'onReconnecting callback threw', err);
475
+ }
476
+ this._internalEvents._emit('reconnecting', reconnectingEvent);
477
+
478
+ if (!this._canReconnect(closeCode)) {
479
+ this._isReconnecting = false;
480
+ if (!this._intentionallyClosed) {
481
+ this._onError(
482
+ null,
483
+ `WebSocket reconnect aborted: non-recoverable close code ${closeCode}`,
484
+ undefined,
485
+ );
486
+ }
487
+ this._emitPermanentClose(
488
+ this._intentionallyClosed ? this._closeCode : closeCode,
489
+ this._intentionallyClosed ? this._closeReason : 'reconnect aborted',
490
+ );
491
+ return;
492
+ }
493
+
494
+ await sleep(actualDelay);
495
+
496
+ if (!this._canReconnect(closeCode)) {
497
+ this._isReconnecting = false;
498
+ if (!this._intentionallyClosed) {
499
+ this._onError(
500
+ null,
501
+ `WebSocket reconnect aborted: non-recoverable close code ${closeCode}`,
502
+ undefined,
503
+ );
504
+ }
505
+ this._emitPermanentClose(
506
+ this._intentionallyClosed ? this._closeCode : closeCode,
507
+ this._intentionallyClosed ? this._closeReason : 'reconnect aborted',
508
+ );
509
+ return;
510
+ }
511
+
512
+ let closeCodePromise: Promise<number> | undefined;
513
+ try {
514
+ const oldSocket = this.socket;
515
+ this.socket = this._connect();
516
+ // Registered synchronously after _connect() and before any
517
+ // await so the code is captured even when ws emits 'close'
518
+ // in the same tick as 'error' (e.g. abortHandshake).
519
+ closeCodePromise = new Promise<number>((resolve) => {
520
+ this.socket.once('close', resolve);
521
+ });
522
+
523
+ await this._awaitOpen(this.socket);
524
+
525
+ this._internalEvents._emit('socketSwap', oldSocket, this.socket);
526
+ this._isReconnecting = false;
527
+ this._flushSendQueue();
528
+ this._emit('reconnected');
529
+ this._internalEvents._emit('reconnected');
530
+ return;
531
+ } catch {
532
+ if (closeCodePromise) {
533
+ // ws may emit 'error' before 'close', so await the code
534
+ // rather than reading it synchronously.
535
+ closeCode = await closeCodePromise;
536
+ }
537
+ }
538
+ }
539
+
540
+ // All retries exhausted — surface an error so consumers can
541
+ // distinguish retry failure from a clean close.
542
+ this._isReconnecting = false;
543
+ this._onError(
544
+ null,
545
+ `WebSocket reconnect failed after ${maxRetries} attempts (close code: ${closeCode})`,
546
+ undefined,
547
+ );
548
+ this._emitPermanentClose(closeCode, `reconnect failed after ${maxRetries} attempts`);
549
+ }
550
+
551
+ /**
552
+ * Resolves once the socket is open, rejects if it errors or closes first
553
+ */
554
+ private _awaitOpen(socket: TSocket): Promise<void> {
555
+ return new Promise<void>((resolve, reject) => {
556
+ const cleanup = () => {
557
+ socket.off('open', onOpen);
558
+ socket.off('error', onError);
559
+ socket.off('close', onFail);
560
+ };
561
+ const onOpen = () => {
562
+ cleanup();
563
+ resolve();
564
+ };
565
+ const onError = (err: Error) => {
566
+ cleanup();
567
+ reject(err);
568
+ };
569
+ const onFail = () => {
570
+ cleanup();
571
+ reject(new Error('socket closed before open'));
572
+ };
573
+ socket.once('open', onOpen);
574
+ socket.once('error', onError);
575
+ socket.once('close', onFail);
576
+ });
577
+ }
578
+
579
+ private _flushSendQueue(): void {
580
+ if (!this.socket) {
581
+ throw new OpenAIError('Internal error: failed to initialize socket. Please report this issue.');
582
+ }
583
+
584
+ try {
585
+ this._sendQueue.flush((data) => this.socket.send(flattenRawData(data)));
586
+ } catch (err) {
587
+ this._onError(null, 'could not send queued data', err);
588
+ }
589
+ }
590
+
591
+ /**
592
+ * Emits the public `close` event with unsent messages and the internal
593
+ * `close` event used by the async iterator.
594
+ */
595
+ private _emitPermanentClose(code: number, reason: string): void {
596
+ this._lastCloseCode = code;
597
+ this._lastCloseReason = reason;
598
+ const unsent = this._sendQueue.drain();
599
+ // Internal close fires first so the async iterator is guaranteed to
600
+ // terminate even if a public 'close' listener throws.
601
+ this._internalEvents._emit('close', code, reason, unsent);
602
+ this._emit('close', code, reason, unsent);
603
+ }
604
+
605
+ protected _authHeaders(): Record<string, string> {
606
+ return { Authorization: `Bearer ${this._client.apiKey}` };
607
+ return {};
608
+ }
609
+ }