@dollhousemcp/mcp-server 2.0.18 → 2.0.20
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 +8 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/ElementCRUDHandler.d.ts +30 -0
- package/dist/handlers/ElementCRUDHandler.d.ts.map +1 -1
- package/dist/handlers/ElementCRUDHandler.js +141 -2
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts +3 -0
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/MCPAQLHandler.js +98 -1
- package/dist/handlers/mcp-aql/OperationRouter.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/OperationRouter.js +6 -1
- package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/OperationSchema.js +17 -1
- package/dist/handlers/mcp-aql/policies/AgentToolPolicyTranslator.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/AgentToolPolicyTranslator.js +2 -1
- package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/ElementPolicies.js +2 -1
- package/dist/handlers/mcp-aql/policies/OperationPolicies.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/OperationPolicies.js +6 -1
- package/dist/handlers/mcp-aql/policies/ToolClassification.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/ToolClassification.js +2 -1
- package/dist/server/tools/MCPAQLTools.js +2 -1
- package/dist/web/console/IngestRoutes.d.ts +6 -0
- package/dist/web/console/IngestRoutes.d.ts.map +1 -1
- package/dist/web/console/IngestRoutes.js +38 -9
- package/dist/web/console/LeaderElection.d.ts +39 -0
- package/dist/web/console/LeaderElection.d.ts.map +1 -1
- package/dist/web/console/LeaderElection.js +147 -29
- package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
- package/dist/web/console/LeaderForwardingSink.js +5 -1
- package/dist/web/console/PromotionManager.d.ts.map +1 -1
- package/dist/web/console/PromotionManager.js +3 -11
- package/dist/web/console/StaleProcessRecovery.d.ts +11 -0
- package/dist/web/console/StaleProcessRecovery.d.ts.map +1 -1
- package/dist/web/console/StaleProcessRecovery.js +229 -63
- package/dist/web/console/UnifiedConsole.d.ts +22 -1
- package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
- package/dist/web/console/UnifiedConsole.js +172 -11
- package/dist/web/public/app.js +62 -1
- package/dist/web/public/index.html +19 -17
- package/dist/web/public/sessions.js +111 -0
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +12 -10
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"StaleProcessRecovery.d.ts","sourceRoot":"","sources":["../../../src/web/console/StaleProcessRecovery.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;
|
|
1
|
+
{"version":3,"file":"StaleProcessRecovery.d.ts","sourceRoot":"","sources":["../../../src/web/console/StaleProcessRecovery.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AA8DH,MAAM,WAAW,uBAAuB;IACtC,MAAM,EAAE,OAAO,CAAC;IAChB,MAAM,EACF,gBAAgB,GAChB,gBAAgB,GAChB,uBAAuB,GACvB,oBAAoB,GACpB,YAAY,GACZ,cAAc,GACd,aAAa,GACb,eAAe,CAAC;IACpB,GAAG,EAAE,MAAM,CAAC;IACZ,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAED,wBAAgB,yBAAyB,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAGlE;AA8LD;;;;;;;GAOG;AACH,wBAAsB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC,CAyBxE;AAED;;;;;;;GAOG;AACH,wBAAsB,gBAAgB,CAAC,GAAG,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,CAGlF;AAED,wBAAsB,wBAAwB,CAAC,GAAG,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,uBAAuB,CAAC,CAyB1G;AAED;;;;;;GAMG;AACH,wBAAsB,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,CAoCrE"}
|
|
@@ -8,6 +8,7 @@
|
|
|
8
8
|
* Extracted to a standalone module so it can be tested without importing
|
|
9
9
|
* the full Express server and its dependency chain.
|
|
10
10
|
*/
|
|
11
|
+
import { UnicodeValidator } from '../../security/validators/unicodeValidator.js';
|
|
11
12
|
// Use lazy import for logger to avoid pulling in the full env.ts/config chain
|
|
12
13
|
// at module load time. This keeps the module independently testable.
|
|
13
14
|
/** Timeout for lsof/fuser/ps system calls (ms) */
|
|
@@ -20,6 +21,12 @@ const KILL_POLL_COUNT = 10;
|
|
|
20
21
|
const SIGKILL_WAIT_MS = 500;
|
|
21
22
|
/** Wait between lock file reads for TOCTOU mitigation (ms) */
|
|
22
23
|
const LOCK_RECHECK_DELAY_MS = 500;
|
|
24
|
+
/** Number of lock-file checks before deciding the port holder is not a fresh leader. */
|
|
25
|
+
const LOCK_RECHECK_ATTEMPTS = 2;
|
|
26
|
+
/** PID used by the OS init/launchd process; direct children are effectively orphaned. */
|
|
27
|
+
const ROOT_PARENT_PID = 1;
|
|
28
|
+
/** Number of `ps` columns requested by inspectProcess: user, pid, ppid, command. */
|
|
29
|
+
const PROCESS_INSPECTION_FIELD_COUNT = 4;
|
|
23
30
|
let _logger = null;
|
|
24
31
|
async function getLogger() {
|
|
25
32
|
if (!_logger) {
|
|
@@ -31,10 +38,190 @@ async function getLogger() {
|
|
|
31
38
|
return _logger;
|
|
32
39
|
}
|
|
33
40
|
const logger = {
|
|
34
|
-
warn: async (...args) => {
|
|
35
|
-
|
|
36
|
-
|
|
41
|
+
warn: async (...args) => {
|
|
42
|
+
const l = await getLogger();
|
|
43
|
+
if (l)
|
|
44
|
+
l.warn(args[0], args[1]);
|
|
45
|
+
else
|
|
46
|
+
console.error('[WARN]', ...args);
|
|
47
|
+
},
|
|
48
|
+
info: async (...args) => {
|
|
49
|
+
const l = await getLogger();
|
|
50
|
+
if (l)
|
|
51
|
+
l.info(args[0], args[1]);
|
|
52
|
+
else
|
|
53
|
+
console.error('[INFO]', ...args);
|
|
54
|
+
},
|
|
55
|
+
debug: async (...args) => {
|
|
56
|
+
const l = await getLogger();
|
|
57
|
+
if (l)
|
|
58
|
+
l.debug(args[0], args[1]);
|
|
59
|
+
},
|
|
37
60
|
};
|
|
61
|
+
const MCP_HOST_PARENT_PATTERNS = [
|
|
62
|
+
/Claude\.app\/Contents\/Helpers\/disclaimer/i,
|
|
63
|
+
/Codex\.app\/Contents\/Resources\/codex app-server/i,
|
|
64
|
+
/Cursor\.app\//i,
|
|
65
|
+
/Windsurf\.app\//i,
|
|
66
|
+
];
|
|
67
|
+
export function isRecognizedMcpHostParent(command) {
|
|
68
|
+
const normalizedCommand = UnicodeValidator.normalize(command).normalizedContent;
|
|
69
|
+
return MCP_HOST_PARENT_PATTERNS.some((pattern) => pattern.test(normalizedCommand));
|
|
70
|
+
}
|
|
71
|
+
function isDollhouseProcessCommand(cmdLine) {
|
|
72
|
+
const normalizedCommand = UnicodeValidator.normalize(cmdLine).normalizedContent;
|
|
73
|
+
const isDollhouseBin = /(?:^|\/)dollhousemcp(?:\s|$)/.test(normalizedCommand) ||
|
|
74
|
+
normalizedCommand.includes('.bin/dollhousemcp');
|
|
75
|
+
const isMcpServerBin = normalizedCommand.includes('.bin/mcp-server') ||
|
|
76
|
+
/(?:dollhousemcp|mcp-server)[/\\]dist[/\\]index\.js/.test(normalizedCommand);
|
|
77
|
+
return isDollhouseBin || isMcpServerBin;
|
|
78
|
+
}
|
|
79
|
+
function parsePidToken(value) {
|
|
80
|
+
if (!value)
|
|
81
|
+
return null;
|
|
82
|
+
for (let i = 0; i < value.length; i++) {
|
|
83
|
+
const codePoint = value.codePointAt(i);
|
|
84
|
+
if (codePoint === undefined || codePoint < 48 || codePoint > 57) {
|
|
85
|
+
return null;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
const parsed = Number.parseInt(value, 10);
|
|
89
|
+
if (!Number.isSafeInteger(parsed) || parsed < ROOT_PARENT_PID) {
|
|
90
|
+
return null;
|
|
91
|
+
}
|
|
92
|
+
return parsed;
|
|
93
|
+
}
|
|
94
|
+
function isWhitespaceChar(value) {
|
|
95
|
+
return value === ' ' || value === '\t' || value === '\n' || value === '\r' || value === '\f' || value === '\v';
|
|
96
|
+
}
|
|
97
|
+
function buildKillOutcome(killed, reason, processInfo, parentCommand, detail) {
|
|
98
|
+
return {
|
|
99
|
+
killed,
|
|
100
|
+
reason,
|
|
101
|
+
pid: processInfo.pid,
|
|
102
|
+
parentPid: processInfo.parentPid,
|
|
103
|
+
command: processInfo.command,
|
|
104
|
+
parentCommand,
|
|
105
|
+
...(detail ? { detail } : {}),
|
|
106
|
+
};
|
|
107
|
+
}
|
|
108
|
+
async function getKillGuardFailure(processInfo, port) {
|
|
109
|
+
const currentUser = (await import('node:os')).userInfo().username;
|
|
110
|
+
if (processInfo.user !== currentUser) {
|
|
111
|
+
await logger.warn(`[WebUI] Port ${port} held by different user (pid ${processInfo.pid}) — not killing`);
|
|
112
|
+
return buildKillOutcome(false, 'different_user', processInfo);
|
|
113
|
+
}
|
|
114
|
+
if (!isDollhouseProcessCommand(processInfo.command)) {
|
|
115
|
+
await logger.warn(`[WebUI] Port ${port} held by non-DollhouseMCP process (pid ${processInfo.pid}) — not killing`, {
|
|
116
|
+
cmdLine: processInfo.command,
|
|
117
|
+
});
|
|
118
|
+
return buildKillOutcome(false, 'not_dollhouse_process', processInfo);
|
|
119
|
+
}
|
|
120
|
+
if (processInfo.parentPid <= ROOT_PARENT_PID || !isPidAlive(processInfo.parentPid)) {
|
|
121
|
+
return null;
|
|
122
|
+
}
|
|
123
|
+
const parentCommand = (await getProcessCommand(processInfo.parentPid)) ?? undefined;
|
|
124
|
+
if (parentCommand && isRecognizedMcpHostParent(parentCommand)) {
|
|
125
|
+
await logger.warn(`[WebUI] Port ${port} held by active client-backed DollhouseMCP process (pid ${processInfo.pid}) — not killing`, {
|
|
126
|
+
cmdLine: processInfo.command,
|
|
127
|
+
parentPid: processInfo.parentPid,
|
|
128
|
+
parentCommand,
|
|
129
|
+
});
|
|
130
|
+
return buildKillOutcome(false, 'active_host_parent', processInfo, parentCommand);
|
|
131
|
+
}
|
|
132
|
+
return null;
|
|
133
|
+
}
|
|
134
|
+
async function terminateProcess(processInfo, port, parentCommand) {
|
|
135
|
+
process.kill(processInfo.pid, 'SIGTERM');
|
|
136
|
+
logger.warn(`[WebUI] Sent SIGTERM to stale process ${processInfo.pid} on port ${port}`, {
|
|
137
|
+
cmdLine: processInfo.command,
|
|
138
|
+
parentPid: processInfo.parentPid,
|
|
139
|
+
parentCommand,
|
|
140
|
+
});
|
|
141
|
+
for (let i = 0; i < KILL_POLL_COUNT; i++) {
|
|
142
|
+
await new Promise(r => setTimeout(r, SIGTERM_POLL_MS));
|
|
143
|
+
if (!isPidAlive(processInfo.pid)) {
|
|
144
|
+
return buildKillOutcome(true, 'terminated', processInfo, parentCommand);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
process.kill(processInfo.pid, 'SIGKILL');
|
|
148
|
+
logger.warn(`[WebUI] Sent SIGKILL to stale process ${processInfo.pid} on port ${port}`);
|
|
149
|
+
await new Promise(r => setTimeout(r, SIGKILL_WAIT_MS));
|
|
150
|
+
return isPidAlive(processInfo.pid)
|
|
151
|
+
? buildKillOutcome(false, 'still_alive', processInfo, parentCommand)
|
|
152
|
+
: buildKillOutcome(true, 'terminated', processInfo, parentCommand);
|
|
153
|
+
}
|
|
154
|
+
function splitProcessInspectionFields(line) {
|
|
155
|
+
const fields = [];
|
|
156
|
+
let index = 0;
|
|
157
|
+
const normalizedLine = UnicodeValidator.normalize(line).normalizedContent.trim();
|
|
158
|
+
while (index < normalizedLine.length && fields.length < PROCESS_INSPECTION_FIELD_COUNT - 1) {
|
|
159
|
+
while (index < normalizedLine.length && isWhitespaceChar(normalizedLine[index])) {
|
|
160
|
+
index++;
|
|
161
|
+
}
|
|
162
|
+
if (index >= normalizedLine.length)
|
|
163
|
+
break;
|
|
164
|
+
const fieldStart = index;
|
|
165
|
+
while (index < normalizedLine.length && !isWhitespaceChar(normalizedLine[index])) {
|
|
166
|
+
index++;
|
|
167
|
+
}
|
|
168
|
+
fields.push(normalizedLine.slice(fieldStart, index));
|
|
169
|
+
}
|
|
170
|
+
while (index < normalizedLine.length && isWhitespaceChar(normalizedLine[index])) {
|
|
171
|
+
index++;
|
|
172
|
+
}
|
|
173
|
+
if (index < normalizedLine.length) {
|
|
174
|
+
fields.push(normalizedLine.slice(index));
|
|
175
|
+
}
|
|
176
|
+
return fields.length === PROCESS_INSPECTION_FIELD_COUNT ? fields : null;
|
|
177
|
+
}
|
|
178
|
+
async function inspectProcess(pid) {
|
|
179
|
+
const { execFile: execFileCb } = await import('node:child_process');
|
|
180
|
+
const { promisify } = await import('node:util');
|
|
181
|
+
const execFileAsync = promisify(execFileCb);
|
|
182
|
+
try {
|
|
183
|
+
const { stdout } = await execFileAsync('ps', ['-p', String(pid), '-o', 'user=,pid=,ppid=,command='], { timeout: COMMAND_TIMEOUT_MS });
|
|
184
|
+
const fields = splitProcessInspectionFields(stdout);
|
|
185
|
+
if (!fields)
|
|
186
|
+
return null;
|
|
187
|
+
const [user, pidToken, parentPidToken, command] = fields;
|
|
188
|
+
const parsedPid = parsePidToken(pidToken);
|
|
189
|
+
const parsedParentPid = parsePidToken(parentPidToken);
|
|
190
|
+
if (parsedPid === null || parsedParentPid === null)
|
|
191
|
+
return null;
|
|
192
|
+
return {
|
|
193
|
+
user: UnicodeValidator.normalize(user).normalizedContent,
|
|
194
|
+
pid: parsedPid,
|
|
195
|
+
parentPid: parsedParentPid,
|
|
196
|
+
command: UnicodeValidator.normalize(command).normalizedContent,
|
|
197
|
+
};
|
|
198
|
+
}
|
|
199
|
+
catch {
|
|
200
|
+
return null;
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
async function getProcessCommand(pid) {
|
|
204
|
+
const { execFile: execFileCb } = await import('node:child_process');
|
|
205
|
+
const { promisify } = await import('node:util');
|
|
206
|
+
const execFileAsync = promisify(execFileCb);
|
|
207
|
+
try {
|
|
208
|
+
const { stdout } = await execFileAsync('ps', ['-p', String(pid), '-o', 'command='], { timeout: COMMAND_TIMEOUT_MS });
|
|
209
|
+
const normalized = UnicodeValidator.normalize(stdout).normalizedContent.trim();
|
|
210
|
+
return normalized || null;
|
|
211
|
+
}
|
|
212
|
+
catch {
|
|
213
|
+
return null;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
function isPidAlive(pid) {
|
|
217
|
+
try {
|
|
218
|
+
process.kill(pid, 0);
|
|
219
|
+
return true;
|
|
220
|
+
}
|
|
221
|
+
catch {
|
|
222
|
+
return false;
|
|
223
|
+
}
|
|
224
|
+
}
|
|
38
225
|
/**
|
|
39
226
|
* Find the PID of the process listening on a given port.
|
|
40
227
|
* Uses lsof on macOS/Linux. Returns null if not found or on error.
|
|
@@ -56,7 +243,10 @@ export async function findPidOnPort(port) {
|
|
|
56
243
|
const { stdout, stderr } = await execFileAsync(cmd.bin, cmd.args, { timeout: COMMAND_TIMEOUT_MS });
|
|
57
244
|
// fuser outputs to stderr on some systems
|
|
58
245
|
const output = (stdout || stderr || '').trim();
|
|
59
|
-
const pids = output
|
|
246
|
+
const pids = output
|
|
247
|
+
.split(/\s+/)
|
|
248
|
+
.map((token) => parsePidToken(token))
|
|
249
|
+
.filter((pid) => pid !== null);
|
|
60
250
|
const otherPid = pids.find(p => p !== process.pid);
|
|
61
251
|
if (otherPid)
|
|
62
252
|
return otherPid;
|
|
@@ -76,63 +266,31 @@ export async function findPidOnPort(port) {
|
|
|
76
266
|
* before escalating to SIGKILL. Total worst case: ~4s.
|
|
77
267
|
*/
|
|
78
268
|
export async function killStaleProcess(pid, port) {
|
|
79
|
-
const
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
try {
|
|
88
|
-
const { stdout } = await execFileAsync('ps', ['-p', String(pid), '-o', 'user=,command='], { timeout: COMMAND_TIMEOUT_MS });
|
|
89
|
-
// Check 1: User ownership — only kill our own processes
|
|
90
|
-
const currentUser = (await import('node:os')).userInfo().username;
|
|
91
|
-
if (!stdout.trim().startsWith(currentUser)) {
|
|
92
|
-
await logger.warn(`[WebUI] Port ${port} held by different user (pid ${pid}) — not killing`);
|
|
93
|
-
return false;
|
|
94
|
-
}
|
|
95
|
-
// Check 2: Binary identity — must be .bin/mcp-server, .bin/dollhousemcp,
|
|
96
|
-
// /bin/dollhousemcp (global install), or dist/index.js (direct node execution).
|
|
97
|
-
// NOT just 'mcp-server' anywhere in the path — that would match Jest workers
|
|
98
|
-
// running from within the mcp-server project directory.
|
|
99
|
-
cmdLine = stdout.trim();
|
|
100
|
-
const isDollhouseBin = /(?:^|\/)dollhousemcp(?:\s|$)/.test(cmdLine) ||
|
|
101
|
-
cmdLine.includes('.bin/dollhousemcp');
|
|
102
|
-
const isMcpServerBin = cmdLine.includes('.bin/mcp-server') ||
|
|
103
|
-
/(?:dollhousemcp|mcp-server)[/\\]dist[/\\]index\.js/.test(cmdLine);
|
|
104
|
-
if (!isDollhouseBin && !isMcpServerBin) {
|
|
105
|
-
await logger.warn(`[WebUI] Port ${port} held by non-DollhouseMCP process (pid ${pid}) — not killing`, { cmdLine });
|
|
106
|
-
return false;
|
|
107
|
-
}
|
|
108
|
-
await logger.debug(`[WebUI] Verified stale process ${pid} is DollhouseMCP`, { cmdLine });
|
|
269
|
+
const outcome = await killStaleProcessDetailed(pid, port);
|
|
270
|
+
return outcome.killed;
|
|
271
|
+
}
|
|
272
|
+
export async function killStaleProcessDetailed(pid, port) {
|
|
273
|
+
const processInfo = await inspectProcess(pid);
|
|
274
|
+
if (!processInfo) {
|
|
275
|
+
await logger.debug(`[WebUI] Cannot verify process ${pid} — skipping kill`);
|
|
276
|
+
return { killed: false, reason: 'inspect_failed', pid };
|
|
109
277
|
}
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
error: err instanceof Error ? err.message : String(err),
|
|
114
|
-
});
|
|
115
|
-
return false;
|
|
278
|
+
const guardFailure = await getKillGuardFailure(processInfo, port);
|
|
279
|
+
if (guardFailure) {
|
|
280
|
+
return guardFailure;
|
|
116
281
|
}
|
|
282
|
+
const parentCommand = processInfo.parentPid > ROOT_PARENT_PID
|
|
283
|
+
? (await getProcessCommand(processInfo.parentPid)) ?? undefined
|
|
284
|
+
: undefined;
|
|
285
|
+
await logger.debug(`[WebUI] Verified stale process ${pid} is DollhouseMCP`, { cmdLine: processInfo.command, parentPid: processInfo.parentPid, parentCommand });
|
|
117
286
|
try {
|
|
118
|
-
|
|
119
|
-
logger.warn(`[WebUI] Sent SIGTERM to stale process ${pid} on port ${port}`, { cmdLine });
|
|
120
|
-
for (let i = 0; i < KILL_POLL_COUNT; i++) {
|
|
121
|
-
await new Promise(r => setTimeout(r, SIGTERM_POLL_MS));
|
|
122
|
-
try {
|
|
123
|
-
process.kill(pid, 0);
|
|
124
|
-
}
|
|
125
|
-
catch {
|
|
126
|
-
return true;
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
process.kill(pid, 'SIGKILL');
|
|
130
|
-
logger.warn(`[WebUI] Sent SIGKILL to stale process ${pid} on port ${port}`);
|
|
131
|
-
await new Promise(r => setTimeout(r, SIGKILL_WAIT_MS));
|
|
132
|
-
return true;
|
|
287
|
+
return await terminateProcess(processInfo, port, parentCommand);
|
|
133
288
|
}
|
|
134
|
-
catch {
|
|
135
|
-
|
|
289
|
+
catch (err) {
|
|
290
|
+
if (!isPidAlive(pid)) {
|
|
291
|
+
return buildKillOutcome(true, 'already_dead', processInfo, parentCommand);
|
|
292
|
+
}
|
|
293
|
+
return buildKillOutcome(false, 'signal_failed', processInfo, parentCommand, err instanceof Error ? err.message : String(err));
|
|
136
294
|
}
|
|
137
295
|
}
|
|
138
296
|
/**
|
|
@@ -150,7 +308,7 @@ export async function recoverStalePort(port) {
|
|
|
150
308
|
// written its lock file. Read the lock, pause, re-read. If the second read
|
|
151
309
|
// now matches the port holder, it's a fresh leader — don't kill.
|
|
152
310
|
const { readLeaderLock } = await import('./LeaderElection.js');
|
|
153
|
-
for (let check = 0; check <
|
|
311
|
+
for (let check = 0; check < LOCK_RECHECK_ATTEMPTS; check++) {
|
|
154
312
|
try {
|
|
155
313
|
const lock = await readLeaderLock();
|
|
156
314
|
if (lock?.pid === stalePid && lock?.port === port && lock.pid !== process.pid) {
|
|
@@ -161,15 +319,23 @@ export async function recoverStalePort(port) {
|
|
|
161
319
|
catch {
|
|
162
320
|
// Can't read lock file — continue to next check or kill
|
|
163
321
|
}
|
|
164
|
-
if (check
|
|
322
|
+
if (check < LOCK_RECHECK_ATTEMPTS - 1) {
|
|
165
323
|
await new Promise(r => setTimeout(r, LOCK_RECHECK_DELAY_MS));
|
|
166
324
|
}
|
|
167
325
|
}
|
|
168
|
-
const
|
|
169
|
-
if (killed) {
|
|
326
|
+
const outcome = await killStaleProcessDetailed(stalePid, port);
|
|
327
|
+
if (outcome.killed) {
|
|
170
328
|
logger.info(`[WebUI] Stale process ${stalePid} removed from port ${port}`);
|
|
171
329
|
await new Promise(r => setTimeout(r, SIGKILL_WAIT_MS)); // brief pause for port release
|
|
172
330
|
}
|
|
173
|
-
|
|
331
|
+
else {
|
|
332
|
+
await logger.debug(`[WebUI] Stale-port recovery skipped for pid ${stalePid}`, {
|
|
333
|
+
reason: outcome.reason,
|
|
334
|
+
parentPid: outcome.parentPid,
|
|
335
|
+
parentCommand: outcome.parentCommand,
|
|
336
|
+
detail: outcome.detail,
|
|
337
|
+
});
|
|
338
|
+
}
|
|
339
|
+
return outcome.killed;
|
|
174
340
|
}
|
|
175
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"StaleProcessRecovery.js","sourceRoot":"","sources":["../../../src/web/console/StaleProcessRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Stale process detection and recovery (#1850).\n *\n * Finds and kills zombie DollhouseMCP processes that squat on the console\n * port after their session has ended. Used by bindAndListen in server.ts\n * when EADDRINUSE occurs.\n *\n * Extracted to a standalone module so it can be tested without importing\n * the full Express server and its dependency chain.\n */\n\n// Use lazy import for logger to avoid pulling in the full env.ts/config chain\n// at module load time. This keeps the module independently testable.\n/** Timeout for lsof/fuser/ps system calls (ms) */\nconst COMMAND_TIMEOUT_MS = 1000;\n/** Polling interval when waiting for SIGTERM to take effect (ms) */\nconst SIGTERM_POLL_MS = 300;\n/** Number of polls before escalating to SIGKILL */\nconst KILL_POLL_COUNT = 10;\n/** Wait after SIGKILL before returning (ms) */\nconst SIGKILL_WAIT_MS = 500;\n/** Wait between lock file reads for TOCTOU mitigation (ms) */\nconst LOCK_RECHECK_DELAY_MS = 500;\n\nlet _logger: typeof import('../../utils/logger.js').logger | null = null;\nasync function getLogger() {\n  if (!_logger) {\n    try { _logger = (await import('../../utils/logger.js')).logger; }\n    catch { /* fallback below */ }\n  }\n  return _logger;\n}\nconst logger = {\n  warn: async (...args: unknown[]) => { const l = await getLogger(); l ? l.warn(args[0] as string, args[1]) : console.error('[WARN]', ...args); },\n  info: async (...args: unknown[]) => { const l = await getLogger(); l ? l.info(args[0] as string, args[1]) : console.error('[INFO]', ...args); },\n  debug: async (...args: unknown[]) => { const l = await getLogger(); l ? l.debug(args[0] as string, args[1]) : void 0; },\n};\n\n/**\n * Find the PID of the process listening on a given port.\n * Uses lsof on macOS/Linux. Returns null if not found or on error.\n *\n * Timeout: 1s — lsof on localhost is typically <100ms. The 1s ceiling\n * handles slow NFS-mounted /dev/fd or overloaded CI runners without\n * delaying startup noticeably.\n */\nexport async function findPidOnPort(port: number): Promise<number | null> {\n  const { execFile: execFileCb } = await import('node:child_process');\n  const { promisify } = await import('node:util');\n  const execFileAsync = promisify(execFileCb);\n\n  // Try lsof first (macOS + most Linux), fall back to fuser (minimal Linux/Docker)\n  for (const cmd of [\n    { bin: 'lsof', args: ['-ti', `:${port}`] },\n    { bin: 'fuser', args: [`${port}/tcp`] },\n  ]) {\n    try {\n      const { stdout, stderr } = await execFileAsync(cmd.bin, cmd.args, { timeout: COMMAND_TIMEOUT_MS });\n      // fuser outputs to stderr on some systems\n      const output = (stdout || stderr || '').trim();\n      const pids = output.split(/\\s+/).map(Number).filter(n => !Number.isNaN(n) && n > 0);\n      const otherPid = pids.find(p => p !== process.pid);\n      if (otherPid) return otherPid;\n    } catch {\n      continue; // command not found or no results — try next\n    }\n  }\n  return null;\n}\n\n/**\n * Kill a stale process holding a port. Sends SIGTERM, waits briefly,\n * then SIGKILL if still alive. Only kills DollhouseMCP processes\n * (verified by checking the command line and user ownership).\n *\n * Timeout: 1s for ps verification. Kill wait: 300ms × 10 polls = 3s\n * before escalating to SIGKILL. Total worst case: ~4s.\n */\nexport async function killStaleProcess(pid: number, port: number): Promise<boolean> {\n  const { execFile: execFileCb } = await import('node:child_process');\n  const { promisify } = await import('node:util');\n  const execFileAsync = promisify(execFileCb);\n\n  // Security verification flow — three checks must pass before we kill:\n  // 1. Process must be owned by the current OS user (prevents cross-user kills)\n  // 2. Command line must match a DollhouseMCP binary path (prevents killing other services)\n  // 3. If both fail or ps can't run, we refuse — safe default is to not kill\n  let cmdLine = '';\n  try {\n    const { stdout } = await execFileAsync('ps', ['-p', String(pid), '-o', 'user=,command='], { timeout: COMMAND_TIMEOUT_MS });\n\n    // Check 1: User ownership — only kill our own processes\n    const currentUser = (await import('node:os')).userInfo().username;\n    if (!stdout.trim().startsWith(currentUser)) {\n      await logger.warn(`[WebUI] Port ${port} held by different user (pid ${pid}) — not killing`);\n      return false;\n    }\n\n    // Check 2: Binary identity — must be .bin/mcp-server, .bin/dollhousemcp,\n    // /bin/dollhousemcp (global install), or dist/index.js (direct node execution).\n    // NOT just 'mcp-server' anywhere in the path — that would match Jest workers\n    // running from within the mcp-server project directory.\n    cmdLine = stdout.trim();\n    const isDollhouseBin = /(?:^|\\/)dollhousemcp(?:\\s|$)/.test(cmdLine) ||\n      cmdLine.includes('.bin/dollhousemcp');\n    const isMcpServerBin = cmdLine.includes('.bin/mcp-server') ||\n      /(?:dollhousemcp|mcp-server)[/\\\\]dist[/\\\\]index\\.js/.test(cmdLine);\n    if (!isDollhouseBin && !isMcpServerBin) {\n      await logger.warn(`[WebUI] Port ${port} held by non-DollhouseMCP process (pid ${pid}) — not killing`, { cmdLine });\n      return false;\n    }\n    await logger.debug(`[WebUI] Verified stale process ${pid} is DollhouseMCP`, { cmdLine });\n  } catch (err) {\n    // Check 3: If we can't verify, don't kill — safe default\n    await logger.debug(`[WebUI] Cannot verify process ${pid} — skipping kill`, {\n      error: err instanceof Error ? err.message : String(err),\n    });\n    return false;\n  }\n\n  try {\n    process.kill(pid, 'SIGTERM');\n    logger.warn(`[WebUI] Sent SIGTERM to stale process ${pid} on port ${port}`, { cmdLine });\n    for (let i = 0; i < KILL_POLL_COUNT; i++) {\n      await new Promise(r => setTimeout(r, SIGTERM_POLL_MS));\n      try { process.kill(pid, 0); } catch { return true; }\n    }\n    process.kill(pid, 'SIGKILL');\n    logger.warn(`[WebUI] Sent SIGKILL to stale process ${pid} on port ${port}`);\n    await new Promise(r => setTimeout(r, SIGKILL_WAIT_MS));\n    return true;\n  } catch {\n    return true; // process already dead\n  }\n}\n\n/**\n * Detect and recover from a stale process squatting on the port.\n * Compares the port holder's PID against the leader lock file to determine\n * if it's a squatter. Returns true if the squatter was killed.\n *\n * Timeouts: lsof 1s, ps 1s, SIGTERM wait 3s — max ~5s total.\n */\nexport async function recoverStalePort(port: number): Promise<boolean> {\n  const stalePid = await findPidOnPort(port);\n  if (!stalePid) return false;\n\n  // TOCTOU mitigation: a new process may have just bound the port but not yet\n  // written its lock file. Read the lock, pause, re-read. If the second read\n  // now matches the port holder, it's a fresh leader — don't kill.\n  const { readLeaderLock } = await import('./LeaderElection.js');\n  for (let check = 0; check < 2; check++) {\n    try {\n      const lock = await readLeaderLock();\n      if (lock?.pid === stalePid && lock?.port === port && lock.pid !== process.pid) {\n        await logger.warn(`[WebUI] Port ${port} held by legitimate leader (pid ${stalePid}) — not killing`);\n        return false;\n      }\n    } catch {\n      // Can't read lock file — continue to next check or kill\n    }\n    if (check === 0) {\n      await new Promise(r => setTimeout(r, LOCK_RECHECK_DELAY_MS));\n    }\n  }\n\n  const killed = await killStaleProcess(stalePid, port);\n  if (killed) {\n    logger.info(`[WebUI] Stale process ${stalePid} removed from port ${port}`);\n    await new Promise(r => setTimeout(r, SIGKILL_WAIT_MS)); // brief pause for port release\n  }\n  return killed;\n}\n"]}
|
|
341
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"StaleProcessRecovery.js","sourceRoot":"","sources":["../../../src/web/console/StaleProcessRecovery.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Stale process detection and recovery (#1850).\n *\n * Finds and kills zombie DollhouseMCP processes that squat on the console\n * port after their session has ended. Used by bindAndListen in server.ts\n * when EADDRINUSE occurs.\n *\n * Extracted to a standalone module so it can be tested without importing\n * the full Express server and its dependency chain.\n */\n\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\n\n// Use lazy import for logger to avoid pulling in the full env.ts/config chain\n// at module load time. This keeps the module independently testable.\n/** Timeout for lsof/fuser/ps system calls (ms) */\nconst COMMAND_TIMEOUT_MS = 1000;\n/** Polling interval when waiting for SIGTERM to take effect (ms) */\nconst SIGTERM_POLL_MS = 300;\n/** Number of polls before escalating to SIGKILL */\nconst KILL_POLL_COUNT = 10;\n/** Wait after SIGKILL before returning (ms) */\nconst SIGKILL_WAIT_MS = 500;\n/** Wait between lock file reads for TOCTOU mitigation (ms) */\nconst LOCK_RECHECK_DELAY_MS = 500;\n/** Number of lock-file checks before deciding the port holder is not a fresh leader. */\nconst LOCK_RECHECK_ATTEMPTS = 2;\n/** PID used by the OS init/launchd process; direct children are effectively orphaned. */\nconst ROOT_PARENT_PID = 1;\n/** Number of `ps` columns requested by inspectProcess: user, pid, ppid, command. */\nconst PROCESS_INSPECTION_FIELD_COUNT = 4;\n\nlet _logger: typeof import('../../utils/logger.js').logger | null = null;\nasync function getLogger() {\n  if (!_logger) {\n    try { _logger = (await import('../../utils/logger.js')).logger; }\n    catch { /* fallback below */ }\n  }\n  return _logger;\n}\nconst logger = {\n  warn: async (...args: unknown[]) => {\n    const l = await getLogger();\n    if (l) l.warn(args[0] as string, args[1]);\n    else console.error('[WARN]', ...args);\n  },\n  info: async (...args: unknown[]) => {\n    const l = await getLogger();\n    if (l) l.info(args[0] as string, args[1]);\n    else console.error('[INFO]', ...args);\n  },\n  debug: async (...args: unknown[]) => {\n    const l = await getLogger();\n    if (l) l.debug(args[0] as string, args[1]);\n  },\n};\n\nconst MCP_HOST_PARENT_PATTERNS = [\n  /Claude\\.app\\/Contents\\/Helpers\\/disclaimer/i,\n  /Codex\\.app\\/Contents\\/Resources\\/codex app-server/i,\n  /Cursor\\.app\\//i,\n  /Windsurf\\.app\\//i,\n];\n\ninterface ProcessInspection {\n  user: string;\n  pid: number;\n  parentPid: number;\n  command: string;\n}\n\nexport interface KillStaleProcessOutcome {\n  killed: boolean;\n  reason:\n    | 'inspect_failed'\n    | 'different_user'\n    | 'not_dollhouse_process'\n    | 'active_host_parent'\n    | 'terminated'\n    | 'already_dead'\n    | 'still_alive'\n    | 'signal_failed';\n  pid: number;\n  parentPid?: number;\n  command?: string;\n  parentCommand?: string;\n  detail?: string;\n}\n\nexport function isRecognizedMcpHostParent(command: string): boolean {\n  const normalizedCommand = UnicodeValidator.normalize(command).normalizedContent;\n  return MCP_HOST_PARENT_PATTERNS.some((pattern) => pattern.test(normalizedCommand));\n}\n\nfunction isDollhouseProcessCommand(cmdLine: string): boolean {\n  const normalizedCommand = UnicodeValidator.normalize(cmdLine).normalizedContent;\n  const isDollhouseBin = /(?:^|\\/)dollhousemcp(?:\\s|$)/.test(normalizedCommand) ||\n    normalizedCommand.includes('.bin/dollhousemcp');\n  const isMcpServerBin = normalizedCommand.includes('.bin/mcp-server') ||\n    /(?:dollhousemcp|mcp-server)[/\\\\]dist[/\\\\]index\\.js/.test(normalizedCommand);\n  return isDollhouseBin || isMcpServerBin;\n}\n\nfunction parsePidToken(value: string): number | null {\n  if (!value) return null;\n  for (let i = 0; i < value.length; i++) {\n    const codePoint = value.codePointAt(i);\n    if (codePoint === undefined || codePoint < 48 || codePoint > 57) {\n      return null;\n    }\n  }\n\n  const parsed = Number.parseInt(value, 10);\n  if (!Number.isSafeInteger(parsed) || parsed < ROOT_PARENT_PID) {\n    return null;\n  }\n  return parsed;\n}\n\nfunction isWhitespaceChar(value: string): boolean {\n  return value === ' ' || value === '\\t' || value === '\\n' || value === '\\r' || value === '\\f' || value === '\\v';\n}\n\nfunction buildKillOutcome(\n  killed: boolean,\n  reason: KillStaleProcessOutcome['reason'],\n  processInfo: ProcessInspection,\n  parentCommand?: string,\n  detail?: string,\n): KillStaleProcessOutcome {\n  return {\n    killed,\n    reason,\n    pid: processInfo.pid,\n    parentPid: processInfo.parentPid,\n    command: processInfo.command,\n    parentCommand,\n    ...(detail ? { detail } : {}),\n  };\n}\n\nasync function getKillGuardFailure(\n  processInfo: ProcessInspection,\n  port: number,\n): Promise<KillStaleProcessOutcome | null> {\n  const currentUser = (await import('node:os')).userInfo().username;\n  if (processInfo.user !== currentUser) {\n    await logger.warn(`[WebUI] Port ${port} held by different user (pid ${processInfo.pid}) — not killing`);\n    return buildKillOutcome(false, 'different_user', processInfo);\n  }\n\n  if (!isDollhouseProcessCommand(processInfo.command)) {\n    await logger.warn(`[WebUI] Port ${port} held by non-DollhouseMCP process (pid ${processInfo.pid}) — not killing`, {\n      cmdLine: processInfo.command,\n    });\n    return buildKillOutcome(false, 'not_dollhouse_process', processInfo);\n  }\n\n  if (processInfo.parentPid <= ROOT_PARENT_PID || !isPidAlive(processInfo.parentPid)) {\n    return null;\n  }\n\n  const parentCommand = (await getProcessCommand(processInfo.parentPid)) ?? undefined;\n  if (parentCommand && isRecognizedMcpHostParent(parentCommand)) {\n    await logger.warn(`[WebUI] Port ${port} held by active client-backed DollhouseMCP process (pid ${processInfo.pid}) — not killing`, {\n      cmdLine: processInfo.command,\n      parentPid: processInfo.parentPid,\n      parentCommand,\n    });\n    return buildKillOutcome(false, 'active_host_parent', processInfo, parentCommand);\n  }\n\n  return null;\n}\n\nasync function terminateProcess(\n  processInfo: ProcessInspection,\n  port: number,\n  parentCommand?: string,\n): Promise<KillStaleProcessOutcome> {\n  process.kill(processInfo.pid, 'SIGTERM');\n  logger.warn(`[WebUI] Sent SIGTERM to stale process ${processInfo.pid} on port ${port}`, {\n    cmdLine: processInfo.command,\n    parentPid: processInfo.parentPid,\n    parentCommand,\n  });\n\n  for (let i = 0; i < KILL_POLL_COUNT; i++) {\n    await new Promise(r => setTimeout(r, SIGTERM_POLL_MS));\n    if (!isPidAlive(processInfo.pid)) {\n      return buildKillOutcome(true, 'terminated', processInfo, parentCommand);\n    }\n  }\n\n  process.kill(processInfo.pid, 'SIGKILL');\n  logger.warn(`[WebUI] Sent SIGKILL to stale process ${processInfo.pid} on port ${port}`);\n  await new Promise(r => setTimeout(r, SIGKILL_WAIT_MS));\n  return isPidAlive(processInfo.pid)\n    ? buildKillOutcome(false, 'still_alive', processInfo, parentCommand)\n    : buildKillOutcome(true, 'terminated', processInfo, parentCommand);\n}\n\nfunction splitProcessInspectionFields(line: string): string[] | null {\n  const fields: string[] = [];\n  let index = 0;\n  const normalizedLine = UnicodeValidator.normalize(line).normalizedContent.trim();\n\n  while (index < normalizedLine.length && fields.length < PROCESS_INSPECTION_FIELD_COUNT - 1) {\n    while (index < normalizedLine.length && isWhitespaceChar(normalizedLine[index])) {\n      index++;\n    }\n    if (index >= normalizedLine.length) break;\n\n    const fieldStart = index;\n    while (index < normalizedLine.length && !isWhitespaceChar(normalizedLine[index])) {\n      index++;\n    }\n    fields.push(normalizedLine.slice(fieldStart, index));\n  }\n\n  while (index < normalizedLine.length && isWhitespaceChar(normalizedLine[index])) {\n    index++;\n  }\n  if (index < normalizedLine.length) {\n    fields.push(normalizedLine.slice(index));\n  }\n\n  return fields.length === PROCESS_INSPECTION_FIELD_COUNT ? fields : null;\n}\n\nasync function inspectProcess(pid: number): Promise<ProcessInspection | null> {\n  const { execFile: execFileCb } = await import('node:child_process');\n  const { promisify } = await import('node:util');\n  const execFileAsync = promisify(execFileCb);\n\n  try {\n    const { stdout } = await execFileAsync(\n      'ps',\n      ['-p', String(pid), '-o', 'user=,pid=,ppid=,command='],\n      { timeout: COMMAND_TIMEOUT_MS },\n    );\n    const fields = splitProcessInspectionFields(stdout);\n    if (!fields) return null;\n    const [user, pidToken, parentPidToken, command] = fields;\n    const parsedPid = parsePidToken(pidToken);\n    const parsedParentPid = parsePidToken(parentPidToken);\n    if (parsedPid === null || parsedParentPid === null) return null;\n\n    return {\n      user: UnicodeValidator.normalize(user).normalizedContent,\n      pid: parsedPid,\n      parentPid: parsedParentPid,\n      command: UnicodeValidator.normalize(command).normalizedContent,\n    };\n  } catch {\n    return null;\n  }\n}\n\nasync function getProcessCommand(pid: number): Promise<string | null> {\n  const { execFile: execFileCb } = await import('node:child_process');\n  const { promisify } = await import('node:util');\n  const execFileAsync = promisify(execFileCb);\n\n  try {\n    const { stdout } = await execFileAsync('ps', ['-p', String(pid), '-o', 'command='], { timeout: COMMAND_TIMEOUT_MS });\n    const normalized = UnicodeValidator.normalize(stdout).normalizedContent.trim();\n    return normalized || null;\n  } catch {\n    return null;\n  }\n}\n\nfunction isPidAlive(pid: number): boolean {\n  try {\n    process.kill(pid, 0);\n    return true;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Find the PID of the process listening on a given port.\n * Uses lsof on macOS/Linux. Returns null if not found or on error.\n *\n * Timeout: 1s — lsof on localhost is typically <100ms. The 1s ceiling\n * handles slow NFS-mounted /dev/fd or overloaded CI runners without\n * delaying startup noticeably.\n */\nexport async function findPidOnPort(port: number): Promise<number | null> {\n  const { execFile: execFileCb } = await import('node:child_process');\n  const { promisify } = await import('node:util');\n  const execFileAsync = promisify(execFileCb);\n\n  // Try lsof first (macOS + most Linux), fall back to fuser (minimal Linux/Docker)\n  for (const cmd of [\n    { bin: 'lsof', args: ['-ti', `:${port}`] },\n    { bin: 'fuser', args: [`${port}/tcp`] },\n  ]) {\n    try {\n      const { stdout, stderr } = await execFileAsync(cmd.bin, cmd.args, { timeout: COMMAND_TIMEOUT_MS });\n      // fuser outputs to stderr on some systems\n      const output = (stdout || stderr || '').trim();\n      const pids = output\n        .split(/\\s+/)\n        .map((token) => parsePidToken(token))\n        .filter((pid): pid is number => pid !== null);\n      const otherPid = pids.find(p => p !== process.pid);\n      if (otherPid) return otherPid;\n    } catch {\n      continue; // command not found or no results — try next\n    }\n  }\n  return null;\n}\n\n/**\n * Kill a stale process holding a port. Sends SIGTERM, waits briefly,\n * then SIGKILL if still alive. Only kills DollhouseMCP processes\n * (verified by checking the command line and user ownership).\n *\n * Timeout: 1s for ps verification. Kill wait: 300ms × 10 polls = 3s\n * before escalating to SIGKILL. Total worst case: ~4s.\n */\nexport async function killStaleProcess(pid: number, port: number): Promise<boolean> {\n  const outcome = await killStaleProcessDetailed(pid, port);\n  return outcome.killed;\n}\n\nexport async function killStaleProcessDetailed(pid: number, port: number): Promise<KillStaleProcessOutcome> {\n  const processInfo = await inspectProcess(pid);\n  if (!processInfo) {\n    await logger.debug(`[WebUI] Cannot verify process ${pid} — skipping kill`);\n    return { killed: false, reason: 'inspect_failed', pid };\n  }\n\n  const guardFailure = await getKillGuardFailure(processInfo, port);\n  if (guardFailure) {\n    return guardFailure;\n  }\n  const parentCommand = processInfo.parentPid > ROOT_PARENT_PID\n    ? (await getProcessCommand(processInfo.parentPid)) ?? undefined\n    : undefined;\n\n  await logger.debug(`[WebUI] Verified stale process ${pid} is DollhouseMCP`, { cmdLine: processInfo.command, parentPid: processInfo.parentPid, parentCommand });\n\n  try {\n    return await terminateProcess(processInfo, port, parentCommand);\n  } catch (err) {\n    if (!isPidAlive(pid)) {\n      return buildKillOutcome(true, 'already_dead', processInfo, parentCommand);\n    }\n    return buildKillOutcome(false, 'signal_failed', processInfo, parentCommand, err instanceof Error ? err.message : String(err));\n  }\n}\n\n/**\n * Detect and recover from a stale process squatting on the port.\n * Compares the port holder's PID against the leader lock file to determine\n * if it's a squatter. Returns true if the squatter was killed.\n *\n * Timeouts: lsof 1s, ps 1s, SIGTERM wait 3s — max ~5s total.\n */\nexport async function recoverStalePort(port: number): Promise<boolean> {\n  const stalePid = await findPidOnPort(port);\n  if (!stalePid) return false;\n\n  // TOCTOU mitigation: a new process may have just bound the port but not yet\n  // written its lock file. Read the lock, pause, re-read. If the second read\n  // now matches the port holder, it's a fresh leader — don't kill.\n  const { readLeaderLock } = await import('./LeaderElection.js');\n  for (let check = 0; check < LOCK_RECHECK_ATTEMPTS; check++) {\n    try {\n      const lock = await readLeaderLock();\n      if (lock?.pid === stalePid && lock?.port === port && lock.pid !== process.pid) {\n        await logger.warn(`[WebUI] Port ${port} held by legitimate leader (pid ${stalePid}) — not killing`);\n        return false;\n      }\n    } catch {\n      // Can't read lock file — continue to next check or kill\n    }\n    if (check < LOCK_RECHECK_ATTEMPTS - 1) {\n      await new Promise(r => setTimeout(r, LOCK_RECHECK_DELAY_MS));\n    }\n  }\n\n  const outcome = await killStaleProcessDetailed(stalePid, port);\n  if (outcome.killed) {\n    logger.info(`[WebUI] Stale process ${stalePid} removed from port ${port}`);\n    await new Promise(r => setTimeout(r, SIGKILL_WAIT_MS)); // brief pause for port release\n  } else {\n    await logger.debug(`[WebUI] Stale-port recovery skipped for pid ${stalePid}`, {\n      reason: outcome.reason,\n      parentPid: outcome.parentPid,\n      parentCommand: outcome.parentCommand,\n      detail: outcome.detail,\n    });\n  }\n  return outcome.killed;\n}\n"]}
|
|
@@ -17,7 +17,8 @@ import type { MetricSnapshot } from '../../metrics/types.js';
|
|
|
17
17
|
import type { MemoryLogSink } from '../../logging/sinks/MemoryLogSink.js';
|
|
18
18
|
import type { MemoryMetricsSink } from '../../metrics/sinks/MemoryMetricsSink.js';
|
|
19
19
|
import { logger } from '../../utils/logger.js';
|
|
20
|
-
import { detectLegacyLeader, type ElectionResult } from './LeaderElection.js';
|
|
20
|
+
import { detectLegacyLeader, readLeaderLock, deleteLeaderLock, type ElectionResult, type ConsoleLeaderInfo } from './LeaderElection.js';
|
|
21
|
+
import { findPidOnPort } from './StaleProcessRecovery.js';
|
|
21
22
|
/**
|
|
22
23
|
* Options for starting the unified console.
|
|
23
24
|
*/
|
|
@@ -79,6 +80,25 @@ export interface UnifiedConsoleResult {
|
|
|
79
80
|
* threw. Exposed so tests can assert the full result shape.
|
|
80
81
|
*/
|
|
81
82
|
export declare function warnIfLegacyConsolePresent(currentPort: number, detect?: typeof detectLegacyLeader, log?: typeof logger): Promise<Awaited<ReturnType<typeof detectLegacyLeader>> | null>;
|
|
83
|
+
export interface PortLeaderDiscovery {
|
|
84
|
+
leaderInfo: ConsoleLeaderInfo | null;
|
|
85
|
+
ownerPid: number | null;
|
|
86
|
+
source: 'api' | 'lock' | 'synthetic' | 'none';
|
|
87
|
+
}
|
|
88
|
+
export interface BindFailureRecoveryResult extends PortLeaderDiscovery {
|
|
89
|
+
lockCleanupAttempted: boolean;
|
|
90
|
+
lockCleanupPerformed: boolean;
|
|
91
|
+
}
|
|
92
|
+
interface DiscoveryDependencies {
|
|
93
|
+
fetchImpl?: typeof fetch;
|
|
94
|
+
findPidOnPortImpl?: typeof findPidOnPort;
|
|
95
|
+
readLeaderLockImpl?: typeof readLeaderLock;
|
|
96
|
+
}
|
|
97
|
+
export declare function discoverLeaderServingPort(port: number, authToken: string | null, deps?: DiscoveryDependencies): Promise<PortLeaderDiscovery>;
|
|
98
|
+
interface BindFailureRecoveryDependencies extends DiscoveryDependencies {
|
|
99
|
+
deleteLeaderLockImpl?: typeof deleteLeaderLock;
|
|
100
|
+
}
|
|
101
|
+
export declare function recoverLeaderBindFailure(provisionalLeader: ConsoleLeaderInfo, port: number, authToken: string | null, deps?: BindFailureRecoveryDependencies): Promise<BindFailureRecoveryResult>;
|
|
82
102
|
/**
|
|
83
103
|
* Start the unified web console.
|
|
84
104
|
*
|
|
@@ -86,4 +106,5 @@ export declare function warnIfLegacyConsolePresent(currentPort: number, detect?:
|
|
|
86
106
|
* or sets up event forwarding (follower).
|
|
87
107
|
*/
|
|
88
108
|
export declare function startUnifiedConsole(options: UnifiedConsoleOptions): Promise<UnifiedConsoleResult>;
|
|
109
|
+
export {};
|
|
89
110
|
//# sourceMappingURL=UnifiedConsole.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"UnifiedConsole.d.ts","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAEH,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAC7D,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,0CAA0C,CAAC;
|
|
1
|
+
{"version":3,"file":"UnifiedConsole.d.ts","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"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"}
|