@dollhousemcp/mcp-server 2.0.27 → 2.0.29

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/dist/generated/version.d.ts +2 -2
  3. package/dist/generated/version.js +3 -3
  4. package/dist/handlers/mcp-aql/OperationSchema.js +2 -2
  5. package/dist/handlers/mcp-aql/evaluatePermission.d.ts.map +1 -1
  6. package/dist/handlers/mcp-aql/evaluatePermission.js +9 -3
  7. package/dist/services/BuildInfoService.d.ts +6 -1
  8. package/dist/services/BuildInfoService.d.ts.map +1 -1
  9. package/dist/services/BuildInfoService.js +23 -3
  10. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -1
  11. package/dist/tools/portfolio/submitToPortfolioTool.js +4 -3
  12. package/dist/utils/permissionHooks.d.ts +18 -0
  13. package/dist/utils/permissionHooks.d.ts.map +1 -1
  14. package/dist/utils/permissionHooks.js +182 -15
  15. package/dist/web/console/IngestRoutes.d.ts +7 -1
  16. package/dist/web/console/IngestRoutes.d.ts.map +1 -1
  17. package/dist/web/console/IngestRoutes.js +28 -6
  18. package/dist/web/console/LeaderForwardingSink.d.ts +6 -1
  19. package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
  20. package/dist/web/console/LeaderForwardingSink.js +8 -2
  21. package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
  22. package/dist/web/console/UnifiedConsole.js +6 -3
  23. package/dist/web/console/sessionClientPlatform.d.ts +11 -0
  24. package/dist/web/console/sessionClientPlatform.d.ts.map +1 -0
  25. package/dist/web/console/sessionClientPlatform.js +83 -0
  26. package/dist/web/public/permissions.js +10 -0
  27. package/dist/web/public/sessions.css +89 -9
  28. package/dist/web/public/sessions.js +160 -4
  29. package/dist/web/public/setup.js +40 -0
  30. package/dist/web/routes/permissionRoutes.d.ts +1 -0
  31. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  32. package/dist/web/routes/permissionRoutes.js +17 -7
  33. package/dist/web/routes/setupRoutes.d.ts +5 -1
  34. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  35. package/dist/web/routes/setupRoutes.js +28 -14
  36. package/dist/web/server.d.ts.map +1 -1
  37. package/dist/web/server.js +5 -1
  38. package/package.json +1 -1
  39. package/scripts/pretooluse-dollhouse.sh +36 -2
  40. package/scripts/pretooluse-vscode.sh +5 -2
  41. package/scripts/pretooluse-windsurf.sh +5 -2
  42. package/server.json +2 -2
