@dollhousemcp/mcp-server 2.0.27 → 2.0.28
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 +6 -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/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/sessions.css +89 -9
- package/dist/web/public/sessions.js +160 -4
- package/dist/web/public/setup.js +4 -0
- 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"]}
|
|
@@ -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 = [
|
|
@@ -1591,6 +1594,7 @@ codex_hooks = true`;
|
|
|
1591
1594
|
intro.innerHTML = `<div class="setup-permissions-note">
|
|
1592
1595
|
<strong>Permissions & Security</strong>
|
|
1593
1596
|
<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>
|
|
1597
|
+
<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
1598
|
</div>`;
|
|
1595
1599
|
};
|
|
1596
1600
|
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dollhousemcp/mcp-server",
|
|
3
|
-
"version": "2.0.
|
|
3
|
+
"version": "2.0.28",
|
|
4
4
|
"description": "DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -20,11 +20,14 @@ RUN_DIR="$HOME/.dollhouse/run"
|
|
|
20
20
|
PORT_FILE="$RUN_DIR/permission-server.port"
|
|
21
21
|
AUTHORITY_FILE="$RUN_DIR/permission-authority.json"
|
|
22
22
|
AUTHORITY_CACHE_TTL_SECONDS=2
|
|
23
|
-
MAX_RETRIES=2
|
|
24
|
-
INITIAL_TIMEOUT=5
|
|
23
|
+
MAX_RETRIES="${DOLLHOUSE_HOOK_MAX_RETRIES:-2}"
|
|
24
|
+
INITIAL_TIMEOUT="${DOLLHOUSE_HOOK_INITIAL_TIMEOUT:-5}"
|
|
25
25
|
HOOK_PLATFORM="${DOLLHOUSE_HOOK_PLATFORM:-claude_code}"
|
|
26
26
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
27
27
|
|
|
28
|
+
[[ "$MAX_RETRIES" =~ ^[0-9]+$ ]] || MAX_RETRIES=2
|
|
29
|
+
[[ "$INITIAL_TIMEOUT" =~ ^[0-9]+$ ]] || INITIAL_TIMEOUT=5
|
|
30
|
+
|
|
28
31
|
# Debug logging helper — writes to stderr so it doesn't pollute stdout
|
|
29
32
|
debug() {
|
|
30
33
|
if [[ "${DOLLHOUSE_HOOK_DEBUG:-0}" == "1" ]]; then
|
|
@@ -114,6 +117,37 @@ normalize_response() {
|
|
|
114
117
|
end
|
|
115
118
|
' 2>/dev/null
|
|
116
119
|
;;
|
|
120
|
+
codex)
|
|
121
|
+
echo "$response" | jq -c '
|
|
122
|
+
if type == "object" and (keys | length) == 0 then
|
|
123
|
+
{
|
|
124
|
+
hookSpecificOutput: {
|
|
125
|
+
hookEventName: "PreToolUse",
|
|
126
|
+
permissionDecision: "allow",
|
|
127
|
+
permissionDecisionReason: ""
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
elif (.hookSpecificOutput.permissionDecision? | type) == "string" then
|
|
131
|
+
{
|
|
132
|
+
hookSpecificOutput: {
|
|
133
|
+
hookEventName: "PreToolUse",
|
|
134
|
+
permissionDecision: (if .hookSpecificOutput.permissionDecision == "allow" then "allow" else "deny" end),
|
|
135
|
+
permissionDecisionReason: (.hookSpecificOutput.permissionDecisionReason // .hookSpecificOutput.reason // .reason // .message // "")
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
elif (.decision? | type) == "string" and (.decision | IN("allow", "deny", "ask")) then
|
|
139
|
+
{
|
|
140
|
+
hookSpecificOutput: {
|
|
141
|
+
hookEventName: "PreToolUse",
|
|
142
|
+
permissionDecision: (if .decision == "allow" then "allow" else "deny" end),
|
|
143
|
+
permissionDecisionReason: (.reason // .message // "")
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
else
|
|
147
|
+
empty
|
|
148
|
+
end
|
|
149
|
+
' 2>/dev/null
|
|
150
|
+
;;
|
|
117
151
|
*)
|
|
118
152
|
echo "$response"
|
|
119
153
|
;;
|
|
@@ -7,11 +7,14 @@
|
|
|
7
7
|
|
|
8
8
|
RUN_DIR="$HOME/.dollhouse/run"
|
|
9
9
|
PORT_FILE="$RUN_DIR/permission-server.port"
|
|
10
|
-
MAX_RETRIES=2
|
|
11
|
-
INITIAL_TIMEOUT=5
|
|
10
|
+
MAX_RETRIES="${DOLLHOUSE_HOOK_MAX_RETRIES:-2}"
|
|
11
|
+
INITIAL_TIMEOUT="${DOLLHOUSE_HOOK_INITIAL_TIMEOUT:-5}"
|
|
12
12
|
HOOK_PLATFORM="vscode"
|
|
13
13
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
14
14
|
|
|
15
|
+
[[ "$MAX_RETRIES" =~ ^[0-9]+$ ]] || MAX_RETRIES=2
|
|
16
|
+
[[ "$INITIAL_TIMEOUT" =~ ^[0-9]+$ ]] || INITIAL_TIMEOUT=5
|
|
17
|
+
|
|
15
18
|
debug() {
|
|
16
19
|
if [[ "${DOLLHOUSE_HOOK_DEBUG:-0}" == "1" ]]; then
|
|
17
20
|
echo "[pretooluse-vscode] $*" >&2
|
|
@@ -7,11 +7,14 @@
|
|
|
7
7
|
|
|
8
8
|
RUN_DIR="$HOME/.dollhouse/run"
|
|
9
9
|
PORT_FILE="$RUN_DIR/permission-server.port"
|
|
10
|
-
MAX_RETRIES=2
|
|
11
|
-
INITIAL_TIMEOUT=5
|
|
10
|
+
MAX_RETRIES="${DOLLHOUSE_HOOK_MAX_RETRIES:-2}"
|
|
11
|
+
INITIAL_TIMEOUT="${DOLLHOUSE_HOOK_INITIAL_TIMEOUT:-5}"
|
|
12
12
|
HOOK_PLATFORM="windsurf"
|
|
13
13
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
14
14
|
|
|
15
|
+
[[ "$MAX_RETRIES" =~ ^[0-9]+$ ]] || MAX_RETRIES=2
|
|
16
|
+
[[ "$INITIAL_TIMEOUT" =~ ^[0-9]+$ ]] || INITIAL_TIMEOUT=5
|
|
17
|
+
|
|
15
18
|
debug() {
|
|
16
19
|
if [[ "${DOLLHOUSE_HOOK_DEBUG:-0}" == "1" ]]; then
|
|
17
20
|
echo "[pretooluse-windsurf] $*" >&2
|
package/server.json
CHANGED
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
"name": "io.github.DollhouseMCP/mcp-server",
|
|
4
4
|
"title": "DollhouseMCP",
|
|
5
5
|
"description": "OSS to create Personas, Skills, Templates, Agents, and Memories to customize your AI experience.",
|
|
6
|
-
"version": "2.0.
|
|
6
|
+
"version": "2.0.28",
|
|
7
7
|
"homepage": "https://dollhousemcp.com",
|
|
8
8
|
"repository": {
|
|
9
9
|
"type": "git",
|
|
@@ -29,7 +29,7 @@
|
|
|
29
29
|
{
|
|
30
30
|
"registryType": "npm",
|
|
31
31
|
"identifier": "@dollhousemcp/mcp-server",
|
|
32
|
-
"version": "2.0.
|
|
32
|
+
"version": "2.0.28",
|
|
33
33
|
"transport": {
|
|
34
34
|
"type": "stdio"
|
|
35
35
|
}
|