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