@langchain/langgraph-sdk 1.9.20 → 1.9.21

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 (90) hide show
  1. package/dist/client/index.cjs +1 -1
  2. package/dist/client/index.js +1 -1
  3. package/dist/client/stream/error.cjs +21 -0
  4. package/dist/client/stream/error.cjs.map +1 -1
  5. package/dist/client/stream/error.js +21 -1
  6. package/dist/client/stream/error.js.map +1 -1
  7. package/dist/client/stream/index.cjs +24 -1
  8. package/dist/client/stream/index.cjs.map +1 -1
  9. package/dist/client/stream/index.d.cts.map +1 -1
  10. package/dist/client/stream/index.d.ts.map +1 -1
  11. package/dist/client/stream/index.js +24 -1
  12. package/dist/client/stream/index.js.map +1 -1
  13. package/dist/client/stream/transport/agent-server.cjs +2 -1
  14. package/dist/client/stream/transport/agent-server.cjs.map +1 -1
  15. package/dist/client/stream/transport/agent-server.d.cts +7 -0
  16. package/dist/client/stream/transport/agent-server.d.cts.map +1 -1
  17. package/dist/client/stream/transport/agent-server.d.ts +7 -0
  18. package/dist/client/stream/transport/agent-server.d.ts.map +1 -1
  19. package/dist/client/stream/transport/agent-server.js +2 -1
  20. package/dist/client/stream/transport/agent-server.js.map +1 -1
  21. package/dist/client/stream/transport/http.cjs +48 -9
  22. package/dist/client/stream/transport/http.cjs.map +1 -1
  23. package/dist/client/stream/transport/http.d.cts +4 -0
  24. package/dist/client/stream/transport/http.d.cts.map +1 -1
  25. package/dist/client/stream/transport/http.d.ts +4 -0
  26. package/dist/client/stream/transport/http.d.ts.map +1 -1
  27. package/dist/client/stream/transport/http.js +48 -9
  28. package/dist/client/stream/transport/http.js.map +1 -1
  29. package/dist/client/stream/transport/index.cjs +2 -1
  30. package/dist/client/stream/transport/index.js +2 -1
  31. package/dist/client/stream/transport/types.d.cts +44 -0
  32. package/dist/client/stream/transport/types.d.cts.map +1 -1
  33. package/dist/client/stream/transport/types.d.ts +44 -0
  34. package/dist/client/stream/transport/types.d.ts.map +1 -1
  35. package/dist/client/stream/transport/websocket.cjs +105 -16
  36. package/dist/client/stream/transport/websocket.cjs.map +1 -1
  37. package/dist/client/stream/transport/websocket.d.cts +19 -0
  38. package/dist/client/stream/transport/websocket.d.cts.map +1 -1
  39. package/dist/client/stream/transport/websocket.d.ts +19 -0
  40. package/dist/client/stream/transport/websocket.d.ts.map +1 -1
  41. package/dist/client/stream/transport/websocket.js +105 -17
  42. package/dist/client/stream/transport/websocket.js.map +1 -1
  43. package/dist/client/stream/types.d.cts +17 -0
  44. package/dist/client/stream/types.d.cts.map +1 -1
  45. package/dist/client/stream/types.d.ts +17 -0
  46. package/dist/client/stream/types.d.ts.map +1 -1
  47. package/dist/client/threads/index.cjs +30 -14
  48. package/dist/client/threads/index.cjs.map +1 -1
  49. package/dist/client/threads/index.js +30 -14
  50. package/dist/client/threads/index.js.map +1 -1
  51. package/dist/client.cjs +1 -1
  52. package/dist/client.js +1 -1
  53. package/dist/index.cjs +1 -1
  54. package/dist/index.js +1 -1
  55. package/dist/stream/controller.cjs +19 -1
  56. package/dist/stream/controller.cjs.map +1 -1
  57. package/dist/stream/controller.d.cts.map +1 -1
  58. package/dist/stream/controller.d.ts.map +1 -1
  59. package/dist/stream/controller.js +19 -1
  60. package/dist/stream/controller.js.map +1 -1
  61. package/dist/stream/index.cjs +2 -0
  62. package/dist/stream/index.d.cts +2 -1
  63. package/dist/stream/index.d.ts +2 -1
  64. package/dist/stream/index.js +2 -1
  65. package/dist/stream/projections/channel-effect.cjs +52 -0
  66. package/dist/stream/projections/channel-effect.cjs.map +1 -0
  67. package/dist/stream/projections/channel-effect.d.cts +35 -0
  68. package/dist/stream/projections/channel-effect.d.cts.map +1 -0
  69. package/dist/stream/projections/channel-effect.d.ts +35 -0
  70. package/dist/stream/projections/channel-effect.d.ts.map +1 -0
  71. package/dist/stream/projections/channel-effect.js +52 -0
  72. package/dist/stream/projections/channel-effect.js.map +1 -0
  73. package/dist/stream/projections/index.cjs +1 -0
  74. package/dist/stream/projections/index.d.ts +1 -0
  75. package/dist/stream/projections/index.js +1 -0
  76. package/dist/stream/root-message-projection.cjs +55 -0
  77. package/dist/stream/root-message-projection.cjs.map +1 -1
  78. package/dist/stream/root-message-projection.js +55 -0
  79. package/dist/stream/root-message-projection.js.map +1 -1
  80. package/dist/ui/branching.d.cts +1 -1
  81. package/dist/ui/branching.d.ts +1 -1
  82. package/dist/ui/orchestrator.d.cts +1 -1
  83. package/dist/ui/orchestrator.d.cts.map +1 -1
  84. package/dist/ui/orchestrator.d.ts +1 -1
  85. package/dist/ui/orchestrator.d.ts.map +1 -1
  86. package/dist/utils/stream.d.cts +1 -1
  87. package/dist/utils/stream.d.cts.map +1 -1
  88. package/dist/utils/stream.d.ts +1 -1
  89. package/dist/utils/stream.d.ts.map +1 -1
  90. package/package.json +5 -2
@@ -1,3 +1,4 @@
1
+ import { AsyncCaller } from "../../../utils/async_caller.cjs";
1
2
  import { CommandResponse, ErrorResponse } from "@langchain/protocol";
2
3
 
3
4
  //#region src/client/stream/transport/types.d.ts
@@ -15,7 +16,28 @@ interface ProtocolSseTransportOptions {
15
16
  onRequest?: ProtocolRequestHook;
16
17
  fetch?: typeof fetch;
17
18
  fetchFactory?: () => typeof fetch | Promise<typeof fetch>;
19
+ /**
20
+ * When set, command and SSE subscription HTTP requests are executed
21
+ * through {@link AsyncCaller} (retries, concurrency). Typically wired
22
+ * from {@link BaseClient} via `client.threads.stream()`.
23
+ */
24
+ asyncCaller?: AsyncCaller;
18
25
  paths?: ProtocolTransportPaths;
26
+ /**
27
+ * Maximum reconnect attempts after an unexpected SSE disconnect.
28
+ * Defaults to 5. Set to 0 to disable automatic reconnection.
29
+ */
30
+ maxReconnectAttempts?: number;
31
+ /** Called before each SSE reconnect attempt (after backoff delay). */
32
+ onReconnect?: (options: {
33
+ attempt: number;
34
+ cause: unknown;
35
+ }) => void;
36
+ /**
37
+ * Backoff before each SSE reconnect attempt. Defaults to
38
+ * {@link webSocketReconnectDelayMs} from `./websocket.js`.
39
+ */
40
+ reconnectDelayMs?: (attempt: number) => number;
19
41
  }
