@agent-relay/dashboard-server 2.0.92 → 2.0.94

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 (205) hide show
  1. package/dist/lib/types.d.ts +3 -2
  2. package/dist/lib/types.d.ts.map +1 -1
  3. package/dist/lib/utils.d.ts +5 -0
  4. package/dist/lib/utils.d.ts.map +1 -1
  5. package/dist/lib/utils.js +16 -0
  6. package/dist/lib/utils.js.map +1 -1
  7. package/dist/proxy-server.d.ts.map +1 -1
  8. package/dist/proxy-server.js +32 -19
  9. package/dist/proxy-server.js.map +1 -1
  10. package/dist/relaycast-provider-helpers.d.ts +6 -0
  11. package/dist/relaycast-provider-helpers.d.ts.map +1 -1
  12. package/dist/relaycast-provider-helpers.js +10 -9
  13. package/dist/relaycast-provider-helpers.js.map +1 -1
  14. package/dist/relaycast-provider.d.ts +3 -7
  15. package/dist/relaycast-provider.d.ts.map +1 -1
  16. package/dist/relaycast-provider.js +5 -28
  17. package/dist/relaycast-provider.js.map +1 -1
  18. package/dist/routes/channels.d.ts.map +1 -1
  19. package/dist/routes/channels.js +7 -8
  20. package/dist/routes/channels.js.map +1 -1
  21. package/dist/routes/health.d.ts.map +1 -1
  22. package/dist/routes/health.js +2 -3
  23. package/dist/routes/health.js.map +1 -1
  24. package/dist/routes/models.d.ts.map +1 -1
  25. package/dist/routes/models.js +39 -3
  26. package/dist/routes/models.js.map +1 -1
  27. package/dist/routes/relay-config.d.ts.map +1 -1
  28. package/dist/routes/relay-config.js +19 -6
  29. package/dist/routes/relay-config.js.map +1 -1
  30. package/dist/routes/thread-replies.d.ts.map +1 -1
  31. package/dist/routes/thread-replies.js +4 -4
  32. package/dist/routes/thread-replies.js.map +1 -1
  33. package/dist/services/index.d.ts +0 -1
  34. package/dist/services/index.d.ts.map +1 -1
  35. package/dist/services/index.js +0 -1
  36. package/dist/services/index.js.map +1 -1
  37. package/dist/types/index.d.ts +0 -75
  38. package/dist/types/index.d.ts.map +1 -1
  39. package/dist/websocket/logs.d.ts +1 -25
  40. package/dist/websocket/logs.d.ts.map +1 -1
  41. package/dist/websocket/logs.js +0 -397
  42. package/dist/websocket/logs.js.map +1 -1
  43. package/out/404.html +1 -1
  44. package/out/_next/static/chunks/{3663-47290254b8f6f5dd.js → 3663-191a9aa9104061af.js} +1 -1
  45. package/out/_next/static/chunks/4201-d11188cf35739bff.js +1 -0
  46. package/out/_next/static/chunks/5787-f6fd7a3fbfe5eae6.js +73 -0
  47. package/out/_next/static/chunks/app/app/[[...slug]]/{page-c1376e695ba19e38.js → page-2b872f60e3d64014.js} +1 -1
  48. package/out/_next/static/chunks/app/{page-f2ebc7d0bc08e395.js → page-ca5511e5d65100a5.js} +1 -1
  49. package/out/about.html +1 -1
  50. package/out/about.txt +1 -1
  51. package/out/app/onboarding.html +1 -1
  52. package/out/app/onboarding.txt +1 -1
  53. package/out/app.html +1 -1
  54. package/out/app.txt +2 -2
  55. package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +1 -1
  56. package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
  57. package/out/blog/let-them-cook-multi-agent-orchestration.html +1 -1
  58. package/out/blog/let-them-cook-multi-agent-orchestration.txt +1 -1
  59. package/out/blog.html +1 -1
  60. package/out/blog.txt +1 -1
  61. package/out/careers.html +1 -1
  62. package/out/careers.txt +1 -1
  63. package/out/changelog.html +1 -1
  64. package/out/changelog.txt +1 -1
  65. package/out/cloud/link.html +1 -1
  66. package/out/cloud/link.txt +1 -1
  67. package/out/complete-profile.html +1 -1
  68. package/out/complete-profile.txt +1 -1
  69. package/out/connect-repos.html +1 -1
  70. package/out/connect-repos.txt +1 -1
  71. package/out/contact.html +1 -1
  72. package/out/contact.txt +1 -1
  73. package/out/dev/cli-tools.html +1 -1
  74. package/out/dev/cli-tools.txt +1 -1
  75. package/out/dev/log-viewer.html +1 -1
  76. package/out/dev/log-viewer.txt +1 -1
  77. package/out/docs.html +1 -1
  78. package/out/docs.txt +1 -1
  79. package/out/history.html +1 -1
  80. package/out/history.txt +2 -2
  81. package/out/index.html +1 -1
  82. package/out/index.txt +2 -2
  83. package/out/login.html +1 -1
  84. package/out/login.txt +1 -1
  85. package/out/metrics.html +1 -1
  86. package/out/metrics.txt +2 -2
  87. package/out/pricing.html +1 -1
  88. package/out/pricing.txt +1 -1
  89. package/out/privacy.html +1 -1
  90. package/out/privacy.txt +1 -1
  91. package/out/providers/setup/claude.html +1 -1
  92. package/out/providers/setup/claude.txt +1 -1
  93. package/out/providers/setup/codex.html +1 -1
  94. package/out/providers/setup/codex.txt +1 -1
  95. package/out/providers/setup/cursor.html +1 -1
  96. package/out/providers/setup/cursor.txt +1 -1
  97. package/out/providers.html +1 -1
  98. package/out/providers.txt +1 -1
  99. package/out/security.html +1 -1
  100. package/out/security.txt +1 -1
  101. package/out/signup.html +1 -1
  102. package/out/signup.txt +1 -1
  103. package/out/terms.html +1 -1
  104. package/out/terms.txt +1 -1
  105. package/package.json +6 -9
  106. package/dist/lib/attachment-storage.d.ts +0 -10
  107. package/dist/lib/attachment-storage.d.ts.map +0 -1
  108. package/dist/lib/attachment-storage.js +0 -53
  109. package/dist/lib/attachment-storage.js.map +0 -1
  110. package/dist/lib/broadcast.d.ts +0 -18
  111. package/dist/lib/broadcast.d.ts.map +0 -1
  112. package/dist/lib/broadcast.js +0 -118
  113. package/dist/lib/broadcast.js.map +0 -1
  114. package/dist/lib/channel-state.d.ts +0 -32
  115. package/dist/lib/channel-state.d.ts.map +0 -1
  116. package/dist/lib/channel-state.js +0 -146
  117. package/dist/lib/channel-state.js.map +0 -1
  118. package/dist/lib/cli-auth.d.ts +0 -40
  119. package/dist/lib/cli-auth.d.ts.map +0 -1
  120. package/dist/lib/cli-auth.js +0 -144
  121. package/dist/lib/cli-auth.js.map +0 -1
  122. package/dist/lib/data-assembly.d.ts +0 -136
  123. package/dist/lib/data-assembly.d.ts.map +0 -1
  124. package/dist/lib/data-assembly.js +0 -550
  125. package/dist/lib/data-assembly.js.map +0 -1
  126. package/dist/lib/server-state.d.ts +0 -115
  127. package/dist/lib/server-state.d.ts.map +0 -1
  128. package/dist/lib/server-state.js +0 -138
  129. package/dist/lib/server-state.js.map +0 -1
  130. package/dist/lib/websocket-runtime.d.ts +0 -17
  131. package/dist/lib/websocket-runtime.d.ts.map +0 -1
  132. package/dist/lib/websocket-runtime.js +0 -76
  133. package/dist/lib/websocket-runtime.js.map +0 -1
  134. package/dist/routes/auth.d.ts +0 -45
  135. package/dist/routes/auth.d.ts.map +0 -1
  136. package/dist/routes/auth.js +0 -261
  137. package/dist/routes/auth.js.map +0 -1
  138. package/dist/routes/channels-integrated.d.ts +0 -84
  139. package/dist/routes/channels-integrated.d.ts.map +0 -1
  140. package/dist/routes/channels-integrated.js +0 -644
  141. package/dist/routes/channels-integrated.js.map +0 -1
  142. package/dist/routes/decisions.d.ts +0 -31
  143. package/dist/routes/decisions.d.ts.map +0 -1
  144. package/dist/routes/decisions.js +0 -109
  145. package/dist/routes/decisions.js.map +0 -1
  146. package/dist/routes/fleet.d.ts +0 -24
  147. package/dist/routes/fleet.d.ts.map +0 -1
  148. package/dist/routes/fleet.js +0 -131
  149. package/dist/routes/fleet.js.map +0 -1
  150. package/dist/routes/history.d.ts +0 -13
  151. package/dist/routes/history.d.ts.map +0 -1
  152. package/dist/routes/history.js +0 -205
  153. package/dist/routes/history.js.map +0 -1
  154. package/dist/routes/messaging.d.ts +0 -34
  155. package/dist/routes/messaging.d.ts.map +0 -1
  156. package/dist/routes/messaging.js +0 -306
  157. package/dist/routes/messaging.js.map +0 -1
  158. package/dist/routes/settings.d.ts +0 -6
  159. package/dist/routes/settings.d.ts.map +0 -1
  160. package/dist/routes/settings.js +0 -119
  161. package/dist/routes/settings.js.map +0 -1
  162. package/dist/routes/spawn.d.ts +0 -75
  163. package/dist/routes/spawn.d.ts.map +0 -1
  164. package/dist/routes/spawn.js +0 -521
  165. package/dist/routes/spawn.js.map +0 -1
  166. package/dist/routes/system.d.ts +0 -21
  167. package/dist/routes/system.d.ts.map +0 -1
  168. package/dist/routes/system.js +0 -186
  169. package/dist/routes/system.js.map +0 -1
  170. package/dist/routes/tasks.d.ts +0 -27
  171. package/dist/routes/tasks.d.ts.map +0 -1
  172. package/dist/routes/tasks.js +0 -103
  173. package/dist/routes/tasks.js.map +0 -1
  174. package/dist/routes/ui.d.ts +0 -6
  175. package/dist/routes/ui.d.ts.map +0 -1
  176. package/dist/routes/ui.js +0 -114
  177. package/dist/routes/ui.js.map +0 -1
  178. package/dist/server.d.ts +0 -4
  179. package/dist/server.d.ts.map +0 -1
  180. package/dist/server.js +0 -481
  181. package/dist/server.js.map +0 -1
  182. package/dist/services/broker-spawn-reader.d.ts +0 -40
  183. package/dist/services/broker-spawn-reader.d.ts.map +0 -1
  184. package/dist/services/broker-spawn-reader.js +0 -155
  185. package/dist/services/broker-spawn-reader.js.map +0 -1
  186. package/dist/services/user-bridge.d.ts +0 -155
  187. package/dist/services/user-bridge.d.ts.map +0 -1
  188. package/dist/services/user-bridge.js +0 -385
  189. package/dist/services/user-bridge.js.map +0 -1
  190. package/dist/websocket/bridge.d.ts +0 -12
  191. package/dist/websocket/bridge.d.ts.map +0 -1
  192. package/dist/websocket/bridge.js +0 -33
  193. package/dist/websocket/bridge.js.map +0 -1
  194. package/dist/websocket/main.d.ts +0 -15
  195. package/dist/websocket/main.d.ts.map +0 -1
  196. package/dist/websocket/main.js +0 -84
  197. package/dist/websocket/main.js.map +0 -1
  198. package/dist/websocket/presence.d.ts +0 -74
  199. package/dist/websocket/presence.d.ts.map +0 -1
  200. package/dist/websocket/presence.js +0 -330
  201. package/dist/websocket/presence.js.map +0 -1
  202. package/out/_next/static/chunks/270-8c0b8109123a0c5f.js +0 -73
  203. package/out/_next/static/chunks/5518-6d77237eefc8d5ae.js +0 -1
  204. /package/out/_next/static/{5cqIVzlh9DbJT28EbNrcC → wgSCX8AQnjVl_Rnx1gUp5}/_buildManifest.js +0 -0
  205. /package/out/_next/static/{5cqIVzlh9DbJT28EbNrcC → wgSCX8AQnjVl_Rnx1gUp5}/_ssgManifest.js +0 -0