@@ -0,0 +1,83 @@
1
+ /**
2
+ * Best-effort MCP client platform detection for session metadata.
3
+ *
4
+ * This is intentionally conservative: when we cannot identify the host
5
+ * confidently, we return null rather than guessing from a session ID.
6
+ */
7
+ const CLIENT_PLATFORM_LABELS = {
8
+ 'claude-code': 'Claude Code',
9
+ 'claude-desktop': 'Claude Desktop',
10
+ codex: 'Codex',
11
+ cursor: 'Cursor',
12
+ vscode: 'VS Code',
13
+ windsurf: 'Windsurf',
14
+ 'gemini-cli': 'Gemini CLI',
15
+ cline: 'Cline',
16
+ lmstudio: 'LM Studio',
17
+ 'web-console': 'Web Console',
18
+ };
19
+ function normalizeText(value) {
20
+ return typeof value === 'string' ? value.trim().toLowerCase() : '';
21
+ }
22
+ function includesAny(value, needles) {
23
+ return needles.some((needle) => value.includes(needle));
24
+ }
25
+ function matchesAnySource(values, needles) {
26
+ return values.some((value) => includesAny(value, needles));
27
+ }
28
+ const TEXT_PLATFORM_MATCHERS = [
29
+ { platform: 'cursor', needles: ['cursor'] },
30
+ { platform: 'windsurf', needles: ['windsurf'] },
31
+ { platform: 'gemini-cli', needles: ['gemini'] },
32
+ { platform: 'cline', needles: ['cline'] },
33
+ { platform: 'lmstudio', needles: ['lmstudio', 'lm studio'] },
34
+ { platform: 'claude-desktop', needles: ['claude desktop'] },
35
+ { platform: 'claude-code', needles: ['claude code'] },
36
+ { platform: 'codex', needles: ['codex'] },
37
+ ];
38
+ export function normalizeSessionClientPlatformId(value) {
39
+ const normalized = normalizeText(value ?? undefined);
40
+ if (!normalized) {
41
+ return null;
42
+ }
43
+ if (normalized === 'gemini') {
44
+ return 'gemini-cli';
45
+ }
46
+ if (normalized in CLIENT_PLATFORM_LABELS) {
47
+ return normalized;
48
+ }
49
+ return null;
50
+ }
51
+ export function getSessionClientPlatformLabel(platform) {
52
+ return platform ? CLIENT_PLATFORM_LABELS[platform] ?? '' : '';
53
+ }
54
+ export function detectSessionClientPlatformId(env = process.env, argv = process.argv, execPath = process.execPath, title = process.title) {
55
+ const termProgram = normalizeText(env.TERM_PROGRAM);
56
+ const argvText = normalizeText(argv.join(' '));
57
+ const execPathText = normalizeText(execPath);
58
+ const titleText = normalizeText(title);
59
+ const textSources = [argvText, execPathText, titleText];
60
+ if (env.CLAUDE_DESKTOP === 'true' || env.CLAUDE_DESKTOP_VERSION) {
61
+ return 'claude-desktop';
62
+ }
63
+ if (env.CLAUDE_CODE === 'true' || termProgram === 'claude-code') {
64
+ return 'claude-code';
65
+ }
66
+ if (env.VSCODE_CWD ||
67
+ env.VSCODE_PID ||
68
+ env.VSCODE_IPC_HOOK ||
69
+ env.VSCODE_NLS_CONFIG ||
70
+ termProgram === 'vscode') {
71
+ return 'vscode';
72
+ }
73
+ if (env.CODEX_HOME || termProgram === 'codex') {
74
+ return 'codex';
75
+ }
76
+ for (const matcher of TEXT_PLATFORM_MATCHERS) {
77
+ if (matchesAnySource(textSources, matcher.needles)) {
78
+ return matcher.platform;
79
+ }
80
+ }
81
+ return null;
82
+ }
83
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sessionClientPlatform.js","sourceRoot":"","sources":["../../../src/web/console/sessionClientPlatform.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Best-effort MCP client platform detection for session metadata.\n *\n * This is intentionally conservative: when we cannot identify the host\n * confidently, we return null rather than guessing from a session ID.\n */\n\nexport type SessionClientPlatformId =\n  | 'claude-code'\n  | 'claude-desktop'\n  | 'codex'\n  | 'cursor'\n  | 'vscode'\n  | 'windsurf'\n  | 'gemini-cli'\n  | 'cline'\n  | 'lmstudio'\n  | 'web-console';\n\nconst CLIENT_PLATFORM_LABELS: Record<SessionClientPlatformId, string> = {\n  'claude-code': 'Claude Code',\n  'claude-desktop': 'Claude Desktop',\n  codex: 'Codex',\n  cursor: 'Cursor',\n  vscode: 'VS Code',\n  windsurf: 'Windsurf',\n  'gemini-cli': 'Gemini CLI',\n  cline: 'Cline',\n  lmstudio: 'LM Studio',\n  'web-console': 'Web Console',\n};\n\nfunction normalizeText(value: string | undefined): string {\n  return typeof value === 'string' ? value.trim().toLowerCase() : '';\n}\n\nfunction includesAny(value: string, needles: readonly string[]): boolean {\n  return needles.some((needle) => value.includes(needle));\n}\n\nfunction matchesAnySource(values: readonly string[], needles: readonly string[]): boolean {\n  return values.some((value) => includesAny(value, needles));\n}\n\nconst TEXT_PLATFORM_MATCHERS: ReadonlyArray<{\n  platform: SessionClientPlatformId;\n  needles: readonly string[];\n}> = [\n  { platform: 'cursor', needles: ['cursor'] },\n  { platform: 'windsurf', needles: ['windsurf'] },\n  { platform: 'gemini-cli', needles: ['gemini'] },\n  { platform: 'cline', needles: ['cline'] },\n  { platform: 'lmstudio', needles: ['lmstudio', 'lm studio'] },\n  { platform: 'claude-desktop', needles: ['claude desktop'] },\n  { platform: 'claude-code', needles: ['claude code'] },\n  { platform: 'codex', needles: ['codex'] },\n];\n\nexport function normalizeSessionClientPlatformId(\n  value: string | null | undefined,\n): SessionClientPlatformId | null {\n  const normalized = normalizeText(value ?? undefined);\n  if (!normalized) {\n    return null;\n  }\n\n  if (normalized === 'gemini') {\n    return 'gemini-cli';\n  }\n\n  if (normalized in CLIENT_PLATFORM_LABELS) {\n    return normalized as SessionClientPlatformId;\n  }\n\n  return null;\n}\n\nexport function getSessionClientPlatformLabel(\n  platform: SessionClientPlatformId | null | undefined,\n): string {\n  return platform ? CLIENT_PLATFORM_LABELS[platform] ?? '' : '';\n}\n\nexport function detectSessionClientPlatformId(\n  env: NodeJS.ProcessEnv = process.env,\n  argv: readonly string[] = process.argv,\n  execPath: string = process.execPath,\n  title: string = process.title,\n): SessionClientPlatformId | null {\n  const termProgram = normalizeText(env.TERM_PROGRAM);\n  const argvText = normalizeText(argv.join(' '));\n  const execPathText = normalizeText(execPath);\n  const titleText = normalizeText(title);\n  const textSources = [argvText, execPathText, titleText];\n\n  if (env.CLAUDE_DESKTOP === 'true' || env.CLAUDE_DESKTOP_VERSION) {\n    return 'claude-desktop';\n  }\n\n  if (env.CLAUDE_CODE === 'true' || termProgram === 'claude-code') {\n    return 'claude-code';\n  }\n\n  if (\n    env.VSCODE_CWD ||\n    env.VSCODE_PID ||\n    env.VSCODE_IPC_HOOK ||\n    env.VSCODE_NLS_CONFIG ||\n    termProgram === 'vscode'\n  ) {\n    return 'vscode';\n  }\n\n  if (env.CODEX_HOME || termProgram === 'codex') {\n    return 'codex';\n  }\n\n  for (const matcher of TEXT_PLATFORM_MATCHERS) {\n    if (matchesAnySource(textSources, matcher.needles)) {\n      return matcher.platform;\n    }\n  }\n\n  return null;\n}\n"]}
@@ -225,6 +225,16 @@
225
225
  } else if (data.permissionPromptActive) {
226
226
  hookDot.dataset.status = 'active';
227
227
  hookLabel.textContent = 'Prompt tool active';
228
+ } else if (data.hookNeedsRepair) {
229
+ hookDot.dataset.status = 'warning';
230
+ hookLabel.textContent = data.hookHost
231
+ ? `Hook needs repair (${data.hookHost})`
232
+ : 'Hook needs repair';
233
+ } else if (data.hookAutoRepaired) {
234
+ hookDot.dataset.status = 'active';
235
+ hookLabel.textContent = data.hookHost
236
+ ? `Hook refreshed (${data.hookHost})`
237
+ : 'Hook refreshed';
228
238
  } else if (data.hookInstalled) {
229
239
  hookDot.dataset.status = 'active';
230
240
  hookLabel.textContent = data.hookHost ? `Hook installed (${data.hookHost})` : 'Hook installed';
@@ -66,7 +66,8 @@
66
66
  position: absolute;
67
67
  top: calc(100% + 0.4rem);
68
68
  right: 0;
69
- min-width: 480px;
69
+ width: min(34rem, calc(100vw - 1rem));
70
+ box-sizing: border-box;
70
71
  background: var(--paper-strong, #fff);
71
72
  border: 1px solid var(--line, #c8d5e9);
72
73
  border-radius: var(--radius-md, 0.85rem);
@@ -184,10 +185,11 @@
184
185
 
185
186
  .session-dropdown-item {
186
187
  display: grid;
187
- grid-template-columns: 1rem 8px 1fr 62px 72px 3.5rem 1.2rem;
188
+ grid-template-columns: 1rem 8px minmax(0, 1fr) 5.5rem 6.75rem 4.5rem 1.2rem;
188
189
  align-items: center;
189
- gap: 0.4rem;
190
- padding: 0.45rem 0.75rem;
190
+ column-gap: 0.7rem;
191
+ row-gap: 0.3rem;
192
+ padding: 0.55rem 0.9rem;
191
193
  font-size: var(--step--1, 0.82rem);
192
194
  cursor: pointer;
193
195
  transition: background 0.1s;
@@ -235,12 +237,51 @@
235
237
  white-space: nowrap;
236
238
  }
237
239
 
240
+ .session-dropdown-primary {
241
+ min-width: 0;
242
+ display: flex;
243
+ flex-direction: column;
244
+ gap: 0.18rem;
245
+ align-self: center;
246
+ }
247
+
248
+ .session-dropdown-meta {
249
+ display: flex;
250
+ align-items: center;
251
+ gap: 0.35rem;
252
+ min-width: 0;
253
+ flex-wrap: wrap;
254
+ }
255
+
256
+ .session-dropdown-version {
257
+ font-size: 0.7rem;
258
+ font-family: var(--font-mono, monospace);
259
+ color: var(--ink-500, #677893);
260
+ white-space: nowrap;
261
+ }
262
+
263
+ .session-dropdown-update {
264
+ display: inline-flex;
265
+ align-items: center;
266
+ padding: 1px 5px;
267
+ border-radius: 999px;
268
+ font-size: 0.62rem;
269
+ font-weight: 700;
270
+ letter-spacing: 0.04em;
271
+ background: #ecfdf5;
272
+ color: #166534;
273
+ border: 1px solid #86efac;
274
+ white-space: nowrap;
275
+ }
276
+
238
277
  .session-dropdown-uptime {
239
278
  font-size: 0.7rem;
240
279
  font-family: var(--font-mono, monospace);
241
280
  color: var(--ink-500, #677893);
242
281
  white-space: nowrap;
243
282
  text-align: right;
283
+ min-width: 4.5rem;
284
+ justify-self: end;
244
285
  }
245
286
 
246
287
  .session-dropdown-role {
@@ -287,6 +328,16 @@
287
328
  color: var(--ink-900, #e0e8f0);
288
329
  }
289
330
 
331
+ [data-theme="dark"] .session-dropdown-version {
332
+ color: var(--ink-500, #8899bb);
333
+ }
334
+
335
+ [data-theme="dark"] .session-dropdown-update {
336
+ background: #052e16;
337
+ color: #bbf7d0;
338
+ border-color: #166534;
339
+ }
340
+
290
341
  [data-theme="dark"] .session-dropdown-toggle-label {
291
342
  color: var(--ink-500, #8899bb);
292
343
  }
@@ -363,15 +414,40 @@
363
414
  * - Color (blue=positive, orange=negative — colorblind-safe pair)
364
415
  */
365
416
  .session-status-badge {
366
- display: inline-block;
367
- padding: 1px 5px;
368
- border-radius: 3px;
369
- font-size: 9px;
417
+ display: inline-flex;
418
+ align-items: center;
419
+ justify-content: center;
420
+ padding: 2px 7px;
421
+ border-radius: 5px;
422
+ font-size: 10px;
370
423
  font-weight: 600;
371
424
  letter-spacing: 0.04em;
372
425
  white-space: nowrap;
373
426
  text-align: center;
374
- min-width: 52px;
427
+ min-width: 4.75rem;
428
+ box-sizing: border-box;
429
+ }
430
+
431
+ .session-dropdown-item > .session-status-badge {
432
+ justify-self: center;
433
+ }
434
+
435
+ .session-dropdown-badge-stack {
436
+ display: flex;
437
+ flex-direction: column;
438
+ align-items: center;
439
+ justify-self: center;
440
+ gap: 0.24rem;
441
+ width: 100%;
442
+ max-width: 6.75rem;
443
+ }
444
+
445
+ .session-dropdown-client-label {
446
+ font-size: 0.62rem;
447
+ line-height: 1.15;
448
+ color: var(--ink-500, #677893);
449
+ text-align: center;
450
+ width: 100%;
375
451
  }
376
452
 
377
453
  .session-status-badge[data-status="positive"] {
@@ -398,6 +474,10 @@
398
474
  border-color: #9a3412;
399
475
  }
400
476
 
477
+ [data-theme="dark"] .session-dropdown-client-label {
478
+ color: var(--ink-500, #8899bb);
479
+ }
480
+
401
481
  /* Session filter in the log viewer */
402
482
  #log-session-filter {
403
483
  min-width: 120px;
@@ -33,6 +33,18 @@
33
33
  var lastReloadTargetVersion = '';
34
34
  var pendingLeaderReloadTimer = null;
35
35
  var showPolicySessions = loadPolicyDebugVisibility();
36
+ var CLIENT_PLATFORM_LABELS = {
37
+ 'claude-code': 'Claude Code',
38
+ 'claude-desktop': 'Claude Desktop',
39
+ 'codex': 'Codex',
40
+ 'cursor': 'Cursor',
41
+ 'vscode': 'VS Code',
42
+ 'windsurf': 'Windsurf',
43
+ 'gemini-cli': 'Gemini CLI',
44
+ 'cline': 'Cline',
45
+ 'lmstudio': 'LM Studio',
46
+ 'web-console': 'Web Console'
47
+ };
36
48
 
37
49
  function loadPolicyDebugVisibility() {
38
50
  try {
@@ -176,6 +188,55 @@
176
188
  /** NFC-normalize a string safely */
177
189
  function nfc(s) { try { return s.normalize('NFC'); } catch(e) { return s; } }
178
190
 
191
+ function normalizeClientPlatform(platform) {
192
+ if (typeof platform !== 'string') return '';
193
+ var normalized = nfc(platform).trim().toLowerCase();
194
+ if (!normalized) return '';
195
+ if (normalized === 'gemini') return 'gemini-cli';
196
+ return Object.prototype.hasOwnProperty.call(CLIENT_PLATFORM_LABELS, normalized) ? normalized : '';
197
+ }
198
+
199
+ function displayPlatform(session) {
200
+ if (!session || typeof session !== 'object') return '';
201
+ if (typeof session.clientPlatformLabel === 'string' && session.clientPlatformLabel.trim()) {
202
+ return nfc(session.clientPlatformLabel.trim());
203
+ }
204
+ var platform = normalizeClientPlatform(session.clientPlatform);
205
+ return platform ? CLIENT_PLATFORM_LABELS[platform] || '' : '';
206
+ }
207
+
208
+ function displayVersion(session) {
209
+ if (!session || typeof session !== 'object') return '';
210
+ var normalized = normalizeSemver(session.serverVersion);
211
+ if (!normalized && typeof session.serverVersion === 'string') {
212
+ normalized = nfc(session.serverVersion).trim();
213
+ }
214
+ if (!normalized) return '';
215
+ return normalized.charAt(0) === 'v' ? normalized : ('v' + normalized);
216
+ }
217
+
218
+ function getNewestKnownSessionVersion(list) {
219
+ if (!Array.isArray(list)) return '';
220
+ var newest = '';
221
+ for (var i = 0; i < list.length; i++) {
222
+ var session = list[i];
223
+ if (!session || isPolicyOnlySession(session) || session.status !== 'active') continue;
224
+ var version = normalizeSemver(session.serverVersion);
225
+ if (!version) continue;
226
+ if (!newest || compareSemver(version, newest) > 0) {
227
+ newest = version;
228
+ }
229
+ }
230
+ return newest;
231
+ }
232
+
233
+ function sessionHasUpdateAvailable(session, newestVersion) {
234
+ if (!session || !newestVersion || isPolicyOnlySession(session)) return false;
235
+ var version = normalizeSemver(session.serverVersion);
236
+ if (!version) return false;
237
+ return compareSemver(version, newestVersion) < 0;
238
+ }
239
+
179
240
  function isPolicyOnlySession(session) {
180
241
  return !!(session && session.isPolicyOnly);
181
242
  }
@@ -231,6 +292,9 @@
231
292
  isLeader: false,
232
293
  authenticated: false,
233
294
  kind: 'policy',
295
+ serverVersion: '',
296
+ clientPlatform: '',
297
+ clientPlatformLabel: '',
234
298
  isPolicyOnly: true,
235
299
  });
236
300
  }
@@ -254,12 +318,63 @@
254
318
  // Build a key from current sessions to detect changes
255
319
  function sessionListKey(list) {
256
320
  return list.map(function(s) {
257
- return s.sessionId + ':' + s.status + ':' + (isPolicyOnlySession(s) ? 'policy' : 'live');
321
+ return [
322
+ s.sessionId,
323
+ s.status,
324
+ s.displayName || '',
325
+ s.serverVersion || '',
326
+ s.clientPlatform || '',
327
+ s.clientPlatformLabel || '',
328
+ s.isLeader ? 'leader' : 'member',
329
+ s.authenticated ? 'auth' : 'noauth',
330
+ isPolicyOnlySession(s) ? 'policy' : 'live'
331
+ ].join(':');
258
332
  }).join(',')
259
333
  + '|policyDebug:' + (showPolicySessions ? 'on' : 'off')
260
334
  + '|knownPolicy:' + policySessions.map(function(session) { return session.sessionId; }).join(',');
261
335
  }
262
336
 
337
+ function normalizeLiveSessions(list) {
338
+ if (!Array.isArray(list)) return [];
339
+ var normalized = [];
340
+ var seen = new Set();
341
+
342
+ for (var i = 0; i < list.length; i++) {
343
+ var item = list[i];
344
+ if (!item || typeof item.sessionId !== 'string') continue;
345
+ var sessionId = nfc(item.sessionId).trim();
346
+ if (!sessionId || seen.has(sessionId)) continue;
347
+ seen.add(sessionId);
348
+
349
+ var clientPlatform = normalizeClientPlatform(item.clientPlatform);
350
+ var clientPlatformLabel = '';
351
+ if (typeof item.clientPlatformLabel === 'string' && item.clientPlatformLabel.trim()) {
352
+ clientPlatformLabel = nfc(item.clientPlatformLabel).trim();
353
+ } else if (clientPlatform) {
354
+ clientPlatformLabel = CLIENT_PLATFORM_LABELS[clientPlatform] || '';
355
+ }
356
+
357
+ normalized.push({
358
+ sessionId: sessionId,
359
+ displayName: nfc(typeof item.displayName === 'string' && item.displayName ? item.displayName : sessionId),
360
+ color: typeof item.color === 'string' ? item.color : '',
361
+ pid: typeof item.pid === 'number' ? item.pid : 0,
362
+ startedAt: typeof item.startedAt === 'string' ? item.startedAt : '',
363
+ lastHeartbeat: typeof item.lastHeartbeat === 'string' ? item.lastHeartbeat : '',
364
+ status: item.status === 'ended' ? 'ended' : 'active',
365
+ isLeader: !!item.isLeader,
366
+ authenticated: !!item.authenticated,
367
+ kind: typeof item.kind === 'string' ? item.kind : 'mcp',
368
+ serverVersion: normalizeSemver(item.serverVersion) || (typeof item.serverVersion === 'string' ? nfc(item.serverVersion).trim() : ''),
369
+ consoleProtocolVersion: typeof item.consoleProtocolVersion === 'number' ? item.consoleProtocolVersion : 0,
370
+ clientPlatform: clientPlatform,
371
+ clientPlatformLabel: clientPlatformLabel,
372
+ });
373
+ }
374
+
375
+ return normalized;
376
+ }
377
+
263
378
  function setPolicyDebugVisibility(nextVisible, keepDropdownOpen) {
264
379
  var normalized = !!nextVisible;
265
380
  if (showPolicySessions === normalized) return;
@@ -513,6 +628,7 @@
513
628
  if (!a.isLeader && b.isLeader) return 1;
514
629
  return 0;
515
630
  });
631
+ var newestKnownVersion = getNewestKnownSessionVersion(sorted);
516
632
 
517
633
  function appendSessionItem(s) {
518
634
  var item = document.createElement('div');
@@ -529,11 +645,37 @@
529
645
  if (s.color) dot.style.background = s.color;
530
646
  item.appendChild(dot);
531
647
 
648
+ var nameWrap = document.createElement('div');
649
+ nameWrap.className = 'session-dropdown-primary';
650
+
532
651
  var nameEl = document.createElement('span');
533
652
  nameEl.className = 'session-dropdown-name';
534
653
  nameEl.textContent = displayName(s);
535
654
  if (s.color) nameEl.style.color = s.color;
536
- item.appendChild(nameEl);
655
+ nameWrap.appendChild(nameEl);
656
+
657
+ var versionText = displayVersion(s);
658
+ if (versionText) {
659
+ var metaRow = document.createElement('div');
660
+ metaRow.className = 'session-dropdown-meta';
661
+
662
+ var versionEl = document.createElement('span');
663
+ versionEl.className = 'session-dropdown-version';
664
+ versionEl.textContent = versionText;
665
+ metaRow.appendChild(versionEl);
666
+
667
+ if (sessionHasUpdateAvailable(s, newestKnownVersion)) {
668
+ var updateBadge = document.createElement('span');
669
+ updateBadge.className = 'session-dropdown-update';
670
+ updateBadge.textContent = 'Update available';
671
+ updateBadge.title = 'A newer local DollhouseMCP session version is active.';
672
+ metaRow.appendChild(updateBadge);
673
+ }
674
+
675
+ nameWrap.appendChild(metaRow);
676
+ }
677
+
678
+ item.appendChild(nameWrap);
537
679
 
538
680
  // Session status badges (#1805) — for persisted policy sessions we
539
681
  // switch from "live/authenticated" semantics to "saved/no client".
@@ -554,6 +696,9 @@
554
696
  }
555
697
  item.appendChild(authBadge);
556
698
 
699
+ var clientWrap = document.createElement('div');
700
+ clientWrap.className = 'session-dropdown-badge-stack';
701
+
557
702
  var clientBadge = document.createElement('span');
558
703
  clientBadge.className = 'session-status-badge';
559
704
  if (isPolicyOnlySession(s)) {
@@ -569,7 +714,17 @@
569
714
  clientBadge.dataset.status = 'negative';
570
715
  clientBadge.title = 'No MCP client attached';
571
716
  }
572
- item.appendChild(clientBadge);
717
+ clientWrap.appendChild(clientBadge);
718
+
719
+ var platformLabel = displayPlatform(s);
720
+ if (platformLabel) {
721
+ var clientLabel = document.createElement('span');
722
+ clientLabel.className = 'session-dropdown-client-label';
723
+ clientLabel.textContent = platformLabel;
724
+ clientWrap.appendChild(clientLabel);
725
+ }
726
+
727
+ item.appendChild(clientWrap);
573
728
 
574
729
  var uptimeEl = document.createElement('span');
575
730
  uptimeEl.className = 'session-dropdown-uptime';
@@ -727,7 +882,7 @@
727
882
  return res.json();
728
883
  }).then(function(data) {
729
884
  if (data && data.sessions) {
730
- sessions = data.sessions;
885
+ sessions = normalizeLiveSessions(data.sessions);
731
886
  maybeForceReloadForNewLeader(sessions);
732
887
  updateSessionIndicator();
733
888
  updateSessionFilterOptions();
@@ -743,6 +898,7 @@
743
898
  window.DollhouseSessions = {
744
899
  getFilterSessionId: function() { return filterSessionId; },
745
900
  displayName: displayName,
901
+ displayPlatform: displayPlatform,
746
902
  getSessions: function() { return sessions; },
747
903
  getLiveSessions: getLiveSessions,
748
904
  getSelectableSessions: getSelectableSessions,
@@ -13,6 +13,9 @@
13
13
  const PKG = '@dollhousemcp/mcp-server';
14
14
  const HOOKS_DIR = '~/.dollhouse/hooks';
15
15
  const HOOK_BASE_SCRIPT_PATH = `${HOOKS_DIR}/pretooluse-dollhouse.sh`;
16
+ const HOOK_CONTRACT_DOC_URL = 'https://github.com/DollhouseMCP/mcp-server/blob/main/docs/architecture/permission-hook-platform-contracts.md';
17
+ // Keep hook entrypoints and output expectations in sync with
18
+ // docs/architecture/permission-hook-platform-contracts.md.
16
19
 
17
20
  /** Platform registry — drives config generation AND panel rendering */
18
21
  const PLATFORMS = [
@@ -677,6 +680,13 @@ codex_hooks = true`;
677
680
  const updatePermissionInstallButton = (btn, detected) => {
678
681
  if (!btn || btn.classList.contains('is-success')) return;
679
682
 
683
+ if (detected?.hookNeedsRepair) {
684
+ btn.textContent = 'Repair hooks';
685
+ btn.disabled = false;
686
+ btn.classList.remove('is-match');
687
+ return;
688
+ }
689
+
680
690
  if (detected?.hookInstalled) {
681
691
  btn.textContent = 'Permissions enabled';
682
692
  btn.disabled = true;
@@ -1116,7 +1126,30 @@ codex_hooks = true`;
1116
1126
  return PERMISSION_SUPPORT_MATRIX[platformId];
1117
1127
  };
1118
1128
 
1129
+ const getHookRepairStatusCopy = (support, detected) => {
1130
+ if (detected?.hookNeedsRepair) {
1131
+ return {
1132
+ tone: 'warning',
1133
+ titleText: `${support.label} hook files need repair.`,
1134
+ messageText: 'DollhouseMCP detected stale local hook assets. Use Configure Now below to rewrite them, or reload the local server so the automatic repair pass can run again.',
1135
+ };
1136
+ }
1137
+
1138
+ if (detected?.hookAutoRepaired) {
1139
+ return {
1140
+ tone: 'info',
1141
+ titleText: `${support.label} hook files were refreshed automatically.`,
1142
+ messageText: 'The installed local hook assets were updated to match this release. Restart the client if it is already running.',
1143
+ };
1144
+ }
1145
+
1146
+ return null;
1147
+ };
1148
+
1119
1149
  const getFullNativePermissionStatusCopy = (support, detected) => {
1150
+ const repairCopy = getHookRepairStatusCopy(support, detected);
1151
+ if (repairCopy) return repairCopy;
1152
+
1120
1153
  if (detected?.hookInstalled) {
1121
1154
  return {
1122
1155
  tone: 'info',
@@ -1142,6 +1175,9 @@ codex_hooks = true`;
1142
1175
 
1143
1176
  const getPartialPermissionStatusCopy = (support, detected) => {
1144
1177
  const activationLabel = support.label === 'Codex' ? 'Bash guardrails' : 'permission hooks';
1178
+ const repairCopy = getHookRepairStatusCopy(support, detected);
1179
+ if (repairCopy) return repairCopy;
1180
+
1145
1181
  if (detected?.hookInstalled) {
1146
1182
  return {
1147
1183
  tone: 'info',
@@ -1182,6 +1218,9 @@ codex_hooks = true`;
1182
1218
  };
1183
1219
 
1184
1220
  const getManualPermissionStatusCopy = (support, detected) => {
1221
+ const repairCopy = getHookRepairStatusCopy(support, detected);
1222
+ if (repairCopy) return repairCopy;
1223
+
1185
1224
  if (detected?.hookAssetsPrepared) {
1186
1225
  return {
1187
1226
  tone: 'info',
@@ -1591,6 +1630,7 @@ codex_hooks = true`;
1591
1630
  intro.innerHTML = `<div class="setup-permissions-note">
1592
1631
  <strong>Permissions &amp; Security</strong>
1593
1632
  <p>Use this mode to turn on permission enforcement for supported clients. Claude Code is fully guided in this release. Gemini CLI, Cursor, VS Code, Windsurf, and Codex have native partial support, while Cline and LM Studio stay in the MCP and fallback lane for now. Other clients will be marked as coming soon.</p>
1633
+ <p class="setup-hint">Need the advanced client-by-client hook contract details? <a href="${HOOK_CONTRACT_DOC_URL}" target="_blank" rel="noopener noreferrer">Read the platform contract reference</a>.</p>
1594
1634
  </div>`;
1595
1635
  };
1596
1636
 
@@ -14,6 +14,7 @@ import type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';
14
14
  */
15
15
  export interface RegisterPermissionRoutesOptions {
16
16
  homeDir?: string;
17
+ autoRepairHookAssets?: boolean;
17
18
  }
18
19
  export declare function registerPermissionRoutes(router: Router, handler: MCPAQLHandler, options?: RegisterPermissionRoutesOptions): void;
19
20
  //# sourceMappingURL=permissionRoutes.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"permissionRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAG1C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,yCAAyC,CAAC;AA8T7E;;;GAGG;AACH,MAAM,WAAW,+BAA+B;IAC9C,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,wBAAgB,wBAAwB,CACtC,MAAM,EAAE,MAAM,EACd,OAAO,EAAE,aAAa,EACtB,OAAO,GAAE,+BAAoC,GAC5C,IAAI,CA+NN"}
1
+ {"version":3,"file":"permissionRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAG1C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,yCAAyC,CAAC;AAyV7E;;;GAGG;AACH,MAAM,WAAW,+BAA+B;IAC9C,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,oBAAoB,CAAC,EAAE,OAAO,CAAC;CAChC;AAED,wBAAgB,wBAAwB,CACtC,MAAM,EAAE,MAAM,EACd,OAAO,EAAE,aAAa,EACtB,OAAO,GAAE,+BAAoC,GAC5C,IAAI,CAiPN"}