@agent-relay/dashboard-server 2.0.20 → 2.0.26

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 (62) hide show
  1. package/dist/index.d.ts +30 -8
  2. package/dist/index.d.ts.map +1 -1
  3. package/dist/index.js +31 -7
  4. package/dist/index.js.map +1 -1
  5. package/dist/mocks/fixtures.d.ts +1 -1
  6. package/dist/mocks/fixtures.d.ts.map +1 -1
  7. package/dist/mocks/fixtures.js.map +1 -1
  8. package/dist/mocks/index.d.ts.map +1 -1
  9. package/dist/mocks/index.js.map +1 -1
  10. package/dist/mocks/routes.d.ts.map +1 -1
  11. package/dist/mocks/routes.js +172 -7
  12. package/dist/mocks/routes.js.map +1 -1
  13. package/dist/mocks/types.d.ts.map +1 -1
  14. package/dist/mocks/types.js.map +1 -1
  15. package/dist/proxy-server.d.ts +45 -0
  16. package/dist/proxy-server.d.ts.map +1 -0
  17. package/dist/proxy-server.js +320 -0
  18. package/dist/proxy-server.js.map +1 -0
  19. package/dist/server.d.ts +3 -44
  20. package/dist/server.d.ts.map +1 -1
  21. package/dist/server.js +5149 -267
  22. package/dist/server.js.map +1 -1
  23. package/dist/services/health-worker-manager.d.ts +62 -0
  24. package/dist/services/health-worker-manager.d.ts.map +1 -0
  25. package/dist/services/health-worker-manager.js +144 -0
  26. package/dist/services/health-worker-manager.js.map +1 -0
  27. package/dist/services/health-worker.d.ts +9 -0
  28. package/dist/services/health-worker.d.ts.map +1 -0
  29. package/dist/services/health-worker.js +79 -0
  30. package/dist/services/health-worker.js.map +1 -0
  31. package/dist/services/index.d.ts +10 -0
  32. package/dist/services/index.d.ts.map +1 -0
  33. package/dist/services/index.js +10 -0
  34. package/dist/services/index.js.map +1 -0
  35. package/dist/services/metrics.d.ts +105 -0
  36. package/dist/services/metrics.d.ts.map +1 -0
  37. package/dist/services/metrics.js +193 -0
  38. package/dist/services/metrics.js.map +1 -0
  39. package/dist/services/needs-attention.d.ts +24 -0
  40. package/dist/services/needs-attention.d.ts.map +1 -0
  41. package/dist/services/needs-attention.js +78 -0
  42. package/dist/services/needs-attention.js.map +1 -0
  43. package/dist/services/user-bridge.d.ts +158 -0
  44. package/dist/services/user-bridge.d.ts.map +1 -0
  45. package/dist/services/user-bridge.js +390 -0
  46. package/dist/services/user-bridge.js.map +1 -0
  47. package/dist/start.d.ts.map +1 -1
  48. package/dist/start.js +1 -1
  49. package/dist/start.js.map +1 -1
  50. package/dist/types/index.d.ts +128 -0
  51. package/dist/types/index.d.ts.map +1 -0
  52. package/dist/types/index.js +7 -0
  53. package/dist/types/index.js.map +1 -0
  54. package/dist/types/threading.d.ts +8 -0
  55. package/dist/types/threading.d.ts.map +1 -0
  56. package/dist/types/threading.js +2 -0
  57. package/dist/types/threading.js.map +1 -0
  58. package/package.json +14 -3
  59. package/dist/vitest.config.d.ts +0 -3
  60. package/dist/vitest.config.d.ts.map +0 -1
  61. package/dist/vitest.config.js +0 -27
  62. package/dist/vitest.config.js.map +0 -1
