@agent-relay/dashboard-server 0.1.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.
@@ -0,0 +1,372 @@
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
+ socketPath;
16
+ createRelayClient;
17
+ loadPersistedChannels;
18
+ lookupUserInfo;
19
+ users = new Map();
20
+ constructor(options) {
21
+ this.socketPath = options.socketPath;
22
+ this.createRelayClient = options.createRelayClient;
23
+ this.loadPersistedChannels = options.loadPersistedChannels;
24
+ this.lookupUserInfo = options.lookupUserInfo;
25
+ }
26
+ /**
27
+ * Get the relay client for a user if they are registered.
28
+ * This allows external code to reuse the userBridge's relay client
29
+ * instead of creating a duplicate connection.
30
+ */
31
+ getRelayClient(username) {
32
+ return this.users.get(username)?.relayClient;
33
+ }
34
+ /**
35
+ * Register a user with the relay daemon.
36
+ * Creates a relay client connection for the user.
37
+ */
38
+ async registerUser(username, webSocket, options) {
39
+ // If user already registered, unregister first
40
+ if (this.users.has(username)) {
41
+ this.unregisterUser(username);
42
+ }
43
+ // Create relay client for this user
44
+ const relayClient = await this.createRelayClient({
45
+ socketPath: this.socketPath,
46
+ agentName: username,
47
+ entityType: 'user',
48
+ displayName: options?.displayName,
49
+ avatarUrl: options?.avatarUrl,
50
+ });
51
+ // Connect to daemon
52
+ await relayClient.connect();
53
+ // Set up message handler to forward direct messages to WebSocket
54
+ relayClient.onMessage = (from, payload, _messageId, _meta, _originalTo) => {
55
+ const body = typeof payload === 'object' && payload !== null && 'body' in payload
56
+ ? payload.body
57
+ : String(payload);
58
+ this.handleIncomingDirectMessage(username, from, body, payload);
59
+ };
60
+ // Set up channel message handler to forward channel messages to WebSocket
61
+ relayClient.onChannelMessage = (from, channel, body, envelope) => {
62
+ this.handleIncomingChannelMessage(username, from, channel, body, envelope);
63
+ };
64
+ // Create session
65
+ const session = {
66
+ username,
67
+ relayClient,
68
+ webSocket,
69
+ channels: new Set(),
70
+ avatarUrl: options?.avatarUrl,
71
+ };
72
+ this.users.set(username, session);
73
+ console.log(`[user-bridge] User registered: ${username} (total: ${this.users.size})`);
74
+ // Auto-join user to #general channel
75
+ // Note: The daemon auto-joins on connect, but we need to track locally too
76
+ session.channels.add('#general');
77
+ if (this.loadPersistedChannels) {
78
+ try {
79
+ const persistedChannels = await this.loadPersistedChannels(username);
80
+ for (const channel of persistedChannels) {
81
+ if (channel === '#general')
82
+ continue;
83
+ if (session.channels.has(channel))
84
+ continue;
85
+ session.relayClient.joinChannel(channel, username);
86
+ session.channels.add(channel);
87
+ }
88
+ }
89
+ catch (err) {
90
+ console.error(`[user-bridge] Failed to restore persisted channels for ${username}:`, err);
91
+ }
92
+ }
93
+ // Set up WebSocket close handler
94
+ webSocket.on('close', () => {
95
+ this.unregisterUser(username);
96
+ });
97
+ console.log(`[user-bridge] User ${username} registered with relay daemon`);
98
+ }
99
+ /**
100
+ * Unregister a user and disconnect their relay client.
101
+ */
102
+ unregisterUser(username) {
103
+ const session = this.users.get(username);
104
+ if (!session)
105
+ return;
106
+ session.relayClient.disconnect();
107
+ this.users.delete(username);
108
+ console.log(`[user-bridge] User ${username} unregistered from relay daemon`);
109
+ }
110
+ /**
111
+ * Check if a user is registered.
112
+ */
113
+ isUserRegistered(username) {
114
+ return this.users.has(username);
115
+ }
116
+ /**
117
+ * Update the WebSocket for an existing user session.
118
+ * This is needed when a user reconnects or opens a new tab,
119
+ * so messages are forwarded to the active WebSocket.
120
+ */
121
+ updateWebSocket(username, newWebSocket) {
122
+ const session = this.users.get(username);
123
+ if (!session) {
124
+ console.log(`[user-bridge] Cannot update WebSocket - user ${username} not registered`);
125
+ return false;
126
+ }
127
+ // Remove the close handler from old WebSocket to prevent auto-unregister
128
+ session.webSocket.removeAllListeners('close');
129
+ // Update to new WebSocket
130
+ session.webSocket = newWebSocket;
131
+ // Set up close handler on new WebSocket
132
+ newWebSocket.on('close', () => {
133
+ // Note: The server manages multi-tab connections and will call
134
+ // unregisterUser when all connections are closed
135
+ });
136
+ console.log(`[user-bridge] Updated WebSocket for user ${username}`);
137
+ return true;
138
+ }
139
+ /**
140
+ * Get list of all registered users.
141
+ */
142
+ getRegisteredUsers() {
143
+ return Array.from(this.users.keys());
144
+ }
145
+ /**
146
+ * Join a channel.
147
+ */
148
+ async joinChannel(username, channel) {
149
+ const session = this.users.get(username);
150
+ if (!session) {
151
+ console.warn(`[user-bridge] Cannot join channel - user ${username} not registered`);
152
+ return false;
153
+ }
154
+ // Send CHANNEL_JOIN via relay client
155
+ const success = session.relayClient.joinChannel(channel, username);
156
+ if (success) {
157
+ // Track membership
158
+ session.channels.add(channel);
159
+ }
160
+ return success;
161
+ }
162
+ /**
163
+ * Leave a channel.
164
+ */
165
+ async leaveChannel(username, channel) {
166
+ const session = this.users.get(username);
167
+ if (!session) {
168
+ console.warn(`[user-bridge] Cannot leave channel - user ${username} not registered`);
169
+ return false;
170
+ }
171
+ // Send CHANNEL_LEAVE via relay client
172
+ const success = session.relayClient.leaveChannel(channel);
173
+ if (success) {
174
+ // Update membership
175
+ session.channels.delete(channel);
176
+ console.log(`[user-bridge] User ${username} left channel ${channel}`);
177
+ }
178
+ return success;
179
+ }
180
+ /**
181
+ * Get channels a user has joined.
182
+ */
183
+ getUserChannels(username) {
184
+ const session = this.users.get(username);
185
+ return session ? Array.from(session.channels) : [];
186
+ }
187
+ /**
188
+ * Send a message to a channel.
189
+ */
190
+ async sendChannelMessage(username, channel, body, options) {
191
+ const session = this.users.get(username);
192
+ if (!session) {
193
+ console.warn(`[user-bridge] Cannot send - user ${username} not registered`);
194
+ return false;
195
+ }
196
+ return session.relayClient.sendChannelMessage(channel, body, {
197
+ thread: options?.thread,
198
+ data: options?.data,
199
+ attachments: options?.attachments,
200
+ });
201
+ }
202
+ /**
203
+ * Send a direct message to another user or agent.
204
+ */
205
+ async sendDirectMessage(fromUsername, toName, body, options) {
206
+ const session = this.users.get(fromUsername);
207
+ if (!session) {
208
+ console.warn(`[user-bridge] Cannot send DM - user ${fromUsername} not registered`);
209
+ return false;
210
+ }
211
+ return session.relayClient.sendMessage(toName, body, 'message', options?.data, options?.thread);
212
+ }
213
+ /**
214
+ * Handle incoming direct message from relay daemon.
215
+ */
216
+ handleIncomingDirectMessage(username, from, body, payload) {
217
+ // Skip channel messages - they are handled by handleIncomingChannelMessage
218
+ // The relay client calls both onMessage and onChannelMessage for channel messages,
219
+ // with _isChannelMessage flag set in the data for onMessage calls
220
+ const payloadObj = payload;
221
+ if (payloadObj?.data?._isChannelMessage) {
222
+ return; // Skip - will be handled by onChannelMessage callback
223
+ }
224
+ const session = this.users.get(username);
225
+ if (!session)
226
+ return;
227
+ const ws = session.webSocket;
228
+ if (ws.readyState !== 1)
229
+ return; // Not OPEN
230
+ // Look up sender's avatar if lookup function is available
231
+ const senderInfo = this.lookupUserInfo?.(from);
232
+ const fromAvatarUrl = senderInfo?.avatarUrl;
233
+ // Determine entity type: user if they have info, agent otherwise
234
+ const fromEntityType = senderInfo ? 'user' : 'agent';
235
+ // Direct message (DELIVER)
236
+ ws.send(JSON.stringify({
237
+ type: 'direct_message',
238
+ from,
239
+ fromAvatarUrl,
240
+ fromEntityType,
241
+ body: payloadObj?.body || body,
242
+ timestamp: new Date().toISOString(),
243
+ }));
244
+ }
245
+ /**
246
+ * Handle incoming channel message from relay daemon.
247
+ */
248
+ handleIncomingChannelMessage(username, from, channel, body, envelope) {
249
+ const session = this.users.get(username);
250
+ if (!session)
251
+ return;
252
+ const ws = session.webSocket;
253
+ if (ws.readyState !== 1)
254
+ return; // Not OPEN
255
+ // Look up sender's avatar if lookup function is available
256
+ const senderInfo = this.lookupUserInfo?.(from);
257
+ const fromAvatarUrl = senderInfo?.avatarUrl;
258
+ // Determine entity type: user if they have info, agent otherwise
259
+ const fromEntityType = senderInfo ? 'user' : 'agent';
260
+ // Channel message
261
+ const env = envelope;
262
+ ws.send(JSON.stringify({
263
+ type: 'channel_message',
264
+ channel,
265
+ from,
266
+ fromAvatarUrl,
267
+ fromEntityType,
268
+ body,
269
+ thread: env?.payload?.thread,
270
+ mentions: env?.payload?.mentions,
271
+ timestamp: new Date().toISOString(),
272
+ }));
273
+ }
274
+ /**
275
+ * Admin: Add a member to a channel (does not require member to be connected).
276
+ * Used to sync channel memberships from database.
277
+ * Uses the first available user session or creates a temporary one.
278
+ */
279
+ async adminJoinChannel(channel, member) {
280
+ // Try to use an existing session
281
+ const sessions = Array.from(this.users.values());
282
+ if (sessions.length > 0) {
283
+ const session = sessions[0];
284
+ if (session.relayClient.adminJoinChannel) {
285
+ console.log(`[user-bridge] Admin join: ${member} -> ${channel} (via ${session.username})`);
286
+ return session.relayClient.adminJoinChannel(channel, member);
287
+ }
288
+ }
289
+ // No sessions available - create a temporary system client
290
+ try {
291
+ console.log(`[user-bridge] Admin join: ${member} -> ${channel} (creating temp client)`);
292
+ const tempClient = await this.createRelayClient({
293
+ socketPath: this.socketPath,
294
+ agentName: '__system__',
295
+ entityType: 'user',
296
+ });
297
+ await tempClient.connect();
298
+ // Give daemon time to complete handshake before sending admin commands.
299
+ // 100ms is sufficient for local Unix socket handshake (typically <10ms),
300
+ // but provides margin for the daemon to process the HELLO message and
301
+ // set up internal state. This is a temporary client created just for
302
+ // the admin operation, not a long-lived session.
303
+ await new Promise(resolve => setTimeout(resolve, 100));
304
+ if (tempClient.adminJoinChannel) {
305
+ const result = tempClient.adminJoinChannel(channel, member);
306
+ // Disconnect after a short delay to allow message to be sent
307
+ setTimeout(() => tempClient.disconnect(), 200);
308
+ return result;
309
+ }
310
+ tempClient.disconnect();
311
+ return false;
312
+ }
313
+ catch (err) {
314
+ console.error('[user-bridge] Failed to create temp client for admin join:', err);
315
+ return false;
316
+ }
317
+ }
318
+ /**
319
+ * Admin: Remove a member from a channel (does not require member to be connected).
320
+ * Used to remove channel members from dashboard.
321
+ * Uses the first available user session or creates a temporary one.
322
+ */
323
+ async adminRemoveMember(channel, member) {
324
+ // Try to use an existing session
325
+ const sessions = Array.from(this.users.values());
326
+ if (sessions.length > 0) {
327
+ const session = sessions[0];
328
+ if (session.relayClient.adminRemoveMember) {
329
+ console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (via ${session.username})`);
330
+ return session.relayClient.adminRemoveMember(channel, member);
331
+ }
332
+ }
333
+ // No sessions available - create a temporary system client
334
+ try {
335
+ console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (creating temp client)`);
336
+ const tempClient = await this.createRelayClient({
337
+ socketPath: this.socketPath,
338
+ agentName: '__system__',
339
+ entityType: 'user',
340
+ });
341
+ await tempClient.connect();
342
+ // Give daemon time to complete handshake before sending admin commands.
343
+ // 100ms is sufficient for local Unix socket handshake (typically <10ms),
344
+ // but provides margin for the daemon to process the HELLO message and
345
+ // set up internal state. This is a temporary client created just for
346
+ // the admin operation, not a long-lived session.
347
+ await new Promise(resolve => setTimeout(resolve, 100));
348
+ if (tempClient.adminRemoveMember) {
349
+ const result = tempClient.adminRemoveMember(channel, member);
350
+ // Disconnect after a short delay to allow message to be sent
351
+ setTimeout(() => tempClient.disconnect(), 200);
352
+ return result;
353
+ }
354
+ tempClient.disconnect();
355
+ return false;
356
+ }
357
+ catch (err) {
358
+ console.error('[user-bridge] Failed to create temp client for admin remove:', err);
359
+ return false;
360
+ }
361
+ }
362
+ /**
363
+ * Dispose of all user sessions.
364
+ */
365
+ dispose() {
366
+ for (const [username] of this.users) {
367
+ this.unregisterUser(username);
368
+ }
369
+ console.log('[user-bridge] Disposed all user sessions');
370
+ }
371
+ }
372
+ //# sourceMappingURL=user-bridge.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"user-bridge.js","sourceRoot":"","sources":["../src/user-bridge.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,49 @@
1
+ {
2
+ "name": "@agent-relay/dashboard-server",
3
+ "version": "0.1.0",
4
+ "description": "Relay dashboard server - HTTP/WebSocket server for agent coordination",
5
+ "type": "module",
6
+ "main": "dist/index.js",
7
+ "types": "dist/index.d.ts",
8
+ "exports": {
9
+ ".": {
10
+ "types": "./dist/index.d.ts",
11
+ "import": "./dist/index.js"
12
+ }
13
+ },
14
+ "files": [
15
+ "dist",
16
+ "README.md"
17
+ ],
18
+ "scripts": {
19
+ "build": "tsc",
20
+ "clean": "rm -rf dist",
21
+ "test": "vitest run",
22
+ "test:watch": "vitest"
23
+ },
24
+ "dependencies": {
25
+ "@agent-relay/protocol": "^0.1.2",
26
+ "@agent-relay/config": "^0.1.0",
27
+ "@agent-relay/storage": "^0.1.0",
28
+ "@agent-relay/bridge": "^0.1.0",
29
+ "@agent-relay/utils": "^0.1.0",
30
+ "@agent-relay/resiliency": "^0.1.0",
31
+ "@agent-relay/trajectory": "^0.1.0",
32
+ "@agent-relay/cloud": "^0.1.0",
33
+ "@agent-relay/daemon": "^0.1.0",
34
+ "@agent-relay/user-directory": "^0.1.0",
35
+ "@agent-relay/sdk": "^0.1.0",
36
+ "express": "^4.21.2",
37
+ "ws": "^8.18.3"
38
+ },
39
+ "devDependencies": {
40
+ "@types/express": "^5.0.1",
41
+ "@types/node": "^22.19.3",
42
+ "@types/ws": "^8.18.1",
43
+ "typescript": "^5.9.3",
44
+ "vitest": "^3.2.4"
45
+ },
46
+ "publishConfig": {
47
+ "access": "public"
48
+ }
49
+ }