@@ -1,385 +0,0 @@
1
- /**
2
- * User Bridge - Bridges dashboard WebSocket users to the relay daemon.
3
- *
4
- * This module allows human users connected via WebSocket to:
5
- * - Register as "user" entities in the relay daemon
6
- * - Join/leave channels
7
- * - Send/receive messages through the relay daemon
8
- * - Communicate with agents and other users
9
- */
10
- /**
11
- * UserBridge manages the connection between dashboard WebSocket users
12
- * and the relay daemon.
13
- */
14
- export class UserBridge {
15
- createRelayClient;
16
- loadPersistedChannels;
17
- lookupUserInfo;
18
- users = new Map();
19
- constructor(options) {
20
- this.createRelayClient = options.createRelayClient;
21
- this.loadPersistedChannels = options.loadPersistedChannels;
22
- this.lookupUserInfo = options.lookupUserInfo;
23
- }
24
- /**
25
- * Get the relay client for a user if they are registered.
26
- * This allows external code to reuse the userBridge's relay client
27
- * instead of creating a duplicate connection.
28
- */
29
- getRelayClient(username) {
30
- return this.users.get(username)?.relayClient;
31
- }
32
- /**
33
- * Register a user with the relay daemon.
34
- * Creates a relay client connection for the user.
35
- */
36
- async registerUser(username, webSocket, options) {
37
- // If user already registered, just update the WebSocket (multi-tab support)
38
- if (this.users.has(username)) {
39
- console.log(`[user-bridge] User ${username} already registered, updating WebSocket`);
40
- this.updateWebSocket(username, webSocket);
41
- return;
42
- }
43
- // Create relay client for this user
44
- const relayClient = await this.createRelayClient({
45
- agentName: username,
46
- entityType: 'user',
47
- displayName: options?.displayName,
48
- avatarUrl: options?.avatarUrl,
49
- });
50
- // Connect to daemon
51
- await relayClient.connect();
52
- // Set up message handler to forward direct messages to WebSocket
53
- relayClient.onMessage = (from, payload, _messageId, _meta, _originalTo) => {
54
- const body = typeof payload === 'object' && payload !== null && 'body' in payload
55
- ? payload.body
56
- : String(payload);
57
- this.handleIncomingDirectMessage(username, from, body, payload);
58
- };
59
- // Set up channel message handler to forward channel messages to WebSocket
60
- relayClient.onChannelMessage = (from, channel, body, envelope) => {
61
- this.handleIncomingChannelMessage(username, from, channel, body, envelope);
62
- };
63
- // Create session with WebSocket set for multi-tab support
64
- const session = {
65
- username,
66
- relayClient,
67
- webSockets: new Set([webSocket]),
68
- channels: new Set(),
69
- avatarUrl: options?.avatarUrl,
70
- };
71
- this.users.set(username, session);
72
- console.log(`[user-bridge] User registered: ${username} (total users: ${this.users.size}, connections: 1)`);
73
- // Auto-join user to #general channel
74
- // Note: The daemon auto-joins on connect, but we need to track locally too
75
- session.channels.add('#general');
76
- if (this.loadPersistedChannels) {
77
- try {
78
- const persistedChannels = await this.loadPersistedChannels(username);
79
- for (const channel of persistedChannels) {
80
- if (channel === '#general')
81
- continue;
82
- if (session.channels.has(channel))
83
- continue;
84
- session.relayClient.joinChannel(channel, username);
85
- session.channels.add(channel);
86
- }
87
- }
88
- catch (err) {
89
- console.error(`[user-bridge] Failed to restore persisted channels for ${username}:`, err);
90
- }
91
- }
92
- // Set up WebSocket close handler to remove this specific connection
93
- webSocket.on('close', () => {
94
- this.removeWebSocket(username, webSocket);
95
- });
96
- console.log(`[user-bridge] User ${username} registered with relay daemon`);
97
- }
98
- /**
99
- * Remove a specific WebSocket connection from a user's session.
100
- * If this was the last connection, unregister the user entirely.
101
- */
102
- removeWebSocket(username, webSocket) {
103
- const session = this.users.get(username);
104
- if (!session)
105
- return;
106
- session.webSockets.delete(webSocket);
107
- console.log(`[user-bridge] WebSocket closed for ${username} (${session.webSockets.size} connections remaining)`);
108
- // Only unregister if ALL connections are closed
109
- if (session.webSockets.size === 0) {
110
- this.unregisterUser(username);
111
- }
112
- }
113
- /**
114
- * Unregister a user and disconnect their relay client.
115
- */
116
- unregisterUser(username) {
117
- const session = this.users.get(username);
118
- if (!session)
119
- return;
120
- session.relayClient.disconnect();
121
- this.users.delete(username);
122
- console.log(`[user-bridge] User ${username} unregistered from relay daemon`);
123
- }
124
- /**
125
- * Check if a user is registered.
126
- */
127
- isUserRegistered(username) {
128
- return this.users.has(username);
129
- }
130
- /**
131
- * Add a new WebSocket connection for an existing user session.
132
- * This is needed when a user opens a new tab.
133
- */
134
- updateWebSocket(username, newWebSocket) {
135
- const session = this.users.get(username);
136
- if (!session) {
137
- console.log(`[user-bridge] Cannot add WebSocket - user ${username} not registered`);
138
- return false;
139
- }
140
- // Add the new WebSocket to the set
141
- session.webSockets.add(newWebSocket);
142
- // Set up close handler to remove this specific connection
143
- newWebSocket.on('close', () => {
144
- this.removeWebSocket(username, newWebSocket);
145
- });
146
- console.log(`[user-bridge] Added WebSocket for user ${username} (${session.webSockets.size} connections)`);
147
- return true;
148
- }
149
- /**
150
- * Get list of all registered users.
151
- */
152
- getRegisteredUsers() {
153
- return Array.from(this.users.keys());
154
- }
155
- /**
156
- * Join a channel.
157
- */
158
- async joinChannel(username, channel) {
159
- const session = this.users.get(username);
160
- if (!session) {
161
- console.warn(`[user-bridge] Cannot join channel - user ${username} not registered`);
162
- return false;
163
- }
164
- // Send CHANNEL_JOIN via relay client
165
- const success = session.relayClient.joinChannel(channel, username);
166
- if (success) {
167
- // Track membership
168
- session.channels.add(channel);
169
- }
170
- return success;
171
- }
172
- /**
173
- * Leave a channel.
174
- */
175
- async leaveChannel(username, channel) {
176
- const session = this.users.get(username);
177
- if (!session) {
178
- console.warn(`[user-bridge] Cannot leave channel - user ${username} not registered`);
179
- return false;
180
- }
181
- // Send CHANNEL_LEAVE via relay client
182
- const success = session.relayClient.leaveChannel(channel);
183
- if (success) {
184
- // Update membership
185
- session.channels.delete(channel);
186
- console.log(`[user-bridge] User ${username} left channel ${channel}`);
187
- }
188
- return success;
189
- }
190
- /**
191
- * Get channels a user has joined.
192
- */
193
- getUserChannels(username) {
194
- const session = this.users.get(username);
195
- return session ? Array.from(session.channels) : [];
196
- }
197
- /**
198
- * Send a message to a channel.
199
- */
200
- async sendChannelMessage(username, channel, body, options) {
201
- const session = this.users.get(username);
202
- if (!session) {
203
- console.warn(`[user-bridge] Cannot send - user ${username} not registered`);
204
- return false;
205
- }
206
- return session.relayClient.sendChannelMessage(channel, body, {
207
- thread: options?.thread,
208
- data: options?.data,
209
- attachments: options?.attachments,
210
- });
211
- }
212
- /**
213
- * Send a direct message to another user or agent.
214
- */
215
- async sendDirectMessage(fromUsername, toName, body, options) {
216
- const session = this.users.get(fromUsername);
217
- if (!session) {
218
- console.warn(`[user-bridge] Cannot send DM - user ${fromUsername} not registered`);
219
- return false;
220
- }
221
- return session.relayClient.sendMessage(toName, body, 'message', options?.data, options?.thread);
222
- }
223
- /**
224
- * Handle incoming direct message from relay daemon.
225
- */
226
- handleIncomingDirectMessage(username, from, body, payload) {
227
- // Skip channel messages - they are handled by handleIncomingChannelMessage
228
- // The relay client calls both onMessage and onChannelMessage for channel messages,
229
- // with _isChannelMessage flag set in the data for onMessage calls
230
- const payloadObj = payload;
231
- if (payloadObj?.data?._isChannelMessage) {
232
- return; // Skip - will be handled by onChannelMessage callback
233
- }
234
- const session = this.users.get(username);
235
- if (!session)
236
- return;
237
- // Look up sender's avatar if lookup function is available
238
- const senderInfo = this.lookupUserInfo?.(from);
239
- const fromAvatarUrl = senderInfo?.avatarUrl;
240
- // Determine entity type: user if they have info, agent otherwise
241
- const fromEntityType = senderInfo ? 'user' : 'agent';
242
- const message = JSON.stringify({
243
- type: 'direct_message',
244
- from,
245
- fromAvatarUrl,
246
- fromEntityType,
247
- body: payloadObj?.body || body,
248
- timestamp: new Date().toISOString(),
249
- });
250
- // Send to ALL open WebSocket connections (multi-tab support)
251
- for (const ws of session.webSockets) {
252
- if (ws.readyState === 1) { // OPEN
253
- ws.send(message);
254
- }
255
- }
256
- }
257
- /**
258
- * Handle incoming channel message from relay daemon.
259
- */
260
- handleIncomingChannelMessage(username, from, channel, body, envelope) {
261
- const session = this.users.get(username);
262
- if (!session)
263
- return;
264
- // Look up sender's avatar if lookup function is available
265
- const senderInfo = this.lookupUserInfo?.(from);
266
- const fromAvatarUrl = senderInfo?.avatarUrl;
267
- // Determine entity type: user if they have info, agent otherwise
268
- const fromEntityType = senderInfo ? 'user' : 'agent';
269
- // Channel message
270
- const env = envelope;
271
- const message = JSON.stringify({
272
- type: 'channel_message',
273
- channel,
274
- from,
275
- fromAvatarUrl,
276
- fromEntityType,
277
- body,
278
- thread: env?.payload?.thread,
279
- mentions: env?.payload?.mentions,
280
- timestamp: new Date().toISOString(),
281
- });
282
- // Send to ALL open WebSocket connections (multi-tab support)
283
- for (const ws of session.webSockets) {
284
- if (ws.readyState === 1) { // OPEN
285
- ws.send(message);
286
- }
287
- }
288
- }
289
- /**
290
- * Admin: Add a member to a channel (does not require member to be connected).
291
- * Used to sync channel memberships from database.
292
- * Uses the first available user session or creates a temporary one.
293
- */
294
- async adminJoinChannel(channel, member) {
295
- // Try to use an existing session
296
- const sessions = Array.from(this.users.values());
297
- if (sessions.length > 0) {
298
- const session = sessions[0];
299
- if (session.relayClient.adminJoinChannel) {
300
- console.log(`[user-bridge] Admin join: ${member} -> ${channel} (via ${session.username})`);
301
- return session.relayClient.adminJoinChannel(channel, member);
302
- }
303
- }
304
- // No sessions available - create a temporary system client
305
- try {
306
- console.log(`[user-bridge] Admin join: ${member} -> ${channel} (creating temp client)`);
307
- const tempClient = await this.createRelayClient({
308
- agentName: '__system__',
309
- entityType: 'user',
310
- });
311
- await tempClient.connect();
312
- // Give daemon time to complete handshake before sending admin commands.
313
- // 100ms is sufficient for local Unix socket handshake (typically <10ms),
314
- // but provides margin for the daemon to process the HELLO message and
315
- // set up internal state. This is a temporary client created just for
316
- // the admin operation, not a long-lived session.
317
- await new Promise(resolve => setTimeout(resolve, 100));
318
- if (tempClient.adminJoinChannel) {
319
- const result = tempClient.adminJoinChannel(channel, member);
320
- // Disconnect after a short delay to allow message to be sent
321
- setTimeout(() => tempClient.disconnect(), 200);
322
- return result;
323
- }
324
- tempClient.disconnect();
325
- return false;
326
- }
327
- catch (err) {
328
- console.error('[user-bridge] Failed to create temp client for admin join:', err);
329
- return false;
330
- }
331
- }
332
- /**
333
- * Admin: Remove a member from a channel (does not require member to be connected).
334
- * Used to remove channel members from dashboard.
335
- * Uses the first available user session or creates a temporary one.
336
- */
337
- async adminRemoveMember(channel, member) {
338
- // Try to use an existing session
339
- const sessions = Array.from(this.users.values());
340
- if (sessions.length > 0) {
341
- const session = sessions[0];
342
- if (session.relayClient.adminRemoveMember) {
343
- console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (via ${session.username})`);
344
- return session.relayClient.adminRemoveMember(channel, member);
345
- }
346
- }
347
- // No sessions available - create a temporary system client
348
- try {
349
- console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (creating temp client)`);
350
- const tempClient = await this.createRelayClient({
351
- agentName: '__system__',
352
- entityType: 'user',
353
- });
354
- await tempClient.connect();
355
- // Give daemon time to complete handshake before sending admin commands.
356
- // 100ms is sufficient for local Unix socket handshake (typically <10ms),
357
- // but provides margin for the daemon to process the HELLO message and
358
- // set up internal state. This is a temporary client created just for
359
- // the admin operation, not a long-lived session.
360
- await new Promise(resolve => setTimeout(resolve, 100));
361
- if (tempClient.adminRemoveMember) {
362
- const result = tempClient.adminRemoveMember(channel, member);
363
- // Disconnect after a short delay to allow message to be sent
364
- setTimeout(() => tempClient.disconnect(), 200);
365
- return result;
366
- }
367
- tempClient.disconnect();
368
- return false;
369
- }
370
- catch (err) {
371
- console.error('[user-bridge] Failed to create temp client for admin remove:', err);
372
- return false;
373
- }
374
- }
375
- /**
376
- * Dispose of all user sessions.
377
- */
378
- dispose() {
379
- for (const [username] of this.users) {
380
- this.unregisterUser(username);
381
- }
382
- console.log('[user-bridge] Disposed all user sessions');
383
- }
384
- }
385
- //# sourceMappingURL=user-bridge.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"user-bridge.js","sourceRoot":"","sources":["../../src/services/user-bridge.ts"],"names":[],"mappings":"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"}
@@ -1,12 +0,0 @@
1
- import { WebSocket, type WebSocketServer } from 'ws';
2
- export interface BridgeWebSocketDeps {
3
- wssBridge: WebSocketServer;
4
- bridgeClientAlive: WeakMap<WebSocket, boolean>;
5
- getBridgeData: () => Promise<unknown>;
6
- debug: (message: string) => void;
7
- }
8
- /**
9
- * Bridge WebSocket handler for cross-project dashboard state.
10
- */
11
- export declare function setupBridgeWebSocket(deps: BridgeWebSocketDeps): void;
12
- //# sourceMappingURL=bridge.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"bridge.d.ts","sourceRoot":"","sources":["../../src/websocket/bridge.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,eAAe,EAAE,MAAM,IAAI,CAAC;AAErD,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,eAAe,CAAC;IAC3B,iBAAiB,EAAE,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IAC/C,aAAa,EAAE,MAAM,OAAO,CAAC,OAAO,CAAC,CAAC;IACtC,KAAK,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC;CAClC;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,mBAAmB,GAAG,IAAI,CAgCpE"}
@@ -1,33 +0,0 @@
1
- import { WebSocket } from 'ws';
2
- /**
3
- * Bridge WebSocket handler for cross-project dashboard state.
4
- */
5
- export function setupBridgeWebSocket(deps) {
6
- const { wssBridge, bridgeClientAlive, getBridgeData, debug } = deps;
7
- wssBridge.on('connection', async (ws) => {
8
- debug('[dashboard] Bridge WebSocket client connected');
9
- // Mark client as alive initially for ping/pong keepalive.
10
- bridgeClientAlive.set(ws, true);
11
- // Handle pong responses (keep connection alive).
12
- ws.on('pong', () => {
13
- bridgeClientAlive.set(ws, true);
14
- });
15
- try {
16
- const data = await getBridgeData();
17
- const payload = JSON.stringify(data);
18
- if (ws.readyState === WebSocket.OPEN) {
19
- ws.send(payload);
20
- }
21
- }
22
- catch (err) {
23
- console.error('[dashboard] Failed to send initial bridge data:', err);
24
- }
25
- ws.on('error', (err) => {
26
- console.error('[dashboard] Bridge WebSocket client error:', err);
27
- });
28
- ws.on('close', (code, reason) => {
29
- debug(`[dashboard] Bridge WebSocket client disconnected, code: ${code}, reason: ${reason?.toString() || 'none'}`);
30
- });
31
- });
32
- }
33
- //# sourceMappingURL=bridge.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"bridge.js","sourceRoot":"","sources":["../../src/websocket/bridge.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAwB,MAAM,IAAI,CAAC;AASrD;;GAEG;AACH,MAAM,UAAU,oBAAoB,CAAC,IAAyB;IAC5D,MAAM,EAAE,SAAS,EAAE,iBAAiB,EAAE,aAAa,EAAE,KAAK,EAAE,GAAG,IAAI,CAAC;IAEpE,SAAS,CAAC,EAAE,CAAC,YAAY,EAAE,KAAK,EAAE,EAAE,EAAE,EAAE;QACtC,KAAK,CAAC,+CAA+C,CAAC,CAAC;QAEvD,0DAA0D;QAC1D,iBAAiB,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAEhC,iDAAiD;QACjD,EAAE,CAAC,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE;YACjB,iBAAiB,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC;YACH,MAAM,IAAI,GAAG,MAAM,aAAa,EAAE,CAAC;YACnC,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YACrC,IAAI,EAAE,CAAC,UAAU,KAAK,SAAS,CAAC,IAAI,EAAE,CAAC;gBACrC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACnB,CAAC;QACH,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,CAAC,KAAK,CAAC,iDAAiD,EAAE,GAAG,CAAC,CAAC;QACxE,CAAC;QAED,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YACrB,OAAO,CAAC,KAAK,CAAC,4CAA4C,EAAE,GAAG,CAAC,CAAC;QACnE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE;YAC9B,KAAK,CAAC,2DAA2D,IAAI,aAAa,MAAM,EAAE,QAAQ,EAAE,IAAI,MAAM,EAAE,CAAC,CAAC;QACpH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -1,15 +0,0 @@
1
- import { WebSocket, type WebSocketServer } from 'ws';
2
- import type { MessageBuffer } from '../messageBuffer.js';
3
- export interface MainWebSocketDeps {
4
- wss: WebSocketServer;
5
- mainClientAlive: WeakMap<WebSocket, boolean>;
6
- mainMessageBuffer: MessageBuffer;
7
- initializingClients: WeakSet<WebSocket>;
8
- getAllData: () => Promise<unknown>;
9
- debug: (message: string) => void;
10
- }
11
- /**
12
- * Main dashboard WebSocket handler: initial snapshot + replay support.
13
- */
14
- export declare function setupMainWebSocket(deps: MainWebSocketDeps): void;
15
- //# sourceMappingURL=main.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"main.d.ts","sourceRoot":"","sources":["../../src/websocket/main.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,eAAe,EAAE,MAAM,IAAI,CAAC;AACrD,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAEzD,MAAM,WAAW,iBAAiB;IAChC,GAAG,EAAE,eAAe,CAAC;IACrB,eAAe,EAAE,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IAC7C,iBAAiB,EAAE,aAAa,CAAC;IACjC,mBAAmB,EAAE,OAAO,CAAC,SAAS,CAAC,CAAC;IACxC,UAAU,EAAE,MAAM,OAAO,CAAC,OAAO,CAAC,CAAC;IACnC,KAAK,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC;CAClC;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,iBAAiB,GAAG,IAAI,CA6FhE"}
@@ -1,84 +0,0 @@
1
- import { WebSocket } from 'ws';
2
- /**
3
- * Main dashboard WebSocket handler: initial snapshot + replay support.
4
- */
5
- export function setupMainWebSocket(deps) {
6
- const { wss, mainClientAlive, mainMessageBuffer, initializingClients, getAllData, debug, } = deps;
7
- wss.on('connection', async (ws, req) => {
8
- debug(`[dashboard] WebSocket client connected from: ${req.socket.remoteAddress}`);
9
- // Mark client as alive initially for ping/pong keepalive.
10
- mainClientAlive.set(ws, true);
11
- // Handle pong responses (keep connection alive).
12
- ws.on('pong', () => {
13
- mainClientAlive.set(ws, true);
14
- });
15
- // Send current sequence ID so client can track its position.
16
- if (ws.readyState === WebSocket.OPEN) {
17
- ws.send(JSON.stringify({ type: 'sync', sequenceId: mainMessageBuffer.currentId() }));
18
- }
19
- // Mark as initializing to prevent broadcaster races.
20
- initializingClients.add(ws);
21
- try {
22
- const data = await getAllData();
23
- if (!data) {
24
- // Team data temporarily unavailable — defer initial send.
25
- debug('[dashboard] Team data unavailable on connect - deferring initial send to next broadcast');
26
- }
27
- else if (ws.readyState === WebSocket.OPEN) {
28
- const payload = JSON.stringify(data);
29
- debug(`[dashboard] Sending initial data, size: ${payload.length}, first 200 chars: ${payload.substring(0, 200)}`);
30
- ws.send(payload);
31
- debug('[dashboard] Initial data sent successfully');
32
- }
33
- else {
34
- console.warn('[dashboard] WebSocket not open, state:', ws.readyState);
35
- }
36
- }
37
- catch (err) {
38
- console.error('[dashboard] Failed to send initial data:', err);
39
- }
40
- finally {
41
- // Now allow broadcastData to send to this client.
42
- initializingClients.delete(ws);
43
- }
44
- // Handle replay requests.
45
- ws.on('message', (data) => {
46
- try {
47
- const msg = JSON.parse(data.toString());
48
- // Handle replay request: { type: "replay", lastSequenceId: N }.
49
- if (msg.type === 'replay' && typeof msg.lastSequenceId === 'number') {
50
- const missed = mainMessageBuffer.getAfter(msg.lastSequenceId);
51
- const gapMs = missed.length > 0 ? Date.now() - missed[0].timestamp : 0;
52
- console.log(`[dashboard] Client replaying ${missed.length} missed messages (gap: ${gapMs}ms)`);
53
- // Send each missed message with its original sequence ID.
54
- for (const buffered of missed) {
55
- if (ws.readyState === WebSocket.OPEN) {
56
- try {
57
- const original = JSON.parse(buffered.payload);
58
- ws.send(JSON.stringify({ seq: buffered.id, ...original }));
59
- }
60
- catch (err) {
61
- console.error('[dashboard] Failed to replay message:', err);
62
- }
63
- }
64
- }
65
- // Send current sync position after replay.
66
- if (ws.readyState === WebSocket.OPEN) {
67
- ws.send(JSON.stringify({ type: 'sync', sequenceId: mainMessageBuffer.currentId() }));
68
- }
69
- }
70
- }
71
- catch (err) {
72
- // Non-JSON messages are ignored (binary, etc.).
73
- debug(`[dashboard] Unhandled main WebSocket message: ${err}`);
74
- }
75
- });
76
- ws.on('error', (err) => {
77
- console.error('[dashboard] WebSocket client error:', err);
78
- });
79
- ws.on('close', (code, reason) => {
80
- debug(`[dashboard] WebSocket client disconnected, code: ${code}, reason: ${reason?.toString() || 'none'}`);
81
- });
82
- });
83
- }
84
- //# sourceMappingURL=main.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"main.js","sourceRoot":"","sources":["../../src/websocket/main.ts"],"names":[],"mappings":"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"}
@@ -1,74 +0,0 @@
1
- import { WebSocket, type WebSocketServer } from 'ws';
2
- import type { MessageBuffer } from '../messageBuffer.js';
3
- export interface UserPresenceInfo {
4
- username: string;
5
- avatarUrl?: string;
6
- connectedAt: string;
7
- lastSeen: string;
8
- }
9
- export interface UserPresenceState {
10
- info: UserPresenceInfo;
11
- connections: Set<WebSocket>;
12
- }
13
- export interface ChannelMessageEvent {
14
- type: 'channel_message';
15
- targetUser: string;
16
- channel: string;
17
- from: string;
18
- fromAvatarUrl?: string;
19
- fromEntityType?: 'user' | 'agent';
20
- body: string;
21
- thread?: string;
22
- mentions?: string[];
23
- timestamp: string;
24
- }
25
- export interface DirectMessageEvent {
26
- type: 'direct_message';
27
- targetUser: string;
28
- from: string;
29
- fromAvatarUrl?: string;
30
- fromEntityType?: 'user' | 'agent';
31
- body: string;
32
- id: string;
33
- messageId: string;
34
- timestamp: string;
35
- }
36
- interface UserBridgeLike {
37
- updateWebSocket: (username: string, ws: WebSocket) => void;
38
- registerUser: (username: string, ws: WebSocket, metadata?: {
39
- avatarUrl?: string;
40
- }) => Promise<void>;
41
- unregisterUser: (username: string) => void;
42
- joinChannel: (username: string, channel: string) => Promise<boolean>;
43
- leaveChannel: (username: string, channel: string) => Promise<boolean>;
44
- sendChannelMessage: (username: string, channel: string, body: string, options?: {
45
- thread?: string;
46
- }) => Promise<boolean>;
47
- sendDirectMessage: (from: string, to: string, body: string, options?: {
48
- thread?: string;
49
- }) => Promise<boolean>;
50
- }
51
- export interface PresenceWebSocketDeps {
52
- wss: WebSocketServer;
53
- wssPresence: WebSocketServer;
54
- mainMessageBuffer: MessageBuffer;
55
- onlineUsers: Map<string, UserPresenceState>;
56
- presenceHealth: WeakMap<WebSocket, {
57
- isAlive: boolean;
58
- }>;
59
- broadcastPresence: (message: object, exclude?: WebSocket) => void;
60
- isValidUsername: (username: unknown) => username is string;
61
- isValidAvatarUrl: (avatarUrl: unknown) => avatarUrl is string | undefined;
62
- getUserBridge: () => UserBridgeLike | undefined;
63
- debug: (message: string) => void;
64
- }
65
- export interface PresenceSetupResult {
66
- broadcastChannelMessage: (message: ChannelMessageEvent) => void;
67
- broadcastDirectMessage: (message: DirectMessageEvent) => void;
68
- }
69
- /**
70
- * Presence WebSocket handler and channel/direct message broadcasters.
71
- */
72
- export declare function setupPresenceWebSocket(deps: PresenceWebSocketDeps): PresenceSetupResult;
73
- export {};
74
- //# sourceMappingURL=presence.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"presence.d.ts","sourceRoot":"","sources":["../../src/websocket/presence.ts"],"names":[],"mappings":"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"}