@@ -0,0 +1,78 @@
1
+ /**
2
+ * Derive which agents currently need attention based on message history.
3
+ *
4
+ * Heuristic:
5
+ * - Track the most recent inbound message to an agent for each conversation key.
6
+ * - Conversation key is thread ID if present, otherwise the counterparty agent.
7
+ * - Track the most recent outbound message from the agent for the same key.
8
+ * - An agent needs attention if their latest inbound message for any key is newer
9
+ * than their latest outbound message for that key.
10
+ */
11
+ // Only consider messages from the last 30 minutes for "needs attention"
12
+ const ATTENTION_WINDOW_MS = 30 * 60 * 1000;
13
+ function updateLatest(map, agent, key, ts) {
14
+ const agentMap = map.get(agent) ?? new Map();
15
+ const prev = agentMap.get(key) ?? -Infinity;
16
+ if (ts > prev) {
17
+ agentMap.set(key, ts);
18
+ map.set(agent, agentMap);
19
+ }
20
+ }
21
+ /**
22
+ * Compute which agents have pending inbound messages they haven't answered.
23
+ * Only considers messages within the attention window (last 30 minutes).
24
+ */
25
+ export function computeNeedsAttention(messages) {
26
+ const latestInbound = new Map(); // agent -> (key -> ts)
27
+ const latestOutbound = new Map(); // agent -> (key -> ts)
28
+ const now = Date.now();
29
+ const cutoffTime = now - ATTENTION_WINDOW_MS;
30
+ for (const message of messages) {
31
+ const ts = Date.parse(message.timestamp);
32
+ if (Number.isNaN(ts))
33
+ continue;
34
+ // Detect broadcasts: either explicit isBroadcast flag or to === '*'
35
+ const isBroadcast = message.isBroadcast || message.to === '*';
36
+ // Inbound: messages directed to a specific agent (ignore broadcasts)
37
+ // Note: isBroadcast indicates the message was originally a broadcast, even though
38
+ // the 'to' field is set to the individual recipient for storage purposes
39
+ if (message.to && !isBroadcast) {
40
+ const inboundKey = message.thread ? `thread:${message.thread}` : `sender:${message.from}`;
41
+ updateLatest(latestInbound, message.to, inboundKey, ts);
42
+ }
43
+ // Outbound: track replies by thread (preferred) or by target agent
44
+ // Also treat broadcasts as clearing attention for all prior senders
45
+ if (message.from) {
46
+ const outboundKey = message.thread
47
+ ? `thread:${message.thread}`
48
+ : (message.to && !isBroadcast)
49
+ ? `sender:${message.to}`
50
+ : null;
51
+ if (outboundKey) {
52
+ updateLatest(latestOutbound, message.from, outboundKey, ts);
53
+ }
54
+ // Broadcasts clear attention: agent is actively participating
55
+ // Track as a "catch-all" outbound timestamp for this agent
56
+ if (isBroadcast) {
57
+ updateLatest(latestOutbound, message.from, '__broadcast__', ts);
58
+ }
59
+ }
60
+ }
61
+ const needsAttention = new Set();
62
+ latestInbound.forEach((keyMap, agent) => {
63
+ keyMap.forEach((inboundTs, key) => {
64
+ // Skip if inbound message is too old (outside attention window)
65
+ if (inboundTs < cutoffTime)
66
+ return;
67
+ const outboundTs = latestOutbound.get(agent)?.get(key) ?? -Infinity;
68
+ // Also check if agent sent a broadcast after the inbound message
69
+ const broadcastTs = latestOutbound.get(agent)?.get('__broadcast__') ?? -Infinity;
70
+ const latestReply = Math.max(outboundTs, broadcastTs);
71
+ if (inboundTs > latestReply) {
72
+ needsAttention.add(agent);
73
+ }
74
+ });
75
+ });
76
+ return needsAttention;
77
+ }
78
+ //# sourceMappingURL=needs-attention.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"needs-attention.js","sourceRoot":"","sources":["../../src/services/needs-attention.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,158 @@
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
+ import type { WebSocket } from 'ws';
11
+ /**
12
+ * Relay client interface (subset of RelayClient for dependency injection)
13
+ */
14
+ export interface IRelayClient {
15
+ connect(): Promise<void>;
16
+ disconnect(): void;
17
+ state: string;
18
+ sendMessage(to: string, body: string, kind?: string, data?: unknown, thread?: string): boolean;
19
+ joinChannel(channel: string, displayName?: string): boolean;
20
+ leaveChannel(channel: string, reason?: string): boolean;
21
+ sendChannelMessage(channel: string, body: string, options?: {
22
+ thread?: string;
23
+ mentions?: string[];
24
+ attachments?: unknown[];
25
+ data?: Record<string, unknown>;
26
+ }): boolean;
27
+ adminJoinChannel?(channel: string, member: string): boolean;
28
+ adminRemoveMember?(channel: string, member: string): boolean;
29
+ onMessage?: (from: string, payload: any, messageId: string, meta?: any, originalTo?: string) => void;
30
+ onChannelMessage?: (from: string, channel: string, body: string, envelope: any) => void;
31
+ }
32
+ /**
33
+ * Factory function type for creating relay clients
34
+ */
35
+ export type RelayClientFactory = (options: {
36
+ socketPath: string;
37
+ agentName: string;
38
+ entityType: 'user';
39
+ displayName?: string;
40
+ avatarUrl?: string;
41
+ }) => Promise<IRelayClient>;
42
+ /**
43
+ * User info for avatar lookups
44
+ */
45
+ export interface UserInfo {
46
+ avatarUrl?: string;
47
+ }
48
+ /**
49
+ * Options for creating a UserBridge
50
+ */
51
+ export interface UserBridgeOptions {
52
+ socketPath: string;
53
+ createRelayClient: RelayClientFactory;
54
+ loadPersistedChannels?: (username: string) => Promise<string[]>;
55
+ /** Optional callback to look up user info (avatar URL) by username */
56
+ lookupUserInfo?: (username: string) => UserInfo | undefined;
57
+ }
58
+ /**
59
+ * Message options for sending
60
+ */
61
+ export interface SendMessageOptions {
62
+ thread?: string;
63
+ data?: Record<string, unknown>;
64
+ attachments?: unknown[];
65
+ }
66
+ /**
67
+ * UserBridge manages the connection between dashboard WebSocket users
68
+ * and the relay daemon.
69
+ */
70
+ export declare class UserBridge {
71
+ private readonly socketPath;
72
+ private readonly createRelayClient;
73
+ private readonly loadPersistedChannels?;
74
+ private readonly lookupUserInfo?;
75
+ private readonly users;
76
+ constructor(options: UserBridgeOptions);
77
+ /**
78
+ * Get the relay client for a user if they are registered.
79
+ * This allows external code to reuse the userBridge's relay client
80
+ * instead of creating a duplicate connection.
81
+ */
82
+ getRelayClient(username: string): IRelayClient | undefined;
83
+ /**
84
+ * Register a user with the relay daemon.
85
+ * Creates a relay client connection for the user.
86
+ */
87
+ registerUser(username: string, webSocket: WebSocket, options?: {
88
+ avatarUrl?: string;
89
+ displayName?: string;
90
+ }): Promise<void>;
91
+ /**
92
+ * Remove a specific WebSocket connection from a user's session.
93
+ * If this was the last connection, unregister the user entirely.
94
+ */
95
+ private removeWebSocket;
96
+ /**
97
+ * Unregister a user and disconnect their relay client.
98
+ */
99
+ unregisterUser(username: string): void;
100
+ /**
101
+ * Check if a user is registered.
102
+ */
103
+ isUserRegistered(username: string): boolean;
104
+ /**
105
+ * Add a new WebSocket connection for an existing user session.
106
+ * This is needed when a user opens a new tab.
107
+ */
108
+ updateWebSocket(username: string, newWebSocket: WebSocket): boolean;
109
+ /**
110
+ * Get list of all registered users.
111
+ */
112
+ getRegisteredUsers(): string[];
113
+ /**
114
+ * Join a channel.
115
+ */
116
+ joinChannel(username: string, channel: string): Promise<boolean>;
117
+ /**
118
+ * Leave a channel.
119
+ */
120
+ leaveChannel(username: string, channel: string): Promise<boolean>;
121
+ /**
122
+ * Get channels a user has joined.
123
+ */
124
+ getUserChannels(username: string): string[];
125
+ /**
126
+ * Send a message to a channel.
127
+ */
128
+ sendChannelMessage(username: string, channel: string, body: string, options?: SendMessageOptions): Promise<boolean>;
129
+ /**
130
+ * Send a direct message to another user or agent.
131
+ */
132
+ sendDirectMessage(fromUsername: string, toName: string, body: string, options?: SendMessageOptions): Promise<boolean>;
133
+ /**
134
+ * Handle incoming direct message from relay daemon.
135
+ */
136
+ private handleIncomingDirectMessage;
137
+ /**
138
+ * Handle incoming channel message from relay daemon.
139
+ */
140
+ private handleIncomingChannelMessage;
141
+ /**
142
+ * Admin: Add a member to a channel (does not require member to be connected).
143
+ * Used to sync channel memberships from database.
144
+ * Uses the first available user session or creates a temporary one.
145
+ */
146
+ adminJoinChannel(channel: string, member: string): Promise<boolean>;
147
+ /**
148
+ * Admin: Remove a member from a channel (does not require member to be connected).
149
+ * Used to remove channel members from dashboard.
150
+ * Uses the first available user session or creates a temporary one.
151
+ */
152
+ adminRemoveMember(channel: string, member: string): Promise<boolean>;
153
+ /**
154
+ * Dispose of all user sessions.
155
+ */
156
+ dispose(): void;
157
+ }
158
+ //# sourceMappingURL=user-bridge.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"user-bridge.d.ts","sourceRoot":"","sources":["../../src/services/user-bridge.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,IAAI,CAAC;AAEpC;;GAEG;AACH,MAAM,WAAW,YAAY;IAC3B,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;IACzB,UAAU,IAAI,IAAI,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,CACT,EAAE,EAAE,MAAM,EACV,IAAI,EAAE,MAAM,EACZ,IAAI,CAAC,EAAE,MAAM,EACb,IAAI,CAAC,EAAE,OAAO,EACd,MAAM,CAAC,EAAE,MAAM,GACd,OAAO,CAAC;IAEX,WAAW,CAAC,OAAO,EAAE,MAAM,EAAE,WAAW,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC;IAC5D,YAAY,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC;IACxD,kBAAkB,CAChB,OAAO,EAAE,MAAM,EACf,IAAI,EAAE,MAAM,EACZ,OAAO,CAAC,EAAE;QAAE,MAAM,CAAC,EAAE,MAAM,CAAC;QAAC,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;QAAC,WAAW,CAAC,EAAE,OAAO,EAAE,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;KAAE,GAC1G,OAAO,CAAC;IAEX,gBAAgB,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC;IAC5D,iBAAiB,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC;IAE7D,SAAS,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,EAAE,SAAS,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,EAAE,UAAU,CAAC,EAAE,MAAM,KAAK,IAAI,CAAC;IAErG,gBAAgB,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,GAAG,KAAK,IAAI,CAAC;CACzF;AAED;;GAEG;AACH,MAAM,MAAM,kBAAkB,GAAG,CAAC,OAAO,EAAE;IACzC,UAAU,EAAE,MAAM,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB,KAAK,OAAO,CAAC,YAAY,CAAC,CAAC;AAc5B;;GAEG;AACH,MAAM,WAAW,QAAQ;IACvB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,UAAU,EAAE,MAAM,CAAC;IACnB,iBAAiB,EAAE,kBAAkB,CAAC;IACtC,qBAAqB,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;IAChE,sEAAsE;IACtE,cAAc,CAAC,EAAE,CAAC,QAAQ,EAAE,MAAM,KAAK,QAAQ,GAAG,SAAS,CAAC;CAC7D;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC/B,WAAW,CAAC,EAAE,OAAO,EAAE,CAAC;CACzB;AAED;;;GAGG;AACH,qBAAa,UAAU;IACrB,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAS;IACpC,OAAO,CAAC,QAAQ,CAAC,iBAAiB,CAAqB;IACvD,OAAO,CAAC,QAAQ,CAAC,qBAAqB,CAAC,CAA0C;IACjF,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAC,CAA6C;IAC7E,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAkC;gBAE5C,OAAO,EAAE,iBAAiB;IAOtC;;;;OAIG;IACH,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAI1D;;;OAGG;IACG,YAAY,CAChB,QAAQ,EAAE,MAAM,EAChB,SAAS,EAAE,SAAS,EACpB,OAAO,CAAC,EAAE;QAAE,SAAS,CAAC,EAAE,MAAM,CAAC;QAAC,WAAW,CAAC,EAAE,MAAM,CAAA;KAAE,GACrD,OAAO,CAAC,IAAI,CAAC;IAuEhB;;;OAGG;IACH,OAAO,CAAC,eAAe;IAavB;;OAEG;IACH,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI;IAUtC;;OAEG;IACH,gBAAgB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAI3C;;;OAGG;IACH,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,EAAE,SAAS,GAAG,OAAO;IAmBnE;;OAEG;IACH,kBAAkB,IAAI,MAAM,EAAE;IAI9B;;OAEG;IACG,WAAW,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAkBtE;;OAEG;IACG,YAAY,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAmBvE;;OAEG;IACH,eAAe,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,EAAE;IAK3C;;OAEG;IACG,kBAAkB,CACtB,QAAQ,EAAE,MAAM,EAChB,OAAO,EAAE,MAAM,EACf,IAAI,EAAE,MAAM,EACZ,OAAO,CAAC,EAAE,kBAAkB,GAC3B,OAAO,CAAC,OAAO,CAAC;IAcnB;;OAEG;IACG,iBAAiB,CACrB,YAAY,EAAE,MAAM,EACpB,MAAM,EAAE,MAAM,EACd,IAAI,EAAE,MAAM,EACZ,OAAO,CAAC,EAAE,kBAAkB,GAC3B,OAAO,CAAC,OAAO,CAAC;IAgBnB;;OAEG;IACH,OAAO,CAAC,2BAA2B;IAwCnC;;OAEG;IACH,OAAO,CAAC,4BAA4B;IAsCpC;;;;OAIG;IACG,gBAAgB,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IA2CzE;;;;OAIG;IACG,iBAAiB,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IA2C1E;;OAEG;IACH,OAAO,IAAI,IAAI;CAMhB"}
@@ -0,0 +1,390 @@
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, just update the WebSocket (multi-tab support)
40
+ if (this.users.has(username)) {
41
+ console.log(`[user-bridge] User ${username} already registered, updating WebSocket`);
42
+ this.updateWebSocket(username, webSocket);
43
+ return;
44
+ }
45
+ // Create relay client for this user
46
+ const relayClient = await this.createRelayClient({
47
+ socketPath: this.socketPath,
48
+ agentName: username,
49
+ entityType: 'user',
50
+ displayName: options?.displayName,
51
+ avatarUrl: options?.avatarUrl,
52
+ });
53
+ // Connect to daemon
54
+ await relayClient.connect();
55
+ // Set up message handler to forward direct messages to WebSocket
56
+ relayClient.onMessage = (from, payload, _messageId, _meta, _originalTo) => {
57
+ const body = typeof payload === 'object' && payload !== null && 'body' in payload
58
+ ? payload.body
59
+ : String(payload);
60
+ this.handleIncomingDirectMessage(username, from, body, payload);
61
+ };
62
+ // Set up channel message handler to forward channel messages to WebSocket
63
+ relayClient.onChannelMessage = (from, channel, body, envelope) => {
64
+ this.handleIncomingChannelMessage(username, from, channel, body, envelope);
65
+ };
66
+ // Create session with WebSocket set for multi-tab support
67
+ const session = {
68
+ username,
69
+ relayClient,
70
+ webSockets: new Set([webSocket]),
71
+ channels: new Set(),
72
+ avatarUrl: options?.avatarUrl,
73
+ };
74
+ this.users.set(username, session);
75
+ console.log(`[user-bridge] User registered: ${username} (total users: ${this.users.size}, connections: 1)`);
76
+ // Auto-join user to #general channel
77
+ // Note: The daemon auto-joins on connect, but we need to track locally too
78
+ session.channels.add('#general');
79
+ if (this.loadPersistedChannels) {
80
+ try {
81
+ const persistedChannels = await this.loadPersistedChannels(username);
82
+ for (const channel of persistedChannels) {
83
+ if (channel === '#general')
84
+ continue;
85
+ if (session.channels.has(channel))
86
+ continue;
87
+ session.relayClient.joinChannel(channel, username);
88
+ session.channels.add(channel);
89
+ }
90
+ }
91
+ catch (err) {
92
+ console.error(`[user-bridge] Failed to restore persisted channels for ${username}:`, err);
93
+ }
94
+ }
95
+ // Set up WebSocket close handler to remove this specific connection
96
+ webSocket.on('close', () => {
97
+ this.removeWebSocket(username, webSocket);
98
+ });
99
+ console.log(`[user-bridge] User ${username} registered with relay daemon`);
100
+ }
101
+ /**
102
+ * Remove a specific WebSocket connection from a user's session.
103
+ * If this was the last connection, unregister the user entirely.
104
+ */
105
+ removeWebSocket(username, webSocket) {
106
+ const session = this.users.get(username);
107
+ if (!session)
108
+ return;
109
+ session.webSockets.delete(webSocket);
110
+ console.log(`[user-bridge] WebSocket closed for ${username} (${session.webSockets.size} connections remaining)`);
111
+ // Only unregister if ALL connections are closed
112
+ if (session.webSockets.size === 0) {
113
+ this.unregisterUser(username);
114
+ }
115
+ }
116
+ /**
117
+ * Unregister a user and disconnect their relay client.
118
+ */
119
+ unregisterUser(username) {
120
+ const session = this.users.get(username);
121
+ if (!session)
122
+ return;
123
+ session.relayClient.disconnect();
124
+ this.users.delete(username);
125
+ console.log(`[user-bridge] User ${username} unregistered from relay daemon`);
126
+ }
127
+ /**
128
+ * Check if a user is registered.
129
+ */
130
+ isUserRegistered(username) {
131
+ return this.users.has(username);
132
+ }
133
+ /**
134
+ * Add a new WebSocket connection for an existing user session.
135
+ * This is needed when a user opens a new tab.
136
+ */
137
+ updateWebSocket(username, newWebSocket) {
138
+ const session = this.users.get(username);
139
+ if (!session) {
140
+ console.log(`[user-bridge] Cannot add WebSocket - user ${username} not registered`);
141
+ return false;
142
+ }
143
+ // Add the new WebSocket to the set
144
+ session.webSockets.add(newWebSocket);
145
+ // Set up close handler to remove this specific connection
146
+ newWebSocket.on('close', () => {
147
+ this.removeWebSocket(username, newWebSocket);
148
+ });
149
+ console.log(`[user-bridge] Added WebSocket for user ${username} (${session.webSockets.size} connections)`);
150
+ return true;
151
+ }
152
+ /**
153
+ * Get list of all registered users.
154
+ */
155
+ getRegisteredUsers() {
156
+ return Array.from(this.users.keys());
157
+ }
158
+ /**
159
+ * Join a channel.
160
+ */
161
+ async joinChannel(username, channel) {
162
+ const session = this.users.get(username);
163
+ if (!session) {
164
+ console.warn(`[user-bridge] Cannot join channel - user ${username} not registered`);
165
+ return false;
166
+ }
167
+ // Send CHANNEL_JOIN via relay client
168
+ const success = session.relayClient.joinChannel(channel, username);
169
+ if (success) {
170
+ // Track membership
171
+ session.channels.add(channel);
172
+ }
173
+ return success;
174
+ }
175
+ /**
176
+ * Leave a channel.
177
+ */
178
+ async leaveChannel(username, channel) {
179
+ const session = this.users.get(username);
180
+ if (!session) {
181
+ console.warn(`[user-bridge] Cannot leave channel - user ${username} not registered`);
182
+ return false;
183
+ }
184
+ // Send CHANNEL_LEAVE via relay client
185
+ const success = session.relayClient.leaveChannel(channel);
186
+ if (success) {
187
+ // Update membership
188
+ session.channels.delete(channel);
189
+ console.log(`[user-bridge] User ${username} left channel ${channel}`);
190
+ }
191
+ return success;
192
+ }
193
+ /**
194
+ * Get channels a user has joined.
195
+ */
196
+ getUserChannels(username) {
197
+ const session = this.users.get(username);
198
+ return session ? Array.from(session.channels) : [];
199
+ }
200
+ /**
201
+ * Send a message to a channel.
202
+ */
203
+ async sendChannelMessage(username, channel, body, options) {
204
+ const session = this.users.get(username);
205
+ if (!session) {
206
+ console.warn(`[user-bridge] Cannot send - user ${username} not registered`);
207
+ return false;
208
+ }
209
+ return session.relayClient.sendChannelMessage(channel, body, {
210
+ thread: options?.thread,
211
+ data: options?.data,
212
+ attachments: options?.attachments,
213
+ });
214
+ }
215
+ /**
216
+ * Send a direct message to another user or agent.
217
+ */
218
+ async sendDirectMessage(fromUsername, toName, body, options) {
219
+ const session = this.users.get(fromUsername);
220
+ if (!session) {
221
+ console.warn(`[user-bridge] Cannot send DM - user ${fromUsername} not registered`);
222
+ return false;
223
+ }
224
+ return session.relayClient.sendMessage(toName, body, 'message', options?.data, options?.thread);
225
+ }
226
+ /**
227
+ * Handle incoming direct message from relay daemon.
228
+ */
229
+ handleIncomingDirectMessage(username, from, body, payload) {
230
+ // Skip channel messages - they are handled by handleIncomingChannelMessage
231
+ // The relay client calls both onMessage and onChannelMessage for channel messages,
232
+ // with _isChannelMessage flag set in the data for onMessage calls
233
+ const payloadObj = payload;
234
+ if (payloadObj?.data?._isChannelMessage) {
235
+ return; // Skip - will be handled by onChannelMessage callback
236
+ }
237
+ const session = this.users.get(username);
238
+ if (!session)
239
+ return;
240
+ // Look up sender's avatar if lookup function is available
241
+ const senderInfo = this.lookupUserInfo?.(from);
242
+ const fromAvatarUrl = senderInfo?.avatarUrl;
243
+ // Determine entity type: user if they have info, agent otherwise
244
+ const fromEntityType = senderInfo ? 'user' : 'agent';
245
+ const message = JSON.stringify({
246
+ type: 'direct_message',
247
+ from,
248
+ fromAvatarUrl,
249
+ fromEntityType,
250
+ body: payloadObj?.body || body,
251
+ timestamp: new Date().toISOString(),
252
+ });
253
+ // Send to ALL open WebSocket connections (multi-tab support)
254
+ for (const ws of session.webSockets) {
255
+ if (ws.readyState === 1) { // OPEN
256
+ ws.send(message);
257
+ }
258
+ }
259
+ }
260
+ /**
261
+ * Handle incoming channel message from relay daemon.
262
+ */
263
+ handleIncomingChannelMessage(username, from, channel, body, envelope) {
264
+ const session = this.users.get(username);
265
+ if (!session)
266
+ return;
267
+ // Look up sender's avatar if lookup function is available
268
+ const senderInfo = this.lookupUserInfo?.(from);
269
+ const fromAvatarUrl = senderInfo?.avatarUrl;
270
+ // Determine entity type: user if they have info, agent otherwise
271
+ const fromEntityType = senderInfo ? 'user' : 'agent';
272
+ // Channel message
273
+ const env = envelope;
274
+ const message = JSON.stringify({
275
+ type: 'channel_message',
276
+ channel,
277
+ from,
278
+ fromAvatarUrl,
279
+ fromEntityType,
280
+ body,
281
+ thread: env?.payload?.thread,
282
+ mentions: env?.payload?.mentions,
283
+ timestamp: new Date().toISOString(),
284
+ });
285
+ // Send to ALL open WebSocket connections (multi-tab support)
286
+ for (const ws of session.webSockets) {
287
+ if (ws.readyState === 1) { // OPEN
288
+ ws.send(message);
289
+ }
290
+ }
291
+ }
292
+ /**
293
+ * Admin: Add a member to a channel (does not require member to be connected).
294
+ * Used to sync channel memberships from database.
295
+ * Uses the first available user session or creates a temporary one.
296
+ */
297
+ async adminJoinChannel(channel, member) {
298
+ // Try to use an existing session
299
+ const sessions = Array.from(this.users.values());
300
+ if (sessions.length > 0) {
301
+ const session = sessions[0];
302
+ if (session.relayClient.adminJoinChannel) {
303
+ console.log(`[user-bridge] Admin join: ${member} -> ${channel} (via ${session.username})`);
304
+ return session.relayClient.adminJoinChannel(channel, member);
305
+ }
306
+ }
307
+ // No sessions available - create a temporary system client
308
+ try {
309
+ console.log(`[user-bridge] Admin join: ${member} -> ${channel} (creating temp client)`);
310
+ const tempClient = await this.createRelayClient({
311
+ socketPath: this.socketPath,
312
+ agentName: '__system__',
313
+ entityType: 'user',
314
+ });
315
+ await tempClient.connect();
316
+ // Give daemon time to complete handshake before sending admin commands.
317
+ // 100ms is sufficient for local Unix socket handshake (typically <10ms),
318
+ // but provides margin for the daemon to process the HELLO message and
319
+ // set up internal state. This is a temporary client created just for
320
+ // the admin operation, not a long-lived session.
321
+ await new Promise(resolve => setTimeout(resolve, 100));
322
+ if (tempClient.adminJoinChannel) {
323
+ const result = tempClient.adminJoinChannel(channel, member);
324
+ // Disconnect after a short delay to allow message to be sent
325
+ setTimeout(() => tempClient.disconnect(), 200);
326
+ return result;
327
+ }
328
+ tempClient.disconnect();
329
+ return false;
330
+ }
331
+ catch (err) {
332
+ console.error('[user-bridge] Failed to create temp client for admin join:', err);
333
+ return false;
334
+ }
335
+ }
336
+ /**
337
+ * Admin: Remove a member from a channel (does not require member to be connected).
338
+ * Used to remove channel members from dashboard.
339
+ * Uses the first available user session or creates a temporary one.
340
+ */
341
+ async adminRemoveMember(channel, member) {
342
+ // Try to use an existing session
343
+ const sessions = Array.from(this.users.values());
344
+ if (sessions.length > 0) {
345
+ const session = sessions[0];
346
+ if (session.relayClient.adminRemoveMember) {
347
+ console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (via ${session.username})`);
348
+ return session.relayClient.adminRemoveMember(channel, member);
349
+ }
350
+ }
351
+ // No sessions available - create a temporary system client
352
+ try {
353
+ console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (creating temp client)`);
354
+ const tempClient = await this.createRelayClient({
355
+ socketPath: this.socketPath,
356
+ agentName: '__system__',
357
+ entityType: 'user',
358
+ });
359
+ await tempClient.connect();
360
+ // Give daemon time to complete handshake before sending admin commands.
361
+ // 100ms is sufficient for local Unix socket handshake (typically <10ms),
362
+ // but provides margin for the daemon to process the HELLO message and
363
+ // set up internal state. This is a temporary client created just for
364
+ // the admin operation, not a long-lived session.
365
+ await new Promise(resolve => setTimeout(resolve, 100));
366
+ if (tempClient.adminRemoveMember) {
367
+ const result = tempClient.adminRemoveMember(channel, member);
368
+ // Disconnect after a short delay to allow message to be sent
369
+ setTimeout(() => tempClient.disconnect(), 200);
370
+ return result;
371
+ }
372
+ tempClient.disconnect();
373
+ return false;
374
+ }
375
+ catch (err) {
376
+ console.error('[user-bridge] Failed to create temp client for admin remove:', err);
377
+ return false;
378
+ }
379
+ }
380
+ /**
381
+ * Dispose of all user sessions.
382
+ */
383
+ dispose() {
384
+ for (const [username] of this.users) {
385
+ this.unregisterUser(username);
386
+ }
387
+ console.log('[user-bridge] Disposed all user sessions');
388
+ }
389
+ }
390
+ //# sourceMappingURL=user-bridge.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"user-bridge.js","sourceRoot":"","sources":["../../src/services/user-bridge.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"start.d.ts","sourceRoot":"","sources":["../start.ts"],"names":[],"mappings":";AACA;;;;;GAKG"}
1
+ {"version":3,"file":"start.d.ts","sourceRoot":"","sources":["../src/start.ts"],"names":[],"mappings":";AACA;;;;;GAKG"}