@dollhousemcp/mcp-server 2.0.12-rc.9 → 2.0.13

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 (69) hide show
  1. package/CHANGELOG.md +66 -0
  2. package/README.md +4 -4
  3. package/README.md.backup +5 -5
  4. package/README.npm.md +4 -4
  5. package/dist/di/Container.d.ts +14 -0
  6. package/dist/di/Container.d.ts.map +1 -1
  7. package/dist/di/Container.js +43 -26
  8. package/dist/elements/agents/AgentManager.d.ts +11 -4
  9. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  10. package/dist/elements/agents/AgentManager.js +38 -11
  11. package/dist/elements/base/BaseElementManager.d.ts +10 -0
  12. package/dist/elements/base/BaseElementManager.d.ts.map +1 -1
  13. package/dist/elements/base/BaseElementManager.js +26 -1
  14. package/dist/elements/ensembles/EnsembleManager.d.ts +1 -0
  15. package/dist/elements/ensembles/EnsembleManager.d.ts.map +1 -1
  16. package/dist/elements/ensembles/EnsembleManager.js +13 -2
  17. package/dist/elements/templates/Template.d.ts +33 -1
  18. package/dist/elements/templates/Template.d.ts.map +1 -1
  19. package/dist/elements/templates/Template.js +74 -15
  20. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  21. package/dist/elements/templates/TemplateManager.js +8 -1
  22. package/dist/generated/version.d.ts +2 -2
  23. package/dist/generated/version.d.ts.map +1 -1
  24. package/dist/generated/version.js +3 -3
  25. package/dist/handlers/element-crud/createElement.js +2 -2
  26. package/dist/handlers/mcp-aql/SchemaDispatcher.d.ts.map +1 -1
  27. package/dist/handlers/mcp-aql/SchemaDispatcher.js +8 -1
  28. package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts +3 -0
  29. package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts.map +1 -1
  30. package/dist/handlers/strategies/EnsembleActivationStrategy.js +48 -9
  31. package/dist/index.js +19 -13
  32. package/dist/portfolio/DefaultElementProvider.d.ts +8 -0
  33. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  34. package/dist/portfolio/DefaultElementProvider.js +43 -1
  35. package/dist/security/contentValidator.d.ts +15 -0
  36. package/dist/security/contentValidator.d.ts.map +1 -1
  37. package/dist/security/contentValidator.js +40 -2
  38. package/dist/utils/TemplateRenderer.d.ts +9 -0
  39. package/dist/utils/TemplateRenderer.d.ts.map +1 -1
  40. package/dist/utils/TemplateRenderer.js +21 -1
  41. package/dist/web/console/IngestRoutes.d.ts.map +1 -1
  42. package/dist/web/console/IngestRoutes.js +193 -59
  43. package/dist/web/console/SessionNames.d.ts +18 -5
  44. package/dist/web/console/SessionNames.d.ts.map +1 -1
  45. package/dist/web/console/SessionNames.js +63 -8
  46. package/dist/web/console/StaleProcessRecovery.d.ts.map +1 -1
  47. package/dist/web/console/StaleProcessRecovery.js +5 -4
  48. package/dist/web/console/UnifiedConsole.js +3 -3
  49. package/dist/web/console/consoleToken.js +3 -3
  50. package/dist/web/portDiscovery.d.ts +1 -1
  51. package/dist/web/portDiscovery.d.ts.map +1 -1
  52. package/dist/web/portDiscovery.js +2 -2
  53. package/dist/web/public/app.js +65 -11
  54. package/dist/web/public/index.html +2 -0
  55. package/dist/web/public/logs.js +24 -2
  56. package/dist/web/public/metrics.js +22 -4
  57. package/dist/web/public/sessions.js +55 -8
  58. package/dist/web/public/setup.js +11 -2
  59. package/dist/web/public/styles.css +12 -0
  60. package/dist/web/routes/permissionRoutes.js +2 -2
  61. package/dist/web/routes/setupRoutes.d.ts +67 -1
  62. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  63. package/dist/web/routes/setupRoutes.js +298 -6
  64. package/dist/web/routes.d.ts.map +1 -1
  65. package/dist/web/routes.js +4 -2
  66. package/dist/web/server.d.ts.map +1 -1
  67. package/dist/web/server.js +14 -5
  68. package/package.json +5 -3
  69. package/server.json +2 -2
@@ -29,6 +29,10 @@ const RATE_LIMIT_WINDOW_MS = 60_000;
29
29
  const REAPER_INTERVAL_MS = 5_000;
30
30
  /** How long since last heartbeat before a session is considered dead (ms) */
31
31
  const SESSION_STALE_MS = 15_000;
32
+ /** Timeout for legacy port federation/proxy requests (ms) */
33
+ const LEGACY_FETCH_TIMEOUT_MS = 2_000;
34
+ /** How long before ended sessions are purged from the Map (ms) */
35
+ const ENDED_PURGE_MS = 5 * 60_000; // 5 minutes
32
36
  /** Normalize a string via UnicodeValidator (DMCP-SEC-004) */
