@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.
- package/CHANGELOG.md +13 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/mcp-aql/OperationSchema.js +2 -2
- package/dist/handlers/mcp-aql/evaluatePermission.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/evaluatePermission.js +9 -3
- package/dist/services/BuildInfoService.d.ts +6 -1
- package/dist/services/BuildInfoService.d.ts.map +1 -1
- package/dist/services/BuildInfoService.js +23 -3
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -1
- package/dist/tools/portfolio/submitToPortfolioTool.js +4 -3
- package/dist/utils/permissionHooks.d.ts +18 -0
- package/dist/utils/permissionHooks.d.ts.map +1 -1
- package/dist/utils/permissionHooks.js +182 -15
- package/dist/web/console/IngestRoutes.d.ts +7 -1
- package/dist/web/console/IngestRoutes.d.ts.map +1 -1
- package/dist/web/console/IngestRoutes.js +28 -6
- package/dist/web/console/LeaderForwardingSink.d.ts +6 -1
- package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
- package/dist/web/console/LeaderForwardingSink.js +8 -2
- package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
- package/dist/web/console/UnifiedConsole.js +6 -3
- package/dist/web/console/sessionClientPlatform.d.ts +11 -0
- package/dist/web/console/sessionClientPlatform.d.ts.map +1 -0
- package/dist/web/console/sessionClientPlatform.js +83 -0
- package/dist/web/public/permissions.js +10 -0
- package/dist/web/public/sessions.css +89 -9
- package/dist/web/public/sessions.js +160 -4
- package/dist/web/public/setup.js +40 -0
- package/dist/web/routes/permissionRoutes.d.ts +1 -0
- package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
- package/dist/web/routes/permissionRoutes.js +17 -7
- package/dist/web/routes/setupRoutes.d.ts +5 -1
- package/dist/web/routes/setupRoutes.d.ts.map +1 -1
- package/dist/web/routes/setupRoutes.js +28 -14
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +5 -1
- package/package.json +1 -1
- package/scripts/pretooluse-dollhouse.sh +36 -2
- package/scripts/pretooluse-vscode.sh +5 -2
- package/scripts/pretooluse-windsurf.sh +5 -2
- 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
|
-
|
|
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
|
|
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.
|
|
190
|
-
|
|
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-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
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:
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
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,
|
package/dist/web/public/setup.js
CHANGED
|
@@ -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 & 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;
|
|
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"}
|