@dollhousemcp/mcp-server 2.0.27-rc.4 → 2.0.27-rc.6

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.
package/CHANGELOG.md CHANGED
@@ -1,6 +1,6 @@
1
1
  # Changelog
2
2
 
3
- ## [2.0.27-rc.4] - 2026-04-19
3
+ ## [2.0.27-rc.6] - 2026-04-19
4
4
 
5
5
  - Version bump
6
6
 
@@ -2,8 +2,8 @@
2
2
  * Auto-generated file - DO NOT EDIT
3
3
  * Generated at build time by scripts/generate-version.js
4
4
  */
5
- export declare const PACKAGE_VERSION = "2.0.27-rc.4";
6
- export declare const BUILD_TIMESTAMP = "2026-04-19T17:27:21.141Z";
5
+ export declare const PACKAGE_VERSION = "2.0.27-rc.6";
6
+ export declare const BUILD_TIMESTAMP = "2026-04-19T19:47:49.427Z";
7
7
  export declare const BUILD_TYPE: 'npm' | 'git';
8
8
  export declare const PACKAGE_NAME = "@dollhousemcp/mcp-server";
9
9
  //# sourceMappingURL=version.d.ts.map
@@ -2,8 +2,8 @@
2
2
  * Auto-generated file - DO NOT EDIT
3
3
  * Generated at build time by scripts/generate-version.js
4
4
  */
5
- export const PACKAGE_VERSION = '2.0.27-rc.4';
6
- export const BUILD_TIMESTAMP = '2026-04-19T17:27:21.141Z';
5
+ export const PACKAGE_VERSION = '2.0.27-rc.6';
6
+ export const BUILD_TIMESTAMP = '2026-04-19T19:47:49.427Z';
7
7
  export const BUILD_TYPE = 'npm';
8
8
  export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