33
37
  function normalizeInput(s) {
34
38
  return UnicodeValidator.normalize(s).normalizedContent;
@@ -44,6 +48,90 @@ export function createIngestRoutes(broadcasts) {
44
48
  const sessions = new Map();
45
49
  const namePool = new SessionNamePool();
46
50
  const rateLimiter = new SlidingWindowRateLimiter(RATE_LIMIT_MAX, RATE_LIMIT_WINDOW_MS);
51
+ // Sessions the user explicitly killed — never come back (#1870).
52
+ // Cleared only on server restart, which is appropriate since that's a new context.
53
+ const killedSessions = new Set();
54
+ // Sessions waiting for a PID so we can SIGTERM them (#1870).
55
+ // When the user dismisses a pid=0 orphan, we add it here. The next heartbeat
56
+ // (every 10s) carries the PID — we SIGTERM immediately and move to killedSessions.
57
+ const pendingKills = new Set();
58
+ /** Execute a deferred kill if we now have a PID. */
59
+ function tryExecutePendingKill(sessionId, pid) {
60
+ const killPid = pid || sessions.get(sessionId)?.pid;
61
+ if (!killPid)
62
+ return;
63
+ try {
64
+ process.kill(killPid, 'SIGTERM');
65
+ }
66
+ catch { /* already dead */ }
67
+ const existing = sessions.get(sessionId);
68
+ if (existing)
69
+ existing.status = 'ended';
70
+ logger.info('[IngestRoutes] Deferred kill executed — PID arrived', {
71
+ displayName: existing?.displayName, sessionId, pid: killPid,
72
+ });
73
+ }
74
+ /** Promote a pending kill to permanent. */
75
+ function finalizePendingKill(sessionId, pid) {
76
+ tryExecutePendingKill(sessionId, pid);
77
+ pendingKills.delete(sessionId);
78
+ killedSessions.add(sessionId);
79
+ }
80
+ /** Create a new session entry for an orphan. Returns null on failure. */
81
+ function autoRegister(sessionId, pid, authenticated = false) {
82
+ try {
83
+ const displayName = namePool.assign(sessionId);
84
+ const color = namePool.getColor(sessionId) ?? '#3b82f6';
85
+ const now = new Date().toISOString();
86
+ const info = {
87
+ sessionId, displayName, color,
88
+ pid: pid || 0,
89
+ startedAt: now, lastHeartbeat: now,
90
+ status: 'active', isLeader: false, authenticated, kind: 'mcp',
91
+ };
92
+ sessions.set(sessionId, info);
93
+ logger.info('[IngestRoutes] Auto-registered orphaned session', {
94
+ displayName, sessionId, source: pid ? 'heartbeat' : 'ingestion',
95
+ });
96
+ broadcasts.sessionBroadcast?.(info);
97
+ return info;
98
+ }
99
+ catch (err) {
100
+ logger.debug('[IngestRoutes] Failed to auto-register orphaned session', {
101
+ sessionId, error: err.message,
102
+ });
103
+ return null;
104
+ }
105
+ }
106
+ /**
107
+ * Auto-register or update an orphaned session from ingestion data.
108
+ * Returns the session (existing or newly created), or null if killed/pending.
109
+ */
110
+ function ensureSession(sessionId, pid, authenticated = false) {
111
+ if (killedSessions.has(sessionId))
112
+ return null;
113
+ if (pendingKills.has(sessionId)) {
114
+ finalizePendingKill(sessionId, pid);
115
+ return null;
116
+ }
117
+ const existing = sessions.get(sessionId);
118
+ if (!existing)
119
+ return autoRegister(sessionId, pid, authenticated);
120
+ if (existing.status === 'ended') {
121
+ existing.status = 'active';
122
+ logger.info('[IngestRoutes] Revived ended session still sending data', {
123
+ displayName: existing.displayName, sessionId,
124
+ });
125
+ }
126
+ existing.lastHeartbeat = new Date().toISOString();
127
+ if (pid && !existing.pid) {
128
+ existing.pid = pid;
129
+ logger.info('[IngestRoutes] Recovered PID for orphaned session', {
130
+ displayName: existing.displayName, sessionId, pid,
131
+ });
132
+ }
133
+ return existing;
134
+ }
47
135
  // JSON body parsing with size limit
48
136
  router.use(express.json({ limit: MAX_PAYLOAD_SIZE }));
49
137
  /**
@@ -76,11 +164,8 @@ export function createIngestRoutes(broadcasts) {
76
164
  broadcasts.logBroadcast(stamped);
77
165
  count++;
78
166
  }
79
- // Update session heartbeat
80
- const session = sessions.get(payload.sessionId);
81
- if (session) {
82
- session.lastHeartbeat = new Date().toISOString();
83
- }
167
+ // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)
168
+ const session = ensureSession(payload.sessionId);
84
169
  if (skipped > 0) {
85
170
  logger.debug(`[IngestRoutes] Log ingest from ${session?.displayName ?? payload.sessionId}: accepted=${count}, skipped=${skipped}`);
86
171
  }
@@ -105,7 +190,8 @@ export function createIngestRoutes(broadcasts) {
105
190
  if (broadcasts.metricsOnSnapshot) {
106
191
  broadcasts.metricsOnSnapshot(payload.snapshot);
107
192
  }
108
- const session = sessions.get(payload.sessionId);
193
+ // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)
194
+ const session = ensureSession(payload.sessionId);
109
195
  logger.debug(`[IngestRoutes] Metrics ingested from ${session?.displayName ?? payload.sessionId}`);
110
196
  res.status(200).json({ accepted: true });
111
197
  });
@@ -124,29 +210,26 @@ export function createIngestRoutes(broadcasts) {
124
210
  const now = new Date().toISOString();
125
211
  switch (payload.event) {
126
212
  case 'started': {
213
+ // Killed sessions stay dead; pending kills get finalized (#1870)
214
+ if (killedSessions.has(payload.sessionId))
215
+ break;
216
+ if (pendingKills.has(payload.sessionId)) {
217
+ finalizePendingKill(payload.sessionId, payload.pid);
218
+ break;
219
+ }
127
220
  const displayName = namePool.assign(payload.sessionId);
128
221
  const color = namePool.getColor(payload.sessionId) ?? '#3b82f6';
129
- // Follower sessions that reach the ingest endpoint are authenticated
130
- // if the auth middleware passed them through (token was valid).
131
222
  const isAuthenticated = Boolean(res.locals?.tokenEntry);
132
- const info = {
133
- sessionId: payload.sessionId,
134
- displayName,
135
- color,
136
- pid: payload.pid,
137
- startedAt: payload.startedAt || now,
138
- lastHeartbeat: now,
139
- status: 'active',
140
- isLeader: false,
141
- authenticated: isAuthenticated,
142
- kind: 'mcp',
143
- };
144
- sessions.set(payload.sessionId, info);
223
+ sessions.set(payload.sessionId, {
224
+ sessionId: payload.sessionId, displayName, color,
225
+ pid: payload.pid, startedAt: payload.startedAt || now, lastHeartbeat: now,
226
+ status: 'active', isLeader: false, authenticated: isAuthenticated, kind: 'mcp',
227
+ });
145
228
  logger.info('[IngestRoutes] Session registered', {
146
229
  displayName, sessionId: payload.sessionId, pid: payload.pid, color,
147
230
  activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length,
148
231
  });
149
- broadcasts.sessionBroadcast?.(info);
232
+ broadcasts.sessionBroadcast?.(sessions.get(payload.sessionId));
150
233
  break;
151
234
  }
152
235
  case 'stopped': {
@@ -164,10 +247,8 @@ export function createIngestRoutes(broadcasts) {
164
247
  break;
165
248
  }
166
249
  case 'heartbeat': {
167
- const existing = sessions.get(payload.sessionId);
168
- if (existing) {
169
- existing.lastHeartbeat = now;
170
- }
250
+ // Auto-register or update — heartbeat includes PID for recovery (#1870)
251
+ ensureSession(payload.sessionId, payload.pid);
171
252
  break;
172
253
  }
173
254
  }
@@ -177,7 +258,9 @@ export function createIngestRoutes(broadcasts) {
177
258
  * GET /api/sessions — List all tracked sessions.
178
259
  */
179
260
  router.get('/api/sessions', async (_req, res) => {
180
- const localSessions = Array.from(sessions.values());
261
+ // Server-side active filter — the frontend also filters, but ended sessions
262
+ // should never leave the API to prevent stale UI (#1870).
263
+ const localSessions = Array.from(sessions.values()).filter(s => s.status === 'active');
181
264
  const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;
182
265
  // Federate with the legacy port (3939) to show all sessions on the
183
266
  // machine, including unauthenticated ones from pre-auth installs.
@@ -185,7 +268,7 @@ export function createIngestRoutes(broadcasts) {
185
268
  if (currentPort !== 3939) {
186
269
  try {
187
270
  const controller = new AbortController();
188
- const timeout = setTimeout(() => controller.abort(), 2000);
271
+ const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);
189
272
  const legacyRes = await fetch('http://127.0.0.1:3939/api/sessions', {
190
273
  signal: controller.signal,
191
274
  });
@@ -213,58 +296,109 @@ export function createIngestRoutes(broadcasts) {
213
296
  /**
214
297
  * POST /api/sessions/:sessionId/kill — Terminate a session's server process.
215
298
  */
216
- router.post('/api/sessions/:sessionId/kill', (req, res) => {
299
+ router.post('/api/sessions/:sessionId/kill', async (req, res) => {
217
300
  const sessionId = req.params['sessionId'];
218
301
  const session = sessions.get(sessionId);
219
302
  if (!session) {
303
+ // Session not in local Map — try proxying kill to legacy port (#1870)
304
+ const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;
305
+ if (currentPort !== 3939) {
306
+ try {
307
+ const controller = new AbortController();
308
+ const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);
309
+ const proxyRes = await fetch(`http://127.0.0.1:3939/api/sessions/${encodeURIComponent(sessionId)}/kill`, {
310
+ method: 'POST',
311
+ signal: controller.signal,
312
+ });
313
+ clearTimeout(timeout);
314
+ if (proxyRes.ok) {
315
+ const data = await proxyRes.json();
316
+ res.json(data);
317
+ return;
318
+ }
319
+ }
320
+ catch {
321
+ // Legacy instance not running — fall through to 404
322
+ }
323
+ }
220
324
  logger.warn('[IngestRoutes] Kill requested for unknown session', { sessionId });
221
325
  res.status(404).json({ error: 'Session not found', sessionId });
222
326
  return;
223
327
  }
224
328
  if (!session.pid) {
225
- res.status(400).json({ error: 'No PID for session', sessionId, displayName: session.displayName });
329
+ // Auto-registered orphan with unknown PID queue for deferred kill (#1870).
330
+ // The next heartbeat (every ~10s) carries the PID. ensureSession() will
331
+ // SIGTERM the process as soon as the PID arrives. Session is gone for good.
332
+ session.status = 'ended';
333
+ namePool.release(sessionId);
334
+ pendingKills.add(sessionId);
335
+ logger.info('[IngestRoutes] Queued deferred kill — waiting for PID via heartbeat', {
336
+ displayName: session.displayName, sessionId,
337
+ });
338
+ res.json({ ok: true, dismissed: session.displayName, reason: 'pending-kill' });
226
339
  return;
227
340
  }
341
+ // SIGTERM the process. Even if it fails (ESRCH = already dead, EPERM = not ours),
342
+ // mark the session as permanently killed so it never reappears (#1870).
343
+ let killed = false;
228
344
  try {
229
345
  process.kill(session.pid, 'SIGTERM');
230
- session.status = 'ended';
231
- namePool.release(sessionId);
232
- logger.info('[IngestRoutes] Session killed', {
233
- displayName: session.displayName, sessionId, pid: session.pid,
234
- activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,
235
- });
236
- res.json({ ok: true, killed: session.displayName, pid: session.pid });
346
+ killed = true;
237
347
  }
238
348
  catch (err) {
239
- const message = err.message;
240
- logger.error('[IngestRoutes] Failed to kill session', {
241
- displayName: session.displayName, sessionId, pid: session.pid, error: message,
242
- });
243
- res.status(500).json({ error: 'Failed to kill session', sessionId, displayName: session.displayName, pid: session.pid, detail: message });
349
+ const code = err.code;
350
+ if (code === 'ESRCH') {
351
+ killed = true; // process already dead — treat as successful kill
352
+ }
353
+ else {
354
+ logger.error('[IngestRoutes] Failed to kill session', {
355
+ displayName: session.displayName, sessionId, pid: session.pid, error: err.message,
356
+ });
357
+ res.status(500).json({ error: 'Failed to kill session', sessionId, displayName: session.displayName, pid: session.pid, detail: err.message });
358
+ return;
359
+ }
244
360
  }
361
+ session.status = 'ended';
362
+ namePool.release(sessionId);
363
+ killedSessions.add(sessionId);
364
+ logger.info('[IngestRoutes] Session killed', {
365
+ displayName: session.displayName, sessionId, pid: session.pid,
366
+ activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,
367
+ });
368
+ res.json({ ok: true, killed: session.displayName, pid: session.pid });
245
369
  });
246
- // Reaper: periodically check for stale sessions whose heartbeat has expired
247
- const reaperInterval = setInterval(() => {
248
- const now = Date.now();
370
+ /** Mark stale active sessions as ended. */
371
+ function reapStaleSessions(now) {
249
372
  for (const [id, session] of sessions) {
250
373
  if (session.status !== 'active')
251
374
  continue;
252
- if (session.isLeader)
253
- continue; // leader manages itself
254
- if (session.kind === 'console')
255
- continue; // console session has no heartbeat (#1805)
375
+ if (session.isLeader || session.kind === 'console')
376
+ continue;
256
377
  const age = now - new Date(session.lastHeartbeat).getTime();
257
- if (age > SESSION_STALE_MS) {
258
- session.status = 'ended';
259
- namePool.release(id);
260
- logger.info('[IngestRoutes] Reaped stale session', {
261
- displayName: session.displayName, sessionId: id, pid: session.pid,
262
- lastHeartbeatAgo: `${Math.round(age / 1000)}s`,
263
- activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,
264
- });
265
- broadcasts.sessionBroadcast?.(session);
378
+ if (age <= SESSION_STALE_MS)
379
+ continue;
380
+ session.status = 'ended';
381
+ namePool.release(id);
382
+ logger.info('[IngestRoutes] Reaped stale session', {
383
+ displayName: session.displayName, sessionId: id, pid: session.pid,
384
+ lastHeartbeatAgo: `${Math.round(age / 1000)}s`,
385
+ activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,
386
+ });
387
+ broadcasts.sessionBroadcast?.(session);
388
+ }
389
+ }
390
+ /** Delete ended sessions to bound memory (#1870). */
391
+ function purgeStaleEntries(now) {
392
+ for (const [id, session] of sessions) {
393
+ if (session.status === 'ended' && now - new Date(session.lastHeartbeat).getTime() > ENDED_PURGE_MS) {
394
+ sessions.delete(id);
266
395
  }
267
396
  }
397
+ }
398
+ const reaperInterval = setInterval(() => {
399
+ const now = Date.now();
400
+ reapStaleSessions(now);
401
+ purgeStaleEntries(now);
268
402
  }, REAPER_INTERVAL_MS);
269
403
  reaperInterval.unref();
270
404
  function getSessions() {
@@ -313,4 +447,4 @@ export function createIngestRoutes(broadcasts) {
313
447
  }
314
448
  return { router, getSessions, registerLeaderSession, registerConsoleSession };
315
449
  }
316
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"IngestRoutes.js","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAO,OAAO,EAAE,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAI1C,OAAO,EAAE,wBAAwB,EAAE,MAAM,yCAAyC,CAAC;AACnF,OAAO,EAAE,gBAAgB,EAAE,MAAM,+CAA+C,CAAC;AACjF,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACpD,OAAO,EAAE,MAAM,EAAE,MAAM,uBAAuB,CAAC;AAC/C,OAAO,EAAE,GAAG,EAAE,MAAM,qBAAqB,CAAC;AAE1C,kDAAkD;AAClD,MAAM,gBAAgB,GAAG,KAAK,CAAC;AAE/B,qDAAqD;AACrD,MAAM,cAAc,GAAG,IAAI,CAAC;AAC5B,MAAM,oBAAoB,GAAG,MAAM,CAAC;AAEpC,iDAAiD;AACjD,MAAM,kBAAkB,GAAG,KAAK,CAAC;AAEjC,6EAA6E;AAC7E,MAAM,gBAAgB,GAAG,MAAM,CAAC;AA6EhC,6DAA6D;AAC7D,SAAS,cAAc,CAAC,CAAS;IAC/B,OAAO,gBAAgB,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC;AACzD,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,kBAAkB,CAAC,UAA4B;IAC7D,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IACxB,MAAM,QAAQ,GAAG,IAAI,GAAG,EAAuB,CAAC;IAChD,MAAM,QAAQ,GAAG,IAAI,eAAe,EAAE,CAAC;IACvC,MAAM,WAAW,GAAG,IAAI,wBAAwB,CAAC,cAAc,EAAE,oBAAoB,CAAC,CAAC;IAEvF,oCAAoC;IACpC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,CAAC,CAAC,CAAC;IAEtD;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QAC9D,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,EAAE,CAAC;YAC9B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAC,CAAC;YACvD,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,GAAG,CAAC,IAAwB,CAAC;QAC7C,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3D,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,oCAAoC,EAAE,EAAE,QAAQ,EAAE,YAAY,EAAE,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,KAAK,CAAC,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC;YACjJ,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,SAAS,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YACjG,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,IAAI,KAAK,GAAG,CAAC,CAAC;QACd,IAAI,OAAO,GAAG,CAAC,CAAC;QAChB,KAAK,MAAM,KAAK,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpC,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,CAAC,OAAO,KAAK,QAAQ,EAAE,CAAC;gBAAC,OAAO,EAAE,CAAC;gBAAC,SAAS;YAAC,CAAC;YACzE,MAAM,OAAO,GAAoB;gBAC/B,GAAG,KAAK;gBACR,IAAI,EAAE,EAAE,GAAG,KAAK,CAAC,IAAI,EAAE,UAAU,EAAE,OAAO,CAAC,SAAS,EAAE;aACvD,CAAC;YACF,UAAU,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;YACjC,KAAK,EAAE,CAAC;QACV,CAAC;QAED,2BAA2B;QAC3B,MAAM,OAAO,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAChD,IAAI,OAAO,EAAE,CAAC;YACZ,OAAO,CAAC,aAAa,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QACnD,CAAC;QAED,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;YAChB,MAAM,CAAC,KAAK,CAAC,kCAAkC,OAAO,EAAE,WAAW,IAAI,OAAO,CAAC,SAAS,cAAc,KAAK,aAAa,OAAO,EAAE,CAAC,CAAC;QACrI,CAAC;QAED,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;IACrD,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QACjE,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,EAAE,CAAC;YAC9B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAC,CAAC;YACvD,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,GAAG,CAAC,IAA4B,CAAC;QACjD,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC7C,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,wCAAwC,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YACpE,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,UAAU,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YAClG,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,IAAI,UAAU,CAAC,iBAAiB,EAAE,CAAC;YACjC,UAAU,CAAC,iBAAiB,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACjD,CAAC;QAED,MAAM,OAAO,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAChD,MAAM,CAAC,KAAK,CAAC,wCAAwC,OAAO,EAAE,WAAW,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC,CAAC;QAClG,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QACjE,MAAM,OAAO,GAAG,GAAG,CAAC,IAA2B,CAAC;QAChD,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;YAC1C,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,8CAA8C,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC1E,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC/F,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAErC,QAAQ,OAAO,CAAC,KAAK,EAAE,CAAC;YACtB,KAAK,SAAS,CAAC,CAAC,CAAC;gBACf,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACvD,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;gBAChE,qEAAqE;gBACrE,gEAAgE;gBAChE,MAAM,eAAe,GAAG,OAAO,CAAE,GAAW,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;gBACjE,MAAM,IAAI,GAAgB;oBACxB,SAAS,EAAE,OAAO,CAAC,SAAS;oBAC5B,WAAW;oBACX,KAAK;oBACL,GAAG,EAAE,OAAO,CAAC,GAAG;oBAChB,SAAS,EAAE,OAAO,CAAC,SAAS,IAAI,GAAG;oBACnC,aAAa,EAAE,GAAG;oBAClB,MAAM,EAAE,QAAQ;oBAChB,QAAQ,EAAE,KAAK;oBACf,aAAa,EAAE,eAAe;oBAC9B,IAAI,EAAE,KAAK;iBACZ,CAAC;gBACF,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;gBACtC,MAAM,CAAC,IAAI,CAAC,mCAAmC,EAAE;oBAC/C,WAAW,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,KAAK;oBAClE,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM;iBACxF,CAAC,CAAC;gBACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,CAAC;gBACpC,MAAM;YACR,CAAC;YACD,KAAK,SAAS,CAAC,CAAC,CAAC;gBACf,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACjD,IAAI,QAAQ,EAAE,CAAC;oBACb,QAAQ,CAAC,MAAM,GAAG,OAAO,CAAC;oBAC1B,QAAQ,CAAC,aAAa,GAAG,GAAG,CAAC;oBAC7B,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;oBACpC,MAAM,CAAC,IAAI,CAAC,gCAAgC,EAAE;wBAC5C,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,QAAQ,CAAC,GAAG;wBAClF,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;qBAC5F,CAAC,CAAC;oBACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC,CAAC;gBAC1C,CAAC;gBACD,MAAM;YACR,CAAC;YACD,KAAK,WAAW,CAAC,CAAC,CAAC;gBACjB,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACjD,IAAI,QAAQ,EAAE,CAAC;oBACb,QAAQ,CAAC,aAAa,GAAG,GAAG,CAAC;gBAC/B,CAAC;gBACD,MAAM;YACR,CAAC;QACH,CAAC;QAED,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,GAAG,CAAC,eAAe,EAAE,KAAK,EAAE,IAAa,EAAE,GAAa,EAAE,EAAE;QACjE,MAAM,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC;QACpD,MAAM,WAAW,GAAG,GAAG,CAAC,0BAA0B,IAAI,KAAK,CAAC;QAE5D,mEAAmE;QACnE,kEAAkE;QAClE,qDAAqD;QACrD,IAAI,WAAW,KAAK,IAAI,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;gBACzC,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,IAAI,CAAC,CAAC;gBAC3D,MAAM,SAAS,GAAG,MAAM,KAAK,CAAC,oCAAoC,EAAE;oBAClE,MAAM,EAAE,UAAU,CAAC,MAAM;iBAC1B,CAAC,CAAC;gBACH,YAAY,CAAC,OAAO,CAAC,CAAC;gBACtB,IAAI,SAAS,CAAC,EAAE,EAAE,CAAC;oBACjB,MAAM,UAAU,GAAG,MAAM,SAAS,CAAC,IAAI,EAAiC,CAAC;oBACzE,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;oBAC9D,KAAK,MAAM,EAAE,IAAI,CAAC,UAAU,CAAC,QAAQ,IAAI,EAAE,CAAC,EAAE,CAAC;wBAC7C,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;4BAC1D,aAAa,CAAC,IAAI,CAAC;gCACjB,GAAG,EAAE;gCACL,aAAa,EAAE,KAAK;gCACpB,IAAI,EAAE,EAAE,CAAC,IAAI,IAAI,KAAK;6BACvB,CAAC,CAAC;wBACL,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,2DAA2D;YAC7D,CAAC;QACH,CAAC;QAED,GAAG,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,+BAA+B,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QAC3E,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,WAAW,CAAW,CAAC;QACpD,MAAM,OAAO,GAAG,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAExC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,CAAC,IAAI,CAAC,mDAAmD,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;YAChF,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,SAAS,EAAE,CAAC,CAAC;YAChE,OAAO;QACT,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;YACjB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,oBAAoB,EAAE,SAAS,EAAE,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC;YACnG,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,SAAS,CAAC,CAAC;YACrC,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;YACzB,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;YAC5B,MAAM,CAAC,IAAI,CAAC,+BAA+B,EAAE;gBAC3C,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG;gBAC7D,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;aAC5F,CAAC,CAAC;YACH,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,WAAW,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;QACxE,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,OAAO,GAAI,GAAa,CAAC,OAAO,CAAC;YACvC,MAAM,CAAC,KAAK,CAAC,uCAAuC,EAAE;gBACpD,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,KAAK,EAAE,OAAO;aAC9E,CAAC,CAAC;YACH,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,wBAAwB,EAAE,SAAS,EAAE,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,CAAC;QAC5I,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,4EAA4E;IAC5E,MAAM,cAAc,GAAG,WAAW,CAAC,GAAG,EAAE;QACtC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,KAAK,MAAM,CAAC,EAAE,EAAE,OAAO,CAAC,IAAI,QAAQ,EAAE,CAAC;YACrC,IAAI,OAAO,CAAC,MAAM,KAAK,QAAQ;gBAAE,SAAS;YAC1C,IAAI,OAAO,CAAC,QAAQ;gBAAE,SAAS,CAAC,wBAAwB;YACxD,IAAI,OAAO,CAAC,IAAI,KAAK,SAAS;gBAAE,SAAS,CAAC,2CAA2C;YACrF,MAAM,GAAG,GAAG,GAAG,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,OAAO,EAAE,CAAC;YAC5D,IAAI,GAAG,GAAG,gBAAgB,EAAE,CAAC;gBAC3B,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;gBACzB,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;gBACrB,MAAM,CAAC,IAAI,CAAC,qCAAqC,EAAE;oBACjD,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,EAAE,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG;oBACjE,gBAAgB,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG;oBAC9C,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;iBAC5F,CAAC,CAAC;gBACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,OAAO,CAAC,CAAC;YACzC,CAAC;QACH,CAAC;IACH,CAAC,EAAE,kBAAkB,CAAC,CAAC;IACvB,cAAc,CAAC,KAAK,EAAE,CAAC;IAEvB,SAAS,WAAW;QAClB,OAAO,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC;IAC1E,CAAC;IAED,SAAS,qBAAqB,CAAC,SAAiB,EAAE,GAAW;QAC3D,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QACrD,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;QACxD,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE;YACtB,SAAS;YACT,WAAW;YACX,KAAK;YACL,GAAG;YACH,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACvC,MAAM,EAAE,QAAQ;YAChB,QAAQ,EAAE,IAAI;YACd,aAAa,EAAE,IAAI;YACnB,IAAI,EAAE,KAAK;SACZ,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,0CAA0C,EAAE,EAAE,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;IAClG,CAAC;IAED;;;;OAIG;IACH,SAAS,sBAAsB;QAC7B,MAAM,SAAS,GAAG,WAAW,OAAO,CAAC,GAAG,EAAE,CAAC;QAC3C,IAAI,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC;YAAE,OAAO;QACpC,MAAM,WAAW,GAAG,aAAa,CAAC;QAClC,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE;YACtB,SAAS,EAAE,SAAS;YACpB,WAAW;YACX,KAAK,EAAE,SAAS,EAAE,6CAA6C;YAC/D,GAAG,EAAE,OAAO,CAAC,GAAG;YAChB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACvC,MAAM,EAAE,QAAQ;YAChB,QAAQ,EAAE,KAAK;YACf,aAAa,EAAE,IAAI;YACnB,IAAI,EAAE,SAAS;SAChB,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,2CAA2C,EAAE,EAAE,SAAS,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACvG,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,qBAAqB,EAAE,sBAAsB,EAAE,CAAC;AAChF,CAAC","sourcesContent":["/**\n * Event ingestion routes for the unified web console.\n *\n * The console leader mounts these routes so follower MCP servers can\n * forward their logs, metrics, and session lifecycle events. All ingested\n * entries are stamped with `_sessionId` in their data field and then\n * broadcast to SSE clients via the existing log/metrics broadcast hooks.\n *\n * Routes:\n * - POST /api/ingest/logs     — Batched log entries from a follower\n * - POST /api/ingest/metrics  — Metric snapshots from a follower\n * - POST /api/ingest/session  — Session lifecycle events (started/stopped/heartbeat)\n * - GET  /api/sessions        — Active session list for the UI\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport express, { Router } from 'express';\nimport type { Request, Response } from 'express';\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { SessionNamePool } from './SessionNames.js';\nimport { logger } from '../../utils/logger.js';\nimport { env } from '../../config/env.js';\n\n/** Maximum payload size for ingestion requests */\nconst MAX_PAYLOAD_SIZE = '1mb';\n\n/** Rate limit: max requests per window per source */\nconst RATE_LIMIT_MAX = 1000;\nconst RATE_LIMIT_WINDOW_MS = 60_000;\n\n/** How often to check for stale sessions (ms) */\nconst REAPER_INTERVAL_MS = 5_000;\n\n/** How long since last heartbeat before a session is considered dead (ms) */\nconst SESSION_STALE_MS = 15_000;\n\n/**\n * Tracked session information.\n */\nexport interface SessionInfo {\n  /** Unique identifier for this session (UUID or `console-<pid>`). */\n  sessionId: string;\n  /** Friendly puppet name (e.g., \"Kermit\", \"Punch\") or \"Web Console\". */\n  displayName: string;\n  /** Canonical hex color for this puppet character. */\n  color: string;\n  /** OS process ID of the MCP server or web console process. */\n  pid: number;\n  /** ISO timestamp when the session started. */\n  startedAt: string;\n  /** ISO timestamp of the most recent heartbeat (followers) or registration (leader/console). */\n  lastHeartbeat: string;\n  /** Lifecycle status — 'active' until ended or reaped for staleness. */\n  status: 'active' | 'ended';\n  /** True if this session won leader election and owns the token file. */\n  isLeader: boolean;\n  /** Whether this session connected with a valid Bearer token (#1805). */\n  authenticated: boolean;\n  /** Session kind — 'mcp' for MCP stdio sessions, 'console' for the web console itself (#1805). */\n  kind: 'mcp' | 'console';\n}\n\n/**\n * Payload for POST /api/ingest/logs\n */\nexport interface IngestLogPayload {\n  sessionId: string;\n  entries: UnifiedLogEntry[];\n}\n\n/**\n * Payload for POST /api/ingest/metrics\n */\nexport interface IngestMetricsPayload {\n  sessionId: string;\n  snapshot: MetricSnapshot;\n}\n\n/**\n * Payload for POST /api/ingest/session\n */\nexport interface SessionEventPayload {\n  sessionId: string;\n  event: 'started' | 'stopped' | 'heartbeat';\n  pid: number;\n  startedAt: string;\n}\n\n/**\n * Callbacks provided by the unified console orchestrator for broadcasting\n * ingested events through the existing SSE infrastructure.\n */\nexport interface IngestBroadcasts {\n  logBroadcast: (entry: UnifiedLogEntry) => void;\n  metricsOnSnapshot?: (snapshot: MetricSnapshot) => void;\n  sessionBroadcast?: (event: SessionInfo) => void;\n}\n\n/**\n * Result of creating ingest routes.\n */\nexport interface IngestRoutesResult {\n  router: Router;\n  /** Get all tracked sessions */\n  getSessions: () => SessionInfo[];\n  /** Register the leader as a session */\n  registerLeaderSession: (sessionId: string, pid: number) => void;\n  /** Register the web console as a session so the indicator is never empty (#1805) */\n  registerConsoleSession: () => void;\n}\n\n/** Normalize a string via UnicodeValidator (DMCP-SEC-004) */\nfunction normalizeInput(s: string): string {\n  return UnicodeValidator.normalize(s).normalizedContent;\n}\n\n/**\n * Create the ingestion routes and session registry.\n *\n * @param broadcasts - Callbacks to forward ingested events to SSE clients\n * @returns Router and session management functions\n */\nexport function createIngestRoutes(broadcasts: IngestBroadcasts): IngestRoutesResult {\n  const router = Router();\n  const sessions = new Map<string, SessionInfo>();\n  const namePool = new SessionNamePool();\n  const rateLimiter = new SlidingWindowRateLimiter(RATE_LIMIT_MAX, RATE_LIMIT_WINDOW_MS);\n\n  // JSON body parsing with size limit\n  router.use(express.json({ limit: MAX_PAYLOAD_SIZE }));\n\n  /**\n   * POST /api/ingest/logs — Receive batched log entries from a follower.\n   */\n  router.post('/api/ingest/logs', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestLogPayload;\n    if (!payload?.sessionId || !Array.isArray(payload.entries)) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid log payload', { received, hasSessionId: !!payload?.sessionId, hasEntries: Array.isArray(payload?.entries) });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'entries'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    let count = 0;\n    let skipped = 0;\n    for (const entry of payload.entries) {\n      if (!entry || typeof entry.message !== 'string') { skipped++; continue; }\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: payload.sessionId },\n      };\n      broadcasts.logBroadcast(stamped);\n      count++;\n    }\n\n    // Update session heartbeat\n    const session = sessions.get(payload.sessionId);\n    if (session) {\n      session.lastHeartbeat = new Date().toISOString();\n    }\n\n    if (skipped > 0) {\n      logger.debug(`[IngestRoutes] Log ingest from ${session?.displayName ?? payload.sessionId}: accepted=${count}, skipped=${skipped}`);\n    }\n\n    res.status(200).json({ accepted: count, skipped });\n  });\n\n  /**\n   * POST /api/ingest/metrics — Receive metric snapshots from a follower.\n   */\n  router.post('/api/ingest/metrics', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestMetricsPayload;\n    if (!payload?.sessionId || !payload.snapshot) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid metrics payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'snapshot'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    if (broadcasts.metricsOnSnapshot) {\n      broadcasts.metricsOnSnapshot(payload.snapshot);\n    }\n\n    const session = sessions.get(payload.sessionId);\n    logger.debug(`[IngestRoutes] Metrics ingested from ${session?.displayName ?? payload.sessionId}`);\n    res.status(200).json({ accepted: true });\n  });\n\n  /**\n   * POST /api/ingest/session — Session lifecycle events.\n   */\n  router.post('/api/ingest/session', (req: Request, res: Response) => {\n    const payload = req.body as SessionEventPayload;\n    if (!payload?.sessionId || !payload.event) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid session event payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'event'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    const now = new Date().toISOString();\n\n    switch (payload.event) {\n      case 'started': {\n        const displayName = namePool.assign(payload.sessionId);\n        const color = namePool.getColor(payload.sessionId) ?? '#3b82f6';\n        // Follower sessions that reach the ingest endpoint are authenticated\n        // if the auth middleware passed them through (token was valid).\n        const isAuthenticated = Boolean((res as any).locals?.tokenEntry);\n        const info: SessionInfo = {\n          sessionId: payload.sessionId,\n          displayName,\n          color,\n          pid: payload.pid,\n          startedAt: payload.startedAt || now,\n          lastHeartbeat: now,\n          status: 'active',\n          isLeader: false,\n          authenticated: isAuthenticated,\n          kind: 'mcp',\n        };\n        sessions.set(payload.sessionId, info);\n        logger.info('[IngestRoutes] Session registered', {\n          displayName, sessionId: payload.sessionId, pid: payload.pid, color,\n          activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length,\n        });\n        broadcasts.sessionBroadcast?.(info);\n        break;\n      }\n      case 'stopped': {\n        const existing = sessions.get(payload.sessionId);\n        if (existing) {\n          existing.status = 'ended';\n          existing.lastHeartbeat = now;\n          namePool.release(payload.sessionId);\n          logger.info('[IngestRoutes] Session stopped', {\n            displayName: existing.displayName, sessionId: payload.sessionId, pid: existing.pid,\n            activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n          });\n          broadcasts.sessionBroadcast?.(existing);\n        }\n        break;\n      }\n      case 'heartbeat': {\n        const existing = sessions.get(payload.sessionId);\n        if (existing) {\n          existing.lastHeartbeat = now;\n        }\n        break;\n      }\n    }\n\n    res.status(200).json({ ok: true });\n  });\n\n  /**\n   * GET /api/sessions — List all tracked sessions.\n   */\n  router.get('/api/sessions', async (_req: Request, res: Response) => {\n    const localSessions = Array.from(sessions.values());\n    const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;\n\n    // Federate with the legacy port (3939) to show all sessions on the\n    // machine, including unauthenticated ones from pre-auth installs.\n    // Server-to-server avoids CORS restrictions (#1805).\n    if (currentPort !== 3939) {\n      try {\n        const controller = new AbortController();\n        const timeout = setTimeout(() => controller.abort(), 2000);\n        const legacyRes = await fetch('http://127.0.0.1:3939/api/sessions', {\n          signal: controller.signal,\n        });\n        clearTimeout(timeout);\n        if (legacyRes.ok) {\n          const legacyData = await legacyRes.json() as { sessions: SessionInfo[] };\n          const localIds = new Set(localSessions.map(s => s.sessionId));\n          for (const ls of (legacyData.sessions || [])) {\n            if (!localIds.has(ls.sessionId) && ls.status === 'active') {\n              localSessions.push({\n                ...ls,\n                authenticated: false,\n                kind: ls.kind || 'mcp',\n              });\n            }\n          }\n        }\n      } catch {\n        // Legacy instance not running or unreachable — that's fine\n      }\n    }\n\n    res.json({ sessions: localSessions });\n  });\n\n  /**\n   * POST /api/sessions/:sessionId/kill — Terminate a session's server process.\n   */\n  router.post('/api/sessions/:sessionId/kill', (req: Request, res: Response) => {\n    const sessionId = req.params['sessionId'] as string;\n    const session = sessions.get(sessionId);\n\n    if (!session) {\n      logger.warn('[IngestRoutes] Kill requested for unknown session', { sessionId });\n      res.status(404).json({ error: 'Session not found', sessionId });\n      return;\n    }\n\n    if (!session.pid) {\n      res.status(400).json({ error: 'No PID for session', sessionId, displayName: session.displayName });\n      return;\n    }\n\n    try {\n      process.kill(session.pid, 'SIGTERM');\n      session.status = 'ended';\n      namePool.release(sessionId);\n      logger.info('[IngestRoutes] Session killed', {\n        displayName: session.displayName, sessionId, pid: session.pid,\n        activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n      });\n      res.json({ ok: true, killed: session.displayName, pid: session.pid });\n    } catch (err) {\n      const message = (err as Error).message;\n      logger.error('[IngestRoutes] Failed to kill session', {\n        displayName: session.displayName, sessionId, pid: session.pid, error: message,\n      });\n      res.status(500).json({ error: 'Failed to kill session', sessionId, displayName: session.displayName, pid: session.pid, detail: message });\n    }\n  });\n\n  // Reaper: periodically check for stale sessions whose heartbeat has expired\n  const reaperInterval = setInterval(() => {\n    const now = Date.now();\n    for (const [id, session] of sessions) {\n      if (session.status !== 'active') continue;\n      if (session.isLeader) continue; // leader manages itself\n      if (session.kind === 'console') continue; // console session has no heartbeat (#1805)\n      const age = now - new Date(session.lastHeartbeat).getTime();\n      if (age > SESSION_STALE_MS) {\n        session.status = 'ended';\n        namePool.release(id);\n        logger.info('[IngestRoutes] Reaped stale session', {\n          displayName: session.displayName, sessionId: id, pid: session.pid,\n          lastHeartbeatAgo: `${Math.round(age / 1000)}s`,\n          activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n        });\n        broadcasts.sessionBroadcast?.(session);\n      }\n    }\n  }, REAPER_INTERVAL_MS);\n  reaperInterval.unref();\n\n  function getSessions(): SessionInfo[] {\n    return Array.from(sessions.values()).filter(s => s.status === 'active');\n  }\n\n  function registerLeaderSession(sessionId: string, pid: number): void {\n    const displayName = namePool.assign(sessionId, true);\n    const color = namePool.getColor(sessionId) ?? '#3b82f6';\n    sessions.set(sessionId, {\n      sessionId,\n      displayName,\n      color,\n      pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: true,\n      authenticated: true,\n      kind: 'mcp',\n    });\n    logger.info('[IngestRoutes] Leader session registered', { displayName, sessionId, pid, color });\n  }\n\n  /**\n   * Register the web console itself as a session (#1805). Ensures the\n   * session indicator always shows at least one entry — the console the\n   * user is currently looking at.\n   */\n  function registerConsoleSession(): void {\n    const consoleId = `console-${process.pid}`;\n    if (sessions.has(consoleId)) return;\n    const displayName = 'Web Console';\n    sessions.set(consoleId, {\n      sessionId: consoleId,\n      displayName,\n      color: '#6366f1', // indigo — distinct from puppet greens/blues\n      pid: process.pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: false,\n      authenticated: true,\n      kind: 'console',\n    });\n    logger.info('[IngestRoutes] Console session registered', { sessionId: consoleId, pid: process.pid });\n  }\n\n  return { router, getSessions, registerLeaderSession, registerConsoleSession };\n}\n"]}
450
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"IngestRoutes.js","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAO,OAAO,EAAE,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAI1C,OAAO,EAAE,wBAAwB,EAAE,MAAM,yCAAyC,CAAC;AACnF,OAAO,EAAE,gBAAgB,EAAE,MAAM,+CAA+C,CAAC;AACjF,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACpD,OAAO,EAAE,MAAM,EAAE,MAAM,uBAAuB,CAAC;AAC/C,OAAO,EAAE,GAAG,EAAE,MAAM,qBAAqB,CAAC;AAE1C,kDAAkD;AAClD,MAAM,gBAAgB,GAAG,KAAK,CAAC;AAE/B,qDAAqD;AACrD,MAAM,cAAc,GAAG,IAAI,CAAC;AAC5B,MAAM,oBAAoB,GAAG,MAAM,CAAC;AAEpC,iDAAiD;AACjD,MAAM,kBAAkB,GAAG,KAAK,CAAC;AAEjC,6EAA6E;AAC7E,MAAM,gBAAgB,GAAG,MAAM,CAAC;AAEhC,6DAA6D;AAC7D,MAAM,uBAAuB,GAAG,KAAK,CAAC;AAEtC,kEAAkE;AAClE,MAAM,cAAc,GAAG,CAAC,GAAG,MAAM,CAAC,CAAC,YAAY;AA6E/C,6DAA6D;AAC7D,SAAS,cAAc,CAAC,CAAS;IAC/B,OAAO,gBAAgB,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC;AACzD,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,kBAAkB,CAAC,UAA4B;IAC7D,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IACxB,MAAM,QAAQ,GAAG,IAAI,GAAG,EAAuB,CAAC;IAChD,MAAM,QAAQ,GAAG,IAAI,eAAe,EAAE,CAAC;IACvC,MAAM,WAAW,GAAG,IAAI,wBAAwB,CAAC,cAAc,EAAE,oBAAoB,CAAC,CAAC;IAEvF,iEAAiE;IACjE,mFAAmF;IACnF,MAAM,cAAc,GAAG,IAAI,GAAG,EAAU,CAAC;IAEzC,6DAA6D;IAC7D,6EAA6E;IAC7E,mFAAmF;IACnF,MAAM,YAAY,GAAG,IAAI,GAAG,EAAU,CAAC;IAEvC,oDAAoD;IACpD,SAAS,qBAAqB,CAAC,SAAiB,EAAE,GAAY;QAC5D,MAAM,OAAO,GAAG,GAAG,IAAI,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,GAAG,CAAC;QACpD,IAAI,CAAC,OAAO;YAAE,OAAO;QACrB,IAAI,CAAC;YAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,kBAAkB,CAAC,CAAC;QACtE,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACzC,IAAI,QAAQ;YAAE,QAAQ,CAAC,MAAM,GAAG,OAAO,CAAC;QACxC,MAAM,CAAC,IAAI,CAAC,qDAAqD,EAAE;YACjE,WAAW,EAAE,QAAQ,EAAE,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO;SAC5D,CAAC,CAAC;IACL,CAAC;IAED,2CAA2C;IAC3C,SAAS,mBAAmB,CAAC,SAAiB,EAAE,GAAY;QAC1D,qBAAqB,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;QACtC,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAC/B,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAChC,CAAC;IAED,yEAAyE;IACzE,SAAS,YAAY,CAAC,SAAiB,EAAE,GAAY,EAAE,aAAa,GAAG,KAAK;QAC1E,IAAI,CAAC;YACH,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;YAC/C,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;YACxD,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;YACrC,MAAM,IAAI,GAAgB;gBACxB,SAAS,EAAE,WAAW,EAAE,KAAK;gBAC7B,GAAG,EAAE,GAAG,IAAI,CAAC;gBACb,SAAS,EAAE,GAAG,EAAE,aAAa,EAAE,GAAG;gBAClC,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,aAAa,EAAE,IAAI,EAAE,KAAK;aAC9D,CAAC;YACF,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;YAC9B,MAAM,CAAC,IAAI,CAAC,iDAAiD,EAAE;gBAC7D,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,WAAW;aAChE,CAAC,CAAC;YACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,CAAC;YACpC,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,yDAAyD,EAAE;gBACtE,SAAS,EAAE,KAAK,EAAG,GAAa,CAAC,OAAO;aACzC,CAAC,CAAC;YACH,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,SAAS,aAAa,CAAC,SAAiB,EAAE,GAAY,EAAE,aAAa,GAAG,KAAK;QAC3E,IAAI,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC;YAAE,OAAO,IAAI,CAAC;QAC/C,IAAI,YAAY,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC;YAChC,mBAAmB,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;YACpC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACzC,IAAI,CAAC,QAAQ;YAAE,OAAO,YAAY,CAAC,SAAS,EAAE,GAAG,EAAE,aAAa,CAAC,CAAC;QAElE,IAAI,QAAQ,CAAC,MAAM,KAAK,OAAO,EAAE,CAAC;YAChC,QAAQ,CAAC,MAAM,GAAG,QAAQ,CAAC;YAC3B,MAAM,CAAC,IAAI,CAAC,yDAAyD,EAAE;gBACrE,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS;aAC7C,CAAC,CAAC;QACL,CAAC;QACD,QAAQ,CAAC,aAAa,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAClD,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;YACzB,QAAQ,CAAC,GAAG,GAAG,GAAG,CAAC;YACnB,MAAM,CAAC,IAAI,CAAC,mDAAmD,EAAE;gBAC/D,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG;aAClD,CAAC,CAAC;QACL,CAAC;QACD,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,oCAAoC;IACpC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,CAAC,CAAC,CAAC;IAEtD;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QAC9D,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,EAAE,CAAC;YAC9B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAC,CAAC;YACvD,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,GAAG,CAAC,IAAwB,CAAC;QAC7C,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3D,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,oCAAoC,EAAE,EAAE,QAAQ,EAAE,YAAY,EAAE,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,KAAK,CAAC,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC;YACjJ,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,SAAS,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YACjG,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,IAAI,KAAK,GAAG,CAAC,CAAC;QACd,IAAI,OAAO,GAAG,CAAC,CAAC;QAChB,KAAK,MAAM,KAAK,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpC,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,CAAC,OAAO,KAAK,QAAQ,EAAE,CAAC;gBAAC,OAAO,EAAE,CAAC;gBAAC,SAAS;YAAC,CAAC;YACzE,MAAM,OAAO,GAAoB;gBAC/B,GAAG,KAAK;gBACR,IAAI,EAAE,EAAE,GAAG,KAAK,CAAC,IAAI,EAAE,UAAU,EAAE,OAAO,CAAC,SAAS,EAAE;aACvD,CAAC;YACF,UAAU,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;YACjC,KAAK,EAAE,CAAC;QACV,CAAC;QAED,4EAA4E;QAC5E,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEjD,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;YAChB,MAAM,CAAC,KAAK,CAAC,kCAAkC,OAAO,EAAE,WAAW,IAAI,OAAO,CAAC,SAAS,cAAc,KAAK,aAAa,OAAO,EAAE,CAAC,CAAC;QACrI,CAAC;QAED,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;IACrD,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QACjE,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,EAAE,CAAC;YAC9B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAC,CAAC;YACvD,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,GAAG,CAAC,IAA4B,CAAC;QACjD,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC7C,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,wCAAwC,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YACpE,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,UAAU,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YAClG,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,IAAI,UAAU,CAAC,iBAAiB,EAAE,CAAC;YACjC,UAAU,CAAC,iBAAiB,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACjD,CAAC;QAED,4EAA4E;QAC5E,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QACjD,MAAM,CAAC,KAAK,CAAC,wCAAwC,OAAO,EAAE,WAAW,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC,CAAC;QAClG,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QACjE,MAAM,OAAO,GAAG,GAAG,CAAC,IAA2B,CAAC;QAChD,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;YAC1C,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,8CAA8C,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC1E,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC/F,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAErC,QAAQ,OAAO,CAAC,KAAK,EAAE,CAAC;YACtB,KAAK,SAAS,CAAC,CAAC,CAAC;gBACf,iEAAiE;gBACjE,IAAI,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC;oBAAE,MAAM;gBACjD,IAAI,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;oBAAC,mBAAmB,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;oBAAC,MAAM;gBAAC,CAAC;gBAExG,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACvD,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;gBAChE,MAAM,eAAe,GAAG,OAAO,CAAE,GAAW,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;gBACjE,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,EAAE;oBAC9B,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,WAAW,EAAE,KAAK;oBAChD,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,IAAI,GAAG,EAAE,aAAa,EAAE,GAAG;oBACzE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,aAAa,EAAE,eAAe,EAAE,IAAI,EAAE,KAAK;iBAC/E,CAAC,CAAC;gBACH,MAAM,CAAC,IAAI,CAAC,mCAAmC,EAAE;oBAC/C,WAAW,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,KAAK;oBAClE,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM;iBACxF,CAAC,CAAC;gBACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAE,CAAC,CAAC;gBAChE,MAAM;YACR,CAAC;YACD,KAAK,SAAS,CAAC,CAAC,CAAC;gBACf,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACjD,IAAI,QAAQ,EAAE,CAAC;oBACb,QAAQ,CAAC,MAAM,GAAG,OAAO,CAAC;oBAC1B,QAAQ,CAAC,aAAa,GAAG,GAAG,CAAC;oBAC7B,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;oBACpC,MAAM,CAAC,IAAI,CAAC,gCAAgC,EAAE;wBAC5C,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,QAAQ,CAAC,GAAG;wBAClF,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;qBAC5F,CAAC,CAAC;oBACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC,CAAC;gBAC1C,CAAC;gBACD,MAAM;YACR,CAAC;YACD,KAAK,WAAW,CAAC,CAAC,CAAC;gBACjB,wEAAwE;gBACxE,aAAa,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;gBAC9C,MAAM;YACR,CAAC;QACH,CAAC;QAED,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,GAAG,CAAC,eAAe,EAAE,KAAK,EAAE,IAAa,EAAE,GAAa,EAAE,EAAE;QACjE,4EAA4E;QAC5E,0DAA0D;QAC1D,MAAM,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC;QACvF,MAAM,WAAW,GAAG,GAAG,CAAC,0BAA0B,IAAI,KAAK,CAAC;QAE5D,mEAAmE;QACnE,kEAAkE;QAClE,qDAAqD;QACrD,IAAI,WAAW,KAAK,IAAI,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;gBACzC,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,uBAAuB,CAAC,CAAC;gBAC9E,MAAM,SAAS,GAAG,MAAM,KAAK,CAAC,oCAAoC,EAAE;oBAClE,MAAM,EAAE,UAAU,CAAC,MAAM;iBAC1B,CAAC,CAAC;gBACH,YAAY,CAAC,OAAO,CAAC,CAAC;gBACtB,IAAI,SAAS,CAAC,EAAE,EAAE,CAAC;oBACjB,MAAM,UAAU,GAAG,MAAM,SAAS,CAAC,IAAI,EAAiC,CAAC;oBACzE,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;oBAC9D,KAAK,MAAM,EAAE,IAAI,CAAC,UAAU,CAAC,QAAQ,IAAI,EAAE,CAAC,EAAE,CAAC;wBAC7C,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;4BAC1D,aAAa,CAAC,IAAI,CAAC;gCACjB,GAAG,EAAE;gCACL,aAAa,EAAE,KAAK;gCACpB,IAAI,EAAE,EAAE,CAAC,IAAI,IAAI,KAAK;6BACvB,CAAC,CAAC;wBACL,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,2DAA2D;YAC7D,CAAC;QACH,CAAC;QAED,GAAG,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,+BAA+B,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;QACjF,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,WAAW,CAAW,CAAC;QACpD,MAAM,OAAO,GAAG,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAExC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,sEAAsE;YACtE,MAAM,WAAW,GAAG,GAAG,CAAC,0BAA0B,IAAI,KAAK,CAAC;YAC5D,IAAI,WAAW,KAAK,IAAI,EAAE,CAAC;gBACzB,IAAI,CAAC;oBACH,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;oBACzC,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,uBAAuB,CAAC,CAAC;oBAC9E,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,sCAAsC,kBAAkB,CAAC,SAAS,CAAC,OAAO,EAAE;wBACvG,MAAM,EAAE,MAAM;wBACd,MAAM,EAAE,UAAU,CAAC,MAAM;qBAC1B,CAAC,CAAC;oBACH,YAAY,CAAC,OAAO,CAAC,CAAC;oBACtB,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;wBAChB,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;wBACnC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;wBACf,OAAO;oBACT,CAAC;gBACH,CAAC;gBAAC,MAAM,CAAC;oBACP,oDAAoD;gBACtD,CAAC;YACH,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,mDAAmD,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;YAChF,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,SAAS,EAAE,CAAC,CAAC;YAChE,OAAO;QACT,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;YACjB,6EAA6E;YAC7E,wEAAwE;YACxE,4EAA4E;YAC5E,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;YACzB,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;YAC5B,YAAY,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YAC5B,MAAM,CAAC,IAAI,CAAC,qEAAqE,EAAE;gBACjF,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS;aAC5C,CAAC,CAAC;YACH,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,CAAC,WAAW,EAAE,MAAM,EAAE,cAAc,EAAE,CAAC,CAAC;YAC/E,OAAO;QACT,CAAC;QAED,kFAAkF;QAClF,wEAAwE;QACxE,IAAI,MAAM,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC;YACH,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,SAAS,CAAC,CAAC;YACrC,MAAM,GAAG,IAAI,CAAC;QAChB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,IAAI,GAAI,GAA6B,CAAC,IAAI,CAAC;YACjD,IAAI,IAAI,KAAK,OAAO,EAAE,CAAC;gBACrB,MAAM,GAAG,IAAI,CAAC,CAAC,kDAAkD;YACnE,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,KAAK,CAAC,uCAAuC,EAAE;oBACpD,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,KAAK,EAAG,GAAa,CAAC,OAAO;iBAC7F,CAAC,CAAC;gBACH,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,wBAAwB,EAAE,SAAS,EAAE,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,MAAM,EAAG,GAAa,CAAC,OAAO,EAAE,CAAC,CAAC;gBACzJ,OAAO;YACT,CAAC;QACH,CAAC;QACD,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;QACzB,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAC5B,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC9B,MAAM,CAAC,IAAI,CAAC,+BAA+B,EAAE;YAC3C,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG;YAC7D,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;SAC5F,CAAC,CAAC;QACH,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,WAAW,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACxE,CAAC,CAAC,CAAC;IAEH,2CAA2C;IAC3C,SAAS,iBAAiB,CAAC,GAAW;QACpC,KAAK,MAAM,CAAC,EAAE,EAAE,OAAO,CAAC,IAAI,QAAQ,EAAE,CAAC;YACrC,IAAI,OAAO,CAAC,MAAM,KAAK,QAAQ;gBAAE,SAAS;YAC1C,IAAI,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,IAAI,KAAK,SAAS;gBAAE,SAAS;YAC7D,MAAM,GAAG,GAAG,GAAG,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,OAAO,EAAE,CAAC;YAC5D,IAAI,GAAG,IAAI,gBAAgB;gBAAE,SAAS;YACtC,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;YACzB,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;YACrB,MAAM,CAAC,IAAI,CAAC,qCAAqC,EAAE;gBACjD,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,EAAE,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG;gBACjE,gBAAgB,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG;gBAC9C,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;aAC5F,CAAC,CAAC;YACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,OAAO,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAED,qDAAqD;IACrD,SAAS,iBAAiB,CAAC,GAAW;QACpC,KAAK,MAAM,CAAC,EAAE,EAAE,OAAO,CAAC,IAAI,QAAQ,EAAE,CAAC;YACrC,IAAI,OAAO,CAAC,MAAM,KAAK,OAAO,IAAI,GAAG,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,OAAO,EAAE,GAAG,cAAc,EAAE,CAAC;gBACnG,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACtB,CAAC;QACH,CAAC;IACH,CAAC;IAED,MAAM,cAAc,GAAG,WAAW,CAAC,GAAG,EAAE;QACtC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,iBAAiB,CAAC,GAAG,CAAC,CAAC;QACvB,iBAAiB,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC,EAAE,kBAAkB,CAAC,CAAC;IACvB,cAAc,CAAC,KAAK,EAAE,CAAC;IAEvB,SAAS,WAAW;QAClB,OAAO,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC;IAC1E,CAAC;IAED,SAAS,qBAAqB,CAAC,SAAiB,EAAE,GAAW;QAC3D,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QACrD,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;QACxD,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE;YACtB,SAAS;YACT,WAAW;YACX,KAAK;YACL,GAAG;YACH,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACvC,MAAM,EAAE,QAAQ;YAChB,QAAQ,EAAE,IAAI;YACd,aAAa,EAAE,IAAI;YACnB,IAAI,EAAE,KAAK;SACZ,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,0CAA0C,EAAE,EAAE,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;IAClG,CAAC;IAED;;;;OAIG;IACH,SAAS,sBAAsB;QAC7B,MAAM,SAAS,GAAG,WAAW,OAAO,CAAC,GAAG,EAAE,CAAC;QAC3C,IAAI,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC;YAAE,OAAO;QACpC,MAAM,WAAW,GAAG,aAAa,CAAC;QAClC,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE;YACtB,SAAS,EAAE,SAAS;YACpB,WAAW;YACX,KAAK,EAAE,SAAS,EAAE,6CAA6C;YAC/D,GAAG,EAAE,OAAO,CAAC,GAAG;YAChB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACvC,MAAM,EAAE,QAAQ;YAChB,QAAQ,EAAE,KAAK;YACf,aAAa,EAAE,IAAI;YACnB,IAAI,EAAE,SAAS;SAChB,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,2CAA2C,EAAE,EAAE,SAAS,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACvG,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,qBAAqB,EAAE,sBAAsB,EAAE,CAAC;AAChF,CAAC","sourcesContent":["/**\n * Event ingestion routes for the unified web console.\n *\n * The console leader mounts these routes so follower MCP servers can\n * forward their logs, metrics, and session lifecycle events. All ingested\n * entries are stamped with `_sessionId` in their data field and then\n * broadcast to SSE clients via the existing log/metrics broadcast hooks.\n *\n * Routes:\n * - POST /api/ingest/logs     — Batched log entries from a follower\n * - POST /api/ingest/metrics  — Metric snapshots from a follower\n * - POST /api/ingest/session  — Session lifecycle events (started/stopped/heartbeat)\n * - GET  /api/sessions        — Active session list for the UI\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport express, { Router } from 'express';\nimport type { Request, Response } from 'express';\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { SessionNamePool } from './SessionNames.js';\nimport { logger } from '../../utils/logger.js';\nimport { env } from '../../config/env.js';\n\n/** Maximum payload size for ingestion requests */\nconst MAX_PAYLOAD_SIZE = '1mb';\n\n/** Rate limit: max requests per window per source */\nconst RATE_LIMIT_MAX = 1000;\nconst RATE_LIMIT_WINDOW_MS = 60_000;\n\n/** How often to check for stale sessions (ms) */\nconst REAPER_INTERVAL_MS = 5_000;\n\n/** How long since last heartbeat before a session is considered dead (ms) */\nconst SESSION_STALE_MS = 15_000;\n\n/** Timeout for legacy port federation/proxy requests (ms) */\nconst LEGACY_FETCH_TIMEOUT_MS = 2_000;\n\n/** How long before ended sessions are purged from the Map (ms) */\nconst ENDED_PURGE_MS = 5 * 60_000; // 5 minutes\n\n/**\n * Tracked session information.\n */\nexport interface SessionInfo {\n  /** Unique identifier for this session (UUID or `console-<pid>`). */\n  sessionId: string;\n  /** Friendly puppet name (e.g., \"Kermit\", \"Punch\") or \"Web Console\". */\n  displayName: string;\n  /** Canonical hex color for this puppet character. */\n  color: string;\n  /** OS process ID of the MCP server or web console process. */\n  pid: number;\n  /** ISO timestamp when the session started. */\n  startedAt: string;\n  /** ISO timestamp of the most recent heartbeat (followers) or registration (leader/console). */\n  lastHeartbeat: string;\n  /** Lifecycle status — 'active' until ended or reaped for staleness. */\n  status: 'active' | 'ended';\n  /** True if this session won leader election and owns the token file. */\n  isLeader: boolean;\n  /** Whether this session connected with a valid Bearer token (#1805). */\n  authenticated: boolean;\n  /** Session kind — 'mcp' for MCP stdio sessions, 'console' for the web console itself (#1805). */\n  kind: 'mcp' | 'console';\n}\n\n/**\n * Payload for POST /api/ingest/logs\n */\nexport interface IngestLogPayload {\n  sessionId: string;\n  entries: UnifiedLogEntry[];\n}\n\n/**\n * Payload for POST /api/ingest/metrics\n */\nexport interface IngestMetricsPayload {\n  sessionId: string;\n  snapshot: MetricSnapshot;\n}\n\n/**\n * Payload for POST /api/ingest/session\n */\nexport interface SessionEventPayload {\n  sessionId: string;\n  event: 'started' | 'stopped' | 'heartbeat';\n  pid: number;\n  startedAt: string;\n}\n\n/**\n * Callbacks provided by the unified console orchestrator for broadcasting\n * ingested events through the existing SSE infrastructure.\n */\nexport interface IngestBroadcasts {\n  logBroadcast: (entry: UnifiedLogEntry) => void;\n  metricsOnSnapshot?: (snapshot: MetricSnapshot) => void;\n  sessionBroadcast?: (event: SessionInfo) => void;\n}\n\n/**\n * Result of creating ingest routes.\n */\nexport interface IngestRoutesResult {\n  router: Router;\n  /** Get all tracked sessions */\n  getSessions: () => SessionInfo[];\n  /** Register the leader as a session */\n  registerLeaderSession: (sessionId: string, pid: number) => void;\n  /** Register the web console as a session so the indicator is never empty (#1805) */\n  registerConsoleSession: () => void;\n}\n\n/** Normalize a string via UnicodeValidator (DMCP-SEC-004) */\nfunction normalizeInput(s: string): string {\n  return UnicodeValidator.normalize(s).normalizedContent;\n}\n\n/**\n * Create the ingestion routes and session registry.\n *\n * @param broadcasts - Callbacks to forward ingested events to SSE clients\n * @returns Router and session management functions\n */\nexport function createIngestRoutes(broadcasts: IngestBroadcasts): IngestRoutesResult {\n  const router = Router();\n  const sessions = new Map<string, SessionInfo>();\n  const namePool = new SessionNamePool();\n  const rateLimiter = new SlidingWindowRateLimiter(RATE_LIMIT_MAX, RATE_LIMIT_WINDOW_MS);\n\n  // Sessions the user explicitly killed — never come back (#1870).\n  // Cleared only on server restart, which is appropriate since that's a new context.\n  const killedSessions = new Set<string>();\n\n  // Sessions waiting for a PID so we can SIGTERM them (#1870).\n  // When the user dismisses a pid=0 orphan, we add it here. The next heartbeat\n  // (every 10s) carries the PID — we SIGTERM immediately and move to killedSessions.\n  const pendingKills = new Set<string>();\n\n  /** Execute a deferred kill if we now have a PID. */\n  function tryExecutePendingKill(sessionId: string, pid?: number): void {\n    const killPid = pid || sessions.get(sessionId)?.pid;\n    if (!killPid) return;\n    try { process.kill(killPid, 'SIGTERM'); } catch { /* already dead */ }\n    const existing = sessions.get(sessionId);\n    if (existing) existing.status = 'ended';\n    logger.info('[IngestRoutes] Deferred kill executed — PID arrived', {\n      displayName: existing?.displayName, sessionId, pid: killPid,\n    });\n  }\n\n  /** Promote a pending kill to permanent. */\n  function finalizePendingKill(sessionId: string, pid?: number): void {\n    tryExecutePendingKill(sessionId, pid);\n    pendingKills.delete(sessionId);\n    killedSessions.add(sessionId);\n  }\n\n  /** Create a new session entry for an orphan. Returns null on failure. */\n  function autoRegister(sessionId: string, pid?: number, authenticated = false): SessionInfo | null {\n    try {\n      const displayName = namePool.assign(sessionId);\n      const color = namePool.getColor(sessionId) ?? '#3b82f6';\n      const now = new Date().toISOString();\n      const info: SessionInfo = {\n        sessionId, displayName, color,\n        pid: pid || 0,\n        startedAt: now, lastHeartbeat: now,\n        status: 'active', isLeader: false, authenticated, kind: 'mcp',\n      };\n      sessions.set(sessionId, info);\n      logger.info('[IngestRoutes] Auto-registered orphaned session', {\n        displayName, sessionId, source: pid ? 'heartbeat' : 'ingestion',\n      });\n      broadcasts.sessionBroadcast?.(info);\n      return info;\n    } catch (err) {\n      logger.debug('[IngestRoutes] Failed to auto-register orphaned session', {\n        sessionId, error: (err as Error).message,\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Auto-register or update an orphaned session from ingestion data.\n   * Returns the session (existing or newly created), or null if killed/pending.\n   */\n  function ensureSession(sessionId: string, pid?: number, authenticated = false): SessionInfo | null {\n    if (killedSessions.has(sessionId)) return null;\n    if (pendingKills.has(sessionId)) {\n      finalizePendingKill(sessionId, pid);\n      return null;\n    }\n\n    const existing = sessions.get(sessionId);\n    if (!existing) return autoRegister(sessionId, pid, authenticated);\n\n    if (existing.status === 'ended') {\n      existing.status = 'active';\n      logger.info('[IngestRoutes] Revived ended session still sending data', {\n        displayName: existing.displayName, sessionId,\n      });\n    }\n    existing.lastHeartbeat = new Date().toISOString();\n    if (pid && !existing.pid) {\n      existing.pid = pid;\n      logger.info('[IngestRoutes] Recovered PID for orphaned session', {\n        displayName: existing.displayName, sessionId, pid,\n      });\n    }\n    return existing;\n  }\n\n  // JSON body parsing with size limit\n  router.use(express.json({ limit: MAX_PAYLOAD_SIZE }));\n\n  /**\n   * POST /api/ingest/logs — Receive batched log entries from a follower.\n   */\n  router.post('/api/ingest/logs', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestLogPayload;\n    if (!payload?.sessionId || !Array.isArray(payload.entries)) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid log payload', { received, hasSessionId: !!payload?.sessionId, hasEntries: Array.isArray(payload?.entries) });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'entries'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    let count = 0;\n    let skipped = 0;\n    for (const entry of payload.entries) {\n      if (!entry || typeof entry.message !== 'string') { skipped++; continue; }\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: payload.sessionId },\n      };\n      broadcasts.logBroadcast(stamped);\n      count++;\n    }\n\n    // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)\n    const session = ensureSession(payload.sessionId);\n\n    if (skipped > 0) {\n      logger.debug(`[IngestRoutes] Log ingest from ${session?.displayName ?? payload.sessionId}: accepted=${count}, skipped=${skipped}`);\n    }\n\n    res.status(200).json({ accepted: count, skipped });\n  });\n\n  /**\n   * POST /api/ingest/metrics — Receive metric snapshots from a follower.\n   */\n  router.post('/api/ingest/metrics', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestMetricsPayload;\n    if (!payload?.sessionId || !payload.snapshot) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid metrics payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'snapshot'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    if (broadcasts.metricsOnSnapshot) {\n      broadcasts.metricsOnSnapshot(payload.snapshot);\n    }\n\n    // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)\n    const session = ensureSession(payload.sessionId);\n    logger.debug(`[IngestRoutes] Metrics ingested from ${session?.displayName ?? payload.sessionId}`);\n    res.status(200).json({ accepted: true });\n  });\n\n  /**\n   * POST /api/ingest/session — Session lifecycle events.\n   */\n  router.post('/api/ingest/session', (req: Request, res: Response) => {\n    const payload = req.body as SessionEventPayload;\n    if (!payload?.sessionId || !payload.event) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid session event payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'event'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    const now = new Date().toISOString();\n\n    switch (payload.event) {\n      case 'started': {\n        // Killed sessions stay dead; pending kills get finalized (#1870)\n        if (killedSessions.has(payload.sessionId)) break;\n        if (pendingKills.has(payload.sessionId)) { finalizePendingKill(payload.sessionId, payload.pid); break; }\n\n        const displayName = namePool.assign(payload.sessionId);\n        const color = namePool.getColor(payload.sessionId) ?? '#3b82f6';\n        const isAuthenticated = Boolean((res as any).locals?.tokenEntry);\n        sessions.set(payload.sessionId, {\n          sessionId: payload.sessionId, displayName, color,\n          pid: payload.pid, startedAt: payload.startedAt || now, lastHeartbeat: now,\n          status: 'active', isLeader: false, authenticated: isAuthenticated, kind: 'mcp',\n        });\n        logger.info('[IngestRoutes] Session registered', {\n          displayName, sessionId: payload.sessionId, pid: payload.pid, color,\n          activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length,\n        });\n        broadcasts.sessionBroadcast?.(sessions.get(payload.sessionId)!);\n        break;\n      }\n      case 'stopped': {\n        const existing = sessions.get(payload.sessionId);\n        if (existing) {\n          existing.status = 'ended';\n          existing.lastHeartbeat = now;\n          namePool.release(payload.sessionId);\n          logger.info('[IngestRoutes] Session stopped', {\n            displayName: existing.displayName, sessionId: payload.sessionId, pid: existing.pid,\n            activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n          });\n          broadcasts.sessionBroadcast?.(existing);\n        }\n        break;\n      }\n      case 'heartbeat': {\n        // Auto-register or update — heartbeat includes PID for recovery (#1870)\n        ensureSession(payload.sessionId, payload.pid);\n        break;\n      }\n    }\n\n    res.status(200).json({ ok: true });\n  });\n\n  /**\n   * GET /api/sessions — List all tracked sessions.\n   */\n  router.get('/api/sessions', async (_req: Request, res: Response) => {\n    // Server-side active filter — the frontend also filters, but ended sessions\n    // should never leave the API to prevent stale UI (#1870).\n    const localSessions = Array.from(sessions.values()).filter(s => s.status === 'active');\n    const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;\n\n    // Federate with the legacy port (3939) to show all sessions on the\n    // machine, including unauthenticated ones from pre-auth installs.\n    // Server-to-server avoids CORS restrictions (#1805).\n    if (currentPort !== 3939) {\n      try {\n        const controller = new AbortController();\n        const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);\n        const legacyRes = await fetch('http://127.0.0.1:3939/api/sessions', {\n          signal: controller.signal,\n        });\n        clearTimeout(timeout);\n        if (legacyRes.ok) {\n          const legacyData = await legacyRes.json() as { sessions: SessionInfo[] };\n          const localIds = new Set(localSessions.map(s => s.sessionId));\n          for (const ls of (legacyData.sessions || [])) {\n            if (!localIds.has(ls.sessionId) && ls.status === 'active') {\n              localSessions.push({\n                ...ls,\n                authenticated: false,\n                kind: ls.kind || 'mcp',\n              });\n            }\n          }\n        }\n      } catch {\n        // Legacy instance not running or unreachable — that's fine\n      }\n    }\n\n    res.json({ sessions: localSessions });\n  });\n\n  /**\n   * POST /api/sessions/:sessionId/kill — Terminate a session's server process.\n   */\n  router.post('/api/sessions/:sessionId/kill', async (req: Request, res: Response) => {\n    const sessionId = req.params['sessionId'] as string;\n    const session = sessions.get(sessionId);\n\n    if (!session) {\n      // Session not in local Map — try proxying kill to legacy port (#1870)\n      const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;\n      if (currentPort !== 3939) {\n        try {\n          const controller = new AbortController();\n          const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);\n          const proxyRes = await fetch(`http://127.0.0.1:3939/api/sessions/${encodeURIComponent(sessionId)}/kill`, {\n            method: 'POST',\n            signal: controller.signal,\n          });\n          clearTimeout(timeout);\n          if (proxyRes.ok) {\n            const data = await proxyRes.json();\n            res.json(data);\n            return;\n          }\n        } catch {\n          // Legacy instance not running — fall through to 404\n        }\n      }\n      logger.warn('[IngestRoutes] Kill requested for unknown session', { sessionId });\n      res.status(404).json({ error: 'Session not found', sessionId });\n      return;\n    }\n\n    if (!session.pid) {\n      // Auto-registered orphan with unknown PID — queue for deferred kill (#1870).\n      // The next heartbeat (every ~10s) carries the PID. ensureSession() will\n      // SIGTERM the process as soon as the PID arrives. Session is gone for good.\n      session.status = 'ended';\n      namePool.release(sessionId);\n      pendingKills.add(sessionId);\n      logger.info('[IngestRoutes] Queued deferred kill — waiting for PID via heartbeat', {\n        displayName: session.displayName, sessionId,\n      });\n      res.json({ ok: true, dismissed: session.displayName, reason: 'pending-kill' });\n      return;\n    }\n\n    // SIGTERM the process. Even if it fails (ESRCH = already dead, EPERM = not ours),\n    // mark the session as permanently killed so it never reappears (#1870).\n    let killed = false;\n    try {\n      process.kill(session.pid, 'SIGTERM');\n      killed = true;\n    } catch (err) {\n      const code = (err as NodeJS.ErrnoException).code;\n      if (code === 'ESRCH') {\n        killed = true; // process already dead — treat as successful kill\n      } else {\n        logger.error('[IngestRoutes] Failed to kill session', {\n          displayName: session.displayName, sessionId, pid: session.pid, error: (err as Error).message,\n        });\n        res.status(500).json({ error: 'Failed to kill session', sessionId, displayName: session.displayName, pid: session.pid, detail: (err as Error).message });\n        return;\n      }\n    }\n    session.status = 'ended';\n    namePool.release(sessionId);\n    killedSessions.add(sessionId);\n    logger.info('[IngestRoutes] Session killed', {\n      displayName: session.displayName, sessionId, pid: session.pid,\n      activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n    });\n    res.json({ ok: true, killed: session.displayName, pid: session.pid });\n  });\n\n  /** Mark stale active sessions as ended. */\n  function reapStaleSessions(now: number): void {\n    for (const [id, session] of sessions) {\n      if (session.status !== 'active') continue;\n      if (session.isLeader || session.kind === 'console') continue;\n      const age = now - new Date(session.lastHeartbeat).getTime();\n      if (age <= SESSION_STALE_MS) continue;\n      session.status = 'ended';\n      namePool.release(id);\n      logger.info('[IngestRoutes] Reaped stale session', {\n        displayName: session.displayName, sessionId: id, pid: session.pid,\n        lastHeartbeatAgo: `${Math.round(age / 1000)}s`,\n        activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n      });\n      broadcasts.sessionBroadcast?.(session);\n    }\n  }\n\n  /** Delete ended sessions to bound memory (#1870). */\n  function purgeStaleEntries(now: number): void {\n    for (const [id, session] of sessions) {\n      if (session.status === 'ended' && now - new Date(session.lastHeartbeat).getTime() > ENDED_PURGE_MS) {\n        sessions.delete(id);\n      }\n    }\n  }\n\n  const reaperInterval = setInterval(() => {\n    const now = Date.now();\n    reapStaleSessions(now);\n    purgeStaleEntries(now);\n  }, REAPER_INTERVAL_MS);\n  reaperInterval.unref();\n\n  function getSessions(): SessionInfo[] {\n    return Array.from(sessions.values()).filter(s => s.status === 'active');\n  }\n\n  function registerLeaderSession(sessionId: string, pid: number): void {\n    const displayName = namePool.assign(sessionId, true);\n    const color = namePool.getColor(sessionId) ?? '#3b82f6';\n    sessions.set(sessionId, {\n      sessionId,\n      displayName,\n      color,\n      pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: true,\n      authenticated: true,\n      kind: 'mcp',\n    });\n    logger.info('[IngestRoutes] Leader session registered', { displayName, sessionId, pid, color });\n  }\n\n  /**\n   * Register the web console itself as a session (#1805). Ensures the\n   * session indicator always shows at least one entry — the console the\n   * user is currently looking at.\n   */\n  function registerConsoleSession(): void {\n    const consoleId = `console-${process.pid}`;\n    if (sessions.has(consoleId)) return;\n    const displayName = 'Web Console';\n    sessions.set(consoleId, {\n      sessionId: consoleId,\n      displayName,\n      color: '#6366f1', // indigo — distinct from puppet greens/blues\n      pid: process.pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: false,\n      authenticated: true,\n      kind: 'console',\n    });\n    logger.info('[IngestRoutes] Console session registered', { sessionId: consoleId, pid: process.pid });\n  }\n\n  return { router, getSessions, registerLeaderSession, registerConsoleSession };\n}\n"]}
@@ -9,12 +9,25 @@
9
9
  * @since v2.1.0 — Issue #1700
10
10
  */
11
11
  /**
12
- * Pick a random puppet name from the pool, independent of the session
13
- * name assignment. Used by the console token module (#1780) to generate
14
- * a friendly default name for newly created tokens. Does not reserve the
15
- * name — multiple callers can receive the same value.
12
+ * Famous puppets, marionettes, and puppet characters from around the world.
13
+ * Order doesn't matter the pool is shuffled on startup.
16
14
  */
17
- export declare function pickRandomPuppetName(): string;
15
+ export declare const ALL_PUPPET_NAMES: readonly string[];
16
+ /**
17
+ * Iconic attire and accessories drawn from famous dolls, puppets, and
18
+ * theatrical characters throughout history. Used to name console tokens
19
+ * so they never collide with the session puppet-name pool (#1871).
20
+ *
21
+ * Names evoke costume pieces — a token is something you wear or carry,
22
+ * not a person.
23
+ */
24
+ export declare const ALL_TOKEN_NAMES: readonly string[];
25
+ /**
26
+ * Pick a random token name from the attire pool.
27
+ * Used by the console token module to name newly created tokens (#1871).
28
+ * Drawn from a separate pool to avoid collision with session puppet names.
29
+ */
30
+ export declare function pickRandomTokenName(): string;
18
31
  /**
19
32
  * Manages friendly session name assignment from the puppet name pool.
20
33
  */
@@ -1 +1 @@
1
- {"version":3,"file":"SessionNames.d.ts","sourceRoot":"","sources":["../../../src/web/console/SessionNames.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AA6GH;;;;;GAKG;AACH,wBAAgB,oBAAoB,IAAI,MAAM,CAE7C;AA8ED;;GAEG;AACH,qBAAa,eAAe;IAC1B,oEAAoE;IACpE,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAA6B;IACtD,uCAAuC;IACvC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAA6B;IAC3D,0CAA0C;IAC1C,OAAO,CAAC,QAAQ,CAAuB;IAEvC;;;;;OAKG;IACH,MAAM,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,UAAQ,GAAG,MAAM;IAuCnD;;OAEG;IACH,OAAO,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAehC;;OAEG;IACH,OAAO,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAI9C;;OAEG;IACH,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAK/C;;OAEG;IACH,MAAM,IAAI,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC;IAI7B,OAAO,CAAC,cAAc;CAIvB"}
1
+ {"version":3,"file":"SessionNames.d.ts","sourceRoot":"","sources":["../../../src/web/console/SessionNames.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAKH;;;GAGG;AACH,eAAO,MAAM,gBAAgB,EAAE,SAAS,MAAM,EAmF7C,CAAC;AAiBF;;;;;;;GAOG;AACH,eAAO,MAAM,eAAe,EAAE,SAAS,MAAM,EAqD5C,CAAC;AAEF;;;;GAIG;AACH,wBAAgB,mBAAmB,IAAI,MAAM,CAE5C;AA8ED;;GAEG;AACH,qBAAa,eAAe;IAC1B,oEAAoE;IACpE,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAA6B;IACtD,uCAAuC;IACvC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAA6B;IAC3D,0CAA0C;IAC1C,OAAO,CAAC,QAAQ,CAAuB;IAEvC;;;;;OAKG;IACH,MAAM,CAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,UAAQ,GAAG,MAAM;IAuCnD;;OAEG;IACH,OAAO,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAehC;;OAEG;IACH,OAAO,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAI9C;;OAEG;IACH,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS;IAK/C;;OAEG;IACH,MAAM,IAAI,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC;IAI7B,OAAO,CAAC,cAAc;CAIvB"}