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

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.6] - 2026-04-19
3
+ ## [2.0.27-rc.7] - 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.6";
6
- export declare const BUILD_TIMESTAMP = "2026-04-19T19:47:49.427Z";
5
+ export declare const PACKAGE_VERSION = "2.0.27-rc.7";
6
+ export declare const BUILD_TIMESTAMP = "2026-04-19T22:30:56.353Z";
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.6';
6
- export const BUILD_TIMESTAMP = '2026-04-19T19:47:49.427Z';
5
+ export const PACKAGE_VERSION = '2.0.27-rc.7';
6
+ export const BUILD_TIMESTAMP = '2026-04-19T22:30:56.353Z';
7
7
  export const BUILD_TYPE = 'npm';
8
8
  export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
9
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsYUFBYSxDQUFDO0FBQzdDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMjctcmMuNic7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMTlUMTk6NDc6NDkuNDI3Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsYUFBYSxDQUFDO0FBQzdDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMjctcmMuNyc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMTlUMjI6MzA6NTYuMzUzWic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
@@ -117,7 +117,7 @@ export declare function detectLegacyLeader(lockPath?: string): Promise<LegacyLea
117
117
  * Read and parse the leader lock file.
118
118
  * Returns null if the file doesn't exist, is unreadable, or has invalid content.
119
119
  */
120
- export declare function readLeaderLock(): Promise<ConsoleLeaderInfo | null>;
120
+ export declare function readLeaderLock(lockPath?: string): Promise<ConsoleLeaderInfo | null>;
121
121
  /**
122
122
  * Check if a leader lock is stale (dead process or expired heartbeat).
123
123
  */
@@ -125,13 +125,14 @@ export declare function isLockStale(info: ConsoleLeaderInfo): boolean;
125
125
  /**
126
126
  * Attempt to atomically claim leadership.
127
127
  *
128
- * Writes to a temp file then renames to the lock path. On POSIX systems
129
- * rename is atomic, so only one writer wins. After renaming, re-reads the
130
- * lock to verify our PID won.
128
+ * Creates the lock file with exclusive-write semantics so only one process
129
+ * can win the initial claim. This avoids startup races where multiple
130
+ * contenders overwrite the lock in quick succession and each briefly believe
131
+ * they are leader.
131
132
  *
132
133
  * @returns true if this process successfully claimed leadership
133
134
  */
134
- export declare function claimLeadership(info: ConsoleLeaderInfo): Promise<boolean>;
135
+ export declare function claimLeadership(info: ConsoleLeaderInfo, lockPath?: string): Promise<boolean>;
135
136
  /**
136
137
  * Delete the leader lock file (for cleanup or takeover).
137
138
  */
@@ -1 +1 @@
1
- {"version":3,"file":"LeaderElection.d.ts","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;GAkBG;AA+CH,uCAAuC;AACvC,eAAO,MAAM,YAAY,IAAI,CAAC;AAE9B;;;GAGG;AACH,eAAO,MAAM,wBAAwB,IAAI,CAAC;AAE1C,kFAAkF;AAClF,eAAO,MAAM,+BAA+B,IAAI,CAAC;AAEjD,kFAAkF;AAClF,eAAO,MAAM,qBAAqB,UAAU,CAAC;AAE7C;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,MAAM,CAAC;IAChB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,sBAAsB,CAAC,EAAE,MAAM,CAAC;CACjC;AAED;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,QAAQ,GAAG,UAAU,CAAC;IAC5B,sEAAsE;IACtE,UAAU,EAAE,iBAAiB,CAAC;CAC/B;AAED,MAAM,WAAW,wBAAwB;IACvC,aAAa,EAAE,OAAO,CAAC;IACvB,MAAM,EAAE,0BAA0B,GAAG,cAAc,GAAG,eAAe,GAAG,uBAAuB,CAAC;IAChG,gBAAgB,EAAE,MAAM,CAAC;IACzB,eAAe,EAAE,MAAM,CAAC;IACxB,wBAAwB,EAAE,MAAM,CAAC;IACjC,uBAAuB,EAAE,MAAM,CAAC;CACjC;AAED;;;GAGG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAQnD;AAED;;;GAGG;AACH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,CAKtE;AAED;;;GAGG;AACH,wBAAgB,+BAA+B,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,CAM/E;AAED;;;GAGG;AACH,wBAAgB,gBAAgB,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,iBAAiB,CAYnF;AAED;;;GAGG;AACH,wBAAgB,wBAAwB,CACtC,SAAS,EAAE,iBAAiB,EAC5B,QAAQ,EAAE,iBAAiB,GAC1B,wBAAwB,CAkD1B;AAED;;;;;GAKG;AACH,MAAM,WAAW,gBAAgB;IAC/B,aAAa,EAAE,OAAO,CAAC;IACvB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED;;;;;;;;;;;;;;;;;GAiBG;AACH,wBAAsB,kBAAkB,CAAC,QAAQ,GAAE,MAAyB,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAiBvG;AAED;;;GAGG;AACH,wBAAsB,cAAc,IAAI,OAAO,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAiBxE;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAM5D;AAED;;;;;;;;GAQG;AACH,wBAAsB,eAAe,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAAC,OAAO,CAAC,CAe/E;AAED;;GAEG;AACH,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,IAAI,CAAC,CAEtD;AAED;;;;;;;;;GASG;AACH,wBAAsB,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CA0E1F;AAED;;;GAGG;AACH,wBAAsB,2BAA2B,CAAC,UAAU,EAAE,iBAAiB,GAAG,OAAO,CAAC,OAAO,CAAC,CAYjG;AAED;;;GAGG;AACH,wBAAsB,oBAAoB,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CAcnG;AAED;;;;;GAKG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,IAAI,CAgBlE;AAED;;;GAGG;AACH,wBAAgB,qBAAqB,IAAI,IAAI,CAM5C"}
1
+ {"version":3,"file":"LeaderElection.d.ts","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;GAkBG;AA+CH,uCAAuC;AACvC,eAAO,MAAM,YAAY,IAAI,CAAC;AAE9B;;;GAGG;AACH,eAAO,MAAM,wBAAwB,IAAI,CAAC;AAE1C,kFAAkF;AAClF,eAAO,MAAM,+BAA+B,IAAI,CAAC;AAEjD,kFAAkF;AAClF,eAAO,MAAM,qBAAqB,UAAU,CAAC;AAE7C;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,MAAM,CAAC;IAChB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,sBAAsB,CAAC,EAAE,MAAM,CAAC;CACjC;AAED;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,QAAQ,GAAG,UAAU,CAAC;IAC5B,sEAAsE;IACtE,UAAU,EAAE,iBAAiB,CAAC;CAC/B;AAED,MAAM,WAAW,wBAAwB;IACvC,aAAa,EAAE,OAAO,CAAC;IACvB,MAAM,EAAE,0BAA0B,GAAG,cAAc,GAAG,eAAe,GAAG,uBAAuB,CAAC;IAChG,gBAAgB,EAAE,MAAM,CAAC;IACzB,eAAe,EAAE,MAAM,CAAC;IACxB,wBAAwB,EAAE,MAAM,CAAC;IACjC,uBAAuB,EAAE,MAAM,CAAC;CACjC;AAED;;;GAGG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAQnD;AAED;;;GAGG;AACH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,CAKtE;AAED;;;GAGG;AACH,wBAAgB,+BAA+B,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,CAM/E;AAED;;;GAGG;AACH,wBAAgB,gBAAgB,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,iBAAiB,CAYnF;AAED;;;GAGG;AACH,wBAAgB,wBAAwB,CACtC,SAAS,EAAE,iBAAiB,EAC5B,QAAQ,EAAE,iBAAiB,GAC1B,wBAAwB,CAkD1B;AAED;;;;;GAKG;AACH,MAAM,WAAW,gBAAgB;IAC/B,aAAa,EAAE,OAAO,CAAC;IACvB,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED;;;;;;;;;;;;;;;;;GAiBG;AACH,wBAAsB,kBAAkB,CAAC,QAAQ,GAAE,MAAyB,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAiBvG;AAED;;;GAGG;AACH,wBAAsB,cAAc,CAAC,QAAQ,GAAE,MAAkB,GAAG,OAAO,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAiBpG;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAM5D;AAED;;;;;;;;;GASG;AACH,wBAAsB,eAAe,CACnC,IAAI,EAAE,iBAAiB,EACvB,QAAQ,GAAE,MAAkB,GAC3B,OAAO,CAAC,OAAO,CAAC,CAmBlB;AAED;;GAEG;AACH,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,IAAI,CAAC,CAEtD;AAED;;;;;;;;;GASG;AACH,wBAAsB,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CA0E1F;AAED;;;GAGG;AACH,wBAAsB,2BAA2B,CAAC,UAAU,EAAE,iBAAiB,GAAG,OAAO,CAAC,OAAO,CAAC,CAYjG;AAED;;;GAGG;AACH,wBAAsB,oBAAoB,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CAcnG;AAED;;;;;GAKG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,IAAI,CAgBlE;AAED;;;GAGG;AACH,wBAAgB,qBAAqB,IAAI,IAAI,CAM5C"}
@@ -18,8 +18,8 @@
18
18
  * @since v2.1.0 — Issue #1700