20
42
  interface ProtocolWebSocketTransportOptions {
21
43
  apiUrl: string;
@@ -24,6 +46,28 @@ interface ProtocolWebSocketTransportOptions {
24
46
  onRequest?: ProtocolRequestHook;
25
47
  webSocketFactory?: (url: string) => WebSocket;
26
48
  paths?: Pick<ProtocolTransportPaths, "stream">;
49
+ /**
50
+ * Maximum reconnect attempts after an unexpected socket close.
51
+ * Defaults to 5. Set to 0 to disable automatic reconnection.
52
+ */
53
+ maxReconnectAttempts?: number;
54
+ /**
55
+ * Called before each reconnect attempt (after backoff delay).
56
+ */
57
+ onReconnect?: (options: {
58
+ attempt: number;
59
+ cause: unknown;
60
+ }) => void;
61
+ /**
62
+ * Invoked after the socket has been re-established. Use to restore
63
+ * server-side subscription state (see `ThreadStream`).
64
+ */
65
+ onReconnected?: () => void | Promise<void>;
66
+ /**
67
+ * Backoff before each reconnect attempt. Defaults to
68
+ * {@link webSocketReconnectDelayMs}.
69
+ */
70
+ reconnectDelayMs?: (attempt: number) => number;
27
71
  }
28
72
  type HeaderValue = string | undefined | null;
29
73
  //#endregion
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.cts","names":[],"sources":["../../../../src/client/stream/transport/types.ts"],"mappings":";;;KAEY,mBAAA,IACV,GAAA,EAAK,GAAA,EACL,IAAA,EAAM,WAAA,KACH,OAAA,CAAQ,WAAA,IAAe,WAAA;AAAA,UAEX,sBAAA;EACf,QAAA;EACA,MAAA;;EAEA,KAAA;AAAA;AAAA,UAGe,2BAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,KAAA,UAAe,KAAA;EACf,YAAA,gBAA4B,KAAA,GAAQ,OAAA,QAAe,KAAA;EACnD,KAAA,GAAQ,sBAAA;AAAA;AAAA,UAGO,iCAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,gBAAA,IAAoB,GAAA,aAAgB,SAAA;EACpC,KAAA,GAAQ,IAAA,CAAK,sBAAA;AAAA;AAAA,KAGH,WAAA"}
1
+ {"version":3,"file":"types.d.cts","names":[],"sources":["../../../../src/client/stream/transport/types.ts"],"mappings":";;;;KAGY,mBAAA,IACV,GAAA,EAAK,GAAA,EACL,IAAA,EAAM,WAAA,KACH,OAAA,CAAQ,WAAA,IAAe,WAAA;AAAA,UAEX,sBAAA;EACf,QAAA;EACA,MAAA;;EAEA,KAAA;AAAA;AAAA,UAGe,2BAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,KAAA,UAAe,KAAA;EACf,YAAA,gBAA4B,KAAA,GAAQ,OAAA,QAAe,KAAA;EAhB7C;;;;;EAsBN,WAAA,GAAc,WAAA;EACd,KAAA,GAAQ,sBAAA;EApBO;;;;EAyBf,oBAAA;EAvBA;EAyBA,WAAA,IAAe,OAAA;IAAW,OAAA;IAAiB,KAAA;EAAA;EApBD;;;;EAyB1C,gBAAA,IAAoB,OAAA;AAAA;AAAA,UAGL,iCAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,gBAAA,IAAoB,GAAA,aAAgB,SAAA;EACpC,KAAA,GAAQ,IAAA,CAAK,sBAAA;EAjCb;;;;EAsCA,oBAAA;EAnCA;;;EAuCA,WAAA,IAAe,OAAA;IAAW,OAAA;IAAiB,KAAA;EAAA;EArCQ;;;;EA0CnD,aAAA,gBAA6B,OAAA;EA9B7B;;;;EAmCA,gBAAA,IAAoB,OAAA;AAAA;AAAA,KAGV,WAAA"}
@@ -1,3 +1,4 @@
1
+ import { AsyncCaller } from "../../../utils/async_caller.js";
1
2
  import { CommandResponse, ErrorResponse } from "@langchain/protocol";
2
3
 
3
4
  //#region src/client/stream/transport/types.d.ts
@@ -15,7 +16,28 @@ interface ProtocolSseTransportOptions {
15
16
  onRequest?: ProtocolRequestHook;
16
17
  fetch?: typeof fetch;
17
18
  fetchFactory?: () => typeof fetch | Promise<typeof fetch>;
19
+ /**
20
+ * When set, command and SSE subscription HTTP requests are executed
21
+ * through {@link AsyncCaller} (retries, concurrency). Typically wired
22
+ * from {@link BaseClient} via `client.threads.stream()`.
23
+ */
24
+ asyncCaller?: AsyncCaller;
18
25
  paths?: ProtocolTransportPaths;
26
+ /**
27
+ * Maximum reconnect attempts after an unexpected SSE disconnect.
28
+ * Defaults to 5. Set to 0 to disable automatic reconnection.
29
+ */
30
+ maxReconnectAttempts?: number;
31
+ /** Called before each SSE reconnect attempt (after backoff delay). */
32
+ onReconnect?: (options: {
33
+ attempt: number;
34
+ cause: unknown;
35
+ }) => void;
36
+ /**
37
+ * Backoff before each SSE reconnect attempt. Defaults to
38
+ * {@link webSocketReconnectDelayMs} from `./websocket.js`.
39
+ */
40
+ reconnectDelayMs?: (attempt: number) => number;
19
41
  }
20
42
  interface ProtocolWebSocketTransportOptions {
21
43
  apiUrl: string;
@@ -24,6 +46,28 @@ interface ProtocolWebSocketTransportOptions {
24
46
  onRequest?: ProtocolRequestHook;
25
47
  webSocketFactory?: (url: string) => WebSocket;
26
48
  paths?: Pick<ProtocolTransportPaths, "stream">;
49
+ /**
50
+ * Maximum reconnect attempts after an unexpected socket close.
51
+ * Defaults to 5. Set to 0 to disable automatic reconnection.
52
+ */
53
+ maxReconnectAttempts?: number;
54
+ /**
55
+ * Called before each reconnect attempt (after backoff delay).
56
+ */
57
+ onReconnect?: (options: {
58
+ attempt: number;
59
+ cause: unknown;
60
+ }) => void;
61
+ /**
62
+ * Invoked after the socket has been re-established. Use to restore
63
+ * server-side subscription state (see `ThreadStream`).
64
+ */
65
+ onReconnected?: () => void | Promise<void>;
66
+ /**
67
+ * Backoff before each reconnect attempt. Defaults to
68
+ * {@link webSocketReconnectDelayMs}.
69
+ */
70
+ reconnectDelayMs?: (attempt: number) => number;
27
71
  }
28
72
  type HeaderValue = string | undefined | null;
29
73
  //#endregion
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.ts","names":[],"sources":["../../../../src/client/stream/transport/types.ts"],"mappings":";;;KAEY,mBAAA,IACV,GAAA,EAAK,GAAA,EACL,IAAA,EAAM,WAAA,KACH,OAAA,CAAQ,WAAA,IAAe,WAAA;AAAA,UAEX,sBAAA;EACf,QAAA;EACA,MAAA;;EAEA,KAAA;AAAA;AAAA,UAGe,2BAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,KAAA,UAAe,KAAA;EACf,YAAA,gBAA4B,KAAA,GAAQ,OAAA,QAAe,KAAA;EACnD,KAAA,GAAQ,sBAAA;AAAA;AAAA,UAGO,iCAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,gBAAA,IAAoB,GAAA,aAAgB,SAAA;EACpC,KAAA,GAAQ,IAAA,CAAK,sBAAA;AAAA;AAAA,KAGH,WAAA"}
1
+ {"version":3,"file":"types.d.ts","names":[],"sources":["../../../../src/client/stream/transport/types.ts"],"mappings":";;;;KAGY,mBAAA,IACV,GAAA,EAAK,GAAA,EACL,IAAA,EAAM,WAAA,KACH,OAAA,CAAQ,WAAA,IAAe,WAAA;AAAA,UAEX,sBAAA;EACf,QAAA;EACA,MAAA;;EAEA,KAAA;AAAA;AAAA,UAGe,2BAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,KAAA,UAAe,KAAA;EACf,YAAA,gBAA4B,KAAA,GAAQ,OAAA,QAAe,KAAA;EAhB7C;;;;;EAsBN,WAAA,GAAc,WAAA;EACd,KAAA,GAAQ,sBAAA;EApBO;;;;EAyBf,oBAAA;EAvBA;EAyBA,WAAA,IAAe,OAAA;IAAW,OAAA;IAAiB,KAAA;EAAA;EApBD;;;;EAyB1C,gBAAA,IAAoB,OAAA;AAAA;AAAA,UAGL,iCAAA;EACf,MAAA;EACA,QAAA;EACA,cAAA,GAAiB,MAAA,SAAe,WAAA;EAChC,SAAA,GAAY,mBAAA;EACZ,gBAAA,IAAoB,GAAA,aAAgB,SAAA;EACpC,KAAA,GAAQ,IAAA,CAAK,sBAAA;EAjCb;;;;EAsCA,oBAAA;EAnCA;;;EAuCA,WAAA,IAAe,OAAA;IAAW,OAAA;IAAiB,KAAA;EAAA;EArCQ;;;;EA0CnD,aAAA,gBAA6B,OAAA;EA9B7B;;;;EAmCA,gBAAA,IAAoB,OAAA;AAAA;AAAA,KAGV,WAAA"}
@@ -1,10 +1,28 @@
1
+ const require_error = require("../error.cjs");
1
2
  const require_queue = require("./queue.cjs");
2
3
  const require_utils = require("./utils.cjs");
3
4
  //#region src/client/stream/transport/websocket.ts
5
+ const WEB_SOCKET_CONNECTING = 0;
6
+ const WEB_SOCKET_OPEN = 1;
7
+ const WEB_SOCKET_CLOSED = 3;
8
+ /**
9
+ * Exponential backoff with jitter for WebSocket reconnect. Mirrors
10
+ * {@link streamWithRetry} in `utils/stream.ts` (capped at 5s + 1s jitter).
11
+ */
12
+ function webSocketReconnectDelayMs(attempt) {
13
+ return Math.min(1e3 * 2 ** (attempt - 1), 5e3) + Math.random() * 1e3;
14
+ }
4
15
  /**
5
16
  * Transport adapter that speaks the thread-centric protocol over a
6
17
  * bidirectional WebSocket. Bound to a specific `threadId` — the socket
7
18
  * connects to `ws://.../threads/:thread_id/stream/events`.
19
+ *
20
+ * On unexpected disconnect the adapter reconnects with exponential
21
+ * backoff (see {@link ProtocolWebSocketTransportOptions.maxReconnectAttempts}).
22
+ * The server replays buffered events on the new socket; the SDK
23
+ * deduplicates by `event_id`. {@link ProtocolWebSocketTransportOptions.onReconnected}
24
+ * runs after each successful reconnect so `ThreadStream` can re-issue
25
+ * `subscription.subscribe` commands.
8
26
  */
9
27
  var ProtocolWebSocketTransportAdapter = class {
10
28
  threadId;
@@ -14,10 +32,15 @@ var ProtocolWebSocketTransportAdapter = class {
14
32
  onRequest;
15
33
  webSocketFactory;
16
34
  streamUrl;
35
+ maxReconnectAttempts;
36
+ onReconnect;
37
+ reconnectDelayMs;
38
+ onReconnected;
17
39
  pending = /* @__PURE__ */ new Map();
18
40
  socket = null;
19
41
  closed = false;
20
42
  intentionalClose = false;
43
+ reconnectInFlight = null;
21
44
  constructor(options) {
22
45
  this.apiUrl = options.apiUrl;
23
46
  this.threadId = options.threadId;
@@ -25,18 +48,32 @@ var ProtocolWebSocketTransportAdapter = class {
25
48
  this.onRequest = options.onRequest;
26
49
  this.webSocketFactory = options.webSocketFactory ?? ((url) => new WebSocket(url));
27
50
  this.streamUrl = options.paths?.stream ?? `/threads/${this.threadId}/stream/events`;
51
+ this.maxReconnectAttempts = options.maxReconnectAttempts ?? 5;
52
+ this.onReconnect = options.onReconnect;
53
+ this.onReconnected = options.onReconnected;
54
+ this.reconnectDelayMs = options.reconnectDelayMs ?? webSocketReconnectDelayMs;
55
+ }
56
+ /**
57
+ * Register a callback invoked after each successful reconnect. Used
58
+ * by {@link ThreadStream} to re-send active `subscription.subscribe`
59
+ * commands.
60
+ */
61
+ setOnReconnected(handler) {
62
+ this.onReconnected = handler;
28
63
  }
29
64
  async open() {
30
- if (this.socket != null) return;
65
+ if (this.closed) throw new Error("Protocol WebSocket transport is closed.");
66
+ if (this.socket?.readyState === WEB_SOCKET_OPEN) return;
67
+ if (this.socket != null) {
68
+ this.#detachSocket(this.socket);
69
+ this.socket = null;
70
+ }
31
71
  this.assertBrowserSafeTransportConfig();
32
72
  const wsUrl = require_utils.toWebSocketUrl(require_utils.toAbsoluteUrl(this.apiUrl, this.streamUrl).toString());
33
73
  const socket = this.webSocketFactory(wsUrl);
34
74
  this.socket = socket;
35
- this.closed = false;
36
75
  this.intentionalClose = false;
37
- socket.addEventListener("message", this.handleMessage);
38
- socket.addEventListener("close", this.handleClose);
39
- socket.addEventListener("error", this.handleSocketError);
76
+ this.#attachSocket(socket);
40
77
  await new Promise((resolve, reject) => {
41
78
  const onOpen = () => {
42
79
  cleanup();
@@ -80,8 +117,9 @@ var ProtocolWebSocketTransportAdapter = class {
80
117
  const socket = this.socket;
81
118
  this.socket = null;
82
119
  if (!socket) return;
120
+ this.#detachSocket(socket);
83
121
  await new Promise((resolve) => {
84
- if (socket.readyState === WebSocket.CLOSED) {
122
+ if (socket.readyState === WEB_SOCKET_CLOSED) {
85
123
  resolve();
86
124
  return;
87
125
  }
@@ -90,7 +128,7 @@ var ProtocolWebSocketTransportAdapter = class {
90
128
  resolve();
91
129
  };
92
130
  socket.addEventListener("close", onClose, { once: true });
93
- if (socket.readyState === WebSocket.OPEN || socket.readyState === WebSocket.CONNECTING) socket.close();
131
+ if (socket.readyState === WEB_SOCKET_OPEN || socket.readyState === WEB_SOCKET_CONNECTING) socket.close();
94
132
  else resolve();
95
133
  });
96
134
  }
@@ -98,8 +136,12 @@ var ProtocolWebSocketTransportAdapter = class {
98
136
  if (require_utils.hasHeaders(this.defaultHeaders) || this.onRequest != null) throw new Error("Browser WebSocket protocol transport does not support defaultHeaders or onRequest hooks. Supply a custom protocolWebSocketFactory if you need custom WebSocket setup.");
99
137
  }
100
138
  async sendCommand(command) {
101
- const socket = this.socket;
102
- if (socket == null || socket.readyState !== WebSocket.OPEN) throw new Error("Protocol WebSocket is not open.");
139
+ let socket = this.socket;
140
+ if (this.reconnectInFlight != null && (socket == null || socket.readyState !== WEB_SOCKET_OPEN)) {
141
+ await this.reconnectInFlight.catch(() => void 0);
142
+ socket = this.socket;
143
+ }
144
+ if (socket == null || socket.readyState !== WEB_SOCKET_OPEN) throw new Error("Protocol WebSocket is not open.");
103
145
  return await new Promise((resolve, reject) => {
104
146
  this.pending.set(command.id, {
105
147
  resolve,
@@ -113,6 +155,16 @@ var ProtocolWebSocketTransportAdapter = class {
113
155
  }
114
156
  });
115
157
  }
158
+ #attachSocket(socket) {
159
+ socket.addEventListener("message", this.handleMessage);
160
+ socket.addEventListener("close", this.handleClose);
161
+ socket.addEventListener("error", this.handleSocketError);
162
+ }
163
+ #detachSocket(socket) {
164
+ socket.removeEventListener("message", this.handleMessage);
165
+ socket.removeEventListener("close", this.handleClose);
166
+ socket.removeEventListener("error", this.handleSocketError);
167
+ }
116
168
  handleMessage = (event) => {
117
169
  let payload;
118
170
  try {
@@ -131,25 +183,62 @@ var ProtocolWebSocketTransportAdapter = class {
131
183
  if (require_utils.isRecord(payload) && payload.type === "event") this.queue.push(payload);
132
184
  };
133
185
  handleClose = () => {
186
+ const socket = this.socket;
187
+ if (socket != null) this.#detachSocket(socket);
134
188
  this.socket = null;
135
189
  if (this.intentionalClose || this.closed) {
136
190
  this.queue.close();
137
191
  return;
138
192
  }
139
- const error = /* @__PURE__ */ new Error("Protocol WebSocket closed unexpectedly.");
140
- for (const { reject } of this.pending.values()) reject(error);
141
- this.pending.clear();
142
- this.queue.close(error);
193
+ this.#handleUnexpectedDisconnect(/* @__PURE__ */ new Error("Protocol WebSocket closed unexpectedly."));
143
194
  };
144
195
  handleSocketError = () => {
145
196
  if (this.closed || this.intentionalClose) return;
146
- const error = /* @__PURE__ */ new Error("Protocol WebSocket encountered an error.");
197
+ this.#handleUnexpectedDisconnect(/* @__PURE__ */ new Error("Protocol WebSocket encountered an error."));
198
+ };
199
+ #handleUnexpectedDisconnect(cause) {
200
+ const error = require_utils.toError(cause);
147
201
  for (const { reject } of this.pending.values()) reject(error);
148
202
  this.pending.clear();
149
- this.queue.close(error);
150
- };
203
+ if (this.maxReconnectAttempts <= 0) {
204
+ this.queue.close(error);
205
+ return;
206
+ }
207
+ this.#scheduleReconnect(cause);
208
+ }
209
+ #scheduleReconnect(cause) {
210
+ if (this.closed || this.intentionalClose) return;
211
+ if (this.reconnectInFlight != null) return;
212
+ this.reconnectInFlight = this.#runReconnectLoop(cause).finally(() => {
213
+ this.reconnectInFlight = null;
214
+ });
215
+ }
216
+ async #runReconnectLoop(initialCause) {
217
+ let lastError = initialCause;
218
+ for (let attempt = 1; attempt <= this.maxReconnectAttempts; attempt += 1) {
219
+ if (this.closed || this.intentionalClose) return;
220
+ this.onReconnect?.({
221
+ attempt,
222
+ cause: lastError
223
+ });
224
+ const delay = this.reconnectDelayMs(attempt);
225
+ if (delay > 0) await new Promise((resolve) => {
226
+ setTimeout(resolve, delay);
227
+ });
228
+ if (this.closed || this.intentionalClose) return;
229
+ try {
230
+ await this.open();
231
+ if (this.onReconnected) await this.onReconnected();
232
+ return;
233
+ } catch (error) {
234
+ lastError = error;
235
+ }
236
+ }
237
+ this.queue.close(new require_error.MaxWebSocketReconnectAttemptsError(this.maxReconnectAttempts, lastError));
238
+ }
151
239
  };
152
240
  //#endregion
153
241
  exports.ProtocolWebSocketTransportAdapter = ProtocolWebSocketTransportAdapter;
242
+ exports.webSocketReconnectDelayMs = webSocketReconnectDelayMs;
154
243
 
155
244
  //# sourceMappingURL=websocket.cjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"websocket.cjs","names":["AsyncQueue","toWebSocketUrl","toAbsoluteUrl","hasHeaders","toError","isRecord"],"sources":["../../../../src/client/stream/transport/websocket.ts"],"sourcesContent":["import { AsyncQueue } from \"./queue.js\";\nimport type {\n Message,\n Command,\n CommandResponse,\n ErrorResponse,\n} from \"@langchain/protocol\";\n\nimport {\n toAbsoluteUrl,\n toWebSocketUrl,\n isRecord,\n hasHeaders,\n toError,\n} from \"./utils.js\";\nimport type {\n HeaderValue,\n ProtocolRequestHook,\n PendingResponse,\n ProtocolWebSocketTransportOptions,\n} from \"./types.js\";\nimport type { TransportAdapter } from \"../transport.js\";\n\n/**\n * Transport adapter that speaks the thread-centric protocol over a\n * bidirectional WebSocket. Bound to a specific `threadId` — the socket\n * connects to `ws://.../threads/:thread_id/stream/events`.\n */\nexport class ProtocolWebSocketTransportAdapter implements TransportAdapter {\n readonly threadId: string;\n\n private readonly queue = new AsyncQueue<Message>();\n\n private readonly apiUrl: string;\n\n private readonly defaultHeaders?: Record<string, HeaderValue>;\n\n private readonly onRequest?: ProtocolRequestHook;\n\n private readonly webSocketFactory: (url: string) => WebSocket;\n\n private readonly streamUrl: string;\n\n private readonly pending = new Map<number, PendingResponse>();\n\n private socket: WebSocket | null = null;\n\n private closed = false;\n\n private intentionalClose = false;\n\n constructor(options: ProtocolWebSocketTransportOptions) {\n this.apiUrl = options.apiUrl;\n this.threadId = options.threadId;\n this.defaultHeaders = options.defaultHeaders;\n this.onRequest = options.onRequest;\n this.webSocketFactory =\n options.webSocketFactory ?? ((url) => new WebSocket(url));\n this.streamUrl =\n options.paths?.stream ?? `/threads/${this.threadId}/stream/events`;\n }\n\n async open(): Promise<void> {\n if (this.socket != null) return;\n this.assertBrowserSafeTransportConfig();\n\n const wsUrl = toWebSocketUrl(\n toAbsoluteUrl(this.apiUrl, this.streamUrl).toString()\n );\n const socket = this.webSocketFactory(wsUrl);\n this.socket = socket;\n this.closed = false;\n this.intentionalClose = false;\n\n socket.addEventListener(\"message\", this.handleMessage);\n socket.addEventListener(\"close\", this.handleClose);\n socket.addEventListener(\"error\", this.handleSocketError);\n\n await new Promise<void>((resolve, reject) => {\n const onOpen = () => {\n cleanup();\n resolve();\n };\n const onError = () => {\n cleanup();\n reject(new Error(\"Failed to open protocol WebSocket.\"));\n };\n const cleanup = () => {\n socket.removeEventListener(\"open\", onOpen);\n socket.removeEventListener(\"error\", onError);\n };\n socket.addEventListener(\"open\", onOpen, { once: true });\n socket.addEventListener(\"error\", onError, { once: true });\n });\n }\n\n async send(\n command: Command\n ): Promise<CommandResponse | ErrorResponse | void> {\n return await this.sendCommand(command);\n }\n\n events(): AsyncIterable<Message> {\n const queue = this.queue;\n return {\n [Symbol.asyncIterator]: () => ({\n next: async () => await queue.shift(),\n return: async () => {\n queue.close();\n return { done: true, value: undefined };\n },\n }),\n };\n }\n\n async close(): Promise<void> {\n if (this.closed) {\n return;\n }\n\n this.closed = true;\n this.intentionalClose = true;\n\n for (const { reject } of this.pending.values()) {\n reject(new Error(\"Protocol WebSocket connection closed.\"));\n }\n this.pending.clear();\n this.queue.close();\n\n const socket = this.socket;\n this.socket = null;\n if (!socket) {\n return;\n }\n\n await new Promise<void>((resolve) => {\n if (socket.readyState === WebSocket.CLOSED) {\n resolve();\n return;\n }\n\n const onClose = () => {\n socket.removeEventListener(\"close\", onClose);\n resolve();\n };\n\n socket.addEventListener(\"close\", onClose, { once: true });\n if (\n socket.readyState === WebSocket.OPEN ||\n socket.readyState === WebSocket.CONNECTING\n ) {\n socket.close();\n } else {\n resolve();\n }\n });\n }\n\n private assertBrowserSafeTransportConfig(): void {\n if (hasHeaders(this.defaultHeaders) || this.onRequest != null) {\n throw new Error(\n \"Browser WebSocket protocol transport does not support defaultHeaders or onRequest hooks. Supply a custom protocolWebSocketFactory if you need custom WebSocket setup.\"\n );\n }\n }\n\n private async sendCommand(\n command: Command\n ): Promise<CommandResponse | ErrorResponse> {\n const socket = this.socket;\n if (socket == null || socket.readyState !== WebSocket.OPEN) {\n throw new Error(\"Protocol WebSocket is not open.\");\n }\n\n return await new Promise<CommandResponse | ErrorResponse>(\n (resolve, reject) => {\n this.pending.set(command.id, { resolve, reject });\n\n try {\n socket.send(JSON.stringify(command));\n } catch (error) {\n this.pending.delete(command.id);\n reject(toError(error));\n }\n }\n );\n }\n\n private readonly handleMessage = (event: MessageEvent): void => {\n let payload: unknown;\n try {\n payload = JSON.parse(String(event.data));\n } catch {\n return;\n }\n\n if (\n isRecord(payload) &&\n typeof payload.id === \"number\" &&\n (payload.type === \"success\" || payload.type === \"error\")\n ) {\n const pending = this.pending.get(payload.id);\n if (pending) {\n this.pending.delete(payload.id);\n pending.resolve(payload as CommandResponse | ErrorResponse);\n }\n return;\n }\n\n if (isRecord(payload) && payload.type === \"event\") {\n this.queue.push(payload as Message);\n }\n };\n\n private readonly handleClose = (): void => {\n this.socket = null;\n\n if (this.intentionalClose || this.closed) {\n this.queue.close();\n return;\n }\n\n const error = new Error(\"Protocol WebSocket closed unexpectedly.\");\n for (const { reject } of this.pending.values()) {\n reject(error);\n }\n this.pending.clear();\n this.queue.close(error);\n };\n\n private readonly handleSocketError = (): void => {\n if (this.closed || this.intentionalClose) {\n return;\n }\n\n const error = new Error(\"Protocol WebSocket encountered an error.\");\n for (const { reject } of this.pending.values()) {\n reject(error);\n }\n this.pending.clear();\n this.queue.close(error);\n };\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"websocket.cjs","names":["AsyncQueue","#detachSocket","toWebSocketUrl","toAbsoluteUrl","#attachSocket","hasHeaders","toError","isRecord","#handleUnexpectedDisconnect","#scheduleReconnect","#runReconnectLoop","MaxWebSocketReconnectAttemptsError"],"sources":["../../../../src/client/stream/transport/websocket.ts"],"sourcesContent":["import { AsyncQueue } from \"./queue.js\";\nimport type {\n Message,\n Command,\n CommandResponse,\n ErrorResponse,\n} from \"@langchain/protocol\";\n\nimport {\n toAbsoluteUrl,\n toWebSocketUrl,\n isRecord,\n hasHeaders,\n toError,\n} from \"./utils.js\";\nimport type {\n HeaderValue,\n ProtocolRequestHook,\n PendingResponse,\n ProtocolWebSocketTransportOptions,\n} from \"./types.js\";\nimport type { TransportAdapter } from \"../transport.js\";\nimport { MaxWebSocketReconnectAttemptsError } from \"../error.js\";\n\nconst WEB_SOCKET_CONNECTING = 0;\nconst WEB_SOCKET_OPEN = 1;\nconst WEB_SOCKET_CLOSED = 3;\n\n/**\n * Reconnect tuning for {@link ProtocolWebSocketTransportAdapter}. A subset\n * of {@link ProtocolWebSocketTransportOptions}.\n */\nexport interface WebSocketReconnectOptions {\n /**\n * Maximum reconnection attempts after an unexpected disconnect.\n * Defaults to 5.\n */\n maxReconnectAttempts?: number;\n\n /**\n * Invoked before each reconnect attempt (after backoff).\n */\n onReconnect?: (options: { attempt: number; cause: unknown }) => void;\n}\n\n/**\n * Exponential backoff with jitter for WebSocket reconnect. Mirrors\n * {@link streamWithRetry} in `utils/stream.ts` (capped at 5s + 1s jitter).\n */\nexport function webSocketReconnectDelayMs(attempt: number): number {\n const baseDelay = Math.min(1000 * 2 ** (attempt - 1), 5000);\n const jitter = Math.random() * 1000;\n return baseDelay + jitter;\n}\n\n/**\n * Transport adapter that speaks the thread-centric protocol over a\n * bidirectional WebSocket. Bound to a specific `threadId` — the socket\n * connects to `ws://.../threads/:thread_id/stream/events`.\n *\n * On unexpected disconnect the adapter reconnects with exponential\n * backoff (see {@link ProtocolWebSocketTransportOptions.maxReconnectAttempts}).\n * The server replays buffered events on the new socket; the SDK\n * deduplicates by `event_id`. {@link ProtocolWebSocketTransportOptions.onReconnected}\n * runs after each successful reconnect so `ThreadStream` can re-issue\n * `subscription.subscribe` commands.\n */\nexport class ProtocolWebSocketTransportAdapter implements TransportAdapter {\n readonly threadId: string;\n\n private readonly queue = new AsyncQueue<Message>();\n\n private readonly apiUrl: string;\n\n private readonly defaultHeaders?: Record<string, HeaderValue>;\n\n private readonly onRequest?: ProtocolRequestHook;\n\n private readonly webSocketFactory: (url: string) => WebSocket;\n\n private readonly streamUrl: string;\n\n private readonly maxReconnectAttempts: number;\n\n private readonly onReconnect?: ProtocolWebSocketTransportOptions[\"onReconnect\"];\n\n private readonly reconnectDelayMs: (attempt: number) => number;\n\n private onReconnected?: () => void | Promise<void>;\n\n private readonly pending = new Map<number, PendingResponse>();\n\n private socket: WebSocket | null = null;\n\n private closed = false;\n\n private intentionalClose = false;\n\n private reconnectInFlight: Promise<void> | null = null;\n\n constructor(options: ProtocolWebSocketTransportOptions) {\n this.apiUrl = options.apiUrl;\n this.threadId = options.threadId;\n this.defaultHeaders = options.defaultHeaders;\n this.onRequest = options.onRequest;\n this.webSocketFactory =\n options.webSocketFactory ?? ((url) => new WebSocket(url));\n this.streamUrl =\n options.paths?.stream ?? `/threads/${this.threadId}/stream/events`;\n this.maxReconnectAttempts = options.maxReconnectAttempts ?? 5;\n this.onReconnect = options.onReconnect;\n this.onReconnected = options.onReconnected;\n this.reconnectDelayMs =\n options.reconnectDelayMs ?? webSocketReconnectDelayMs;\n }\n\n /**\n * Register a callback invoked after each successful reconnect. Used\n * by {@link ThreadStream} to re-send active `subscription.subscribe`\n * commands.\n */\n setOnReconnected(handler: () => void | Promise<void>): void {\n this.onReconnected = handler;\n }\n\n async open(): Promise<void> {\n if (this.closed) {\n throw new Error(\"Protocol WebSocket transport is closed.\");\n }\n if (this.socket?.readyState === WEB_SOCKET_OPEN) {\n return;\n }\n if (this.socket != null) {\n this.#detachSocket(this.socket);\n this.socket = null;\n }\n\n this.assertBrowserSafeTransportConfig();\n\n const wsUrl = toWebSocketUrl(\n toAbsoluteUrl(this.apiUrl, this.streamUrl).toString()\n );\n const socket = this.webSocketFactory(wsUrl);\n this.socket = socket;\n this.intentionalClose = false;\n\n this.#attachSocket(socket);\n\n await new Promise<void>((resolve, reject) => {\n const onOpen = () => {\n cleanup();\n resolve();\n };\n const onError = () => {\n cleanup();\n reject(new Error(\"Failed to open protocol WebSocket.\"));\n };\n const cleanup = () => {\n socket.removeEventListener(\"open\", onOpen);\n socket.removeEventListener(\"error\", onError);\n };\n socket.addEventListener(\"open\", onOpen, { once: true });\n socket.addEventListener(\"error\", onError, { once: true });\n });\n }\n\n async send(\n command: Command\n ): Promise<CommandResponse | ErrorResponse | void> {\n return await this.sendCommand(command);\n }\n\n events(): AsyncIterable<Message> {\n const queue = this.queue;\n return {\n [Symbol.asyncIterator]: () => ({\n next: async () => await queue.shift(),\n return: async () => {\n queue.close();\n return { done: true, value: undefined };\n },\n }),\n };\n }\n\n async close(): Promise<void> {\n if (this.closed) {\n return;\n }\n\n this.closed = true;\n this.intentionalClose = true;\n\n for (const { reject } of this.pending.values()) {\n reject(new Error(\"Protocol WebSocket connection closed.\"));\n }\n this.pending.clear();\n this.queue.close();\n\n const socket = this.socket;\n this.socket = null;\n if (!socket) {\n return;\n }\n\n this.#detachSocket(socket);\n\n await new Promise<void>((resolve) => {\n if (socket.readyState === WEB_SOCKET_CLOSED) {\n resolve();\n return;\n }\n\n const onClose = () => {\n socket.removeEventListener(\"close\", onClose);\n resolve();\n };\n\n socket.addEventListener(\"close\", onClose, { once: true });\n if (\n socket.readyState === WEB_SOCKET_OPEN ||\n socket.readyState === WEB_SOCKET_CONNECTING\n ) {\n socket.close();\n } else {\n resolve();\n }\n });\n }\n\n private assertBrowserSafeTransportConfig(): void {\n if (hasHeaders(this.defaultHeaders) || this.onRequest != null) {\n throw new Error(\n \"Browser WebSocket protocol transport does not support defaultHeaders or onRequest hooks. Supply a custom protocolWebSocketFactory if you need custom WebSocket setup.\"\n );\n }\n }\n\n private async sendCommand(\n command: Command\n ): Promise<CommandResponse | ErrorResponse> {\n // Wait for an in-flight reconnect only when the socket is not usable.\n // After `open()` succeeds, `#runReconnectLoop` may still be awaiting\n // `onReconnected` (e.g. ThreadStream resubscribe). Those callbacks call\n // `sendCommand` and must not await the same `reconnectInFlight` promise.\n let socket = this.socket;\n if (\n this.reconnectInFlight != null &&\n (socket == null || socket.readyState !== WEB_SOCKET_OPEN)\n ) {\n await this.reconnectInFlight.catch(() => undefined);\n socket = this.socket;\n }\n\n if (socket == null || socket.readyState !== WEB_SOCKET_OPEN) {\n throw new Error(\"Protocol WebSocket is not open.\");\n }\n\n return await new Promise<CommandResponse | ErrorResponse>(\n (resolve, reject) => {\n this.pending.set(command.id, { resolve, reject });\n\n try {\n socket.send(JSON.stringify(command));\n } catch (error) {\n this.pending.delete(command.id);\n reject(toError(error));\n }\n }\n );\n }\n\n #attachSocket(socket: WebSocket): void {\n socket.addEventListener(\"message\", this.handleMessage);\n socket.addEventListener(\"close\", this.handleClose);\n socket.addEventListener(\"error\", this.handleSocketError);\n }\n\n #detachSocket(socket: WebSocket): void {\n socket.removeEventListener(\"message\", this.handleMessage);\n socket.removeEventListener(\"close\", this.handleClose);\n socket.removeEventListener(\"error\", this.handleSocketError);\n }\n\n private readonly handleMessage = (event: MessageEvent): void => {\n let payload: unknown;\n try {\n payload = JSON.parse(String(event.data));\n } catch {\n return;\n }\n\n if (\n isRecord(payload) &&\n typeof payload.id === \"number\" &&\n (payload.type === \"success\" || payload.type === \"error\")\n ) {\n const pending = this.pending.get(payload.id);\n if (pending) {\n this.pending.delete(payload.id);\n pending.resolve(payload as CommandResponse | ErrorResponse);\n }\n return;\n }\n\n if (isRecord(payload) && payload.type === \"event\") {\n this.queue.push(payload as Message);\n }\n };\n\n private readonly handleClose = (): void => {\n const socket = this.socket;\n if (socket != null) {\n this.#detachSocket(socket);\n }\n this.socket = null;\n\n if (this.intentionalClose || this.closed) {\n this.queue.close();\n return;\n }\n\n this.#handleUnexpectedDisconnect(\n new Error(\"Protocol WebSocket closed unexpectedly.\")\n );\n };\n\n private readonly handleSocketError = (): void => {\n if (this.closed || this.intentionalClose) {\n return;\n }\n\n this.#handleUnexpectedDisconnect(\n new Error(\"Protocol WebSocket encountered an error.\")\n );\n };\n\n #handleUnexpectedDisconnect(cause: unknown): void {\n const error = toError(cause);\n for (const { reject } of this.pending.values()) {\n reject(error);\n }\n this.pending.clear();\n\n if (this.maxReconnectAttempts <= 0) {\n this.queue.close(error);\n return;\n }\n\n this.#scheduleReconnect(cause);\n }\n\n #scheduleReconnect(cause: unknown): void {\n if (this.closed || this.intentionalClose) {\n return;\n }\n if (this.reconnectInFlight != null) {\n return;\n }\n\n this.reconnectInFlight = this.#runReconnectLoop(cause).finally(() => {\n this.reconnectInFlight = null;\n });\n }\n\n async #runReconnectLoop(initialCause: unknown): Promise<void> {\n let lastError: unknown = initialCause;\n\n for (let attempt = 1; attempt <= this.maxReconnectAttempts; attempt += 1) {\n if (this.closed || this.intentionalClose) {\n return;\n }\n\n this.onReconnect?.({ attempt, cause: lastError });\n\n const delay = this.reconnectDelayMs(attempt);\n if (delay > 0) {\n await new Promise<void>((resolve) => {\n setTimeout(resolve, delay);\n });\n }\n\n if (this.closed || this.intentionalClose) {\n return;\n }\n\n try {\n await this.open();\n if (this.onReconnected) {\n await this.onReconnected();\n }\n return;\n } catch (error) {\n lastError = error;\n }\n }\n\n this.queue.close(\n new MaxWebSocketReconnectAttemptsError(\n this.maxReconnectAttempts,\n lastError\n )\n );\n }\n}\n"],"mappings":"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"}
@@ -7,8 +7,16 @@ import { Command, CommandResponse, ErrorResponse, Message } from "@langchain/pro
7
7
  * Transport adapter that speaks the thread-centric protocol over a
8
8
  * bidirectional WebSocket. Bound to a specific `threadId` — the socket
9
9
  * connects to `ws://.../threads/:thread_id/stream/events`.
10
+ *
11
+ * On unexpected disconnect the adapter reconnects with exponential
12
+ * backoff (see {@link ProtocolWebSocketTransportOptions.maxReconnectAttempts}).
13
+ * The server replays buffered events on the new socket; the SDK
14
+ * deduplicates by `event_id`. {@link ProtocolWebSocketTransportOptions.onReconnected}
15
+ * runs after each successful reconnect so `ThreadStream` can re-issue
16
+ * `subscription.subscribe` commands.
10
17
  */
11
18
  declare class ProtocolWebSocketTransportAdapter implements TransportAdapter {
19
+ #private;
12
20
  readonly threadId: string;
13
21
  private readonly queue;
14
22
  private readonly apiUrl;
@@ -16,11 +24,22 @@ declare class ProtocolWebSocketTransportAdapter implements TransportAdapter {
16
24
  private readonly onRequest?;
17
25
  private readonly webSocketFactory;
18
26
  private readonly streamUrl;
27
+ private readonly maxReconnectAttempts;
28
+ private readonly onReconnect?;
29
+ private readonly reconnectDelayMs;
30
+ private onReconnected?;
19
31
  private readonly pending;
20
32
  private socket;
21
33
  private closed;
22
34
  private intentionalClose;
35
+ private reconnectInFlight;
23
36
  constructor(options: ProtocolWebSocketTransportOptions);
37
+ /**
38
+ * Register a callback invoked after each successful reconnect. Used
39
+ * by {@link ThreadStream} to re-send active `subscription.subscribe`
40
+ * commands.
41
+ */
42
+ setOnReconnected(handler: () => void | Promise<void>): void;
24
43
  open(): Promise<void>;
25
44
  send(command: Command): Promise<CommandResponse | ErrorResponse | void>;
26
45
  events(): AsyncIterable<Message>;
@@ -1 +1 @@
1
- {"version":3,"file":"websocket.d.cts","names":[],"sources":["../../../../src/client/stream/transport/websocket.ts"],"mappings":";;;;;;;AA4BA;;;cAAa,iCAAA,YAA6C,gBAAA;EAAA,SAC/C,QAAA;EAAA,iBAEQ,KAAA;EAAA,iBAEA,MAAA;EAAA,iBAEA,cAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,gBAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,OAAA;EAAA,QAET,MAAA;EAAA,QAEA,MAAA;EAAA,QAEA,gBAAA;EAER,WAAA,CAAY,OAAA,EAAS,iCAAA;EAWf,IAAA,CAAA,GAAQ,OAAA;EAkCR,IAAA,CACJ,OAAA,EAAS,OAAA,GACR,OAAA,CAAQ,eAAA,GAAkB,aAAA;EAI7B,MAAA,CAAA,GAAU,aAAA,CAAc,OAAA;EAalB,KAAA,CAAA,GAAS,OAAA;EAAA,QA2CP,gCAAA;EAAA,QAQM,WAAA;EAAA,iBAsBG,aAAA;EAAA,iBA0BA,WAAA;EAAA,iBAgBA,iBAAA;AAAA"}
1
+ {"version":3,"file":"websocket.d.cts","names":[],"sources":["../../../../src/client/stream/transport/websocket.ts"],"mappings":";;;;;;;;;;;;;;;;;cAmEa,iCAAA,YAA6C,gBAAA;EAAA;WAC/C,QAAA;EAAA,iBAEQ,KAAA;EAAA,iBAEA,MAAA;EAAA,iBAEA,cAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,gBAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,oBAAA;EAAA,iBAEA,WAAA;EAAA,iBAEA,gBAAA;EAAA,QAET,aAAA;EAAA,iBAES,OAAA;EAAA,QAET,MAAA;EAAA,QAEA,MAAA;EAAA,QAEA,gBAAA;EAAA,QAEA,iBAAA;EAER,WAAA,CAAY,OAAA,EAAS,iCAAA;EAqFN;;;;;EAhEf,gBAAA,CAAiB,OAAA,eAAsB,OAAA;EAIjC,IAAA,CAAA,GAAQ,OAAA;EAyCR,IAAA,CACJ,OAAA,EAAS,OAAA,GACR,OAAA,CAAQ,eAAA,GAAkB,aAAA;EAI7B,MAAA,CAAA,GAAU,aAAA,CAAc,OAAA;EAalB,KAAA,CAAA,GAAS,OAAA;EAAA,QA6CP,gCAAA;EAAA,QAQM,WAAA;EAAA,iBA8CG,aAAA;EAAA,iBA0BA,WAAA;EAAA,iBAiBA,iBAAA;AAAA"}
@@ -7,8 +7,16 @@ import { Command, CommandResponse, ErrorResponse, Message } from "@langchain/pro
7
7
  * Transport adapter that speaks the thread-centric protocol over a
8
8
  * bidirectional WebSocket. Bound to a specific `threadId` — the socket
9
9
  * connects to `ws://.../threads/:thread_id/stream/events`.
10
+ *
11
+ * On unexpected disconnect the adapter reconnects with exponential
12
+ * backoff (see {@link ProtocolWebSocketTransportOptions.maxReconnectAttempts}).
13
+ * The server replays buffered events on the new socket; the SDK
14
+ * deduplicates by `event_id`. {@link ProtocolWebSocketTransportOptions.onReconnected}
15
+ * runs after each successful reconnect so `ThreadStream` can re-issue
16
+ * `subscription.subscribe` commands.
10
17
  */
11
18
  declare class ProtocolWebSocketTransportAdapter implements TransportAdapter {
19
+ #private;
12
20
  readonly threadId: string;
13
21
  private readonly queue;
14
22
  private readonly apiUrl;
@@ -16,11 +24,22 @@ declare class ProtocolWebSocketTransportAdapter implements TransportAdapter {
16
24
  private readonly onRequest?;
17
25
  private readonly webSocketFactory;
18
26
  private readonly streamUrl;
27
+ private readonly maxReconnectAttempts;
28
+ private readonly onReconnect?;
29
+ private readonly reconnectDelayMs;
30
+ private onReconnected?;
19
31
  private readonly pending;
20
32
  private socket;
21
33
  private closed;
22
34
  private intentionalClose;
35
+ private reconnectInFlight;
23
36
  constructor(options: ProtocolWebSocketTransportOptions);
37
+ /**
38
+ * Register a callback invoked after each successful reconnect. Used
39
+ * by {@link ThreadStream} to re-send active `subscription.subscribe`
40
+ * commands.
41
+ */
42
+ setOnReconnected(handler: () => void | Promise<void>): void;
24
43
  open(): Promise<void>;
25
44
  send(command: Command): Promise<CommandResponse | ErrorResponse | void>;
26
45
  events(): AsyncIterable<Message>;
@@ -1 +1 @@
1
- {"version":3,"file":"websocket.d.ts","names":[],"sources":["../../../../src/client/stream/transport/websocket.ts"],"mappings":";;;;;;;AA4BA;;;cAAa,iCAAA,YAA6C,gBAAA;EAAA,SAC/C,QAAA;EAAA,iBAEQ,KAAA;EAAA,iBAEA,MAAA;EAAA,iBAEA,cAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,gBAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,OAAA;EAAA,QAET,MAAA;EAAA,QAEA,MAAA;EAAA,QAEA,gBAAA;EAER,WAAA,CAAY,OAAA,EAAS,iCAAA;EAWf,IAAA,CAAA,GAAQ,OAAA;EAkCR,IAAA,CACJ,OAAA,EAAS,OAAA,GACR,OAAA,CAAQ,eAAA,GAAkB,aAAA;EAI7B,MAAA,CAAA,GAAU,aAAA,CAAc,OAAA;EAalB,KAAA,CAAA,GAAS,OAAA;EAAA,QA2CP,gCAAA;EAAA,QAQM,WAAA;EAAA,iBAsBG,aAAA;EAAA,iBA0BA,WAAA;EAAA,iBAgBA,iBAAA;AAAA"}
1
+ {"version":3,"file":"websocket.d.ts","names":[],"sources":["../../../../src/client/stream/transport/websocket.ts"],"mappings":";;;;;;;;;;;;;;;;;cAmEa,iCAAA,YAA6C,gBAAA;EAAA;WAC/C,QAAA;EAAA,iBAEQ,KAAA;EAAA,iBAEA,MAAA;EAAA,iBAEA,cAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,gBAAA;EAAA,iBAEA,SAAA;EAAA,iBAEA,oBAAA;EAAA,iBAEA,WAAA;EAAA,iBAEA,gBAAA;EAAA,QAET,aAAA;EAAA,iBAES,OAAA;EAAA,QAET,MAAA;EAAA,QAEA,MAAA;EAAA,QAEA,gBAAA;EAAA,QAEA,iBAAA;EAER,WAAA,CAAY,OAAA,EAAS,iCAAA;EAqFN;;;;;EAhEf,gBAAA,CAAiB,OAAA,eAAsB,OAAA;EAIjC,IAAA,CAAA,GAAQ,OAAA;EAyCR,IAAA,CACJ,OAAA,EAAS,OAAA,GACR,OAAA,CAAQ,eAAA,GAAkB,aAAA;EAI7B,MAAA,CAAA,GAAU,aAAA,CAAc,OAAA;EAalB,KAAA,CAAA,GAAS,OAAA;EAAA,QA6CP,gCAAA;EAAA,QAQM,WAAA;EAAA,iBA8CG,aAAA;EAAA,iBA0BA,WAAA;EAAA,iBAiBA,iBAAA;AAAA"}