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

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.5] - 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.5";
6
+ export declare const BUILD_TIMESTAMP = "2026-04-19T18:04:08.471Z";
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.5';
6
+ export const BUILD_TIMESTAMP = '2026-04-19T18:04:08.471Z';
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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsYUFBYSxDQUFDO0FBQzdDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMjctcmMuNSc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMTlUMTg6MDQ6MDguNDcxWic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
@@ -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"]}
@@ -21,6 +21,7 @@ import { isLeaderWebConsoleReachable, forceClaimLeadership, detectLegacyLeader,
21
21
  import { LeaderForwardingLogSink, SessionHeartbeat } from './LeaderForwardingSink.js';
22
22
  import { PromotionManager } from './PromotionManager.js';
23
23
  import { findPidOnPort } from './StaleProcessRecovery.js';
24
+ import type { SessionInfo } from './IngestRoutes.js';
24
25
  /**
25
26
  * Options for starting the unified console.
26
27
  */
@@ -121,6 +122,7 @@ interface DiscoveryDependencies {
121
122
  findPidOnPortImpl?: typeof findPidOnPort;
122
123
  readLeaderLockImpl?: typeof readLeaderLock;
123
124
  }
125
+ export declare function fetchLeaderSessionsSnapshot(port: number, authToken: string | null, fetchImpl?: typeof fetch): Promise<SessionInfo[]>;
124
126
  export declare function discoverLeaderServingPort(port: number, authToken: string | null, deps?: DiscoveryDependencies): Promise<PortLeaderDiscovery>;
125
127
  interface BindFailureRecoveryDependencies extends DiscoveryDependencies {
126
128
  deleteLeaderLockImpl?: typeof deleteLeaderLock;
@@ -1 +1 @@
1
- {"version":3,"file":"UnifiedConsole.d.ts","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"UnifiedConsole.d.ts","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"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"}
@@ -99,6 +99,27 @@ export async function warnIfLegacyConsolePresent(currentPort, detect = detectLeg
99
99
  function buildDiscoveryHeaders(authToken) {
100
100
  return authToken ? { Authorization: `Bearer ${authToken}` } : {};
101
101
  }
102
+ export async function fetchLeaderSessionsSnapshot(port, authToken, fetchImpl = fetch) {
103
+ const controller = new AbortController();
104
+ const timeout = setTimeout(() => controller.abort(), LEADER_DISCOVERY_TIMEOUT_MS);
105
+ try {
106
+ const response = await fetchImpl(`http://127.0.0.1:${port}/api/sessions`, {
107
+ headers: buildDiscoveryHeaders(authToken),
108
+ signal: controller.signal,
109
+ });
110
+ if (!response.ok) {
111
+ return [];
112
+ }
113
+ const data = await response.json();
114
+ return Array.isArray(data.sessions) ? data.sessions : [];
115
+ }
116
+ catch {
117
+ return [];
118
+ }
119
+ finally {
120
+ clearTimeout(timeout);
121
+ }
122
+ }
102
123
  function buildLeaderInfoFromSession(port, ownerPid, leaderSession) {
103
124
  return {
104
125
  version: LOCK_VERSION,
@@ -452,12 +473,14 @@ async function attemptForceTakeover(options, currentElection, consolePort, prima
452
473
  election: currentElection,
453
474
  fallback: initialFallback,
454
475
  replacement: initialReplacement,
476
+ recoveredSessions: [],
455
477
  forcedKill: null,
456
478
  takeoverAttempted: false,
457
479
  reboundLockClaimed: false,
458
480
  };
459
481
  }
460
482
  const latestFallback = await discoverLeaderServingPort(consolePort, primaryToken);
483
+ const recoveredSessions = await fetchLeaderSessionsSnapshot(consolePort, primaryToken);
461
484
  const latestReplacement = evaluatePortOwnerReplacement(currentElection.leaderInfo, latestFallback);
462
485
  if (!latestReplacement.shouldEvict || latestReplacement.ownerPid === null) {
463
486
  logger.warn('[UnifiedConsole] Forced takeover target changed before eviction; skipping forced kill', {
@@ -469,6 +492,7 @@ async function attemptForceTakeover(options, currentElection, consolePort, prima
469
492
  election: currentElection,
470
493
  fallback: latestFallback,
471
494
  replacement: latestReplacement,
495
+ recoveredSessions,
472
496
  forcedKill: null,
473
497
  takeoverAttempted: false,
474
498
  reboundLockClaimed: false,
@@ -489,6 +513,7 @@ async function attemptForceTakeover(options, currentElection, consolePort, prima
489
513
  election: currentElection,
490
514
  fallback: latestFallback,
491
515
  replacement: latestReplacement,
516
+ recoveredSessions,
492
517
  forcedKill,
493
518
  takeoverAttempted: true,
494
519
  reboundLockClaimed: false,
@@ -517,6 +542,7 @@ async function attemptForceTakeover(options, currentElection, consolePort, prima
517
542
  election: reboundElection,
518
543
  fallback: latestFallback,
519
544
  replacement: latestReplacement,
545
+ recoveredSessions,
520
546
  forcedKill,
521
547
  takeoverAttempted: true,
522
548
  reboundLockClaimed,
@@ -597,10 +623,12 @@ async function startAsLeader(options, election, consolePort = DEFAULT_CONSOLE_PO
597
623
  // bindAndListen now handles EADDRINUSE by finding and killing the stale
598
624
  // process on the port, then retrying. No external retry loop needed.
599
625
  let webResult = await startWebServer(serverOpts);
626
+ let recoveredFollowerSessions = [];
600
627
  if (webResult.bindResult && !webResult.bindResult.success) {
601
628
  const forceTakeover = await attemptForceTakeover(options, election, consolePort, primaryToken.token, serverOpts, startWebServer);
602
629
  webResult = forceTakeover.webResult;
603
630
  election = forceTakeover.election;
631
+ recoveredFollowerSessions = forceTakeover.recoveredSessions;
604
632
  if (webResult.bindResult && !webResult.bindResult.success) {
605
633
  if (forceTakeover.fallback.leaderInfo) {
606
634
  logger.warn('[UnifiedConsole] Leader role aborted: bind failed, falling back to follower', {
@@ -624,6 +652,13 @@ async function startAsLeader(options, election, consolePort = DEFAULT_CONSOLE_PO
624
652
  ingestResult.registerLeaderSession(options.sessionId, process.pid);
625
653
  // Register the web console itself so the session indicator is never empty (#1805)
626
654
  ingestResult.registerConsoleSession();
655
+ if (recoveredFollowerSessions.length > 0) {
656
+ ingestResult.importSessions(recoveredFollowerSessions);
657
+ logger.info('[UnifiedConsole] Recovered follower session snapshot from displaced leader', {
658
+ sessionId: options.sessionId,
659
+ recoveredSessions: recoveredFollowerSessions.length,
660
+ });
661
+ }
627
662
  // Wire SSE broadcasts for this leader's own events
628
663
  options.wireSSEBroadcasts(webResult, options.metricsSink);
629
664
  // Now wire the live broadcast functions into the ingest routes
@@ -707,4 +742,4 @@ async function startAsFollower(options, election, consolePort = DEFAULT_CONSOLE_
707
742
  },
708
743
  };
709
744
  }
710
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UnifiedConsole.js","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Unified web console orchestrator.\n *\n * Ties together leader election, console startup, follower wiring,\n * and session lifecycle management. This is the main entry point\n * called by the DI container during deferred setup.\n *\n * Flow:\n * 1. Run leader election (read lock file, claim or follow)\n * 2. If leader: start web server on fixed port, mount ingest routes, start heartbeat\n * 3. If follower: register forwarding sinks with LogManager, start session heartbeat\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport type { MemoryLogSink } from '../../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../../metrics/sinks/MemoryMetricsSink.js';\nimport type { WebServerOptions, WebServerResult } from '../server.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { logger } from '../../utils/logger.js';\nimport {\n  electLeader,\n  isLeaderWebConsoleReachable,\n  forceClaimLeadership,\n  startHeartbeat,\n  registerLeaderCleanup,\n  detectLegacyLeader,\n  readLeaderLock,\n  deleteLeaderLock,\n  claimLeadership,\n  createLeaderInfo,\n  LOCK_VERSION,\n  CONSOLE_PROTOCOL_VERSION,\n  LEGACY_SERVER_VERSION,\n  evaluateLeaderPreference,\n  type ElectionResult,\n  type ConsoleLeaderInfo,\n  type LeaderPreferenceDecision,\n} from './LeaderElection.js';\nimport { createIngestRoutes } from './IngestRoutes.js';\nimport {\n  LeaderForwardingLogSink,\n  SessionHeartbeat,\n} from './LeaderForwardingSink.js';\nimport { PromotionManager } from './PromotionManager.js';\nimport { ConsoleTokenStore } from './consoleToken.js';\nimport {\n  findPidOnPort,\n  killStaleProcessDetailed,\n  type KillStaleProcessOutcome,\n} from './StaleProcessRecovery.js';\nimport { env } from '../../config/env.js';\n\n/**\n * Default console port from the env var. Used as fallback when no port\n * is provided via config file or options. The resolution hierarchy is:\n *   1. options.port (from config file, resolved by the DI container)\n *   2. DOLLHOUSE_WEB_CONSOLE_PORT env var\n *   3. 41715 (hardcoded default in env.ts)\n */\nconst DEFAULT_CONSOLE_PORT = env.DOLLHOUSE_WEB_CONSOLE_PORT;\nconst LEGACY_CONSOLE_FALLBACK_PORT = 3939;\nconst SYNTHETIC_PORT_OWNER_SESSION_PREFIX = 'port-owner-';\nconst LEADER_DISCOVERY_TIMEOUT_MS = 2_000;\nconst FOLLOWER_AUTHORITY_MONITOR_CONFIG = {\n  intervalMs: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_MS,\n  jitterMs: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_JITTER_MS,\n  failureThreshold: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_FAILURE_THRESHOLD,\n  failureCooldownMs: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_FAILURE_COOLDOWN_MS,\n} as const;\n\nfunction currentTimestamp(): string {\n  return new Date().toISOString();\n}\n\nfunction computeFollowerAuthorityRecheckInterval(sessionId: string): number {\n  const normalizedSessionId = UnicodeValidator.normalize(sessionId).normalizedContent;\n  let hash = 0;\n  for (let index = 0; index < normalizedSessionId.length; index += 1) {\n    const codePoint = normalizedSessionId.codePointAt(index) ?? 0;\n    hash = (hash * 31 + codePoint) >>> 0;\n    if (codePoint > 0xffff) {\n      index += 1;\n    }\n  }\n  return FOLLOWER_AUTHORITY_MONITOR_CONFIG.intervalMs + (hash % (FOLLOWER_AUTHORITY_MONITOR_CONFIG.jitterMs + 1));\n}\n\n/**\n * Options for starting the unified console.\n */\nexport interface UnifiedConsoleOptions {\n  /** This process's unique session ID */\n  sessionId: string;\n  /** Stable Dollhouse session identity shown to humans and used for persistence. */\n  stableSessionId: string;\n  /** Portfolio base directory (for startWebServer) */\n  portfolioDir: string;\n  /** Log memory sink (for console history) */\n  memorySink: MemoryLogSink;\n  /** Metrics memory sink */\n  metricsSink?: MemoryMetricsSink;\n  /** MCP-AQL handler for permission routes (typed as any to avoid circular imports) */\n  mcpAqlHandler?: any;\n  /** Callback to register a log sink with the LogManager */\n  registerLogSink: (sink: { write(entry: UnifiedLogEntry): void; flush(): Promise<void>; close(): Promise<void> }) => void;\n  /** Callback to wire SSE broadcasts after web server starts */\n  wireSSEBroadcasts: (webResult: { logBroadcast?: (entry: UnifiedLogEntry) => void; metricsOnSnapshot?: (snapshot: MetricSnapshot) => void }, metricsSink?: MemoryMetricsSink) => void;\n  /** Console port override from config file. Falls back to env var if not provided. */\n  port?: number;\n}\n\n/**\n * Result of starting the unified console.\n */\nexport interface UnifiedConsoleResult {\n  role: 'leader' | 'follower';\n  election: ElectionResult;\n  /** Port the console is running on (leader only) */\n  port?: number;\n  /** Cleanup function to call on shutdown */\n  cleanup: () => Promise<void>;\n}\n\n/**\n * Check for a running legacy (pre-authentication) DollhouseMCP console and\n * log a WARN-level message if one is found (#1794).\n *\n * Extracted from `startUnifiedConsole` so the wiring can be integration-\n * tested in isolation without spinning up a full web server and leader\n * election. The implementation is fire-and-forget: detection failures\n * are logged at DEBUG and never propagate, because a failure here must\n * not block leader election of the authenticated console.\n *\n * @param currentPort - The port the authenticated console intends to\n *                      bind to. Used in the warning message to help the\n *                      user tell the two consoles apart.\n * @param detect      - Optional injection point for the detection\n *                      function. Defaults to `detectLegacyLeader`. Tests\n *                      pass a stub.\n * @param log         - Optional injection point for the logger. Defaults\n *                      to the module logger. Tests pass a spy.\n * @returns The legacy leader info from `detect()`, or null if detection\n *          threw. Exposed so tests can assert the full result shape.\n */\nexport async function warnIfLegacyConsolePresent(\n  currentPort: number,\n  detect: typeof detectLegacyLeader = detectLegacyLeader,\n  log: typeof logger = logger,\n): Promise<Awaited<ReturnType<typeof detectLegacyLeader>> | null> {\n  try {\n    const legacy = await detect();\n    if (legacy.legacyRunning) {\n      log.warn(\n        `[UnifiedConsole] Legacy (pre-authentication) DollhouseMCP console detected ` +\n        `(pid=${legacy.pid}, port=${legacy.port}). Both consoles will run ` +\n        `independently on different ports with different security posture. ` +\n        `The authenticated console (this process) uses port ${currentPort}; ` +\n        `the legacy console uses port ${legacy.port ?? LEGACY_CONSOLE_FALLBACK_PORT}. ` +\n        `For consistent security, update the legacy installation to a ` +\n        `version with the authenticated console.`,\n      );\n    }\n    return legacy;\n  } catch (err) {\n    // Best-effort — never block election on a detection failure\n    log.debug('[UnifiedConsole] Legacy leader detection failed', {\n      error: err instanceof Error ? err.message : String(err),\n    });\n    return null;\n  }\n}\n\ninterface SessionApiRecord {\n  sessionId: string;\n  pid: number;\n  startedAt?: string;\n  lastHeartbeat?: string;\n  status?: string;\n  isLeader?: boolean;\n  kind?: string;\n  serverVersion?: string;\n  consoleProtocolVersion?: number;\n}\n\nexport interface PortLeaderDiscovery {\n  leaderInfo: ConsoleLeaderInfo | null;\n  ownerPid: number | null;\n  source: 'api' | 'lock' | 'synthetic' | 'none';\n}\n\nexport interface BindFailureRecoveryResult extends PortLeaderDiscovery {\n  lockCleanupAttempted: boolean;\n  lockCleanupPerformed: boolean;\n}\n\nexport interface PortOwnerReplacementDecision {\n  shouldEvict: boolean;\n  ownerPid: number | null;\n  preference: LeaderPreferenceDecision | null;\n}\n\ninterface ForceTakeoverAttemptResult {\n  webResult: WebServerResult;\n  election: ElectionResult;\n  fallback: PortLeaderDiscovery;\n  replacement: PortOwnerReplacementDecision;\n  forcedKill: KillStaleProcessOutcome | null;\n  takeoverAttempted: boolean;\n  reboundLockClaimed: boolean;\n}\n\ninterface FollowerAuthorityResolution {\n  election: ElectionResult;\n  discovery: PortLeaderDiscovery | null;\n  replacement: PortOwnerReplacementDecision | null;\n  forcedClaim: boolean;\n}\n\ninterface FollowerAuthorityDependencies {\n  isLeaderWebConsoleReachableImpl?: typeof isLeaderWebConsoleReachable;\n  discoverLeaderServingPortImpl?: typeof discoverLeaderServingPort;\n  forceClaimLeadershipImpl?: typeof forceClaimLeadership;\n  deleteLeaderLockImpl?: typeof deleteLeaderLock;\n}\n\ninterface FollowerAuthorityMonitorDependencies extends FollowerAuthorityDependencies {\n  resolveFollowerAuthorityImpl?: typeof resolveFollowerAuthority;\n  setIntervalImpl?: typeof setInterval;\n  clearIntervalImpl?: typeof clearInterval;\n  nowImpl?: () => number;\n}\n\ninterface DiscoveryDependencies {\n  fetchImpl?: typeof fetch;\n  findPidOnPortImpl?: typeof findPidOnPort;\n  readLeaderLockImpl?: typeof readLeaderLock;\n}\n\nfunction buildDiscoveryHeaders(authToken: string | null): Record<string, string> {\n  return authToken ? { Authorization: `Bearer ${authToken}` } : {};\n}\n\nfunction buildLeaderInfoFromSession(port: number, ownerPid: number, leaderSession: SessionApiRecord): ConsoleLeaderInfo {\n  return {\n    version: LOCK_VERSION,\n    pid: ownerPid,\n    port,\n    sessionId: UnicodeValidator.normalize(leaderSession.sessionId).normalizedContent,\n    startedAt: leaderSession.startedAt ?? currentTimestamp(),\n    heartbeat: leaderSession.lastHeartbeat ?? currentTimestamp(),\n    serverVersion: leaderSession.serverVersion ?? LEGACY_SERVER_VERSION,\n    consoleProtocolVersion: leaderSession.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\nfunction buildSyntheticLeaderInfo(port: number, ownerPid: number): ConsoleLeaderInfo {\n  const now = currentTimestamp();\n  return {\n    version: LOCK_VERSION,\n    pid: ownerPid,\n    port,\n    sessionId: `${SYNTHETIC_PORT_OWNER_SESSION_PREFIX}${ownerPid}`,\n    startedAt: now,\n    heartbeat: now,\n    serverVersion: LEGACY_SERVER_VERSION,\n    consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\nasync function discoverLeaderViaSessionsApi(\n  port: number,\n  ownerPid: number,\n  authToken: string | null,\n  fetchImpl: typeof fetch,\n): Promise<ConsoleLeaderInfo | null> {\n  const controller = new AbortController();\n  const timeout = setTimeout(() => controller.abort(), LEADER_DISCOVERY_TIMEOUT_MS);\n\n  try {\n    const response = await fetchImpl(`http://127.0.0.1:${port}/api/sessions`, {\n      headers: buildDiscoveryHeaders(authToken),\n      signal: controller.signal,\n    });\n    if (!response.ok) {\n      return null;\n    }\n\n    const payload = await response.json() as { sessions?: SessionApiRecord[] };\n    const sessions = Array.isArray(payload.sessions) ? payload.sessions : [];\n    const leaderSession = sessions.find((session) =>\n      session.pid === ownerPid &&\n      session.isLeader === true &&\n      session.kind === 'mcp' &&\n      session.status !== 'stopped'\n    );\n    return leaderSession ? buildLeaderInfoFromSession(port, ownerPid, leaderSession) : null;\n  } finally {\n    clearTimeout(timeout);\n  }\n}\n\nexport async function discoverLeaderServingPort(\n  port: number,\n  authToken: string | null,\n  deps: DiscoveryDependencies = {},\n): Promise<PortLeaderDiscovery> {\n  const fetchImpl = deps.fetchImpl ?? fetch;\n  const findPidOnPortImpl = deps.findPidOnPortImpl ?? findPidOnPort;\n  const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;\n  const ownerPid = await findPidOnPortImpl(port);\n\n  if (ownerPid !== null) {\n    try {\n      const leaderInfo = await discoverLeaderViaSessionsApi(port, ownerPid, authToken, fetchImpl);\n      if (leaderInfo) {\n        return { ownerPid, source: 'api', leaderInfo };\n      }\n    } catch (err) {\n      logger.debug('[UnifiedConsole] Failed to query active leader sessions', {\n        port,\n        ownerPid,\n        error: err instanceof Error ? err.message : String(err),\n      });\n    }\n  }\n\n  const lock = await readLeaderLockImpl();\n  if (lock?.port === port && (ownerPid === null || lock.pid === ownerPid)) {\n    return {\n      ownerPid: ownerPid ?? lock.pid,\n      source: 'lock',\n      leaderInfo: {\n        ...lock,\n        sessionId: UnicodeValidator.normalize(lock.sessionId).normalizedContent,\n      },\n    };\n  }\n\n  if (ownerPid !== null) {\n    return {\n      ownerPid,\n      source: 'synthetic',\n      leaderInfo: buildSyntheticLeaderInfo(port, ownerPid),\n    };\n  }\n\n  return { leaderInfo: null, ownerPid: null, source: 'none' };\n}\n\ninterface BindFailureRecoveryDependencies extends DiscoveryDependencies {\n  deleteLeaderLockImpl?: typeof deleteLeaderLock;\n}\n\nexport async function recoverLeaderBindFailure(\n  provisionalLeader: ConsoleLeaderInfo,\n  port: number,\n  authToken: string | null,\n  deps: BindFailureRecoveryDependencies = {},\n): Promise<BindFailureRecoveryResult> {\n  const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;\n  const deleteLeaderLockImpl = deps.deleteLeaderLockImpl ?? deleteLeaderLock;\n  logger.info('[UnifiedConsole] Leader bind recovery initiated', {\n    provisionalSessionId: provisionalLeader.sessionId,\n    provisionalPid: provisionalLeader.pid,\n    port,\n  });\n\n  let fallback = await discoverLeaderServingPort(port, authToken, deps);\n  let lockCleanupAttempted = false;\n  let lockCleanupPerformed = false;\n  const currentLock = await readLeaderLockImpl();\n  const provisionalLockMatches = (\n    currentLock?.pid === provisionalLeader.pid &&\n    currentLock.port === provisionalLeader.port &&\n    currentLock.sessionId === provisionalLeader.sessionId\n  );\n  const fallbackPointsToProvisionalLeader = (\n    fallback.leaderInfo?.pid === provisionalLeader.pid &&\n    fallback.leaderInfo.port === provisionalLeader.port &&\n    fallback.leaderInfo.sessionId === provisionalLeader.sessionId\n  );\n\n  if (provisionalLockMatches) {\n    lockCleanupAttempted = true;\n    await deleteLeaderLockImpl();\n    lockCleanupPerformed = true;\n    logger.info('[UnifiedConsole] Removed provisional leader lock after bind failure', {\n      provisionalSessionId: provisionalLeader.sessionId,\n      provisionalPid: provisionalLeader.pid,\n      port,\n    });\n    if (fallbackPointsToProvisionalLeader) {\n      fallback = await discoverLeaderServingPort(port, authToken, deps);\n    }\n  }\n\n  logger.info('[UnifiedConsole] Leader bind recovery completed', {\n    provisionalSessionId: provisionalLeader.sessionId,\n    provisionalPid: provisionalLeader.pid,\n    port,\n    discoverySource: fallback.source,\n    ownerPid: fallback.ownerPid,\n    lockCleanupAttempted,\n    lockCleanupPerformed,\n  });\n\n  return {\n    ...fallback,\n    lockCleanupAttempted,\n    lockCleanupPerformed,\n  };\n}\n\nexport function evaluatePortOwnerReplacement(\n  candidateLeader: ConsoleLeaderInfo,\n  fallback: PortLeaderDiscovery,\n): PortOwnerReplacementDecision {\n  if (!fallback.leaderInfo || fallback.ownerPid === null || fallback.ownerPid === candidateLeader.pid) {\n    return {\n      shouldEvict: false,\n      ownerPid: fallback.ownerPid,\n      preference: null,\n    };\n  }\n\n  const preference = evaluateLeaderPreference(candidateLeader, fallback.leaderInfo);\n  return {\n    shouldEvict: preference.shouldReplace,\n    ownerPid: fallback.ownerPid,\n    preference,\n  };\n}\n\nfunction buildBindFailureLogContext(\n  consolePort: number,\n  provisionalLeader: ConsoleLeaderInfo,\n  bindResult: WebServerResult['bindResult'],\n  fallback: PortLeaderDiscovery,\n  replacement?: PortOwnerReplacementDecision,\n  forcedKill?: KillStaleProcessOutcome | null,\n) {\n  return {\n    port: consolePort,\n    bindError: bindResult?.error,\n    bindDetail: bindResult?.detail,\n    provisionalLeaderPid: provisionalLeader.pid,\n    provisionalLeaderSessionId: provisionalLeader.sessionId,\n    provisionalLeaderVersion: provisionalLeader.serverVersion ?? LEGACY_SERVER_VERSION,\n    provisionalLeaderProtocolVersion: provisionalLeader.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    fallbackOwnerPid: fallback.ownerPid,\n    fallbackSource: fallback.source,\n    fallbackLeaderPid: fallback.leaderInfo?.pid,\n    fallbackLeaderSessionId: fallback.leaderInfo?.sessionId,\n    fallbackLeaderVersion: fallback.leaderInfo?.serverVersion ?? LEGACY_SERVER_VERSION,\n    fallbackLeaderProtocolVersion: fallback.leaderInfo?.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    replacementShouldEvict: replacement?.shouldEvict ?? false,\n    replacementReason: replacement?.preference?.reason,\n    forcedKillReason: forcedKill?.reason,\n    forcedKillPid: forcedKill?.pid,\n    forcedKillDetail: forcedKill?.detail,\n  };\n}\n\nfunction buildAuthorityResolutionLogContext(\n  consolePort: number,\n  electedLeader: ConsoleLeaderInfo,\n  discovery: PortLeaderDiscovery | null,\n  replacement: PortOwnerReplacementDecision | null,\n) {\n  return {\n    port: consolePort,\n    electedLeaderPid: electedLeader.pid,\n    electedLeaderSessionId: electedLeader.sessionId,\n    electedLeaderVersion: electedLeader.serverVersion ?? LEGACY_SERVER_VERSION,\n    electedLeaderProtocolVersion: electedLeader.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    servingOwnerPid: discovery?.ownerPid ?? null,\n    servingSource: discovery?.source ?? 'none',\n    servingLeaderPid: discovery?.leaderInfo?.pid ?? null,\n    servingLeaderSessionId: discovery?.leaderInfo?.sessionId ?? null,\n    servingLeaderVersion: discovery?.leaderInfo?.serverVersion ?? LEGACY_SERVER_VERSION,\n    servingLeaderProtocolVersion: discovery?.leaderInfo?.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    replacementShouldEvict: replacement?.shouldEvict ?? false,\n    replacementReason: replacement?.preference?.reason ?? null,\n  };\n}\n\nexport async function resolveFollowerAuthority(\n  sessionId: string,\n  consolePort: number,\n  election: ElectionResult,\n  deps: FollowerAuthorityDependencies = {},\n): Promise<FollowerAuthorityResolution> {\n  const isLeaderWebConsoleReachableImpl = deps.isLeaderWebConsoleReachableImpl ?? isLeaderWebConsoleReachable;\n  const discoverLeaderServingPortImpl = deps.discoverLeaderServingPortImpl ?? discoverLeaderServingPort;\n  const forceClaimLeadershipImpl = deps.forceClaimLeadershipImpl ?? forceClaimLeadership;\n  const deleteLeaderLockImpl = deps.deleteLeaderLockImpl ?? deleteLeaderLock;\n\n  const reachable = await isLeaderWebConsoleReachableImpl(election.leaderInfo);\n  if (!reachable) {\n    logger.warn('[UnifiedConsole] Elected leader is not serving the console port; forcing takeover', {\n      port: consolePort,\n      electedLeaderPid: election.leaderInfo.pid,\n      electedLeaderSessionId: election.leaderInfo.sessionId,\n    });\n    return {\n      election: await forceClaimLeadershipImpl(sessionId, consolePort),\n      discovery: null,\n      replacement: null,\n      forcedClaim: true,\n    };\n  }\n\n  const candidateLeader = createLeaderInfo(sessionId, consolePort);\n  const discovery = await discoverLeaderServingPortImpl(consolePort, null);\n  if (!discovery.leaderInfo || discovery.ownerPid === null) {\n    return {\n      election,\n      discovery,\n      replacement: null,\n      forcedClaim: false,\n    };\n  }\n\n  const replacement = evaluatePortOwnerReplacement(candidateLeader, discovery);\n  if (replacement.shouldEvict) {\n    await deleteLeaderLockImpl();\n    logger.warn(\n      discovery.ownerPid === election.leaderInfo.pid\n        ? '[UnifiedConsole] Older console leader detected on the console port; newer session will take over'\n        : '[UnifiedConsole] Split-brain console authority detected; newer session will replace the actual port owner',\n      buildAuthorityResolutionLogContext(\n        consolePort,\n        election.leaderInfo,\n        discovery,\n        replacement,\n      ),\n    );\n    return {\n      election: { role: 'leader', leaderInfo: candidateLeader },\n      discovery,\n      replacement,\n      forcedClaim: false,\n    };\n  }\n\n  if (discovery.ownerPid !== election.leaderInfo.pid) {\n    logger.warn('[UnifiedConsole] Split-brain console authority detected; following the actual port owner', buildAuthorityResolutionLogContext(\n      consolePort,\n      election.leaderInfo,\n      discovery,\n      replacement,\n    ));\n    return {\n      election: { role: 'follower', leaderInfo: discovery.leaderInfo },\n      discovery,\n      replacement,\n      forcedClaim: false,\n    };\n  }\n\n  return {\n    election,\n    discovery,\n    replacement,\n    forcedClaim: false,\n  };\n}\n\nexport function startFollowerAuthorityMonitor(\n  options: UnifiedConsoleOptions,\n  consolePort: number,\n  election: ElectionResult,\n  promotionMgr: PromotionManager,\n  forwardingSink: LeaderForwardingLogSink,\n  sessionHeartbeat: SessionHeartbeat,\n  deps: FollowerAuthorityMonitorDependencies = {},\n): () => void {\n  const resolveFollowerAuthorityImpl = deps.resolveFollowerAuthorityImpl ?? resolveFollowerAuthority;\n  const setIntervalImpl = deps.setIntervalImpl ?? setInterval;\n  const clearIntervalImpl = deps.clearIntervalImpl ?? clearInterval;\n  const nowImpl = deps.nowImpl ?? Date.now;\n  let currentElection = election;\n  let checkInProgress = false;\n  let promotionQueued = false;\n  let consecutiveFailures = 0;\n  let circuitOpenUntilMs = 0;\n  let authorityTimer: ReturnType<typeof setInterval> | null = null;\n  const recheckIntervalMs = computeFollowerAuthorityRecheckInterval(options.sessionId);\n\n  const queueAuthorityPromotion = () => {\n    queueMicrotask(() => {\n      promotionMgr.promote(forwardingSink, sessionHeartbeat)\n        .catch((err) => logger.error('[UnifiedConsole] Authority-based promotion crashed', {\n          error: String(err),\n        }));\n    });\n  };\n\n  const handleResolvedAuthority = (resolved: FollowerAuthorityResolution) => {\n    currentElection = resolved.election;\n    consecutiveFailures = 0;\n    circuitOpenUntilMs = 0;\n\n    if (resolved.election.role !== 'leader') {\n      return;\n    }\n\n    promotionQueued = true;\n    if (authorityTimer) {\n      clearIntervalImpl(authorityTimer);\n      authorityTimer = null;\n    }\n\n    logger.warn('[UnifiedConsole] Follower authority re-evaluation queued a leader takeover', {\n      sessionId: options.sessionId,\n      stableSessionId: options.stableSessionId,\n      port: consolePort,\n      recheckIntervalMs,\n      electedLeaderPid: election.leaderInfo.pid,\n      electedLeaderSessionId: election.leaderInfo.sessionId,\n      resolvedLeaderPid: resolved.election.leaderInfo.pid,\n      resolvedLeaderSessionId: resolved.election.leaderInfo.sessionId,\n      replacementReason: resolved.replacement?.preference?.reason ?? null,\n      forcedClaim: resolved.forcedClaim,\n    });\n\n    queueAuthorityPromotion();\n  };\n\n  const handleAuthorityFailure = (err: unknown) => {\n    consecutiveFailures += 1;\n    if (consecutiveFailures >= FOLLOWER_AUTHORITY_MONITOR_CONFIG.failureThreshold) {\n      circuitOpenUntilMs = nowImpl() + FOLLOWER_AUTHORITY_MONITOR_CONFIG.failureCooldownMs;\n      logger.warn('[UnifiedConsole] Follower authority re-evaluation circuit opened after repeated failures', {\n        sessionId: options.sessionId,\n        port: consolePort,\n        recheckIntervalMs,\n        consecutiveFailures,\n        circuitOpenUntilMs: new Date(circuitOpenUntilMs).toISOString(),\n      });\n      consecutiveFailures = 0;\n    }\n\n    logger.debug('[UnifiedConsole] Follower authority re-evaluation failed', {\n      error: err instanceof Error ? err.message : String(err),\n      sessionId: options.sessionId,\n      port: consolePort,\n      recheckIntervalMs,\n      circuitOpenUntilMs: circuitOpenUntilMs ? new Date(circuitOpenUntilMs).toISOString() : null,\n    });\n  };\n\n  authorityTimer = setIntervalImpl(() => {\n    if (checkInProgress || promotionQueued) {\n      return;\n    }\n\n    if (circuitOpenUntilMs > nowImpl()) {\n      return;\n    }\n\n    if (circuitOpenUntilMs !== 0) {\n      logger.info('[UnifiedConsole] Follower authority re-evaluation circuit closed; resuming checks', {\n        sessionId: options.sessionId,\n        port: consolePort,\n        recheckIntervalMs,\n      });\n      circuitOpenUntilMs = 0;\n    }\n\n    checkInProgress = true;\n    resolveFollowerAuthorityImpl(options.sessionId, consolePort, currentElection)\n      .then(handleResolvedAuthority)\n      .catch(handleAuthorityFailure)\n      .finally(() => {\n        checkInProgress = false;\n      });\n  }, recheckIntervalMs);\n  authorityTimer.unref();\n\n  return () => {\n    if (authorityTimer) {\n      clearIntervalImpl(authorityTimer);\n      authorityTimer = null;\n    }\n  };\n}\n\nasync function attemptForceTakeover(\n  options: UnifiedConsoleOptions,\n  currentElection: ElectionResult,\n  consolePort: number,\n  primaryToken: string,\n  serverOpts: WebServerOptions,\n  startWebServerImpl: (options: WebServerOptions) => Promise<WebServerResult>,\n): Promise<ForceTakeoverAttemptResult> {\n  const initialFallback = await recoverLeaderBindFailure(currentElection.leaderInfo, consolePort, primaryToken);\n  const initialReplacement = evaluatePortOwnerReplacement(currentElection.leaderInfo, initialFallback);\n\n  if (!initialReplacement.shouldEvict || initialReplacement.ownerPid === null) {\n    return {\n      webResult: { bindResult: { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` } },\n      election: currentElection,\n      fallback: initialFallback,\n      replacement: initialReplacement,\n      forcedKill: null,\n      takeoverAttempted: false,\n      reboundLockClaimed: false,\n    };\n  }\n\n  const latestFallback = await discoverLeaderServingPort(consolePort, primaryToken);\n  const latestReplacement = evaluatePortOwnerReplacement(currentElection.leaderInfo, latestFallback);\n  if (!latestReplacement.shouldEvict || latestReplacement.ownerPid === null) {\n    logger.warn('[UnifiedConsole] Forced takeover target changed before eviction; skipping forced kill', {\n      ...buildBindFailureLogContext(\n        consolePort,\n        currentElection.leaderInfo,\n        { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` },\n        latestFallback,\n        latestReplacement,\n      ),\n      previousOwnerPid: initialReplacement.ownerPid,\n    });\n    return {\n      webResult: { bindResult: { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` } },\n      election: currentElection,\n      fallback: latestFallback,\n      replacement: latestReplacement,\n      forcedKill: null,\n      takeoverAttempted: false,\n      reboundLockClaimed: false,\n    };\n  }\n\n  logger.warn('[UnifiedConsole] Attempting forced takeover from older or incompatible active leader', {\n    ...buildBindFailureLogContext(\n      consolePort,\n      currentElection.leaderInfo,\n      { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` },\n      latestFallback,\n      latestReplacement,\n    ),\n  });\n\n  const forcedKill = await killStaleProcessDetailed(latestReplacement.ownerPid, consolePort, {\n    allowActiveHostParent: true,\n  });\n  if (!forcedKill.killed) {\n    logger.warn('[UnifiedConsole] Forced takeover skipped or failed after identifying replaceable leader', {\n      ...buildBindFailureLogContext(\n        consolePort,\n        currentElection.leaderInfo,\n        { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` },\n        latestFallback,\n        latestReplacement,\n        forcedKill,\n      ),\n    });\n    return {\n      webResult: { bindResult: { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` } },\n      election: currentElection,\n      fallback: latestFallback,\n      replacement: latestReplacement,\n      forcedKill,\n      takeoverAttempted: true,\n      reboundLockClaimed: false,\n    };\n  }\n\n  const reboundWebResult = await startWebServerImpl(serverOpts);\n  let reboundElection = currentElection;\n  let reboundLockClaimed = false;\n\n  if (!reboundWebResult.bindResult || reboundWebResult.bindResult.success) {\n    const reboundLeaderInfo = createLeaderInfo(options.sessionId, consolePort);\n    reboundLockClaimed = await claimLeadership(reboundLeaderInfo);\n    if (!reboundLockClaimed) {\n      logger.warn('[UnifiedConsole] Rebound leader bound port but could not immediately re-claim lock', {\n        ...buildBindFailureLogContext(\n          consolePort,\n          reboundLeaderInfo,\n          reboundWebResult.bindResult,\n          latestFallback,\n          latestReplacement,\n          forcedKill,\n        ),\n      });\n    }\n    reboundElection = { role: 'leader', leaderInfo: reboundLeaderInfo };\n  } else {\n    logger.warn('[UnifiedConsole] Forced takeover killed old leader but bind retry still failed', {\n      ...buildBindFailureLogContext(\n        consolePort,\n        currentElection.leaderInfo,\n        reboundWebResult.bindResult,\n        latestFallback,\n        latestReplacement,\n        forcedKill,\n      ),\n    });\n  }\n\n  return {\n    webResult: reboundWebResult,\n    election: reboundElection,\n    fallback: latestFallback,\n    replacement: latestReplacement,\n    forcedKill,\n    takeoverAttempted: true,\n    reboundLockClaimed,\n  };\n}\n\n/**\n * Start the unified web console.\n *\n * Runs leader election, then either starts the full console (leader)\n * or sets up event forwarding (follower).\n */\nexport async function startUnifiedConsole(options: UnifiedConsoleOptions): Promise<UnifiedConsoleResult> {\n  // Resolve port: options (config file) → env var → default\n  const consolePort = options.port || DEFAULT_CONSOLE_PORT;\n  logger.debug(`[UnifiedConsole] Port resolved: ${consolePort}` +\n    (options.port ? ' (from config file)' : ` (from env/default)`));\n\n  // Legacy-leader detection (#1794) — warn the user if a pre-auth\n  // DollhouseMCP console is running alongside this authenticated one.\n  // They will coexist fine because of port + lock + token file isolation,\n  // but the user should know both exist so the differing security posture\n  // between them doesn't look like a bug.\n  await warnIfLegacyConsolePresent(consolePort);\n\n  let election = await electLeader(options.sessionId, consolePort);\n\n  if (election.role === 'follower') {\n    const resolved = await resolveFollowerAuthority(options.sessionId, consolePort, election);\n    election = resolved.election;\n  }\n\n  if (election.role === 'leader') {\n    return startAsLeader(options, election, consolePort);\n  } else {\n    return startAsFollower(options, election, consolePort);\n  }\n}\n\n/**\n * Start as the console leader.\n * Binds the resolved console port (config file → env var → default),\n * mounts all routes including ingestion, starts heartbeat.\n */\nasync function startAsLeader(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n): Promise<UnifiedConsoleResult> {\n  const { startWebServer } = await import('../server.js');\n  const { pickRandomTokenName } = await import('./SessionNames.js');\n\n  // Initialize the console token store (#1780). Creates the token file on\n  // first run, reads the existing tokens on subsequent runs. The token is\n  // persistent across restarts — only rotated on explicit request (Phase 2).\n  // Feature flag DOLLHOUSE_WEB_AUTH_ENABLED controls enforcement; the file\n  // is generated regardless so consumers can attach tokens preemptively.\n  const tokenStore = new ConsoleTokenStore(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  const primaryToken = await tokenStore.ensureInitialized(pickRandomTokenName());\n  logger.info('[UnifiedConsole] Console token store initialized', {\n    tokenId: primaryToken.id,\n    tokenName: primaryToken.name,\n    file: tokenStore.getFilePath(),\n    authEnforced: env.DOLLHOUSE_WEB_AUTH_ENABLED,\n  });\n\n  // Pre-create a placeholder broadcast that we'll wire up after the server starts\n  let liveBroadcast: ((entry: UnifiedLogEntry) => void) | undefined;\n  let liveMetricsOnSnapshot: ((snapshot: MetricSnapshot) => void) | undefined;\n\n  // Create ingestion routes with a deferred broadcast (wired after server starts)\n  const ingestResult = createIngestRoutes({\n    logBroadcast: (entry) => liveBroadcast?.(entry),\n    metricsOnSnapshot: (snapshot) => liveMetricsOnSnapshot?.(snapshot),\n    storeMetricsSnapshot: (snapshot) => options.metricsSink?.onSnapshot(snapshot),\n  });\n\n  // Start the web server with ingest routes mounted before the SPA fallback.\n  // If the port is occupied by a stale process, retry with exponential backoff.\n  const serverOpts = {\n    portfolioDir: options.portfolioDir,\n    memorySink: options.memorySink,\n    metricsSink: options.metricsSink,\n    port: consolePort,\n    sessionId: options.stableSessionId,\n    runtimeSessionId: options.sessionId,\n    additionalRouters: [ingestResult.router],\n    tokenStore,\n    ...(options.mcpAqlHandler ? { mcpAqlHandler: options.mcpAqlHandler } : {}),\n  };\n  // bindAndListen now handles EADDRINUSE by finding and killing the stale\n  // process on the port, then retrying. No external retry loop needed.\n  let webResult = await startWebServer(serverOpts);\n\n  if (webResult.bindResult && !webResult.bindResult.success) {\n    const forceTakeover = await attemptForceTakeover(\n      options,\n      election,\n      consolePort,\n      primaryToken.token,\n      serverOpts,\n      startWebServer,\n    );\n    webResult = forceTakeover.webResult;\n    election = forceTakeover.election;\n\n    if (webResult.bindResult && !webResult.bindResult.success) {\n      if (forceTakeover.fallback.leaderInfo) {\n      logger.warn('[UnifiedConsole] Leader role aborted: bind failed, falling back to follower', {\n        ...buildBindFailureLogContext(\n          consolePort,\n          election.leaderInfo,\n          webResult.bindResult,\n          forceTakeover.fallback,\n          forceTakeover.replacement,\n          forceTakeover.forcedKill,\n        ),\n        takeoverAttempted: forceTakeover.takeoverAttempted,\n        reboundLockClaimed: forceTakeover.reboundLockClaimed,\n        lockCleanupAttempted: forceTakeover.fallback.source !== 'none',\n      });\n      const followerElection: ElectionResult = { role: 'follower', leaderInfo: forceTakeover.fallback.leaderInfo };\n      return startAsFollower(options, followerElection, consolePort, primaryToken.token);\n      }\n\n      logger.error('[UnifiedConsole] Leader failed to bind and no active leader could be identified', {\n        ...buildBindFailureLogContext(\n          consolePort,\n          election.leaderInfo,\n          webResult.bindResult,\n          forceTakeover.fallback,\n          forceTakeover.replacement,\n          forceTakeover.forcedKill,\n        ),\n        takeoverAttempted: forceTakeover.takeoverAttempted,\n        reboundLockClaimed: forceTakeover.reboundLockClaimed,\n      });\n      throw new Error(`Leader failed to bind port ${consolePort} and no active leader was discoverable`);\n    }\n  }\n\n  // Register the leader only after the HTTP listener is actually serving the port.\n  ingestResult.registerLeaderSession(options.sessionId, process.pid);\n\n  // Register the web console itself so the session indicator is never empty (#1805)\n  ingestResult.registerConsoleSession();\n\n  // Wire SSE broadcasts for this leader's own events\n  options.wireSSEBroadcasts(webResult, options.metricsSink);\n\n  // Now wire the live broadcast functions into the ingest routes\n  if (webResult.logBroadcast) {\n    const originalBroadcast = webResult.logBroadcast;\n    // Stamp leader's own entries with session ID\n    liveBroadcast = (entry: UnifiedLogEntry) => {\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: options.sessionId },\n      };\n      originalBroadcast(stamped);\n    };\n  }\n  liveMetricsOnSnapshot = webResult.metricsOnSnapshot;\n\n  logger.info('[UnifiedConsole] Ingestion routes mounted');\n\n  // Start heartbeat and register cleanup\n  const stopHeartbeat = startHeartbeat(election.leaderInfo);\n  registerLeaderCleanup();\n\n  logger.info('[UnifiedConsole] Leader started', {\n    sessionId: options.sessionId, port: consolePort, pid: process.pid,\n    role: 'leader', ingestRoutes: ['/api/ingest/logs', '/api/ingest/metrics', '/api/ingest/session', '/api/sessions'],\n  });\n\n  return {\n    role: 'leader',\n    election,\n    port: consolePort,\n    cleanup: async () => {\n      stopHeartbeat();\n    },\n  };\n}\n\n/**\n * Start as a follower.\n * Registers forwarding sinks with the LogManager, starts session heartbeat.\n */\nasync function startAsFollower(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n  initialAuthToken: string | null = null,\n): Promise<UnifiedConsoleResult> {\n  const leaderUrl = `http://127.0.0.1:${election.leaderInfo.port}`;\n\n  // Read the console auth token (#1780) written by the leader. May be null\n  // if the file doesn't exist yet — the sinks handle that gracefully and\n  // simply omit the Bearer header, which is fine when auth is not enforced.\n  let authToken = initialAuthToken;\n  if (authToken === null) {\n    const { getPrimaryTokenFromFile } = await import('./consoleToken.js');\n    authToken = await getPrimaryTokenFromFile(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  }\n  if (authToken) {\n    logger.debug('[UnifiedConsole] Follower loaded console auth token');\n  } else {\n    logger.debug('[UnifiedConsole] No console auth token file found; follower will POST without Bearer header');\n  }\n\n  // Per-instance promotion manager — tracks its own attempt counter so\n  // multiple followers don't interfere with each other's promotion budgets.\n  const promotionMgr = new PromotionManager(options, consolePort, startAsLeader, startAsFollower);\n\n  // Declare sessionHeartbeat before the sink so the closure can capture it.\n  // Both are initialized before the callback could possibly fire (needs 5+ failed flushes).\n  let sessionHeartbeat: SessionHeartbeat;\n\n  // Register a forwarding log sink with leader-death callback (#1850).\n  const forwardingSink = new LeaderForwardingLogSink(leaderUrl, options.sessionId, authToken, () => {\n    promotionMgr.promote(forwardingSink, sessionHeartbeat)\n      .catch(err => logger.error('[UnifiedConsole] Promotion crashed', { error: String(err) }));\n  });\n  options.registerLogSink(forwardingSink);\n\n  // Start session heartbeat to the leader\n  sessionHeartbeat = new SessionHeartbeat(leaderUrl, options.sessionId, process.pid, authToken);\n  await sessionHeartbeat.start();\n\n  const stopAuthorityMonitor = startFollowerAuthorityMonitor(\n    options,\n    consolePort,\n    election,\n    promotionMgr,\n    forwardingSink,\n    sessionHeartbeat,\n  );\n\n  logger.info('[UnifiedConsole] Follower started', {\n    sessionId: options.sessionId, pid: process.pid, role: 'follower',\n    leaderSession: election.leaderInfo.sessionId, leaderPid: election.leaderInfo.pid,\n    leaderPort: election.leaderInfo.port, leaderUrl,\n  });\n\n  return {\n    role: 'follower',\n    election,\n    cleanup: async () => {\n      stopAuthorityMonitor();\n      await sessionHeartbeat.stop();\n      await forwardingSink.close();\n    },\n  };\n}\n"]}
745
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UnifiedConsole.js","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Unified web console orchestrator.\n *\n * Ties together leader election, console startup, follower wiring,\n * and session lifecycle management. This is the main entry point\n * called by the DI container during deferred setup.\n *\n * Flow:\n * 1. Run leader election (read lock file, claim or follow)\n * 2. If leader: start web server on fixed port, mount ingest routes, start heartbeat\n * 3. If follower: register forwarding sinks with LogManager, start session heartbeat\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport type { MemoryLogSink } from '../../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../../metrics/sinks/MemoryMetricsSink.js';\nimport type { WebServerOptions, WebServerResult } from '../server.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { logger } from '../../utils/logger.js';\nimport {\n  electLeader,\n  isLeaderWebConsoleReachable,\n  forceClaimLeadership,\n  startHeartbeat,\n  registerLeaderCleanup,\n  detectLegacyLeader,\n  readLeaderLock,\n  deleteLeaderLock,\n  claimLeadership,\n  createLeaderInfo,\n  LOCK_VERSION,\n  CONSOLE_PROTOCOL_VERSION,\n  LEGACY_SERVER_VERSION,\n  evaluateLeaderPreference,\n  type ElectionResult,\n  type ConsoleLeaderInfo,\n  type LeaderPreferenceDecision,\n} from './LeaderElection.js';\nimport { createIngestRoutes } from './IngestRoutes.js';\nimport {\n  LeaderForwardingLogSink,\n  SessionHeartbeat,\n} from './LeaderForwardingSink.js';\nimport { PromotionManager } from './PromotionManager.js';\nimport { ConsoleTokenStore } from './consoleToken.js';\nimport {\n  findPidOnPort,\n  killStaleProcessDetailed,\n  type KillStaleProcessOutcome,\n} from './StaleProcessRecovery.js';\nimport { env } from '../../config/env.js';\nimport type { SessionInfo } from './IngestRoutes.js';\n\n/**\n * Default console port from the env var. Used as fallback when no port\n * is provided via config file or options. The resolution hierarchy is:\n *   1. options.port (from config file, resolved by the DI container)\n *   2. DOLLHOUSE_WEB_CONSOLE_PORT env var\n *   3. 41715 (hardcoded default in env.ts)\n */\nconst DEFAULT_CONSOLE_PORT = env.DOLLHOUSE_WEB_CONSOLE_PORT;\nconst LEGACY_CONSOLE_FALLBACK_PORT = 3939;\nconst SYNTHETIC_PORT_OWNER_SESSION_PREFIX = 'port-owner-';\nconst LEADER_DISCOVERY_TIMEOUT_MS = 2_000;\nconst FOLLOWER_AUTHORITY_MONITOR_CONFIG = {\n  intervalMs: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_MS,\n  jitterMs: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_JITTER_MS,\n  failureThreshold: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_FAILURE_THRESHOLD,\n  failureCooldownMs: env.DOLLHOUSE_CONSOLE_AUTHORITY_RECHECK_FAILURE_COOLDOWN_MS,\n} as const;\n\nfunction currentTimestamp(): string {\n  return new Date().toISOString();\n}\n\nfunction computeFollowerAuthorityRecheckInterval(sessionId: string): number {\n  const normalizedSessionId = UnicodeValidator.normalize(sessionId).normalizedContent;\n  let hash = 0;\n  for (let index = 0; index < normalizedSessionId.length; index += 1) {\n    const codePoint = normalizedSessionId.codePointAt(index) ?? 0;\n    hash = (hash * 31 + codePoint) >>> 0;\n    if (codePoint > 0xffff) {\n      index += 1;\n    }\n  }\n  return FOLLOWER_AUTHORITY_MONITOR_CONFIG.intervalMs + (hash % (FOLLOWER_AUTHORITY_MONITOR_CONFIG.jitterMs + 1));\n}\n\n/**\n * Options for starting the unified console.\n */\nexport interface UnifiedConsoleOptions {\n  /** This process's unique session ID */\n  sessionId: string;\n  /** Stable Dollhouse session identity shown to humans and used for persistence. */\n  stableSessionId: string;\n  /** Portfolio base directory (for startWebServer) */\n  portfolioDir: string;\n  /** Log memory sink (for console history) */\n  memorySink: MemoryLogSink;\n  /** Metrics memory sink */\n  metricsSink?: MemoryMetricsSink;\n  /** MCP-AQL handler for permission routes (typed as any to avoid circular imports) */\n  mcpAqlHandler?: any;\n  /** Callback to register a log sink with the LogManager */\n  registerLogSink: (sink: { write(entry: UnifiedLogEntry): void; flush(): Promise<void>; close(): Promise<void> }) => void;\n  /** Callback to wire SSE broadcasts after web server starts */\n  wireSSEBroadcasts: (webResult: { logBroadcast?: (entry: UnifiedLogEntry) => void; metricsOnSnapshot?: (snapshot: MetricSnapshot) => void }, metricsSink?: MemoryMetricsSink) => void;\n  /** Console port override from config file. Falls back to env var if not provided. */\n  port?: number;\n}\n\n/**\n * Result of starting the unified console.\n */\nexport interface UnifiedConsoleResult {\n  role: 'leader' | 'follower';\n  election: ElectionResult;\n  /** Port the console is running on (leader only) */\n  port?: number;\n  /** Cleanup function to call on shutdown */\n  cleanup: () => Promise<void>;\n}\n\n/**\n * Check for a running legacy (pre-authentication) DollhouseMCP console and\n * log a WARN-level message if one is found (#1794).\n *\n * Extracted from `startUnifiedConsole` so the wiring can be integration-\n * tested in isolation without spinning up a full web server and leader\n * election. The implementation is fire-and-forget: detection failures\n * are logged at DEBUG and never propagate, because a failure here must\n * not block leader election of the authenticated console.\n *\n * @param currentPort - The port the authenticated console intends to\n *                      bind to. Used in the warning message to help the\n *                      user tell the two consoles apart.\n * @param detect      - Optional injection point for the detection\n *                      function. Defaults to `detectLegacyLeader`. Tests\n *                      pass a stub.\n * @param log         - Optional injection point for the logger. Defaults\n *                      to the module logger. Tests pass a spy.\n * @returns The legacy leader info from `detect()`, or null if detection\n *          threw. Exposed so tests can assert the full result shape.\n */\nexport async function warnIfLegacyConsolePresent(\n  currentPort: number,\n  detect: typeof detectLegacyLeader = detectLegacyLeader,\n  log: typeof logger = logger,\n): Promise<Awaited<ReturnType<typeof detectLegacyLeader>> | null> {\n  try {\n    const legacy = await detect();\n    if (legacy.legacyRunning) {\n      log.warn(\n        `[UnifiedConsole] Legacy (pre-authentication) DollhouseMCP console detected ` +\n        `(pid=${legacy.pid}, port=${legacy.port}). Both consoles will run ` +\n        `independently on different ports with different security posture. ` +\n        `The authenticated console (this process) uses port ${currentPort}; ` +\n        `the legacy console uses port ${legacy.port ?? LEGACY_CONSOLE_FALLBACK_PORT}. ` +\n        `For consistent security, update the legacy installation to a ` +\n        `version with the authenticated console.`,\n      );\n    }\n    return legacy;\n  } catch (err) {\n    // Best-effort — never block election on a detection failure\n    log.debug('[UnifiedConsole] Legacy leader detection failed', {\n      error: err instanceof Error ? err.message : String(err),\n    });\n    return null;\n  }\n}\n\ninterface SessionApiRecord {\n  sessionId: string;\n  pid: number;\n  startedAt?: string;\n  lastHeartbeat?: string;\n  status?: string;\n  isLeader?: boolean;\n  kind?: string;\n  serverVersion?: string;\n  consoleProtocolVersion?: number;\n}\n\nexport interface PortLeaderDiscovery {\n  leaderInfo: ConsoleLeaderInfo | null;\n  ownerPid: number | null;\n  source: 'api' | 'lock' | 'synthetic' | 'none';\n}\n\nexport interface BindFailureRecoveryResult extends PortLeaderDiscovery {\n  lockCleanupAttempted: boolean;\n  lockCleanupPerformed: boolean;\n}\n\nexport interface PortOwnerReplacementDecision {\n  shouldEvict: boolean;\n  ownerPid: number | null;\n  preference: LeaderPreferenceDecision | null;\n}\n\ninterface ForceTakeoverAttemptResult {\n  webResult: WebServerResult;\n  election: ElectionResult;\n  fallback: PortLeaderDiscovery;\n  replacement: PortOwnerReplacementDecision;\n  recoveredSessions: SessionInfo[];\n  forcedKill: KillStaleProcessOutcome | null;\n  takeoverAttempted: boolean;\n  reboundLockClaimed: boolean;\n}\n\ninterface FollowerAuthorityResolution {\n  election: ElectionResult;\n  discovery: PortLeaderDiscovery | null;\n  replacement: PortOwnerReplacementDecision | null;\n  forcedClaim: boolean;\n}\n\ninterface FollowerAuthorityDependencies {\n  isLeaderWebConsoleReachableImpl?: typeof isLeaderWebConsoleReachable;\n  discoverLeaderServingPortImpl?: typeof discoverLeaderServingPort;\n  forceClaimLeadershipImpl?: typeof forceClaimLeadership;\n  deleteLeaderLockImpl?: typeof deleteLeaderLock;\n}\n\ninterface FollowerAuthorityMonitorDependencies extends FollowerAuthorityDependencies {\n  resolveFollowerAuthorityImpl?: typeof resolveFollowerAuthority;\n  setIntervalImpl?: typeof setInterval;\n  clearIntervalImpl?: typeof clearInterval;\n  nowImpl?: () => number;\n}\n\ninterface DiscoveryDependencies {\n  fetchImpl?: typeof fetch;\n  findPidOnPortImpl?: typeof findPidOnPort;\n  readLeaderLockImpl?: typeof readLeaderLock;\n}\n\nfunction buildDiscoveryHeaders(authToken: string | null): Record<string, string> {\n  return authToken ? { Authorization: `Bearer ${authToken}` } : {};\n}\n\nexport async function fetchLeaderSessionsSnapshot(\n  port: number,\n  authToken: string | null,\n  fetchImpl: typeof fetch = fetch,\n): Promise<SessionInfo[]> {\n  const controller = new AbortController();\n  const timeout = setTimeout(() => controller.abort(), LEADER_DISCOVERY_TIMEOUT_MS);\n  try {\n    const response = await fetchImpl(`http://127.0.0.1:${port}/api/sessions`, {\n      headers: buildDiscoveryHeaders(authToken),\n      signal: controller.signal,\n    });\n    if (!response.ok) {\n      return [];\n    }\n\n    const data = await response.json() as { sessions?: SessionInfo[] };\n    return Array.isArray(data.sessions) ? data.sessions : [];\n  } catch {\n    return [];\n  } finally {\n    clearTimeout(timeout);\n  }\n}\n\nfunction buildLeaderInfoFromSession(port: number, ownerPid: number, leaderSession: SessionApiRecord): ConsoleLeaderInfo {\n  return {\n    version: LOCK_VERSION,\n    pid: ownerPid,\n    port,\n    sessionId: UnicodeValidator.normalize(leaderSession.sessionId).normalizedContent,\n    startedAt: leaderSession.startedAt ?? currentTimestamp(),\n    heartbeat: leaderSession.lastHeartbeat ?? currentTimestamp(),\n    serverVersion: leaderSession.serverVersion ?? LEGACY_SERVER_VERSION,\n    consoleProtocolVersion: leaderSession.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\nfunction buildSyntheticLeaderInfo(port: number, ownerPid: number): ConsoleLeaderInfo {\n  const now = currentTimestamp();\n  return {\n    version: LOCK_VERSION,\n    pid: ownerPid,\n    port,\n    sessionId: `${SYNTHETIC_PORT_OWNER_SESSION_PREFIX}${ownerPid}`,\n    startedAt: now,\n    heartbeat: now,\n    serverVersion: LEGACY_SERVER_VERSION,\n    consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\nasync function discoverLeaderViaSessionsApi(\n  port: number,\n  ownerPid: number,\n  authToken: string | null,\n  fetchImpl: typeof fetch,\n): Promise<ConsoleLeaderInfo | null> {\n  const controller = new AbortController();\n  const timeout = setTimeout(() => controller.abort(), LEADER_DISCOVERY_TIMEOUT_MS);\n\n  try {\n    const response = await fetchImpl(`http://127.0.0.1:${port}/api/sessions`, {\n      headers: buildDiscoveryHeaders(authToken),\n      signal: controller.signal,\n    });\n    if (!response.ok) {\n      return null;\n    }\n\n    const payload = await response.json() as { sessions?: SessionApiRecord[] };\n    const sessions = Array.isArray(payload.sessions) ? payload.sessions : [];\n    const leaderSession = sessions.find((session) =>\n      session.pid === ownerPid &&\n      session.isLeader === true &&\n      session.kind === 'mcp' &&\n      session.status !== 'stopped'\n    );\n    return leaderSession ? buildLeaderInfoFromSession(port, ownerPid, leaderSession) : null;\n  } finally {\n    clearTimeout(timeout);\n  }\n}\n\nexport async function discoverLeaderServingPort(\n  port: number,\n  authToken: string | null,\n  deps: DiscoveryDependencies = {},\n): Promise<PortLeaderDiscovery> {\n  const fetchImpl = deps.fetchImpl ?? fetch;\n  const findPidOnPortImpl = deps.findPidOnPortImpl ?? findPidOnPort;\n  const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;\n  const ownerPid = await findPidOnPortImpl(port);\n\n  if (ownerPid !== null) {\n    try {\n      const leaderInfo = await discoverLeaderViaSessionsApi(port, ownerPid, authToken, fetchImpl);\n      if (leaderInfo) {\n        return { ownerPid, source: 'api', leaderInfo };\n      }\n    } catch (err) {\n      logger.debug('[UnifiedConsole] Failed to query active leader sessions', {\n        port,\n        ownerPid,\n        error: err instanceof Error ? err.message : String(err),\n      });\n    }\n  }\n\n  const lock = await readLeaderLockImpl();\n  if (lock?.port === port && (ownerPid === null || lock.pid === ownerPid)) {\n    return {\n      ownerPid: ownerPid ?? lock.pid,\n      source: 'lock',\n      leaderInfo: {\n        ...lock,\n        sessionId: UnicodeValidator.normalize(lock.sessionId).normalizedContent,\n      },\n    };\n  }\n\n  if (ownerPid !== null) {\n    return {\n      ownerPid,\n      source: 'synthetic',\n      leaderInfo: buildSyntheticLeaderInfo(port, ownerPid),\n    };\n  }\n\n  return { leaderInfo: null, ownerPid: null, source: 'none' };\n}\n\ninterface BindFailureRecoveryDependencies extends DiscoveryDependencies {\n  deleteLeaderLockImpl?: typeof deleteLeaderLock;\n}\n\nexport async function recoverLeaderBindFailure(\n  provisionalLeader: ConsoleLeaderInfo,\n  port: number,\n  authToken: string | null,\n  deps: BindFailureRecoveryDependencies = {},\n): Promise<BindFailureRecoveryResult> {\n  const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;\n  const deleteLeaderLockImpl = deps.deleteLeaderLockImpl ?? deleteLeaderLock;\n  logger.info('[UnifiedConsole] Leader bind recovery initiated', {\n    provisionalSessionId: provisionalLeader.sessionId,\n    provisionalPid: provisionalLeader.pid,\n    port,\n  });\n\n  let fallback = await discoverLeaderServingPort(port, authToken, deps);\n  let lockCleanupAttempted = false;\n  let lockCleanupPerformed = false;\n  const currentLock = await readLeaderLockImpl();\n  const provisionalLockMatches = (\n    currentLock?.pid === provisionalLeader.pid &&\n    currentLock.port === provisionalLeader.port &&\n    currentLock.sessionId === provisionalLeader.sessionId\n  );\n  const fallbackPointsToProvisionalLeader = (\n    fallback.leaderInfo?.pid === provisionalLeader.pid &&\n    fallback.leaderInfo.port === provisionalLeader.port &&\n    fallback.leaderInfo.sessionId === provisionalLeader.sessionId\n  );\n\n  if (provisionalLockMatches) {\n    lockCleanupAttempted = true;\n    await deleteLeaderLockImpl();\n    lockCleanupPerformed = true;\n    logger.info('[UnifiedConsole] Removed provisional leader lock after bind failure', {\n      provisionalSessionId: provisionalLeader.sessionId,\n      provisionalPid: provisionalLeader.pid,\n      port,\n    });\n    if (fallbackPointsToProvisionalLeader) {\n      fallback = await discoverLeaderServingPort(port, authToken, deps);\n    }\n  }\n\n  logger.info('[UnifiedConsole] Leader bind recovery completed', {\n    provisionalSessionId: provisionalLeader.sessionId,\n    provisionalPid: provisionalLeader.pid,\n    port,\n    discoverySource: fallback.source,\n    ownerPid: fallback.ownerPid,\n    lockCleanupAttempted,\n    lockCleanupPerformed,\n  });\n\n  return {\n    ...fallback,\n    lockCleanupAttempted,\n    lockCleanupPerformed,\n  };\n}\n\nexport function evaluatePortOwnerReplacement(\n  candidateLeader: ConsoleLeaderInfo,\n  fallback: PortLeaderDiscovery,\n): PortOwnerReplacementDecision {\n  if (!fallback.leaderInfo || fallback.ownerPid === null || fallback.ownerPid === candidateLeader.pid) {\n    return {\n      shouldEvict: false,\n      ownerPid: fallback.ownerPid,\n      preference: null,\n    };\n  }\n\n  const preference = evaluateLeaderPreference(candidateLeader, fallback.leaderInfo);\n  return {\n    shouldEvict: preference.shouldReplace,\n    ownerPid: fallback.ownerPid,\n    preference,\n  };\n}\n\nfunction buildBindFailureLogContext(\n  consolePort: number,\n  provisionalLeader: ConsoleLeaderInfo,\n  bindResult: WebServerResult['bindResult'],\n  fallback: PortLeaderDiscovery,\n  replacement?: PortOwnerReplacementDecision,\n  forcedKill?: KillStaleProcessOutcome | null,\n) {\n  return {\n    port: consolePort,\n    bindError: bindResult?.error,\n    bindDetail: bindResult?.detail,\n    provisionalLeaderPid: provisionalLeader.pid,\n    provisionalLeaderSessionId: provisionalLeader.sessionId,\n    provisionalLeaderVersion: provisionalLeader.serverVersion ?? LEGACY_SERVER_VERSION,\n    provisionalLeaderProtocolVersion: provisionalLeader.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    fallbackOwnerPid: fallback.ownerPid,\n    fallbackSource: fallback.source,\n    fallbackLeaderPid: fallback.leaderInfo?.pid,\n    fallbackLeaderSessionId: fallback.leaderInfo?.sessionId,\n    fallbackLeaderVersion: fallback.leaderInfo?.serverVersion ?? LEGACY_SERVER_VERSION,\n    fallbackLeaderProtocolVersion: fallback.leaderInfo?.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    replacementShouldEvict: replacement?.shouldEvict ?? false,\n    replacementReason: replacement?.preference?.reason,\n    forcedKillReason: forcedKill?.reason,\n    forcedKillPid: forcedKill?.pid,\n    forcedKillDetail: forcedKill?.detail,\n  };\n}\n\nfunction buildAuthorityResolutionLogContext(\n  consolePort: number,\n  electedLeader: ConsoleLeaderInfo,\n  discovery: PortLeaderDiscovery | null,\n  replacement: PortOwnerReplacementDecision | null,\n) {\n  return {\n    port: consolePort,\n    electedLeaderPid: electedLeader.pid,\n    electedLeaderSessionId: electedLeader.sessionId,\n    electedLeaderVersion: electedLeader.serverVersion ?? LEGACY_SERVER_VERSION,\n    electedLeaderProtocolVersion: electedLeader.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    servingOwnerPid: discovery?.ownerPid ?? null,\n    servingSource: discovery?.source ?? 'none',\n    servingLeaderPid: discovery?.leaderInfo?.pid ?? null,\n    servingLeaderSessionId: discovery?.leaderInfo?.sessionId ?? null,\n    servingLeaderVersion: discovery?.leaderInfo?.serverVersion ?? LEGACY_SERVER_VERSION,\n    servingLeaderProtocolVersion: discovery?.leaderInfo?.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n    replacementShouldEvict: replacement?.shouldEvict ?? false,\n    replacementReason: replacement?.preference?.reason ?? null,\n  };\n}\n\nexport async function resolveFollowerAuthority(\n  sessionId: string,\n  consolePort: number,\n  election: ElectionResult,\n  deps: FollowerAuthorityDependencies = {},\n): Promise<FollowerAuthorityResolution> {\n  const isLeaderWebConsoleReachableImpl = deps.isLeaderWebConsoleReachableImpl ?? isLeaderWebConsoleReachable;\n  const discoverLeaderServingPortImpl = deps.discoverLeaderServingPortImpl ?? discoverLeaderServingPort;\n  const forceClaimLeadershipImpl = deps.forceClaimLeadershipImpl ?? forceClaimLeadership;\n  const deleteLeaderLockImpl = deps.deleteLeaderLockImpl ?? deleteLeaderLock;\n\n  const reachable = await isLeaderWebConsoleReachableImpl(election.leaderInfo);\n  if (!reachable) {\n    logger.warn('[UnifiedConsole] Elected leader is not serving the console port; forcing takeover', {\n      port: consolePort,\n      electedLeaderPid: election.leaderInfo.pid,\n      electedLeaderSessionId: election.leaderInfo.sessionId,\n    });\n    return {\n      election: await forceClaimLeadershipImpl(sessionId, consolePort),\n      discovery: null,\n      replacement: null,\n      forcedClaim: true,\n    };\n  }\n\n  const candidateLeader = createLeaderInfo(sessionId, consolePort);\n  const discovery = await discoverLeaderServingPortImpl(consolePort, null);\n  if (!discovery.leaderInfo || discovery.ownerPid === null) {\n    return {\n      election,\n      discovery,\n      replacement: null,\n      forcedClaim: false,\n    };\n  }\n\n  const replacement = evaluatePortOwnerReplacement(candidateLeader, discovery);\n  if (replacement.shouldEvict) {\n    await deleteLeaderLockImpl();\n    logger.warn(\n      discovery.ownerPid === election.leaderInfo.pid\n        ? '[UnifiedConsole] Older console leader detected on the console port; newer session will take over'\n        : '[UnifiedConsole] Split-brain console authority detected; newer session will replace the actual port owner',\n      buildAuthorityResolutionLogContext(\n        consolePort,\n        election.leaderInfo,\n        discovery,\n        replacement,\n      ),\n    );\n    return {\n      election: { role: 'leader', leaderInfo: candidateLeader },\n      discovery,\n      replacement,\n      forcedClaim: false,\n    };\n  }\n\n  if (discovery.ownerPid !== election.leaderInfo.pid) {\n    logger.warn('[UnifiedConsole] Split-brain console authority detected; following the actual port owner', buildAuthorityResolutionLogContext(\n      consolePort,\n      election.leaderInfo,\n      discovery,\n      replacement,\n    ));\n    return {\n      election: { role: 'follower', leaderInfo: discovery.leaderInfo },\n      discovery,\n      replacement,\n      forcedClaim: false,\n    };\n  }\n\n  return {\n    election,\n    discovery,\n    replacement,\n    forcedClaim: false,\n  };\n}\n\nexport function startFollowerAuthorityMonitor(\n  options: UnifiedConsoleOptions,\n  consolePort: number,\n  election: ElectionResult,\n  promotionMgr: PromotionManager,\n  forwardingSink: LeaderForwardingLogSink,\n  sessionHeartbeat: SessionHeartbeat,\n  deps: FollowerAuthorityMonitorDependencies = {},\n): () => void {\n  const resolveFollowerAuthorityImpl = deps.resolveFollowerAuthorityImpl ?? resolveFollowerAuthority;\n  const setIntervalImpl = deps.setIntervalImpl ?? setInterval;\n  const clearIntervalImpl = deps.clearIntervalImpl ?? clearInterval;\n  const nowImpl = deps.nowImpl ?? Date.now;\n  let currentElection = election;\n  let checkInProgress = false;\n  let promotionQueued = false;\n  let consecutiveFailures = 0;\n  let circuitOpenUntilMs = 0;\n  let authorityTimer: ReturnType<typeof setInterval> | null = null;\n  const recheckIntervalMs = computeFollowerAuthorityRecheckInterval(options.sessionId);\n\n  const queueAuthorityPromotion = () => {\n    queueMicrotask(() => {\n      promotionMgr.promote(forwardingSink, sessionHeartbeat)\n        .catch((err) => logger.error('[UnifiedConsole] Authority-based promotion crashed', {\n          error: String(err),\n        }));\n    });\n  };\n\n  const handleResolvedAuthority = (resolved: FollowerAuthorityResolution) => {\n    currentElection = resolved.election;\n    consecutiveFailures = 0;\n    circuitOpenUntilMs = 0;\n\n    if (resolved.election.role !== 'leader') {\n      return;\n    }\n\n    promotionQueued = true;\n    if (authorityTimer) {\n      clearIntervalImpl(authorityTimer);\n      authorityTimer = null;\n    }\n\n    logger.warn('[UnifiedConsole] Follower authority re-evaluation queued a leader takeover', {\n      sessionId: options.sessionId,\n      stableSessionId: options.stableSessionId,\n      port: consolePort,\n      recheckIntervalMs,\n      electedLeaderPid: election.leaderInfo.pid,\n      electedLeaderSessionId: election.leaderInfo.sessionId,\n      resolvedLeaderPid: resolved.election.leaderInfo.pid,\n      resolvedLeaderSessionId: resolved.election.leaderInfo.sessionId,\n      replacementReason: resolved.replacement?.preference?.reason ?? null,\n      forcedClaim: resolved.forcedClaim,\n    });\n\n    queueAuthorityPromotion();\n  };\n\n  const handleAuthorityFailure = (err: unknown) => {\n    consecutiveFailures += 1;\n    if (consecutiveFailures >= FOLLOWER_AUTHORITY_MONITOR_CONFIG.failureThreshold) {\n      circuitOpenUntilMs = nowImpl() + FOLLOWER_AUTHORITY_MONITOR_CONFIG.failureCooldownMs;\n      logger.warn('[UnifiedConsole] Follower authority re-evaluation circuit opened after repeated failures', {\n        sessionId: options.sessionId,\n        port: consolePort,\n        recheckIntervalMs,\n        consecutiveFailures,\n        circuitOpenUntilMs: new Date(circuitOpenUntilMs).toISOString(),\n      });\n      consecutiveFailures = 0;\n    }\n\n    logger.debug('[UnifiedConsole] Follower authority re-evaluation failed', {\n      error: err instanceof Error ? err.message : String(err),\n      sessionId: options.sessionId,\n      port: consolePort,\n      recheckIntervalMs,\n      circuitOpenUntilMs: circuitOpenUntilMs ? new Date(circuitOpenUntilMs).toISOString() : null,\n    });\n  };\n\n  authorityTimer = setIntervalImpl(() => {\n    if (checkInProgress || promotionQueued) {\n      return;\n    }\n\n    if (circuitOpenUntilMs > nowImpl()) {\n      return;\n    }\n\n    if (circuitOpenUntilMs !== 0) {\n      logger.info('[UnifiedConsole] Follower authority re-evaluation circuit closed; resuming checks', {\n        sessionId: options.sessionId,\n        port: consolePort,\n        recheckIntervalMs,\n      });\n      circuitOpenUntilMs = 0;\n    }\n\n    checkInProgress = true;\n    resolveFollowerAuthorityImpl(options.sessionId, consolePort, currentElection)\n      .then(handleResolvedAuthority)\n      .catch(handleAuthorityFailure)\n      .finally(() => {\n        checkInProgress = false;\n      });\n  }, recheckIntervalMs);\n  authorityTimer.unref();\n\n  return () => {\n    if (authorityTimer) {\n      clearIntervalImpl(authorityTimer);\n      authorityTimer = null;\n    }\n  };\n}\n\nasync function attemptForceTakeover(\n  options: UnifiedConsoleOptions,\n  currentElection: ElectionResult,\n  consolePort: number,\n  primaryToken: string,\n  serverOpts: WebServerOptions,\n  startWebServerImpl: (options: WebServerOptions) => Promise<WebServerResult>,\n): Promise<ForceTakeoverAttemptResult> {\n  const initialFallback = await recoverLeaderBindFailure(currentElection.leaderInfo, consolePort, primaryToken);\n  const initialReplacement = evaluatePortOwnerReplacement(currentElection.leaderInfo, initialFallback);\n\n  if (!initialReplacement.shouldEvict || initialReplacement.ownerPid === null) {\n    return {\n      webResult: { bindResult: { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` } },\n      election: currentElection,\n      fallback: initialFallback,\n      replacement: initialReplacement,\n      recoveredSessions: [],\n      forcedKill: null,\n      takeoverAttempted: false,\n      reboundLockClaimed: false,\n    };\n  }\n\n  const latestFallback = await discoverLeaderServingPort(consolePort, primaryToken);\n  const recoveredSessions = await fetchLeaderSessionsSnapshot(consolePort, primaryToken);\n  const latestReplacement = evaluatePortOwnerReplacement(currentElection.leaderInfo, latestFallback);\n  if (!latestReplacement.shouldEvict || latestReplacement.ownerPid === null) {\n    logger.warn('[UnifiedConsole] Forced takeover target changed before eviction; skipping forced kill', {\n      ...buildBindFailureLogContext(\n        consolePort,\n        currentElection.leaderInfo,\n        { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` },\n        latestFallback,\n        latestReplacement,\n      ),\n      previousOwnerPid: initialReplacement.ownerPid,\n    });\n    return {\n      webResult: { bindResult: { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` } },\n      election: currentElection,\n      fallback: latestFallback,\n      replacement: latestReplacement,\n      recoveredSessions,\n      forcedKill: null,\n      takeoverAttempted: false,\n      reboundLockClaimed: false,\n    };\n  }\n\n  logger.warn('[UnifiedConsole] Attempting forced takeover from older or incompatible active leader', {\n    ...buildBindFailureLogContext(\n      consolePort,\n      currentElection.leaderInfo,\n      { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` },\n      latestFallback,\n      latestReplacement,\n    ),\n  });\n\n  const forcedKill = await killStaleProcessDetailed(latestReplacement.ownerPid, consolePort, {\n    allowActiveHostParent: true,\n  });\n  if (!forcedKill.killed) {\n    logger.warn('[UnifiedConsole] Forced takeover skipped or failed after identifying replaceable leader', {\n      ...buildBindFailureLogContext(\n        consolePort,\n        currentElection.leaderInfo,\n        { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` },\n        latestFallback,\n        latestReplacement,\n        forcedKill,\n      ),\n    });\n    return {\n      webResult: { bindResult: { success: false, error: 'EADDRINUSE', detail: `Port ${consolePort} already in use` } },\n      election: currentElection,\n      fallback: latestFallback,\n      replacement: latestReplacement,\n      recoveredSessions,\n      forcedKill,\n      takeoverAttempted: true,\n      reboundLockClaimed: false,\n    };\n  }\n\n  const reboundWebResult = await startWebServerImpl(serverOpts);\n  let reboundElection = currentElection;\n  let reboundLockClaimed = false;\n\n  if (!reboundWebResult.bindResult || reboundWebResult.bindResult.success) {\n    const reboundLeaderInfo = createLeaderInfo(options.sessionId, consolePort);\n    reboundLockClaimed = await claimLeadership(reboundLeaderInfo);\n    if (!reboundLockClaimed) {\n      logger.warn('[UnifiedConsole] Rebound leader bound port but could not immediately re-claim lock', {\n        ...buildBindFailureLogContext(\n          consolePort,\n          reboundLeaderInfo,\n          reboundWebResult.bindResult,\n          latestFallback,\n          latestReplacement,\n          forcedKill,\n        ),\n      });\n    }\n    reboundElection = { role: 'leader', leaderInfo: reboundLeaderInfo };\n  } else {\n    logger.warn('[UnifiedConsole] Forced takeover killed old leader but bind retry still failed', {\n      ...buildBindFailureLogContext(\n        consolePort,\n        currentElection.leaderInfo,\n        reboundWebResult.bindResult,\n        latestFallback,\n        latestReplacement,\n        forcedKill,\n      ),\n    });\n  }\n\n  return {\n    webResult: reboundWebResult,\n    election: reboundElection,\n    fallback: latestFallback,\n    replacement: latestReplacement,\n    recoveredSessions,\n    forcedKill,\n    takeoverAttempted: true,\n    reboundLockClaimed,\n  };\n}\n\n/**\n * Start the unified web console.\n *\n * Runs leader election, then either starts the full console (leader)\n * or sets up event forwarding (follower).\n */\nexport async function startUnifiedConsole(options: UnifiedConsoleOptions): Promise<UnifiedConsoleResult> {\n  // Resolve port: options (config file) → env var → default\n  const consolePort = options.port || DEFAULT_CONSOLE_PORT;\n  logger.debug(`[UnifiedConsole] Port resolved: ${consolePort}` +\n    (options.port ? ' (from config file)' : ` (from env/default)`));\n\n  // Legacy-leader detection (#1794) — warn the user if a pre-auth\n  // DollhouseMCP console is running alongside this authenticated one.\n  // They will coexist fine because of port + lock + token file isolation,\n  // but the user should know both exist so the differing security posture\n  // between them doesn't look like a bug.\n  await warnIfLegacyConsolePresent(consolePort);\n\n  let election = await electLeader(options.sessionId, consolePort);\n\n  if (election.role === 'follower') {\n    const resolved = await resolveFollowerAuthority(options.sessionId, consolePort, election);\n    election = resolved.election;\n  }\n\n  if (election.role === 'leader') {\n    return startAsLeader(options, election, consolePort);\n  } else {\n    return startAsFollower(options, election, consolePort);\n  }\n}\n\n/**\n * Start as the console leader.\n * Binds the resolved console port (config file → env var → default),\n * mounts all routes including ingestion, starts heartbeat.\n */\nasync function startAsLeader(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n): Promise<UnifiedConsoleResult> {\n  const { startWebServer } = await import('../server.js');\n  const { pickRandomTokenName } = await import('./SessionNames.js');\n\n  // Initialize the console token store (#1780). Creates the token file on\n  // first run, reads the existing tokens on subsequent runs. The token is\n  // persistent across restarts — only rotated on explicit request (Phase 2).\n  // Feature flag DOLLHOUSE_WEB_AUTH_ENABLED controls enforcement; the file\n  // is generated regardless so consumers can attach tokens preemptively.\n  const tokenStore = new ConsoleTokenStore(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  const primaryToken = await tokenStore.ensureInitialized(pickRandomTokenName());\n  logger.info('[UnifiedConsole] Console token store initialized', {\n    tokenId: primaryToken.id,\n    tokenName: primaryToken.name,\n    file: tokenStore.getFilePath(),\n    authEnforced: env.DOLLHOUSE_WEB_AUTH_ENABLED,\n  });\n\n  // Pre-create a placeholder broadcast that we'll wire up after the server starts\n  let liveBroadcast: ((entry: UnifiedLogEntry) => void) | undefined;\n  let liveMetricsOnSnapshot: ((snapshot: MetricSnapshot) => void) | undefined;\n\n  // Create ingestion routes with a deferred broadcast (wired after server starts)\n  const ingestResult = createIngestRoutes({\n    logBroadcast: (entry) => liveBroadcast?.(entry),\n    metricsOnSnapshot: (snapshot) => liveMetricsOnSnapshot?.(snapshot),\n    storeMetricsSnapshot: (snapshot) => options.metricsSink?.onSnapshot(snapshot),\n  });\n\n  // Start the web server with ingest routes mounted before the SPA fallback.\n  // If the port is occupied by a stale process, retry with exponential backoff.\n  const serverOpts = {\n    portfolioDir: options.portfolioDir,\n    memorySink: options.memorySink,\n    metricsSink: options.metricsSink,\n    port: consolePort,\n    sessionId: options.stableSessionId,\n    runtimeSessionId: options.sessionId,\n    additionalRouters: [ingestResult.router],\n    tokenStore,\n    ...(options.mcpAqlHandler ? { mcpAqlHandler: options.mcpAqlHandler } : {}),\n  };\n  // bindAndListen now handles EADDRINUSE by finding and killing the stale\n  // process on the port, then retrying. No external retry loop needed.\n  let webResult = await startWebServer(serverOpts);\n  let recoveredFollowerSessions: SessionInfo[] = [];\n\n  if (webResult.bindResult && !webResult.bindResult.success) {\n    const forceTakeover = await attemptForceTakeover(\n      options,\n      election,\n      consolePort,\n      primaryToken.token,\n      serverOpts,\n      startWebServer,\n    );\n    webResult = forceTakeover.webResult;\n    election = forceTakeover.election;\n    recoveredFollowerSessions = forceTakeover.recoveredSessions;\n\n    if (webResult.bindResult && !webResult.bindResult.success) {\n      if (forceTakeover.fallback.leaderInfo) {\n      logger.warn('[UnifiedConsole] Leader role aborted: bind failed, falling back to follower', {\n        ...buildBindFailureLogContext(\n          consolePort,\n          election.leaderInfo,\n          webResult.bindResult,\n          forceTakeover.fallback,\n          forceTakeover.replacement,\n          forceTakeover.forcedKill,\n        ),\n        takeoverAttempted: forceTakeover.takeoverAttempted,\n        reboundLockClaimed: forceTakeover.reboundLockClaimed,\n        lockCleanupAttempted: forceTakeover.fallback.source !== 'none',\n      });\n      const followerElection: ElectionResult = { role: 'follower', leaderInfo: forceTakeover.fallback.leaderInfo };\n      return startAsFollower(options, followerElection, consolePort, primaryToken.token);\n      }\n\n      logger.error('[UnifiedConsole] Leader failed to bind and no active leader could be identified', {\n        ...buildBindFailureLogContext(\n          consolePort,\n          election.leaderInfo,\n          webResult.bindResult,\n          forceTakeover.fallback,\n          forceTakeover.replacement,\n          forceTakeover.forcedKill,\n        ),\n        takeoverAttempted: forceTakeover.takeoverAttempted,\n        reboundLockClaimed: forceTakeover.reboundLockClaimed,\n      });\n      throw new Error(`Leader failed to bind port ${consolePort} and no active leader was discoverable`);\n    }\n  }\n\n  // Register the leader only after the HTTP listener is actually serving the port.\n  ingestResult.registerLeaderSession(options.sessionId, process.pid);\n\n  // Register the web console itself so the session indicator is never empty (#1805)\n  ingestResult.registerConsoleSession();\n\n  if (recoveredFollowerSessions.length > 0) {\n    ingestResult.importSessions(recoveredFollowerSessions);\n    logger.info('[UnifiedConsole] Recovered follower session snapshot from displaced leader', {\n      sessionId: options.sessionId,\n      recoveredSessions: recoveredFollowerSessions.length,\n    });\n  }\n\n  // Wire SSE broadcasts for this leader's own events\n  options.wireSSEBroadcasts(webResult, options.metricsSink);\n\n  // Now wire the live broadcast functions into the ingest routes\n  if (webResult.logBroadcast) {\n    const originalBroadcast = webResult.logBroadcast;\n    // Stamp leader's own entries with session ID\n    liveBroadcast = (entry: UnifiedLogEntry) => {\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: options.sessionId },\n      };\n      originalBroadcast(stamped);\n    };\n  }\n  liveMetricsOnSnapshot = webResult.metricsOnSnapshot;\n\n  logger.info('[UnifiedConsole] Ingestion routes mounted');\n\n  // Start heartbeat and register cleanup\n  const stopHeartbeat = startHeartbeat(election.leaderInfo);\n  registerLeaderCleanup();\n\n  logger.info('[UnifiedConsole] Leader started', {\n    sessionId: options.sessionId, port: consolePort, pid: process.pid,\n    role: 'leader', ingestRoutes: ['/api/ingest/logs', '/api/ingest/metrics', '/api/ingest/session', '/api/sessions'],\n  });\n\n  return {\n    role: 'leader',\n    election,\n    port: consolePort,\n    cleanup: async () => {\n      stopHeartbeat();\n    },\n  };\n}\n\n/**\n * Start as a follower.\n * Registers forwarding sinks with the LogManager, starts session heartbeat.\n */\nasync function startAsFollower(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n  initialAuthToken: string | null = null,\n): Promise<UnifiedConsoleResult> {\n  const leaderUrl = `http://127.0.0.1:${election.leaderInfo.port}`;\n\n  // Read the console auth token (#1780) written by the leader. May be null\n  // if the file doesn't exist yet — the sinks handle that gracefully and\n  // simply omit the Bearer header, which is fine when auth is not enforced.\n  let authToken = initialAuthToken;\n  if (authToken === null) {\n    const { getPrimaryTokenFromFile } = await import('./consoleToken.js');\n    authToken = await getPrimaryTokenFromFile(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  }\n  if (authToken) {\n    logger.debug('[UnifiedConsole] Follower loaded console auth token');\n  } else {\n    logger.debug('[UnifiedConsole] No console auth token file found; follower will POST without Bearer header');\n  }\n\n  // Per-instance promotion manager — tracks its own attempt counter so\n  // multiple followers don't interfere with each other's promotion budgets.\n  const promotionMgr = new PromotionManager(options, consolePort, startAsLeader, startAsFollower);\n\n  // Declare sessionHeartbeat before the sink so the closure can capture it.\n  // Both are initialized before the callback could possibly fire (needs 5+ failed flushes).\n  let sessionHeartbeat: SessionHeartbeat;\n\n  // Register a forwarding log sink with leader-death callback (#1850).\n  const forwardingSink = new LeaderForwardingLogSink(leaderUrl, options.sessionId, authToken, () => {\n    promotionMgr.promote(forwardingSink, sessionHeartbeat)\n      .catch(err => logger.error('[UnifiedConsole] Promotion crashed', { error: String(err) }));\n  });\n  options.registerLogSink(forwardingSink);\n\n  // Start session heartbeat to the leader\n  sessionHeartbeat = new SessionHeartbeat(leaderUrl, options.sessionId, process.pid, authToken);\n  await sessionHeartbeat.start();\n\n  const stopAuthorityMonitor = startFollowerAuthorityMonitor(\n    options,\n    consolePort,\n    election,\n    promotionMgr,\n    forwardingSink,\n    sessionHeartbeat,\n  );\n\n  logger.info('[UnifiedConsole] Follower started', {\n    sessionId: options.sessionId, pid: process.pid, role: 'follower',\n    leaderSession: election.leaderInfo.sessionId, leaderPid: election.leaderInfo.pid,\n    leaderPort: election.leaderInfo.port, leaderUrl,\n  });\n\n  return {\n    role: 'follower',\n    election,\n    cleanup: async () => {\n      stopAuthorityMonitor();\n      await sessionHeartbeat.stop();\n      await forwardingSink.close();\n    },\n  };\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dollhousemcp/mcp-server",
3
- "version": "2.0.27-rc.4",
3
+ "version": "2.0.27-rc.5",
4
4
  "description": "DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
package/server.json CHANGED
@@ -3,7 +3,7 @@
3
3
  "name": "io.github.DollhouseMCP/mcp-server",
4
4
  "title": "DollhouseMCP",
5
5
  "description": "OSS to create Personas, Skills, Templates, Agents, and Memories to customize your AI experience.",
6
- "version": "2.0.27-rc.4",
6
+ "version": "2.0.27-rc.5",
7
7
  "homepage": "https://dollhousemcp.com",
8
8
  "repository": {
9
9
  "type": "git",
@@ -29,7 +29,7 @@
29
29
  {
30
30
  "registryType": "npm",
31
31
  "identifier": "@dollhousemcp/mcp-server",
32
- "version": "2.0.27-rc.4",
32
+ "version": "2.0.27-rc.5",
33
33
  "transport": {
34
34
  "type": "stdio"
35
35
  }