19
19
  */
20
20
  import { homedir } from 'node:os';
21
- import { join } from 'node:path';
22
- import { mkdir, readFile, writeFile, rename, unlink } from 'node:fs/promises';
21
+ import { dirname, join } from 'node:path';
22
+ import { mkdir, open, readFile, rename, unlink, writeFile } from 'node:fs/promises';
23
23
  import { UnicodeValidator } from '../../security/validators/unicodeValidator.js';
24
24
  import { env } from '../../config/env.js';
25
25
  import { PACKAGE_VERSION } from '../../generated/version.js';
@@ -209,9 +209,9 @@ export async function detectLegacyLeader(lockPath = LEGACY_LOCK_FILE) {
209
209
  * Read and parse the leader lock file.
210
210
  * Returns null if the file doesn't exist, is unreadable, or has invalid content.
211
211
  */
212
- export async function readLeaderLock() {
212
+ export async function readLeaderLock(lockPath = LOCK_FILE) {
213
213
  try {
214
- const content = await readFile(LOCK_FILE, 'utf-8');
214
+ const content = await readFile(lockPath, 'utf-8');
215
215
  const data = JSON.parse(content);
216
216
  if (data.version !== LOCK_VERSION || !data.pid || !data.port || !data.sessionId) {
217
217
  return null;
@@ -221,7 +221,7 @@ export async function readLeaderLock() {
221
221
  catch (error) {
222
222
  if (error.code !== 'ENOENT') {
223
223
  logger.debug('[LeaderElection] Ignoring unreadable or invalid leader lock', {
224
- lockFile: LOCK_FILE,
224
+ lockFile: lockPath,
225
225
  error: error instanceof Error ? error.message : String(error),
226
226
  });
227
227
  }
@@ -241,28 +241,31 @@ export function isLockStale(info) {
241
241
  /**
242
242
  * Attempt to atomically claim leadership.
243
243
  *
244
- * Writes to a temp file then renames to the lock path. On POSIX systems
245
- * rename is atomic, so only one writer wins. After renaming, re-reads the
246
- * lock to verify our PID won.
244
+ * Creates the lock file with exclusive-write semantics so only one process
245
+ * can win the initial claim. This avoids startup races where multiple
246
+ * contenders overwrite the lock in quick succession and each briefly believe
247
+ * they are leader.
247
248
  *
248
249
  * @returns true if this process successfully claimed leadership
249
250
  */
250
- export async function claimLeadership(info) {
251
- await mkdir(RUN_DIR, { recursive: true });
252
- const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);
251
+ export async function claimLeadership(info, lockPath = LOCK_FILE) {
252
+ await mkdir(dirname(lockPath), { recursive: true });
253
253
  try {
254
- await writeFile(tmpFile, JSON.stringify(info, null, 2), 'utf-8');
255
- await rename(tmpFile, LOCK_FILE);
256
- // Verify we won the race
257
- const written = await readLeaderLock();
254
+ const handle = await open(lockPath, 'wx', 0o600);
255
+ try {
256
+ await handle.writeFile(JSON.stringify(info, null, 2), 'utf-8');
257
+ }
258
+ finally {
259
+ await handle.close();
260
+ }
261
+ // Verify we won the race.
262
+ const written = await readLeaderLock(lockPath);
258
263
  return written !== null && written.pid === info.pid;
259
264
  }
260
- catch {
261
- // Clean up temp file on failure
262
- try {
263
- await unlink(tmpFile);
265
+ catch (error) {
266
+ if (error.code === 'EEXIST') {
267
+ return false;
264
268
  }
265
- catch { /* ignore */ }
266
269
  return false;
267
270
  }
268
271
  }
@@ -425,4 +428,4 @@ export function registerLeaderCleanup() {
425
428
  process.once('SIGINT', cleanup);
426
429
  process.once('SIGHUP', cleanup);
427
430
  }
428
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LeaderElection.js","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Leader election for the unified web console.\n *\n * When multiple MCP server instances run concurrently, only one should host\n * the web console (the \"leader\"). Others become \"followers\" that forward\n * events to the leader. This module handles:\n *\n * 1. Reading/writing a leader lock file at ~/.dollhouse/run/console-leader.lock\n * 2. Atomic claim via temp+rename to prevent TOCTOU races\n * 3. PID-based stale detection (signal-0 liveness check)\n * 4. Heartbeat updates (10s interval) so followers can detect hung leaders\n * 5. Cleanup on process exit\n *\n * The configured port binding is the ultimate tiebreaker: even if two\n * processes both write the lock file, only one can bind the port (see\n * `DOLLHOUSE_WEB_CONSOLE_PORT` in `src/config/env.ts`).\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport { homedir } from 'node:os';\nimport { join } from 'node:path';\nimport { mkdir, readFile, writeFile, rename, unlink } from 'node:fs/promises';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { env } from '../../config/env.js';\nimport { PACKAGE_VERSION } from '../../generated/version.js';\nimport { logger } from '../../utils/logger.js';\nimport { compareVersions } from '../../utils/version.js';\n\n/** Directory for runtime state files */\nconst RUN_DIR = join(homedir(), '.dollhouse', 'run');\n\n/**\n * Built-in default filename for the authenticated console's leader lock.\n *\n * The `.auth` suffix isolates this from any legacy no-authentication\n * DollhouseMCP installation that may also be running on the same\n * machine. Those older installs use `console-leader.lock` (no suffix);\n * the authenticated console uses `console-leader.auth.lock`. Combined\n * with the port separation, this means the two generations of the\n * console can coexist with zero interference — different port, different\n * lock file, different token file, independent leader election spaces.\n */\nconst DEFAULT_LOCK_FILENAME = 'console-leader.auth.lock';\n\n/** Legacy lock filename from the pre-authentication console. Used only for detection. */\nconst LEGACY_LOCK_FILENAME = 'console-leader.lock';\n\n/**\n * Path to the leader lock file. Prefers the `DOLLHOUSE_CONSOLE_LEADER_LOCK_FILE`\n * env var when set, otherwise uses `DEFAULT_LOCK_FILENAME` under RUN_DIR.\n * The env var is the single source of truth when present, so a deployment\n * can relocate the lock without code changes (see `src/config/env.ts`).\n */\nconst LOCK_FILE = env.DOLLHOUSE_CONSOLE_LEADER_LOCK_FILE ?? join(RUN_DIR, DEFAULT_LOCK_FILENAME);\n\n/** Path to the legacy pre-auth lock file (used by `detectLegacyLeader` only). */\nconst LEGACY_LOCK_FILE = join(RUN_DIR, LEGACY_LOCK_FILENAME);\n\n/** How often the leader updates its heartbeat (ms) */\nconst HEARTBEAT_INTERVAL_MS = 10_000;\n\n/** How long before a heartbeat is considered stale (ms) */\nconst HEARTBEAT_STALE_MS = 30_000;\n\n/** Current lock file schema version */\nexport const LOCK_VERSION = 1;\n\n/**\n * Version of the leader-election/session metadata contract used by the\n * authenticated web console. Older leaders will not have this field.\n */\nexport const CONSOLE_PROTOCOL_VERSION = 1;\n\n/** Missing protocol metadata means the leader predates version-aware election. */\nexport const LEGACY_CONSOLE_PROTOCOL_VERSION = 0;\n\n/** Old lock files do not carry package version metadata. Treat them as oldest. */\nexport const LEGACY_SERVER_VERSION = '0.0.0';\n\n/**\n * Information stored in the leader lock file.\n */\nexport interface ConsoleLeaderInfo {\n  version: number;\n  pid: number;\n  port: number;\n  sessionId: string;\n  startedAt: string;\n  heartbeat: string;\n  serverVersion?: string;\n  consoleProtocolVersion?: number;\n}\n\n/**\n * Result of a leader election attempt.\n */\nexport interface ElectionResult {\n  role: 'leader' | 'follower';\n  /** Leader info — for followers, this is the existing leader's info */\n  leaderInfo: ConsoleLeaderInfo;\n}\n\nexport interface LeaderPreferenceDecision {\n  shouldReplace: boolean;\n  reason: 'newer-compatible-version' | 'same-version' | 'older-version' | 'incompatible-protocol';\n  candidateVersion: string;\n  existingVersion: string;\n  candidateProtocolVersion: number;\n  existingProtocolVersion: number;\n}\n\n/**\n * Check whether a process with the given PID is alive.\n * Uses signal 0 which checks existence without sending a signal.\n */\nexport function isProcessAlive(pid: number): boolean {\n  try {\n    process.kill(pid, 0);\n    return true;\n  } catch (err: any) {\n    // EPERM = process exists but owned by another user — still alive\n    return err?.code === 'EPERM';\n  }\n}\n\n/**\n * Normalize the server version present in the leader lock.\n * Missing metadata means \"legacy leader\" for election purposes.\n */\nexport function getLeaderServerVersion(info: ConsoleLeaderInfo): string {\n  if (typeof info.serverVersion === 'string' && info.serverVersion.trim().length > 0) {\n    return info.serverVersion.trim();\n  }\n  return LEGACY_SERVER_VERSION;\n}\n\n/**\n * Normalize the console protocol version present in the leader lock.\n * Missing metadata means a leader from before version-aware election.\n */\nexport function getLeaderConsoleProtocolVersion(info: ConsoleLeaderInfo): number {\n  const raw = info.consoleProtocolVersion;\n  if (typeof raw === 'number' && Number.isInteger(raw) && raw >= 0) {\n    return raw;\n  }\n  return LEGACY_CONSOLE_PROTOCOL_VERSION;\n}\n\n/**\n * Create this process's leader metadata in one place so all leadership paths\n * publish the same version and protocol information.\n */\nexport function createLeaderInfo(sessionId: string, port: number): ConsoleLeaderInfo {\n  const now = new Date().toISOString();\n  return {\n    version: LOCK_VERSION,\n    pid: process.pid,\n    port,\n    sessionId: UnicodeValidator.normalize(sessionId).normalizedContent,\n    startedAt: now,\n    heartbeat: now,\n    serverVersion: PACKAGE_VERSION,\n    consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\n/**\n * Decide whether this process should replace the current live leader based on\n * compatibility first, then package version.\n */\nexport function evaluateLeaderPreference(\n  candidate: ConsoleLeaderInfo,\n  existing: ConsoleLeaderInfo,\n): LeaderPreferenceDecision {\n  const candidateVersion = getLeaderServerVersion(candidate);\n  const existingVersion = getLeaderServerVersion(existing);\n  const candidateProtocolVersion = getLeaderConsoleProtocolVersion(candidate);\n  const existingProtocolVersion = getLeaderConsoleProtocolVersion(existing);\n\n  const compatible =\n    existingProtocolVersion === candidateProtocolVersion ||\n    existingProtocolVersion === LEGACY_CONSOLE_PROTOCOL_VERSION;\n\n  if (!compatible) {\n    return {\n      shouldReplace: false,\n      reason: 'incompatible-protocol',\n      candidateVersion,\n      existingVersion,\n      candidateProtocolVersion,\n      existingProtocolVersion,\n    };\n  }\n\n  const versionComparison = compareVersions(candidateVersion, existingVersion);\n  if (versionComparison > 0) {\n    return {\n      shouldReplace: true,\n      reason: 'newer-compatible-version',\n      candidateVersion,\n      existingVersion,\n      candidateProtocolVersion,\n      existingProtocolVersion,\n    };\n  }\n  if (versionComparison === 0) {\n    return {\n      shouldReplace: false,\n      reason: 'same-version',\n      candidateVersion,\n      existingVersion,\n      candidateProtocolVersion,\n      existingProtocolVersion,\n    };\n  }\n  return {\n    shouldReplace: false,\n    reason: 'older-version',\n    candidateVersion,\n    existingVersion,\n    candidateProtocolVersion,\n    existingProtocolVersion,\n  };\n}\n\n/**\n * Result of a legacy-leader detection scan.\n * `legacyRunning === true` means a pre-authentication DollhouseMCP console\n * is currently running on this machine (its lock file exists and its pid\n * is alive). Callers can surface this to the user as a warning.\n */\nexport interface LegacyLeaderInfo {\n  legacyRunning: boolean;\n  pid?: number;\n  port?: number;\n  lockPath: string;\n}\n\n/**\n * Detect whether a legacy (pre-authentication) DollhouseMCP console is\n * currently running on this machine (#1794).\n *\n * The pre-authentication console writes its lock to\n * `~/.dollhouse/run/console-leader.lock` (no `.auth` suffix). An\n * authenticated console on a different port will not interfere with\n * it — they have fully independent ports, lock files, and token files —\n * but the user probably wants to know the two exist simultaneously\n * because the security posture of each console is different.\n *\n * Returns info about the legacy leader if one is detected, or\n * `{ legacyRunning: false }` otherwise.\n *\n * @param lockPath - Optional override for the legacy lock file path.\n *                   Defaults to the built-in legacy location. Primarily\n *                   used by tests to point at a temp directory.\n */\nexport async function detectLegacyLeader(lockPath: string = LEGACY_LOCK_FILE): Promise<LegacyLeaderInfo> {\n  try {\n    const content = await readFile(lockPath, 'utf-8');\n    const data = JSON.parse(content) as ConsoleLeaderInfo;\n    if (!data.pid || !isProcessAlive(data.pid)) {\n      return { legacyRunning: false, lockPath };\n    }\n    return {\n      legacyRunning: true,\n      pid: data.pid,\n      port: data.port,\n      lockPath,\n    };\n  } catch {\n    // File missing, unreadable, or invalid JSON — no legacy leader detected\n    return { legacyRunning: false, lockPath };\n  }\n}\n\n/**\n * Read and parse the leader lock file.\n * Returns null if the file doesn't exist, is unreadable, or has invalid content.\n */\nexport async function readLeaderLock(): Promise<ConsoleLeaderInfo | null> {\n  try {\n    const content = await readFile(LOCK_FILE, 'utf-8');\n    const data = JSON.parse(content) as ConsoleLeaderInfo;\n    if (data.version !== LOCK_VERSION || !data.pid || !data.port || !data.sessionId) {\n      return null;\n    }\n    return data;\n  } catch (error) {\n    if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {\n      logger.debug('[LeaderElection] Ignoring unreadable or invalid leader lock', {\n        lockFile: LOCK_FILE,\n        error: error instanceof Error ? error.message : String(error),\n      });\n    }\n    return null;\n  }\n}\n\n/**\n * Check if a leader lock is stale (dead process or expired heartbeat).\n */\nexport function isLockStale(info: ConsoleLeaderInfo): boolean {\n  if (!isProcessAlive(info.pid)) {\n    return true;\n  }\n  const heartbeatAge = Date.now() - new Date(info.heartbeat).getTime();\n  return heartbeatAge > HEARTBEAT_STALE_MS;\n}\n\n/**\n * Attempt to atomically claim leadership.\n *\n * Writes to a temp file then renames to the lock path. On POSIX systems\n * rename is atomic, so only one writer wins. After renaming, re-reads the\n * lock to verify our PID won.\n *\n * @returns true if this process successfully claimed leadership\n */\nexport async function claimLeadership(info: ConsoleLeaderInfo): Promise<boolean> {\n  await mkdir(RUN_DIR, { recursive: true });\n  const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);\n  try {\n    await writeFile(tmpFile, JSON.stringify(info, null, 2), 'utf-8');\n    await rename(tmpFile, LOCK_FILE);\n\n    // Verify we won the race\n    const written = await readLeaderLock();\n    return written !== null && written.pid === info.pid;\n  } catch {\n    // Clean up temp file on failure\n    try { await unlink(tmpFile); } catch { /* ignore */ }\n    return false;\n  }\n}\n\n/**\n * Delete the leader lock file (for cleanup or takeover).\n */\nexport async function deleteLeaderLock(): Promise<void> {\n  try { await unlink(LOCK_FILE); } catch { /* already gone */ }\n}\n\n/**\n * Run the leader election protocol.\n *\n * 1. If no lock exists or lock is stale → claim leadership\n * 2. If lock exists with a live, responsive leader → become follower\n *\n * @param sessionId - This process's unique session identifier\n * @param port - The port this process would use as leader (see `DOLLHOUSE_WEB_CONSOLE_PORT`)\n * @returns Election result with role and leader info\n */\nexport async function electLeader(sessionId: string, port: number): Promise<ElectionResult> {\n  const myInfo = createLeaderInfo(sessionId, port);\n  sessionId = myInfo.sessionId;\n  const existingLock = await readLeaderLock();\n\n  if (existingLock && !isLockStale(existingLock)) {\n    const preference = evaluateLeaderPreference(myInfo, existingLock);\n    if (preference.shouldReplace) {\n      logger.info('[LeaderElection] Replacing leader with newer compatible version', {\n        staleSession: existingLock.sessionId,\n        stalePid: existingLock.pid,\n        stalePort: existingLock.port,\n        staleVersion: preference.existingVersion,\n        staleProtocolVersion: preference.existingProtocolVersion,\n        mySession: sessionId,\n        myPid: process.pid,\n        myVersion: preference.candidateVersion,\n        myProtocolVersion: preference.candidateProtocolVersion,\n      });\n      await deleteLeaderLock();\n    } else {\n      logger.info('[LeaderElection] Existing leader found — becoming follower', {\n        leaderSession: existingLock.sessionId,\n        leaderPid: existingLock.pid,\n        leaderPort: existingLock.port,\n        leaderVersion: preference.existingVersion,\n        leaderProtocolVersion: preference.existingProtocolVersion,\n        mySession: sessionId,\n        myPid: process.pid,\n        myVersion: preference.candidateVersion,\n        myProtocolVersion: preference.candidateProtocolVersion,\n        reason: preference.reason,\n      });\n      return { role: 'follower', leaderInfo: existingLock };\n    }\n  }\n\n  if (existingLock && !isLockStale(existingLock)) {\n    // Leader was intentionally replaced above. Continue to the claim path.\n  } else if (existingLock) {\n    // No valid leader — try to claim\n    const alive = isProcessAlive(existingLock.pid);\n    const heartbeatAge = Date.now() - new Date(existingLock.heartbeat).getTime();\n    logger.info('[LeaderElection] Stale leader lock — taking over', {\n      stalePid: existingLock.pid, alive, heartbeatAgeMs: heartbeatAge,\n      staleSession: existingLock.sessionId, mySession: sessionId,\n    });\n    await deleteLeaderLock();\n  }\n\n  const claimed = await claimLeadership(myInfo);\n  if (claimed) {\n    logger.info('[LeaderElection] Claimed leadership', { sessionId, port, pid: process.pid });\n    return { role: 'leader', leaderInfo: myInfo };\n  }\n\n  // Another process won the race — re-read and become follower\n  const winner = await readLeaderLock();\n  if (winner) {\n    logger.info('[LeaderElection] Lost election — becoming follower', {\n      winnerPid: winner.pid, winnerSession: winner.sessionId, mySession: sessionId, myPid: process.pid,\n    });\n    return { role: 'follower', leaderInfo: winner };\n  }\n\n  // Extremely unlikely: lock disappeared between our claim and re-read. Retry once.\n  logger.warn('[LeaderElection] Lock vanished after failed claim. Retrying.');\n  const retryInfo: ConsoleLeaderInfo = { ...createLeaderInfo(sessionId, port) };\n  const retryClaimed = await claimLeadership(retryInfo);\n  if (retryClaimed) {\n    return { role: 'leader', leaderInfo: retryInfo };\n  }\n  const actualLeader = await readLeaderLock();\n  return { role: 'follower', leaderInfo: actualLeader ?? retryInfo };\n}\n\n/**\n * Probe whether the leader's web console is reachable.\n * Returns true if the leader's ingest endpoint responds, false otherwise.\n */\nexport async function isLeaderWebConsoleReachable(leaderInfo: ConsoleLeaderInfo): Promise<boolean> {\n  try {\n    const controller = new AbortController();\n    const timeout = setTimeout(() => controller.abort(), 2_000);\n    const res = await fetch(`http://127.0.0.1:${leaderInfo.port}/api/logs/stats`, {\n      signal: controller.signal,\n    });\n    clearTimeout(timeout);\n    return res.ok;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Force claim leadership by deleting the existing lock and claiming.\n * Used when the existing leader is alive but not running a web console.\n */\nexport async function forceClaimLeadership(sessionId: string, port: number): Promise<ElectionResult> {\n  logger.info('[LeaderElection] Forcing leadership takeover — existing leader has no web console');\n  await deleteLeaderLock();\n  const myInfo = createLeaderInfo(sessionId, port);\n\n  const claimed = await claimLeadership(myInfo);\n  if (claimed) {\n    logger.info('[LeaderElection] Forced leadership claimed', { sessionId, port, pid: process.pid });\n    return { role: 'leader', leaderInfo: myInfo };\n  }\n\n  // Failed — fall back to follower\n  const winner = await readLeaderLock();\n  return { role: 'follower', leaderInfo: winner ?? myInfo };\n}\n\n/**\n * Start the leader heartbeat loop.\n * Updates the lock file every HEARTBEAT_INTERVAL_MS so followers know the leader is alive.\n *\n * @returns A stop function to clear the interval\n */\nexport function startHeartbeat(info: ConsoleLeaderInfo): () => void {\n  const interval = setInterval(async () => {\n    try {\n      const updated: ConsoleLeaderInfo = { ...info, heartbeat: new Date().toISOString() };\n      const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);\n      await writeFile(tmpFile, JSON.stringify(updated, null, 2), 'utf-8');\n      await rename(tmpFile, LOCK_FILE);\n    } catch (err) {\n      logger.debug('[LeaderElection] Heartbeat write failed:', err);\n    }\n  }, HEARTBEAT_INTERVAL_MS);\n\n  // Don't let the heartbeat interval keep the process alive\n  interval.unref();\n\n  return () => clearInterval(interval);\n}\n\n/**\n * Register cleanup handlers to remove the leader lock on process exit.\n * Should only be called by the leader.\n */\nexport function registerLeaderCleanup(): void {\n  const cleanup = () => { deleteLeaderLock().catch(() => {}); };\n  process.once('exit', cleanup);\n  process.once('SIGTERM', cleanup);\n  process.once('SIGINT', cleanup);\n  process.once('SIGHUP', cleanup);\n}\n"]}
431
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LeaderElection.js","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Leader election for the unified web console.\n *\n * When multiple MCP server instances run concurrently, only one should host\n * the web console (the \"leader\"). Others become \"followers\" that forward\n * events to the leader. This module handles:\n *\n * 1. Reading/writing a leader lock file at ~/.dollhouse/run/console-leader.lock\n * 2. Atomic claim via temp+rename to prevent TOCTOU races\n * 3. PID-based stale detection (signal-0 liveness check)\n * 4. Heartbeat updates (10s interval) so followers can detect hung leaders\n * 5. Cleanup on process exit\n *\n * The configured port binding is the ultimate tiebreaker: even if two\n * processes both write the lock file, only one can bind the port (see\n * `DOLLHOUSE_WEB_CONSOLE_PORT` in `src/config/env.ts`).\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport { homedir } from 'node:os';\nimport { dirname, join } from 'node:path';\nimport { mkdir, open, readFile, rename, unlink, writeFile } from 'node:fs/promises';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { env } from '../../config/env.js';\nimport { PACKAGE_VERSION } from '../../generated/version.js';\nimport { logger } from '../../utils/logger.js';\nimport { compareVersions } from '../../utils/version.js';\n\n/** Directory for runtime state files */\nconst RUN_DIR = join(homedir(), '.dollhouse', 'run');\n\n/**\n * Built-in default filename for the authenticated console's leader lock.\n *\n * The `.auth` suffix isolates this from any legacy no-authentication\n * DollhouseMCP installation that may also be running on the same\n * machine. Those older installs use `console-leader.lock` (no suffix);\n * the authenticated console uses `console-leader.auth.lock`. Combined\n * with the port separation, this means the two generations of the\n * console can coexist with zero interference — different port, different\n * lock file, different token file, independent leader election spaces.\n */\nconst DEFAULT_LOCK_FILENAME = 'console-leader.auth.lock';\n\n/** Legacy lock filename from the pre-authentication console. Used only for detection. */\nconst LEGACY_LOCK_FILENAME = 'console-leader.lock';\n\n/**\n * Path to the leader lock file. Prefers the `DOLLHOUSE_CONSOLE_LEADER_LOCK_FILE`\n * env var when set, otherwise uses `DEFAULT_LOCK_FILENAME` under RUN_DIR.\n * The env var is the single source of truth when present, so a deployment\n * can relocate the lock without code changes (see `src/config/env.ts`).\n */\nconst LOCK_FILE = env.DOLLHOUSE_CONSOLE_LEADER_LOCK_FILE ?? join(RUN_DIR, DEFAULT_LOCK_FILENAME);\n\n/** Path to the legacy pre-auth lock file (used by `detectLegacyLeader` only). */\nconst LEGACY_LOCK_FILE = join(RUN_DIR, LEGACY_LOCK_FILENAME);\n\n/** How often the leader updates its heartbeat (ms) */\nconst HEARTBEAT_INTERVAL_MS = 10_000;\n\n/** How long before a heartbeat is considered stale (ms) */\nconst HEARTBEAT_STALE_MS = 30_000;\n\n/** Current lock file schema version */\nexport const LOCK_VERSION = 1;\n\n/**\n * Version of the leader-election/session metadata contract used by the\n * authenticated web console. Older leaders will not have this field.\n */\nexport const CONSOLE_PROTOCOL_VERSION = 1;\n\n/** Missing protocol metadata means the leader predates version-aware election. */\nexport const LEGACY_CONSOLE_PROTOCOL_VERSION = 0;\n\n/** Old lock files do not carry package version metadata. Treat them as oldest. */\nexport const LEGACY_SERVER_VERSION = '0.0.0';\n\n/**\n * Information stored in the leader lock file.\n */\nexport interface ConsoleLeaderInfo {\n  version: number;\n  pid: number;\n  port: number;\n  sessionId: string;\n  startedAt: string;\n  heartbeat: string;\n  serverVersion?: string;\n  consoleProtocolVersion?: number;\n}\n\n/**\n * Result of a leader election attempt.\n */\nexport interface ElectionResult {\n  role: 'leader' | 'follower';\n  /** Leader info — for followers, this is the existing leader's info */\n  leaderInfo: ConsoleLeaderInfo;\n}\n\nexport interface LeaderPreferenceDecision {\n  shouldReplace: boolean;\n  reason: 'newer-compatible-version' | 'same-version' | 'older-version' | 'incompatible-protocol';\n  candidateVersion: string;\n  existingVersion: string;\n  candidateProtocolVersion: number;\n  existingProtocolVersion: number;\n}\n\n/**\n * Check whether a process with the given PID is alive.\n * Uses signal 0 which checks existence without sending a signal.\n */\nexport function isProcessAlive(pid: number): boolean {\n  try {\n    process.kill(pid, 0);\n    return true;\n  } catch (err: any) {\n    // EPERM = process exists but owned by another user — still alive\n    return err?.code === 'EPERM';\n  }\n}\n\n/**\n * Normalize the server version present in the leader lock.\n * Missing metadata means \"legacy leader\" for election purposes.\n */\nexport function getLeaderServerVersion(info: ConsoleLeaderInfo): string {\n  if (typeof info.serverVersion === 'string' && info.serverVersion.trim().length > 0) {\n    return info.serverVersion.trim();\n  }\n  return LEGACY_SERVER_VERSION;\n}\n\n/**\n * Normalize the console protocol version present in the leader lock.\n * Missing metadata means a leader from before version-aware election.\n */\nexport function getLeaderConsoleProtocolVersion(info: ConsoleLeaderInfo): number {\n  const raw = info.consoleProtocolVersion;\n  if (typeof raw === 'number' && Number.isInteger(raw) && raw >= 0) {\n    return raw;\n  }\n  return LEGACY_CONSOLE_PROTOCOL_VERSION;\n}\n\n/**\n * Create this process's leader metadata in one place so all leadership paths\n * publish the same version and protocol information.\n */\nexport function createLeaderInfo(sessionId: string, port: number): ConsoleLeaderInfo {\n  const now = new Date().toISOString();\n  return {\n    version: LOCK_VERSION,\n    pid: process.pid,\n    port,\n    sessionId: UnicodeValidator.normalize(sessionId).normalizedContent,\n    startedAt: now,\n    heartbeat: now,\n    serverVersion: PACKAGE_VERSION,\n    consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\n/**\n * Decide whether this process should replace the current live leader based on\n * compatibility first, then package version.\n */\nexport function evaluateLeaderPreference(\n  candidate: ConsoleLeaderInfo,\n  existing: ConsoleLeaderInfo,\n): LeaderPreferenceDecision {\n  const candidateVersion = getLeaderServerVersion(candidate);\n  const existingVersion = getLeaderServerVersion(existing);\n  const candidateProtocolVersion = getLeaderConsoleProtocolVersion(candidate);\n  const existingProtocolVersion = getLeaderConsoleProtocolVersion(existing);\n\n  const compatible =\n    existingProtocolVersion === candidateProtocolVersion ||\n    existingProtocolVersion === LEGACY_CONSOLE_PROTOCOL_VERSION;\n\n  if (!compatible) {\n    return {\n      shouldReplace: false,\n      reason: 'incompatible-protocol',\n      candidateVersion,\n      existingVersion,\n      candidateProtocolVersion,\n      existingProtocolVersion,\n    };\n  }\n\n  const versionComparison = compareVersions(candidateVersion, existingVersion);\n  if (versionComparison > 0) {\n    return {\n      shouldReplace: true,\n      reason: 'newer-compatible-version',\n      candidateVersion,\n      existingVersion,\n      candidateProtocolVersion,\n      existingProtocolVersion,\n    };\n  }\n  if (versionComparison === 0) {\n    return {\n      shouldReplace: false,\n      reason: 'same-version',\n      candidateVersion,\n      existingVersion,\n      candidateProtocolVersion,\n      existingProtocolVersion,\n    };\n  }\n  return {\n    shouldReplace: false,\n    reason: 'older-version',\n    candidateVersion,\n    existingVersion,\n    candidateProtocolVersion,\n    existingProtocolVersion,\n  };\n}\n\n/**\n * Result of a legacy-leader detection scan.\n * `legacyRunning === true` means a pre-authentication DollhouseMCP console\n * is currently running on this machine (its lock file exists and its pid\n * is alive). Callers can surface this to the user as a warning.\n */\nexport interface LegacyLeaderInfo {\n  legacyRunning: boolean;\n  pid?: number;\n  port?: number;\n  lockPath: string;\n}\n\n/**\n * Detect whether a legacy (pre-authentication) DollhouseMCP console is\n * currently running on this machine (#1794).\n *\n * The pre-authentication console writes its lock to\n * `~/.dollhouse/run/console-leader.lock` (no `.auth` suffix). An\n * authenticated console on a different port will not interfere with\n * it — they have fully independent ports, lock files, and token files —\n * but the user probably wants to know the two exist simultaneously\n * because the security posture of each console is different.\n *\n * Returns info about the legacy leader if one is detected, or\n * `{ legacyRunning: false }` otherwise.\n *\n * @param lockPath - Optional override for the legacy lock file path.\n *                   Defaults to the built-in legacy location. Primarily\n *                   used by tests to point at a temp directory.\n */\nexport async function detectLegacyLeader(lockPath: string = LEGACY_LOCK_FILE): Promise<LegacyLeaderInfo> {\n  try {\n    const content = await readFile(lockPath, 'utf-8');\n    const data = JSON.parse(content) as ConsoleLeaderInfo;\n    if (!data.pid || !isProcessAlive(data.pid)) {\n      return { legacyRunning: false, lockPath };\n    }\n    return {\n      legacyRunning: true,\n      pid: data.pid,\n      port: data.port,\n      lockPath,\n    };\n  } catch {\n    // File missing, unreadable, or invalid JSON — no legacy leader detected\n    return { legacyRunning: false, lockPath };\n  }\n}\n\n/**\n * Read and parse the leader lock file.\n * Returns null if the file doesn't exist, is unreadable, or has invalid content.\n */\nexport async function readLeaderLock(lockPath: string = LOCK_FILE): Promise<ConsoleLeaderInfo | null> {\n  try {\n    const content = await readFile(lockPath, 'utf-8');\n    const data = JSON.parse(content) as ConsoleLeaderInfo;\n    if (data.version !== LOCK_VERSION || !data.pid || !data.port || !data.sessionId) {\n      return null;\n    }\n    return data;\n  } catch (error) {\n    if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {\n      logger.debug('[LeaderElection] Ignoring unreadable or invalid leader lock', {\n        lockFile: lockPath,\n        error: error instanceof Error ? error.message : String(error),\n      });\n    }\n    return null;\n  }\n}\n\n/**\n * Check if a leader lock is stale (dead process or expired heartbeat).\n */\nexport function isLockStale(info: ConsoleLeaderInfo): boolean {\n  if (!isProcessAlive(info.pid)) {\n    return true;\n  }\n  const heartbeatAge = Date.now() - new Date(info.heartbeat).getTime();\n  return heartbeatAge > HEARTBEAT_STALE_MS;\n}\n\n/**\n * Attempt to atomically claim leadership.\n *\n * Creates the lock file with exclusive-write semantics so only one process\n * can win the initial claim. This avoids startup races where multiple\n * contenders overwrite the lock in quick succession and each briefly believe\n * they are leader.\n *\n * @returns true if this process successfully claimed leadership\n */\nexport async function claimLeadership(\n  info: ConsoleLeaderInfo,\n  lockPath: string = LOCK_FILE,\n): Promise<boolean> {\n  await mkdir(dirname(lockPath), { recursive: true });\n  try {\n    const handle = await open(lockPath, 'wx', 0o600);\n    try {\n      await handle.writeFile(JSON.stringify(info, null, 2), 'utf-8');\n    } finally {\n      await handle.close();\n    }\n\n    // Verify we won the race.\n    const written = await readLeaderLock(lockPath);\n    return written !== null && written.pid === info.pid;\n  } catch (error) {\n    if ((error as NodeJS.ErrnoException).code === 'EEXIST') {\n      return false;\n    }\n    return false;\n  }\n}\n\n/**\n * Delete the leader lock file (for cleanup or takeover).\n */\nexport async function deleteLeaderLock(): Promise<void> {\n  try { await unlink(LOCK_FILE); } catch { /* already gone */ }\n}\n\n/**\n * Run the leader election protocol.\n *\n * 1. If no lock exists or lock is stale → claim leadership\n * 2. If lock exists with a live, responsive leader → become follower\n *\n * @param sessionId - This process's unique session identifier\n * @param port - The port this process would use as leader (see `DOLLHOUSE_WEB_CONSOLE_PORT`)\n * @returns Election result with role and leader info\n */\nexport async function electLeader(sessionId: string, port: number): Promise<ElectionResult> {\n  const myInfo = createLeaderInfo(sessionId, port);\n  sessionId = myInfo.sessionId;\n  const existingLock = await readLeaderLock();\n\n  if (existingLock && !isLockStale(existingLock)) {\n    const preference = evaluateLeaderPreference(myInfo, existingLock);\n    if (preference.shouldReplace) {\n      logger.info('[LeaderElection] Replacing leader with newer compatible version', {\n        staleSession: existingLock.sessionId,\n        stalePid: existingLock.pid,\n        stalePort: existingLock.port,\n        staleVersion: preference.existingVersion,\n        staleProtocolVersion: preference.existingProtocolVersion,\n        mySession: sessionId,\n        myPid: process.pid,\n        myVersion: preference.candidateVersion,\n        myProtocolVersion: preference.candidateProtocolVersion,\n      });\n      await deleteLeaderLock();\n    } else {\n      logger.info('[LeaderElection] Existing leader found — becoming follower', {\n        leaderSession: existingLock.sessionId,\n        leaderPid: existingLock.pid,\n        leaderPort: existingLock.port,\n        leaderVersion: preference.existingVersion,\n        leaderProtocolVersion: preference.existingProtocolVersion,\n        mySession: sessionId,\n        myPid: process.pid,\n        myVersion: preference.candidateVersion,\n        myProtocolVersion: preference.candidateProtocolVersion,\n        reason: preference.reason,\n      });\n      return { role: 'follower', leaderInfo: existingLock };\n    }\n  }\n\n  if (existingLock && !isLockStale(existingLock)) {\n    // Leader was intentionally replaced above. Continue to the claim path.\n  } else if (existingLock) {\n    // No valid leader — try to claim\n    const alive = isProcessAlive(existingLock.pid);\n    const heartbeatAge = Date.now() - new Date(existingLock.heartbeat).getTime();\n    logger.info('[LeaderElection] Stale leader lock — taking over', {\n      stalePid: existingLock.pid, alive, heartbeatAgeMs: heartbeatAge,\n      staleSession: existingLock.sessionId, mySession: sessionId,\n    });\n    await deleteLeaderLock();\n  }\n\n  const claimed = await claimLeadership(myInfo);\n  if (claimed) {\n    logger.info('[LeaderElection] Claimed leadership', { sessionId, port, pid: process.pid });\n    return { role: 'leader', leaderInfo: myInfo };\n  }\n\n  // Another process won the race — re-read and become follower\n  const winner = await readLeaderLock();\n  if (winner) {\n    logger.info('[LeaderElection] Lost election — becoming follower', {\n      winnerPid: winner.pid, winnerSession: winner.sessionId, mySession: sessionId, myPid: process.pid,\n    });\n    return { role: 'follower', leaderInfo: winner };\n  }\n\n  // Extremely unlikely: lock disappeared between our claim and re-read. Retry once.\n  logger.warn('[LeaderElection] Lock vanished after failed claim. Retrying.');\n  const retryInfo: ConsoleLeaderInfo = { ...createLeaderInfo(sessionId, port) };\n  const retryClaimed = await claimLeadership(retryInfo);\n  if (retryClaimed) {\n    return { role: 'leader', leaderInfo: retryInfo };\n  }\n  const actualLeader = await readLeaderLock();\n  return { role: 'follower', leaderInfo: actualLeader ?? retryInfo };\n}\n\n/**\n * Probe whether the leader's web console is reachable.\n * Returns true if the leader's ingest endpoint responds, false otherwise.\n */\nexport async function isLeaderWebConsoleReachable(leaderInfo: ConsoleLeaderInfo): Promise<boolean> {\n  try {\n    const controller = new AbortController();\n    const timeout = setTimeout(() => controller.abort(), 2_000);\n    const res = await fetch(`http://127.0.0.1:${leaderInfo.port}/api/logs/stats`, {\n      signal: controller.signal,\n    });\n    clearTimeout(timeout);\n    return res.ok;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Force claim leadership by deleting the existing lock and claiming.\n * Used when the existing leader is alive but not running a web console.\n */\nexport async function forceClaimLeadership(sessionId: string, port: number): Promise<ElectionResult> {\n  logger.info('[LeaderElection] Forcing leadership takeover — existing leader has no web console');\n  await deleteLeaderLock();\n  const myInfo = createLeaderInfo(sessionId, port);\n\n  const claimed = await claimLeadership(myInfo);\n  if (claimed) {\n    logger.info('[LeaderElection] Forced leadership claimed', { sessionId, port, pid: process.pid });\n    return { role: 'leader', leaderInfo: myInfo };\n  }\n\n  // Failed — fall back to follower\n  const winner = await readLeaderLock();\n  return { role: 'follower', leaderInfo: winner ?? myInfo };\n}\n\n/**\n * Start the leader heartbeat loop.\n * Updates the lock file every HEARTBEAT_INTERVAL_MS so followers know the leader is alive.\n *\n * @returns A stop function to clear the interval\n */\nexport function startHeartbeat(info: ConsoleLeaderInfo): () => void {\n  const interval = setInterval(async () => {\n    try {\n      const updated: ConsoleLeaderInfo = { ...info, heartbeat: new Date().toISOString() };\n      const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);\n      await writeFile(tmpFile, JSON.stringify(updated, null, 2), 'utf-8');\n      await rename(tmpFile, LOCK_FILE);\n    } catch (err) {\n      logger.debug('[LeaderElection] Heartbeat write failed:', err);\n    }\n  }, HEARTBEAT_INTERVAL_MS);\n\n  // Don't let the heartbeat interval keep the process alive\n  interval.unref();\n\n  return () => clearInterval(interval);\n}\n\n/**\n * Register cleanup handlers to remove the leader lock on process exit.\n * Should only be called by the leader.\n */\nexport function registerLeaderCleanup(): void {\n  const cleanup = () => { deleteLeaderLock().catch(() => {}); };\n  process.once('exit', cleanup);\n  process.once('SIGTERM', cleanup);\n  process.once('SIGINT', cleanup);\n  process.once('SIGHUP', cleanup);\n}\n"]}
@@ -121,6 +121,7 @@ interface LeaderLeaseReconciliationDependencies {
121
121
  readLeaderLockImpl?: typeof readLeaderLock;
122
122
  findPidOnPortImpl?: typeof findPidOnPort;
123
123
  claimLeadershipImpl?: typeof claimLeadership;
124
+ deleteLeaderLockImpl?: typeof deleteLeaderLock;
124
125
  killStaleProcessDetailedImpl?: typeof killStaleProcessDetailed;
125
126
  setTimeoutImpl?: typeof setTimeout;
126
127
  clearTimeoutImpl?: typeof clearTimeout;
@@ -139,7 +140,7 @@ export declare function recoverLeaderBindFailure(provisionalLeader: ConsoleLeade
139
140
  export declare function evaluatePortOwnerReplacement(candidateLeader: ConsoleLeaderInfo, fallback: PortLeaderDiscovery): PortOwnerReplacementDecision;
140
141
  export declare function resolveFollowerAuthority(sessionId: string, consolePort: number, election: ElectionResult, deps?: FollowerAuthorityDependencies): Promise<FollowerAuthorityResolution>;
141
142
  export declare function startFollowerAuthorityMonitor(options: UnifiedConsoleOptions, consolePort: number, election: ElectionResult, promotionMgr: PromotionManager, forwardingSink: LeaderForwardingLogSink, sessionHeartbeat: SessionHeartbeat, deps?: FollowerAuthorityMonitorDependencies): () => void;
142
- export declare function reconcileLeaderLease(sessionId: string, consolePort: number, deps?: LeaderLeaseReconciliationDependencies): Promise<'not-port-owner' | 'already-owned' | 'reconciled'>;
143
+ export declare function reconcileLeaderLease(sessionId: string, consolePort: number, deps?: LeaderLeaseReconciliationDependencies): Promise<'not-port-owner' | 'already-owned' | 'reconciled' | 'reclaim-failed'>;
143
144
  export declare function startLeaderLeaseMonitor(sessionId: string, consolePort: number, deps?: LeaderLeaseReconciliationDependencies): () => void;
144
145
  /**
145
146
  * Start the unified web console.
@@ -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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"}