9
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsYUFBYSxDQUFDO0FBQzdDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMjctcmMuNCc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMTlUMTc6Mjc6MjEuMTQxWic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsYUFBYSxDQUFDO0FBQzdDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMjctcmMuNic7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMTlUMTk6NDc6NDkuNDI3Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
@@ -88,6 +88,8 @@ export interface IngestRoutesResult {
88
88
  router: Router;
89
89
  /** Get all tracked sessions */
90
90
  getSessions: () => SessionInfo[];
91
+ /** Import active follower sessions from a displaced leader during takeover. */
92
+ importSessions: (sessions: SessionInfo[]) => void;
91
93
  /** Register the leader as a session */
92
94
  registerLeaderSession: (sessionId: string, pid: number) => void;
93
95
  /** Register the web console as a session so the indicator is never empty (#1805) */
@@ -1 +1 @@
1
- {"version":3,"file":"IngestRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAE1C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AA+B7D;;GAEG;AACH,MAAM,WAAW,WAAW;IAC1B,oEAAoE;IACpE,SAAS,EAAE,MAAM,CAAC;IAClB,uEAAuE;IACvE,WAAW,EAAE,MAAM,CAAC;IACpB,qDAAqD;IACrD,KAAK,EAAE,MAAM,CAAC;IACd,8DAA8D;IAC9D,GAAG,EAAE,MAAM,CAAC;IACZ,8CAA8C;IAC9C,SAAS,EAAE,MAAM,CAAC;IAClB,+FAA+F;IAC/F,aAAa,EAAE,MAAM,CAAC;IACtB,uEAAuE;IACvE,MAAM,EAAE,QAAQ,GAAG,OAAO,CAAC;IAC3B,wEAAwE;IACxE,QAAQ,EAAE,OAAO,CAAC;IAClB,wEAAwE;IACxE,aAAa,EAAE,OAAO,CAAC;IACvB,iGAAiG;IACjG,IAAI,EAAE,KAAK,GAAG,SAAS,CAAC;IACxB,4DAA4D;IAC5D,aAAa,EAAE,MAAM,CAAC;IACtB,8EAA8E;IAC9E,sBAAsB,EAAE,MAAM,CAAC;CAChC;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,eAAe,EAAE,CAAC;CAC5B;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,cAAc,CAAC;CAC1B;AAED;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,SAAS,GAAG,SAAS,GAAG,WAAW,CAAC;IAC3C,GAAG,EAAE,MAAM,CAAC;IACZ,SAAS,EAAE,MAAM,CAAC;IAClB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,sBAAsB,CAAC,EAAE,MAAM,CAAC;CACjC;AAED;;;GAGG;AACH,MAAM,WAAW,gBAAgB;IAC/B,YAAY,EAAE,CAAC,KAAK,EAAE,eAAe,KAAK,IAAI,CAAC;IAC/C,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAC;IACvD,oBAAoB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,KAAK,IAAI,CAAC;IAC7E,gBAAgB,CAAC,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,IAAI,CAAC;CACjD;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,MAAM,CAAC;IACf,+BAA+B;IAC/B,WAAW,EAAE,MAAM,WAAW,EAAE,CAAC;IACjC,uCAAuC;IACvC,qBAAqB,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,KAAK,IAAI,CAAC;IAChE,oFAAoF;IACpF,sBAAsB,EAAE,MAAM,IAAI,CAAC;CACpC;AAqBD;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAAC,UAAU,EAAE,gBAAgB,GAAG,kBAAkB,CAucnF"}
1
+ {"version":3,"file":"IngestRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAE1C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAgC7D;;GAEG;AACH,MAAM,WAAW,WAAW;IAC1B,oEAAoE;IACpE,SAAS,EAAE,MAAM,CAAC;IAClB,uEAAuE;IACvE,WAAW,EAAE,MAAM,CAAC;IACpB,qDAAqD;IACrD,KAAK,EAAE,MAAM,CAAC;IACd,8DAA8D;IAC9D,GAAG,EAAE,MAAM,CAAC;IACZ,8CAA8C;IAC9C,SAAS,EAAE,MAAM,CAAC;IAClB,+FAA+F;IAC/F,aAAa,EAAE,MAAM,CAAC;IACtB,uEAAuE;IACvE,MAAM,EAAE,QAAQ,GAAG,OAAO,CAAC;IAC3B,wEAAwE;IACxE,QAAQ,EAAE,OAAO,CAAC;IAClB,wEAAwE;IACxE,aAAa,EAAE,OAAO,CAAC;IACvB,iGAAiG;IACjG,IAAI,EAAE,KAAK,GAAG,SAAS,CAAC;IACxB,4DAA4D;IAC5D,aAAa,EAAE,MAAM,CAAC;IACtB,8EAA8E;IAC9E,sBAAsB,EAAE,MAAM,CAAC;CAChC;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,eAAe,EAAE,CAAC;CAC5B;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,cAAc,CAAC;CAC1B;AAED;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,SAAS,GAAG,SAAS,GAAG,WAAW,CAAC;IAC3C,GAAG,EAAE,MAAM,CAAC;IACZ,SAAS,EAAE,MAAM,CAAC;IAClB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,sBAAsB,CAAC,EAAE,MAAM,CAAC;CACjC;AAED;;;GAGG;AACH,MAAM,WAAW,gBAAgB;IAC/B,YAAY,EAAE,CAAC,KAAK,EAAE,eAAe,KAAK,IAAI,CAAC;IAC/C,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAC;IACvD,oBAAoB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,KAAK,IAAI,CAAC;IAC7E,gBAAgB,CAAC,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,IAAI,CAAC;CACjD;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,MAAM,CAAC;IACf,+BAA+B;IAC/B,WAAW,EAAE,MAAM,WAAW,EAAE,CAAC;IACjC,+EAA+E;IAC/E,cAAc,EAAE,CAAC,QAAQ,EAAE,WAAW,EAAE,KAAK,IAAI,CAAC;IAClD,uCAAuC;IACvC,qBAAqB,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,KAAK,IAAI,CAAC;IAChE,oFAAoF;IACpF,sBAAsB,EAAE,MAAM,IAAI,CAAC;CACpC;AAqBD;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAAC,UAAU,EAAE,gBAAgB,GAAG,kBAAkB,CAwfnF"}
@@ -31,6 +31,7 @@ const RATE_LIMIT_WINDOW_MS = 60_000;
31
31
  const REAPER_INTERVAL_MS = 5_000;
32
32
  /** How long since last heartbeat before a session is considered dead (ms) */
33
33
  const SESSION_STALE_MS = 15_000;
34
+ const TAKEOVER_IMPORTED_SESSION_GRACE_MS = 60_000;
34
35
  /** Timeout for legacy port federation/proxy requests (ms) */
35
36
  const LEGACY_FETCH_TIMEOUT_MS = 2_000;
36
37
  /** How long before ended sessions are purged from the Map (ms) */
@@ -69,6 +70,7 @@ export function createIngestRoutes(broadcasts) {
69
70
  // When the user dismisses a pid=0 orphan, we add it here. The next heartbeat
70
71
  // (every 10s) carries the PID — we SIGTERM immediately and move to killedSessions.
71
72
  const pendingKills = new Set();
73
+ const importedSessionGraceUntil = new Map();
72
74
  /** Execute a deferred kill if we now have a PID. */
73
75
  function tryExecutePendingKill(sessionId, pid) {
74
76
  const killPid = pid || sessions.get(sessionId)?.pid;
@@ -134,6 +136,7 @@ export function createIngestRoutes(broadcasts) {
134
136
  if (!existing) {
135
137
  return autoRegister(sessionId, pid, authenticated, serverVersion, consoleProtocolVersion);
136
138
  }
139
+ importedSessionGraceUntil.delete(sessionId);
137
140
  if (existing.status === 'ended') {
138
141
  existing.status = 'active';
139
142
  logger.info('[IngestRoutes] Revived ended session still sending data', {
@@ -403,6 +406,12 @@ export function createIngestRoutes(broadcasts) {
403
406
  if (session.isLeader || session.kind === 'console')
404
407
  continue;
405
408
  const age = now - new Date(session.lastHeartbeat).getTime();
409
+ const graceUntil = importedSessionGraceUntil.get(id) ?? 0;
410
+ if (graceUntil > now)
411
+ continue;
412
+ if (graceUntil !== 0) {
413
+ importedSessionGraceUntil.delete(id);
414
+ }
406
415
  if (age <= SESSION_STALE_MS)
407
416
  continue;
408
417
  session.status = 'ended';
@@ -419,6 +428,7 @@ export function createIngestRoutes(broadcasts) {
419
428
  function purgeStaleEntries(now) {
420
429
  for (const [id, session] of sessions) {
421
430
  if (session.status === 'ended' && now - new Date(session.lastHeartbeat).getTime() > ENDED_PURGE_MS) {
431
+ importedSessionGraceUntil.delete(id);
422
432
  sessions.delete(id);
423
433
  }
424
434
  }
@@ -432,6 +442,43 @@ export function createIngestRoutes(broadcasts) {
432
442
  function getSessions() {
433
443
  return Array.from(sessions.values()).filter(s => s.status === 'active');
434
444
  }
445
+ function importSessions(importedSessions) {
446
+ const now = Date.now();
447
+ for (const imported of importedSessions) {
448
+ if (imported.status !== 'active')
449
+ continue;
450
+ if (imported.isLeader || imported.kind === 'console')
451
+ continue;
452
+ if (killedSessions.has(imported.sessionId) || pendingKills.has(imported.sessionId))
453
+ continue;
454
+ const normalizedSessionId = normalizeInput(imported.sessionId);
455
+ const existing = sessions.get(normalizedSessionId);
456
+ if (existing?.isLeader || existing?.kind === 'console') {
457
+ continue;
458
+ }
459
+ const displayName = imported.displayName
460
+ ? namePool.adopt(normalizedSessionId, imported.displayName)
461
+ : namePool.assign(normalizedSessionId);
462
+ const color = imported.color || namePool.getColor(normalizedSessionId) || '#3b82f6';
463
+ const merged = {
464
+ sessionId: normalizedSessionId,
465
+ displayName,
466
+ color,
467
+ pid: imported.pid || existing?.pid || 0,
468
+ startedAt: imported.startedAt || existing?.startedAt || new Date(now).toISOString(),
469
+ lastHeartbeat: imported.lastHeartbeat || existing?.lastHeartbeat || new Date(now).toISOString(),
470
+ status: 'active',
471
+ isLeader: false,
472
+ authenticated: imported.authenticated ?? existing?.authenticated ?? false,
473
+ kind: 'mcp',
474
+ serverVersion: normalizeServerVersion(imported.serverVersion || existing?.serverVersion),
475
+ consoleProtocolVersion: normalizeConsoleProtocolVersion(imported.consoleProtocolVersion ?? existing?.consoleProtocolVersion),
476
+ };
477
+ sessions.set(normalizedSessionId, merged);
478
+ importedSessionGraceUntil.set(normalizedSessionId, now + TAKEOVER_IMPORTED_SESSION_GRACE_MS);
479
+ broadcasts.sessionBroadcast?.(merged);
480
+ }
481
+ }
435
482
  function registerLeaderSession(sessionId, pid) {
436
483
  const displayName = namePool.assign(sessionId, true);
437
484
  const color = namePool.getColor(sessionId) ?? '#3b82f6';
@@ -477,6 +524,6 @@ export function createIngestRoutes(broadcasts) {
477
524
  });
478
525
  logger.info('[IngestRoutes] Console session registered', { sessionId: consoleId, pid: process.pid });
479
526
  }
480
- return { router, getSessions, registerLeaderSession, registerConsoleSession };
527
+ return { router, getSessions, importSessions, registerLeaderSession, registerConsoleSession };
481
528
  }
482
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"IngestRoutes.js","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"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","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';\nimport { PACKAGE_VERSION } from '../../generated/version.js';\nimport {\n  CONSOLE_PROTOCOL_VERSION,\n  LEGACY_CONSOLE_PROTOCOL_VERSION,\n} from './LeaderElection.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  /** DollhouseMCP package version reported by the session. */\n  serverVersion: string;\n  /** Console/session contract version used for compatibility-aware takeover. */\n  consoleProtocolVersion: number;\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  serverVersion?: string;\n  consoleProtocolVersion?: number;\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  storeMetricsSnapshot?: (snapshot: MetricSnapshot, sessionId: string) => 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\nfunction normalizeServerVersion(version?: string): string {\n  if (typeof version === 'string' && version.trim().length > 0) {\n    return version.trim();\n  }\n  return 'unknown';\n}\n\nfunction normalizeConsoleProtocolVersion(version?: number): number {\n  if (typeof version === 'number' && Number.isInteger(version) && version >= 0) {\n    return version;\n  }\n  return LEGACY_CONSOLE_PROTOCOL_VERSION;\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(\n    sessionId: string,\n    pid?: number,\n    authenticated = false,\n    serverVersion?: string,\n    consoleProtocolVersion?: number,\n  ): 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        serverVersion: normalizeServerVersion(serverVersion),\n        consoleProtocolVersion: normalizeConsoleProtocolVersion(consoleProtocolVersion),\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(\n    sessionId: string,\n    pid?: number,\n    authenticated = false,\n    serverVersion?: string,\n    consoleProtocolVersion?: number,\n  ): 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) {\n      return autoRegister(sessionId, pid, authenticated, serverVersion, consoleProtocolVersion);\n    }\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    if (serverVersion) {\n      existing.serverVersion = normalizeServerVersion(serverVersion);\n    }\n    if (consoleProtocolVersion !== undefined) {\n      existing.consoleProtocolVersion = normalizeConsoleProtocolVersion(consoleProtocolVersion);\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    if (broadcasts.storeMetricsSnapshot) {\n      broadcasts.storeMetricsSnapshot(payload.snapshot, payload.sessionId);\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          serverVersion: normalizeServerVersion(payload.serverVersion),\n          consoleProtocolVersion: normalizeConsoleProtocolVersion(payload.consoleProtocolVersion),\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(\n          payload.sessionId,\n          payload.pid,\n          false,\n          payload.serverVersion,\n          payload.consoleProtocolVersion,\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    // 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                serverVersion: normalizeServerVersion(ls.serverVersion),\n                consoleProtocolVersion: normalizeConsoleProtocolVersion(ls.consoleProtocolVersion),\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    try {\n      process.kill(session.pid, 'SIGTERM');\n    } catch (err) {\n      const code = (err as NodeJS.ErrnoException).code;\n      if (code === 'ESRCH') {\n        // 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      serverVersion: PACKAGE_VERSION,\n      consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\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      serverVersion: PACKAGE_VERSION,\n      consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\n    });\n    logger.info('[IngestRoutes] Console session registered', { sessionId: consoleId, pid: process.pid });\n  }\n\n  return { router, getSessions, registerLeaderSession, registerConsoleSession };\n}\n"]}
529
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"IngestRoutes.js","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"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","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';\nimport { PACKAGE_VERSION } from '../../generated/version.js';\nimport {\n  CONSOLE_PROTOCOL_VERSION,\n  LEGACY_CONSOLE_PROTOCOL_VERSION,\n} from './LeaderElection.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;\nconst TAKEOVER_IMPORTED_SESSION_GRACE_MS = 60_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  /** DollhouseMCP package version reported by the session. */\n  serverVersion: string;\n  /** Console/session contract version used for compatibility-aware takeover. */\n  consoleProtocolVersion: number;\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  serverVersion?: string;\n  consoleProtocolVersion?: number;\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  storeMetricsSnapshot?: (snapshot: MetricSnapshot, sessionId: string) => 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  /** Import active follower sessions from a displaced leader during takeover. */\n  importSessions: (sessions: SessionInfo[]) => void;\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\nfunction normalizeServerVersion(version?: string): string {\n  if (typeof version === 'string' && version.trim().length > 0) {\n    return version.trim();\n  }\n  return 'unknown';\n}\n\nfunction normalizeConsoleProtocolVersion(version?: number): number {\n  if (typeof version === 'number' && Number.isInteger(version) && version >= 0) {\n    return version;\n  }\n  return LEGACY_CONSOLE_PROTOCOL_VERSION;\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  const importedSessionGraceUntil = new Map<string, number>();\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(\n    sessionId: string,\n    pid?: number,\n    authenticated = false,\n    serverVersion?: string,\n    consoleProtocolVersion?: number,\n  ): 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        serverVersion: normalizeServerVersion(serverVersion),\n        consoleProtocolVersion: normalizeConsoleProtocolVersion(consoleProtocolVersion),\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(\n    sessionId: string,\n    pid?: number,\n    authenticated = false,\n    serverVersion?: string,\n    consoleProtocolVersion?: number,\n  ): 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) {\n      return autoRegister(sessionId, pid, authenticated, serverVersion, consoleProtocolVersion);\n    }\n\n    importedSessionGraceUntil.delete(sessionId);\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    if (serverVersion) {\n      existing.serverVersion = normalizeServerVersion(serverVersion);\n    }\n    if (consoleProtocolVersion !== undefined) {\n      existing.consoleProtocolVersion = normalizeConsoleProtocolVersion(consoleProtocolVersion);\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    if (broadcasts.storeMetricsSnapshot) {\n      broadcasts.storeMetricsSnapshot(payload.snapshot, payload.sessionId);\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          serverVersion: normalizeServerVersion(payload.serverVersion),\n          consoleProtocolVersion: normalizeConsoleProtocolVersion(payload.consoleProtocolVersion),\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(\n          payload.sessionId,\n          payload.pid,\n          false,\n          payload.serverVersion,\n          payload.consoleProtocolVersion,\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    // 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                serverVersion: normalizeServerVersion(ls.serverVersion),\n                consoleProtocolVersion: normalizeConsoleProtocolVersion(ls.consoleProtocolVersion),\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    try {\n      process.kill(session.pid, 'SIGTERM');\n    } catch (err) {\n      const code = (err as NodeJS.ErrnoException).code;\n      if (code === 'ESRCH') {\n        // 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      const graceUntil = importedSessionGraceUntil.get(id) ?? 0;\n      if (graceUntil > now) continue;\n      if (graceUntil !== 0) {\n        importedSessionGraceUntil.delete(id);\n      }\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        importedSessionGraceUntil.delete(id);\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 importSessions(importedSessions: SessionInfo[]): void {\n    const now = Date.now();\n    for (const imported of importedSessions) {\n      if (imported.status !== 'active') continue;\n      if (imported.isLeader || imported.kind === 'console') continue;\n      if (killedSessions.has(imported.sessionId) || pendingKills.has(imported.sessionId)) continue;\n\n      const normalizedSessionId = normalizeInput(imported.sessionId);\n      const existing = sessions.get(normalizedSessionId);\n      if (existing?.isLeader || existing?.kind === 'console') {\n        continue;\n      }\n\n      const displayName = imported.displayName\n        ? namePool.adopt(normalizedSessionId, imported.displayName)\n        : namePool.assign(normalizedSessionId);\n      const color = imported.color || namePool.getColor(normalizedSessionId) || '#3b82f6';\n      const merged: SessionInfo = {\n        sessionId: normalizedSessionId,\n        displayName,\n        color,\n        pid: imported.pid || existing?.pid || 0,\n        startedAt: imported.startedAt || existing?.startedAt || new Date(now).toISOString(),\n        lastHeartbeat: imported.lastHeartbeat || existing?.lastHeartbeat || new Date(now).toISOString(),\n        status: 'active',\n        isLeader: false,\n        authenticated: imported.authenticated ?? existing?.authenticated ?? false,\n        kind: 'mcp',\n        serverVersion: normalizeServerVersion(imported.serverVersion || existing?.serverVersion),\n        consoleProtocolVersion: normalizeConsoleProtocolVersion(\n          imported.consoleProtocolVersion ?? existing?.consoleProtocolVersion,\n        ),\n      };\n\n      sessions.set(normalizedSessionId, merged);\n      importedSessionGraceUntil.set(normalizedSessionId, now + TAKEOVER_IMPORTED_SESSION_GRACE_MS);\n      broadcasts.sessionBroadcast?.(merged);\n    }\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      serverVersion: PACKAGE_VERSION,\n      consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\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      serverVersion: PACKAGE_VERSION,\n      consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\n    });\n    logger.info('[IngestRoutes] Console session registered', { sessionId: consoleId, pid: process.pid });\n  }\n\n  return { router, getSessions, importSessions, registerLeaderSession, registerConsoleSession };\n}\n"]}
@@ -45,6 +45,12 @@ export declare class SessionNamePool {
45
45
  * @param isLeader - If true, follower-only names (e.g., Punch) are excluded
46
46
  */
47
47
  assign(sessionId: string, isLeader?: boolean): string;
48
+ /**
49
+ * Preserve an existing human-facing assignment during leadership handoff.
50
+ * If the requested name is already taken by another live session, the pool
51
+ * falls back to normal assignment logic rather than creating a duplicate.
52
+ */
53
+ adopt(sessionId: string, name: string, isLeader?: boolean): string;
48
54
  /**
49
55
  * Release a name back to the pool with a cooldown period.
50
56
  */
@@ -1 +1 @@
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"}
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;;;;OAIG;IACH,KAAK,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,UAAQ,GAAG,MAAM;IAiBhE;;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"}
@@ -279,6 +279,25 @@ export class SessionNamePool {
279
279
  logger.warn(`[SessionNames] Name pool exhausted, using fallback '${fallback}' for ${sessionId}`);
280
280
  return fallback;
281
281
  }
282
+ /**
283
+ * Preserve an existing human-facing assignment during leadership handoff.
284
+ * If the requested name is already taken by another live session, the pool
285
+ * falls back to normal assignment logic rather than creating a duplicate.
286
+ */
287
+ adopt(sessionId, name, isLeader = false) {
288
+ const existing = this.assigned.get(sessionId);
289
+ if (existing)
290
+ return existing;
291
+ this.flushCooldowns();
292
+ if (!this.nameToSession.has(name) && !(isLeader && FOLLOWER_ONLY_NAMES.has(name))) {
293
+ this.assigned.set(sessionId, name);
294
+ this.nameToSession.set(name, sessionId);
295
+ this.cooldown = this.cooldown.filter(entry => entry.name !== name);
296
+ logger.debug(`[SessionNames] Adopted '${name}' for ${sessionId}`);
297
+ return name;
298
+ }
299
+ return this.assign(sessionId, isLeader);
300
+ }
282
301
  /**
283
302
  * Release a name back to the pool with a cooldown period.
284
303
  */
@@ -318,4 +337,4 @@ export class SessionNamePool {
318
337
  this.cooldown = this.cooldown.filter(c => (now - c.releasedAt) < NAME_COOLDOWN_MS);
319
338
  }
320
339
  }
321
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SessionNames.js","sourceRoot":"","sources":["../../../src/web/console/SessionNames.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Friendly session names drawn from famous puppets, marionettes,\n * and puppet characters throughout history.\n *\n * Names are assigned from a pool and returned after a cooldown period\n * when sessions end. This keeps the active session list human-readable\n * and on-brand for DollhouseMCP.\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport { randomInt } from 'node:crypto';\nimport { logger } from '../../utils/logger.js';\n\n/**\n * Famous puppets, marionettes, and puppet characters from around the world.\n * Order doesn't matter — the pool is shuffled on startup.\n */\nexport const ALL_PUPPET_NAMES: readonly string[] = [\n  // Classic & traditional\n  'Punch',\n  'Judy',\n  'Pinocchio',\n  'Petrouchka',\n  'Pulcinella',\n  'Guignol',\n  'Kasperle',\n\n  // Muppets & Jim Henson\n  'Kermit',\n  'Piggy',\n  'Fozzie',\n  'Gonzo',\n  'Scooter',\n  'Rowlf',\n  'Waldorf',\n  'Statler',\n\n  // Kukla, Fran and Ollie\n  'Kukla',\n  'Ollie',\n\n  // Howdy Doody era\n  'Howdy',\n  'Clarabell',\n\n  // Sesame Street\n  'Grover',\n  'Elmo',\n  'Ernie',\n  'Bert',\n  'Oscar',\n\n  // Ventriloquist & variety\n  'Mortimer',\n  'Lambchop',\n  'Madame',\n  'Topo',\n\n  // International puppetry\n  'Bunraku',\n  'Wayang',\n  'Petrushka',\n  'Hanneschen',\n  'Vitezslav',\n\n  // Modern & pop culture\n  'Salem',\n  'Triumph',\n  'Peanut',\n  'Achmed',\n\n  // Marionette traditions\n  'Fantoccini',\n  'Saltimbanque',\n  'Burattino',\n  'Harlequin',\n\n  // Dollhouse (TV series)\n  'Echo',\n\n  // Inside jokes\n  'Spike',\n  'Angel',\n\n  // Classic dolls\n  'Barbie',\n  'Ken',\n  'Skipper',\n  'Midge',\n  'Christie',\n  'Annie',\n  'Andy',\n  'Cathy',\n  'Teddy',\n  'Xavier',\n  'Strawberry',\n  'Blythe',\n  'Ginny',\n  'Betsy',\n  'Madeline',\n];\n\n/** Names that can never be assigned to a leader session */\nconst FOLLOWER_ONLY_NAMES = new Set(['Punch']);\n\n/** Fisher-Yates shuffle using crypto.randomInt (unbiased, no modulo) */\nfunction shuffleArray<T>(arr: T[]): T[] {\n  for (let i = arr.length - 1; i > 0; i--) {\n    const j = randomInt(i + 1);\n    [arr[i], arr[j]] = [arr[j], arr[i]];\n  }\n  return arr;\n}\n\n/** Shuffled copy of the name pool — randomized on each process start */\nconst PUPPET_NAMES: string[] = shuffleArray([...ALL_PUPPET_NAMES]);\n\n/**\n * Iconic attire and accessories drawn from famous dolls, puppets, and\n * theatrical characters throughout history. Used to name console tokens\n * so they never collide with the session puppet-name pool (#1871).\n *\n * Names evoke costume pieces — a token is something you wear or carry,\n * not a person.\n */\nexport const ALL_TOKEN_NAMES: readonly string[] = [\n  // Victorian & theatrical\n  'Top Hat',\n  'Monocle',\n  'Trench Coat',\n  'Opera Cape',\n  'Opera Gloves',\n  'Velvet Cloak',\n  'Lace Collar',\n  'Silk Cravat',\n  'Waistcoat',\n  'Gilt Button',\n\n  // Phantom, masks, mystery\n  'Half Mask',\n  'Domino Mask',\n  'Feathered Mask',\n\n  // Punch & Judy / Harlequin\n  'Jester Bells',\n  'Diamond Suit',\n  'Bell Cap',\n  'Slapstick',\n  'Red Nose',\n\n  // Puppet traditions\n  'Marionette Strings',\n  'Cracked Porcelain',\n  'Papier-Mâché',\n\n  // Classic dolls & characters\n  'Pink Corvette',\n  'Red Yarn Hair',\n  'Sailor Suit',\n  'Yellow Hat',\n  'Ruby Slippers',\n  'Glass Slipper',\n  'Blue Ribbon',\n  'Striped Stockings',\n\n  // Wizard / witch / fantasy\n  'Pointed Hat',\n  'Broomstick',\n  'Silver Wand',\n  'Tin Crown',\n  'Straw Hat',\n\n  // Adventure & mystery\n  'Deerstalker',\n  'Magnifying Glass',\n  'Feathered Cap',\n  'Silver Buckle',\n  'Wicker Basket',\n];\n\n/**\n * Pick a random token name from the attire pool.\n * Used by the console token module to name newly created tokens (#1871).\n * Drawn from a separate pool to avoid collision with session puppet names.\n */\nexport function pickRandomTokenName(): string {\n  return ALL_TOKEN_NAMES[randomInt(ALL_TOKEN_NAMES.length)];\n}\n\n/**\n * Canonical colors for each puppet character.\n * Adjusted from true canonical colors for UI readability in both light/dark themes.\n */\nconst PUPPET_COLORS: Record<string, string> = {\n  'Punch':        '#DC143C', // crimson red costume\n  'Judy':         '#1E90FF', // blue dress\n  'Pinocchio':    '#DAA520', // goldenrod (wooden, yellow hat)\n  'Petrouchka':   '#B0BEC5', // blue-gray (white costume, adjusted for visibility)\n  'Pulcinella':   '#90A4AE', // gray-blue (white costume, adjusted)\n  'Guignol':      '#8B4513', // saddle brown\n  'Kasperle':     '#FF0000', // red pointed cap\n  'Kermit':       '#4CAF50', // green frog\n  'Piggy':        '#E91E8C', // hot pink (glamorous pig)\n  'Fozzie':       '#CC7722', // ochre brown bear\n  'Gonzo':        '#4169E1', // royal blue\n  'Scooter':      '#FF8C00', // dark orange\n  'Rowlf':        '#8B6914', // dark goldenrod brown dog\n  'Waldorf':      '#556B2F', // dark olive green\n  'Statler':      '#708090', // slate gray\n  'Kukla':        '#FF0000', // red nose and costume\n  'Ollie':        '#228B22', // forest green dragon\n  'Howdy':        '#E2725B', // terra cotta\n  'Clarabell':    '#FFCC00', // bright yellow clown\n  'Grover':       '#4682B4', // steel blue\n  'Elmo':         '#FF2400', // scarlet red\n  'Ernie':        '#F4A460', // sandy brown\n  'Bert':         '#FFD700', // gold yellow\n  'Oscar':        '#6B8E23', // olive drab green\n  'Mortimer':     '#DEB887', // burlywood\n  'Lambchop':     '#D4C5A9', // warm cream (adjusted from pure white)\n  'Madame':       '#800080', // purple\n  'Topo':         '#A0A0A0', // silver gray\n  'Bunraku':      '#B22222', // firebrick red\n  'Wayang':       '#6B4226', // dark leather brown (lightened for visibility)\n  'Petrushka':    '#FF4500', // orange red\n  'Hanneschen':   '#CD5C5C', // indian red\n  'Vitezslav':    '#B8860B', // dark goldenrod\n  'Salem':        '#4A4A4A', // dark gray (black cat, lightened for visibility)\n  'Triumph':      '#6F4E37', // coffee brown\n  'Peanut':       '#9370DB', // medium purple\n  'Achmed':       '#C8BFA9', // bone/parchment (lightened from beige)\n  'Fantoccini':   '#C41E3A', // cardinal red\n  'Saltimbanque': '#DAA520', // goldenrod\n  'Burattino':    '#D2691E', // chocolate brown\n  'Harlequin':    '#E60026', // diamond red\n  'Echo':         '#5C6370', // slate (dark attire, lightened for visibility)\n  'Spike':        '#E8DCC8', // platinum/bleach (lightened for readability)\n  'Angel':        '#3D3D3D', // charcoal (black duster, lightened for visibility)\n\n  // Classic dolls\n  'Barbie':       '#E91E90', // Barbie pink\n  'Ken':          '#4A90D9', // Ken blue\n  'Skipper':      '#FF6B6B', // coral red\n  'Midge':        '#E87040', // warm auburn\n  'Christie':     '#C06030', // warm brown\n  'Annie':        '#E03030', // Raggedy Ann red yarn hair\n  'Andy':         '#3070C0', // Raggedy Andy blue sailor outfit\n  'Cathy':        '#D4A574', // Chatty Cathy vintage tan\n  'Teddy':        '#A0784A', // Teddy Ruxpin bear brown\n  'Xavier':       '#5AAF4A', // Xavier Roberts / Cabbage Patch green\n  'Strawberry':   '#E8445A', // strawberry red-pink\n  'Blythe':       '#7B68EE', // big-eyed purple\n  'Ginny':        '#5B9BD5', // classic blue dress\n  'Betsy':        '#DD7694', // rose pink\n  'Madeline':     '#FFD700', // yellow hat\n};\n\n/** Cooldown period before a released name can be reused (ms) */\nconst NAME_COOLDOWN_MS = 5 * 60_000; // 5 minutes\n\ninterface CooldownEntry {\n  name: string;\n  releasedAt: number;\n}\n\n/**\n * Manages friendly session name assignment from the puppet name pool.\n */\nexport class SessionNamePool {\n  /** Names currently assigned to active sessions: sessionId → name */\n  private readonly assigned = new Map<string, string>();\n  /** Reverse lookup: name → sessionId */\n  private readonly nameToSession = new Map<string, string>();\n  /** Names in cooldown after session end */\n  private cooldown: CooldownEntry[] = [];\n\n  /**\n   * Assign a friendly name to a session.\n   * Returns an existing assignment if the session already has one.\n   *\n   * @param isLeader - If true, follower-only names (e.g., Punch) are excluded\n   */\n  assign(sessionId: string, isLeader = false): string {\n    // Already assigned?\n    const existing = this.assigned.get(sessionId);\n    if (existing) return existing;\n\n    // Flush expired cooldowns\n    this.flushCooldowns();\n\n    // Find an available name, respecting leader restrictions\n    const cooldownNames = new Set(this.cooldown.map(c => c.name));\n    const availableName = PUPPET_NAMES.find(\n      name => !this.nameToSession.has(name) &&\n              !cooldownNames.has(name) &&\n              !(isLeader && FOLLOWER_ONLY_NAMES.has(name))\n    );\n\n    if (availableName) {\n      this.assigned.set(sessionId, availableName);\n      this.nameToSession.set(availableName, sessionId);\n      logger.debug(`[SessionNames] Assigned '${availableName}' to ${sessionId}`);\n      return availableName;\n    }\n\n    // All names in use or cooling down — try cooldown names (oldest first)\n    if (this.cooldown.length > 0) {\n      const oldest = this.cooldown.shift()!;\n      this.assigned.set(sessionId, oldest.name);\n      this.nameToSession.set(oldest.name, sessionId);\n      logger.debug(`[SessionNames] Assigned '${oldest.name}' to ${sessionId} (early cooldown release)`);\n      return oldest.name;\n    }\n\n    // Truly exhausted — fall back to truncated session ID\n    const fallback = sessionId.split('-')[1] || sessionId.slice(0, 8);\n    this.assigned.set(sessionId, fallback);\n    logger.warn(`[SessionNames] Name pool exhausted, using fallback '${fallback}' for ${sessionId}`);\n    return fallback;\n  }\n\n  /**\n   * Release a name back to the pool with a cooldown period.\n   */\n  release(sessionId: string): void {\n    const name = this.assigned.get(sessionId);\n    if (!name) return;\n\n    this.assigned.delete(sessionId);\n    this.nameToSession.delete(name);\n\n    // Only cooldown puppet names, not fallback IDs\n    if (PUPPET_NAMES.includes(name)) {\n      this.cooldown.push({ name, releasedAt: Date.now() });\n    }\n\n    logger.debug(`[SessionNames] Released '${name}' from ${sessionId} (cooldown ${NAME_COOLDOWN_MS / 1000}s)`);\n  }\n\n  /**\n   * Get the friendly name for a session, or undefined if not assigned.\n   */\n  getName(sessionId: string): string | undefined {\n    return this.assigned.get(sessionId);\n  }\n\n  /**\n   * Get the canonical color for an assigned session name.\n   */\n  getColor(sessionId: string): string | undefined {\n    const name = this.assigned.get(sessionId);\n    return name ? (PUPPET_COLORS[name] ?? undefined) : undefined;\n  }\n\n  /**\n   * Get all current assignments.\n   */\n  getAll(): Map<string, string> {\n    return new Map(this.assigned);\n  }\n\n  private flushCooldowns(): void {\n    const now = Date.now();\n    this.cooldown = this.cooldown.filter(c => (now - c.releasedAt) < NAME_COOLDOWN_MS);\n  }\n}\n"]}
340
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SessionNames.js","sourceRoot":"","sources":["../../../src/web/console/SessionNames.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Friendly session names drawn from famous puppets, marionettes,\n * and puppet characters throughout history.\n *\n * Names are assigned from a pool and returned after a cooldown period\n * when sessions end. This keeps the active session list human-readable\n * and on-brand for DollhouseMCP.\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport { randomInt } from 'node:crypto';\nimport { logger } from '../../utils/logger.js';\n\n/**\n * Famous puppets, marionettes, and puppet characters from around the world.\n * Order doesn't matter — the pool is shuffled on startup.\n */\nexport const ALL_PUPPET_NAMES: readonly string[] = [\n  // Classic & traditional\n  'Punch',\n  'Judy',\n  'Pinocchio',\n  'Petrouchka',\n  'Pulcinella',\n  'Guignol',\n  'Kasperle',\n\n  // Muppets & Jim Henson\n  'Kermit',\n  'Piggy',\n  'Fozzie',\n  'Gonzo',\n  'Scooter',\n  'Rowlf',\n  'Waldorf',\n  'Statler',\n\n  // Kukla, Fran and Ollie\n  'Kukla',\n  'Ollie',\n\n  // Howdy Doody era\n  'Howdy',\n  'Clarabell',\n\n  // Sesame Street\n  'Grover',\n  'Elmo',\n  'Ernie',\n  'Bert',\n  'Oscar',\n\n  // Ventriloquist & variety\n  'Mortimer',\n  'Lambchop',\n  'Madame',\n  'Topo',\n\n  // International puppetry\n  'Bunraku',\n  'Wayang',\n  'Petrushka',\n  'Hanneschen',\n  'Vitezslav',\n\n  // Modern & pop culture\n  'Salem',\n  'Triumph',\n  'Peanut',\n  'Achmed',\n\n  // Marionette traditions\n  'Fantoccini',\n  'Saltimbanque',\n  'Burattino',\n  'Harlequin',\n\n  // Dollhouse (TV series)\n  'Echo',\n\n  // Inside jokes\n  'Spike',\n  'Angel',\n\n  // Classic dolls\n  'Barbie',\n  'Ken',\n  'Skipper',\n  'Midge',\n  'Christie',\n  'Annie',\n  'Andy',\n  'Cathy',\n  'Teddy',\n  'Xavier',\n  'Strawberry',\n  'Blythe',\n  'Ginny',\n  'Betsy',\n  'Madeline',\n];\n\n/** Names that can never be assigned to a leader session */\nconst FOLLOWER_ONLY_NAMES = new Set(['Punch']);\n\n/** Fisher-Yates shuffle using crypto.randomInt (unbiased, no modulo) */\nfunction shuffleArray<T>(arr: T[]): T[] {\n  for (let i = arr.length - 1; i > 0; i--) {\n    const j = randomInt(i + 1);\n    [arr[i], arr[j]] = [arr[j], arr[i]];\n  }\n  return arr;\n}\n\n/** Shuffled copy of the name pool — randomized on each process start */\nconst PUPPET_NAMES: string[] = shuffleArray([...ALL_PUPPET_NAMES]);\n\n/**\n * Iconic attire and accessories drawn from famous dolls, puppets, and\n * theatrical characters throughout history. Used to name console tokens\n * so they never collide with the session puppet-name pool (#1871).\n *\n * Names evoke costume pieces — a token is something you wear or carry,\n * not a person.\n */\nexport const ALL_TOKEN_NAMES: readonly string[] = [\n  // Victorian & theatrical\n  'Top Hat',\n  'Monocle',\n  'Trench Coat',\n  'Opera Cape',\n  'Opera Gloves',\n  'Velvet Cloak',\n  'Lace Collar',\n  'Silk Cravat',\n  'Waistcoat',\n  'Gilt Button',\n\n  // Phantom, masks, mystery\n  'Half Mask',\n  'Domino Mask',\n  'Feathered Mask',\n\n  // Punch & Judy / Harlequin\n  'Jester Bells',\n  'Diamond Suit',\n  'Bell Cap',\n  'Slapstick',\n  'Red Nose',\n\n  // Puppet traditions\n  'Marionette Strings',\n  'Cracked Porcelain',\n  'Papier-Mâché',\n\n  // Classic dolls & characters\n  'Pink Corvette',\n  'Red Yarn Hair',\n  'Sailor Suit',\n  'Yellow Hat',\n  'Ruby Slippers',\n  'Glass Slipper',\n  'Blue Ribbon',\n  'Striped Stockings',\n\n  // Wizard / witch / fantasy\n  'Pointed Hat',\n  'Broomstick',\n  'Silver Wand',\n  'Tin Crown',\n  'Straw Hat',\n\n  // Adventure & mystery\n  'Deerstalker',\n  'Magnifying Glass',\n  'Feathered Cap',\n  'Silver Buckle',\n  'Wicker Basket',\n];\n\n/**\n * Pick a random token name from the attire pool.\n * Used by the console token module to name newly created tokens (#1871).\n * Drawn from a separate pool to avoid collision with session puppet names.\n */\nexport function pickRandomTokenName(): string {\n  return ALL_TOKEN_NAMES[randomInt(ALL_TOKEN_NAMES.length)];\n}\n\n/**\n * Canonical colors for each puppet character.\n * Adjusted from true canonical colors for UI readability in both light/dark themes.\n */\nconst PUPPET_COLORS: Record<string, string> = {\n  'Punch':        '#DC143C', // crimson red costume\n  'Judy':         '#1E90FF', // blue dress\n  'Pinocchio':    '#DAA520', // goldenrod (wooden, yellow hat)\n  'Petrouchka':   '#B0BEC5', // blue-gray (white costume, adjusted for visibility)\n  'Pulcinella':   '#90A4AE', // gray-blue (white costume, adjusted)\n  'Guignol':      '#8B4513', // saddle brown\n  'Kasperle':     '#FF0000', // red pointed cap\n  'Kermit':       '#4CAF50', // green frog\n  'Piggy':        '#E91E8C', // hot pink (glamorous pig)\n  'Fozzie':       '#CC7722', // ochre brown bear\n  'Gonzo':        '#4169E1', // royal blue\n  'Scooter':      '#FF8C00', // dark orange\n  'Rowlf':        '#8B6914', // dark goldenrod brown dog\n  'Waldorf':      '#556B2F', // dark olive green\n  'Statler':      '#708090', // slate gray\n  'Kukla':        '#FF0000', // red nose and costume\n  'Ollie':        '#228B22', // forest green dragon\n  'Howdy':        '#E2725B', // terra cotta\n  'Clarabell':    '#FFCC00', // bright yellow clown\n  'Grover':       '#4682B4', // steel blue\n  'Elmo':         '#FF2400', // scarlet red\n  'Ernie':        '#F4A460', // sandy brown\n  'Bert':         '#FFD700', // gold yellow\n  'Oscar':        '#6B8E23', // olive drab green\n  'Mortimer':     '#DEB887', // burlywood\n  'Lambchop':     '#D4C5A9', // warm cream (adjusted from pure white)\n  'Madame':       '#800080', // purple\n  'Topo':         '#A0A0A0', // silver gray\n  'Bunraku':      '#B22222', // firebrick red\n  'Wayang':       '#6B4226', // dark leather brown (lightened for visibility)\n  'Petrushka':    '#FF4500', // orange red\n  'Hanneschen':   '#CD5C5C', // indian red\n  'Vitezslav':    '#B8860B', // dark goldenrod\n  'Salem':        '#4A4A4A', // dark gray (black cat, lightened for visibility)\n  'Triumph':      '#6F4E37', // coffee brown\n  'Peanut':       '#9370DB', // medium purple\n  'Achmed':       '#C8BFA9', // bone/parchment (lightened from beige)\n  'Fantoccini':   '#C41E3A', // cardinal red\n  'Saltimbanque': '#DAA520', // goldenrod\n  'Burattino':    '#D2691E', // chocolate brown\n  'Harlequin':    '#E60026', // diamond red\n  'Echo':         '#5C6370', // slate (dark attire, lightened for visibility)\n  'Spike':        '#E8DCC8', // platinum/bleach (lightened for readability)\n  'Angel':        '#3D3D3D', // charcoal (black duster, lightened for visibility)\n\n  // Classic dolls\n  'Barbie':       '#E91E90', // Barbie pink\n  'Ken':          '#4A90D9', // Ken blue\n  'Skipper':      '#FF6B6B', // coral red\n  'Midge':        '#E87040', // warm auburn\n  'Christie':     '#C06030', // warm brown\n  'Annie':        '#E03030', // Raggedy Ann red yarn hair\n  'Andy':         '#3070C0', // Raggedy Andy blue sailor outfit\n  'Cathy':        '#D4A574', // Chatty Cathy vintage tan\n  'Teddy':        '#A0784A', // Teddy Ruxpin bear brown\n  'Xavier':       '#5AAF4A', // Xavier Roberts / Cabbage Patch green\n  'Strawberry':   '#E8445A', // strawberry red-pink\n  'Blythe':       '#7B68EE', // big-eyed purple\n  'Ginny':        '#5B9BD5', // classic blue dress\n  'Betsy':        '#DD7694', // rose pink\n  'Madeline':     '#FFD700', // yellow hat\n};\n\n/** Cooldown period before a released name can be reused (ms) */\nconst NAME_COOLDOWN_MS = 5 * 60_000; // 5 minutes\n\ninterface CooldownEntry {\n  name: string;\n  releasedAt: number;\n}\n\n/**\n * Manages friendly session name assignment from the puppet name pool.\n */\nexport class SessionNamePool {\n  /** Names currently assigned to active sessions: sessionId → name */\n  private readonly assigned = new Map<string, string>();\n  /** Reverse lookup: name → sessionId */\n  private readonly nameToSession = new Map<string, string>();\n  /** Names in cooldown after session end */\n  private cooldown: CooldownEntry[] = [];\n\n  /**\n   * Assign a friendly name to a session.\n   * Returns an existing assignment if the session already has one.\n   *\n   * @param isLeader - If true, follower-only names (e.g., Punch) are excluded\n   */\n  assign(sessionId: string, isLeader = false): string {\n    // Already assigned?\n    const existing = this.assigned.get(sessionId);\n    if (existing) return existing;\n\n    // Flush expired cooldowns\n    this.flushCooldowns();\n\n    // Find an available name, respecting leader restrictions\n    const cooldownNames = new Set(this.cooldown.map(c => c.name));\n    const availableName = PUPPET_NAMES.find(\n      name => !this.nameToSession.has(name) &&\n              !cooldownNames.has(name) &&\n              !(isLeader && FOLLOWER_ONLY_NAMES.has(name))\n    );\n\n    if (availableName) {\n      this.assigned.set(sessionId, availableName);\n      this.nameToSession.set(availableName, sessionId);\n      logger.debug(`[SessionNames] Assigned '${availableName}' to ${sessionId}`);\n      return availableName;\n    }\n\n    // All names in use or cooling down — try cooldown names (oldest first)\n    if (this.cooldown.length > 0) {\n      const oldest = this.cooldown.shift()!;\n      this.assigned.set(sessionId, oldest.name);\n      this.nameToSession.set(oldest.name, sessionId);\n      logger.debug(`[SessionNames] Assigned '${oldest.name}' to ${sessionId} (early cooldown release)`);\n      return oldest.name;\n    }\n\n    // Truly exhausted — fall back to truncated session ID\n    const fallback = sessionId.split('-')[1] || sessionId.slice(0, 8);\n    this.assigned.set(sessionId, fallback);\n    logger.warn(`[SessionNames] Name pool exhausted, using fallback '${fallback}' for ${sessionId}`);\n    return fallback;\n  }\n\n  /**\n   * Preserve an existing human-facing assignment during leadership handoff.\n   * If the requested name is already taken by another live session, the pool\n   * falls back to normal assignment logic rather than creating a duplicate.\n   */\n  adopt(sessionId: string, name: string, isLeader = false): string {\n    const existing = this.assigned.get(sessionId);\n    if (existing) return existing;\n\n    this.flushCooldowns();\n\n    if (!this.nameToSession.has(name) && !(isLeader && FOLLOWER_ONLY_NAMES.has(name))) {\n      this.assigned.set(sessionId, name);\n      this.nameToSession.set(name, sessionId);\n      this.cooldown = this.cooldown.filter(entry => entry.name !== name);\n      logger.debug(`[SessionNames] Adopted '${name}' for ${sessionId}`);\n      return name;\n    }\n\n    return this.assign(sessionId, isLeader);\n  }\n\n  /**\n   * Release a name back to the pool with a cooldown period.\n   */\n  release(sessionId: string): void {\n    const name = this.assigned.get(sessionId);\n    if (!name) return;\n\n    this.assigned.delete(sessionId);\n    this.nameToSession.delete(name);\n\n    // Only cooldown puppet names, not fallback IDs\n    if (PUPPET_NAMES.includes(name)) {\n      this.cooldown.push({ name, releasedAt: Date.now() });\n    }\n\n    logger.debug(`[SessionNames] Released '${name}' from ${sessionId} (cooldown ${NAME_COOLDOWN_MS / 1000}s)`);\n  }\n\n  /**\n   * Get the friendly name for a session, or undefined if not assigned.\n   */\n  getName(sessionId: string): string | undefined {\n    return this.assigned.get(sessionId);\n  }\n\n  /**\n   * Get the canonical color for an assigned session name.\n   */\n  getColor(sessionId: string): string | undefined {\n    const name = this.assigned.get(sessionId);\n    return name ? (PUPPET_COLORS[name] ?? undefined) : undefined;\n  }\n\n  /**\n   * Get all current assignments.\n   */\n  getAll(): Map<string, string> {\n    return new Map(this.assigned);\n  }\n\n  private flushCooldowns(): void {\n    const now = Date.now();\n    this.cooldown = this.cooldown.filter(c => (now - c.releasedAt) < NAME_COOLDOWN_MS);\n  }\n}\n"]}